2015-07-10 06:01:56 -07:00
|
|
|
/**
|
|
|
|
* @file engine.cpp
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* This might be a http://en.wikipedia.org/wiki/God_object but that's best way I can
|
|
|
|
* express myself in C/C++. I am open for suggestions :)
|
|
|
|
*
|
|
|
|
* @date May 21, 2014
|
2020-01-07 21:02:40 -08:00
|
|
|
* @author Andrey Belomutskiy, (c) 2012-2020
|
2015-07-10 06:01:56 -07:00
|
|
|
*/
|
|
|
|
|
2021-07-25 18:23:23 -07:00
|
|
|
#include "pch.h"
|
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
#include "trigger_central.h"
|
|
|
|
#include "fuel_math.h"
|
|
|
|
#include "advance_map.h"
|
|
|
|
#include "speed_density.h"
|
2016-01-01 16:02:59 -08:00
|
|
|
#include "advance_map.h"
|
2022-09-07 12:56:45 -07:00
|
|
|
|
2017-11-26 19:30:37 -08:00
|
|
|
#include "aux_valves.h"
|
2018-02-06 12:47:19 -08:00
|
|
|
#include "map_averaging.h"
|
2019-10-11 17:43:21 -07:00
|
|
|
#include "perf_trace.h"
|
2020-09-07 12:08:54 -07:00
|
|
|
#include "backup_ram.h"
|
2020-11-11 18:47:19 -08:00
|
|
|
#include "idle_thread.h"
|
2020-11-03 15:06:32 -08:00
|
|
|
#include "idle_hardware.h"
|
2020-04-26 11:06:28 -07:00
|
|
|
#include "gppwm.h"
|
2020-08-21 12:36:43 -07:00
|
|
|
#include "tachometer.h"
|
2020-12-04 17:28:48 -08:00
|
|
|
#include "dynoview.h"
|
2021-01-18 04:04:14 -08:00
|
|
|
#include "boost_control.h"
|
2021-06-11 03:25:12 -07:00
|
|
|
#include "fan_control.h"
|
2021-06-27 15:51:34 -07:00
|
|
|
#include "ac_control.h"
|
2021-07-28 04:44:44 -07:00
|
|
|
#include "vr_pwm.h"
|
2020-10-19 19:04:06 -07:00
|
|
|
#if EFI_MC33816
|
|
|
|
#include "mc33816.h"
|
|
|
|
#endif // EFI_MC33816
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2019-04-12 19:07:03 -07:00
|
|
|
#if EFI_PROD_CODE
|
2020-10-11 19:46:08 -07:00
|
|
|
#include "trigger_emulator_algo.h"
|
2020-03-26 05:03:55 -07:00
|
|
|
#include "bench_test.h"
|
2015-07-10 06:01:56 -07:00
|
|
|
#else
|
|
|
|
#define isRunningBenchTest() true
|
2016-01-23 15:01:40 -08:00
|
|
|
#endif /* EFI_PROD_CODE */
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2019-09-06 17:30:27 -07:00
|
|
|
#if (BOARD_TLE8888_COUNT > 0)
|
|
|
|
#include "gpio/tle8888.h"
|
|
|
|
#endif
|
|
|
|
|
2019-12-21 17:43:11 -08:00
|
|
|
#if EFI_ENGINE_SNIFFER
|
|
|
|
#include "engine_sniffer.h"
|
|
|
|
extern int waveChartUsedSize;
|
|
|
|
extern WaveChart waveChart;
|
|
|
|
#endif /* EFI_ENGINE_SNIFFER */
|
|
|
|
|
|
|
|
void Engine::resetEngineSnifferIfInTestMode() {
|
|
|
|
#if EFI_ENGINE_SNIFFER
|
2020-08-08 08:25:17 -07:00
|
|
|
if (isFunctionalTestMode) {
|
2019-12-21 20:27:54 -08:00
|
|
|
// TODO: what is the exact reasoning for the exact engine sniffer pause time I wonder
|
|
|
|
waveChart.pauseEngineSnifferUntilNt = getTimeNowNt() + MS2NT(300);
|
2019-12-21 17:43:11 -08:00
|
|
|
waveChart.reset();
|
|
|
|
}
|
|
|
|
#endif /* EFI_ENGINE_SNIFFER */
|
|
|
|
}
|
|
|
|
|
2021-10-24 14:05:37 -07:00
|
|
|
/**
|
|
|
|
* VVT decoding delegates to universal trigger decoder. Here we map vvt_mode_e into corresponding trigger_type_e
|
|
|
|
*/
|
2020-08-26 10:30:55 -07:00
|
|
|
trigger_type_e getVvtTriggerType(vvt_mode_e vvtMode) {
|
|
|
|
switch (vvtMode) {
|
2021-12-31 12:47:25 -08:00
|
|
|
case VVT_INACTIVE:
|
|
|
|
return TT_ONE;
|
2020-08-26 10:30:55 -07:00
|
|
|
case VVT_2JZ:
|
|
|
|
return TT_VVT_JZ;
|
2022-01-31 15:20:43 -08:00
|
|
|
case VVT_MIATA_NB:
|
|
|
|
return TT_VVT_MIATA_NB;
|
2020-08-26 21:43:23 -07:00
|
|
|
case VVT_BOSCH_QUICK_START:
|
|
|
|
return TT_VVT_BOSCH_QUICK_START;
|
2022-09-24 20:43:00 -07:00
|
|
|
case VVT_HONDA_K_EXHAUST:
|
|
|
|
return TT_HONDA_K_CAM_4_1;
|
2022-09-24 20:25:36 -07:00
|
|
|
case VVT_HONDA_K_INTAKE:
|
2020-08-26 10:30:55 -07:00
|
|
|
case VVT_FIRST_HALF:
|
|
|
|
case VVT_SECOND_HALF:
|
2022-08-31 21:18:06 -07:00
|
|
|
case VVT_MAP_V_TWIN:
|
2020-08-26 10:30:55 -07:00
|
|
|
return TT_ONE;
|
2020-12-03 20:54:08 -08:00
|
|
|
case VVT_FORD_ST170:
|
|
|
|
return TT_FORD_ST170;
|
2021-03-22 04:48:29 -07:00
|
|
|
case VVT_BARRA_3_PLUS_1:
|
|
|
|
return TT_VVT_BARRA_3_PLUS_1;
|
2022-08-31 19:12:45 -07:00
|
|
|
case VVT_MAZDA_SKYACTIV:
|
|
|
|
return TT_VVT_MAZDA_SKYACTIV;
|
2021-07-03 06:43:27 -07:00
|
|
|
case VVT_NISSAN_VQ:
|
2021-07-16 21:27:56 -07:00
|
|
|
return TT_VVT_NISSAN_VQ35;
|
2022-03-22 03:34:22 -07:00
|
|
|
case VVT_TOYOTA_4_1:
|
|
|
|
return TT_VVT_TOYOTA_4_1;
|
2022-03-19 19:06:19 -07:00
|
|
|
case VVT_MITSUBISHI_3A92:
|
|
|
|
return TT_VVT_MITSUBISHI_3A92;
|
|
|
|
case VVT_MITSUBISHI_6G75:
|
2021-11-05 12:33:22 -07:00
|
|
|
case VVT_NISSAN_MR:
|
|
|
|
return TT_NISSAN_MR18_CAM_VVT;
|
2020-08-26 10:30:55 -07:00
|
|
|
default:
|
2021-07-03 06:43:27 -07:00
|
|
|
firmwareError(OBD_PCM_Processor_Fault, "getVvtTriggerType for %s", getVvt_mode_e(vvtMode));
|
2021-07-03 06:48:59 -07:00
|
|
|
return TT_ONE; // we have to return something for the sake of -Werror=return-type
|
2020-08-26 10:30:55 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-02 16:27:18 -07:00
|
|
|
void Engine::updateTriggerWaveform() {
|
2020-08-26 20:35:11 -07:00
|
|
|
|
2020-10-05 13:42:50 -07:00
|
|
|
|
2019-04-12 19:07:03 -07:00
|
|
|
#if EFI_ENGINE_CONTROL && EFI_SHAFT_POSITION_INPUT
|
2018-12-25 18:18:14 -08:00
|
|
|
// we have a confusing threading model so some synchronization would not hurt
|
2020-11-19 03:56:02 -08:00
|
|
|
chibios_rt::CriticalSectionLocker csl;
|
2018-12-25 18:18:14 -08:00
|
|
|
|
2022-09-13 23:45:31 -07:00
|
|
|
engine->triggerCentral.updateWaveform();
|
2020-08-26 10:30:55 -07:00
|
|
|
|
2022-06-09 14:21:22 -07:00
|
|
|
|
2022-09-13 23:45:31 -07:00
|
|
|
if (!engine->triggerCentral.triggerShape.shapeDefinitionError) {
|
2021-11-16 01:15:29 -08:00
|
|
|
prepareOutputSignals();
|
2018-12-25 18:18:14 -08:00
|
|
|
}
|
2019-01-31 14:55:23 -08:00
|
|
|
#endif /* EFI_ENGINE_CONTROL && EFI_SHAFT_POSITION_INPUT */
|
2019-01-14 07:58:38 -08:00
|
|
|
}
|
|
|
|
|
2020-12-27 14:41:28 -08:00
|
|
|
#if ANALOG_HW_CHECK_MODE
|
2022-04-10 14:41:41 -07:00
|
|
|
static void assertCloseTo(const char* msg, float actual, float expected) {
|
|
|
|
if (actual < 0.95f * expected || actual > 1.05f * expected) {
|
2022-04-09 16:13:55 -07:00
|
|
|
firmwareError(OBD_PCM_Processor_Fault, "%s validation failed actual=%f vs expected=%f", msg, actual, expected);
|
2020-07-27 19:49:59 -07:00
|
|
|
}
|
|
|
|
}
|
2020-12-27 14:41:28 -08:00
|
|
|
#endif // ANALOG_HW_CHECK_MODE
|
2020-07-27 19:49:59 -07:00
|
|
|
|
2021-11-16 01:15:29 -08:00
|
|
|
void Engine::periodicSlowCallback() {
|
2019-10-11 17:43:21 -07:00
|
|
|
ScopePerf perf(PE::EnginePeriodicSlowCallback);
|
2020-10-05 13:42:50 -07:00
|
|
|
|
|
|
|
// Re-read config in case it's changed
|
2022-09-13 22:53:17 -07:00
|
|
|
triggerCentral.primaryTriggerConfiguration.update();
|
2021-02-08 19:21:02 -08:00
|
|
|
for (int camIndex = 0;camIndex < CAMS_PER_BANK;camIndex++) {
|
2022-09-13 22:53:17 -07:00
|
|
|
triggerCentral.vvtTriggerConfiguration[camIndex].update();
|
2021-02-08 19:21:02 -08:00
|
|
|
}
|
2020-10-05 13:42:50 -07:00
|
|
|
|
2022-09-09 15:02:38 -07:00
|
|
|
efiWatchdog();
|
2021-11-16 01:15:29 -08:00
|
|
|
updateSlowSensors();
|
|
|
|
checkShutdown();
|
2019-01-14 07:58:38 -08:00
|
|
|
|
2022-01-23 12:31:39 -08:00
|
|
|
tpsAccelEnrichment.onNewValue(Sensor::getOrZero(SensorType::Tps1));
|
|
|
|
|
2021-11-16 01:15:29 -08:00
|
|
|
updateVrPwm();
|
2021-07-28 04:44:44 -07:00
|
|
|
|
2022-06-23 11:32:34 -07:00
|
|
|
enginePins.o2heater.setValue(engineConfiguration->forceO2Heating || engine->rpmCalculator.isRunning());
|
2022-02-01 12:42:04 -08:00
|
|
|
enginePins.starterRelayDisable.setValue(Sensor::getOrZero(SensorType::Rpm) < engineConfiguration->cranking.rpm);
|
2018-12-25 18:18:14 -08:00
|
|
|
|
2020-04-26 11:06:28 -07:00
|
|
|
updateGppwm();
|
|
|
|
|
2021-11-17 10:45:10 -08:00
|
|
|
engine->engineModules.apply_all([](auto & m) { m.onSlowCallback(); });
|
2020-11-11 18:47:19 -08:00
|
|
|
|
2021-11-25 04:59:31 -08:00
|
|
|
updateFans(module<AcController>().unmock().isAcEnabled());
|
|
|
|
|
2021-01-18 04:04:14 -08:00
|
|
|
#if EFI_BOOST_CONTROL
|
|
|
|
updateBoostControl();
|
|
|
|
#endif // EFI_BOOST_CONTROL
|
|
|
|
|
2019-09-06 17:30:27 -07:00
|
|
|
#if (BOARD_TLE8888_COUNT > 0)
|
2021-05-14 07:54:40 -07:00
|
|
|
tle8888startup();
|
2019-09-06 17:30:27 -07:00
|
|
|
#endif
|
|
|
|
|
2020-12-04 17:28:48 -08:00
|
|
|
#if EFI_DYNO_VIEW
|
2021-11-16 01:15:29 -08:00
|
|
|
updateDynoView();
|
2020-12-04 17:28:48 -08:00
|
|
|
#endif
|
|
|
|
|
2020-07-27 19:49:59 -07:00
|
|
|
slowCallBackWasInvoked = true;
|
|
|
|
|
2021-05-14 07:54:40 -07:00
|
|
|
#if EFI_PROD_CODE
|
|
|
|
void baroLps25Update();
|
|
|
|
baroLps25Update();
|
|
|
|
#endif // EFI_PROD_CODE
|
2021-03-04 19:39:20 -08:00
|
|
|
|
2020-12-27 14:41:28 -08:00
|
|
|
#if ANALOG_HW_CHECK_MODE
|
2021-11-17 00:54:21 -08:00
|
|
|
efiAssertVoid(OBD_PCM_Processor_Fault, isAdcChannelValid(engineConfiguration->clt.adcChannel), "No CLT setting");
|
2022-09-11 10:06:03 -07:00
|
|
|
efitimesec_t secondsNow = getTimeNowS();
|
2022-04-09 16:13:55 -07:00
|
|
|
|
|
|
|
#if ! HW_CHECK_ALWAYS_STIMULATE
|
|
|
|
fail("HW_CHECK_ALWAYS_STIMULATE required to have self-stimulation")
|
|
|
|
#endif
|
|
|
|
|
2020-10-11 16:05:56 -07:00
|
|
|
if (secondsNow > 2 && secondsNow < 180) {
|
2022-04-10 14:41:41 -07:00
|
|
|
assertCloseTo("RPM", Sensor::get(SensorType::Rpm).Value, HW_CHECK_RPM);
|
2020-10-11 19:46:08 -07:00
|
|
|
} else if (!hasFirmwareError() && secondsNow > 180) {
|
2020-10-11 19:57:31 -07:00
|
|
|
static bool isHappyTest = false;
|
2020-10-11 19:46:08 -07:00
|
|
|
if (!isHappyTest) {
|
|
|
|
setTriggerEmulatorRPM(5 * HW_CHECK_RPM);
|
2021-04-19 05:11:59 -07:00
|
|
|
efiPrintf("TEST PASSED");
|
2020-10-11 19:46:08 -07:00
|
|
|
isHappyTest = true;
|
|
|
|
}
|
2020-10-11 16:05:56 -07:00
|
|
|
}
|
2022-04-10 14:41:41 -07:00
|
|
|
|
|
|
|
assertCloseTo("clt", Sensor::getRaw(SensorType::Clt), 1.351f);
|
|
|
|
assertCloseTo("iat", Sensor::getRaw(SensorType::Iat), 2.245f);
|
|
|
|
assertCloseTo("aut1", Sensor::getRaw(SensorType::AuxTemp1), 2.750f);
|
|
|
|
assertCloseTo("aut2", Sensor::getRaw(SensorType::AuxTemp2), 3.176f);
|
2020-12-27 14:41:28 -08:00
|
|
|
#endif // ANALOG_HW_CHECK_MODE
|
2018-12-25 18:18:14 -08:00
|
|
|
}
|
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
/**
|
|
|
|
* We are executing these heavy (logarithm) methods from outside the trigger callbacks for performance reasons.
|
2016-01-01 14:02:49 -08:00
|
|
|
* See also periodicFastCallback
|
2015-07-10 06:01:56 -07:00
|
|
|
*/
|
2021-11-16 01:15:29 -08:00
|
|
|
void Engine::updateSlowSensors() {
|
|
|
|
updateSwitchInputs();
|
2020-12-26 18:31:41 -08:00
|
|
|
|
2019-04-12 19:07:03 -07:00
|
|
|
#if EFI_ENGINE_CONTROL
|
2022-01-20 19:42:13 -08:00
|
|
|
int rpm = Sensor::getOrZero(SensorType::Rpm);
|
2022-09-13 23:17:04 -07:00
|
|
|
triggerCentral.isEngineSnifferEnabled = rpm < engineConfiguration->engineSnifferRpmThreshold;
|
2022-09-15 18:45:48 -07:00
|
|
|
getEngineState()->sensorChartMode = rpm < engineConfiguration->sensorSnifferRpmThreshold ? engineConfiguration->sensorChartMode : SC_OFF;
|
2016-01-30 19:03:36 -08:00
|
|
|
|
2021-11-16 01:15:29 -08:00
|
|
|
engineState.updateSlowSensors();
|
2019-01-31 08:57:15 -08:00
|
|
|
#endif
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
2021-12-15 16:20:07 -08:00
|
|
|
#if EFI_GPIO_HARDWARE
|
2022-08-27 19:09:54 -07:00
|
|
|
static bool getClutchUpState() {
|
2021-12-15 14:48:21 -08:00
|
|
|
if (isBrainPinValid(engineConfiguration->clutchUpPin)) {
|
|
|
|
return engineConfiguration->clutchUpPinInverted ^ efiReadPin(engineConfiguration->clutchUpPin);
|
|
|
|
}
|
2022-06-29 20:34:00 -07:00
|
|
|
return engine->engineState.lua.clutchUpState;
|
2021-12-15 14:48:21 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
static bool getBrakePedalState() {
|
2021-12-17 13:33:43 -08:00
|
|
|
if (isBrainPinValid(engineConfiguration->brakePedalPin)) {
|
2021-12-15 14:48:21 -08:00
|
|
|
return efiReadPin(engineConfiguration->brakePedalPin);
|
|
|
|
}
|
2022-06-29 20:34:00 -07:00
|
|
|
return engine->engineState.lua.brakePedalState;
|
2021-12-15 14:48:21 -08:00
|
|
|
}
|
2022-08-27 19:09:54 -07:00
|
|
|
#endif // EFI_GPIO_HARDWARE
|
2021-12-15 14:48:21 -08:00
|
|
|
|
2021-11-16 01:15:29 -08:00
|
|
|
void Engine::updateSwitchInputs() {
|
2020-12-26 18:31:41 -08:00
|
|
|
#if EFI_GPIO_HARDWARE
|
|
|
|
// this value is not used yet
|
2021-11-17 00:54:21 -08:00
|
|
|
if (isBrainPinValid(engineConfiguration->clutchDownPin)) {
|
2022-06-28 23:37:24 -07:00
|
|
|
engine->engineState.clutchDownState = engineConfiguration->clutchDownPinInverted ^ efiReadPin(engineConfiguration->clutchDownPin);
|
2020-12-26 18:31:41 -08:00
|
|
|
}
|
2022-07-07 21:51:53 -07:00
|
|
|
{
|
|
|
|
bool currentState;
|
|
|
|
if (hasAcToggle()) {
|
|
|
|
currentState = getAcToggle();
|
|
|
|
} else {
|
|
|
|
currentState = engine->engineState.lua.acRequestState;
|
|
|
|
}
|
2022-05-09 03:12:17 -07:00
|
|
|
AcController & acController = engine->module<AcController>().unmock();
|
2022-07-07 21:51:53 -07:00
|
|
|
if (acController.acButtonState != currentState) {
|
|
|
|
acController.acButtonState = currentState;
|
2022-05-09 03:12:17 -07:00
|
|
|
acController.acSwitchLastChangeTimeMs = US2MS(getTimeNowUs());
|
2020-12-26 18:31:41 -08:00
|
|
|
}
|
|
|
|
}
|
2022-08-25 19:24:35 -07:00
|
|
|
engine->engineState.clutchUpState = getClutchUpState();
|
2021-12-15 14:48:21 -08:00
|
|
|
|
2021-11-17 00:54:21 -08:00
|
|
|
if (isBrainPinValid(engineConfiguration->throttlePedalUpPin)) {
|
2022-01-10 16:48:45 -08:00
|
|
|
engine->module<IdleController>().unmock().throttlePedalUpState = efiReadPin(engineConfiguration->throttlePedalUpPin);
|
2020-12-26 18:31:41 -08:00
|
|
|
}
|
|
|
|
|
2022-08-25 19:24:35 -07:00
|
|
|
engine->engineState.brakePedalState = getBrakePedalState();
|
2021-12-15 14:48:21 -08:00
|
|
|
|
2020-12-26 18:31:41 -08:00
|
|
|
#endif // EFI_GPIO_HARDWARE
|
|
|
|
}
|
|
|
|
|
2020-10-04 16:29:26 -07:00
|
|
|
Engine::Engine() {
|
2016-12-18 07:02:38 -08:00
|
|
|
reset();
|
|
|
|
}
|
|
|
|
|
2019-01-15 18:51:09 -08:00
|
|
|
int Engine::getGlobalConfigurationVersion(void) const {
|
|
|
|
return globalConfigurationVersion;
|
|
|
|
}
|
|
|
|
|
2016-12-18 07:02:38 -08:00
|
|
|
void Engine::reset() {
|
2016-01-14 21:01:42 -08:00
|
|
|
/**
|
|
|
|
* it's important for fixAngle() that engineCycle field never has zero
|
|
|
|
*/
|
2022-09-05 01:24:17 -07:00
|
|
|
engineState.engineCycle = getEngineCycle(FOUR_STROKE_CRANK_SENSOR);
|
2022-07-14 09:36:55 -07:00
|
|
|
resetLua();
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
2022-07-14 09:36:55 -07:00
|
|
|
void Engine::resetLua() {
|
|
|
|
// todo: https://github.com/rusefi/rusefi/issues/4308
|
|
|
|
engineState.lua = {};
|
|
|
|
engineState.lua.fuelMult = 1;
|
2022-07-14 11:04:21 -07:00
|
|
|
#if EFI_BOOST_CONTROL
|
2022-07-14 10:03:08 -07:00
|
|
|
boostController.resetLua();
|
2022-07-14 11:04:21 -07:00
|
|
|
#endif // EFI_BOOST_CONTROL
|
2022-07-14 09:36:55 -07:00
|
|
|
ignitionState.luaTimingAdd = 0;
|
|
|
|
ignitionState.luaTimingMult = 1;
|
2022-08-20 21:58:59 -07:00
|
|
|
module<IdleController>().unmock().luaAdd = 0;
|
2022-07-14 09:36:55 -07:00
|
|
|
}
|
2016-01-01 14:02:49 -08:00
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
/**
|
|
|
|
* Here we have a bunch of stuff which should invoked after configuration change
|
|
|
|
* so that we can prepare some helper structures
|
|
|
|
*/
|
2021-11-16 01:15:29 -08:00
|
|
|
void Engine::preCalculate() {
|
2020-05-09 19:43:39 -07:00
|
|
|
#if EFI_TUNER_STUDIO
|
|
|
|
// we take 2 bytes of crc32, no idea if it's right to call it crc16 or not
|
2020-05-09 21:59:32 -07:00
|
|
|
// we have a hack here - we rely on the fact that engineMake is the first of three relevant fields
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.engineMakeCodeNameCrc16 = crc32(engineConfiguration->engineMake, 3 * VEHICLE_INFO_SIZE);
|
2020-06-04 17:43:52 -07:00
|
|
|
|
2020-06-04 21:43:05 -07:00
|
|
|
// we need and can empty warning message for CRC purposes
|
2021-02-16 13:58:54 -08:00
|
|
|
memset(config->warning_message, 0, sizeof(error_message_t));
|
2021-12-07 17:18:47 -08:00
|
|
|
engine->outputChannels.tuneCrc16 = crc32(config, sizeof(persistent_config_s));
|
2020-05-09 19:43:39 -07:00
|
|
|
#endif /* EFI_TUNER_STUDIO */
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
2020-02-26 15:16:35 -08:00
|
|
|
#if EFI_SHAFT_POSITION_INPUT
|
2019-12-05 22:57:11 -08:00
|
|
|
void Engine::OnTriggerStateProperState(efitick_t nowNt) {
|
2020-09-05 15:49:42 -07:00
|
|
|
rpmCalculator.setSpinningUp(nowNt);
|
2019-12-05 22:57:11 -08:00
|
|
|
}
|
|
|
|
|
2020-01-26 09:02:54 -08:00
|
|
|
void Engine::OnTriggerSynchronizationLost() {
|
|
|
|
// Needed for early instant-RPM detection
|
2021-02-16 13:58:54 -08:00
|
|
|
rpmCalculator.setStopSpinning();
|
2021-12-27 09:09:38 -08:00
|
|
|
|
|
|
|
triggerCentral.triggerState.resetTriggerState();
|
|
|
|
|
|
|
|
for (size_t i = 0; i < efi::size(triggerCentral.vvtState); i++) {
|
|
|
|
for (size_t j = 0; j < efi::size(triggerCentral.vvtState[0]); j++) {
|
|
|
|
triggerCentral.vvtState[i][j].resetTriggerState();
|
|
|
|
}
|
|
|
|
}
|
2020-01-26 09:02:54 -08:00
|
|
|
}
|
|
|
|
|
2022-05-10 13:55:28 -07:00
|
|
|
void Engine::OnTriggerSyncronization(bool wasSynchronized, bool isDecodingError) {
|
|
|
|
// TODO: this logic probably shouldn't be part of engine.cpp
|
2020-01-26 00:33:45 -08:00
|
|
|
|
2020-01-23 10:39:50 -08:00
|
|
|
// We only care about trigger shape once we have synchronized trigger. Anything could happen
|
|
|
|
// during first revolution and it's fine
|
|
|
|
if (wasSynchronized) {
|
|
|
|
enginePins.triggerDecoderErrorPin.setValue(isDecodingError);
|
|
|
|
|
|
|
|
// 'triggerStateListener is not null' means we are running a real engine and now just preparing trigger shape
|
|
|
|
// that's a bit of a hack, a sweet OOP solution would be a real callback or at least 'needDecodingErrorLogic' method?
|
|
|
|
if (isDecodingError) {
|
2022-09-04 06:15:24 -07:00
|
|
|
#if EFI_PROD_CODE
|
|
|
|
if (engineConfiguration->verboseTriggerSynchDetails || (triggerCentral.triggerState.someSortOfTriggerError() && !engineConfiguration->silentTriggerError)) {
|
|
|
|
efiPrintf("error: synchronizationPoint @ index %d expected %d/%d got %d/%d",
|
|
|
|
triggerCentral.triggerState.currentCycle.current_index,
|
2022-09-11 00:46:50 -07:00
|
|
|
TRIGGER_WAVEFORM(getExpectedEventCount(TriggerWheel::T_PRIMARY)),
|
|
|
|
TRIGGER_WAVEFORM(getExpectedEventCount(TriggerWheel::T_SECONDARY)),
|
2022-09-04 06:15:24 -07:00
|
|
|
triggerCentral.triggerState.currentCycle.eventCount[0],
|
|
|
|
triggerCentral.triggerState.currentCycle.eventCount[1]);
|
|
|
|
}
|
|
|
|
#endif /* EFI_PROD_CODE */
|
2020-01-23 10:39:50 -08:00
|
|
|
}
|
|
|
|
|
2022-09-13 23:35:55 -07:00
|
|
|
engine->triggerCentral.triggerErrorDetection.add(isDecodingError);
|
2020-01-23 10:39:50 -08:00
|
|
|
}
|
2020-01-22 10:25:35 -08:00
|
|
|
|
|
|
|
}
|
2020-02-26 15:16:35 -08:00
|
|
|
#endif
|
2019-12-05 22:57:11 -08:00
|
|
|
|
2020-10-04 16:29:26 -07:00
|
|
|
void Engine::injectEngineReferences() {
|
2022-09-13 22:53:17 -07:00
|
|
|
triggerCentral.primaryTriggerConfiguration.update();
|
2021-02-08 19:21:02 -08:00
|
|
|
for (int camIndex = 0;camIndex < CAMS_PER_BANK;camIndex++) {
|
2022-09-13 22:53:17 -07:00
|
|
|
triggerCentral.vvtTriggerConfiguration[camIndex].update();
|
2021-02-08 19:21:02 -08:00
|
|
|
}
|
2020-10-04 16:29:26 -07:00
|
|
|
}
|
|
|
|
|
2021-11-16 01:15:29 -08:00
|
|
|
void Engine::setConfig() {
|
2021-05-20 17:00:32 -07:00
|
|
|
efi::clear(config);
|
2020-10-04 16:29:26 -07:00
|
|
|
|
|
|
|
injectEngineReferences();
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
2022-09-09 15:02:38 -07:00
|
|
|
void Engine::efiWatchdog() {
|
2015-07-10 06:01:56 -07:00
|
|
|
#if EFI_ENGINE_CONTROL
|
|
|
|
if (isRunningPwmTest)
|
|
|
|
return;
|
2021-12-08 13:20:19 -08:00
|
|
|
|
|
|
|
if (module<PrimeController>()->isPriming()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-09-11 18:15:25 -07:00
|
|
|
static efitimems_t mostRecentMs = 0;
|
2022-09-11 18:27:20 -07:00
|
|
|
efitimems_t msNow = getTimeNowMs();
|
2022-09-11 13:08:11 -07:00
|
|
|
if (engineConfiguration->tempBooleanForVerySpecialLogic && engine->configBurnTimer.hasElapsedSec(5)) {
|
2022-09-09 15:27:16 -07:00
|
|
|
|
2022-09-11 02:46:55 -07:00
|
|
|
if (mostRecentMs != 0) {
|
|
|
|
efitimems_t gapInMs = msNow - mostRecentMs;
|
2022-09-09 17:17:02 -07:00
|
|
|
if (gapInMs > 500) {
|
2022-09-11 02:46:55 -07:00
|
|
|
firmwareError(WATCH_DOG_SECONDS, "gap in time: now=%d mS, was %d mS, gap=%dmS",
|
|
|
|
msNow, mostRecentMs, gapInMs);
|
2022-09-09 15:27:16 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-09-11 18:15:25 -07:00
|
|
|
mostRecentMs = msNow;
|
2022-09-09 15:27:16 -07:00
|
|
|
|
2022-09-13 23:24:41 -07:00
|
|
|
if (!getTriggerCentral()->isSpinningJustForWatchdog) {
|
2016-11-03 20:02:58 -07:00
|
|
|
if (!isRunningBenchTest() && enginePins.stopPins()) {
|
2016-11-04 19:02:42 -07:00
|
|
|
// todo: make this a firmwareError assuming functional tests would run
|
|
|
|
warning(CUSTOM_ERR_2ND_WATCHDOG, "Some pins were turned off by 2nd pass watchdog");
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
2021-03-07 14:11:50 -08:00
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
/**
|
|
|
|
* todo: better watch dog implementation should be implemented - see
|
|
|
|
* http://sourceforge.net/p/rusefi/tickets/96/
|
|
|
|
*/
|
2022-01-17 18:00:49 -08:00
|
|
|
if (engine->triggerCentral.engineMovedRecently()) {
|
2021-03-07 14:11:50 -08:00
|
|
|
// Engine moved recently, no need to safe pins.
|
2015-07-10 06:01:56 -07:00
|
|
|
return;
|
|
|
|
}
|
2022-09-13 23:24:41 -07:00
|
|
|
getTriggerCentral()->isSpinningJustForWatchdog = false;
|
2016-12-18 07:02:38 -08:00
|
|
|
ignitionEvents.isReady = false;
|
2019-04-12 19:07:03 -07:00
|
|
|
#if EFI_PROD_CODE || EFI_SIMULATOR
|
2021-04-19 05:11:59 -07:00
|
|
|
efiPrintf("engine has STOPPED");
|
2015-07-10 06:01:56 -07:00
|
|
|
triggerInfo();
|
|
|
|
#endif
|
|
|
|
|
2016-11-03 20:02:58 -07:00
|
|
|
enginePins.stopPins();
|
2015-07-10 06:01:56 -07:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2021-11-16 01:15:29 -08:00
|
|
|
void Engine::checkShutdown() {
|
2019-04-12 19:07:03 -07:00
|
|
|
#if EFI_MAIN_RELAY_CONTROL
|
2020-09-07 11:41:04 -07:00
|
|
|
// if we are already in the "ignition_on" mode, then do nothing
|
2022-09-28 19:42:08 -07:00
|
|
|
/* this logic is not alive
|
2020-09-07 11:41:04 -07:00
|
|
|
if (ignitionOnTimeNt > 0) {
|
|
|
|
return;
|
|
|
|
}
|
2022-09-28 19:42:08 -07:00
|
|
|
todo: move to shutdown_controller.cpp
|
|
|
|
*/
|
2017-06-24 22:35:46 -07:00
|
|
|
|
2020-09-07 11:41:04 -07:00
|
|
|
// here we are in the shutdown (the ignition is off) or initial mode (after the firmware fresh start)
|
|
|
|
const efitick_t engineStopWaitTimeoutUs = 500000LL; // 0.5 sec
|
|
|
|
// in shutdown mode, we need a small cooldown time between the ignition off and on
|
2022-02-01 17:47:17 -08:00
|
|
|
/* this needs work or tests
|
2022-09-28 19:42:08 -07:00
|
|
|
todo: move to shutdown_controller.cpp
|
2020-09-07 11:41:04 -07:00
|
|
|
if (stopEngineRequestTimeNt == 0 || (getTimeNowNt() - stopEngineRequestTimeNt) > US2NT(engineStopWaitTimeoutUs)) {
|
|
|
|
// if the ignition key is turned on again,
|
|
|
|
// we cancel the shutdown mode, but only if all shutdown procedures are complete
|
|
|
|
const float vBattThresholdOn = 8.0f;
|
2021-12-01 08:22:13 -08:00
|
|
|
// we fallback into zero instead of VBAT_FALLBACK_VALUE because it's not safe to false-trigger the "ignition on" event,
|
|
|
|
// and we want to turn on the main relay only when 100% sure.
|
|
|
|
if ((Sensor::get(SensorType::BatteryVoltage).value_or(0) > vBattThresholdOn) && !isInShutdownMode()) {
|
2020-09-07 11:41:04 -07:00
|
|
|
ignitionOnTimeNt = getTimeNowNt();
|
2021-10-08 01:58:11 -07:00
|
|
|
efiPrintf("Ignition voltage detected!");
|
|
|
|
if (stopEngineRequestTimeNt != 0) {
|
|
|
|
efiPrintf("Cancel the engine shutdown!");
|
|
|
|
stopEngineRequestTimeNt = 0;
|
|
|
|
}
|
2020-09-07 11:41:04 -07:00
|
|
|
}
|
2017-06-24 22:35:46 -07:00
|
|
|
}
|
2022-02-01 17:47:17 -08:00
|
|
|
*/
|
2017-06-24 22:35:46 -07:00
|
|
|
#endif /* EFI_MAIN_RELAY_CONTROL */
|
|
|
|
}
|
|
|
|
|
2021-11-16 01:15:29 -08:00
|
|
|
bool Engine::isInMainRelayBench() {
|
2021-01-10 20:46:50 -08:00
|
|
|
if (mainRelayBenchStartNt == 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return (getTimeNowNt() - mainRelayBenchStartNt) < NT_PER_SECOND;
|
|
|
|
}
|
|
|
|
|
2021-11-16 01:15:29 -08:00
|
|
|
bool Engine::isInShutdownMode() const {
|
2022-02-01 17:47:17 -08:00
|
|
|
// TODO: this logic is currently broken
|
|
|
|
#if 0 && EFI_MAIN_RELAY_CONTROL && EFI_PROD_CODE
|
2020-09-07 11:41:04 -07:00
|
|
|
// if we are in "ignition_on" mode and not in shutdown mode
|
|
|
|
if (stopEngineRequestTimeNt == 0 && ignitionOnTimeNt > 0) {
|
|
|
|
const float vBattThresholdOff = 5.0f;
|
|
|
|
// start the shutdown process if the ignition voltage dropped low
|
2021-03-11 19:52:34 -08:00
|
|
|
if (Sensor::get(SensorType::BatteryVoltage).value_or(VBAT_FALLBACK_VALUE) <= vBattThresholdOff) {
|
2020-09-07 11:41:04 -07:00
|
|
|
scheduleStopEngine();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// we are not in the shutdown mode?
|
|
|
|
if (stopEngineRequestTimeNt == 0) {
|
2017-06-24 22:35:46 -07:00
|
|
|
return false;
|
2020-09-07 11:41:04 -07:00
|
|
|
}
|
|
|
|
|
2021-01-10 20:46:50 -08:00
|
|
|
const efitick_t turnOffWaitTimeoutNt = NT_PER_SECOND;
|
2020-09-07 11:41:04 -07:00
|
|
|
// We don't want any transients to step in, so we wait at least 1 second whatever happens.
|
|
|
|
// Also it's good to give the stepper motor some time to start moving to the initial position (or parking)
|
2021-01-10 20:46:50 -08:00
|
|
|
if ((getTimeNowNt() - stopEngineRequestTimeNt) < turnOffWaitTimeoutNt)
|
2020-09-07 11:41:04 -07:00
|
|
|
return true;
|
|
|
|
|
2021-01-10 20:46:50 -08:00
|
|
|
const efitick_t engineSpinningWaitTimeoutNt = 5 * NT_PER_SECOND;
|
2017-06-24 22:35:46 -07:00
|
|
|
// The engine is still spinning! Give it some time to stop (but wait no more than 5 secs)
|
2021-01-10 20:46:50 -08:00
|
|
|
if (isSpinning && (getTimeNowNt() - stopEngineRequestTimeNt) < engineSpinningWaitTimeoutNt)
|
2020-09-07 11:41:04 -07:00
|
|
|
return true;
|
|
|
|
|
|
|
|
// The idle motor valve is still moving! Give it some time to park (but wait no more than 10 secs)
|
|
|
|
// Usually it can move to the initial 'cranking' position or zero 'parking' position.
|
2021-01-10 20:46:50 -08:00
|
|
|
const efitick_t idleMotorWaitTimeoutNt = 10 * NT_PER_SECOND;
|
2021-11-16 01:15:29 -08:00
|
|
|
if (isIdleMotorBusy() && (getTimeNowNt() - stopEngineRequestTimeNt) < idleMotorWaitTimeoutNt)
|
2017-06-24 22:35:46 -07:00
|
|
|
return true;
|
|
|
|
#endif /* EFI_MAIN_RELAY_CONTROL */
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-11-16 01:15:29 -08:00
|
|
|
bool Engine::isMainRelayEnabled() const {
|
2020-09-07 11:41:04 -07:00
|
|
|
#if EFI_MAIN_RELAY_CONTROL
|
|
|
|
return enginePins.mainRelay.getLogicValue();
|
|
|
|
#else
|
|
|
|
// if no main relay control, we assume it's always turned on
|
|
|
|
return true;
|
|
|
|
#endif /* EFI_MAIN_RELAY_CONTROL */
|
|
|
|
}
|
|
|
|
|
2022-09-04 23:44:06 -07:00
|
|
|
injection_mode_e getCurrentInjectionMode() {
|
|
|
|
return getEngineRotationState()->isCranking() ? engineConfiguration->crankingInjectionMode : engineConfiguration->injectionMode;
|
2016-11-30 19:06:43 -08:00
|
|
|
}
|
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
/**
|
|
|
|
* The idea of this method is to execute all heavy calculations in a lower-priority thread,
|
2016-01-01 14:02:49 -08:00
|
|
|
* so that trigger event handler/IO scheduler tasks are faster.
|
2015-07-10 06:01:56 -07:00
|
|
|
*/
|
2021-11-16 01:15:29 -08:00
|
|
|
void Engine::periodicFastCallback() {
|
2019-10-11 17:43:21 -07:00
|
|
|
ScopePerf pc(PE::EnginePeriodicFastCallback);
|
2016-01-01 14:02:49 -08:00
|
|
|
|
2018-02-06 12:47:19 -08:00
|
|
|
#if EFI_MAP_AVERAGING
|
2021-11-16 01:15:29 -08:00
|
|
|
refreshMapAveragingPreCalc();
|
2018-02-06 12:47:19 -08:00
|
|
|
#endif
|
2016-01-01 14:02:49 -08:00
|
|
|
|
2021-11-16 01:15:29 -08:00
|
|
|
engineState.periodicFastCallback();
|
2020-08-21 12:36:43 -07:00
|
|
|
|
2021-11-16 01:15:29 -08:00
|
|
|
tachSignalCallback();
|
2021-11-17 09:13:19 -08:00
|
|
|
|
2021-11-17 10:45:10 -08:00
|
|
|
engine->engineModules.apply_all([](auto & m) { m.onFastCallback(); });
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
2019-01-19 19:09:37 -08:00
|
|
|
|
2022-09-04 23:28:46 -07:00
|
|
|
EngineRotationState * getEngineRotationState() {
|
|
|
|
return &engine->rpmCalculator;
|
|
|
|
}
|
|
|
|
|
2022-09-05 00:38:44 -07:00
|
|
|
EngineState * getEngineState() {
|
|
|
|
return &engine->engineState;
|
|
|
|
}
|
|
|
|
|
2022-09-07 16:35:52 -07:00
|
|
|
TunerStudioOutputChannels *getTunerStudioOutputChannels() {
|
|
|
|
return &engine->outputChannels;
|
|
|
|
}
|
2022-09-04 23:28:46 -07:00
|
|
|
|
2022-09-07 17:25:49 -07:00
|
|
|
ExecutorInterface *getExecutorInterface() {
|
|
|
|
return &engine->executor;
|
|
|
|
}
|
2022-09-13 23:06:52 -07:00
|
|
|
|
|
|
|
TriggerCentral * getTriggerCentral() {
|
|
|
|
return &engine->triggerCentral;
|
|
|
|
}
|
2022-09-13 23:17:04 -07:00
|
|
|
|
|
|
|
LimpManager * getLimpManager() {
|
|
|
|
return &engine->limpManager;
|
|
|
|
}
|
2022-09-15 20:41:06 -07:00
|
|
|
|
|
|
|
FuelSchedule *getFuelSchedule() {
|
|
|
|
return &engine->injectionEvents;
|
|
|
|
}
|
|
|
|
|
|
|
|
IgnitionEventList *getIgnitionEvents() {
|
|
|
|
return &engine->ignitionEvents;
|
|
|
|
}
|