2022-09-01 16:00:16 -07:00
|
|
|
|
2024-07-03 17:48:49 -07:00
|
|
|
#include "pch.h"
|
2022-09-01 16:00:16 -07:00
|
|
|
|
|
|
|
#include "dfco.h"
|
2022-01-23 16:44:41 -08:00
|
|
|
|
|
|
|
bool DfcoController::getState() const {
|
|
|
|
if (!engineConfiguration->coastingFuelCutEnabled) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-07-28 00:04:28 -07:00
|
|
|
const auto tps = Sensor::get(SensorType::DriverThrottleIntent);
|
|
|
|
const auto clt = Sensor::get(SensorType::Clt);
|
|
|
|
const auto map = Sensor::get(SensorType::Map);
|
2022-01-23 16:44:41 -08:00
|
|
|
|
|
|
|
// If some sensor is broken, inhibit DFCO
|
2022-07-28 00:04:28 -07:00
|
|
|
if (!tps || !clt || !map) {
|
2022-01-23 16:44:41 -08:00
|
|
|
return false;
|
|
|
|
}
|
2024-07-03 17:48:49 -07:00
|
|
|
if (engine->engineState.lua.disableDecelerationFuelCutOff) {
|
|
|
|
// Lua might have reasons to disable
|
|
|
|
return false;
|
|
|
|
}
|
2022-01-23 16:44:41 -08:00
|
|
|
|
2022-01-27 11:05:18 -08:00
|
|
|
float rpm = Sensor::getOrZero(SensorType::Rpm);
|
|
|
|
float vss = Sensor::getOrZero(SensorType::VehicleSpeed);
|
|
|
|
|
2022-07-28 00:04:28 -07:00
|
|
|
bool mapActivate = map.Value < engineConfiguration->coastingFuelCutMap;
|
|
|
|
bool tpsActivate = tps.Value < engineConfiguration->coastingFuelCutTps;
|
|
|
|
bool cltActivate = clt.Value > engineConfiguration->coastingFuelCutClt;
|
2022-01-23 16:44:41 -08:00
|
|
|
// True if throttle, MAP, and CLT are all acceptable for DFCO to occur
|
|
|
|
bool dfcoAllowed = mapActivate && tpsActivate && cltActivate;
|
|
|
|
|
|
|
|
bool rpmActivate = (rpm > engineConfiguration->coastingFuelCutRpmHigh);
|
|
|
|
bool rpmDeactivate = (rpm < engineConfiguration->coastingFuelCutRpmLow);
|
|
|
|
|
2022-01-27 11:05:18 -08:00
|
|
|
// greater than or equal so that it works if both config params are set to 0
|
|
|
|
bool vssActivate = (vss >= engineConfiguration->coastingFuelCutVssHigh);
|
|
|
|
bool vssDeactivate = (vss < engineConfiguration->coastingFuelCutVssLow);
|
|
|
|
|
|
|
|
// RPM is high enough, VSS high enough, and DFCO allowed
|
|
|
|
if (dfcoAllowed && rpmActivate && vssActivate) {
|
2022-01-23 16:44:41 -08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2022-01-27 11:05:18 -08:00
|
|
|
// RPM too low, VSS too low, or DFCO not allowed
|
|
|
|
if (!dfcoAllowed || rpmDeactivate || vssDeactivate) {
|
2022-01-23 16:44:41 -08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// No conditions hit, no change to state (provides hysteresis)
|
|
|
|
return m_isDfco;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DfcoController::update() {
|
|
|
|
// Run state machine
|
2022-01-28 18:35:23 -08:00
|
|
|
bool newState = getState();
|
|
|
|
|
|
|
|
// If fuel is cut, reset the timer
|
|
|
|
if (newState) {
|
|
|
|
m_timeSinceCut.reset();
|
2022-08-25 18:26:17 -07:00
|
|
|
} else {
|
|
|
|
// If fuel is not cut, reset the not-cut timer
|
|
|
|
m_timeSinceNoCut.reset();
|
2022-01-28 18:35:23 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
m_isDfco = newState;
|
2022-01-23 16:44:41 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool DfcoController::cutFuel() const {
|
2022-08-25 18:26:17 -07:00
|
|
|
float cutDelay = engineConfiguration->dfcoDelay;
|
|
|
|
|
|
|
|
// 0 delay means cut immediately, aka timer has always expired
|
|
|
|
bool hasBeenDelay = (cutDelay == 0) || m_timeSinceNoCut.hasElapsedSec(cutDelay);
|
|
|
|
|
|
|
|
return m_isDfco && hasBeenDelay;
|
2022-01-23 16:44:41 -08:00
|
|
|
}
|
2022-01-28 18:35:23 -08:00
|
|
|
|
|
|
|
float DfcoController::getTimeSinceCut() const {
|
|
|
|
return m_timeSinceCut.getElapsedSeconds();
|
|
|
|
}
|