2015-07-10 06:01:56 -07:00
|
|
|
/**
|
|
|
|
* @file advance_map.cpp
|
|
|
|
*
|
|
|
|
* @date Mar 27, 2013
|
2020-01-07 21:02:40 -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/>.
|
|
|
|
*/
|
|
|
|
|
2021-07-25 22:05:17 -07:00
|
|
|
#include "pch.h"
|
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
#include "advance_map.h"
|
2019-03-22 19:55:51 -07:00
|
|
|
#include "idle_thread.h"
|
2020-03-24 16:55:12 -07:00
|
|
|
#include "launch_control.h"
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2020-02-26 15:16:35 -08:00
|
|
|
#if EFI_ENGINE_CONTROL
|
|
|
|
|
2021-06-03 11:30:25 -07:00
|
|
|
static ign_Map3D_t advanceMap;
|
|
|
|
static ign_Map3D_t iatAdvanceCorrectionMap;
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2021-05-23 10:54:49 -07:00
|
|
|
// todo: reset this between cranking attempts?! #2735
|
2021-06-02 02:02:52 -07:00
|
|
|
int minCrankingRpm = 0;
|
2017-12-03 10:25:24 -08:00
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
/**
|
|
|
|
* @return ignition timing angle advance before TDC
|
|
|
|
*/
|
2021-11-16 01:15:29 -08:00
|
|
|
static angle_t getRunningAdvance(int rpm, float engineLoad) {
|
2020-03-24 16:55:12 -07:00
|
|
|
if (CONFIG(timingMode) == TM_FIXED) {
|
2017-12-03 10:22:29 -08:00
|
|
|
return engineConfiguration->fixedTiming;
|
2020-03-24 16:55:12 -07:00
|
|
|
}
|
2017-12-03 10:25:24 -08:00
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
if (cisnan(engineLoad)) {
|
2016-11-02 20:01:48 -07:00
|
|
|
warning(CUSTOM_NAN_ENGINE_LOAD, "NaN engine load");
|
2015-07-10 06:01:56 -07:00
|
|
|
return NAN;
|
|
|
|
}
|
2020-03-24 16:55:12 -07:00
|
|
|
|
2018-07-25 20:30:00 -07:00
|
|
|
efiAssert(CUSTOM_ERR_ASSERT, !cisnan(engineLoad), "invalid el", NAN);
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2020-07-21 00:08:41 -07:00
|
|
|
float advanceAngle = advanceMap.getValue((float) rpm, engineLoad);
|
2020-03-24 16:55:12 -07:00
|
|
|
|
2018-01-22 20:47:58 -08:00
|
|
|
// get advance from the separate table for Idle
|
2021-10-06 09:05:20 -07:00
|
|
|
if (CONFIG(useSeparateAdvanceForIdle) && isIdlingOrTaper()) {
|
2021-02-16 06:32:16 -08:00
|
|
|
float idleAdvance = interpolate2d(rpm, config->idleAdvanceBins, config->idleAdvance);
|
2020-04-03 16:59:08 -07:00
|
|
|
|
|
|
|
auto [valid, tps] = Sensor::get(SensorType::DriverThrottleIntent);
|
|
|
|
if (valid) {
|
|
|
|
// interpolate between idle table and normal (running) table using TPS threshold
|
|
|
|
advanceAngle = interpolateClamped(0.0f, idleAdvance, CONFIG(idlePidDeactivationTpsThreshold), advanceAngle, tps);
|
|
|
|
}
|
2018-01-22 20:47:58 -08:00
|
|
|
}
|
|
|
|
|
2020-03-24 16:55:12 -07:00
|
|
|
#if EFI_LAUNCH_CONTROL
|
2021-11-15 17:09:03 -08:00
|
|
|
if (engine->launchController.isLaunchCondition && CONFIG(enableLaunchRetard)) {
|
2020-03-24 16:55:12 -07:00
|
|
|
if (CONFIG(launchSmoothRetard)) {
|
|
|
|
float launchAngle = CONFIG(launchTimingRetard);
|
|
|
|
int launchAdvanceRpmRange = CONFIG(launchTimingRpmRange);
|
|
|
|
int launchRpm = CONFIG(launchRpm);
|
|
|
|
// interpolate timing from rpm at launch triggered to full retard at launch launchRpm + launchTimingRpmRange
|
|
|
|
return interpolateClamped(launchRpm, advanceAngle, (launchRpm + launchAdvanceRpmRange), launchAngle, rpm);
|
|
|
|
} else {
|
|
|
|
return engineConfiguration->launchTimingRetard;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif /* EFI_LAUNCH_CONTROL */
|
|
|
|
|
2018-01-22 20:47:58 -08:00
|
|
|
return advanceAngle;
|
|
|
|
}
|
|
|
|
|
2021-11-16 01:15:29 -08:00
|
|
|
angle_t getAdvanceCorrections(int rpm) {
|
2016-05-17 21:03:11 -07:00
|
|
|
float iatCorrection;
|
2020-04-18 15:45:30 -07:00
|
|
|
|
|
|
|
const auto [iatValid, iat] = Sensor::get(SensorType::Iat);
|
|
|
|
|
|
|
|
if (!iatValid) {
|
2016-05-17 21:03:11 -07:00
|
|
|
iatCorrection = 0;
|
|
|
|
} else {
|
2021-01-07 05:06:36 -08:00
|
|
|
iatCorrection = iatAdvanceCorrectionMap.getValue(rpm, iat);
|
2016-05-17 21:03:11 -07:00
|
|
|
}
|
2020-04-18 15:45:30 -07:00
|
|
|
|
2021-01-07 05:06:36 -08:00
|
|
|
float pidTimingCorrection = getIdleTimingAdjustment(rpm);
|
2019-03-22 19:55:51 -07:00
|
|
|
|
2017-02-09 09:03:46 -08:00
|
|
|
if (engineConfiguration->debugMode == DBG_IGNITION_TIMING) {
|
2019-04-12 19:07:03 -07:00
|
|
|
#if EFI_TUNER_STUDIO
|
2017-01-05 18:12:06 -08:00
|
|
|
tsOutputChannels.debugFloatField1 = iatCorrection;
|
|
|
|
tsOutputChannels.debugFloatField2 = engine->engineState.cltTimingCorrection;
|
2019-03-22 19:55:51 -07:00
|
|
|
tsOutputChannels.debugFloatField4 = pidTimingCorrection;
|
2020-03-25 22:49:36 -07:00
|
|
|
tsOutputChannels.debugIntField1 = engine->engineState.multispark.count;
|
2018-11-16 04:40:06 -08:00
|
|
|
#endif /* EFI_TUNER_STUDIO */
|
2017-01-05 18:12:06 -08:00
|
|
|
}
|
2020-03-25 22:49:36 -07:00
|
|
|
|
2018-01-22 20:47:58 -08:00
|
|
|
return iatCorrection
|
|
|
|
+ engine->engineState.cltTimingCorrection
|
2021-11-01 20:33:59 -07:00
|
|
|
+ pidTimingCorrection;
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
2018-02-02 05:18:00 -08:00
|
|
|
/**
|
|
|
|
* @return ignition timing angle advance before TDC for Cranking
|
|
|
|
*/
|
2021-11-16 01:15:29 -08:00
|
|
|
static angle_t getCrankingAdvance(int rpm, float engineLoad) {
|
2018-02-02 05:18:00 -08:00
|
|
|
// get advance from the separate table for Cranking
|
|
|
|
if (CONFIG(useSeparateAdvanceForCranking)) {
|
2021-02-16 06:32:16 -08:00
|
|
|
return interpolate2d(rpm, CONFIG(crankingAdvanceBins), CONFIG(crankingAdvance));
|
2018-02-02 05:18:00 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Interpolate the cranking timing angle to the earlier running angle for faster engine start
|
2021-11-16 01:15:29 -08:00
|
|
|
angle_t crankingToRunningTransitionAngle = getRunningAdvance(CONFIG(cranking.rpm), engineLoad);
|
2018-02-02 05:18:00 -08:00
|
|
|
// interpolate not from zero, but starting from min. possible rpm detected
|
|
|
|
if (rpm < minCrankingRpm || minCrankingRpm == 0)
|
|
|
|
minCrankingRpm = rpm;
|
|
|
|
return interpolateClamped(minCrankingRpm, CONFIG(crankingTimingAngle), CONFIG(cranking.rpm), crankingToRunningTransitionAngle, rpm);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-11-16 01:15:29 -08:00
|
|
|
angle_t getAdvance(int rpm, float engineLoad) {
|
2019-04-12 19:07:03 -07:00
|
|
|
#if EFI_ENGINE_CONTROL && EFI_SHAFT_POSITION_INPUT
|
2017-12-03 13:45:39 -08:00
|
|
|
if (cisnan(engineLoad)) {
|
|
|
|
return 0; // any error should already be reported
|
|
|
|
}
|
2020-04-02 05:54:55 -07:00
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
angle_t angle;
|
2020-04-02 05:54:55 -07:00
|
|
|
|
2020-09-05 15:49:42 -07:00
|
|
|
bool isCranking = ENGINE(rpmCalculator).isCranking();
|
2020-04-02 05:54:55 -07:00
|
|
|
if (isCranking) {
|
2021-11-16 01:15:29 -08:00
|
|
|
angle = getCrankingAdvance(rpm, engineLoad);
|
2021-10-02 22:30:42 -07:00
|
|
|
assertAngleRange(angle, "crAngle", CUSTOM_ERR_ANGLE_CR);
|
2018-09-16 19:11:59 -07:00
|
|
|
efiAssert(CUSTOM_ERR_ASSERT, !cisnan(angle), "cr_AngleN", 0);
|
2015-07-10 06:01:56 -07:00
|
|
|
} else {
|
2021-11-16 01:15:29 -08:00
|
|
|
angle = getRunningAdvance(rpm, engineLoad);
|
2020-04-02 05:54:55 -07:00
|
|
|
|
2018-09-16 19:11:59 -07:00
|
|
|
if (cisnan(angle)) {
|
|
|
|
warning(CUSTOM_ERR_6610, "NaN angle from table");
|
|
|
|
return 0;
|
|
|
|
}
|
2020-04-02 05:54:55 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Allow correction only if set to dynamic
|
|
|
|
// AND we're either not cranking OR allowed to correct in cranking
|
|
|
|
bool allowCorrections = CONFIG(timingMode) == TM_DYNAMIC
|
|
|
|
&& (!isCranking || CONFIG(useAdvanceCorrectionsForCranking));
|
|
|
|
|
|
|
|
if (allowCorrections) {
|
2021-11-16 01:15:29 -08:00
|
|
|
angle_t correction = getAdvanceCorrections(rpm);
|
2018-10-07 11:22:05 -07:00
|
|
|
if (!cisnan(correction)) { // correction could be NaN during settings update
|
|
|
|
angle += correction;
|
|
|
|
}
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
2020-04-02 05:54:55 -07:00
|
|
|
|
2018-09-29 11:11:25 -07:00
|
|
|
efiAssert(CUSTOM_ERR_ASSERT, !cisnan(angle), "_AngleN5", 0);
|
2019-01-11 06:24:24 -08:00
|
|
|
fixAngle(angle, "getAdvance", CUSTOM_ERR_ADCANCE_CALC_ANGLE);
|
2015-07-10 06:01:56 -07:00
|
|
|
return angle;
|
2019-01-31 14:55:23 -08:00
|
|
|
#else
|
|
|
|
return 0;
|
|
|
|
#endif
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
2021-11-16 01:15:29 -08:00
|
|
|
size_t getMultiSparkCount(int rpm) {
|
2020-03-25 22:49:36 -07:00
|
|
|
// Compute multispark (if enabled)
|
|
|
|
if (CONFIG(multisparkEnable)
|
|
|
|
&& rpm <= CONFIG(multisparkMaxRpm)
|
|
|
|
&& CONFIG(multisparkMaxExtraSparkCount) > 0) {
|
|
|
|
// For zero RPM, disable multispark. We don't yet know the engine speed, so multispark may not be safe.
|
|
|
|
if (rpm == 0) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
floatus_t multiDelay = CONFIG(multisparkSparkDuration);
|
|
|
|
floatus_t multiDwell = CONFIG(multisparkDwell);
|
|
|
|
|
2020-12-03 10:32:01 -08:00
|
|
|
// dwell times are below 10 seconds here so we use 32 bit type for performance reasons
|
|
|
|
ENGINE(engineState.multispark.delay) = (uint32_t)USF2NT(multiDelay);
|
|
|
|
ENGINE(engineState.multispark.dwell) = (uint32_t)USF2NT(multiDwell);
|
2020-03-25 22:49:36 -07:00
|
|
|
|
|
|
|
constexpr float usPerDegreeAt1Rpm = 60e6 / 360;
|
|
|
|
floatus_t usPerDegree = usPerDegreeAt1Rpm / rpm;
|
|
|
|
|
|
|
|
// How long is there for sparks? The user configured an angle, convert to time.
|
|
|
|
floatus_t additionalSparksUs = usPerDegree * CONFIG(multisparkMaxSparkingAngle);
|
|
|
|
// How long does one spark take?
|
|
|
|
floatus_t oneSparkTime = multiDelay + multiDwell;
|
|
|
|
|
|
|
|
// How many sparks can we fit in the alloted time?
|
|
|
|
float sparksFitInTime = additionalSparksUs / oneSparkTime;
|
|
|
|
|
|
|
|
// Take the floor (convert to uint8_t) - we want to undershoot, not overshoot
|
|
|
|
uint32_t floored = sparksFitInTime;
|
|
|
|
|
|
|
|
// Allow no more than the maximum number of extra sparks
|
|
|
|
return minI(floored, CONFIG(multisparkMaxExtraSparkCount));
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-11-16 01:15:29 -08:00
|
|
|
void initTimingMap() {
|
2018-09-29 09:16:36 -07:00
|
|
|
// We init both tables in RAM because here we're at a very early stage, with no config settings loaded.
|
2015-07-10 06:01:56 -07:00
|
|
|
advanceMap.init(config->ignitionTable, config->ignitionLoadBins,
|
|
|
|
config->ignitionRpmBins);
|
|
|
|
iatAdvanceCorrectionMap.init(config->ignitionIatCorrTable, config->ignitionIatCorrLoadBins,
|
|
|
|
config->ignitionIatCorrRpmBins);
|
|
|
|
}
|
2015-12-24 11:02:03 -08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @param octane gas octane number
|
|
|
|
* @param bore in mm
|
|
|
|
*/
|
|
|
|
float getTopAdvanceForBore(chamber_style_e style, int octane, double compression, double bore) {
|
|
|
|
int octaneCorrection;
|
|
|
|
if ( octane <= 90) {
|
|
|
|
octaneCorrection = -2;
|
|
|
|
} else if (octane < 94) {
|
|
|
|
octaneCorrection = -1;
|
|
|
|
} else {
|
|
|
|
octaneCorrection = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int compressionCorrection;
|
|
|
|
if (compression <= 9) {
|
|
|
|
compressionCorrection = 2;
|
|
|
|
} else if (compression <= 10) {
|
|
|
|
compressionCorrection = 1;
|
|
|
|
} else if (compression <= 11) {
|
|
|
|
compressionCorrection = 0;
|
|
|
|
} else {
|
|
|
|
// compression ratio above 11
|
|
|
|
compressionCorrection = -2;
|
|
|
|
}
|
|
|
|
int base;
|
|
|
|
if (style == CS_OPEN) {
|
|
|
|
base = 33;
|
|
|
|
} else if (style == CS_CLOSED) {
|
|
|
|
base = 28;
|
|
|
|
} else {
|
|
|
|
// CS_SWIRL_TUMBLE
|
|
|
|
base = 22;
|
|
|
|
}
|
|
|
|
|
|
|
|
float boreCorrection = (bore - 4 * 25.4) / 25.4 * 6;
|
|
|
|
float result = base + octaneCorrection + compressionCorrection + boreCorrection;
|
|
|
|
return ((int)(result * 10)) / 10.0;
|
|
|
|
}
|
2015-12-31 10:02:19 -08:00
|
|
|
|
|
|
|
float getAdvanceForRpm(int rpm, float advanceMax) {
|
|
|
|
if (rpm >= 3000)
|
|
|
|
return advanceMax;
|
|
|
|
if (rpm < 600)
|
|
|
|
return 10;
|
2018-06-12 02:45:11 -07:00
|
|
|
return interpolateMsg("advance", 600, 10, 3000, advanceMax, rpm);
|
2015-12-31 10:02:19 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
#define round10(x) efiRound(x, 0.1)
|
|
|
|
|
|
|
|
float getInitialAdvance(int rpm, float map, float advanceMax) {
|
2016-01-21 19:01:31 -08:00
|
|
|
map = minF(map, 100);
|
2015-12-31 10:02:19 -08:00
|
|
|
float advance = getAdvanceForRpm(rpm, advanceMax);
|
2016-01-21 19:01:31 -08:00
|
|
|
|
|
|
|
if (rpm >= 3000)
|
|
|
|
return round10(advance + 0.1 * (100 - map));
|
|
|
|
return round10(advance + 0.1 * (100 - map) * rpm / 3000);
|
2015-12-31 10:02:19 -08:00
|
|
|
}
|
|
|
|
|
2016-01-21 14:02:40 -08:00
|
|
|
/**
|
|
|
|
* this method builds a good-enough base timing advance map bases on a number of heuristics
|
|
|
|
*/
|
2021-11-16 01:15:29 -08:00
|
|
|
void buildTimingMap(float advanceMax) {
|
2020-05-05 12:04:55 -07:00
|
|
|
if (engineConfiguration->fuelAlgorithm != LM_SPEED_DENSITY) {
|
2016-11-02 20:01:48 -07:00
|
|
|
warning(CUSTOM_WRONG_ALGORITHM, "wrong algorithm for MAP-based timing");
|
2015-12-31 10:02:19 -08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* good enough (but do not trust us!) default timing map in case of MAP-based engine load
|
|
|
|
*/
|
|
|
|
for (int loadIndex = 0; loadIndex < IGN_LOAD_COUNT; loadIndex++) {
|
|
|
|
float load = config->ignitionLoadBins[loadIndex];
|
|
|
|
for (int rpmIndex = 0;rpmIndex<IGN_RPM_COUNT;rpmIndex++) {
|
2016-01-21 19:01:31 -08:00
|
|
|
float rpm = config->ignitionRpmBins[rpmIndex];
|
2015-12-31 10:02:19 -08:00
|
|
|
config->ignitionTable[loadIndex][rpmIndex] = getInitialAdvance(rpm, load, advanceMax);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-26 15:16:35 -08:00
|
|
|
#endif // EFI_ENGINE_CONTROL
|