2015-07-10 06:01:56 -07:00
|
|
|
/**
|
|
|
|
* @file SingleTimerExecutor.cpp
|
|
|
|
*
|
|
|
|
* This class combines the powers of a 1MHz hardware timer from microsecond_timer.c
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
* http://sourceforge.net/p/rusefi/tickets/24/
|
|
|
|
*
|
|
|
|
* @date: Apr 18, 2014
|
2018-01-20 17:55:31 -08:00
|
|
|
* @author Andrey Belomutskiy, (c) 2012-2018
|
2015-07-10 06:01:56 -07:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "SingleTimerExecutor.h"
|
|
|
|
#include "efitime.h"
|
|
|
|
#include "efilib2.h"
|
|
|
|
|
2017-06-07 19:55:05 -07:00
|
|
|
#if EFI_PROD_CODE || defined(__DOXYGEN__)
|
2015-07-10 06:01:56 -07:00
|
|
|
#include "microsecond_timer.h"
|
2017-06-07 19:55:05 -07:00
|
|
|
#include "tunerstudio_configuration.h"
|
2015-07-10 06:01:56 -07:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if (EFI_SIGNAL_EXECUTOR_ONE_TIMER && EFI_PROD_CODE )|| defined(__DOXYGEN__)
|
|
|
|
#include "rfiutil.h"
|
|
|
|
|
|
|
|
static Executor instance;
|
|
|
|
|
|
|
|
extern schfunc_t globalTimerCallback;
|
|
|
|
|
|
|
|
//static int timerIsLate = 0;
|
|
|
|
//static efitime_t callbackTime = 0;
|
|
|
|
/**
|
|
|
|
* these fields are global in order to facilitate debugging
|
|
|
|
*/
|
|
|
|
static efitime_t nextEventTimeNt = 0;
|
|
|
|
|
2017-05-21 07:25:35 -07:00
|
|
|
uint32_t hwSetTimerDuration;
|
2015-07-10 06:01:56 -07:00
|
|
|
uint32_t lastExecutionCount;
|
|
|
|
|
|
|
|
static void executorCallback(void *arg) {
|
|
|
|
(void)arg;
|
2017-03-27 19:52:43 -07:00
|
|
|
efiAssertVoid(getRemainingStack(chThdGetSelfX()) > 256, "lowstck#2y");
|
2015-07-10 06:01:56 -07:00
|
|
|
|
|
|
|
// callbackTime = getTimeNowNt();
|
|
|
|
// if((callbackTime > nextEventTimeNt) && (callbackTime - nextEventTimeNt > US2NT(5000))) {
|
|
|
|
// timerIsLate++;
|
|
|
|
// }
|
|
|
|
|
|
|
|
instance.onTimerCallback();
|
|
|
|
}
|
|
|
|
|
|
|
|
Executor::Executor() {
|
|
|
|
reentrantFlag = false;
|
2017-06-07 19:55:05 -07:00
|
|
|
doExecuteCounter = scheduleCounter = timerCallbackCounter = 0;
|
2015-07-10 06:01:56 -07:00
|
|
|
/**
|
|
|
|
* todo: a good comment
|
|
|
|
*/
|
|
|
|
queue.setLateDelay(US2NT(100));
|
|
|
|
}
|
|
|
|
|
2018-01-28 08:27:33 -08:00
|
|
|
void Executor::scheduleByTimestamp(scheduling_s *scheduling, efitimeus_t timeUs, schfunc_t callback,
|
2015-07-10 06:01:56 -07:00
|
|
|
void *param) {
|
2017-06-07 19:55:05 -07:00
|
|
|
scheduleCounter++;
|
2015-07-10 06:01:56 -07:00
|
|
|
// if (delayUs < 0) {
|
2016-10-10 11:02:17 -07:00
|
|
|
// firmwareError(OBD_PCM_Processor_Fault, "Negative delayUs %s: %d", prefix, delayUs);
|
2015-07-10 06:01:56 -07:00
|
|
|
// return;
|
|
|
|
// }
|
|
|
|
// if (delayUs == 0) {
|
|
|
|
// callback(param);
|
|
|
|
// return;
|
|
|
|
// }
|
2017-05-25 11:51:21 -07:00
|
|
|
bool alreadyLocked = true;
|
2015-07-10 06:01:56 -07:00
|
|
|
if (!reentrantFlag) {
|
|
|
|
// this would guard the queue and disable interrupts
|
2017-05-25 11:51:21 -07:00
|
|
|
alreadyLocked = lockAnyContext();
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
2016-01-11 14:01:33 -08:00
|
|
|
bool needToResetTimer = queue.insertTask(scheduling, US2NT(timeUs), callback, param);
|
2015-07-10 06:01:56 -07:00
|
|
|
if (!reentrantFlag) {
|
|
|
|
doExecute();
|
|
|
|
if (needToResetTimer) {
|
|
|
|
scheduleTimerCallback();
|
|
|
|
}
|
2017-05-25 11:51:21 -07:00
|
|
|
if (!alreadyLocked)
|
|
|
|
unlockAnyContext();
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Executor::onTimerCallback() {
|
2017-06-07 19:55:05 -07:00
|
|
|
timerCallbackCounter++;
|
2017-05-25 11:51:21 -07:00
|
|
|
bool alreadyLocked = lockAnyContext();
|
2015-07-10 06:01:56 -07:00
|
|
|
doExecute();
|
|
|
|
scheduleTimerCallback();
|
2017-05-25 11:51:21 -07:00
|
|
|
if (!alreadyLocked)
|
|
|
|
unlockAnyContext();
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* this private method is executed under lock
|
|
|
|
*/
|
|
|
|
void Executor::doExecute() {
|
2017-06-07 19:55:05 -07:00
|
|
|
doExecuteCounter++;
|
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;
|
|
|
|
int shouldExecute = 1;
|
|
|
|
/**
|
|
|
|
* 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?
|
|
|
|
*/
|
|
|
|
int totalExecuted = 0;
|
|
|
|
while (shouldExecute > 0) {
|
|
|
|
/**
|
|
|
|
* It's worth noting that that the actions might be adding new actions into the queue
|
|
|
|
*/
|
|
|
|
efitick_t nowNt = getTimeNowNt();
|
|
|
|
shouldExecute = queue.executeAll(nowNt);
|
|
|
|
totalExecuted += shouldExecute;
|
|
|
|
}
|
|
|
|
lastExecutionCount = totalExecuted;
|
|
|
|
if (!isLocked()) {
|
2017-05-21 07:15:57 -07:00
|
|
|
firmwareError(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 Executor::scheduleTimerCallback() {
|
|
|
|
/**
|
|
|
|
* Let's grab fresh time value
|
|
|
|
*/
|
|
|
|
efitick_t nowNt = getTimeNowNt();
|
|
|
|
nextEventTimeNt = queue.getNextEventTime(nowNt);
|
|
|
|
efiAssertVoid(nextEventTimeNt > nowNt, "setTimer constraint");
|
|
|
|
if (nextEventTimeNt == EMPTY_QUEUE)
|
|
|
|
return; // no pending events in the queue
|
|
|
|
int32_t hwAlarmTime = NT2US((int32_t)nextEventTimeNt - (int32_t)nowNt);
|
2017-05-21 07:25:35 -07:00
|
|
|
uint32_t beforeHwSetTimer = GET_TIMESTAMP();
|
2015-07-10 06:01:56 -07:00
|
|
|
setHardwareUsTimer(hwAlarmTime == 0 ? 1 : hwAlarmTime);
|
2017-05-21 07:25:35 -07:00
|
|
|
hwSetTimerDuration = GET_TIMESTAMP() - beforeHwSetTimer;
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2018-01-28 08:27:33 -08:00
|
|
|
* @brief Schedule an event at specific delay after now
|
2015-07-10 06:01:56 -07:00
|
|
|
*
|
|
|
|
* Invokes event callback after the specified amount of time.
|
|
|
|
*
|
|
|
|
* @param [in, out] scheduling Data structure to keep this event in the collection.
|
|
|
|
* @param [in] delayUs the number of microseconds before the output signal immediate output if delay is zero.
|
|
|
|
* @param [in] dwell the number of ticks of output duration.
|
|
|
|
*/
|
2018-01-28 08:27:33 -08:00
|
|
|
void scheduleForLater(scheduling_s *scheduling, int delayUs, schfunc_t callback, void *param) {
|
|
|
|
instance.scheduleByTimestamp(scheduling, getTimeNowUs() + delayUs, callback, param);
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
2018-01-28 08:27:33 -08:00
|
|
|
/**
|
|
|
|
* @brief Schedule an event at specified timestamp
|
|
|
|
*
|
|
|
|
* @param [in] timeUs absolute time of the event, since ECU boot
|
|
|
|
*/
|
|
|
|
void scheduleByTimestamp(scheduling_s *scheduling, efitimeus_t time, schfunc_t callback, void *param) {
|
|
|
|
instance.scheduleByTimestamp(scheduling, time, callback, param);
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void initSignalExecutorImpl(void) {
|
|
|
|
globalTimerCallback = executorCallback;
|
|
|
|
initMicrosecondTimer();
|
|
|
|
}
|
|
|
|
|
2017-06-07 19:55:05 -07:00
|
|
|
extern TunerStudioOutputChannels tsOutputChannels;
|
|
|
|
#include "engine.h"
|
|
|
|
EXTERN_ENGINE;
|
|
|
|
|
|
|
|
|
|
|
|
void executorStatistics() {
|
2017-06-07 20:04:56 -07:00
|
|
|
if (engineConfiguration->debugMode == DBG_EXECUTOR) {
|
2017-06-07 19:55:05 -07:00
|
|
|
tsOutputChannels.debugIntField1 = instance.timerCallbackCounter;
|
|
|
|
tsOutputChannels.debugIntField2 = instance.doExecuteCounter;
|
|
|
|
tsOutputChannels.debugIntField3 = instance.scheduleCounter;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* EFI_SIGNAL_EXECUTOR_ONE_TIMER */
|
2015-07-10 06:01:56 -07:00
|
|
|
|