2015-07-10 06:01:56 -07:00
|
|
|
/**
|
|
|
|
* @file efitime.h
|
|
|
|
*
|
|
|
|
* By the way, there are 86400000 milliseconds in a day
|
|
|
|
*
|
|
|
|
* @date Apr 14, 2014
|
2020-01-13 18:57:43 -08:00
|
|
|
* @author Andrey Belomutskiy, (c) 2012-2020
|
2015-07-10 06:01:56 -07:00
|
|
|
*/
|
|
|
|
|
2020-01-20 22:40:11 -08:00
|
|
|
#pragma once
|
2015-07-10 06:01:56 -07:00
|
|
|
|
|
|
|
#include "efifeatures.h"
|
|
|
|
#include "rusefi_types.h"
|
|
|
|
|
2022-09-11 10:42:07 -07:00
|
|
|
#if EFI_PROD_CODE
|
2022-09-11 10:06:03 -07:00
|
|
|
// for US_TO_NT_MULTIPLIER which is port-specific
|
|
|
|
#include "port_mpu_util.h"
|
2022-09-11 10:42:07 -07:00
|
|
|
#endif
|
2022-09-11 10:06:03 -07:00
|
|
|
|
2022-04-10 16:22:39 -07:00
|
|
|
#define MS_PER_SECOND 1000
|
2015-07-10 06:01:56 -07:00
|
|
|
#define US_PER_SECOND 1000000
|
2019-09-08 18:05:03 -07:00
|
|
|
#define US_PER_SECOND_F 1000000.0
|
2015-07-10 06:01:56 -07:00
|
|
|
#define US_PER_SECOND_LL 1000000LL
|
2020-01-19 19:23:41 -08:00
|
|
|
#define NT_PER_SECOND (US2NT(US_PER_SECOND_LL))
|
2015-07-10 06:01:56 -07:00
|
|
|
|
|
|
|
#define MS2US(MS_TIME) ((MS_TIME) * 1000)
|
2021-11-11 13:39:23 -08:00
|
|
|
#define US2MS(US_TIME) ((US_TIME) / 1000)
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2020-11-23 20:33:46 -08:00
|
|
|
// microseconds to ticks
|
2022-09-11 13:08:11 -07:00
|
|
|
// since only about 20 seconds of ticks fit in 32 bits this macro is casting parameter into 64 bits 'efitick_t' type
|
2020-12-03 07:31:16 -08:00
|
|
|
// please note that int64 <-> float is a heavy operation thus we have 'USF2NT' below
|
2022-09-11 13:08:11 -07:00
|
|
|
#define US2NT(us) (((efitick_t)(us)) * US_TO_NT_MULTIPLIER)
|
2020-12-03 07:26:50 -08:00
|
|
|
|
|
|
|
// microseconds to ticks, but floating point
|
|
|
|
// If converting a floating point time period, use this macro to avoid
|
|
|
|
// the expensive conversions from int64 <-> float
|
2020-11-23 20:33:46 -08:00
|
|
|
#define USF2NT(us_float) ((us_float) * US_TO_NT_MULTIPLIER)
|
2021-07-25 17:42:26 -07:00
|
|
|
#define USF2MS(us_float) (0.001f * (us_float))
|
|
|
|
|
2020-11-23 20:33:46 -08:00
|
|
|
// And back
|
|
|
|
#define NT2US(x) ((x) / US_TO_NT_MULTIPLIER)
|
2022-02-03 08:24:31 -08:00
|
|
|
#define NT2USF(x) (((float)(x)) / US_TO_NT_MULTIPLIER)
|
2020-11-23 20:33:46 -08:00
|
|
|
|
2019-12-21 18:18:38 -08:00
|
|
|
// milliseconds to ticks
|
|
|
|
#define MS2NT(msTime) US2NT(MS2US(msTime))
|
2020-12-03 07:26:50 -08:00
|
|
|
// See USF2NT above for when to use MSF2NT
|
2020-11-23 20:33:46 -08:00
|
|
|
#define MSF2NT(msTimeFloat) USF2NT(MS2US(msTimeFloat))
|
2019-12-21 18:18:38 -08:00
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
#ifdef __cplusplus
|
2021-11-08 11:24:31 -08:00
|
|
|
/**
|
|
|
|
* 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<<shift, but then the result would span both halves of
|
|
|
|
// the 64-bit result. Doing it this way means we only operate on one half at a
|
|
|
|
// time. Source will supply those bits anyways, so we don't need them from
|
|
|
|
// upper...
|
|
|
|
return (efitick_t(upper >> (32 - shift)) << 32) | source;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
volatile uint32_t m_upper = 0;
|
|
|
|
};
|
|
|
|
|
2022-09-11 10:06:03 -07:00
|
|
|
/**
|
|
|
|
* Get a monotonically increasing (but wrapping) 32-bit timer value
|
|
|
|
* Implemented at port level, based on timer or CPU tick counter
|
|
|
|
* Main source of EFI clock, SW-extended to 64bits
|
|
|
|
*/
|
|
|
|
uint32_t getTimeNowLowerNt();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 64-bit counter CPU/timer cycles since MCU reset
|
|
|
|
*
|
|
|
|
* See getTimeNowLowerNt for a quicker version which returns only lower 32 bits
|
|
|
|
* Lower 32 bits are enough if all we need is to measure relatively short time durations
|
|
|
|
* (BTW 2^32 cpu cycles at 168MHz is 25.59 seconds)
|
|
|
|
*/
|
|
|
|
efitick_t getTimeNowNt();
|
2015-07-10 06:01:56 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* 64-bit counter of microseconds (1/1 000 000 of a second) since MCU reset
|
|
|
|
*
|
|
|
|
* By using 64 bit, we can achieve a very precise timestamp which does not overflow.
|
|
|
|
* The primary implementation counts the number of CPU cycles from MCU reset.
|
|
|
|
*
|
|
|
|
* WARNING: you should use getTimeNowNt where possible for performance reasons.
|
|
|
|
* The heaviest part is '__aeabi_ildivmod' - non-native 64 bit division
|
|
|
|
*/
|
2022-01-11 19:40:52 -08:00
|
|
|
efitimeus_t getTimeNowUs();
|
2015-07-10 06:01:56 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Returns the number of milliseconds since the board initialization.
|
|
|
|
*/
|
2022-09-11 10:06:03 -07:00
|
|
|
efitimems_t getTimeNowMs();
|
2015-07-10 06:01:56 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Current system time in seconds.
|
|
|
|
*/
|
2022-09-11 10:06:03 -07:00
|
|
|
efitimesec_t getTimeNowS();
|
2020-11-18 11:17:00 -08:00
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
#endif /* __cplusplus */
|
|
|
|
|