rusefi/firmware/controllers/system/timer/single_timer_executor.cpp

175 lines
5.0 KiB
C++
Raw Normal View History

2015-07-10 06:01:56 -07:00
/**
* @file SingleTimerExecutor.cpp
*
2019-05-05 06:41:44 -07:00
* This class combines the powers of a 1MHz hardware timer from microsecond_timer.cpp
2015-07-10 06:01:56 -07:00
* and pending events queue event_queue.cpp
*
* As of version 2.6.x, ChibiOS tick-based kernel is not capable of scheduling events
* with the level of precision we need, and realistically it should not.
*
2019-05-05 06:41:44 -07:00
* Update: actually newer ChibiOS has tickless mode and what we have here is pretty much the same thing :)
* open question if rusEfi should simply migrate to ChibiOS tickless scheduling (which would increase coupling with ChibiOS)
*
* See https://rusefi.com/forum/viewtopic.php?f=5&t=373&start=360#p30895
2018-12-24 16:36:03 -08:00
* for some performance data: with 'debug' firmware we spend about 5% of CPU in TIM5 handler which seem to be executed
* about 1500 times a second
*
2015-07-10 06:01:56 -07:00
* http://sourceforge.net/p/rusefi/tickets/24/
*
* @date: Apr 18, 2014
2020-01-13 18:57:43 -08:00
* @author Andrey Belomutskiy, (c) 2012-2020
2015-07-10 06:01:56 -07:00
*/
#include "pch.h"
2022-09-07 12:56:45 -07:00
2019-03-29 06:11:13 -07:00
#include "single_timer_executor.h"
2015-07-10 06:01:56 -07:00
#include "efitime.h"
2019-04-12 19:07:03 -07:00
#if EFI_SIGNAL_EXECUTOR_ONE_TIMER
2015-07-10 06:01:56 -07:00
#include "microsecond_timer.h"
2019-07-06 17:15:49 -07:00
#include "os_util.h"
2015-07-10 06:01:56 -07:00
2017-05-21 07:25:35 -07:00
uint32_t hwSetTimerDuration;
2015-07-10 06:01:56 -07:00
2020-04-24 18:21:04 -07:00
void globalTimerCallback() {
2023-06-24 23:08:53 -07:00
efiAssertVoid(ObdCode::CUSTOM_ERR_6624, hasLotsOfRemainingStack(), "lowstck#2y");
2015-07-10 06:01:56 -07:00
2024-07-11 17:03:13 -07:00
___engine.scheduler.onTimerCallback();
2015-07-10 06:01:56 -07:00
}
SingleTimerExecutor::SingleTimerExecutor()
// 8us is roughly the cost of the interrupt + overhead of a single timer event
: queue(US2NT(8))
{
2015-07-10 06:01:56 -07:00
}
2024-07-11 17:03:13 -07:00
void SingleTimerExecutor::schedule(const char *msg, scheduling_s* scheduling, efitick_t nt, action_s action) {
2019-10-13 13:14:08 -07:00
ScopePerf perf(PE::SingleTimerExecutorScheduleByTimestamp);
2020-01-18 21:28:58 -08:00
#if EFI_ENABLE_ASSERTS
efidur_t deltaTimeNt = nt - getTimeNowNt();
2020-01-18 21:28:58 -08:00
if (deltaTimeNt >= TOO_FAR_INTO_FUTURE_NT) {
2020-01-18 21:28:58 -08:00
// we are trying to set callback for too far into the future. This does not look right at all
int32_t intDeltaTimeNt = (int32_t)deltaTimeNt;
2024-07-11 17:03:13 -07:00
firmwareError(ObdCode::CUSTOM_ERR_TASK_TIMER_OVERFLOW, "schedule() too far: %ld %s", intDeltaTimeNt, msg);
2020-01-18 21:28:58 -08:00
return;
}
#endif
2017-06-07 19:55:05 -07:00
scheduleCounter++;
// Lock for queue insertion - we may already be locked, but that's ok
chibios_rt::CriticalSectionLocker csl;
bool needToResetTimer = queue.insertTask(scheduling, nt, action);
2015-07-10 06:01:56 -07:00
if (!reentrantFlag) {
executeAllPendingActions();
2015-07-10 06:01:56 -07:00
if (needToResetTimer) {
scheduleTimerCallback();
}
}
}
void SingleTimerExecutor::cancel(scheduling_s* scheduling) {
// Lock for queue removal - we may already be locked, but that's ok
chibios_rt::CriticalSectionLocker csl;
queue.remove(scheduling);
}
void SingleTimerExecutor::onTimerCallback() {
2017-06-07 19:55:05 -07:00
timerCallbackCounter++;
chibios_rt::CriticalSectionLocker csl;
executeAllPendingActions();
2015-07-10 06:01:56 -07:00
scheduleTimerCallback();
}
/*
* this private method is executed under lock
*/
void SingleTimerExecutor::executeAllPendingActions() {
2019-10-11 17:43:21 -07:00
ScopePerf perf(PE::SingleTimerExecutorDoExecute);
executeAllPendingActionsInvocationCounter++;
2015-07-10 06:01:56 -07:00
/**
* Let's execute actions we should execute at this point.
* reentrantFlag takes care of the use case where the actions we are executing are scheduling
* further invocations
*/
reentrantFlag = true;
2015-07-10 06:01:56 -07:00
/**
* in real life it could be that while we executing listeners time passes and it's already time to execute
* next listeners.
* TODO: add a counter & figure out a limit of iterations?
*/
2020-10-05 05:57:00 -07:00
// starts at -1 because do..while will run a minimum of once
executeCounter = -1;
2020-10-05 05:57:00 -07:00
bool didExecute;
do {
2015-07-10 06:01:56 -07:00
efitick_t nowNt = getTimeNowNt();
2020-10-05 05:57:00 -07:00
didExecute = queue.executeOne(nowNt);
// if we're stuck in a loop executing lots of events, panic!
if (executeCounter++ == 500) {
firmwareError(ObdCode::CUSTOM_ERR_LOCK_ISSUE, "Maximum scheduling run length exceeded - CPU load too high");
}
2020-10-05 05:57:00 -07:00
} while (didExecute);
maxExecuteCounter = maxI(maxExecuteCounter, executeCounter);
2020-10-05 05:57:00 -07:00
2015-07-10 06:01:56 -07:00
if (!isLocked()) {
firmwareError(ObdCode::CUSTOM_ERR_LOCK_ISSUE, "Someone has stolen my lock");
2015-07-10 06:01:56 -07:00
return;
}
reentrantFlag = false;
}
/**
* This method is always invoked under a lock
*/
void SingleTimerExecutor::scheduleTimerCallback() {
2019-10-13 13:14:08 -07:00
ScopePerf perf(PE::SingleTimerExecutorScheduleTimerCallback);
2015-07-10 06:01:56 -07:00
/**
* Let's grab fresh time value
*/
efitick_t nowNt = getTimeNowNt();
expected<efitick_t> nextEventTimeNt = queue.getNextEventTime(nowNt);
if (!nextEventTimeNt) {
2015-07-10 06:01:56 -07:00
return; // no pending events in the queue
}
efiAssertVoid(ObdCode::CUSTOM_ERR_6625, nextEventTimeNt.Value > nowNt, "setTimer constraint");
setHardwareSchedulerTimer(nowNt, nextEventTimeNt.Value);
2015-07-10 06:01:56 -07:00
}
2023-05-23 21:09:06 -07:00
void initSingleTimerExecutorHardware() {
2015-07-10 06:01:56 -07:00
initMicrosecondTimer();
}
2017-06-07 19:55:05 -07:00
void executorStatistics() {
2017-06-07 20:04:56 -07:00
if (engineConfiguration->debugMode == DBG_EXECUTOR) {
#if EFI_TUNER_STUDIO
2024-07-11 17:03:13 -07:00
engine->outputChannels.debugIntField1 = ___engine.scheduler.timerCallbackCounter;
engine->outputChannels.debugIntField2 = ___engine.scheduler.executeAllPendingActionsInvocationCounter;
engine->outputChannels.debugIntField3 = ___engine.scheduler.scheduleCounter;
engine->outputChannels.debugIntField4 = ___engine.scheduler.executeCounter;
engine->outputChannels.debugIntField5 = ___engine.scheduler.maxExecuteCounter;
#endif /* EFI_TUNER_STUDIO */
2017-06-07 19:55:05 -07:00
}
}
#endif /* EFI_SIGNAL_EXECUTOR_ONE_TIMER */
2015-07-10 06:01:56 -07:00