rusefi/firmware/controllers/algo/fuel_math.cpp

435 lines
14 KiB
C++
Raw Normal View History

2015-07-10 06:01:56 -07:00
/**
* @file fuel_math.cpp
* @brief Fuel amount calculation logic
*
*
* @date May 27, 2013
2020-01-13 18:57:43 -08:00
* @author Andrey Belomutskiy, (c) 2012-2020
2015-07-10 06:01:56 -07:00
*
* This file is part of rusEfi - see http://rusefi.com
*
* rusEfi is free software; you can redistribute it and/or modify it under the terms of
* the GNU General Public License as published by the Free Software Foundation; either
* version 3 of the License, or (at your option) any later version.
*
* rusEfi is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without
* even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License along with this program.
* If not, see <http://www.gnu.org/licenses/>.
*
*/
#include "pch.h"
#include "airmass.h"
2020-07-25 01:14:35 -07:00
#include "alphan_airmass.h"
2020-07-23 01:23:57 -07:00
#include "maf_airmass.h"
2020-07-24 19:30:12 -07:00
#include "speed_density_airmass.h"
2015-07-10 06:01:56 -07:00
#include "fuel_math.h"
2020-08-10 21:41:03 -07:00
#include "fuel_computer.h"
#include "injector_model.h"
2015-07-10 06:01:56 -07:00
#include "speed_density.h"
2020-07-22 13:11:07 -07:00
#include "speed_density_base.h"
#include "lua_hooks.h"
2015-07-10 06:01:56 -07:00
static fuel_Map3D_t fuelPhaseMap;
extern fuel_Map3D_t veMap;
extern lambda_Map3D_t lambdaMap;
2015-07-10 06:01:56 -07:00
extern baroCorr_Map3D_t baroCorrMap;
static mapEstimate_Map3D_t mapEstimationTable;
2015-07-10 06:01:56 -07:00
2019-04-12 19:07:03 -07:00
#if EFI_ENGINE_CONTROL
2019-01-31 08:57:15 -08:00
float getCrankingFuel3(
float baseFuel,
uint32_t revolutionCounterSinceStart) {
2019-08-26 20:41:04 -07:00
// these magic constants are in Celsius
2020-02-18 07:47:39 -08:00
float baseCrankingFuel;
if (engineConfiguration->useRunningMathForCranking) {
baseCrankingFuel = baseFuel;
2020-02-18 07:47:39 -08:00
} else {
// parameter is in milligrams, convert to grams
baseCrankingFuel = engineConfiguration->cranking.baseFuel * 0.001f;
2020-02-18 07:47:39 -08:00
}
2019-08-26 20:41:04 -07:00
/**
* Cranking fuel changes over time
*/
engine->engineState.cranking.durationCoefficient = interpolate2d(revolutionCounterSinceStart, config->crankingCycleBins,
2019-08-26 20:41:04 -07:00
config->crankingCycleCoef);
/**
* Cranking fuel is different depending on engine coolant temperature
2020-06-18 05:54:02 -07:00
* If the sensor is failed, use 20 deg C
2019-08-26 20:41:04 -07:00
*/
2020-06-18 05:54:02 -07:00
auto clt = Sensor::get(SensorType::Clt);
engine->engineState.cranking.coolantTemperatureCoefficient =
interpolate2d(clt.value_or(20), config->crankingFuelBins, config->crankingFuelCoef);
2019-08-26 20:41:04 -07:00
auto tps = Sensor::get(SensorType::DriverThrottleIntent);
2019-08-26 20:41:04 -07:00
engine->engineState.cranking.tpsCoefficient = tps.Valid ? 1 : interpolate2d(tps.Value, engineConfiguration->crankingTpsBins,
2019-08-26 20:41:04 -07:00
engineConfiguration->crankingTpsCoef);
2020-06-18 05:54:02 -07:00
/*
engine->engineState.cranking.tpsCoefficient =
2020-06-18 05:54:02 -07:00
tps.Valid
? interpolate2d(tps.Value, engineConfiguration->crankingTpsBins, engineConfiguration->crankingTpsCoef)
2020-06-18 05:54:02 -07:00
: 1; // in case of failed TPS, don't correct.*/
2019-08-26 20:41:04 -07:00
2019-10-14 23:06:15 -07:00
floatms_t crankingFuel = baseCrankingFuel
2019-08-26 20:41:04 -07:00
* engine->engineState.cranking.durationCoefficient
* engine->engineState.cranking.coolantTemperatureCoefficient
* engine->engineState.cranking.tpsCoefficient;
engine->engineState.cranking.fuel = crankingFuel * 1000;
2019-10-14 23:06:15 -07:00
2019-08-26 20:41:04 -07:00
if (crankingFuel <= 0) {
warning(CUSTOM_ERR_ZERO_CRANKING_FUEL, "Cranking fuel value %f", crankingFuel);
}
return crankingFuel;
}
float getRunningFuel(float baseFuel) {
2019-10-13 13:14:08 -07:00
ScopePerf perf(PE::GetRunningFuel);
engine->engineState.running.baseFuel = baseFuel;
2019-08-27 21:29:55 -07:00
float iatCorrection = engine->engineState.running.intakeTemperatureCoefficient;
2019-08-27 21:29:55 -07:00
float cltCorrection = engine->engineState.running.coolantTemperatureCoefficient;
2019-08-27 21:29:55 -07:00
float postCrankingFuelCorrection = engine->engineState.running.postCrankingFuelCorrection;
2019-08-27 21:29:55 -07:00
float baroCorrection = engine->engineState.baroCorrection;
2019-08-26 20:41:04 -07:00
efiAssert(CUSTOM_ERR_ASSERT, !cisnan(iatCorrection), "NaN iatCorrection", 0);
efiAssert(CUSTOM_ERR_ASSERT, !cisnan(cltCorrection), "NaN cltCorrection", 0);
efiAssert(CUSTOM_ERR_ASSERT, !cisnan(postCrankingFuelCorrection), "NaN postCrankingFuelCorrection", 0);
float runningFuel = baseFuel * baroCorrection * iatCorrection * cltCorrection * postCrankingFuelCorrection;
2019-08-26 20:41:04 -07:00
efiAssert(CUSTOM_ERR_ASSERT, !cisnan(runningFuel), "NaN runningFuel", 0);
2019-08-27 21:29:55 -07:00
engine->engineState.running.fuel = runningFuel * 1000;
2019-08-26 20:41:04 -07:00
return runningFuel;
}
/* DISPLAY_ENDIF */
static SpeedDensityAirmass sdAirmass(veMap, mapEstimationTable);
2020-07-23 01:23:57 -07:00
static MafAirmass mafAirmass(veMap);
2020-07-25 01:14:35 -07:00
static AlphaNAirmass alphaNAirmass(veMap);
2020-07-23 01:23:57 -07:00
AirmassModelBase* getAirmassModel(engine_load_mode_e mode) {
2021-05-14 04:17:22 -07:00
switch (mode) {
2020-07-24 19:38:48 -07:00
case LM_SPEED_DENSITY: return &sdAirmass;
case LM_REAL_MAF: return &mafAirmass;
2020-11-27 12:44:31 -08:00
case LM_ALPHA_N: return &alphaNAirmass;
#if EFI_LUA
case LM_LUA: return &(getLuaAirmassModel());
#endif
2020-07-26 20:33:18 -07:00
#if EFI_UNIT_TEST
case LM_MOCK: return engine->mockAirmassModel;
#endif
default:
// this is a bad work-around for https://github.com/rusefi/rusefi/issues/1690 issue
warning(CUSTOM_ERR_ASSERT, "Invalid airmass mode %d", engineConfiguration->fuelAlgorithm);
return &sdAirmass;
/* todo: this should be the implementation
return nullptr;
*/
}
}
// Per-cylinder base fuel mass
static float getBaseFuelMass(int rpm) {
2019-10-13 13:14:08 -07:00
ScopePerf perf(PE::GetBaseFuel);
2020-07-30 21:20:10 -07:00
// airmass modes - get airmass first, then convert to fuel
auto model = getAirmassModel(engineConfiguration->fuelAlgorithm);
2020-07-30 21:20:10 -07:00
efiAssert(CUSTOM_ERR_ASSERT, model != nullptr, "Invalid airmass mode", 0.0f);
2020-07-30 21:20:10 -07:00
auto airmass = model->getAirmass(rpm);
// Plop some state for others to read
engine->engineState.sd.airMassInOneCylinder = airmass.CylinderAirmass;
engine->engineState.fuelingLoad = airmass.EngineLoadPercent;
engine->engineState.ignitionLoad = getLoadOverride(airmass.EngineLoadPercent, engineConfiguration->ignOverrideMode);
2020-07-30 21:20:10 -07:00
float baseFuelMass = engine->fuelComputer->getCycleFuel(airmass.CylinderAirmass, rpm, airmass.EngineLoadPercent);
// Fudge it by the global correction factor
baseFuelMass *= engineConfiguration->globalFuelCorrection;
engine->engineState.baseFuel = baseFuelMass;
2015-12-02 17:10:06 -08:00
2020-10-26 11:04:22 -07:00
if (cisnan(baseFuelMass)) {
// todo: we should not have this here but https://github.com/rusefi/rusefi/issues/1690
return 0;
}
return baseFuelMass;
2015-07-10 06:01:56 -07:00
}
angle_t getInjectionOffset(float rpm, float load) {
2018-01-01 15:59:50 -08:00
if (cisnan(rpm)) {
return 0; // error already reported
}
2020-07-20 23:11:48 -07:00
if (cisnan(load)) {
2017-12-03 12:30:42 -08:00
return 0; // error already reported
}
2020-07-20 23:11:48 -07:00
angle_t value = fuelPhaseMap.getValue(rpm, load);
2018-01-01 15:59:50 -08:00
if (cisnan(value)) {
2018-09-01 06:12:33 -07:00
// we could be here while resetting configuration for example
warning(CUSTOM_ERR_6569, "phase map not ready");
2018-01-01 15:59:50 -08:00
return 0;
}
angle_t result = value + engineConfiguration->extraInjectionOffset;
2018-07-23 18:38:05 -07:00
fixAngle(result, "inj offset#2", CUSTOM_ERR_6553);
2016-08-28 14:02:14 -07:00
return result;
2015-07-10 06:01:56 -07:00
}
/**
2017-11-06 19:29:39 -08:00
* Number of injections using each injector per engine cycle
2017-03-06 22:28:26 -08:00
* @see getNumberOfSparks
2015-07-10 06:01:56 -07:00
*/
int getNumberOfInjections(injection_mode_e mode) {
2015-07-10 06:01:56 -07:00
switch (mode) {
case IM_SIMULTANEOUS:
2017-11-06 16:00:30 -08:00
case IM_SINGLE_POINT:
2017-11-06 19:29:39 -08:00
return engineConfiguration->specs.cylindersCount;
2015-07-10 06:01:56 -07:00
case IM_BATCH:
2016-07-23 18:04:30 -07:00
return 2;
2017-11-06 19:29:39 -08:00
case IM_SEQUENTIAL:
return 1;
2015-07-10 06:01:56 -07:00
default:
2016-10-10 13:02:39 -07:00
firmwareError(CUSTOM_ERR_INVALID_INJECTION_MODE, "Unexpected injection_mode_e %d", mode);
2015-07-10 06:01:56 -07:00
return 1;
}
}
float getInjectionModeDurationMultiplier() {
injection_mode_e mode = engine->getCurrentInjectionMode();
2020-08-11 20:35:10 -07:00
switch (mode) {
2020-08-12 22:18:56 -07:00
case IM_SIMULTANEOUS: {
auto cylCount = engineConfiguration->specs.cylindersCount;
if (cylCount == 0) {
// we can end up here during configuration reset
return 0;
}
return 1.0f / cylCount;
}
2020-08-11 20:35:10 -07:00
case IM_SEQUENTIAL:
case IM_SINGLE_POINT:
return 1;
case IM_BATCH:
return 0.5f;
default:
firmwareError(CUSTOM_ERR_INVALID_INJECTION_MODE, "Unexpected injection_mode_e %d", mode);
return 0;
}
}
2017-03-06 22:28:26 -08:00
/**
2017-12-12 14:40:48 -08:00
* This is more like MOSFET duty cycle since durations include injector lag
2017-03-06 22:28:26 -08:00
* @see getCoilDutyCycle
*/
percent_t getInjectorDutyCycle(int rpm) {
floatms_t totalInjectiorAmountPerCycle = engine->injectionDuration * getNumberOfInjections(engineConfiguration->injectionMode);
floatms_t engineCycleDuration = getEngineCycleDuration(rpm);
2017-11-06 19:29:39 -08:00
return 100 * totalInjectiorAmountPerCycle / engineCycleDuration;
2015-07-10 06:01:56 -07:00
}
static float getCycleFuelMass(bool isCranking, float baseFuelMass) {
if (isCranking) {
return getCrankingFuel(baseFuelMass);
} else {
return getRunningFuel(baseFuelMass);
}
}
2015-07-10 06:01:56 -07:00
/**
* @returns Mass of each individual fuel injection, in grams
2017-11-06 19:29:39 -08:00
* in case of single point injection mode the amount of fuel into all cylinders, otherwise the amount for one cylinder
2015-07-10 06:01:56 -07:00
*/
float getInjectionMass(int rpm) {
2019-10-13 13:14:08 -07:00
ScopePerf perf(PE::GetInjectionDuration);
#if EFI_SHAFT_POSITION_INPUT
// Always update base fuel - some cranking modes use it
float baseFuelMass = getBaseFuelMass(rpm);
bool isCranking = engine->rpmCalculator.isCranking();
float cycleFuelMass = getCycleFuelMass(isCranking, baseFuelMass);
efiAssert(CUSTOM_ERR_ASSERT, !cisnan(cycleFuelMass), "NaN cycleFuelMass", 0);
// Fuel cut-off isn't just 0 or 1, it can be tapered
cycleFuelMass *= engine->engineState.fuelCutoffCorrection;
float durationMultiplier = getInjectionModeDurationMultiplier();
float injectionFuelMass = cycleFuelMass * durationMultiplier;
// Prepare injector flow rate & deadtime
engine->module<InjectorModel>()->prepare();
floatms_t tpsAccelEnrich = engine->tpsAccelEnrichment.getTpsEnrichment();
efiAssert(CUSTOM_ERR_ASSERT, !cisnan(tpsAccelEnrich), "NaN tpsAccelEnrich", 0);
engine->engineState.tpsAccelEnrich = tpsAccelEnrich;
// For legacy reasons, the TPS accel table is in units of milliseconds, so we have to convert BACK to mass
float tpsAccelPerInjection = durationMultiplier * tpsAccelEnrich;
float tpsFuelMass = engine->module<InjectorModel>()->getFuelMassForDuration(tpsAccelPerInjection);
return injectionFuelMass + tpsFuelMass;
2019-01-31 14:55:23 -08:00
#else
return 0;
#endif
2015-07-10 06:01:56 -07:00
}
static FuelComputer fuelComputer(lambdaMap);
2015-07-10 06:01:56 -07:00
/**
* @brief Initialize fuel map data structure
* @note this method has nothing to do with fuel map VALUES - it's job
* is to prepare the fuel map data structure for 3d interpolation
*/
void initFuelMap() {
2020-07-23 01:23:57 -07:00
engine->fuelComputer = &fuelComputer;
mapEstimationTable.init(config->mapEstimateTable, config->mapEstimateTpsBins, config->mapEstimateRpmBins);
2019-06-10 12:45:18 -07:00
#if (IGN_LOAD_COUNT == FUEL_LOAD_COUNT) && (IGN_RPM_COUNT == FUEL_RPM_COUNT)
2015-07-10 06:01:56 -07:00
fuelPhaseMap.init(config->injectionPhase, config->injPhaseLoadBins, config->injPhaseRpmBins);
2019-06-10 12:45:18 -07:00
#endif /* (IGN_LOAD_COUNT == FUEL_LOAD_COUNT) && (IGN_RPM_COUNT == FUEL_RPM_COUNT) */
2015-07-10 06:01:56 -07:00
}
/**
* @brief Engine warm-up fuel correction.
*/
float getCltFuelCorrection() {
const auto [valid, clt] = Sensor::get(SensorType::Clt);
if (!valid)
2015-07-10 06:01:56 -07:00
return 1; // this error should be already reported somewhere else, let's just handle it
return interpolate2d(clt, config->cltFuelCorrBins, config->cltFuelCorr);
2015-07-10 06:01:56 -07:00
}
angle_t getCltTimingCorrection() {
const auto [valid, clt] = Sensor::get(SensorType::Clt);
if (!valid)
2017-01-05 18:12:06 -08:00
return 0; // this error should be already reported somewhere else, let's just handle it
return interpolate2d(clt, engineConfiguration->cltTimingBins, engineConfiguration->cltTimingExtra);
2017-01-05 18:12:06 -08:00
}
float getIatFuelCorrection() {
const auto [valid, iat] = Sensor::get(SensorType::Iat);
if (!valid)
2015-07-10 06:01:56 -07:00
return 1; // this error should be already reported somewhere else, let's just handle it
return interpolate2d(iat, config->iatFuelCorrBins, config->iatFuelCorr);
2015-07-10 06:01:56 -07:00
}
/**
* @brief Called from EngineState::periodicFastCallback to update the state.
* @note The returned value is float, not boolean - to implement taper (smoothed correction).
* @return Fuel duration correction for fuel cut-off control (ex. if coasting). No correction if 1.0
*/
float getFuelCutOffCorrection(efitick_t nowNt, int rpm) {
// no corrections by default
float fuelCorr = 1.0f;
// coasting fuel cut-off correction
if (engineConfiguration->coastingFuelCutEnabled) {
auto [tpsValid, tpsPos] = Sensor::get(SensorType::Tps1);
if (!tpsValid) {
return 1.0f;
}
const auto [cltValid, clt] = Sensor::get(SensorType::Clt);
if (!cltValid) {
return 1.0f;
}
const auto [mapValid, map] = Sensor::get(SensorType::Map);
if (!mapValid) {
return 1.0f;
}
// gather events
bool mapDeactivate = (map >= engineConfiguration->coastingFuelCutMap);
bool tpsDeactivate = (tpsPos >= engineConfiguration->coastingFuelCutTps);
2019-11-04 19:52:37 -08:00
// If no CLT sensor (or broken), don't allow DFCO
bool cltDeactivate = clt < (float)engineConfiguration->coastingFuelCutClt;
bool rpmDeactivate = (rpm < engineConfiguration->coastingFuelCutRpmLow);
bool rpmActivate = (rpm > engineConfiguration->coastingFuelCutRpmHigh);
// state machine (coastingFuelCutStartTime is also used as a flag)
if (!mapDeactivate && !tpsDeactivate && !cltDeactivate && rpmActivate) {
engine->engineState.coastingFuelCutStartTime = nowNt;
} else if (mapDeactivate || tpsDeactivate || rpmDeactivate || cltDeactivate) {
engine->engineState.coastingFuelCutStartTime = 0;
}
// enable fuelcut?
if (engine->engineState.coastingFuelCutStartTime != 0) {
// todo: add taper - interpolate using (nowNt - coastingFuelCutStartTime)?
fuelCorr = 0.0f;
}
}
// todo: add other fuel cut-off checks here (possibly cutFuelOnHardLimit?)
return fuelCorr;
}
float getBaroCorrection() {
if (Sensor::hasSensor(SensorType::BarometricPressure)) {
// Default to 1atm if failed
float pressure = Sensor::get(SensorType::BarometricPressure).value_or(101.325f);
float correction = baroCorrMap.getValue(GET_RPM(), pressure);
if (cisnan(correction) || correction < 0.01) {
warning(OBD_Barometric_Press_Circ_Range_Perf, "Invalid baro correction %f", correction);
return 1;
}
return correction;
2015-07-10 06:01:56 -07:00
} else {
return 1;
}
}
2019-04-12 19:07:03 -07:00
#if EFI_ENGINE_CONTROL
2015-07-10 06:01:56 -07:00
/**
* @return Duration of fuel injection while craning
*/
float getCrankingFuel(float baseFuel) {
return getCrankingFuel3(baseFuel, engine->rpmCalculator.getRevolutionCounterSinceStart());
2015-07-10 06:01:56 -07:00
}
2020-05-05 05:01:40 -07:00
float getStandardAirCharge() {
float totalDisplacement = engineConfiguration->specs.displacement;
float cylDisplacement = totalDisplacement / engineConfiguration->specs.cylindersCount;
2020-05-05 05:01:40 -07:00
2020-07-22 13:11:07 -07:00
// Calculation of 100% VE air mass in g/cyl - 1 cylinder filling at 1.204/L
// 101.325kpa, 20C
return idealGasLaw(cylDisplacement, 101.325f, 273.15f + 20.0f);
2020-05-05 05:01:40 -07:00
}
2015-07-10 06:01:56 -07:00
#endif
2019-01-31 08:57:15 -08:00
#endif