rusefi/firmware/controllers/core/fsio_impl.cpp

387 lines
12 KiB
C++
Raw Normal View History

2015-07-10 06:01:56 -07:00
/**
* @file fsio_impl.cpp
* @brief FSIO as it's used for GPIO
*
2017-12-17 09:02:35 -08:00
* set debug_mode 23
* https://rusefi.com/wiki/index.php?title=Manual:Flexible_Logic
*
2019-09-11 17:46:50 -07:00
* 'fsioinfo' command in console shows current state of FSIO - formulas and current value
*
2015-07-10 06:01:56 -07:00
* @date Oct 5, 2014
2020-01-07 21:02:40 -08:00
* @author Andrey Belomutskiy, (c) 2012-2020
2015-07-10 06:01:56 -07:00
*/
#include "pch.h"
#include "fsio_impl.h"
2015-07-10 06:01:56 -07:00
#if EFI_PROD_CODE
// todo: that's about bench test mode, wrong header for sure!
#include "bench_test.h"
#endif // EFI_PROD_CODE
2019-04-12 19:07:03 -07:00
#if EFI_FSIO
2015-07-10 06:01:56 -07:00
#include "os_access.h"
2015-07-10 06:01:56 -07:00
2017-07-14 18:35:38 -07:00
/**
* in case of zero frequency pin is operating as simple on/off. '1' for ON and '0' for OFF
*
*/
2015-07-10 06:01:56 -07:00
#define NO_PWM 0
2021-11-14 07:39:47 -08:00
static fsio8_Map3D_f32t scriptTable1;
static fsio8_Map3D_u8t scriptTable2;
static fsio8_Map3D_u8t scriptTable3;
static fsio8_Map3D_u8t scriptTable4;
2016-04-04 07:01:43 -07:00
2015-07-10 06:01:56 -07:00
/**
* Here we define all rusEfi-specific methods
*/
static LENameOrdinalPair leRpm(LE_METHOD_RPM, "rpm");
static LENameOrdinalPair leTps(LE_METHOD_TPS, "tps");
static LENameOrdinalPair lePps(LE_METHOD_PPS, "pps");
2015-07-10 06:01:56 -07:00
static LENameOrdinalPair leMaf(LE_METHOD_MAF, "maf");
2016-07-07 20:01:43 -07:00
static LENameOrdinalPair leMap(LE_METHOD_MAP, "map");
2015-07-10 06:01:56 -07:00
static LENameOrdinalPair leVBatt(LE_METHOD_VBATT, "vbatt");
static LENameOrdinalPair leFan(LE_METHOD_FAN, "fan");
static LENameOrdinalPair leCoolant(LE_METHOD_COOLANT, "coolant");
2020-10-04 18:33:54 -07:00
static LENameOrdinalPair leIntakeTemp(LE_METHOD_INTAKE_AIR, "iat");
2017-06-04 12:00:19 -07:00
static LENameOrdinalPair leIsCoolantBroken(LE_METHOD_IS_COOLANT_BROKEN, "is_clt_broken");
static LENameOrdinalPair leOilPressure(LE_METHOD_OIL_PRESSURE, "oilp");
2019-09-11 17:46:50 -07:00
// @returns boolean state of A/C toggle switch
2015-07-10 06:01:56 -07:00
static LENameOrdinalPair leAcToggle(LE_METHOD_AC_TOGGLE, "ac_on_switch");
2019-09-11 17:46:50 -07:00
// @returns float number of seconds since last A/C toggle
2019-09-08 18:09:45 -07:00
static LENameOrdinalPair leTimeSinceAcToggle(LE_METHOD_TIME_SINCE_AC_TOGGLE, "time_since_ac_on_switch");
2015-07-10 06:01:56 -07:00
static LENameOrdinalPair leTimeSinceBoot(LE_METHOD_TIME_SINCE_BOOT, "time_since_boot");
2020-04-18 18:01:23 -07:00
static LENameOrdinalPair leFsioSetting(LE_METHOD_FSIO_SETTING, FSIO_METHOD_FSIO_SETTING);
static LENameOrdinalPair leFsioAnalogInput(LE_METHOD_FSIO_ANALOG_INPUT, FSIO_METHOD_FSIO_ANALOG_INPUT);
static LENameOrdinalPair leFsioDigitalInput(LE_METHOD_FSIO_DIGITAL_INPUT, FSIO_METHOD_FSIO_DIGITAL_INPUT);
2016-12-17 09:03:02 -08:00
static LENameOrdinalPair leIntakeVVT(LE_METHOD_INTAKE_VVT, "ivvt");
static LENameOrdinalPair leExhaustVVT(LE_METHOD_EXHAUST_VVT, "evvt");
2017-06-14 02:39:00 -07:00
static LENameOrdinalPair leCrankingRpm(LE_METHOD_CRANKING_RPM, "cranking_rpm");
static LENameOrdinalPair leStartupFuelPumpDuration(LE_METHOD_STARTUP_FUEL_PUMP_DURATION, "startup_fuel_pump_duration");
static LENameOrdinalPair leInShutdown(LE_METHOD_IN_SHUTDOWN, "in_shutdown");
2021-01-10 20:46:50 -08:00
static LENameOrdinalPair leInMrBench(LE_METHOD_IN_MR_BENCH, "in_mr_bench");
static LENameOrdinalPair leTimeSinceTrigger(LE_METHOD_TIME_SINCE_TRIGGER_EVENT, "time_since_trigger");
static LENameOrdinalPair leFuelRate(LE_METHOD_FUEL_FLOW_RATE, "fuel_flow");
2015-07-10 06:01:56 -07:00
2019-09-11 16:30:36 -07:00
#include "fsio_names.def"
2021-08-18 12:25:46 -07:00
#define SYS_ELEMENT_POOL_SIZE 24
#define UD_ELEMENT_POOL_SIZE 64
2015-07-10 06:01:56 -07:00
2017-01-06 08:02:49 -08:00
static LEElement sysElements[SYS_ELEMENT_POOL_SIZE] CCM_OPTIONAL;
CCM_OPTIONAL LEElementPool sysPool(sysElements, SYS_ELEMENT_POOL_SIZE);
2015-07-10 06:01:56 -07:00
static LEElement * fuelPumpLogic;
2020-03-18 13:32:03 -07:00
static LEElement * starterRelayDisableLogic;
2015-07-10 06:01:56 -07:00
2019-04-12 19:07:03 -07:00
#if EFI_MAIN_RELAY_CONTROL
2017-06-12 15:14:39 -07:00
static LEElement * mainRelayLogic;
#endif /* EFI_MAIN_RELAY_CONTROL */
2018-12-24 05:37:25 -08:00
#if EFI_PROD_CODE || EFI_SIMULATOR
2015-07-10 06:01:56 -07:00
FsioResult getEngineValue(le_action_e action) {
efiAssert(CUSTOM_ERR_ASSERT, engine!=NULL, "getLEValue", unexpected);
2015-07-10 06:01:56 -07:00
switch (action) {
case LE_METHOD_FAN:
return enginePins.fanRelay.getLogicValue();
2019-09-08 18:05:03 -07:00
case LE_METHOD_TIME_SINCE_AC_TOGGLE:
return (getTimeNowUs() - engine->acSwitchLastChangeTime) / US_PER_SECOND_F;
2015-07-10 06:01:56 -07:00
case LE_METHOD_AC_TOGGLE:
return getAcToggle();
2015-07-10 06:01:56 -07:00
case LE_METHOD_COOLANT:
2021-10-05 16:59:07 -07:00
return Sensor::getOrZero(SensorType::Clt);
2017-06-04 12:00:19 -07:00
case LE_METHOD_IS_COOLANT_BROKEN:
return !Sensor::get(SensorType::Clt).Valid;
2015-07-10 06:01:56 -07:00
case LE_METHOD_INTAKE_AIR:
2021-10-05 16:59:07 -07:00
return Sensor::getOrZero(SensorType::Iat);
2015-07-10 06:01:56 -07:00
case LE_METHOD_RPM:
2021-10-05 16:59:07 -07:00
return Sensor::getOrZero(SensorType::Rpm);
2016-07-07 20:01:43 -07:00
case LE_METHOD_MAF:
2021-10-05 16:59:07 -07:00
return Sensor::getOrZero(SensorType::Maf);
2016-07-07 20:01:43 -07:00
case LE_METHOD_MAP:
2021-10-05 16:59:07 -07:00
return Sensor::getOrZero(SensorType::Map);
#if EFI_SHAFT_POSITION_INPUT
2016-12-17 09:03:02 -08:00
case LE_METHOD_INTAKE_VVT:
return engine->triggerCentral.getVVTPosition(0, 0);
2016-12-17 09:03:02 -08:00
case LE_METHOD_EXHAUST_VVT:
return engine->triggerCentral.getVVTPosition(0, 1);
#endif
case LE_METHOD_TIME_SINCE_TRIGGER_EVENT:
return engine->triggerCentral.getTimeSinceTriggerEvent(getTimeNowNt());
2015-07-10 06:01:56 -07:00
case LE_METHOD_TIME_SINCE_BOOT:
#if EFI_MAIN_RELAY_CONTROL
// in main relay control mode, we return the number of seconds since the ignition is turned on
// (or negative if the ignition key is switched off)
return engine->getTimeIgnitionSeconds();
#else
2015-07-10 06:01:56 -07:00
return getTimeNowSeconds();
#endif /* EFI_MAIN_RELAY_CONTROL */
case LE_METHOD_STARTUP_FUEL_PUMP_DURATION:
return engineConfiguration->startUpFuelPumpDuration;
2017-06-14 02:39:00 -07:00
case LE_METHOD_CRANKING_RPM:
return engineConfiguration->cranking.rpm;
case LE_METHOD_IN_SHUTDOWN:
return engine->isInShutdownMode();
2021-01-10 20:46:50 -08:00
case LE_METHOD_IN_MR_BENCH:
return engine->isInMainRelayBench();
2015-07-10 06:01:56 -07:00
case LE_METHOD_VBATT:
2021-10-05 16:59:07 -07:00
return Sensor::getOrZero(SensorType::BatteryVoltage);
case LE_METHOD_TPS:
2021-10-05 16:59:07 -07:00
return Sensor::getOrZero(SensorType::DriverThrottleIntent);
case LE_METHOD_FUEL_FLOW_RATE:
return engine->engineState.fuelConsumption.getConsumptionGramPerSecond();
case LE_METHOD_OIL_PRESSURE:
2021-10-05 16:59:07 -07:00
return Sensor::getOrZero(SensorType::OilPressure);
// cfg_xxx references are code generated
2019-09-10 20:49:21 -07:00
#include "fsio_getters.def"
2015-07-10 06:01:56 -07:00
default:
2020-04-10 14:45:04 -07:00
warning(CUSTOM_FSIO_UNEXPECTED, "FSIO ERROR no data for action=%d", action);
return unexpected;
2015-07-10 06:01:56 -07:00
}
}
2016-01-12 09:01:43 -08:00
#endif
void onConfigurationChangeFsioCallback(engine_configuration_s *previousConfiguration) {
(void)previousConfiguration;
}
static LECalculator calc CCM_OPTIONAL;
2015-07-10 06:01:56 -07:00
static const char * action2String(le_action_e action) {
2016-08-09 21:04:24 -07:00
static char buffer[_MAX_FILLER];
2015-07-10 06:01:56 -07:00
switch(action) {
case LE_METHOD_RPM:
return "RPM";
2017-06-14 02:39:00 -07:00
case LE_METHOD_CRANKING_RPM:
return "cranking_rpm";
2015-07-10 06:01:56 -07:00
case LE_METHOD_COOLANT:
return "CLT";
case LE_METHOD_FAN:
return "fan";
case LE_METHOD_STARTUP_FUEL_PUMP_DURATION:
2021-01-10 20:46:50 -08:00
return leStartupFuelPumpDuration.name;
case LE_METHOD_IN_SHUTDOWN:
2021-01-10 20:46:50 -08:00
return leInShutdown.name;
case LE_METHOD_IN_MR_BENCH:
return leInMrBench.name;
2019-09-11 16:30:36 -07:00
#include "fsio_strings.def"
2015-07-10 06:01:56 -07:00
default: {
// this is here to make compiler happy
}
}
itoa10(buffer, (int)action);
return buffer;
}
static void setPinState(const char * msg, OutputPin *pin, LEElement *element) {
2018-12-24 05:37:25 -08:00
#if EFI_PROD_CODE
if (isRunningBenchTest()) {
return; // let's not mess with bench testing
}
2018-12-24 05:37:25 -08:00
#endif /* EFI_PROD_CODE */
if (!element) {
2017-07-24 16:38:22 -07:00
warning(CUSTOM_FSIO_INVALID_EXPRESSION, "invalid expression for %s", msg);
2015-07-10 06:01:56 -07:00
} else {
int value = (int)calc.evaluate(msg, pin->getLogicValue(), element);
2015-07-10 06:01:56 -07:00
if (pin->isInitialized() && value != pin->getLogicValue()) {
for (int i = 0;i < calc.currentCalculationLogPosition;i++) {
efiPrintf("calc %d: action %s value %.2f", i, action2String(calc.calcLogAction[i]), calc.calcLogValue[i]);
2015-07-10 06:01:56 -07:00
}
efiPrintf("setPin %s %s", msg, value ? "on" : "off");
2015-07-10 06:01:56 -07:00
pin->setValue(value);
}
}
}
2017-12-16 21:03:26 -08:00
/**
* this method should be invoked periodically to calculate FSIO and toggle corresponding FSIO outputs
*/
void runFsio() {
2019-04-12 19:07:03 -07:00
#if EFI_FUEL_PUMP
if (isBrainPinValid(CONFIG(fuelPumpPin))) {
setPinState("pump", &enginePins.fuelPumpRelay, fuelPumpLogic);
2016-01-12 07:04:06 -08:00
}
#endif /* EFI_FUEL_PUMP */
2019-04-12 19:07:03 -07:00
#if EFI_MAIN_RELAY_CONTROL
if (isBrainPinValid(CONFIG(mainRelayPin)))
// the MAIN_RELAY_LOGIC calls engine->isInShutdownMode()
setPinState("main_relay", &enginePins.mainRelay, mainRelayLogic);
2017-06-12 15:14:39 -07:00
#else /* EFI_MAIN_RELAY_CONTROL */
2016-01-12 07:04:06 -08:00
/**
* main relay is always on if ECU is on, that's a good enough initial implementation
*/
if (isBrainPinValid(CONFIG(mainRelayPin)))
enginePins.mainRelay.setValue(!engine->isInMainRelayBench());
2017-06-12 15:14:39 -07:00
#endif /* EFI_MAIN_RELAY_CONTROL */
2016-01-12 07:04:06 -08:00
if (isBrainPinValid(CONFIG(starterRelayDisablePin)))
setPinState("starter_relay", &enginePins.starterRelayDisable, starterRelayDisableLogic);
2017-07-06 18:26:35 -07:00
/**
* o2 heater is off during cranking
2017-07-08 12:46:34 -07:00
* todo: convert to FSIO?
* open question if heater should be ON during cranking
2017-07-06 18:26:35 -07:00
*/
enginePins.o2heater.setValue(engine->rpmCalculator.isRunning());
2016-01-12 07:04:06 -08:00
}
2015-07-10 06:01:56 -07:00
static void showFsio(const char *msg, LEElement *element) {
2016-01-12 09:01:43 -08:00
#if EFI_PROD_CODE || EFI_SIMULATOR
2015-07-10 06:01:56 -07:00
if (msg != NULL)
efiPrintf("%s:", msg);
while (element->action != LE_METHOD_RETURN) {
efiPrintf("action %d: fValue=%.2f", element->action, element->fValue);
element++;
2015-07-10 06:01:56 -07:00
}
efiPrintf("<end>");
2016-01-12 09:01:43 -08:00
#endif
2015-07-10 06:01:56 -07:00
}
2021-11-14 06:49:02 -08:00
// todo: move somewhere else
static void showFsioInfo() {
2016-01-12 09:01:43 -08:00
#if EFI_PROD_CODE || EFI_SIMULATOR
2015-07-10 06:01:56 -07:00
showFsio("fuel", fuelPumpLogic);
2021-11-14 06:49:02 -08:00
for (int i = 0; i < SCRIPT_SETTING_COUNT; i++) {
2021-11-13 20:10:38 -08:00
float v = CONFIG(scriptSetting)[i];
2015-07-10 06:01:56 -07:00
if (!cisnan(v)) {
efiPrintf("user property #%d: %.2f", i + 1, v);
2015-07-10 06:01:56 -07:00
}
}
2016-01-12 09:01:43 -08:00
#endif
2015-07-10 06:01:56 -07:00
}
2021-11-14 07:39:47 -08:00
ValueProvider3D *getscriptTable(int index) {
2019-07-12 05:31:38 -07:00
switch (index) {
default:
2021-11-14 07:39:47 -08:00
return &scriptTable1;
2019-07-12 05:31:38 -07:00
case 1:
2021-11-14 07:39:47 -08:00
return &scriptTable2;
2019-07-12 05:31:38 -07:00
case 2:
2021-11-14 07:39:47 -08:00
return &scriptTable3;
2019-07-12 05:31:38 -07:00
case 3:
2021-11-14 07:39:47 -08:00
return &scriptTable4;
2019-07-12 05:31:38 -07:00
}
}
2021-11-14 09:51:41 -08:00
// todo: template this copy-pasta
2021-11-05 12:25:29 -07:00
/**
* @return zero-based index of curve with given name
*/
int getCurveIndexByName(const char *name) {
2021-11-05 12:25:29 -07:00
for (int i = 0;i<SCRIPT_CURVE_COUNT;i++) {
if (strEqualCaseInsensitive(name, engineConfiguration->scriptCurveName[i])) {
return i;
}
}
return EFI_ERROR_CODE;
}
int getTableIndexByName(const char *name) {
2021-11-13 07:02:01 -08:00
for (int i = 0;i<SCRIPT_TABLE_COUNT;i++) {
if (strEqualCaseInsensitive(name, engineConfiguration->scriptTableName[i])) {
return i;
}
}
return EFI_ERROR_CODE;
}
int getSettingIndexByName(const char *name) {
2021-11-14 09:51:41 -08:00
for (int i = 0;i<SCRIPT_SETTING_COUNT;i++) {
if (strEqualCaseInsensitive(name, engineConfiguration->scriptSettingName[i])) {
return i;
}
}
return EFI_ERROR_CODE;
}
float getCurveValue(int index, float key) {
2021-11-04 20:22:37 -07:00
// not great code at all :(
switch (index) {
default:
return interpolate2d(key, engineConfiguration->scriptCurve1Bins, engineConfiguration->scriptCurve1);
2021-11-04 20:59:04 -07:00
case 1:
return interpolate2d(key, engineConfiguration->scriptCurve2Bins, engineConfiguration->scriptCurve2);
case 2:
return interpolate2d(key, engineConfiguration->scriptCurve3Bins, engineConfiguration->scriptCurve3);
case 3:
return interpolate2d(key, engineConfiguration->scriptCurve4Bins, engineConfiguration->scriptCurve4);
case 4:
2021-11-04 20:22:37 -07:00
return interpolate2d(key, engineConfiguration->scriptCurve5Bins, engineConfiguration->scriptCurve5);
2021-11-04 20:59:04 -07:00
case 5:
return interpolate2d(key, engineConfiguration->scriptCurve6Bins, engineConfiguration->scriptCurve6);
2021-11-04 20:22:37 -07:00
}
}
void initFsioImpl() {
#if EFI_UNIT_TEST
2019-05-12 17:31:28 -07:00
// only unit test needs this
sysPool.reset();
2016-01-12 09:01:43 -08:00
#endif
2015-07-10 06:01:56 -07:00
2019-04-12 19:07:03 -07:00
#if EFI_FUEL_PUMP
2015-07-10 06:01:56 -07:00
fuelPumpLogic = sysPool.parseExpression(FUEL_PUMP_LOGIC);
2016-01-12 07:04:06 -08:00
#endif /* EFI_FUEL_PUMP */
2015-07-10 06:01:56 -07:00
2019-04-12 19:07:03 -07:00
#if EFI_MAIN_RELAY_CONTROL
if (isBrainPinValid(CONFIG(mainRelayPin)))
2017-06-12 15:14:39 -07:00
mainRelayLogic = sysPool.parseExpression(MAIN_RELAY_LOGIC);
#endif /* EFI_MAIN_RELAY_CONTROL */
if (isBrainPinValid(CONFIG(starterRelayDisablePin)))
2020-03-18 13:32:03 -07:00
starterRelayDisableLogic = sysPool.parseExpression(STARTER_RELAY_LOGIC);
2015-07-10 06:01:56 -07:00
2021-11-14 07:39:47 -08:00
scriptTable1.init(config->scriptTable1, config->scriptTable1LoadBins,
config->scriptTable1RpmBins);
scriptTable2.init(config->scriptTable2, config->scriptTable2LoadBins,
config->scriptTable2RpmBins);
scriptTable3.init(config->scriptTable3, config->scriptTable3LoadBins,
config->scriptTable3RpmBins);
scriptTable4.init(config->scriptTable4, config->scriptTable4LoadBins,
config->scriptTable4RpmBins);
2016-04-04 07:01:43 -07:00
2015-07-10 06:01:56 -07:00
}
#else /* !EFI_FSIO */
// "Limp-mode" implementation for some RAM-limited configs without FSIO
void runHardcodedFsio() {
#if EFI_PROD_CODE
if (isRunningBenchTest()) {
return; // let's not mess with bench testing
}
#endif /* EFI_PROD_CODE */
// see MAIN_RELAY_LOGIC
if (isBrainPinValid(CONFIG(mainRelayPin))) {
2021-10-05 16:59:07 -07:00
enginePins.mainRelay.setValue((getTimeNowSeconds() < 2) || (Sensor::getOrZero(SensorType::BatteryVoltage) > LOW_VBATT) || engine->isInShutdownMode());
}
// see STARTER_RELAY_LOGIC
if (isBrainPinValid(CONFIG(starterRelayDisablePin))) {
2020-03-18 16:41:12 -07:00
enginePins.starterRelayDisable.setValue(engine->rpmCalculator.getRpm() < engineConfiguration->cranking.rpm);
}
// see FUEL_PUMP_LOGIC
if (isBrainPinValid(CONFIG(fuelPumpPin))) {
int triggerActivityOrEcuStartSecond = maxI(0, engine->triggerActivityMs / 1000);
2021-11-11 16:58:43 -08:00
enginePins.fuelPumpRelay.setValue((getTimeNowSeconds() < triggerActivityOrEcuStartSecond + engineConfiguration->startUpFuelPumpDuration) || (engine->rpmCalculator.getRpm() > 0));
}
enginePins.o2heater.setValue(engine->rpmCalculator.isRunning());
}
2015-07-10 06:01:56 -07:00
#endif /* EFI_FSIO */