rusefi/firmware/controllers/algo/accel_enrichment.cpp

233 lines
6.8 KiB
C++
Raw Normal View History

2015-07-10 06:01:56 -07:00
/**
* @file accel_enrichment.cpp
* @brief Acceleration enrichment calculator
*
2015-08-21 12:01:36 -07:00
* In this file we have three strategies for acceleration/deceleration fuel correction
*
* 1) MAP rate-of-change correction
* 2) TPS rate-of-change correction
* 3) fuel film/wal wetting correction
* AWC Added to Wall Coefficient, %
* AWA Added to Wall Amount
* SOC Sucked Off wall Coefficient, %
* SOA Sucked Off wall amount
* WF current on-Wall Fuel amount
*
*
* http://rusefi.com/wiki/index.php?title=Manual:Software:Fuel_Control
2015-07-10 06:01:56 -07:00
* @date Apr 21, 2014
* @author Dmitry Sidin
2020-01-13 18:57:43 -08:00
* @author Andrey Belomutskiy, (c) 2012-2020
* @author Matthew Kennedy
2015-07-10 06:01:56 -07:00
*/
#include "pch.h"
2015-07-10 06:01:56 -07:00
#include "accel_enrichment.h"
static tps_tps_Map3D_t tpsTpsMap;
2016-03-11 08:03:18 -08:00
2016-01-02 13:01:27 -08:00
// todo: eliminate code duplication between these two methods! Some pointer magic would help.
floatms_t TpsAccelEnrichment::getTpsEnrichment(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
2019-10-13 13:14:08 -07:00
ScopePerf perf(PE::GetTpsEnrichment);
2019-03-16 08:14:52 -07:00
int maxDeltaIndex = getMaxDeltaIndex(PASS_ENGINE_PARAMETER_SIGNATURE);
2016-03-11 08:03:18 -08:00
// FuelSchedule *fs = engineConfiguration->injectionEvents;
2019-03-16 08:14:52 -07:00
percent_t tpsTo = cb.get(maxDeltaIndex);
percent_t tpsFrom = cb.get(maxDeltaIndex - 1);
percent_t deltaTps = tpsTo - tpsFrom;
2016-03-11 08:03:18 -08:00
isAboveAccelThreshold = deltaTps > engineConfiguration->tpsAccelEnrichmentThreshold;
isBelowDecelThreshold = deltaTps < -engineConfiguration->tpsDecelEnleanmentThreshold;
if (isAboveAccelThreshold) {
valueFromTable = tpsTpsMap.getValue(tpsFrom, tpsTo);
2016-09-17 16:02:34 -07:00
extraFuel = valueFromTable;
} else if (isBelowDecelThreshold) {
2019-03-16 08:14:52 -07:00
extraFuel = deltaTps * engineConfiguration->tpsDecelEnleanmentMultiplier;
2016-03-11 11:03:23 -08:00
} else {
2016-03-12 17:03:40 -08:00
extraFuel = 0;
2016-03-11 11:03:23 -08:00
}
// Fractional enrichment (fuel portions are accumulated and split between several engine cycles.
// This is a crude imitation of carburetor's acceleration pump.
isFractionalEnrichment = CONFIG(tpsAccelFractionPeriod) > 1 || CONFIG(tpsAccelFractionDivisor) > 1.0f;
if (isFractionalEnrichment) {
// make sure both values are non-zero
float periodF = (float)maxI(CONFIG(tpsAccelFractionPeriod), 1);
float divisor = maxF(CONFIG(tpsAccelFractionDivisor), 1.0f);
// if current extra fuel portion is not "strong" enough, then we keep up the "pump pressure" with the accumulated portion
floatms_t maxExtraFuel = maxF(extraFuel, accumulatedValue);
// use only a fixed fraction of the accumulated portion
fractionalInjFuel = maxExtraFuel / divisor;
// update max counters
maxExtraPerCycle = maxF(extraFuel, maxExtraPerCycle);
maxInjectedPerPeriod = maxF(fractionalInjFuel, maxInjectedPerPeriod);
// evenly split it between several engine cycles
extraFuel = fractionalInjFuel / periodF;
} else {
resetFractionValues();
}
2019-04-12 19:07:03 -07:00
#if EFI_TUNER_STUDIO
if (engineConfiguration->debugMode == DBG_TPS_ACCEL) {
2016-03-11 08:03:18 -08:00
tsOutputChannels.debugFloatField1 = tpsFrom;
tsOutputChannels.debugFloatField2 = tpsTo;
2016-09-17 16:02:34 -07:00
tsOutputChannels.debugFloatField3 = valueFromTable;
2016-03-12 17:03:40 -08:00
tsOutputChannels.debugFloatField4 = extraFuel;
tsOutputChannels.debugFloatField5 = accumulatedValue;
tsOutputChannels.debugFloatField6 = maxExtraPerPeriod;
tsOutputChannels.debugFloatField7 = maxInjectedPerPeriod;
tsOutputChannels.debugIntField1 = cycleCnt;
2016-03-11 08:03:18 -08:00
}
#endif /* EFI_TUNER_STUDIO */
2016-03-11 11:03:23 -08:00
2016-03-12 17:03:40 -08:00
return extraFuel;
2015-07-10 06:01:56 -07:00
}
void TpsAccelEnrichment::onEngineCycleTps(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
// we update values in handleFuel() directly by calling onNewValue()
2019-10-16 19:10:38 -07:00
onUpdateInvocationCounter++;
// we used some extra fuel during the current cycle, so we "charge" our "acceleration pump" with it
accumulatedValue -= maxExtraPerPeriod;
maxExtraPerPeriod = maxF(maxExtraPerCycle, maxExtraPerPeriod);
maxExtraPerCycle = 0;
accumulatedValue += maxExtraPerPeriod;
// update the accumulated value every 'Period' engine cycles
isTimeToResetAccumulator = --cycleCnt <= 0;
if (isTimeToResetAccumulator) {
maxExtraPerPeriod = 0;
// we've injected this portion during the cycle, so we set what's left for the next cycle
accumulatedValue -= maxInjectedPerPeriod;
maxInjectedPerPeriod = 0;
// it's an infinitely convergent series, so we set a limit at some point
// (also make sure that accumulatedValue is positive, for safety)
static const floatms_t smallEpsilon = 0.001f;
belowEpsilon = accumulatedValue < smallEpsilon;
if (belowEpsilon) {
accumulatedValue = 0;
}
// reset the counter
cycleCnt = CONFIG(tpsAccelFractionPeriod);
}
2015-07-10 06:01:56 -07:00
}
2021-10-31 09:07:44 -07:00
int AccelEnrichment::getMaxDeltaIndex(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
int len = minI(cb.getSize(), cb.getCount());
if (len < 2)
return 0;
int ci = cb.currentIndex - 1;
float maxValue = cb.get(ci) - cb.get(ci - 1);
int resultIndex = ci;
// todo: 'get' method is maybe a bit heavy because of the branching
// todo: this could be optimized with some careful magic
for (int i = 1; i<len - 1;i++) {
float v = cb.get(ci - i) - cb.get(ci - i - 1);
if (v > maxValue) {
maxValue = v;
resultIndex = ci - i;
}
}
return resultIndex;
}
float AccelEnrichment::getMaxDelta(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
int index = getMaxDeltaIndex(PASS_ENGINE_PARAMETER_SIGNATURE);
return (cb.get(index) - (cb.get(index - 1)));
}
void AccelEnrichment::resetAE() {
cb.clear();
}
void TpsAccelEnrichment::resetAE() {
AccelEnrichment::resetAE();
resetFractionValues();
}
void TpsAccelEnrichment::resetFractionValues() {
accumulatedValue = 0;
maxExtraPerCycle = 0;
maxExtraPerPeriod = 0;
maxInjectedPerPeriod = 0;
cycleCnt = 0;
}
void AccelEnrichment::setLength(int length) {
cb.setSize(length);
}
void AccelEnrichment::onNewValue(float currentValue DECLARE_ENGINE_PARAMETER_SUFFIX) {
cb.add(currentValue);
}
AccelEnrichment::AccelEnrichment() {
resetAE();
2015-07-10 06:01:56 -07:00
cb.setSize(4);
}
2019-04-12 19:07:03 -07:00
#if ! EFI_UNIT_TEST
2015-07-10 06:01:56 -07:00
static void accelInfo() {
// efiPrintf("TPS accel length=%d", tpsInstance.cb.getSize());
efiPrintf("TPS accel th=%.2f/mult=%.2f", engineConfiguration->tpsAccelEnrichmentThreshold, -1);
2015-08-23 22:01:28 -07:00
efiPrintf("beta=%.2f/tau=%.2f", engineConfiguration->wwaeBeta, engineConfiguration->wwaeTau);
2015-07-10 06:01:56 -07:00
}
2017-01-06 07:04:41 -08:00
void setTpsAccelThr(float value) {
2015-07-10 06:01:56 -07:00
engineConfiguration->tpsAccelEnrichmentThreshold = value;
accelInfo();
}
2017-01-06 07:04:41 -08:00
void setTpsDecelThr(float value) {
2015-12-31 15:02:17 -08:00
engineConfiguration->tpsDecelEnleanmentThreshold = value;
accelInfo();
}
2017-01-06 07:04:41 -08:00
void setTpsDecelMult(float value) {
2015-12-31 15:02:17 -08:00
engineConfiguration->tpsDecelEnleanmentMultiplier = value;
accelInfo();
}
2017-01-06 07:04:41 -08:00
void setTpsAccelLen(int length) {
2016-03-09 20:02:39 -08:00
if (length < 1) {
efiPrintf("Length should be positive");
2015-07-10 06:01:56 -07:00
return;
}
2016-03-09 20:02:39 -08:00
engine->tpsAccelEnrichment.setLength(length);
2015-07-10 06:01:56 -07:00
accelInfo();
}
2016-01-31 12:01:41 -08:00
void updateAccelParameters() {
setTpsAccelLen(engineConfiguration->tpsAccelLength);
}
2016-03-11 08:03:18 -08:00
#endif /* ! EFI_UNIT_TEST */
void initAccelEnrichment(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
2016-03-11 08:03:18 -08:00
tpsTpsMap.init(config->tpsTpsAccelTable, config->tpsTpsAccelFromRpmBins, config->tpsTpsAccelToRpmBins);
2019-04-12 19:07:03 -07:00
#if ! EFI_UNIT_TEST
2017-01-06 13:03:41 -08:00
2015-07-10 06:01:56 -07:00
addConsoleAction("accelinfo", accelInfo);
2016-01-31 12:01:41 -08:00
updateAccelParameters();
2015-07-10 06:01:56 -07:00
#endif /* ! EFI_UNIT_TEST */
2016-03-11 08:03:18 -08:00
}