2015-07-10 06:01:56 -07:00
|
|
|
/**
|
|
|
|
* @file fsio_impl.cpp
|
|
|
|
* @brief FSIO as it's used for GPIO
|
|
|
|
*
|
|
|
|
* @date Oct 5, 2014
|
2015-12-31 13:02:30 -08:00
|
|
|
* @author Andrey Belomutskiy, (c) 2012-2016
|
2015-07-10 06:01:56 -07:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "main.h"
|
|
|
|
|
|
|
|
#if EFI_FSIO || defined(__DOXYGEN__)
|
|
|
|
|
|
|
|
#include "fsio_impl.h"
|
|
|
|
#include "allsensors.h"
|
|
|
|
#include "rpm_calculator.h"
|
|
|
|
#include "efiGpio.h"
|
|
|
|
|
|
|
|
#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");
|
|
|
|
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");
|
|
|
|
static LENameOrdinalPair leAcToggle(LE_METHOD_AC_TOGGLE, "ac_on_switch");
|
|
|
|
static LENameOrdinalPair leFanOnSetting(LE_METHOD_FAN_ON_SETTING, "fan_on_setting");
|
|
|
|
static LENameOrdinalPair leFanOffSetting(LE_METHOD_FAN_OFF_SETTING, "fan_off_setting");
|
|
|
|
static LENameOrdinalPair leTimeSinceBoot(LE_METHOD_TIME_SINCE_BOOT, "time_since_boot");
|
|
|
|
static LENameOrdinalPair leFsioSetting(LE_METHOD_FSIO_SETTING, "fsio_setting");
|
2016-03-22 11:03:44 -07:00
|
|
|
static LENameOrdinalPair leFsioTable(LE_METHOD_FSIO_TABLE, "fsio_table");
|
2016-10-22 21:03:08 -07:00
|
|
|
static LENameOrdinalPair leFsioAnalogInput(LE_METHOD_FSIO_ANALOG_INPUT, "fsio_input");
|
2015-07-10 06:01:56 -07:00
|
|
|
static LENameOrdinalPair leKnock(LE_METHOD_KNOCK, "knock");
|
|
|
|
|
|
|
|
#define LE_EVAL_POOL_SIZE 32
|
|
|
|
|
|
|
|
extern engine_pins_s enginePins;
|
|
|
|
|
|
|
|
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
|
|
|
|
|
|
|
|
static LEElement sysElements[SYS_ELEMENT_POOL_SIZE];
|
|
|
|
LEElementPool sysPool(sysElements, SYS_ELEMENT_POOL_SIZE);
|
|
|
|
|
|
|
|
static LEElement userElements[UD_ELEMENT_POOL_SIZE];
|
|
|
|
LEElementPool userPool(userElements, UD_ELEMENT_POOL_SIZE);
|
|
|
|
static LEElement * fsioLogics[LE_COMMAND_COUNT] CCM_OPTIONAL;
|
|
|
|
|
|
|
|
static LEElement * acRelayLogic;
|
|
|
|
static LEElement * fuelPumpLogic;
|
|
|
|
static LEElement * radiatorFanLogic;
|
|
|
|
static LEElement * alternatorLogic;
|
|
|
|
|
|
|
|
EXTERN_ENGINE
|
|
|
|
;
|
|
|
|
|
|
|
|
#if EFI_PROD_CODE || EFI_SIMULATOR
|
|
|
|
static Logging *logger;
|
|
|
|
|
|
|
|
float getLEValue(Engine *engine, calc_stack_t *s, le_action_e action) {
|
|
|
|
engine_configuration_s *engineConfiguration = engine->engineConfiguration;
|
|
|
|
efiAssert(engine!=NULL, "getLEValue", NAN);
|
|
|
|
switch (action) {
|
|
|
|
case LE_METHOD_FAN:
|
|
|
|
return enginePins.fanRelay.getLogicValue();
|
|
|
|
case LE_METHOD_AC_TOGGLE:
|
|
|
|
return getAcToggle(PASS_ENGINE_PARAMETER_F);
|
|
|
|
case LE_METHOD_COOLANT:
|
|
|
|
return getCoolantTemperature(PASS_ENGINE_PARAMETER_F);
|
|
|
|
case LE_METHOD_INTAKE_AIR:
|
|
|
|
return getIntakeAirTemperature(PASS_ENGINE_PARAMETER_F);
|
|
|
|
case LE_METHOD_RPM:
|
2016-01-18 09:03:32 -08:00
|
|
|
return engine->rpmCalculator.getRpm();
|
2016-07-07 20:01:43 -07:00
|
|
|
case LE_METHOD_MAF:
|
|
|
|
return getMaf(PASS_ENGINE_PARAMETER_F);
|
|
|
|
case LE_METHOD_MAP:
|
|
|
|
return getMap();
|
2015-07-10 06:01:56 -07:00
|
|
|
case LE_METHOD_TIME_SINCE_BOOT:
|
|
|
|
return getTimeNowSeconds();
|
|
|
|
case LE_METHOD_FAN_OFF_SETTING:
|
|
|
|
return engineConfiguration->fanOffTemperature;
|
|
|
|
case LE_METHOD_FAN_ON_SETTING:
|
|
|
|
return engineConfiguration->fanOnTemperature;
|
|
|
|
case LE_METHOD_VBATT:
|
|
|
|
return getVBatt(PASS_ENGINE_PARAMETER_F);
|
|
|
|
default:
|
2016-07-13 18:03:05 -07:00
|
|
|
warning(CUSTOM_OBD_9, "FSIO unexpected %d", action);
|
2015-07-10 06:01:56 -07:00
|
|
|
return NAN;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#if EFI_PROD_CODE || defined(__DOXYGEN__)
|
|
|
|
|
|
|
|
#include "pin_repository.h"
|
|
|
|
#include "pwm_generator.h"
|
|
|
|
// todo: that's about bench test mode, wrong header for sure!
|
|
|
|
#include "injector_central.h"
|
|
|
|
|
|
|
|
static void setFsioInputPin(const char *indexStr, const char *pinName) {
|
|
|
|
int index = atoi(indexStr) - 1;
|
|
|
|
if (index < 0 || index >= LE_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;
|
|
|
|
}
|
|
|
|
boardConfiguration->fsioDigitalInputs[index] = pin;
|
|
|
|
scheduleMsg(logger, "FSIO input pin #%d [%s]", (index + 1), hwPortname(pin));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void setFsioOutputPin(const char *indexStr, const char *pinName) {
|
|
|
|
int index = atoi(indexStr) - 1;
|
|
|
|
if (index < 0 || index >= LE_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;
|
|
|
|
}
|
|
|
|
boardConfiguration->fsioPins[index] = pin;
|
|
|
|
scheduleMsg(logger, "FSIO output pin #%d [%s]", (index + 1), hwPortname(pin));
|
|
|
|
}
|
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
|
|
|
|
*/
|
2015-07-10 06:01:56 -07:00
|
|
|
void setFsioExt(int index, brain_pin_e pin, const char * exp, int freq DECLARE_ENGINE_PARAMETER_S) {
|
|
|
|
boardConfiguration->fsioPins[index] = pin;
|
|
|
|
int len = strlen(exp);
|
|
|
|
if (len >= LE_COMMAND_LENGTH) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
strcpy(config->le_formulas[index], exp);
|
|
|
|
boardConfiguration->fsioFrequency[index] = freq;
|
|
|
|
}
|
|
|
|
|
|
|
|
void setFsio(int index, brain_pin_e pin, const char * exp DECLARE_ENGINE_PARAMETER_S) {
|
|
|
|
setFsioExt(index, pin, exp, NO_PWM PASS_ENGINE_PARAMETER);
|
|
|
|
}
|
|
|
|
|
|
|
|
void applyFsioConfiguration(DECLARE_ENGINE_PARAMETER_F) {
|
2016-06-26 17:03:27 -07:00
|
|
|
userPool.reset();
|
2015-07-10 06:01:56 -07:00
|
|
|
for (int i = 0; i < LE_COMMAND_COUNT; i++) {
|
|
|
|
brain_pin_e brainPin = boardConfiguration->fsioPins[i];
|
|
|
|
|
|
|
|
if (brainPin != GPIO_UNASSIGNED) {
|
|
|
|
const char *formula = config->le_formulas[i];
|
|
|
|
LEElement *logic = userPool.parseExpression(formula);
|
|
|
|
if (logic == NULL) {
|
2016-07-13 18:03:05 -07:00
|
|
|
warning(CUSTOM_OBD_10, "parsing [%s]", formula);
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
fsioLogics[i] = logic;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#if EFI_PROD_CODE || defined(__DOXYGEN__)
|
|
|
|
|
|
|
|
static SimplePwm fsioPwm[LE_COMMAND_COUNT] CCM_OPTIONAL;
|
|
|
|
|
|
|
|
static LECalculator calc;
|
|
|
|
extern LEElement * fsioLogics[LE_COMMAND_COUNT];
|
|
|
|
|
|
|
|
// 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:
|
|
|
|
return "GPIO_0";
|
|
|
|
case 1:
|
|
|
|
return "GPIO_1";
|
|
|
|
case 10:
|
|
|
|
return "GPIO_10";
|
|
|
|
case 11:
|
|
|
|
return "GPIO_11";
|
|
|
|
case 12:
|
|
|
|
return "GPIO_12";
|
|
|
|
case 13:
|
|
|
|
return "GPIO_13";
|
|
|
|
case 14:
|
|
|
|
return "GPIO_14";
|
|
|
|
case 15:
|
|
|
|
return "GPIO_15";
|
|
|
|
case 2:
|
|
|
|
return "GPIO_2";
|
|
|
|
case 3:
|
|
|
|
return "GPIO_3";
|
|
|
|
case 4:
|
|
|
|
return "GPIO_4";
|
|
|
|
case 5:
|
|
|
|
return "GPIO_5";
|
|
|
|
case 6:
|
|
|
|
return "GPIO_6";
|
|
|
|
case 7:
|
|
|
|
return "GPIO_7";
|
|
|
|
case 8:
|
|
|
|
return "GPIO_8";
|
|
|
|
case 9:
|
|
|
|
return "GPIO_9";
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2016-10-01 06:02:04 -07:00
|
|
|
/**
|
|
|
|
* @param index from zero for (LE_COMMAND_COUNT - 1)
|
|
|
|
*/
|
2015-07-10 06:01:56 -07:00
|
|
|
static void handleFsio(Engine *engine, int index) {
|
|
|
|
if (boardConfiguration->fsioPins[index] == GPIO_UNASSIGNED)
|
|
|
|
return;
|
|
|
|
|
2016-01-11 14:01:33 -08:00
|
|
|
bool isPwmMode = boardConfiguration->fsioFrequency[index] != NO_PWM;
|
2015-07-10 06:01:56 -07:00
|
|
|
|
2016-10-01 06:02:04 -07:00
|
|
|
float fvalue;
|
|
|
|
if (fsioLogics[index] == NULL) {
|
|
|
|
warning(CUSTOM_NO_FSIO, "no FSIO for #%d %s", index + 1, hwPortname(boardConfiguration->fsioPins[index]));
|
|
|
|
fvalue = NAN;
|
|
|
|
} else {
|
|
|
|
fvalue = calc.getValue2(engine->engineConfiguration2->fsioLastValue[index], fsioLogics[index], engine);
|
|
|
|
}
|
2015-07-10 06:01:56 -07:00
|
|
|
engine->engineConfiguration2->fsioLastValue[index] = fvalue;
|
|
|
|
|
|
|
|
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";
|
|
|
|
case LE_METHOD_COOLANT:
|
|
|
|
return "CLT";
|
|
|
|
case LE_METHOD_FAN_ON_SETTING:
|
|
|
|
return "fan_on";
|
|
|
|
case LE_METHOD_FAN_OFF_SETTING:
|
|
|
|
return "fan_off";
|
|
|
|
case LE_METHOD_FAN:
|
|
|
|
return "fan";
|
|
|
|
default: {
|
|
|
|
// this is here to make compiler happy
|
|
|
|
}
|
|
|
|
}
|
|
|
|
itoa10(buffer, (int)action);
|
|
|
|
return buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void setPinState(const char * msg, OutputPin *pin, LEElement *element, Engine *engine) {
|
|
|
|
if (element == NULL) {
|
2016-07-13 18:03:05 -07:00
|
|
|
warning(CUSTOM_OBD_11, "invalid expression for %s", msg);
|
2015-07-10 06:01:56 -07:00
|
|
|
} else {
|
2016-03-02 19:02:37 -08:00
|
|
|
int value = calc.getValue2(pin->getLogicValue(), element, engine);
|
2015-07-10 06:01:56 -07:00
|
|
|
if (pin->isInitialized() && value != pin->getLogicValue()) {
|
|
|
|
if (isRunningBenchTest()) {
|
|
|
|
return; // let's not mess with bench testing
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0;i < calc.currentCalculationLogPosition;i++) {
|
|
|
|
scheduleMsg(logger, "calc %d: action %s value %f", i, action2String(calc.calcLogAction[i]), calc.calcLogValue[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
scheduleMsg(logger, "setPin %s %s", msg, value ? "on" : "off");
|
|
|
|
pin->setValue(value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-01-12 07:04:06 -08:00
|
|
|
static void setFsioFrequency(int index, int frequency) {
|
|
|
|
index--;
|
|
|
|
if (index < 0 || index >= LE_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;
|
|
|
|
}
|
|
|
|
boardConfiguration->fsioFrequency[index] = frequency;
|
|
|
|
scheduleMsg(logger, "Setting FSIO frequency %d on #%d", frequency, index + 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
void runFsio(void) {
|
|
|
|
for (int i = 0; i < LE_COMMAND_COUNT; i++) {
|
|
|
|
handleFsio(engine, i);
|
|
|
|
}
|
|
|
|
|
|
|
|
#if EFI_FUEL_PUMP || defined(__DOXYGEN__)
|
|
|
|
if (boardConfiguration->fuelPumpPin != GPIO_UNASSIGNED && engineConfiguration->isFuelPumpEnabled) {
|
|
|
|
setPinState("pump", &enginePins.fuelPumpRelay, fuelPumpLogic, engine);
|
|
|
|
}
|
|
|
|
#endif /* EFI_FUEL_PUMP */
|
|
|
|
|
|
|
|
/**
|
|
|
|
* main relay is always on if ECU is on, that's a good enough initial implementation
|
|
|
|
*/
|
|
|
|
if (boardConfiguration->mainRelayPin != GPIO_UNASSIGNED)
|
|
|
|
enginePins.mainRelay.setValue(true);
|
|
|
|
|
|
|
|
enginePins.o2heater.setValue(engine->rpmCalculator.isRunning());
|
|
|
|
|
|
|
|
if (boardConfiguration->acRelayPin != GPIO_UNASSIGNED) {
|
|
|
|
setPinState("A/C", &enginePins.acRelay, acRelayLogic, engine);
|
|
|
|
}
|
|
|
|
|
|
|
|
// if (boardConfiguration->alternatorControlPin != GPIO_UNASSIGNED) {
|
|
|
|
// setPinState("alternator", &enginePins.alternatorField, alternatorLogic, engine);
|
|
|
|
// }
|
|
|
|
|
|
|
|
if (boardConfiguration->fanPin != GPIO_UNASSIGNED) {
|
|
|
|
setPinState("fan", &enginePins.fanRelay, radiatorFanLogic, engine);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
static pin_output_mode_e defa = OM_DEFAULT;
|
|
|
|
|
|
|
|
#endif /* EFI_PROD_CODE */
|
|
|
|
|
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);
|
|
|
|
while (element != NULL) {
|
|
|
|
scheduleMsg(logger, "action %d: fValue=%f iValue=%d", element->action, element->fValue, element->iValue);
|
|
|
|
element = element->next;
|
|
|
|
}
|
|
|
|
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);
|
|
|
|
|
|
|
|
for (int i = 0; i < LE_COMMAND_COUNT; i++) {
|
|
|
|
char * exp = config->le_formulas[i];
|
|
|
|
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
|
|
|
|
*/
|
2016-07-07 20:01:43 -07:00
|
|
|
scheduleMsg(logger, "FSIO #%d [%s] at %s@%dHz value=%f", (i + 1), exp,
|
2015-07-10 06:01:56 -07:00
|
|
|
hwPortname(boardConfiguration->fsioPins[i]), boardConfiguration->fsioFrequency[i],
|
|
|
|
engineConfiguration2->fsioLastValue[i]);
|
|
|
|
// scheduleMsg(logger, "user-defined #%d value=%f", i, engine->engineConfiguration2->fsioLastValue[i]);
|
|
|
|
showFsio(NULL, fsioLogics[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (int i = 0; i < LE_COMMAND_COUNT; i++) {
|
|
|
|
float v = boardConfiguration->fsio_setting[i];
|
|
|
|
if (!cisnan(v)) {
|
|
|
|
scheduleMsg(logger, "user property #%d: %f", i + 1, v);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (int i = 0; i < LE_COMMAND_COUNT; i++) {
|
|
|
|
brain_pin_e inputPin = boardConfiguration->fsioDigitalInputs[i];
|
|
|
|
if (inputPin != GPIO_UNASSIGNED) {
|
|
|
|
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
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* set_fsio_setting 0 0.11
|
|
|
|
*/
|
|
|
|
static void setFsioSetting(float indexF, float value) {
|
2016-01-12 09:01:43 -08:00
|
|
|
#if EFI_PROD_CODE || EFI_SIMULATOR
|
2015-07-10 06:01:56 -07:00
|
|
|
int index = indexF;
|
|
|
|
if (index < 0 || index >= LE_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;
|
|
|
|
}
|
|
|
|
engineConfiguration->bc.fsio_setting[index] = value;
|
|
|
|
showFsioInfo();
|
2016-01-12 09:01:43 -08:00
|
|
|
#endif
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void setFsioExpression(const char *indexStr, const char *quotedLine, Engine *engine) {
|
2016-01-12 09:01:43 -08:00
|
|
|
#if EFI_PROD_CODE || EFI_SIMULATOR
|
2015-07-10 06:01:56 -07:00
|
|
|
int index = atoi(indexStr) - 1;
|
|
|
|
if (index < 0 || index >= LE_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);
|
|
|
|
strcpy(engine->config->le_formulas[index], l);
|
|
|
|
// this would apply the changes
|
|
|
|
applyFsioConfiguration(PASS_ENGINE_PARAMETER_F);
|
|
|
|
showFsioInfo();
|
2016-01-12 09:01:43 -08:00
|
|
|
#endif
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void eval(char *line, Engine *engine) {
|
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 {
|
2016-03-02 19:02:37 -08:00
|
|
|
float result = evalCalc.getValue2(0, e, engine);
|
2015-07-10 06:01:56 -07:00
|
|
|
scheduleMsg(logger, "Eval result: %f", result);
|
|
|
|
}
|
2016-01-12 09:01:43 -08:00
|
|
|
#endif
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
|
2016-01-12 07:04:06 -08:00
|
|
|
void initFsioImpl(Logging *sharedLogger DECLARE_ENGINE_PARAMETER_S) {
|
2016-01-12 09:01:43 -08:00
|
|
|
#if EFI_PROD_CODE || EFI_SIMULATOR
|
2015-07-10 06:01:56 -07:00
|
|
|
logger = sharedLogger;
|
2016-01-12 09:01:43 -08:00
|
|
|
#endif
|
2015-07-10 06:01:56 -07:00
|
|
|
for (int i = 0; i < LE_COMMAND_COUNT; i++) {
|
|
|
|
fsioLogics[i] = NULL;
|
|
|
|
}
|
|
|
|
|
2016-01-12 07:04:06 -08:00
|
|
|
#if EFI_FUEL_PUMP || defined(__DOXYGEN__)
|
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);
|
|
|
|
|
|
|
|
#if EFI_PROD_CODE || defined(__DOXYGEN__)
|
|
|
|
for (int i = 0; i < LE_COMMAND_COUNT; i++) {
|
|
|
|
brain_pin_e brainPin = boardConfiguration->fsioPins[i];
|
|
|
|
|
|
|
|
if (brainPin != GPIO_UNASSIGNED) {
|
|
|
|
int frequency = boardConfiguration->fsioFrequency[i];
|
|
|
|
if (frequency == 0) {
|
2016-09-14 16:03:00 -07:00
|
|
|
outputPinRegisterExt2(getGpioPinName(i), &enginePins.fsioOutputs[i], boardConfiguration->fsioPins[i], &defa);
|
2015-07-10 06:01:56 -07:00
|
|
|
} else {
|
2016-09-14 16:03:00 -07:00
|
|
|
startSimplePwmExt(&fsioPwm[i], "FSIOpwm", brainPin, &enginePins.fsioOutputs[i], frequency, 0.5f, applyPinState);
|
2015-07-10 06:01:56 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; i < LE_COMMAND_COUNT; i++) {
|
|
|
|
brain_pin_e inputPin = boardConfiguration->fsioDigitalInputs[i];
|
|
|
|
|
|
|
|
if (inputPin != GPIO_UNASSIGNED) {
|
|
|
|
mySetPadMode2("FSIO input", inputPin, getInputMode(engineConfiguration->fsioInputModes[i]));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-01-12 07:04:06 -08:00
|
|
|
addConsoleActionSS("set_fsio_output_pin", (VoidCharPtrCharPtr) setFsioOutputPin);
|
|
|
|
addConsoleActionII("set_fsio_output_frequency", (VoidIntInt) setFsioFrequency);
|
|
|
|
addConsoleActionSS("set_fsio_input_pin", (VoidCharPtrCharPtr) setFsioInputPin);
|
|
|
|
|
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
|
2015-07-10 06:01:56 -07:00
|
|
|
addConsoleActionSSP("set_fsio_expression", (VoidCharPtrCharPtrVoidPtr) setFsioExpression, engine);
|
|
|
|
addConsoleActionFF("set_fsio_setting", setFsioSetting);
|
|
|
|
addConsoleAction("fsioinfo", showFsioInfo);
|
|
|
|
addConsoleActionSP("eval", (VoidCharPtrVoidPtr) eval, engine);
|
2016-01-12 09:01:43 -08:00
|
|
|
#endif
|
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
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#endif /* EFI_FSIO */
|