rusefi-full/firmware/controllers/math/engine_math.cpp

557 lines
18 KiB
C++
Raw Normal View History

2015-07-10 06:01:56 -07:00
/**
* @file engine_math.cpp
* @brief
*
* @date Jul 13, 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/>.
*/
2018-09-16 19:26:57 -07:00
#include "global.h"
2015-07-10 06:01:56 -07:00
#include "engine_math.h"
#include "engine_configuration.h"
#include "interpolation.h"
#include "allsensors.h"
#include "sensor.h"
2015-07-10 06:01:56 -07:00
#include "event_registry.h"
2019-03-29 06:11:13 -07:00
#include "efi_gpio.h"
2015-07-10 06:01:56 -07:00
#include "fuel_math.h"
2016-01-08 12:01:38 -08:00
#include "advance_map.h"
2019-01-28 00:25:28 -08:00
#include "config_engine_specs.h"
2015-07-10 06:01:56 -07:00
EXTERN_ENGINE;
2019-08-24 22:15:18 -07:00
#if EFI_UNIT_TEST
extern bool verboseMode;
#endif /* EFI_UNIT_TEST */
2015-07-10 06:01:56 -07:00
floatms_t getEngineCycleDuration(int rpm DECLARE_ENGINE_PARAMETER_SUFFIX) {
return getCrankshaftRevolutionTimeMs(rpm) * (engine->getOperationMode(PASS_ENGINE_PARAMETER_SIGNATURE) == TWO_STROKE ? 1 : 2);
2017-03-08 21:51:27 -08:00
}
2015-07-10 06:01:56 -07:00
/**
2016-01-08 12:01:38 -08:00
* @return number of milliseconds in one crank shaft revolution
2015-07-10 06:01:56 -07:00
*/
2015-09-07 06:03:24 -07:00
floatms_t getCrankshaftRevolutionTimeMs(int rpm) {
2016-08-30 19:02:21 -07:00
if (rpm == 0) {
return NAN;
}
2015-07-10 06:01:56 -07:00
return 360 * getOneDegreeTimeMs(rpm);
}
/**
* @brief Returns engine load according to selected engine_load_mode
*
*/
2017-05-15 20:28:49 -07:00
float getEngineLoadT(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
2018-07-25 20:30:00 -07:00
efiAssert(CUSTOM_ERR_ASSERT, engine!=NULL, "engine 2NULL", NAN);
efiAssert(CUSTOM_ERR_ASSERT, engineConfiguration!=NULL, "engineConfiguration 2NULL", NAN);
2016-08-28 13:02:34 -07:00
switch (engineConfiguration->fuelAlgorithm) {
2015-07-10 06:01:56 -07:00
case LM_PLAIN_MAF:
2017-05-15 20:28:49 -07:00
if (!hasMafSensor(PASS_ENGINE_PARAMETER_SIGNATURE)) {
2017-04-19 19:03:14 -07:00
warning(CUSTOM_MAF_NEEDED, "MAF sensor needed for current fuel algorithm");
2015-07-10 06:01:56 -07:00
return NAN;
}
return getMafVoltage(PASS_ENGINE_PARAMETER_SIGNATURE);
2015-07-10 06:01:56 -07:00
case LM_SPEED_DENSITY:
// SD engine load is used for timing lookup but not for fuel calculation,
// so fall thru to the MAP case.
2015-07-10 06:01:56 -07:00
case LM_MAP:
return getMap(PASS_ENGINE_PARAMETER_SIGNATURE);
2015-07-10 06:01:56 -07:00
case LM_ALPHA_N:
return Sensor::get(SensorType::Tps1).value_or(0);
2015-07-10 06:01:56 -07:00
case LM_REAL_MAF: {
2017-05-15 20:28:49 -07:00
return getRealMaf(PASS_ENGINE_PARAMETER_SIGNATURE);
2015-07-10 06:01:56 -07:00
}
default:
2017-01-22 06:03:08 -08:00
warning(CUSTOM_UNKNOWN_ALGORITHM, "Unexpected engine load parameter: %d", engineConfiguration->fuelAlgorithm);
2015-07-10 06:01:56 -07:00
return -1;
}
}
2019-11-05 20:17:44 -08:00
/**
* see also setConstantDwell
*/
void setSingleCoilDwell(DECLARE_CONFIG_PARAMETER_SIGNATURE) {
2015-07-10 06:01:56 -07:00
for (int i = 0; i < DWELL_CURVE_SIZE; i++) {
2017-02-18 12:01:47 -08:00
engineConfiguration->sparkDwellRpmBins[i] = i + 1;
engineConfiguration->sparkDwellValues[i] = 4;
2015-07-10 06:01:56 -07:00
}
2017-02-18 12:01:47 -08:00
engineConfiguration->sparkDwellRpmBins[5] = 10;
engineConfiguration->sparkDwellValues[5] = 4;
2015-07-10 06:01:56 -07:00
2017-02-18 12:01:47 -08:00
engineConfiguration->sparkDwellRpmBins[6] = 4500;
engineConfiguration->sparkDwellValues[6] = 4;
2015-07-10 06:01:56 -07:00
2017-02-18 12:01:47 -08:00
engineConfiguration->sparkDwellRpmBins[7] = 12500;
engineConfiguration->sparkDwellValues[7] = 0;
2015-07-10 06:01:56 -07:00
}
2019-04-12 19:07:03 -07:00
#if EFI_ENGINE_CONTROL
2015-07-10 06:01:56 -07:00
2016-11-30 17:02:41 -08:00
FuelSchedule::FuelSchedule() {
clear();
2018-07-28 11:28:52 -07:00
for (int cylinderIndex = 0; cylinderIndex < MAX_INJECTION_OUTPUT_COUNT; cylinderIndex++) {
InjectionEvent *ev = &elements[cylinderIndex];
ev->ownIndex = cylinderIndex;
}
2016-11-30 17:02:41 -08:00
}
void FuelSchedule::clear() {
2016-11-30 19:06:43 -08:00
isReady = false;
2016-11-30 17:02:41 -08:00
}
2017-03-12 12:57:33 -07:00
/**
* @returns false in case of error, true if success
*/
2017-05-15 20:28:49 -07:00
bool FuelSchedule::addFuelEventsForCylinder(int i DECLARE_ENGINE_PARAMETER_SUFFIX) {
2018-07-25 20:30:00 -07:00
efiAssert(CUSTOM_ERR_ASSERT, engine!=NULL, "engine is NULL", false);
2016-11-30 17:02:41 -08:00
2017-04-12 14:51:35 -07:00
floatus_t oneDegreeUs = ENGINE(rpmCalculator.oneDegreeUs); // local copy
2017-04-12 12:44:24 -07:00
if (cisnan(oneDegreeUs)) {
2016-11-30 17:02:41 -08:00
// in order to have fuel schedule we need to have current RPM
// wonder if this line slows engine startup?
2016-12-25 19:02:40 -08:00
return false;
2016-11-30 17:02:41 -08:00
}
/**
* injection phase is scheduled by injection end, so we need to step the angle back
* for the duration of the injection
*
* todo: since this method is not invoked within trigger event handler and
* engineState.injectionOffset is calculated from the same utility timer should we more that logic here?
*/
2017-11-06 19:29:39 -08:00
floatms_t fuelMs = ENGINE(injectionDuration);
2018-07-25 20:30:00 -07:00
efiAssert(CUSTOM_ERR_ASSERT, !cisnan(fuelMs), "NaN fuelMs", false);
2017-04-12 14:51:35 -07:00
angle_t injectionDuration = MS2US(fuelMs) / oneDegreeUs;
2018-07-25 20:30:00 -07:00
efiAssert(CUSTOM_ERR_ASSERT, !cisnan(injectionDuration), "NaN injectionDuration", false);
2018-09-10 19:00:13 -07:00
assertAngleRange(injectionDuration, "injectionDuration_r", CUSTOM_INJ_DURATION);
2017-04-12 14:51:35 -07:00
floatus_t injectionOffset = ENGINE(engineState.injectionOffset);
2017-04-13 08:19:36 -07:00
if (cisnan(injectionOffset)) {
// injection offset map not ready - we are not ready to schedule fuel events
return false;
}
2017-08-26 17:10:50 -07:00
angle_t baseAngle = injectionOffset - injectionDuration;
2018-07-25 20:30:00 -07:00
efiAssert(CUSTOM_ERR_ASSERT, !cisnan(baseAngle), "NaN baseAngle", false);
2018-07-23 18:03:20 -07:00
assertAngleRange(baseAngle, "baseAngle_r", CUSTOM_ERR_6554);
2016-11-30 17:02:41 -08:00
2017-11-06 18:30:39 -08:00
int injectorIndex;
2016-11-30 17:02:41 -08:00
2017-05-15 20:28:49 -07:00
injection_mode_e mode = engine->getCurrentInjectionMode(PASS_ENGINE_PARAMETER_SIGNATURE);
2016-11-30 19:06:43 -08:00
2017-11-06 16:00:30 -08:00
if (mode == IM_SIMULTANEOUS || mode == IM_SINGLE_POINT) {
2017-11-06 18:30:39 -08:00
injectorIndex = 0;
2016-11-30 17:02:41 -08:00
} else if (mode == IM_SEQUENTIAL) {
2017-11-06 18:30:39 -08:00
injectorIndex = getCylinderId(i PASS_ENGINE_PARAMETER_SUFFIX) - 1;
2016-11-30 17:02:41 -08:00
} else if (mode == IM_BATCH) {
// does not look exactly right, not too consistent with IM_SEQUENTIAL
2017-11-06 18:30:39 -08:00
injectorIndex = i % (engineConfiguration->specs.cylindersCount / 2);
2016-11-30 17:02:41 -08:00
} else {
2016-12-18 16:02:00 -08:00
warning(CUSTOM_OBD_UNEXPECTED_INJECTION_MODE, "Unexpected injection mode %d", mode);
2017-11-06 18:30:39 -08:00
injectorIndex = 0;
2016-11-30 17:02:41 -08:00
}
bool isSimultanious = mode == IM_SIMULTANEOUS;
2018-09-10 19:00:13 -07:00
assertAngleRange(baseAngle, "addFbaseAngle", CUSTOM_ADD_BASE);
2016-12-06 19:03:16 -08:00
2016-12-20 20:02:52 -08:00
int cylindersCount = CONFIG(specs.cylindersCount);
2016-12-20 21:02:53 -08:00
if (cylindersCount < 1) {
2016-12-25 19:02:40 -08:00
warning(CUSTOM_OBD_ZERO_CYLINDER_COUNT, "temp cylindersCount %d", cylindersCount);
return false;
2016-12-20 21:02:53 -08:00
}
2016-12-20 20:02:52 -08:00
2016-11-30 17:02:41 -08:00
float angle = baseAngle
2016-12-20 20:02:52 -08:00
+ i * ENGINE(engineCycle) / cylindersCount;
2016-11-30 17:02:41 -08:00
InjectorOutputPin *secondOutput;
if (mode == IM_BATCH && CONFIG(twoWireBatchInjection)) {
/**
* also fire the 2nd half of the injectors so that we can implement a batch mode on individual wires
*/
2017-11-06 18:30:39 -08:00
int secondIndex = injectorIndex + (CONFIG(specs.cylindersCount) / 2);
2016-11-30 17:02:41 -08:00
secondOutput = &enginePins.injectors[secondIndex];
} else {
secondOutput = NULL;
}
2015-08-23 19:01:55 -07:00
2017-11-06 18:30:39 -08:00
InjectorOutputPin *output = &enginePins.injectors[injectorIndex];
2015-08-23 19:01:55 -07:00
2017-07-10 18:47:51 -07:00
if (!isSimultanious && !output->isInitialized()) {
2016-08-28 13:02:34 -07:00
// todo: extract method for this index math
2016-12-18 16:02:00 -08:00
warning(CUSTOM_OBD_INJECTION_NO_PIN_ASSIGNED, "no_pin_inj #%s", output->name);
2015-07-10 06:01:56 -07:00
}
2016-11-30 19:06:43 -08:00
InjectionEvent *ev = &elements[i];
ev->ownIndex = i;
2019-12-02 20:08:18 -08:00
INJECT_ENGINE_REFERENCE(ev);
2018-07-23 18:38:05 -07:00
fixAngle(angle, "addFuel#1", CUSTOM_ERR_6554);
2015-07-10 06:01:56 -07:00
2016-11-30 16:01:43 -08:00
ev->outputs[0] = output;
2016-11-30 17:02:41 -08:00
ev->outputs[1] = secondOutput;
2015-07-10 06:01:56 -07:00
ev->isSimultanious = isSimultanious;
if (TRIGGER_WAVEFORM(getSize()) < 1) {
warning(CUSTOM_ERR_NOT_INITIALIZED_TRIGGER, "uninitialized TriggerWaveform");
2017-03-12 12:57:33 -07:00
return false;
}
2015-07-10 06:01:56 -07:00
2018-07-25 20:30:00 -07:00
efiAssert(CUSTOM_ERR_ASSERT, !cisnan(angle), "findAngle#3", false);
2018-07-23 18:03:20 -07:00
assertAngleRange(angle, "findAngle#a33", CUSTOM_ERR_6544);
2019-11-23 21:15:44 -08:00
ev->injectionStart.setAngle(angle PASS_ENGINE_PARAMETER_SUFFIX);
2019-04-12 19:07:03 -07:00
#if EFI_UNIT_TEST
2019-10-07 21:54:19 -07:00
printf("registerInjectionEvent angle=%.2f trgIndex=%d inj %d\r\n", angle, ev->injectionStart.triggerEventIndex, injectorIndex);
2016-01-30 19:03:36 -08:00
#endif
2016-12-25 19:02:40 -08:00
return true;
2015-07-10 06:01:56 -07:00
}
2017-05-15 20:28:49 -07:00
void FuelSchedule::addFuelEvents(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
2016-11-30 18:06:24 -08:00
clear();
2015-07-10 06:01:56 -07:00
2018-07-28 11:28:52 -07:00
for (int cylinderIndex = 0; cylinderIndex < CONFIG(specs.cylindersCount); cylinderIndex++) {
InjectionEvent *ev = &elements[cylinderIndex];
ev->ownIndex = cylinderIndex; // todo: is this assignment needed here? we now initialize in constructor
bool result = addFuelEventsForCylinder(cylinderIndex PASS_ENGINE_PARAMETER_SUFFIX);
2016-12-25 19:02:40 -08:00
if (!result)
return;
2015-07-10 06:01:56 -07:00
}
2016-11-30 19:06:43 -08:00
isReady = true;
2015-07-10 06:01:56 -07:00
}
#endif
2017-07-06 08:19:02 -07:00
static floatms_t getCrankingSparkDwell(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
2016-11-11 16:02:29 -08:00
if (engineConfiguration->useConstantDwellDuringCranking) {
return engineConfiguration->ignitionDwellForCrankingMs;
} else {
// technically this could be implemented via interpolate2d
float angle = engineConfiguration->crankingChargeAngle;
2019-01-21 17:33:21 -08:00
return getOneDegreeTimeMs(GET_RPM_VALUE) * angle;
2016-11-11 16:02:29 -08:00
}
}
2015-07-10 06:01:56 -07:00
/**
2017-04-18 05:16:53 -07:00
* @return Spark dwell time, in milliseconds. 0 if tables are not ready.
2015-07-10 06:01:56 -07:00
*/
2017-05-15 20:28:49 -07:00
floatms_t getSparkDwell(int rpm DECLARE_ENGINE_PARAMETER_SUFFIX) {
2019-04-12 19:07:03 -07:00
#if EFI_ENGINE_CONTROL && EFI_SHAFT_POSITION_INPUT
2016-11-30 12:02:43 -08:00
float dwellMs;
2017-07-06 05:49:55 -07:00
if (ENGINE(rpmCalculator).isCranking(PASS_ENGINE_PARAMETER_SIGNATURE)) {
2017-07-06 08:19:02 -07:00
dwellMs = getCrankingSparkDwell(PASS_ENGINE_PARAMETER_SIGNATURE);
2016-11-30 12:02:43 -08:00
} else {
2018-07-25 20:30:00 -07:00
efiAssert(CUSTOM_ERR_ASSERT, !cisnan(rpm), "invalid rpm", NAN);
2016-11-30 12:02:43 -08:00
dwellMs = interpolate2d("dwell", rpm, engineConfiguration->sparkDwellRpmBins, engineConfiguration->sparkDwellValues);
2015-07-10 06:01:56 -07:00
}
2017-04-18 05:16:53 -07:00
if (cisnan(dwellMs) || dwellMs <= 0) {
// this could happen during engine configuration reset
2018-01-23 09:05:14 -08:00
warning(CUSTOM_ERR_DWELL_DURATION, "invalid dwell: %.2f at rpm=%d", dwellMs, rpm);
2017-04-18 05:16:53 -07:00
return 0;
2016-11-30 12:02:43 -08:00
}
return dwellMs;
2019-01-31 14:55:23 -08:00
#else
return 0;
#endif
2015-07-10 06:01:56 -07:00
}
2019-01-05 05:06:18 -08:00
static const int order_1_2[] = {1, 2};
2017-02-07 22:03:36 -08:00
2019-01-05 05:06:18 -08:00
static const int order_1_2_3[] = {1, 2, 3};
2017-02-07 22:03:36 -08:00
// 4 cylinder
2019-01-05 05:06:18 -08:00
static const int order_1_THEN_3_THEN_4_THEN2[] = { 1, 3, 4, 2 };
static const int order_1_THEN_2_THEN_4_THEN3[] = { 1, 2, 4, 3 };
static const int order_1_THEN_3_THEN_2_THEN4[] = { 1, 3, 2, 4 };
2019-08-15 16:36:16 -07:00
static const int order_1_THEN_4_THEN_3_THEN2[] = { 1, 4, 3, 2 };
2015-07-10 06:01:56 -07:00
2017-02-07 22:03:36 -08:00
// 5 cylinder
2019-01-05 05:06:18 -08:00
static const int order_1_2_4_5_3[] = {1, 2, 4, 5, 3};
2015-07-10 06:01:56 -07:00
2017-02-07 22:03:36 -08:00
// 6 cylinder
2019-01-05 05:06:18 -08:00
static const int order_1_THEN_5_THEN_3_THEN_6_THEN_2_THEN_4[] = { 1, 5, 3, 6, 2, 4 };
static const int order_1_THEN_4_THEN_2_THEN_5_THEN_3_THEN_6[] = { 1, 4, 2, 5, 3, 6 };
static const int order_1_THEN_2_THEN_3_THEN_4_THEN_5_THEN_6[] = { 1, 2, 3, 4, 5, 6 };
static const int order_1_6_3_2_5_4[] = {1, 6, 3, 2, 5, 4};
2015-07-10 06:01:56 -07:00
2017-02-07 22:03:36 -08:00
// 8 cylinder
2019-01-05 05:06:18 -08:00
static const int order_1_8_4_3_6_5_7_2[] = { 1, 8, 4, 3, 6, 5, 7, 2 };
static const int order_1_8_7_2_6_5_4_3[] = { 1, 8, 7, 2, 6, 5, 4, 3 };
static const int order_1_5_4_2_6_3_7_8[] = { 1, 5, 4, 2, 6, 3, 7, 8 };
2019-09-28 07:25:57 -07:00
static const int order_1_2_7_8_4_5_6_3[] = { 1, 2, 7, 8, 4, 5, 6, 3 };
2016-05-22 10:07:12 -07:00
2017-02-07 22:03:36 -08:00
// 10 cylinder
2019-01-05 05:06:18 -08:00
static const int order_1_10_9_4_3_6_5_8_7_2[] = {1, 10, 9, 4, 3, 6, 5, 8, 7, 2};
2015-07-10 06:01:56 -07:00
2017-02-07 22:03:36 -08:00
// 12 cyliner
2019-01-05 05:06:18 -08:00
static const int order_1_7_5_11_3_9_6_12_2_8_4_10[] = {1, 7, 5, 11, 3, 9, 6, 12, 2, 8, 4, 10};
static const int order_1_7_4_10_2_8_6_12_3_9_5_11[] = {1, 7, 4, 10, 2, 8, 6, 12, 3, 9, 5, 11};
static const int order_1_12_5_8_3_10_6_7_2_11_4_9[] = {1, 12, 5, 8, 3, 10, 6, 7, 2, 11, 4, 9};
2015-09-10 16:01:32 -07:00
2017-05-15 20:28:49 -07:00
static int getFiringOrderLength(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
2017-03-23 19:01:10 -07:00
switch (CONFIG(specs.firingOrder)) {
case FO_1:
return 1;
// 2 cylinder
case FO_1_2:
return 2;
// 3 cylinder
case FO_1_2_3:
return 3;
// 4 cylinder
case FO_1_3_4_2:
case FO_1_2_4_3:
case FO_1_3_2_4:
2019-08-15 16:36:16 -07:00
case FO_1_4_3_2:
2017-03-23 19:01:10 -07:00
return 4;
// 5 cylinder
case FO_1_2_4_5_3:
return 5;
// 6 cylinder
case FO_1_5_3_6_2_4:
case FO_1_4_2_5_3_6:
case FO_1_2_3_4_5_6:
case FO_1_6_3_2_5_4:
return 6;
// 8 cylinder
case FO_1_8_4_3_6_5_7_2:
case FO_1_8_7_2_6_5_4_3:
case FO_1_5_4_2_6_3_7_8:
2019-09-28 07:25:57 -07:00
case FO_1_2_7_8_4_5_6_3:
2017-03-23 19:01:10 -07:00
return 8;
// 10 cylinder
case FO_1_10_9_4_3_6_5_8_7_2:
return 10;
// 12 cylinder
case FO_1_7_5_11_3_9_6_12_2_8_4_10:
2017-06-08 17:35:42 -07:00
case FO_1_7_4_10_2_8_6_12_3_9_5_11:
case FO_1_12_5_8_3_10_6_7_2_11_4_9:
2017-03-23 19:01:10 -07:00
return 12;
default:
warning(CUSTOM_OBD_UNKNOWN_FIRING_ORDER, "getCylinderId not supported for %d", CONFIG(specs.firingOrder));
}
return 1;
}
2015-07-10 06:01:56 -07:00
/**
* @param index from zero to cylindersCount - 1
* @return cylinderId from one to cylindersCount
*/
2017-05-15 20:28:49 -07:00
int getCylinderId(int index DECLARE_ENGINE_PARAMETER_SUFFIX) {
2017-03-23 19:01:10 -07:00
2018-07-28 10:58:19 -07:00
const int firingOrderLength = getFiringOrderLength(PASS_ENGINE_PARAMETER_SIGNATURE);
2018-07-28 11:01:28 -07:00
if (firingOrderLength < 1 || firingOrderLength > INJECTION_PIN_COUNT) {
2018-07-28 10:58:19 -07:00
firmwareError(CUSTOM_ERR_6687, "fol %d", firingOrderLength);
return 1;
}
if (engineConfiguration->specs.cylindersCount != firingOrderLength) {
warning(CUSTOM_OBD_WRONG_FIRING_ORDER, "Wrong firing order %d/%d", engineConfiguration->specs.cylindersCount, firingOrderLength);
return 1;
}
if (index < 0 || index >= firingOrderLength) {
2018-07-28 12:36:47 -07:00
// todo: open question when does this happen? reproducible with functional tests?
warning(CUSTOM_ERR_6686, "index %d", index);
2017-03-23 19:01:10 -07:00
return 1;
}
2015-07-10 06:01:56 -07:00
2017-03-23 19:01:10 -07:00
switch (CONFIG(specs.firingOrder)) {
2016-07-20 16:04:27 -07:00
case FO_1:
2015-07-10 06:01:56 -07:00
return 1;
2015-09-10 16:01:32 -07:00
// 2 cylinder
2016-07-20 16:04:27 -07:00
case FO_1_2:
2015-07-10 06:01:56 -07:00
return order_1_2[index];
2015-09-10 16:01:32 -07:00
// 3 cylinder
case FO_1_2_3:
return order_1_2_3[index];
2015-07-10 06:01:56 -07:00
// 4 cylinder
2016-07-20 16:04:27 -07:00
case FO_1_3_4_2:
2015-07-10 06:01:56 -07:00
return order_1_THEN_3_THEN_4_THEN2[index];
2016-07-20 16:04:27 -07:00
case FO_1_2_4_3:
2015-07-10 06:01:56 -07:00
return order_1_THEN_2_THEN_4_THEN3[index];
2016-07-20 16:04:27 -07:00
case FO_1_3_2_4:
2015-07-10 06:01:56 -07:00
return order_1_THEN_3_THEN_2_THEN4[index];
2019-08-15 16:36:16 -07:00
case FO_1_4_3_2:
return order_1_THEN_4_THEN_3_THEN2[index];
2015-07-10 06:01:56 -07:00
// 5 cylinder
case FO_1_2_4_5_3:
return order_1_2_4_5_3[index];
// 6 cylinder
2016-07-20 16:04:27 -07:00
case FO_1_5_3_6_2_4:
2015-07-10 06:01:56 -07:00
return order_1_THEN_5_THEN_3_THEN_6_THEN_2_THEN_4[index];
2016-07-20 16:04:27 -07:00
case FO_1_4_2_5_3_6:
2015-07-10 06:01:56 -07:00
return order_1_THEN_4_THEN_2_THEN_5_THEN_3_THEN_6[index];
2016-07-20 16:04:27 -07:00
case FO_1_2_3_4_5_6:
2015-09-05 20:02:46 -07:00
return order_1_THEN_2_THEN_3_THEN_4_THEN_5_THEN_6[index];
2017-01-12 21:03:30 -08:00
case FO_1_6_3_2_5_4:
return order_1_6_3_2_5_4[index];
2015-09-05 20:02:46 -07:00
2015-07-10 06:01:56 -07:00
// 8 cylinder
case FO_1_8_4_3_6_5_7_2:
return order_1_8_4_3_6_5_7_2[index];
2016-05-22 10:07:12 -07:00
case FO_1_8_7_2_6_5_4_3:
return order_1_8_7_2_6_5_4_3[index];
2016-06-05 17:03:16 -07:00
case FO_1_5_4_2_6_3_7_8:
return order_1_5_4_2_6_3_7_8[index];
2019-09-28 07:25:57 -07:00
case FO_1_2_7_8_4_5_6_3:
return order_1_2_7_8_4_5_6_3[index];
2015-07-10 06:01:56 -07:00
2017-02-07 22:03:36 -08:00
// 10 cylinder
case FO_1_10_9_4_3_6_5_8_7_2:
return order_1_10_9_4_3_6_5_8_7_2[index];
// 12 cylinder
case FO_1_7_5_11_3_9_6_12_2_8_4_10:
return order_1_7_5_11_3_9_6_12_2_8_4_10[index];
2017-06-08 17:35:42 -07:00
case FO_1_7_4_10_2_8_6_12_3_9_5_11:
return order_1_7_4_10_2_8_6_12_3_9_5_11[index];
case FO_1_12_5_8_3_10_6_7_2_11_4_9:
return order_1_12_5_8_3_10_6_7_2_11_4_9[index];
2017-02-07 22:03:36 -08:00
2015-07-10 06:01:56 -07:00
default:
2017-03-23 19:01:10 -07:00
warning(CUSTOM_OBD_UNKNOWN_FIRING_ORDER, "getCylinderId not supported for %d", CONFIG(specs.firingOrder));
2015-07-10 06:01:56 -07:00
}
return 1;
}
2019-10-26 18:53:22 -07:00
/**
2019-11-17 06:18:43 -08:00
* @param cylinderIndex from 0 to cylinderCount, not cylinder number
2019-10-26 18:53:22 -07:00
*/
static int getIgnitionPinForIndex(int cylinderIndex DECLARE_ENGINE_PARAMETER_SUFFIX) {
switch (getCurrentIgnitionMode(PASS_ENGINE_PARAMETER_SIGNATURE)) {
2015-07-10 06:01:56 -07:00
case IM_ONE_COIL:
2016-01-16 14:02:38 -08:00
return 0;
2015-07-10 06:01:56 -07:00
case IM_WASTED_SPARK: {
2019-10-10 04:07:59 -07:00
if (CONFIG(specs.cylindersCount) == 1) {
// we do not want to divide by zero
return 0;
}
2019-10-26 18:53:22 -07:00
return cylinderIndex % (CONFIG(specs.cylindersCount) / 2);
2015-07-10 06:01:56 -07:00
}
case IM_INDIVIDUAL_COILS:
2019-10-26 18:53:22 -07:00
return cylinderIndex;
2019-10-27 13:19:59 -07:00
case IM_TWO_COILS:
return cylinderIndex % 2;
2015-07-10 06:01:56 -07:00
default:
warning(CUSTOM_OBD_IGNITION_MODE, "unsupported ignitionMode %d in getIgnitionPinForIndex()", engineConfiguration->ignitionMode);
2016-01-16 14:02:38 -08:00
return 0;
2015-07-10 06:01:56 -07:00
}
}
void prepareIgnitionPinIndices(ignition_mode_e ignitionMode DECLARE_ENGINE_PARAMETER_SUFFIX) {
2019-01-27 23:29:13 -08:00
if (ignitionMode != engine->ignitionModeForPinIndices) {
2019-04-12 19:07:03 -07:00
#if EFI_ENGINE_CONTROL
2019-10-26 18:53:22 -07:00
for (int cylinderIndex = 0; cylinderIndex < CONFIG(specs.cylindersCount); cylinderIndex++) {
ENGINE(ignitionPin[cylinderIndex]) = getIgnitionPinForIndex(cylinderIndex PASS_ENGINE_PARAMETER_SUFFIX);
}
#endif /* EFI_ENGINE_CONTROL */
2019-01-27 23:29:13 -08:00
engine->ignitionModeForPinIndices = ignitionMode;
}
}
/**
* @return IM_WASTED_SPARK if in SPINNING mode and IM_INDIVIDUAL_COILS setting
* @return CONFIG(ignitionMode) otherwise
*/
ignition_mode_e getCurrentIgnitionMode(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
ignition_mode_e ignitionMode = CONFIG(ignitionMode);
2019-04-12 19:07:03 -07:00
#if EFI_SHAFT_POSITION_INPUT
// In spin-up cranking mode we don't have full phase sync. info yet, so wasted spark mode is better
if (ignitionMode == IM_INDIVIDUAL_COILS && ENGINE(rpmCalculator.isSpinningUp(PASS_ENGINE_PARAMETER_SIGNATURE)))
ignitionMode = IM_WASTED_SPARK;
2019-01-31 14:55:23 -08:00
#endif /* EFI_SHAFT_POSITION_INPUT */
return ignitionMode;
}
2019-04-12 19:07:03 -07:00
#if EFI_ENGINE_CONTROL
2015-07-10 06:01:56 -07:00
2016-01-14 21:01:42 -08:00
/**
* This heavy method is only invoked in case of a configuration change or initialization.
*/
2017-05-15 20:28:49 -07:00
void prepareOutputSignals(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
ENGINE(engineCycle) = getEngineCycle(engine->getOperationMode(PASS_ENGINE_PARAMETER_SIGNATURE));
2015-07-10 06:01:56 -07:00
2016-11-11 16:02:29 -08:00
angle_t maxTimingCorrMap = -720.0f;
angle_t maxTimingMap = -720.0f;
for (int rpmIndex = 0;rpmIndex<IGN_RPM_COUNT;rpmIndex++) {
for (int l = 0;l<IGN_LOAD_COUNT;l++) {
maxTimingCorrMap = maxF(maxTimingCorrMap, config->ignitionIatCorrTable[l][rpmIndex]);
maxTimingMap = maxF(maxTimingMap, config->ignitionTable[l][rpmIndex]);
}
}
2016-10-31 19:02:12 -07:00
#if EFI_UNIT_TEST
2019-08-24 22:15:18 -07:00
if (verboseMode) {
printf("prepareOutputSignals %d onlyEdge=%s %s\r\n", engineConfiguration->trigger.type, boolToString(engineConfiguration->useOnlyRisingEdgeForTrigger),
getIgnition_mode_e(engineConfiguration->ignitionMode));
}
2018-03-03 06:11:49 -08:00
#endif /* EFI_UNIT_TEST */
2016-10-31 19:02:12 -07:00
2015-07-10 06:01:56 -07:00
for (int i = 0; i < CONFIG(specs.cylindersCount); i++) {
2019-10-26 18:53:22 -07:00
ENGINE(ignitionPositionWithinEngineCycle[i]) = ENGINE(engineCycle) * i / CONFIG(specs.cylindersCount);
2015-07-10 06:01:56 -07:00
}
prepareIgnitionPinIndices(CONFIG(ignitionMode) PASS_ENGINE_PARAMETER_SUFFIX);
TRIGGER_WAVEFORM(prepareShape());
2015-07-10 06:01:56 -07:00
}
void setFuelRpmBin(float from, float to DECLARE_CONFIG_PARAMETER_SUFFIX) {
setLinearCurve(config->fuelRpmBins, from, to);
2015-07-10 06:01:56 -07:00
}
void setFuelLoadBin(float from, float to DECLARE_CONFIG_PARAMETER_SUFFIX) {
setLinearCurve(config->fuelLoadBins, from, to);
2015-07-10 06:01:56 -07:00
}
void setTimingRpmBin(float from, float to DECLARE_CONFIG_PARAMETER_SUFFIX) {
2016-01-21 19:01:31 -08:00
setRpmBin(config->ignitionRpmBins, IGN_RPM_COUNT, from, to);
2015-07-10 06:01:56 -07:00
}
void setTimingLoadBin(float from, float to DECLARE_CONFIG_PARAMETER_SUFFIX) {
setLinearCurve(config->ignitionLoadBins, from, to);
2015-07-10 06:01:56 -07:00
}
/**
* this method sets algorithm and ignition table scale
*/
void setAlgorithm(engine_load_mode_e algo DECLARE_CONFIG_PARAMETER_SUFFIX) {
2016-08-28 13:02:34 -07:00
engineConfiguration->fuelAlgorithm = algo;
2015-07-10 06:01:56 -07:00
if (algo == LM_ALPHA_N) {
setTimingLoadBin(20, 120 PASS_CONFIG_PARAMETER_SUFFIX);
2015-07-10 06:01:56 -07:00
} else if (algo == LM_SPEED_DENSITY) {
setLinearCurve(config->ignitionLoadBins, 20, 120, 3);
buildTimingMap(35 PASS_CONFIG_PARAMETER_SUFFIX);
2015-07-10 06:01:56 -07:00
}
}
2016-07-24 20:02:52 -07:00
void setFlatInjectorLag(float value DECLARE_CONFIG_PARAMETER_SUFFIX) {
setArrayValues(engineConfiguration->injector.battLagCorr, value);
2016-07-24 20:02:52 -07:00
}
#endif /* EFI_ENGINE_CONTROL */