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
|
|
|
*/
|
|
|
|
|
2018-09-16 19:26:57 -07:00
|
|
|
#include "global.h"
|
2019-11-25 17:08:01 -08:00
|
|
|
#include "fsio_impl.h"
|
|
|
|
#include "allsensors.h"
|
2020-04-18 15:45:30 -07:00
|
|
|
#include "sensor.h"
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2020-03-29 16:06:03 -07:00
|
|
|
EXTERN_ENGINE;
|
|
|
|
|
2020-09-10 18:51:58 -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
|
|
|
|
2019-07-04 00:25:35 -07:00
|
|
|
#include "os_access.h"
|
2019-01-05 20:48:37 -08:00
|
|
|
#include "settings.h"
|
2015-07-10 06:01:56 -07:00
|
|
|
#include "rpm_calculator.h"
|
2019-03-29 06:11:13 -07:00
|
|
|
#include "efi_gpio.h"
|
2018-12-24 05:37:25 -08:00
|
|
|
#include "pwm_generator_logic.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
|
|
|
|
|
2016-06-29 22:01:38 -07:00
|
|
|
fsio8_Map3D_f32t fsioTable1("fsio#1");
|
2016-07-01 20:01:22 -07:00
|
|
|
fsio8_Map3D_u8t fsioTable2("fsio#2");
|
|
|
|
fsio8_Map3D_u8t fsioTable3("fsio#3");
|
|
|
|
fsio8_Map3D_u8t fsioTable4("fsio#4");
|
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");
|
2020-04-10 16:02:37 -07:00
|
|
|
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");
|
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 leFsioTable(LE_METHOD_FSIO_TABLE, FSIO_METHOD_FSIO_TABLE);
|
|
|
|
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);
|
2015-07-10 06:01:56 -07:00
|
|
|
static LENameOrdinalPair leKnock(LE_METHOD_KNOCK, "knock");
|
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");
|
2017-06-24 22:35:46 -07:00
|
|
|
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");
|
2020-12-03 08:13:45 -08:00
|
|
|
static LENameOrdinalPair leTimeSinceTrigger(LE_METHOD_TIME_SINCE_TRIGGER_EVENT, "time_since_trigger");
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2019-09-11 16:30:36 -07:00
|
|
|
#include "fsio_names.def"
|
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
#define LE_EVAL_POOL_SIZE 32
|
|
|
|
|
|
|
|
static LECalculator evalCalc;
|
|
|
|
static LEElement evalPoolElements[LE_EVAL_POOL_SIZE];
|
|
|
|
static LEElementPool evalPool(evalPoolElements, LE_EVAL_POOL_SIZE);
|
|
|
|
|
|
|
|
#define SYS_ELEMENT_POOL_SIZE 128
|
|
|
|
#define UD_ELEMENT_POOL_SIZE 128
|
|
|
|
|
2017-01-06 08:02:49 -08:00
|
|
|
static LEElement sysElements[SYS_ELEMENT_POOL_SIZE] CCM_OPTIONAL;
|
2015-07-10 06:01:56 -07:00
|
|
|
LEElementPool sysPool(sysElements, SYS_ELEMENT_POOL_SIZE);
|
|
|
|
|
2017-01-06 08:02:49 -08:00
|
|
|
static LEElement userElements[UD_ELEMENT_POOL_SIZE] CCM_OPTIONAL;
|
2015-07-10 06:01:56 -07:00
|
|
|
LEElementPool userPool(userElements, UD_ELEMENT_POOL_SIZE);
|
2019-01-10 20:53:59 -08:00
|
|
|
|
|
|
|
class FsioPointers {
|
|
|
|
public:
|
|
|
|
FsioPointers();
|
|
|
|
LEElement * fsioLogics[FSIO_COMMAND_COUNT];
|
|
|
|
};
|
|
|
|
|
2019-01-19 18:10:58 -08:00
|
|
|
FsioPointers::FsioPointers() : fsioLogics() {
|
2019-01-10 20:53:59 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
static FsioPointers state;
|
2015-07-10 06:01:56 -07:00
|
|
|
|
|
|
|
static LEElement * acRelayLogic;
|
|
|
|
static LEElement * fuelPumpLogic;
|
|
|
|
static LEElement * radiatorFanLogic;
|
|
|
|
static LEElement * alternatorLogic;
|
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 */
|
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
static Logging *logger;
|
2018-12-24 05:37:25 -08:00
|
|
|
#if EFI_PROD_CODE || EFI_SIMULATOR
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2020-12-25 05:40:54 -08:00
|
|
|
FsioResult getEngineValue(le_action_e action DECLARE_ENGINE_PARAMETER_SUFFIX) {
|
2020-10-06 19:33:00 -07:00
|
|
|
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:
|
2017-05-15 20:28:49 -07:00
|
|
|
return getAcToggle(PASS_ENGINE_PARAMETER_SIGNATURE);
|
2015-07-10 06:01:56 -07:00
|
|
|
case LE_METHOD_COOLANT:
|
2020-04-15 13:20:45 -07:00
|
|
|
return Sensor::get(SensorType::Clt).value_or(0);
|
2017-06-04 12:00:19 -07:00
|
|
|
case LE_METHOD_IS_COOLANT_BROKEN:
|
2020-04-15 13:20:45 -07:00
|
|
|
return !Sensor::get(SensorType::Clt).Valid;
|
2015-07-10 06:01:56 -07:00
|
|
|
case LE_METHOD_INTAKE_AIR:
|
2020-04-18 12:53:04 -07:00
|
|
|
return Sensor::get(SensorType::Iat).value_or(0);
|
2015-07-10 06:01:56 -07:00
|
|
|
case LE_METHOD_RPM:
|
2020-09-07 08:09:48 -07:00
|
|
|
return Sensor::get(SensorType::Rpm).value_or(0);
|
2016-07-07 20:01:43 -07:00
|
|
|
case LE_METHOD_MAF:
|
2019-07-09 23:14:47 -07:00
|
|
|
return getRealMaf(PASS_ENGINE_PARAMETER_SIGNATURE);
|
2016-07-07 20:01:43 -07:00
|
|
|
case LE_METHOD_MAP:
|
2021-01-02 16:13:10 -08:00
|
|
|
return Sensor::get(SensorType::Map).value_or(0);
|
2020-02-26 15:16:35 -08:00
|
|
|
#if EFI_SHAFT_POSITION_INPUT
|
2016-12-17 09:03:02 -08:00
|
|
|
case LE_METHOD_INTAKE_VVT:
|
2021-03-25 04:39:23 -07:00
|
|
|
return engine->triggerCentral.getVVTPosition(0, 0);
|
2016-12-17 09:03:02 -08:00
|
|
|
case LE_METHOD_EXHAUST_VVT:
|
2021-03-25 04:39:23 -07:00
|
|
|
return engine->triggerCentral.getVVTPosition(0, 1);
|
2020-02-26 15:16:35 -08:00
|
|
|
#endif
|
2020-12-02 20:09:39 -08:00
|
|
|
case LE_METHOD_TIME_SINCE_TRIGGER_EVENT:
|
2020-12-06 19:28:29 -08:00
|
|
|
return engine->triggerCentral.getTimeSinceTriggerEvent(getTimeNowNt());
|
2015-07-10 06:01:56 -07:00
|
|
|
case LE_METHOD_TIME_SINCE_BOOT:
|
2020-09-10 18:08:39 -07:00
|
|
|
#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();
|
2020-09-10 18:08:39 -07:00
|
|
|
#endif /* EFI_MAIN_RELAY_CONTROL */
|
2019-09-08 19:17:51 -07:00
|
|
|
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;
|
2017-06-24 22:35:46 -07:00
|
|
|
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-03-07 17:18:32 -08:00
|
|
|
return Sensor::get(SensorType::BatteryVoltage).value_or(0);
|
2020-04-10 14:49:13 -07:00
|
|
|
case LE_METHOD_TPS:
|
|
|
|
return Sensor::get(SensorType::DriverThrottleIntent).value_or(0);
|
2020-07-01 17:45:37 -07:00
|
|
|
// 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);
|
2020-10-06 19:33:00 -07:00
|
|
|
return unexpected;
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-04-12 19:07:03 -07:00
|
|
|
#if EFI_PROD_CODE
|
2015-07-10 06:01:56 -07:00
|
|
|
|
|
|
|
#include "pin_repository.h"
|
2020-04-26 14:40:12 -07:00
|
|
|
#include "pwm_generator_logic.h"
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2018-08-01 20:27:22 -07:00
|
|
|
static void setFsioAnalogInputPin(const char *indexStr, const char *pinName) {
|
|
|
|
// todo: reduce code duplication between all "set pin methods"
|
|
|
|
int index = atoi(indexStr) - 1;
|
2019-04-16 07:27:34 -07:00
|
|
|
if (index < 0 || index >= FSIO_ANALOG_INPUT_COUNT) {
|
2018-08-01 20:27:22 -07:00
|
|
|
scheduleMsg(logger, "invalid FSIO index: %d", index);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
brain_pin_e pin = parseBrainPin(pinName);
|
|
|
|
// todo: extract method - code duplication with other 'set_xxx_pin' methods?
|
|
|
|
if (pin == GPIO_INVALID) {
|
|
|
|
scheduleMsg(logger, "invalid pin name [%s]", pinName);
|
|
|
|
return;
|
|
|
|
}
|
2018-08-01 20:33:13 -07:00
|
|
|
engineConfiguration->fsioAdc[index] = (adc_channel_e) pin;
|
2018-08-01 20:27:22 -07:00
|
|
|
scheduleMsg(logger, "FSIO analog input pin #%d [%s]", (index + 1), hwPortname(pin));
|
|
|
|
}
|
|
|
|
|
2018-08-01 20:02:19 -07:00
|
|
|
static void setFsioDigitalInputPin(const char *indexStr, const char *pinName) {
|
2018-08-01 20:27:22 -07:00
|
|
|
// todo: reduce code duplication between all "set pin methods"
|
2015-07-10 06:01:56 -07:00
|
|
|
int index = atoi(indexStr) - 1;
|
2017-06-25 23:14:31 -07:00
|
|
|
if (index < 0 || index >= FSIO_COMMAND_COUNT) {
|
2016-01-12 19:01:53 -08:00
|
|
|
scheduleMsg(logger, "invalid FSIO index: %d", index);
|
2015-07-10 06:01:56 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
brain_pin_e pin = parseBrainPin(pinName);
|
|
|
|
// todo: extract method - code duplication with other 'set_xxx_pin' methods?
|
|
|
|
if (pin == GPIO_INVALID) {
|
|
|
|
scheduleMsg(logger, "invalid pin name [%s]", pinName);
|
|
|
|
return;
|
|
|
|
}
|
2019-11-30 07:24:37 -08:00
|
|
|
CONFIG(fsioDigitalInputs)[index] = pin;
|
2018-08-01 20:27:22 -07:00
|
|
|
scheduleMsg(logger, "FSIO digital input pin #%d [%s]", (index + 1), hwPortname(pin));
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
2016-12-27 21:02:03 -08:00
|
|
|
static void setFsioPidOutputPin(const char *indexStr, const char *pinName) {
|
|
|
|
int index = atoi(indexStr) - 1;
|
2021-02-10 16:51:14 -08:00
|
|
|
if (index < 0 || index >= CAM_INPUTS_COUNT) {
|
|
|
|
scheduleMsg(logger, "invalid VVT index: %d", index);
|
2016-12-27 21:02:03 -08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
brain_pin_e pin = parseBrainPin(pinName);
|
|
|
|
// todo: extract method - code duplication with other 'set_xxx_pin' methods?
|
|
|
|
if (pin == GPIO_INVALID) {
|
|
|
|
scheduleMsg(logger, "invalid pin name [%s]", pinName);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
engineConfiguration->auxPidPins[index] = pin;
|
2021-02-10 16:51:14 -08:00
|
|
|
scheduleMsg(logger, "VVT pid pin #%d [%s]", (index + 1), hwPortname(pin));
|
2016-12-27 21:02:03 -08:00
|
|
|
}
|
|
|
|
|
2017-12-17 09:02:35 -08:00
|
|
|
static void showFsioInfo(void);
|
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
static void setFsioOutputPin(const char *indexStr, const char *pinName) {
|
|
|
|
int index = atoi(indexStr) - 1;
|
2017-06-25 23:14:31 -07:00
|
|
|
if (index < 0 || index >= FSIO_COMMAND_COUNT) {
|
2016-01-12 19:01:53 -08:00
|
|
|
scheduleMsg(logger, "invalid FSIO index: %d", index);
|
2015-07-10 06:01:56 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
brain_pin_e pin = parseBrainPin(pinName);
|
|
|
|
// todo: extract method - code duplication with other 'set_xxx_pin' methods?
|
|
|
|
if (pin == GPIO_INVALID) {
|
|
|
|
scheduleMsg(logger, "invalid pin name [%s]", pinName);
|
|
|
|
return;
|
|
|
|
}
|
2019-12-11 14:48:55 -08:00
|
|
|
CONFIG(fsioOutputPins)[index] = pin;
|
2015-07-10 06:01:56 -07:00
|
|
|
scheduleMsg(logger, "FSIO output pin #%d [%s]", (index + 1), hwPortname(pin));
|
2018-08-01 05:00:02 -07:00
|
|
|
scheduleMsg(logger, "please writeconfig and reboot for pin to take effect");
|
2017-12-17 09:02:35 -08:00
|
|
|
showFsioInfo();
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
2016-01-12 07:04:06 -08:00
|
|
|
#endif /* EFI_PROD_CODE */
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2016-01-12 09:01:43 -08:00
|
|
|
#endif
|
|
|
|
|
2016-01-12 19:01:53 -08:00
|
|
|
/**
|
|
|
|
* index is between zero and LE_COMMAND_LENGTH-1
|
|
|
|
*/
|
2019-05-03 15:41:43 -07:00
|
|
|
void setFsioExt(int index, brain_pin_e pin, const char * formula, int pwmFrequency DECLARE_CONFIG_PARAMETER_SUFFIX) {
|
2019-12-11 14:48:55 -08:00
|
|
|
CONFIG(fsioOutputPins)[index] = pin;
|
2017-12-17 09:02:35 -08:00
|
|
|
int len = strlen(formula);
|
2015-07-10 06:01:56 -07:00
|
|
|
if (len >= LE_COMMAND_LENGTH) {
|
|
|
|
return;
|
|
|
|
}
|
2017-12-17 09:02:35 -08:00
|
|
|
strcpy(config->fsioFormulas[index], formula);
|
2019-12-11 14:48:55 -08:00
|
|
|
CONFIG(fsioFrequency)[index] = pwmFrequency;
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
2019-05-03 15:41:43 -07:00
|
|
|
void setFsio(int index, brain_pin_e pin, const char * exp DECLARE_CONFIG_PARAMETER_SUFFIX) {
|
|
|
|
setFsioExt(index, pin, exp, NO_PWM PASS_CONFIG_PARAMETER_SUFFIX);
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
2017-05-15 20:28:49 -07:00
|
|
|
void applyFsioConfiguration(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
|
2016-06-26 17:03:27 -07:00
|
|
|
userPool.reset();
|
2017-06-25 23:14:31 -07:00
|
|
|
for (int i = 0; i < FSIO_COMMAND_COUNT; i++) {
|
2019-09-18 19:07:02 -07:00
|
|
|
const char *formula = config->fsioFormulas[i];
|
2020-07-01 17:45:37 -07:00
|
|
|
int len = strlen(formula);
|
2019-09-18 19:07:02 -07:00
|
|
|
LEElement *logic = userPool.parseExpression(formula);
|
2020-07-01 17:45:37 -07:00
|
|
|
if (len > 0 && logic == NULL) {
|
2019-09-18 19:07:02 -07:00
|
|
|
warning(CUSTOM_FSIO_PARSING, "parsing [%s]", formula);
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
2019-09-18 19:07:02 -07:00
|
|
|
|
|
|
|
state.fsioLogics[i] = logic;
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-03 08:42:50 -08:00
|
|
|
void onConfigurationChangeFsioCallback(engine_configuration_s *previousConfiguration DECLARE_ENGINE_PARAMETER_SUFFIX) {
|
2019-11-05 17:07:55 -08:00
|
|
|
(void)previousConfiguration;
|
2019-04-12 19:07:03 -07:00
|
|
|
#if EFI_FSIO
|
2018-02-03 08:42:50 -08:00
|
|
|
applyFsioConfiguration(PASS_ENGINE_PARAMETER_SIGNATURE);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
static LECalculator calc;
|
|
|
|
|
2018-12-24 05:37:25 -08:00
|
|
|
static SimplePwm fsioPwm[FSIO_COMMAND_COUNT] CCM_OPTIONAL;
|
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
// that's crazy, but what's an alternative? we need const char *, a shared buffer would not work for pin repository
|
|
|
|
static const char *getGpioPinName(int index) {
|
|
|
|
switch (index) {
|
|
|
|
case 0:
|
2017-05-12 11:37:07 -07:00
|
|
|
return "FSIO_OUT_0";
|
2015-07-10 06:01:56 -07:00
|
|
|
case 1:
|
2017-05-12 11:37:07 -07:00
|
|
|
return "FSIO_OUT_1";
|
2015-07-10 06:01:56 -07:00
|
|
|
case 10:
|
2017-05-12 11:37:07 -07:00
|
|
|
return "FSIO_OUT_10";
|
2015-07-10 06:01:56 -07:00
|
|
|
case 11:
|
2017-05-12 11:37:07 -07:00
|
|
|
return "FSIO_OUT_11";
|
2015-07-10 06:01:56 -07:00
|
|
|
case 12:
|
2017-05-12 11:37:07 -07:00
|
|
|
return "FSIO_OUT_12";
|
2015-07-10 06:01:56 -07:00
|
|
|
case 13:
|
2017-05-12 11:37:07 -07:00
|
|
|
return "FSIO_OUT_13";
|
2015-07-10 06:01:56 -07:00
|
|
|
case 14:
|
2017-05-12 11:37:07 -07:00
|
|
|
return "FSIO_OUT_14";
|
2015-07-10 06:01:56 -07:00
|
|
|
case 15:
|
2017-05-12 11:37:07 -07:00
|
|
|
return "FSIO_OUT_15";
|
2015-07-10 06:01:56 -07:00
|
|
|
case 2:
|
2017-05-12 11:37:07 -07:00
|
|
|
return "FSIO_OUT_2";
|
2015-07-10 06:01:56 -07:00
|
|
|
case 3:
|
2017-05-12 11:37:07 -07:00
|
|
|
return "FSIO_OUT_3";
|
2015-07-10 06:01:56 -07:00
|
|
|
case 4:
|
2017-05-12 11:37:07 -07:00
|
|
|
return "FSIO_OUT_4";
|
2015-07-10 06:01:56 -07:00
|
|
|
case 5:
|
2017-05-12 11:37:07 -07:00
|
|
|
return "FSIO_OUT_5";
|
2015-07-10 06:01:56 -07:00
|
|
|
case 6:
|
2017-05-12 11:37:07 -07:00
|
|
|
return "FSIO_OUT_6";
|
2015-07-10 06:01:56 -07:00
|
|
|
case 7:
|
2017-05-12 11:37:07 -07:00
|
|
|
return "FSIO_OUT_7";
|
2015-07-10 06:01:56 -07:00
|
|
|
case 8:
|
2017-05-12 11:37:07 -07:00
|
|
|
return "FSIO_OUT_8";
|
2015-07-10 06:01:56 -07:00
|
|
|
case 9:
|
2017-05-12 11:37:07 -07:00
|
|
|
return "FSIO_OUT_9";
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2017-12-16 21:03:26 -08:00
|
|
|
float getFsioOutputValue(int index DECLARE_ENGINE_PARAMETER_SUFFIX) {
|
2019-01-10 20:53:59 -08:00
|
|
|
if (state.fsioLogics[index] == NULL) {
|
2019-12-11 14:48:55 -08:00
|
|
|
warning(CUSTOM_NO_FSIO, "no FSIO for #%d %s", index + 1, hwPortname(CONFIG(fsioOutputPins)[index]));
|
2017-12-16 21:03:26 -08:00
|
|
|
return NAN;
|
|
|
|
} else {
|
2021-02-23 13:57:14 -08:00
|
|
|
return calc.evaluate(engine->fsioState.fsioLastValue[index], state.fsioLogics[index] PASS_ENGINE_PARAMETER_SUFFIX);
|
2017-12-16 21:03:26 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-10-01 06:02:04 -07:00
|
|
|
/**
|
2017-06-25 23:14:31 -07:00
|
|
|
* @param index from zero for (FSIO_COMMAND_COUNT - 1)
|
2016-10-01 06:02:04 -07:00
|
|
|
*/
|
2020-07-01 17:45:37 -07:00
|
|
|
static void runFsioCalculation(int index DECLARE_ENGINE_PARAMETER_SUFFIX) {
|
|
|
|
if (strlen(config->fsioFormulas[index]) == 0) {
|
2019-01-19 19:09:37 -08:00
|
|
|
engine->fsioState.fsioLastValue[index] = NAN;
|
2015-07-10 06:01:56 -07:00
|
|
|
return;
|
2017-12-17 09:02:35 -08:00
|
|
|
}
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2019-12-11 14:48:55 -08:00
|
|
|
bool isPwmMode = CONFIG(fsioFrequency)[index] != NO_PWM;
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2017-12-16 21:07:43 -08:00
|
|
|
float fvalue = getFsioOutputValue(index PASS_ENGINE_PARAMETER_SUFFIX);
|
2019-01-19 19:09:37 -08:00
|
|
|
engine->fsioState.fsioLastValue[index] = fvalue;
|
2015-07-10 06:01:56 -07:00
|
|
|
|
|
|
|
if (isPwmMode) {
|
|
|
|
fsioPwm[index].setSimplePwmDutyCycle(fvalue);
|
|
|
|
} else {
|
|
|
|
int value = (int) fvalue;
|
2016-09-14 16:03:00 -07:00
|
|
|
if (value != enginePins.fsioOutputs[index].getLogicValue()) {
|
2015-07-10 06:01:56 -07:00
|
|
|
// scheduleMsg(logger, "setting %s %s", getIo_pin_e(pin), boolToString(value));
|
2016-09-14 16:03:00 -07:00
|
|
|
enginePins.fsioOutputs[index].setValue(value);
|
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";
|
2017-06-24 22:35:46 -07:00
|
|
|
case LE_METHOD_STARTUP_FUEL_PUMP_DURATION:
|
2021-01-10 20:46:50 -08:00
|
|
|
return leStartupFuelPumpDuration.name;
|
2017-06-24 22:35:46 -07:00
|
|
|
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"
|
2017-06-24 22:35:46 -07:00
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
default: {
|
|
|
|
// this is here to make compiler happy
|
|
|
|
}
|
|
|
|
}
|
|
|
|
itoa10(buffer, (int)action);
|
|
|
|
return buffer;
|
|
|
|
}
|
|
|
|
|
2018-12-24 05:37:25 -08:00
|
|
|
static void setPinState(const char * msg, OutputPin *pin, LEElement *element DECLARE_ENGINE_PARAMETER_SUFFIX) {
|
|
|
|
#if EFI_PROD_CODE
|
2017-07-24 17:34:40 -07:00
|
|
|
if (isRunningBenchTest()) {
|
|
|
|
return; // let's not mess with bench testing
|
|
|
|
}
|
2018-12-24 05:37:25 -08:00
|
|
|
#endif /* EFI_PROD_CODE */
|
2017-07-24 17:34:40 -07:00
|
|
|
|
2020-01-21 21:45:09 -08:00
|
|
|
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 {
|
2021-02-23 13:57:14 -08:00
|
|
|
int value = (int)calc.evaluate(pin->getLogicValue(), element PASS_ENGINE_PARAMETER_SUFFIX);
|
2015-07-10 06:01:56 -07:00
|
|
|
if (pin->isInitialized() && value != pin->getLogicValue()) {
|
|
|
|
|
|
|
|
for (int i = 0;i < calc.currentCalculationLogPosition;i++) {
|
2018-01-23 09:05:14 -08:00
|
|
|
scheduleMsg(logger, "calc %d: action %s value %.2f", i, action2String(calc.calcLogAction[i]), calc.calcLogValue[i]);
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
scheduleMsg(logger, "setPin %s %s", msg, value ? "on" : "off");
|
|
|
|
pin->setValue(value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-24 05:37:25 -08:00
|
|
|
|
2019-04-12 19:07:03 -07:00
|
|
|
#if EFI_PROD_CODE
|
2016-01-12 07:04:06 -08:00
|
|
|
static void setFsioFrequency(int index, int frequency) {
|
|
|
|
index--;
|
2017-06-25 23:14:31 -07:00
|
|
|
if (index < 0 || index >= FSIO_COMMAND_COUNT) {
|
2016-01-12 19:01:53 -08:00
|
|
|
scheduleMsg(logger, "invalid FSIO index: %d", index);
|
2016-01-12 07:04:06 -08:00
|
|
|
return;
|
|
|
|
}
|
2019-12-11 14:48:55 -08:00
|
|
|
CONFIG(fsioFrequency)[index] = frequency;
|
2017-03-12 11:47:21 -07:00
|
|
|
if (frequency == 0) {
|
2019-12-11 14:48:55 -08:00
|
|
|
scheduleMsg(logger, "FSIO output #%d@%s set to on/off mode", index + 1, hwPortname(CONFIG(fsioOutputPins)[index]));
|
2017-03-12 11:47:21 -07:00
|
|
|
} else {
|
2019-12-11 14:48:55 -08:00
|
|
|
scheduleMsg(logger, "Setting FSIO frequency %dHz on #%d@%s", frequency, index + 1, hwPortname(CONFIG(fsioOutputPins)[index]));
|
2017-03-12 11:47:21 -07:00
|
|
|
}
|
2016-01-12 07:04:06 -08:00
|
|
|
}
|
2018-12-24 05:37:25 -08:00
|
|
|
#endif /* EFI_PROD_CODE */
|
2016-01-12 07:04:06 -08:00
|
|
|
|
2019-01-05 16:31:56 -08:00
|
|
|
/**
|
|
|
|
* @param out param! current and new value as long as element is not NULL
|
2019-01-05 20:48:37 -08:00
|
|
|
* @return 'true' if value has changed
|
2019-01-05 16:31:56 -08:00
|
|
|
*/
|
2020-07-01 17:45:37 -07:00
|
|
|
static bool updateValueOrWarning(int humanIndex, const char *msg, float *value DECLARE_ENGINE_PARAMETER_SUFFIX) {
|
|
|
|
int fsioIndex = humanIndex - 1;
|
2019-01-10 20:53:59 -08:00
|
|
|
LEElement * element = state.fsioLogics[fsioIndex];
|
2018-03-18 09:15:51 -07:00
|
|
|
if (element == NULL) {
|
2019-01-05 16:31:56 -08:00
|
|
|
warning(CUSTOM_FSIO_INVALID_EXPRESSION, "invalid expression for %s", msg);
|
2019-01-05 20:48:37 -08:00
|
|
|
return false;
|
2018-03-18 09:15:51 -07:00
|
|
|
} else {
|
2019-01-05 20:48:37 -08:00
|
|
|
float beforeValue = *value;
|
2021-02-23 13:57:14 -08:00
|
|
|
*value = calc.evaluate(beforeValue, element PASS_ENGINE_PARAMETER_SUFFIX);
|
2019-01-05 20:48:37 -08:00
|
|
|
// floating '==' comparison without EPS seems fine here
|
|
|
|
return (beforeValue != *value);
|
2018-03-18 09:15:51 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-05 16:31:56 -08:00
|
|
|
static void useFsioForServo(int servoIndex DECLARE_ENGINE_PARAMETER_SUFFIX) {
|
2020-07-01 17:45:37 -07:00
|
|
|
updateValueOrWarning(8 + servoIndex, "servo", &engine->fsioState.servoValues[servoIndex] PASS_ENGINE_PARAMETER_SUFFIX);
|
2019-01-05 16:31:56 -08:00
|
|
|
}
|
|
|
|
|
2017-12-16 21:03:26 -08:00
|
|
|
/**
|
|
|
|
* this method should be invoked periodically to calculate FSIO and toggle corresponding FSIO outputs
|
|
|
|
*/
|
|
|
|
void runFsio(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
|
|
|
|
for (int index = 0; index < FSIO_COMMAND_COUNT; index++) {
|
2020-07-01 17:45:37 -07:00
|
|
|
runFsioCalculation(index PASS_ENGINE_PARAMETER_SUFFIX);
|
2016-01-12 07:04:06 -08:00
|
|
|
}
|
|
|
|
|
2019-04-12 19:07:03 -07:00
|
|
|
#if EFI_FUEL_PUMP
|
2021-01-08 17:01:26 -08:00
|
|
|
if (isBrainPinValid(CONFIG(fuelPumpPin))) {
|
2018-12-24 05:37:25 -08:00
|
|
|
setPinState("pump", &enginePins.fuelPumpRelay, fuelPumpLogic PASS_ENGINE_PARAMETER_SUFFIX);
|
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
|
2021-01-08 17:01:26 -08:00
|
|
|
if (isBrainPinValid(CONFIG(mainRelayPin)))
|
2020-09-10 18:08:39 -07:00
|
|
|
// the MAIN_RELAY_LOGIC calls engine->isInShutdownMode()
|
2018-12-24 05:37:25 -08:00
|
|
|
setPinState("main_relay", &enginePins.mainRelay, mainRelayLogic PASS_ENGINE_PARAMETER_SUFFIX);
|
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
|
|
|
|
*/
|
2021-01-08 17:01:26 -08:00
|
|
|
if (isBrainPinValid(CONFIG(mainRelayPin)))
|
2021-01-11 17:01:20 -08:00
|
|
|
enginePins.mainRelay.setValue(!engine->isInMainRelayBench(PASS_ENGINE_PARAMETER_SIGNATURE));
|
2017-06-12 15:14:39 -07:00
|
|
|
#endif /* EFI_MAIN_RELAY_CONTROL */
|
2016-01-12 07:04:06 -08:00
|
|
|
|
2021-01-08 17:01:26 -08:00
|
|
|
if (isBrainPinValid(CONFIG(starterRelayDisablePin)))
|
2020-03-18 13:32:03 -07:00
|
|
|
setPinState("starter_relay", &enginePins.starterRelayDisable, starterRelayDisableLogic PASS_ENGINE_PARAMETER_SUFFIX);
|
2019-11-25 17:08:01 -08:00
|
|
|
|
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
|
|
|
*/
|
2020-09-05 15:49:42 -07:00
|
|
|
enginePins.o2heater.setValue(engine->rpmCalculator.isRunning());
|
2016-01-12 07:04:06 -08:00
|
|
|
|
2021-01-08 17:01:26 -08:00
|
|
|
if (isBrainPinValid(CONFIG(acRelayPin))) {
|
2018-12-24 05:37:25 -08:00
|
|
|
setPinState("A/C", &enginePins.acRelay, acRelayLogic PASS_ENGINE_PARAMETER_SUFFIX);
|
2016-01-12 07:04:06 -08:00
|
|
|
}
|
|
|
|
|
2021-01-08 17:01:26 -08:00
|
|
|
// if (isBrainPinValid(CONFIG(alternatorControlPin))) {
|
2018-12-24 05:37:25 -08:00
|
|
|
// setPinState("alternator", &enginePins.alternatorField, alternatorLogic, engine PASS_ENGINE_PARAMETER_SUFFIX);
|
2016-01-12 07:04:06 -08:00
|
|
|
// }
|
|
|
|
|
2021-01-08 17:01:26 -08:00
|
|
|
if (isBrainPinValid(CONFIG(fanPin))) {
|
2018-12-24 05:37:25 -08:00
|
|
|
setPinState("fan", &enginePins.fanRelay, radiatorFanLogic PASS_ENGINE_PARAMETER_SUFFIX);
|
2016-01-12 07:04:06 -08:00
|
|
|
}
|
2018-07-29 15:02:37 -07:00
|
|
|
|
2019-01-05 20:48:37 -08:00
|
|
|
#if EFI_ENABLE_ENGINE_WARNING
|
|
|
|
if (engineConfiguration->useFSIO4ForSeriousEngineWarning) {
|
|
|
|
updateValueOrWarning(MAGIC_OFFSET_FOR_ENGINE_WARNING, "eng warning", &ENGINE(fsioState.isEngineWarning) PASS_ENGINE_PARAMETER_SUFFIX);
|
|
|
|
}
|
2019-01-06 04:37:29 -08:00
|
|
|
#endif /* EFI_ENABLE_ENGINE_WARNING */
|
2019-01-05 20:48:37 -08:00
|
|
|
|
|
|
|
#if EFI_ENABLE_CRITICAL_ENGINE_STOP
|
|
|
|
if (engineConfiguration->useFSIO5ForCriticalIssueEngineStop) {
|
|
|
|
bool changed = updateValueOrWarning(MAGIC_OFFSET_FOR_CRITICAL_ENGINE, "eng critical", &ENGINE(fsioState.isCriticalEngineCondition) PASS_ENGINE_PARAMETER_SUFFIX);
|
|
|
|
if (changed && float2bool(ENGINE(fsioState.isCriticalEngineCondition))) {
|
2019-01-19 19:09:37 -08:00
|
|
|
doScheduleStopEngine(PASS_ENGINE_PARAMETER_SIGNATURE);
|
2019-01-05 20:48:37 -08:00
|
|
|
}
|
|
|
|
}
|
2019-01-06 04:37:29 -08:00
|
|
|
#endif /* EFI_ENABLE_CRITICAL_ENGINE_STOP */
|
2019-01-05 20:48:37 -08:00
|
|
|
|
2019-09-11 17:46:50 -07:00
|
|
|
if (engineConfiguration->useFSIO12ForIdleOffset) {
|
|
|
|
updateValueOrWarning(MAGIC_OFFSET_FOR_IDLE_OFFSET, "idle offset", &ENGINE(fsioState.fsioIdleOffset) PASS_ENGINE_PARAMETER_SUFFIX);
|
|
|
|
}
|
|
|
|
if (engineConfiguration->useFSIO13ForIdleMinValue) {
|
|
|
|
updateValueOrWarning(MAGIC_OFFSET_FOR_IDLE_MIN_VALUE, "idle minValue", &ENGINE(fsioState.fsioIdleMinValue) PASS_ENGINE_PARAMETER_SUFFIX);
|
|
|
|
}
|
2018-07-29 15:02:37 -07:00
|
|
|
if (engineConfiguration->useFSIO15ForIdleRpmAdjustment) {
|
2019-01-19 18:10:58 -08:00
|
|
|
updateValueOrWarning(MAGIC_OFFSET_FOR_IDLE_TARGET_RPM, "RPM target", &ENGINE(fsioState.fsioIdleTargetRPMAdjustment) PASS_ENGINE_PARAMETER_SUFFIX);
|
2018-07-29 15:02:37 -07:00
|
|
|
}
|
2017-07-25 17:37:46 -07:00
|
|
|
if (engineConfiguration->useFSIO16ForTimingAdjustment) {
|
2019-01-19 18:10:58 -08:00
|
|
|
updateValueOrWarning(MAGIC_OFFSET_FOR_TIMING_FSIO, "timing", &ENGINE(fsioState.fsioTimingAdjustment) PASS_ENGINE_PARAMETER_SUFFIX);
|
2017-07-25 17:37:46 -07:00
|
|
|
}
|
2016-01-12 07:04:06 -08:00
|
|
|
|
2020-06-01 06:09:55 -07:00
|
|
|
if (engineConfiguration->useFSIO6ForRevLimiter) {
|
2020-07-01 17:45:37 -07:00
|
|
|
updateValueOrWarning(6, "rpm limit", &ENGINE(fsioState.fsioRpmHardLimit) PASS_ENGINE_PARAMETER_SUFFIX);
|
2020-06-01 06:09:55 -07:00
|
|
|
}
|
|
|
|
|
2018-03-18 09:15:51 -07:00
|
|
|
if (engineConfiguration->useFSIO8ForServo1) {
|
|
|
|
useFsioForServo(0 PASS_ENGINE_PARAMETER_SUFFIX);
|
|
|
|
}
|
|
|
|
if (engineConfiguration->useFSIO9ForServo2) {
|
|
|
|
useFsioForServo(1 PASS_ENGINE_PARAMETER_SUFFIX);
|
|
|
|
}
|
|
|
|
if (engineConfiguration->useFSIO10ForServo3) {
|
|
|
|
useFsioForServo(2 PASS_ENGINE_PARAMETER_SUFFIX);
|
|
|
|
}
|
|
|
|
if (engineConfiguration->useFSIO11ForServo4) {
|
|
|
|
useFsioForServo(3 PASS_ENGINE_PARAMETER_SUFFIX);
|
|
|
|
}
|
|
|
|
if (engineConfiguration->useFSIO12ForServo5) {
|
|
|
|
useFsioForServo(4 PASS_ENGINE_PARAMETER_SUFFIX);
|
|
|
|
}
|
|
|
|
|
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)
|
|
|
|
scheduleMsg(logger, "%s:", msg);
|
2021-02-17 20:32:44 -08:00
|
|
|
while (element->action != LE_METHOD_RETURN) {
|
2020-12-25 06:47:33 -08:00
|
|
|
scheduleMsg(logger, "action %d: fValue=%.2f", element->action, element->fValue);
|
2021-02-17 20:32:44 -08:00
|
|
|
element++;
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
scheduleMsg(logger, "<end>");
|
2016-01-12 09:01:43 -08:00
|
|
|
#endif
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void showFsioInfo(void) {
|
2016-01-12 09:01:43 -08:00
|
|
|
#if EFI_PROD_CODE || EFI_SIMULATOR
|
2015-07-10 06:01:56 -07:00
|
|
|
scheduleMsg(logger, "sys used %d/user used %d", sysPool.getSize(), userPool.getSize());
|
|
|
|
showFsio("a/c", acRelayLogic);
|
|
|
|
showFsio("fuel", fuelPumpLogic);
|
|
|
|
showFsio("fan", radiatorFanLogic);
|
|
|
|
showFsio("alt", alternatorLogic);
|
|
|
|
|
2021-02-10 16:51:14 -08:00
|
|
|
for (int i = 0; i < CAM_INPUTS_COUNT ; i++) {
|
2016-12-27 21:02:03 -08:00
|
|
|
brain_pin_e pin = engineConfiguration->auxPidPins[i];
|
2021-01-08 17:01:26 -08:00
|
|
|
if (isBrainPinValid(pin)) {
|
2021-02-10 16:51:14 -08:00
|
|
|
scheduleMsg(logger, "VVT pid #%d [%s]", (i + 1),
|
2016-12-27 21:02:03 -08:00
|
|
|
hwPortname(pin));
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2017-06-25 23:14:31 -07:00
|
|
|
for (int i = 0; i < FSIO_COMMAND_COUNT; i++) {
|
2017-06-25 23:23:26 -07:00
|
|
|
char * exp = config->fsioFormulas[i];
|
2015-07-10 06:01:56 -07:00
|
|
|
if (exp[0] != 0) {
|
|
|
|
/**
|
|
|
|
* in case of FSIO user interface indexes are starting with 0, the argument for that
|
|
|
|
* is the fact that the target audience is more software developers
|
|
|
|
*/
|
2020-04-10 12:49:44 -07:00
|
|
|
int freq = CONFIG(fsioFrequency)[i];
|
|
|
|
const char *modeMessage = freq == 0 ? " (on/off mode)" : "";
|
|
|
|
scheduleMsg(logger, "FSIO #%d [%s] at %s@%dHz%s value=%.2f", (i + 1), exp,
|
|
|
|
hwPortname(CONFIG(fsioOutputPins)[i]),
|
|
|
|
freq, modeMessage,
|
2019-01-19 19:09:37 -08:00
|
|
|
engine->fsioState.fsioLastValue[i]);
|
2019-01-04 21:11:17 -08:00
|
|
|
// scheduleMsg(logger, "user-defined #%d value=%.2f", i, engine->engineConfigurationPtr2->fsioLastValue[i]);
|
2019-01-10 20:53:59 -08:00
|
|
|
showFsio(NULL, state.fsioLogics[i]);
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
}
|
2017-06-25 23:14:31 -07:00
|
|
|
for (int i = 0; i < FSIO_COMMAND_COUNT; i++) {
|
2019-12-11 14:48:55 -08:00
|
|
|
float v = CONFIG(fsio_setting)[i];
|
2015-07-10 06:01:56 -07:00
|
|
|
if (!cisnan(v)) {
|
2018-01-23 09:05:14 -08:00
|
|
|
scheduleMsg(logger, "user property #%d: %.2f", i + 1, v);
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
}
|
2017-06-25 23:14:31 -07:00
|
|
|
for (int i = 0; i < FSIO_COMMAND_COUNT; i++) {
|
2019-11-30 07:24:37 -08:00
|
|
|
brain_pin_e inputPin = CONFIG(fsioDigitalInputs)[i];
|
2021-01-08 17:01:26 -08:00
|
|
|
if (isBrainPinValid(inputPin)) {
|
2015-07-10 06:01:56 -07:00
|
|
|
scheduleMsg(logger, "FSIO digital input #%d: %s", i, hwPortname(inputPin));
|
|
|
|
}
|
|
|
|
}
|
2016-01-12 09:01:43 -08:00
|
|
|
#endif
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-03-12 11:47:21 -07:00
|
|
|
* set_fsio_setting 1 0.11
|
2015-07-10 06:01:56 -07:00
|
|
|
*/
|
2017-03-12 12:22:07 -07:00
|
|
|
static void setFsioSetting(float humanIndexF, float value) {
|
2016-01-12 09:01:43 -08:00
|
|
|
#if EFI_PROD_CODE || EFI_SIMULATOR
|
2017-03-12 12:22:07 -07:00
|
|
|
int index = (int)humanIndexF - 1;
|
2017-06-25 23:14:31 -07:00
|
|
|
if (index < 0 || index >= FSIO_COMMAND_COUNT) {
|
2017-03-12 12:22:07 -07:00
|
|
|
scheduleMsg(logger, "invalid FSIO index: %d", (int)humanIndexF);
|
2015-07-10 06:01:56 -07:00
|
|
|
return;
|
|
|
|
}
|
2019-12-11 14:48:55 -08:00
|
|
|
engineConfiguration->fsio_setting[index] = value;
|
2015-07-10 06:01:56 -07:00
|
|
|
showFsioInfo();
|
2016-01-12 09:01:43 -08:00
|
|
|
#endif
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
2020-07-29 19:48:41 -07:00
|
|
|
void setFsioExpression(const char *indexStr, const char *quotedLine DECLARE_CONFIG_PARAMETER_SUFFIX) {
|
2015-07-10 06:01:56 -07:00
|
|
|
int index = atoi(indexStr) - 1;
|
2017-06-25 23:14:31 -07:00
|
|
|
if (index < 0 || index >= FSIO_COMMAND_COUNT) {
|
2016-01-12 19:01:53 -08:00
|
|
|
scheduleMsg(logger, "invalid FSIO index: %d", index);
|
2015-07-10 06:01:56 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
char * l = unquote((char*) quotedLine);
|
|
|
|
if (strlen(l) > LE_COMMAND_LENGTH - 1) {
|
|
|
|
scheduleMsg(logger, "Too long %d", strlen(l));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
scheduleMsg(logger, "setting user out #%d to [%s]", index + 1, l);
|
2017-11-26 16:35:08 -08:00
|
|
|
strcpy(config->fsioFormulas[index], l);
|
2020-07-29 19:48:41 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void applyFsioExpression(const char *indexStr, const char *quotedLine DECLARE_ENGINE_PARAMETER_SUFFIX) {
|
|
|
|
setFsioExpression(indexStr, quotedLine PASS_CONFIG_PARAMETER_SUFFIX);
|
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
// this would apply the changes
|
2017-05-15 20:28:49 -07:00
|
|
|
applyFsioConfiguration(PASS_ENGINE_PARAMETER_SIGNATURE);
|
2015-07-10 06:01:56 -07:00
|
|
|
showFsioInfo();
|
|
|
|
}
|
|
|
|
|
2017-06-04 12:25:37 -07:00
|
|
|
static void rpnEval(char *line) {
|
2016-01-12 09:01:43 -08:00
|
|
|
#if EFI_PROD_CODE || EFI_SIMULATOR
|
2015-07-10 06:01:56 -07:00
|
|
|
line = unquote(line);
|
|
|
|
scheduleMsg(logger, "Parsing [%s]", line);
|
|
|
|
evalPool.reset();
|
|
|
|
LEElement * e = evalPool.parseExpression(line);
|
|
|
|
if (e == NULL) {
|
|
|
|
scheduleMsg(logger, "parsing failed");
|
|
|
|
} else {
|
2021-02-23 13:57:14 -08:00
|
|
|
float result = evalCalc.evaluate(0, e PASS_ENGINE_PARAMETER_SUFFIX);
|
2018-01-23 09:05:14 -08:00
|
|
|
scheduleMsg(logger, "Evaluate result: %.2f", result);
|
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
|
|
|
}
|
|
|
|
|
2019-07-12 05:31:38 -07:00
|
|
|
ValueProvider3D *getFSIOTable(int index) {
|
|
|
|
switch (index) {
|
|
|
|
default:
|
|
|
|
return &fsioTable1;
|
|
|
|
case 1:
|
|
|
|
return &fsioTable2;
|
|
|
|
case 2:
|
|
|
|
return &fsioTable3;
|
|
|
|
case 3:
|
|
|
|
return &fsioTable4;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-15 20:28:49 -07:00
|
|
|
void initFsioImpl(Logging *sharedLogger DECLARE_ENGINE_PARAMETER_SUFFIX) {
|
2016-01-12 09:01:43 -08:00
|
|
|
#if EFI_PROD_CODE || EFI_SIMULATOR
|
2015-07-10 06:01:56 -07:00
|
|
|
logger = sharedLogger;
|
2019-05-12 17:31:28 -07:00
|
|
|
#else
|
|
|
|
// 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
|
|
|
|
|
|
|
acRelayLogic = sysPool.parseExpression(AC_RELAY_LOGIC);
|
|
|
|
radiatorFanLogic = sysPool.parseExpression(FAN_CONTROL_LOGIC);
|
|
|
|
|
|
|
|
alternatorLogic = sysPool.parseExpression(ALTERNATOR_LOGIC);
|
2017-06-12 15:14:39 -07:00
|
|
|
|
2019-04-12 19:07:03 -07:00
|
|
|
#if EFI_MAIN_RELAY_CONTROL
|
2021-01-08 17:01:26 -08:00
|
|
|
if (isBrainPinValid(CONFIG(mainRelayPin)))
|
2017-06-12 15:14:39 -07:00
|
|
|
mainRelayLogic = sysPool.parseExpression(MAIN_RELAY_LOGIC);
|
|
|
|
#endif /* EFI_MAIN_RELAY_CONTROL */
|
2021-01-08 17:01:26 -08:00
|
|
|
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
|
|
|
|
2019-04-12 19:07:03 -07:00
|
|
|
#if EFI_PROD_CODE
|
2017-06-25 23:14:31 -07:00
|
|
|
for (int i = 0; i < FSIO_COMMAND_COUNT; i++) {
|
2019-12-11 14:48:55 -08:00
|
|
|
brain_pin_e brainPin = CONFIG(fsioOutputPins)[i];
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2021-01-08 17:01:26 -08:00
|
|
|
if (isBrainPinValid(brainPin)) {
|
2019-12-11 14:48:55 -08:00
|
|
|
int frequency = CONFIG(fsioFrequency)[i];
|
2015-07-10 06:01:56 -07:00
|
|
|
if (frequency == 0) {
|
2020-04-04 12:59:13 -07:00
|
|
|
enginePins.fsioOutputs[i].initPin(getGpioPinName(i), CONFIG(fsioOutputPins)[i]);
|
2015-07-10 06:01:56 -07:00
|
|
|
} else {
|
2019-01-09 05:50:51 -08:00
|
|
|
startSimplePwmExt(&fsioPwm[i], "FSIOpwm",
|
|
|
|
&engine->executor,
|
2020-03-25 15:14:09 -07:00
|
|
|
brainPin, &enginePins.fsioOutputs[i], frequency, 0.5f);
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-25 23:14:31 -07:00
|
|
|
for (int i = 0; i < FSIO_COMMAND_COUNT; i++) {
|
2019-11-30 07:24:37 -08:00
|
|
|
brain_pin_e inputPin = CONFIG(fsioDigitalInputs)[i];
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2021-01-08 17:01:26 -08:00
|
|
|
if (isBrainPinValid(inputPin)) {
|
2017-05-15 05:40:54 -07:00
|
|
|
efiSetPadMode("FSIO input", inputPin, getInputMode(engineConfiguration->fsioInputModes[i]));
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-27 21:02:03 -08:00
|
|
|
addConsoleActionSS("set_fsio_pid_output_pin", (VoidCharPtrCharPtr) setFsioPidOutputPin);
|
2016-01-12 07:04:06 -08:00
|
|
|
addConsoleActionSS("set_fsio_output_pin", (VoidCharPtrCharPtr) setFsioOutputPin);
|
|
|
|
addConsoleActionII("set_fsio_output_frequency", (VoidIntInt) setFsioFrequency);
|
2018-08-01 20:02:19 -07:00
|
|
|
addConsoleActionSS("set_fsio_digital_input_pin", (VoidCharPtrCharPtr) setFsioDigitalInputPin);
|
2018-08-01 20:27:22 -07:00
|
|
|
addConsoleActionSS("set_fsio_analog_input_pin", (VoidCharPtrCharPtr) setFsioAnalogInputPin);
|
2016-01-12 07:04:06 -08:00
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
#endif /* EFI_PROD_CODE */
|
|
|
|
|
2016-01-12 09:01:43 -08:00
|
|
|
#if EFI_PROD_CODE || EFI_SIMULATOR
|
2020-07-29 19:48:41 -07:00
|
|
|
addConsoleActionSS("set_rpn_expression", applyFsioExpression);
|
2015-07-10 06:01:56 -07:00
|
|
|
addConsoleActionFF("set_fsio_setting", setFsioSetting);
|
|
|
|
addConsoleAction("fsioinfo", showFsioInfo);
|
2017-06-04 12:25:37 -07:00
|
|
|
addConsoleActionS("rpn_eval", (VoidCharPtr) rpnEval);
|
|
|
|
#endif /* EFI_PROD_CODE || EFI_SIMULATOR */
|
2016-04-04 07:01:43 -07:00
|
|
|
|
|
|
|
fsioTable1.init(config->fsioTable1, config->fsioTable1LoadBins,
|
|
|
|
config->fsioTable1RpmBins);
|
|
|
|
fsioTable2.init(config->fsioTable2, config->fsioTable2LoadBins,
|
|
|
|
config->fsioTable2RpmBins);
|
2016-07-01 20:01:22 -07:00
|
|
|
fsioTable3.init(config->fsioTable3, config->fsioTable3LoadBins,
|
|
|
|
config->fsioTable3RpmBins);
|
|
|
|
fsioTable4.init(config->fsioTable4, config->fsioTable4LoadBins,
|
|
|
|
config->fsioTable4RpmBins);
|
2016-04-04 07:01:43 -07:00
|
|
|
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
2019-11-25 17:08:01 -08:00
|
|
|
#else /* !EFI_FSIO */
|
|
|
|
|
|
|
|
// "Limp-mode" implementation for some RAM-limited configs without FSIO
|
|
|
|
void runHardcodedFsio(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
|
2020-09-10 18:08:39 -07:00
|
|
|
#if EFI_PROD_CODE
|
|
|
|
if (isRunningBenchTest()) {
|
|
|
|
return; // let's not mess with bench testing
|
|
|
|
}
|
|
|
|
#endif /* EFI_PROD_CODE */
|
|
|
|
|
2019-11-25 17:08:01 -08:00
|
|
|
// see MAIN_RELAY_LOGIC
|
2021-01-08 17:01:26 -08:00
|
|
|
if (isBrainPinValid(CONFIG(mainRelayPin))) {
|
2021-03-07 17:18:32 -08:00
|
|
|
enginePins.mainRelay.setValue((getTimeNowSeconds() < 2) || (Sensor::get(SensorType::BatteryVoltage).value_or(0) > LOW_VBATT) || engine->isInShutdownMode());
|
2019-11-25 17:08:01 -08:00
|
|
|
}
|
|
|
|
// see STARTER_RELAY_LOGIC
|
2021-01-08 17:01:26 -08:00
|
|
|
if (isBrainPinValid(CONFIG(starterRelayDisablePin))) {
|
2020-03-18 16:41:12 -07:00
|
|
|
enginePins.starterRelayDisable.setValue(engine->rpmCalculator.getRpm() < engineConfiguration->cranking.rpm);
|
2019-11-25 17:08:01 -08:00
|
|
|
}
|
|
|
|
// see FAN_CONTROL_LOGIC
|
2021-01-08 17:01:26 -08:00
|
|
|
if (isBrainPinValid(CONFIG(fanPin))) {
|
2020-04-18 15:45:30 -07:00
|
|
|
auto clt = Sensor::get(SensorType::Clt);
|
|
|
|
enginePins.fanRelay.setValue(!clt.Valid || (enginePins.fanRelay.getLogicValue() && (clt.Value > engineConfiguration->fanOffTemperature)) ||
|
2021-02-16 13:58:54 -08:00
|
|
|
(clt.Value > engineConfiguration->fanOnTemperature) || !clt.Valid);
|
2019-11-25 17:08:01 -08:00
|
|
|
}
|
|
|
|
// see AC_RELAY_LOGIC
|
2021-01-08 17:01:26 -08:00
|
|
|
if (isBrainPinValid(CONFIG(acRelayPin))) {
|
2019-11-25 17:08:01 -08:00
|
|
|
enginePins.acRelay.setValue(getAcToggle(PASS_ENGINE_PARAMETER_SIGNATURE) && engine->rpmCalculator.getRpm() > 850);
|
|
|
|
}
|
|
|
|
// see FUEL_PUMP_LOGIC
|
2021-01-08 17:01:26 -08:00
|
|
|
if (isBrainPinValid(CONFIG(fuelPumpPin))) {
|
2020-12-02 20:09:39 -08:00
|
|
|
enginePins.fuelPumpRelay.setValue((getTimeNowSeconds() < engine->triggerActivitySecond + engineConfiguration->startUpFuelPumpDuration) || (engine->rpmCalculator.getRpm() > 0));
|
2019-11-25 17:08:01 -08:00
|
|
|
}
|
|
|
|
|
2020-09-05 15:49:42 -07:00
|
|
|
enginePins.o2heater.setValue(engine->rpmCalculator.isRunning());
|
2019-11-25 17:08:01 -08:00
|
|
|
}
|
2015-07-10 06:01:56 -07:00
|
|
|
|
|
|
|
#endif /* EFI_FSIO */
|