2015-07-10 06:01:56 -07:00
|
|
|
/**
|
|
|
|
* @file rpm_calculator.cpp
|
|
|
|
* @brief RPM calculator
|
|
|
|
*
|
|
|
|
* Here we listen to position sensor events in order to figure our if engine is currently running or not.
|
|
|
|
* Actual getRpm() is calculated once per crankshaft revolution, based on the amount of time passed
|
|
|
|
* since the start of previous shaft revolution.
|
|
|
|
*
|
|
|
|
* @date Jan 1, 2013
|
2018-01-20 17:55:31 -08:00
|
|
|
* @author Andrey Belomutskiy, (c) 2012-2018
|
2015-07-10 06:01:56 -07:00
|
|
|
*/
|
|
|
|
|
2018-09-16 19:26:57 -07:00
|
|
|
#include "global.h"
|
2015-07-10 06:01:56 -07:00
|
|
|
#include "rpm_calculator.h"
|
|
|
|
|
|
|
|
#if EFI_SHAFT_POSITION_INPUT || defined(__DOXYGEN__)
|
|
|
|
|
|
|
|
#include "trigger_central.h"
|
|
|
|
#include "engine_configuration.h"
|
|
|
|
#include "engine_math.h"
|
|
|
|
|
|
|
|
#if EFI_PROD_CODE
|
|
|
|
#include "rfiutil.h"
|
|
|
|
#include "engine.h"
|
|
|
|
#endif
|
|
|
|
|
2015-09-13 09:01:42 -07:00
|
|
|
#if EFI_SENSOR_CHART || defined(__DOXYGEN__)
|
2015-09-12 16:01:20 -07:00
|
|
|
#include "sensor_chart.h"
|
2015-07-10 06:01:56 -07:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "efilib2.h"
|
|
|
|
|
2015-07-15 18:01:45 -07:00
|
|
|
#if EFI_ENGINE_SNIFFER
|
|
|
|
#include "engine_sniffer.h"
|
2015-07-10 06:01:56 -07:00
|
|
|
extern WaveChart waveChart;
|
2015-07-15 18:01:45 -07:00
|
|
|
#endif /* EFI_ENGINE_SNIFFER */
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2018-03-03 05:55:19 -08:00
|
|
|
// See RpmCalculator::checkIfSpinning()
|
|
|
|
#ifndef NO_RPM_EVENTS_TIMEOUT_SECS
|
|
|
|
#define NO_RPM_EVENTS_TIMEOUT_SECS 2
|
|
|
|
#endif /* NO_RPM_EVENTS_TIMEOUT_SECS */
|
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
EXTERN_ENGINE
|
|
|
|
;
|
|
|
|
|
2016-06-01 17:01:36 -07:00
|
|
|
extern bool hasFirmwareErrorFlag;
|
2015-07-10 06:01:56 -07:00
|
|
|
|
|
|
|
static Logging * logger;
|
|
|
|
|
2016-09-04 22:03:25 -07:00
|
|
|
int revolutionCounterSinceBootForUnitTest = 0;
|
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
RpmCalculator::RpmCalculator() {
|
|
|
|
#if !EFI_PROD_CODE
|
|
|
|
mockRpm = MOCK_UNDEFINED;
|
2017-07-06 18:21:45 -07:00
|
|
|
#endif /* EFI_PROD_CODE */
|
2018-03-03 06:11:49 -08:00
|
|
|
// todo: reuse assignRpmValue() method which needs PASS_ENGINE_PARAMETER_SUFFIX
|
|
|
|
// which we cannot provide inside this parameter-less consutructor. need a solution for this minor mess
|
|
|
|
previousRpmValue = rpmValue = 0;
|
|
|
|
oneDegreeUs = NAN;
|
2017-07-08 08:02:28 -07:00
|
|
|
state = STOPPED;
|
2018-03-10 17:58:51 -08:00
|
|
|
isSpinning = false;
|
2015-07-10 06:01:56 -07:00
|
|
|
|
|
|
|
// we need this initial to have not_running at first invocation
|
|
|
|
lastRpmEventTimeNt = (efitime_t) -10 * US2NT(US_PER_SECOND_LL);
|
|
|
|
revolutionCounterSinceStart = 0;
|
2016-09-04 22:03:25 -07:00
|
|
|
revolutionCounterSinceBootForUnitTest = revolutionCounterSinceBoot = 0;
|
2015-07-10 06:01:56 -07:00
|
|
|
|
|
|
|
lastRpmEventTimeNt = 0;
|
|
|
|
oneDegreeUs = NAN;
|
|
|
|
}
|
|
|
|
|
2017-07-06 05:43:15 -07:00
|
|
|
bool RpmCalculator::isStopped(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
|
2018-03-10 17:58:51 -08:00
|
|
|
// Spinning-up with zero RPM means that the engine is not ready yet, and is treated as 'stopped'.
|
|
|
|
return state == STOPPED || (state == SPINNING_UP && rpmValue == 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool RpmCalculator::isSpinningUp(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
|
|
|
|
return state == SPINNING_UP;
|
2017-07-06 05:35:09 -07:00
|
|
|
}
|
|
|
|
|
2017-07-06 05:43:15 -07:00
|
|
|
bool RpmCalculator::isCranking(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
|
2018-03-10 17:58:51 -08:00
|
|
|
// Spinning-up with non-zero RPM is suitable for all engine math, as good as cranking
|
|
|
|
return state == CRANKING || (state == SPINNING_UP && rpmValue > 0);
|
2017-07-06 05:35:09 -07:00
|
|
|
}
|
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
/**
|
|
|
|
* @return true if there was a full shaft revolution within the last second
|
|
|
|
*/
|
2017-05-15 20:28:49 -07:00
|
|
|
bool RpmCalculator::isRunning(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
|
2017-07-07 05:10:06 -07:00
|
|
|
return state == RUNNING;
|
2017-07-06 17:10:34 -07:00
|
|
|
}
|
|
|
|
|
2019-01-05 20:33:04 -08:00
|
|
|
/**
|
|
|
|
* @return true if engine is spinning (cranking or running)
|
|
|
|
*/
|
2017-07-06 17:10:34 -07:00
|
|
|
bool RpmCalculator::checkIfSpinning(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
|
|
|
|
if (lastRpmEventTimeNt == 0) {
|
|
|
|
// here we assume 64 bit time does not overflow
|
|
|
|
// zero value is the default meaning no RPM events since reboot
|
|
|
|
return false;
|
|
|
|
}
|
2015-07-10 06:01:56 -07:00
|
|
|
efitick_t nowNt = getTimeNowNt();
|
2019-01-05 20:33:04 -08:00
|
|
|
if (ENGINE(needToStopEngine(nowNt))) {
|
|
|
|
setStopped(PASS_ENGINE_PARAMETER_SIGNATURE);
|
|
|
|
return false;
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
2017-07-06 17:10:34 -07:00
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
/**
|
|
|
|
* note that the result of this subtraction could be negative, that would happen if
|
|
|
|
* we have a trigger event between the time we've invoked 'getTimeNow' and here
|
|
|
|
*/
|
2018-03-03 05:55:19 -08:00
|
|
|
bool noRpmEventsForTooLong = nowNt - lastRpmEventTimeNt >= US2NT(NO_RPM_EVENTS_TIMEOUT_SECS * US_PER_SECOND_LL); // Anything below 60 rpm is not running
|
2018-03-10 17:58:51 -08:00
|
|
|
/**
|
|
|
|
* Also check if there were no trigger events
|
|
|
|
*/
|
|
|
|
bool noTriggerEventsForTooLong = nowNt - engine->triggerCentral.previousShaftEventTimeNt >= US2NT(US_PER_SECOND_LL);
|
|
|
|
if (noRpmEventsForTooLong || noTriggerEventsForTooLong) {
|
|
|
|
setStopSpinning(PASS_ENGINE_PARAMETER_SIGNATURE);
|
2017-07-06 17:10:34 -07:00
|
|
|
return false;
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
2017-07-06 17:10:34 -07:00
|
|
|
return true;
|
2017-07-06 16:33:25 -07:00
|
|
|
}
|
2016-03-15 19:03:43 -07:00
|
|
|
|
2018-03-03 05:55:19 -08:00
|
|
|
void RpmCalculator::assignRpmValue(int value DECLARE_ENGINE_PARAMETER_SUFFIX) {
|
2015-07-10 06:01:56 -07:00
|
|
|
previousRpmValue = rpmValue;
|
|
|
|
rpmValue = value;
|
|
|
|
if (rpmValue <= 0) {
|
|
|
|
oneDegreeUs = NAN;
|
|
|
|
} else {
|
|
|
|
oneDegreeUs = getOneDegreeTimeUs(rpmValue);
|
2018-03-10 17:58:51 -08:00
|
|
|
if (previousRpmValue == 0) {
|
|
|
|
/**
|
|
|
|
* this would make sure that we have good numbers for first cranking revolution
|
|
|
|
* #275 cranking could be improved
|
|
|
|
*/
|
|
|
|
ENGINE(periodicFastCallback(PASS_ENGINE_PARAMETER_SIGNATURE));
|
|
|
|
}
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-15 20:28:49 -07:00
|
|
|
void RpmCalculator::setRpmValue(int value DECLARE_ENGINE_PARAMETER_SUFFIX) {
|
2018-03-03 05:55:19 -08:00
|
|
|
assignRpmValue(value PASS_ENGINE_PARAMETER_SUFFIX);
|
2018-03-10 17:58:51 -08:00
|
|
|
spinning_state_e oldState = state;
|
|
|
|
// Change state
|
2017-07-07 05:10:06 -07:00
|
|
|
if (rpmValue == 0) {
|
|
|
|
state = STOPPED;
|
2017-07-08 10:42:14 -07:00
|
|
|
} else if (rpmValue >= CONFIG(cranking.rpm)) {
|
2017-07-07 05:10:06 -07:00
|
|
|
state = RUNNING;
|
2018-03-10 17:58:51 -08:00
|
|
|
} else if (state == STOPPED || state == SPINNING_UP) {
|
2017-07-08 12:46:34 -07:00
|
|
|
/**
|
|
|
|
* We are here if RPM is above zero but we have not seen running RPM yet.
|
|
|
|
* This gives us cranking hysteresis - a drop of RPM during running is still running, not cranking.
|
|
|
|
*/
|
2017-07-08 10:42:14 -07:00
|
|
|
state = CRANKING;
|
2017-07-07 05:10:06 -07:00
|
|
|
}
|
2018-03-10 17:58:51 -08:00
|
|
|
#if EFI_ENGINE_CONTROL || defined(__DOXYGEN__)
|
|
|
|
// This presumably fixes injection mode change for cranking-to-running transition.
|
|
|
|
// 'isSimultanious' flag should be updated for events if injection modes differ for cranking and running.
|
|
|
|
if (state != oldState) {
|
|
|
|
engine->injectionEvents.addFuelEvents(PASS_ENGINE_PARAMETER_SIGNATURE);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
spinning_state_e RpmCalculator::getState(void) {
|
|
|
|
return state;
|
2016-03-15 19:03:43 -07:00
|
|
|
}
|
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
void RpmCalculator::onNewEngineCycle() {
|
|
|
|
revolutionCounterSinceBoot++;
|
|
|
|
revolutionCounterSinceStart++;
|
2016-09-04 22:03:25 -07:00
|
|
|
#if EFI_UNIT_TEST
|
|
|
|
revolutionCounterSinceBootForUnitTest = revolutionCounterSinceBoot;
|
2017-05-25 19:37:43 -07:00
|
|
|
#endif /* EFI_UNIT_TEST */
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t RpmCalculator::getRevolutionCounter(void) {
|
|
|
|
return revolutionCounterSinceBoot;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t RpmCalculator::getRevolutionCounterSinceStart(void) {
|
|
|
|
return revolutionCounterSinceStart;
|
|
|
|
}
|
|
|
|
|
|
|
|
float RpmCalculator::getRpmAcceleration() {
|
|
|
|
return 1.0 * previousRpmValue / rpmValue;
|
|
|
|
}
|
|
|
|
|
2017-07-06 16:33:25 -07:00
|
|
|
void RpmCalculator::setStopped(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
|
|
|
|
revolutionCounterSinceStart = 0;
|
|
|
|
if (rpmValue != 0) {
|
2018-03-03 05:55:19 -08:00
|
|
|
assignRpmValue(0 PASS_ENGINE_PARAMETER_SUFFIX);
|
2017-07-06 17:10:34 -07:00
|
|
|
scheduleMsg(logger, "engine stopped");
|
2017-07-06 16:33:25 -07:00
|
|
|
}
|
2017-07-07 04:20:04 -07:00
|
|
|
state = STOPPED;
|
2017-07-06 16:33:25 -07:00
|
|
|
}
|
|
|
|
|
2018-03-10 17:58:51 -08:00
|
|
|
void RpmCalculator::setStopSpinning(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
|
|
|
|
isSpinning = false;
|
|
|
|
setStopped(PASS_ENGINE_PARAMETER_SIGNATURE);
|
|
|
|
}
|
|
|
|
|
|
|
|
void RpmCalculator::setSpinningUp(efitime_t nowNt DECLARE_ENGINE_PARAMETER_SUFFIX) {
|
2019-01-09 19:57:33 -08:00
|
|
|
if (!CONFIGB(isFasterEngineSpinUpEnabled))
|
2018-03-10 17:58:51 -08:00
|
|
|
return;
|
|
|
|
// Only a completely stopped and non-spinning engine can enter the spinning-up state.
|
|
|
|
if (isStopped(PASS_ENGINE_PARAMETER_SIGNATURE) && !isSpinning) {
|
|
|
|
state = SPINNING_UP;
|
|
|
|
isSpinning = true;
|
|
|
|
}
|
|
|
|
// update variables needed by early instant RPM calc.
|
|
|
|
if (isSpinningUp(PASS_ENGINE_PARAMETER_SIGNATURE)) {
|
|
|
|
engine->triggerCentral.triggerState.setLastEventTimeForInstantRpm(nowNt PASS_ENGINE_PARAMETER_SUFFIX);
|
|
|
|
}
|
|
|
|
// Update ignition pin indices if needed
|
|
|
|
prepareIgnitionPinIndices(getIgnitionMode(PASS_ENGINE_PARAMETER_SIGNATURE) PASS_ENGINE_PARAMETER_SUFFIX);
|
|
|
|
}
|
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
/**
|
|
|
|
* WARNING: this is a heavy method because 'getRpm()' is relatively heavy
|
|
|
|
*
|
|
|
|
* @return -1 in case of isNoisySignal(), current RPM otherwise
|
|
|
|
*/
|
|
|
|
// todo: migrate to float return result or add a float version? this would have with calculations
|
|
|
|
// todo: add a version which does not check time & saves time? need to profile
|
2017-05-15 20:28:49 -07:00
|
|
|
int RpmCalculator::getRpm(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
|
2015-07-10 06:01:56 -07:00
|
|
|
#if !EFI_PROD_CODE
|
2017-07-06 18:21:45 -07:00
|
|
|
if (mockRpm != MOCK_UNDEFINED) {
|
|
|
|
return mockRpm;
|
|
|
|
}
|
|
|
|
#endif /* EFI_PROD_CODE */
|
2015-07-10 06:01:56 -07:00
|
|
|
return rpmValue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Shaft position callback used by RPM calculation logic.
|
|
|
|
*
|
|
|
|
* This callback should always be the first of trigger callbacks because other callbacks depend of values
|
|
|
|
* updated here.
|
|
|
|
* This callback is invoked on interrupt thread.
|
|
|
|
*/
|
|
|
|
void rpmShaftPositionCallback(trigger_event_e ckpSignalType,
|
2017-05-15 20:28:49 -07:00
|
|
|
uint32_t index DECLARE_ENGINE_PARAMETER_SUFFIX) {
|
2015-07-10 06:01:56 -07:00
|
|
|
efitick_t nowNt = getTimeNowNt();
|
|
|
|
#if EFI_PROD_CODE
|
2018-07-25 20:03:04 -07:00
|
|
|
efiAssertVoid(CUSTOM_ERR_6632, getRemainingStack(chThdGetSelfX()) > 256, "lowstckRCL");
|
2015-07-10 06:01:56 -07:00
|
|
|
#endif
|
|
|
|
|
2018-03-10 17:58:51 -08:00
|
|
|
RpmCalculator *rpmState = &engine->rpmCalculator;
|
|
|
|
|
2017-05-25 19:49:40 -07:00
|
|
|
if (index == 0) {
|
|
|
|
ENGINE(m.beforeRpmCb) = GET_TIMESTAMP();
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2017-07-06 18:28:51 -07:00
|
|
|
bool hadRpmRecently = rpmState->checkIfSpinning(PASS_ENGINE_PARAMETER_SIGNATURE);
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2017-05-25 19:49:40 -07:00
|
|
|
if (hadRpmRecently) {
|
|
|
|
efitime_t diffNt = nowNt - rpmState->lastRpmEventTimeNt;
|
2015-07-10 06:01:56 -07:00
|
|
|
/**
|
|
|
|
* Four stroke cycle is two crankshaft revolutions
|
|
|
|
*
|
|
|
|
* We always do '* 2' because the event signal is already adjusted to 'per engine cycle'
|
|
|
|
* and each revolution of crankshaft consists of two engine cycles revolutions
|
|
|
|
*
|
|
|
|
*/
|
2017-05-25 19:49:40 -07:00
|
|
|
if (diffNt == 0) {
|
|
|
|
rpmState->setRpmValue(NOISY_RPM PASS_ENGINE_PARAMETER_SUFFIX);
|
|
|
|
} else {
|
|
|
|
int mult = (int)getEngineCycle(engineConfiguration->operationMode) / 360;
|
|
|
|
int rpm = (int) (60 * US2NT(US_PER_SECOND_LL) * mult / diffNt);
|
|
|
|
rpmState->setRpmValue(rpm > UNREALISTIC_RPM ? NOISY_RPM : rpm PASS_ENGINE_PARAMETER_SUFFIX);
|
|
|
|
}
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
2017-05-25 19:49:40 -07:00
|
|
|
rpmState->onNewEngineCycle();
|
|
|
|
rpmState->lastRpmEventTimeNt = nowNt;
|
|
|
|
ENGINE(m.rpmCbTime) = GET_TIMESTAMP() - ENGINE(m.beforeRpmCb);
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
2017-05-25 19:49:40 -07:00
|
|
|
|
|
|
|
|
2015-09-13 09:01:42 -07:00
|
|
|
#if EFI_SENSOR_CHART || defined(__DOXYGEN__)
|
2016-05-28 21:01:59 -07:00
|
|
|
// this 'index==0' case is here so that it happens after cycle callback so
|
|
|
|
// it goes into sniffer report into the first position
|
2016-01-30 19:03:36 -08:00
|
|
|
if (ENGINE(sensorChartMode) == SC_TRIGGER) {
|
2017-05-25 19:49:40 -07:00
|
|
|
angle_t crankAngle = getCrankshaftAngleNt(nowNt PASS_ENGINE_PARAMETER_SUFFIX);
|
|
|
|
int signal = 1000 * ckpSignalType + index;
|
2015-07-26 21:01:35 -07:00
|
|
|
scAddData(crankAngle, signal);
|
|
|
|
}
|
2015-07-10 06:01:56 -07:00
|
|
|
#endif
|
2017-05-25 19:49:40 -07:00
|
|
|
|
2018-03-10 17:58:51 -08:00
|
|
|
// Replace 'normal' RPM with instant RPM for the initial spin-up period
|
|
|
|
if (rpmState->isSpinningUp(PASS_ENGINE_PARAMETER_SIGNATURE)) {
|
|
|
|
int prevIndex;
|
|
|
|
int iRpm = engine->triggerCentral.triggerState.calculateInstantRpm(&prevIndex, nowNt PASS_ENGINE_PARAMETER_SUFFIX);
|
|
|
|
// validate instant RPM - we shouldn't skip the cranking state
|
|
|
|
iRpm = minI(iRpm, CONFIG(cranking.rpm) - 1);
|
|
|
|
rpmState->assignRpmValue(iRpm PASS_ENGINE_PARAMETER_SUFFIX);
|
|
|
|
#if 0
|
|
|
|
scheduleMsg(logger, "** RPM: idx=%d sig=%d iRPM=%d", index, ckpSignalType, iRpm);
|
|
|
|
#endif
|
|
|
|
}
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static scheduling_s tdcScheduler[2];
|
|
|
|
|
2016-08-09 21:04:24 -07:00
|
|
|
static char rpmBuffer[_MAX_FILLER];
|
2015-07-10 06:01:56 -07:00
|
|
|
|
|
|
|
#if (EFI_PROD_CODE || EFI_SIMULATOR) || defined(__DOXYGEN__)
|
|
|
|
/**
|
|
|
|
* This callback has nothing to do with actual engine control, it just sends a Top Dead Center mark to the dev console
|
|
|
|
* digital sniffer.
|
|
|
|
*/
|
|
|
|
static void onTdcCallback(void) {
|
2016-01-18 09:03:32 -08:00
|
|
|
itoa10(rpmBuffer, getRpmE(engine));
|
2018-09-10 19:29:43 -07:00
|
|
|
addEngineSnifferEvent(TOP_DEAD_CENTER_MESSAGE, (char* ) rpmBuffer);
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This trigger callback schedules the actual physical TDC callback in relation to trigger synchronization point.
|
|
|
|
*/
|
|
|
|
static void tdcMarkCallback(trigger_event_e ckpSignalType,
|
2017-05-15 20:28:49 -07:00
|
|
|
uint32_t index0 DECLARE_ENGINE_PARAMETER_SUFFIX) {
|
2015-07-10 06:01:56 -07:00
|
|
|
(void) ckpSignalType;
|
|
|
|
bool isTriggerSynchronizationPoint = index0 == 0;
|
2016-01-30 19:03:36 -08:00
|
|
|
if (isTriggerSynchronizationPoint && ENGINE(isEngineChartEnabled)) {
|
2015-07-10 06:01:56 -07:00
|
|
|
int revIndex2 = engine->rpmCalculator.getRevolutionCounter() % 2;
|
2017-05-25 19:45:12 -07:00
|
|
|
int rpm = ENGINE(rpmCalculator.getRpm(PASS_ENGINE_PARAMETER_SIGNATURE));
|
2018-08-31 18:11:22 -07:00
|
|
|
// todo: use tooth event-based scheduling, not just time-based scheduling
|
2015-07-10 06:01:56 -07:00
|
|
|
if (isValidRpm(rpm)) {
|
|
|
|
scheduleByAngle(rpm, &tdcScheduler[revIndex2], tdcPosition(),
|
|
|
|
(schfunc_t) onTdcCallback, NULL, &engine->rpmCalculator);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if EFI_PROD_CODE || EFI_SIMULATOR
|
|
|
|
int getRevolutionCounter() {
|
|
|
|
return engine->rpmCalculator.getRevolutionCounter();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return Current crankshaft angle, 0 to 720 for four-stroke
|
|
|
|
*/
|
2017-05-15 20:28:49 -07:00
|
|
|
float getCrankshaftAngleNt(efitime_t timeNt DECLARE_ENGINE_PARAMETER_SUFFIX) {
|
2015-07-10 06:01:56 -07:00
|
|
|
efitime_t timeSinceZeroAngleNt = timeNt
|
|
|
|
- engine->rpmCalculator.lastRpmEventTimeNt;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* even if we use 'getOneDegreeTimeUs' macros here, it looks like the
|
|
|
|
* compiler is not smart enough to figure out that "A / ( B / C)" could be optimized into
|
|
|
|
* "A * C / B" in order to replace a slower division with a faster multiplication.
|
|
|
|
*/
|
2017-05-15 20:28:49 -07:00
|
|
|
int rpm = engine->rpmCalculator.getRpm(PASS_ENGINE_PARAMETER_SIGNATURE);
|
2015-07-10 06:01:56 -07:00
|
|
|
return rpm == 0 ? NAN : timeSinceZeroAngleNt / getOneDegreeTimeNt(rpm);
|
|
|
|
}
|
|
|
|
|
|
|
|
void initRpmCalculator(Logging *sharedLogger, Engine *engine) {
|
|
|
|
logger = sharedLogger;
|
2016-06-01 17:01:36 -07:00
|
|
|
if (hasFirmwareError()) {
|
|
|
|
return;
|
|
|
|
}
|
2015-07-10 06:01:56 -07:00
|
|
|
#if (EFI_PROD_CODE || EFI_SIMULATOR) || defined(__DOXYGEN__)
|
|
|
|
|
|
|
|
addTriggerEventListener(tdcMarkCallback, "chart TDC mark", engine);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
addTriggerEventListener(rpmShaftPositionCallback, "rpm reporter", engine);
|
|
|
|
}
|
|
|
|
|
|
|
|
#if (EFI_PROD_CODE || EFI_SIMULATOR) || defined(__DOXYGEN__)
|
|
|
|
/**
|
|
|
|
* Schedules a callback 'angle' degree of crankshaft from now.
|
|
|
|
* The callback would be executed once after the duration of time which
|
|
|
|
* it takes the crankshaft to rotate to the specified angle.
|
|
|
|
*/
|
|
|
|
void scheduleByAngle(int rpm, scheduling_s *timer, angle_t angle,
|
2019-01-09 06:37:16 -08:00
|
|
|
schfunc_t callback, void *param, RpmCalculator *calc DECLARE_ENGINE_PARAMETER_SUFFIX) {
|
2018-09-10 19:10:55 -07:00
|
|
|
efiAssertVoid(CUSTOM_ANGLE_NAN, !cisnan(angle), "NaN angle?");
|
2018-07-25 20:03:04 -07:00
|
|
|
efiAssertVoid(CUSTOM_ERR_6634, isValidRpm(rpm), "RPM check expected");
|
2015-07-10 06:01:56 -07:00
|
|
|
float delayUs = calc->oneDegreeUs * angle;
|
2018-07-25 20:03:04 -07:00
|
|
|
efiAssertVoid(CUSTOM_ERR_6635, !cisnan(delayUs), "NaN delay?");
|
2019-01-09 06:37:16 -08:00
|
|
|
engine->executor.scheduleForLater(timer, (int) delayUs, callback, param);
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#else
|
|
|
|
RpmCalculator::RpmCalculator() {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* EFI_SHAFT_POSITION_INPUT */
|
|
|
|
|