2015-07-10 06:01:56 -07:00
|
|
|
/**
|
|
|
|
* @file microsecond_timer.cpp
|
|
|
|
*
|
|
|
|
* Here we have a 1MHz timer dedicated to event scheduling. We are using one of the 32-bit timers here,
|
|
|
|
* so this timer can schedule events up to 4B/100M ~ 4000 seconds ~ 1 hour from current time.
|
|
|
|
*
|
|
|
|
* GPT5 timer clock: 84000000Hz
|
|
|
|
* If only it was a better multiplier of 2 (84000000 = 328125 * 256)
|
|
|
|
*
|
|
|
|
* @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
|
|
|
*/
|
|
|
|
|
2021-08-03 19:05:01 -07:00
|
|
|
#include "pch.h"
|
2015-07-10 06:01:56 -07:00
|
|
|
#include "microsecond_timer.h"
|
2020-11-22 17:39:32 -08:00
|
|
|
#include "port_microsecond_timer.h"
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2020-11-22 17:39:32 -08:00
|
|
|
#if EFI_PROD_CODE
|
2019-03-31 14:44:34 -07:00
|
|
|
|
2019-07-13 08:08:17 -07:00
|
|
|
#include "periodic_task.h"
|
2021-08-03 19:05:01 -07:00
|
|
|
|
2019-11-10 20:54:08 -08:00
|
|
|
// Just in case we have a mechanism to validate that hardware timer is clocked right and all the
|
|
|
|
// conversions between wall clock and hardware frequencies are done right
|
|
|
|
// delay in milliseconds
|
2021-03-29 19:59:38 -07:00
|
|
|
#define TEST_CALLBACK_DELAY 10
|
2020-03-28 17:56:58 -07:00
|
|
|
// if hardware timer is 20% off we throw a critical error and call it a day
|
2019-11-10 20:54:08 -08:00
|
|
|
// maybe this threshold should be 5%? 10%?
|
|
|
|
#define TIMER_PRECISION_THRESHOLD 0.2
|
|
|
|
|
2017-05-21 07:25:35 -07:00
|
|
|
/**
|
|
|
|
* Maximum duration of complete timer callback, all pending events together
|
|
|
|
* See also 'maxEventCallbackDuration' for maximum duration of one event
|
|
|
|
*/
|
|
|
|
uint32_t maxPrecisionCallbackDuration = 0;
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2020-11-22 17:39:32 -08:00
|
|
|
static efitick_t lastSetTimerTimeNt;
|
|
|
|
static bool isTimerPending = false;
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2020-11-22 17:39:32 -08:00
|
|
|
static int timerCallbackCounter = 0;
|
|
|
|
static int timerRestartCounter = 0;
|
2015-07-10 06:01:56 -07:00
|
|
|
|
|
|
|
static const char * msg;
|
|
|
|
|
2016-01-30 19:03:36 -08:00
|
|
|
static int timerFreezeCounter = 0;
|
2020-11-22 17:39:32 -08:00
|
|
|
static int setHwTimerCounter = 0;
|
|
|
|
static bool hwStarted = false;
|
2016-01-30 19:03:36 -08:00
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
/**
|
|
|
|
* sets the alarm to the specified number of microseconds from now.
|
|
|
|
* This function should be invoked under kernel lock which would disable interrupts.
|
|
|
|
*/
|
2020-11-16 15:50:13 -08:00
|
|
|
void setHardwareSchedulerTimer(efitick_t nowNt, efitick_t setTimeNt) {
|
2019-02-27 14:12:52 -08:00
|
|
|
efiAssertVoid(OBD_PCM_Processor_Fault, hwStarted, "HW.started");
|
2020-10-05 10:53:34 -07:00
|
|
|
|
2020-11-22 17:39:32 -08:00
|
|
|
// How many ticks in the future is this event?
|
|
|
|
auto timeDeltaNt = setTimeNt - nowNt;
|
2020-11-16 15:50:13 -08:00
|
|
|
|
2019-01-11 14:40:44 -08:00
|
|
|
setHwTimerCounter++;
|
2020-10-05 10:53:34 -07:00
|
|
|
|
2016-01-30 19:03:36 -08:00
|
|
|
/**
|
2020-11-22 17:39:32 -08:00
|
|
|
* #259 BUG error: not positive deltaTimeNt
|
2016-01-30 19:03:36 -08:00
|
|
|
* Once in a while we night get an interrupt where we do not expect it
|
|
|
|
*/
|
2020-11-22 17:39:32 -08:00
|
|
|
if (timeDeltaNt <= 0) {
|
2016-01-30 19:03:36 -08:00
|
|
|
timerFreezeCounter++;
|
2017-03-08 22:10:33 -08:00
|
|
|
warning(CUSTOM_OBD_LOCAL_FREEZE, "local freeze cnt=%d", timerFreezeCounter);
|
2016-01-30 19:03:36 -08:00
|
|
|
}
|
2020-10-05 10:53:34 -07:00
|
|
|
|
|
|
|
// We need the timer to fire after we return - 1 doesn't work as it may actually schedule in the past
|
2020-11-22 17:39:32 -08:00
|
|
|
if (timeDeltaNt < US2NT(2)) {
|
|
|
|
timeDeltaNt = US2NT(2);
|
2020-10-05 10:53:34 -07:00
|
|
|
}
|
|
|
|
|
2020-11-22 17:39:32 -08:00
|
|
|
if (timeDeltaNt >= TOO_FAR_INTO_FUTURE_NT) {
|
2020-01-18 21:16:19 -08:00
|
|
|
// we are trying to set callback for too far into the future. This does not look right at all
|
2020-11-22 17:39:32 -08:00
|
|
|
firmwareError(CUSTOM_ERR_TIMER_OVERFLOW, "setHardwareSchedulerTimer() too far: %d", timeDeltaNt);
|
2018-05-20 10:36:15 -07:00
|
|
|
return;
|
|
|
|
}
|
2020-10-05 10:53:34 -07:00
|
|
|
|
2020-11-22 17:39:32 -08:00
|
|
|
// Skip scheduling if there's a firmware error active
|
2019-11-10 09:39:47 -08:00
|
|
|
if (hasFirmwareError()) {
|
2016-01-18 16:01:48 -08:00
|
|
|
return;
|
2019-11-10 09:39:47 -08:00
|
|
|
}
|
2020-10-05 10:53:34 -07:00
|
|
|
|
2020-11-22 17:39:32 -08:00
|
|
|
// Do the actual hardware-specific timer set operation
|
|
|
|
portSetHardwareSchedulerTimer(nowNt, setTimeNt);
|
2015-07-10 06:01:56 -07:00
|
|
|
|
|
|
|
lastSetTimerTimeNt = getTimeNowNt();
|
2020-04-24 18:21:04 -07:00
|
|
|
isTimerPending = true;
|
2015-07-10 06:01:56 -07:00
|
|
|
timerRestartCounter++;
|
|
|
|
}
|
|
|
|
|
2020-04-24 18:21:04 -07:00
|
|
|
void globalTimerCallback();
|
|
|
|
|
2020-11-22 17:39:32 -08:00
|
|
|
void portMicrosecondTimerCallback() {
|
2015-07-10 06:01:56 -07:00
|
|
|
timerCallbackCounter++;
|
|
|
|
isTimerPending = false;
|
|
|
|
|
2019-05-07 16:32:08 -07:00
|
|
|
uint32_t before = getTimeNowLowerNt();
|
2020-04-24 18:21:04 -07:00
|
|
|
globalTimerCallback();
|
2019-05-07 16:32:08 -07:00
|
|
|
uint32_t precisionCallbackDuration = getTimeNowLowerNt() - before;
|
2017-05-21 07:25:35 -07:00
|
|
|
if (precisionCallbackDuration > maxPrecisionCallbackDuration) {
|
|
|
|
maxPrecisionCallbackDuration = precisionCallbackDuration;
|
2017-05-21 07:03:17 -07:00
|
|
|
}
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
2019-07-13 08:08:17 -07:00
|
|
|
class MicrosecondTimerWatchdogController : public PeriodicTimerController {
|
|
|
|
void PeriodicTask() override {
|
2019-12-21 18:11:09 -08:00
|
|
|
efitick_t nowNt = getTimeNowNt();
|
2019-01-06 18:15:49 -08:00
|
|
|
if (nowNt >= lastSetTimerTimeNt + 2 * CORE_CLOCK) {
|
2021-11-24 17:34:24 -08:00
|
|
|
firmwareError(CUSTOM_ERR_SCHEDULING_ERROR, "watchdog: no events since %d", lastSetTimerTimeNt);
|
2019-01-06 18:15:49 -08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
msg = isTimerPending ? "No_cb too long" : "Timer not awhile";
|
|
|
|
// 2 seconds of inactivity would not look right
|
2021-10-02 22:30:42 -07:00
|
|
|
efiAssertVoid(CUSTOM_TIMER_WATCHDOG, nowNt < lastSetTimerTimeNt + 2 * CORE_CLOCK, msg);
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
2019-07-13 08:08:17 -07:00
|
|
|
|
|
|
|
int getPeriodMs() override {
|
|
|
|
return 500;
|
|
|
|
}
|
2019-01-06 18:15:49 -08:00
|
|
|
};
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2019-02-10 16:52:06 -08:00
|
|
|
static MicrosecondTimerWatchdogController watchdogControllerInstance;
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2019-07-29 19:47:39 -07:00
|
|
|
static scheduling_s watchDogBuddy;
|
|
|
|
|
2020-10-05 10:53:34 -07:00
|
|
|
static void watchDogBuddyCallback(void*) {
|
2019-07-29 19:47:39 -07:00
|
|
|
/**
|
|
|
|
* the purpose of this periodic activity is to make watchdogControllerInstance
|
|
|
|
* watchdog happy by ensuring that we have scheduler activity even in case of very broken configuration
|
|
|
|
* without any PWM or input pins
|
|
|
|
*/
|
2022-05-14 18:24:19 -07:00
|
|
|
engine->executor.scheduleForLater("watch", &watchDogBuddy, MS2US(1000), watchDogBuddyCallback);
|
2019-07-29 19:47:39 -07:00
|
|
|
}
|
|
|
|
|
2019-11-10 20:54:08 -08:00
|
|
|
static volatile bool testSchedulingHappened = false;
|
|
|
|
static efitimems_t testSchedulingStart;
|
|
|
|
|
2020-10-05 10:53:34 -07:00
|
|
|
static void timerValidationCallback(void*) {
|
2019-11-10 20:54:08 -08:00
|
|
|
testSchedulingHappened = true;
|
|
|
|
efitimems_t actualTimeSinceScheduling = (currentTimeMillis() - testSchedulingStart);
|
2020-09-07 11:09:00 -07:00
|
|
|
|
2019-11-10 20:54:08 -08:00
|
|
|
if (absI(actualTimeSinceScheduling - TEST_CALLBACK_DELAY) > TEST_CALLBACK_DELAY * TIMER_PRECISION_THRESHOLD) {
|
2020-09-07 11:09:00 -07:00
|
|
|
firmwareError(CUSTOM_ERR_TIMER_TEST_CALLBACK_WRONG_TIME, "hwTimer broken precision: %ld ms", actualTimeSinceScheduling);
|
2019-11-10 20:54:08 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This method would validate that hardware timer callbacks happen with some reasonable precision
|
|
|
|
* helps to make sure our GPT hardware settings are somewhat right
|
|
|
|
*/
|
|
|
|
static void validateHardwareTimer() {
|
2019-11-12 19:56:56 -08:00
|
|
|
if (hasFirmwareError()) {
|
|
|
|
return;
|
|
|
|
}
|
2019-11-10 20:54:08 -08:00
|
|
|
testSchedulingStart = currentTimeMillis();
|
|
|
|
|
|
|
|
// to save RAM let's use 'watchDogBuddy' here once before we enable watchdog
|
2022-05-14 18:24:19 -07:00
|
|
|
engine->executor.scheduleForLater("hw-validate", &watchDogBuddy, MS2US(TEST_CALLBACK_DELAY), timerValidationCallback);
|
2019-11-10 20:54:08 -08:00
|
|
|
|
2021-03-29 19:59:38 -07:00
|
|
|
chThdSleepMilliseconds(TEST_CALLBACK_DELAY + 2);
|
2019-11-10 20:54:08 -08:00
|
|
|
if (!testSchedulingHappened) {
|
|
|
|
firmwareError(CUSTOM_ERR_TIMER_TEST_CALLBACK_NOT_HAPPENED, "hwTimer not alive");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-16 15:50:13 -08:00
|
|
|
void initMicrosecondTimer() {
|
2020-11-22 17:39:32 -08:00
|
|
|
portInitMicrosecondTimer();
|
|
|
|
|
2019-02-27 14:12:52 -08:00
|
|
|
hwStarted = true;
|
2015-07-10 06:01:56 -07:00
|
|
|
|
|
|
|
lastSetTimerTimeNt = getTimeNowNt();
|
2019-07-29 19:47:39 -07:00
|
|
|
|
2019-11-12 18:42:44 -08:00
|
|
|
validateHardwareTimer();
|
2019-11-10 20:54:08 -08:00
|
|
|
|
2019-07-29 19:47:39 -07:00
|
|
|
watchDogBuddyCallback(NULL);
|
2015-07-10 06:01:56 -07:00
|
|
|
#if EFI_EMULATE_POSITION_SENSORS
|
2019-01-06 18:15:49 -08:00
|
|
|
watchdogControllerInstance.Start();
|
2019-07-13 08:08:17 -07:00
|
|
|
#endif /* EFI_EMULATE_POSITION_SENSORS */
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* EFI_PROD_CODE */
|