diff --git a/firmware/controllers/engine_controller_misc.cpp b/firmware/controllers/engine_controller_misc.cpp index d858aa26db..9d826d0804 100644 --- a/firmware/controllers/engine_controller_misc.cpp +++ b/firmware/controllers/engine_controller_misc.cpp @@ -64,23 +64,10 @@ efitimeus_t getTimeNowUs(void) { } -// this is bits 30-61, not 32-63. We only support 62-bit time. You can fire me in 36,533 years -// (1,461 on the simulator). -static volatile uint32_t upperTimeNt = 0; +static WrapAround62 timeNt; efitick_t getTimeNowNt() { - // Shift cannot be 31, as we wouldn't be able to tell if time is moving forward or backward - // relative to upperTimeNt. We do need to handle both directions as our "thread" can be - // racing with other "threads" in sampling stamp and updating upperTimeNt. - constexpr unsigned shift = 30; - - uint32_t stamp = getTimeNowLowerNt(); - uint32_t upper = upperTimeNt; - uint32_t relative_unsigned = stamp - (upper << shift); - efitick_t time64 = (efitick_t(upper) << shift) + (int32_t)relative_unsigned; - upperTimeNt = time64 >> shift; - - return time64; + return timeNt.update(getTimeNowLowerNt()); } #endif /* !EFI_UNIT_TEST */ diff --git a/firmware/util/efitime.h b/firmware/util/efitime.h index a42bade031..b36c7e0794 100644 --- a/firmware/util/efitime.h +++ b/firmware/util/efitime.h @@ -51,6 +51,37 @@ #ifdef __cplusplus extern "C" { + +/** + * Provide a 62-bit counter from a 32-bit counter source that wraps around. + * + * If you'd like it use it with a 16-bit counter, shift the source by 16 before passing it here. + * This class is thread/interrupt-safe. + */ +struct WrapAround62 { + uint64_t update(uint32_t source) { + // Shift cannot be 31, as we wouldn't be able to tell if time is moving forward or + // backward relative to m_upper. We do need to handle both directions as our + // "thread" can be racing with other "threads" in sampling stamp and updating + // m_upper. + constexpr unsigned shift = 30; + + uint32_t upper = m_upper; + uint32_t relative_unsigned = source - (upper << shift); + upper += int32_t(relative_unsigned) >> shift; + m_upper = upper; + + // Yes we could just do upper<> (32 - shift)) << 32) | source; + } + +private: + volatile uint32_t m_upper = 0; +}; + #endif /* __cplusplus */ diff --git a/unit_tests/tests/test_util.cpp b/unit_tests/tests/test_util.cpp index 8ee1866e2e..7281a26c84 100644 --- a/unit_tests/tests/test_util.cpp +++ b/unit_tests/tests/test_util.cpp @@ -499,3 +499,69 @@ TEST(util, PeakDetect) { // Small value past the timeout is used EXPECT_EQ(dut.detect(500, startTime + timeout + 1), 500); } + +TEST(util, WrapAround62) { + // Random test + { + WrapAround62 t; + uint32_t source = 0; + uint64_t actual = 0; + + // Test random progression, positive and negative. + uint32_t seed = time(NULL); + printf("Testing with seed 0x%08x\n", seed); + srandom(seed); + for (unsigned i = 0; i < 10000; i++) { + int32_t delta = random(); + if (delta < 0) { + delta = ~delta; + } + delta -= RAND_MAX >> 1; + + // Cap negative test + if (delta < 0 && -delta > actual) { + delta = -actual; + } + + source += delta; + actual += delta; + + uint64_t next = t.update(source); + EXPECT_EQ(actual, next); + } + } + + // More pointed test for expected edge conditions + { + WrapAround62 t; + + EXPECT_EQ(t.update(0x03453455), 0x003453455LL); + EXPECT_EQ(t.update(0x42342323), 0x042342323LL); + EXPECT_EQ(t.update(0x84356345), 0x084356345LL); + EXPECT_EQ(t.update(0x42342323), 0x042342323LL); + EXPECT_EQ(t.update(0x84356345), 0x084356345LL); + EXPECT_EQ(t.update(0xC5656565), 0x0C5656565LL); + EXPECT_EQ(t.update(0x01122112), 0x101122112LL); // Wrap around! + EXPECT_EQ(t.update(0xC5656565), 0x0C5656565LL); + EXPECT_EQ(t.update(0x84356345), 0x084356345LL); + EXPECT_EQ(t.update(0xC5656565), 0x0C5656565LL); + EXPECT_EQ(t.update(0x01122112), 0x101122112LL); // Wrap around! + EXPECT_EQ(t.update(0x42342323), 0x142342323LL); + EXPECT_EQ(t.update(0x84356345), 0x184356345LL); + EXPECT_EQ(t.update(0x42342323), 0x142342323LL); + EXPECT_EQ(t.update(0x84356345), 0x184356345LL); + EXPECT_EQ(t.update(0xC5656565), 0x1C5656565LL); + EXPECT_EQ(t.update(0x01122112), 0x201122112LL); // Wrap around! + EXPECT_EQ(t.update(0xC5656565), 0x1C5656565LL); + EXPECT_EQ(t.update(0x84356345), 0x184356345LL); + EXPECT_EQ(t.update(0xC5656565), 0x1C5656565LL); + EXPECT_EQ(t.update(0x01122112), 0x201122112LL); // Wrap around! + EXPECT_EQ(t.update(0xC5656565), 0x1C5656565LL); + EXPECT_EQ(t.update(0x84356345), 0x184356345LL); + EXPECT_EQ(t.update(0x42342323), 0x142342323LL); + EXPECT_EQ(t.update(0x01122112), 0x101122112LL); + EXPECT_EQ(t.update(0x84356345), 0x084356345LL); + EXPECT_EQ(t.update(0x42342323), 0x042342323LL); + EXPECT_EQ(t.update(0x03453455), 0x003453455LL); + } +}