rusefi/firmware/controllers/system/efi_gpio.cpp

576 lines
18 KiB
C++
Raw Normal View History

2015-07-10 06:01:56 -07:00
/**
2019-03-29 06:11:13 -07:00
* @file efi_gpio.cpp
2017-04-21 10:36:51 -07:00
* @brief EFI-related GPIO code
2015-07-10 06:01:56 -07:00
*
* @date Sep 26, 2014
2020-01-13 18:57:43 -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"
2017-04-21 12:14:37 -07:00
#include "engine.h"
2019-03-29 06:11:13 -07:00
#include "efi_gpio.h"
#include "drivers/gpio/gpio_ext.h"
2019-10-11 17:43:21 -07:00
#include "perf_trace.h"
2015-07-10 06:01:56 -07:00
2019-04-12 19:07:03 -07:00
#if EFI_GPIO_HARDWARE
2018-12-18 20:50:29 -08:00
#include "pin_repository.h"
2017-04-21 12:14:37 -07:00
#include "io_pins.h"
#endif /* EFI_GPIO_HARDWARE */
2015-07-10 06:01:56 -07:00
#if EFI_ELECTRONIC_THROTTLE_BODY
#include "electronic_throttle.h"
#endif /* EFI_ELECTRONIC_THROTTLE_BODY */
2017-04-21 12:14:37 -07:00
EXTERN_ENGINE;
2015-07-10 06:01:56 -07:00
2019-04-12 19:07:03 -07:00
#if EFI_ENGINE_SNIFFER
2017-04-21 16:23:20 -07:00
#include "engine_sniffer.h"
extern WaveChart waveChart;
#endif /* EFI_ENGINE_SNIFFER */
2017-04-21 13:36:50 -07:00
// todo: clean this mess, this should become 'static'/private
EnginePins enginePins;
static Logging* logger;
2017-04-21 13:36:50 -07:00
static pin_output_mode_e DEFAULT_OUTPUT = OM_DEFAULT;
2020-09-09 15:18:59 -07:00
pin_output_mode_e INVERTED_OUTPUT = OM_INVERTED;
2016-09-03 21:03:27 -07:00
static const char *sparkNames[] = { "Coil 1", "Coil 2", "Coil 3", "Coil 4", "Coil 5", "Coil 6", "Coil 7", "Coil 8",
"Coil 9", "Coil 10", "Coil 11", "Coil 12"};
// these short names are part of engine sniffer protocol
static const char *sparkShortNames[] = { PROTOCOL_COIL1_SHORT_NAME, "c2", "c3", "c4", "c5", "c6", "c7", "c8",
2016-09-27 08:01:57 -07:00
"c9", "cA", "cB", "cD"};
static const char *injectorNames[] = { "Injector 1", "Injector 2", "Injector 3", "Injector 4", "Injector 5", "Injector 6",
"Injector 7", "Injector 8", "Injector 9", "Injector 10", "Injector 11", "Injector 12"};
static const char *injectorShortNames[] = { PROTOCOL_INJ1_SHORT_NAME, "i2", "i3", "i4", "i5", "i6", "i7", "i8",
2016-09-27 08:01:57 -07:00
"j9", "iA", "iB", "iC"};
2019-06-10 12:11:53 -07:00
static const char *auxValveShortNames[] = { "a1", "a2"};
2017-11-26 19:30:37 -08:00
static RegisteredOutputPin * registeredOutputHead = nullptr;
RegisteredOutputPin::RegisteredOutputPin(const char *name, short pinOffset,
short pinModeOffset) {
this->name = name;
this->pinOffset = pinOffset;
this->pinModeOffset = pinModeOffset;
// adding into head of the list is so easy and since we do not care about order that's what we shall do
this->next = registeredOutputHead;
registeredOutputHead = this;
}
void RegisteredOutputPin::unregister() {
#if EFI_PROD_CODE
brain_pin_e curPin = *(brain_pin_e *) ((void *) (&((char*)&activeConfiguration)[pinOffset]));
brain_pin_e newPin = *(brain_pin_e *) ((void *) (&((char*) engineConfiguration)[pinOffset]));
pin_output_mode_e curMode = *(pin_output_mode_e *) ((void *) (&((char*)&activeConfiguration)[pinModeOffset]));
pin_output_mode_e newMode = *(pin_output_mode_e *) ((void *) (&((char*) engineConfiguration)[pinModeOffset]));
if (curPin != newPin || curMode != newMode) {
unregisterOutput(curPin);
}
#endif // EFI_PROD_CODE
}
#define CONFIG_OFFSET(x) x##_offset
// todo: pin and pinMode should be combined into a composite entity
// todo: one of the impediments is code generator hints handling (we need custom hints and those are not handled nice for fields of structs?)
#define CONFIG_PIN_OFFSETS(x) CONFIG_OFFSET(x##Pin), CONFIG_OFFSET(x##PinMode)
EnginePins::EnginePins() :
mainRelay("mainRelay", CONFIG_OFFSET(mainRelayPin), CONFIG_OFFSET(mainRelayPinMode)),
starterControl("starterControl", CONFIG_PIN_OFFSETS(starterControl)),
starterRelayDisable("starterRelayDisable", CONFIG_PIN_OFFSETS(starterRelayDisable)),
fanRelay("fanRelay", CONFIG_PIN_OFFSETS(fan)),
acRelay("acRelay", CONFIG_PIN_OFFSETS(acRelay)),
fuelPumpRelay("fuelPump", CONFIG_PIN_OFFSETS(fuelPump)),
boostPin("boostPin", CONFIG_PIN_OFFSETS(boostControl)),
idleSolenoidPin("idleSolenoid", idle_solenoidPin_offset, idle_solenoidPinMode_offset),
secondIdleSolenoidPin("secondIdleSolenoid", CONFIG_OFFSET(secondSolenoidPin), idle_solenoidPinMode_offset),
alternatorPin("alternatorPin", CONFIG_PIN_OFFSETS(alternatorControl)),
checkEnginePin("checkEnginePin", CONFIG_PIN_OFFSETS(malfunctionIndicator)),
// todo: NamedOutputPin vs RegisteredOutputPin
// tachOut("tachOut", CONFIG_PIN_OFFSETS(tachOutput)),
triggerDecoderErrorPin("triggerDecoderErrorPin", CONFIG_PIN_OFFSETS(triggerError)),
hipCs("hipCs", CONFIG_PIN_OFFSETS(hip9011Cs))
{
tachOut.name = PROTOCOL_TACH_NAME;
2016-09-27 08:01:57 -07:00
static_assert(efi::size(sparkNames) >= IGNITION_PIN_COUNT, "Too many ignition pins");
2016-09-27 08:01:57 -07:00
for (int i = 0; i < IGNITION_PIN_COUNT;i++) {
enginePins.coils[i].name = sparkNames[i];
enginePins.coils[i].shortName = sparkShortNames[i];
2016-09-27 08:01:57 -07:00
}
2019-11-13 05:42:16 -08:00
static_assert(efi::size(injectorNames) >= INJECTION_PIN_COUNT, "Too many injection pins");
for (int i = 0; i < INJECTION_PIN_COUNT;i++) {
2016-11-30 15:02:19 -08:00
enginePins.injectors[i].injectorIndex = i;
2016-09-27 08:01:57 -07:00
enginePins.injectors[i].name = injectorNames[i];
enginePins.injectors[i].shortName = injectorShortNames[i];
2016-09-27 08:01:57 -07:00
}
2019-11-13 05:42:16 -08:00
static_assert(efi::size(auxValveShortNames) >= AUX_DIGITAL_VALVE_COUNT, "Too many aux valve pins");
2017-11-26 19:30:37 -08:00
for (int i = 0; i < AUX_DIGITAL_VALVE_COUNT;i++) {
enginePins.auxValve[i].name = auxValveShortNames[i];
2017-11-26 19:30:37 -08:00
}
2016-09-06 21:02:11 -07:00
}
2017-04-21 13:52:02 -07:00
/**
* Sets the value of the pin. On this layer the value is assigned as is, without any conversion.
*/
2017-06-04 13:35:13 -07:00
#if EFI_PROD_CODE
2017-04-21 13:52:02 -07:00
#define setPinValue(outputPin, electricalValue, logicValue) \
{ \
if ((outputPin)->currentLogicValue != (logicValue)) { \
palWritePad((outputPin)->port, (outputPin)->pin, (electricalValue)); \
(outputPin)->currentLogicValue = (logicValue); \
} \
}
#define unregisterOutputIfPinChanged(output, pin) { \
if (isConfigurationChanged(pin)) { \
(output).unregisterOutput(activeConfiguration.pin); \
} \
}
#define unregisterOutputIfPinOrModeChanged(output, pin, mode) { \
if (isPinOrModeChanged(pin, mode)) { \
(output).unregisterOutput(activeConfiguration.pin); \
} \
}
2017-04-21 13:52:02 -07:00
#else /* EFI_PROD_CODE */
2017-04-21 13:52:02 -07:00
#define setPinValue(outputPin, electricalValue, logicValue) \
{ \
if ((outputPin)->currentLogicValue != (logicValue)) { \
(outputPin)->currentLogicValue = (logicValue); \
} \
}
#endif /* EFI_PROD_CODE */
2016-11-03 20:02:58 -07:00
bool EnginePins::stopPins() {
bool result = false;
for (int i = 0; i < IGNITION_PIN_COUNT; i++) {
result |= coils[i].stop();
}
for (int i = 0; i < INJECTION_PIN_COUNT; i++) {
result |= injectors[i].stop();
}
2017-11-26 19:30:37 -08:00
for (int i = 0; i < AUX_DIGITAL_VALVE_COUNT; i++) {
result |= auxValve[i].stop();
}
2016-11-03 20:02:58 -07:00
return result;
}
2017-06-04 13:35:13 -07:00
void EnginePins::unregisterPins() {
#if EFI_ELECTRONIC_THROTTLE_BODY
unregisterEtbPins();
#endif /* EFI_ELECTRONIC_THROTTLE_BODY */
2019-04-12 19:07:03 -07:00
#if EFI_PROD_CODE
unregisterOutputIfPinOrModeChanged(tachOut, tachOutputPin, tachOutputPinMode);
// todo: add pinMode
unregisterOutputIfPinChanged(sdCsPin, sdCardCsPin);
unregisterOutputIfPinChanged(accelerometerCs, LIS302DLCsPin);
2017-06-04 15:29:57 -07:00
2017-06-25 23:14:31 -07:00
for (int i = 0;i < FSIO_COMMAND_COUNT;i++) {
unregisterOutputIfPinChanged(fsioOutputs[i], fsioOutputPins[i]);
2017-06-04 15:29:57 -07:00
}
RegisteredOutputPin * pin = registeredOutputHead;
while (pin != nullptr) {
pin->unregister();
pin = pin->next;
}
2020-03-24 22:28:37 -07:00
2017-06-04 13:35:13 -07:00
#endif /* EFI_PROD_CODE */
}
void EnginePins::startPins() {
startInjectionPins();
startIgnitionPins();
startAuxValves();
}
2016-11-03 20:02:58 -07:00
void EnginePins::reset() {
2016-11-01 06:02:29 -07:00
for (int i = 0; i < INJECTION_PIN_COUNT;i++) {
injectors[i].reset();
}
for (int i = 0; i < IGNITION_PIN_COUNT;i++) {
coils[i].reset();
}
}
2017-06-04 15:53:43 -07:00
void EnginePins::stopIgnitionPins(void) {
2019-04-12 19:07:03 -07:00
#if EFI_PROD_CODE
2017-06-04 15:53:43 -07:00
for (int i = 0; i < IGNITION_PIN_COUNT; i++) {
unregisterOutputIfPinOrModeChanged(enginePins.coils[i], ignitionPins[i], ignitionPinMode);
2017-06-04 15:53:43 -07:00
}
#endif /* EFI_PROD_CODE */
}
void EnginePins::stopInjectionPins(void) {
2019-04-12 19:07:03 -07:00
#if EFI_PROD_CODE
2017-06-04 15:53:43 -07:00
for (int i = 0; i < INJECTION_PIN_COUNT; i++) {
unregisterOutputIfPinOrModeChanged(enginePins.injectors[i], injectionPins[i], injectionPinMode);
2017-06-04 15:53:43 -07:00
}
#endif /* EFI_PROD_CODE */
}
2017-11-26 19:30:37 -08:00
void EnginePins::startAuxValves(void) {
2019-04-12 19:07:03 -07:00
#if EFI_PROD_CODE
2017-11-26 19:30:37 -08:00
for (int i = 0; i < AUX_DIGITAL_VALVE_COUNT; i++) {
NamedOutputPin *output = &enginePins.auxValve[i];
output->initPin(output->name, engineConfiguration->auxValves[i]);
}
#endif /* EFI_PROD_CODE */
}
2017-06-04 15:53:43 -07:00
void EnginePins::startIgnitionPins(void) {
2019-04-12 19:07:03 -07:00
#if EFI_PROD_CODE
2017-06-04 15:53:43 -07:00
for (int i = 0; i < engineConfiguration->specs.cylindersCount; i++) {
NamedOutputPin *output = &enginePins.coils[i];
if (isPinOrModeChanged(ignitionPins[i], ignitionPinMode)) {
output->initPin(output->name, CONFIG(ignitionPins)[i], &CONFIG(ignitionPinMode));
2017-06-04 15:53:43 -07:00
}
}
#endif /* EFI_PROD_CODE */
}
void EnginePins::startInjectionPins(void) {
2019-04-12 19:07:03 -07:00
#if EFI_PROD_CODE
2017-06-04 15:53:43 -07:00
// todo: should we move this code closer to the injection logic?
for (int i = 0; i < engineConfiguration->specs.cylindersCount; i++) {
NamedOutputPin *output = &enginePins.injectors[i];
if (isPinOrModeChanged(injectionPins[i], injectionPinMode)) {
output->initPin(output->name, CONFIG(injectionPins)[i],
&CONFIG(injectionPinMode));
2017-06-04 15:53:43 -07:00
}
}
#endif /* EFI_PROD_CODE */
}
2017-04-21 12:14:37 -07:00
NamedOutputPin::NamedOutputPin() : OutputPin() {
name = NULL;
}
const char *NamedOutputPin::getName() const {
return name;
}
const char *NamedOutputPin::getShortName() const {
return shortName == NULL ? name : shortName;
}
2017-04-21 12:14:37 -07:00
NamedOutputPin::NamedOutputPin(const char *name) : OutputPin() {
this->name = name;
}
2017-04-21 16:23:20 -07:00
void NamedOutputPin::setHigh() {
2019-04-12 19:07:03 -07:00
#if EFI_DEFAILED_LOGGING
2017-04-21 16:23:20 -07:00
// signal->hi_time = hTimeNow();
#endif /* EFI_DEFAILED_LOGGING */
// turn the output level ACTIVE
setValue(true);
2019-04-12 19:07:03 -07:00
#if EFI_ENGINE_SNIFFER
addEngineSnifferEvent(getShortName(), PROTOCOL_ES_UP);
2017-04-21 16:23:20 -07:00
#endif /* EFI_ENGINE_SNIFFER */
}
void NamedOutputPin::setLow() {
// turn off the output
setValue(false);
2019-04-12 19:07:03 -07:00
#if EFI_ENGINE_SNIFFER
addEngineSnifferEvent(getShortName(), PROTOCOL_ES_DOWN);
2017-04-21 16:23:20 -07:00
#endif /* EFI_ENGINE_SNIFFER */
}
2017-04-21 12:14:37 -07:00
InjectorOutputPin::InjectorOutputPin() : NamedOutputPin() {
reset();
injectorIndex = -1;
}
2016-11-03 20:02:58 -07:00
bool NamedOutputPin::stop() {
2019-04-12 19:07:03 -07:00
#if EFI_GPIO_HARDWARE
2016-11-03 20:02:58 -07:00
if (isInitialized() && getLogicValue()) {
setValue(false);
scheduleMsg(logger, "turning off %s", name);
2016-11-03 20:02:58 -07:00
return true;
}
2017-04-21 13:36:50 -07:00
#endif /* EFI_GPIO_HARDWARE */
2016-11-03 20:02:58 -07:00
return false;
}
2016-09-03 21:03:27 -07:00
void InjectorOutputPin::reset() {
2020-07-16 23:55:41 -07:00
// If this injector was open, close it and reset state
if (overlappingCounter != 0) {
overlappingCounter = 0;
setValue(0);
}
2016-09-04 22:03:25 -07:00
// todo: this could be refactored by calling some super-reset method
currentLogicValue = INITIAL_PIN_STATE;
2016-09-03 21:03:27 -07:00
}
2016-10-31 19:02:12 -07:00
IgnitionOutputPin::IgnitionOutputPin() {
reset();
}
void IgnitionOutputPin::reset() {
2016-11-01 20:01:54 -07:00
outOfOrder = false;
signalFallSparkId = 0;
2016-10-31 19:02:12 -07:00
}
2015-07-10 06:01:56 -07:00
OutputPin::OutputPin() {
2017-04-21 15:11:36 -07:00
modePtr = &DEFAULT_OUTPUT;
2015-07-10 06:01:56 -07:00
}
2016-01-11 14:01:33 -08:00
bool OutputPin::isInitialized() {
2019-09-19 18:41:52 -07:00
#if EFI_GPIO_HARDWARE && EFI_PROD_CODE
2019-04-13 09:02:34 -07:00
#if (BOARD_EXT_GPIOCHIPS > 0)
if (ext)
return true;
#endif /* (BOARD_EXT_GPIOCHIPS > 0) */
2015-07-10 06:01:56 -07:00
return port != NULL;
2017-04-21 12:14:37 -07:00
#else /* EFI_GPIO_HARDWARE */
2017-07-10 18:43:03 -07:00
return true;
2017-04-21 12:14:37 -07:00
#endif /* EFI_GPIO_HARDWARE */
2015-07-10 06:01:56 -07:00
}
2018-01-28 08:27:33 -08:00
void OutputPin::toggle() {
2018-01-28 08:08:37 -08:00
setValue(!getLogicValue());
2020-03-30 22:06:19 -07:00
}
2018-01-28 08:08:37 -08:00
2020-03-30 22:06:19 -07:00
bool OutputPin::getAndSet(int logicValue) {
bool oldValue = currentLogicValue;
setValue(logicValue);
return oldValue;
2018-01-28 08:08:37 -08:00
}
2020-03-30 22:06:19 -07:00
2015-07-10 06:01:56 -07:00
void OutputPin::setValue(int logicValue) {
2020-07-25 16:04:15 -07:00
#if ENABLE_PERF_TRACE
// todo: https://github.com/rusefi/rusefi/issues/1638
// ScopePerf perf(PE::OutputPinSetValue);
2020-07-25 16:04:15 -07:00
#endif // ENABLE_PERF_TRACE
2019-10-13 13:14:08 -07:00
2017-04-21 13:52:02 -07:00
#if EFI_PROD_CODE
efiAssertVoid(CUSTOM_ERR_6621, modePtr!=NULL, "pin mode not initialized");
pin_output_mode_e mode = *modePtr;
efiAssertVoid(CUSTOM_ERR_6622, mode <= OM_OPENDRAIN_INVERTED, "invalid pin_output_mode_e");
int eValue = getElectricalValue(logicValue, mode);
#if (BOARD_EXT_GPIOCHIPS > 0)
if (!this->ext) {
/* onchip pin */
if (port != GPIO_NULL) {
setPinValue(this, eValue, logicValue);
}
} else {
/* external pin */
gpiochips_writePad(this->brainPin, logicValue);
/* TODO: check return value */
currentLogicValue = logicValue;
}
#else
if (port != GPIO_NULL) {
setPinValue(this, eValue, logicValue);
} else {
// even without physical pin sometimes it's nice to track logic pin value
currentLogicValue = logicValue;
}
#endif
2017-04-21 13:52:02 -07:00
#else /* EFI_PROD_CODE */
setPinValue(this, eValue, logicValue);
#endif /* EFI_PROD_CODE */
2015-07-10 06:01:56 -07:00
}
bool OutputPin::getLogicValue() const {
2015-07-10 06:01:56 -07:00
return currentLogicValue;
}
2019-01-16 05:24:37 -08:00
void OutputPin::setDefaultPinState(const pin_output_mode_e *outputMode) {
2015-07-10 06:01:56 -07:00
pin_output_mode_e mode = *outputMode;
/* may be*/UNUSED(mode);
2015-07-10 06:01:56 -07:00
assertOMode(mode);
this->modePtr = outputMode;
setValue(false); // initial state
}
2019-09-19 18:41:52 -07:00
void initOutputPins(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
2019-04-12 19:07:03 -07:00
#if EFI_GPIO_HARDWARE
2017-04-21 12:14:37 -07:00
/**
* want to make sure it's all zeros so that we can compare in initOutputPinExt() method
*/
// todo: it's too late to clear now? this breaks default status LEDs
// todo: fix this?
// memset(&outputs, 0, sizeof(outputs));
2019-04-12 19:07:03 -07:00
#if HAL_USE_SPI
enginePins.sdCsPin.initPin("SD CS", CONFIG(sdCardCsPin));
2017-04-21 12:14:37 -07:00
#endif /* HAL_USE_SPI */
// todo: should we move this code closer to the fuel pump logic?
enginePins.fuelPumpRelay.initPin("Fuel pump", CONFIG(fuelPumpPin), &CONFIG(fuelPumpPinMode));
2017-04-21 12:14:37 -07:00
enginePins.mainRelay.initPin("Main relay", CONFIG(mainRelayPin), &CONFIG(mainRelayPinMode));
enginePins.starterRelayDisable.initPin("Starter disable", CONFIG(starterRelayDisablePin), &CONFIG(starterRelayDisableMode));
enginePins.starterControl.initPin("Starter control", CONFIG(starterControlPin));
2017-04-21 12:14:37 -07:00
enginePins.fanRelay.initPin("Fan", CONFIG(fanPin), &CONFIG(fanPinMode));
enginePins.o2heater.initPin("O2 heater", CONFIG(o2heaterPin));
enginePins.acRelay.initPin("A/C relay", CONFIG(acRelayPin), &CONFIG(acRelayPinMode));
2017-04-21 12:14:37 -07:00
// digit 1
/*
ledRegister(LED_HUGE_0, GPIOB, 2);
ledRegister(LED_HUGE_1, GPIOE, 7);
ledRegister(LED_HUGE_2, GPIOE, 8);
ledRegister(LED_HUGE_3, GPIOE, 9);
ledRegister(LED_HUGE_4, GPIOE, 10);
ledRegister(LED_HUGE_5, GPIOE, 11);
ledRegister(LED_HUGE_6, GPIOE, 12);
// digit 2
ledRegister(LED_HUGE_7, GPIOE, 13);
ledRegister(LED_HUGE_8, GPIOE, 14);
ledRegister(LED_HUGE_9, GPIOE, 15);
ledRegister(LED_HUGE_10, GPIOB, 10);
ledRegister(LED_HUGE_11, GPIOB, 11);
ledRegister(LED_HUGE_12, GPIOB, 12);
ledRegister(LED_HUGE_13, GPIOB, 13);
// digit 3
ledRegister(LED_HUGE_14, GPIOE, 0);
ledRegister(LED_HUGE_15, GPIOE, 2);
ledRegister(LED_HUGE_16, GPIOE, 4);
ledRegister(LED_HUGE_17, GPIOE, 6);
ledRegister(LED_HUGE_18, GPIOE, 5);
ledRegister(LED_HUGE_19, GPIOE, 3);
ledRegister(LED_HUGE_20, GPIOE, 1);
*/
#endif /* EFI_GPIO_HARDWARE */
}
2017-04-21 15:11:36 -07:00
void OutputPin::initPin(const char *msg, brain_pin_e brainPin) {
initPin(msg, brainPin, &DEFAULT_OUTPUT);
2017-04-21 14:50:28 -07:00
}
2019-01-16 05:24:37 -08:00
void OutputPin::initPin(const char *msg, brain_pin_e brainPin, const pin_output_mode_e *outputMode) {
2019-09-19 18:41:52 -07:00
#if EFI_GPIO_HARDWARE && EFI_PROD_CODE
2017-04-21 14:14:14 -07:00
if (brainPin == GPIO_UNASSIGNED)
return;
2017-04-21 12:14:37 -07:00
2020-03-05 18:16:45 -08:00
if (*outputMode > OM_OPENDRAIN_INVERTED) {
firmwareError(CUSTOM_INVALID_MODE_SETTING, "%s invalid pin_output_mode_e", msg);
return;
}
2017-04-21 12:14:37 -07:00
iomode_t mode = (*outputMode == OM_DEFAULT || *outputMode == OM_INVERTED) ?
2017-04-21 14:14:14 -07:00
PAL_MODE_OUTPUT_PUSHPULL : PAL_MODE_OUTPUT_OPENDRAIN;
2017-04-21 12:14:37 -07:00
#if (BOARD_EXT_GPIOCHIPS > 0)
this->ext = false;
#endif
if (brain_pin_is_onchip(brainPin)) {
ioportid_t port = getHwPort(msg, brainPin);
int pin = getHwPin(msg, brainPin);
2017-04-21 14:26:50 -07:00
/**
* This method is used for digital GPIO pins only, for peripheral pins see mySetPadMode
2017-04-21 14:26:50 -07:00
*/
if (port == GPIO_NULL) {
// that's for GRIO_NONE
this->port = port;
return;
}
/**
* @brief Initialize the hardware output pin while also assigning it a logical name
*/
if (this->port != NULL && (this->port != port || this->pin != pin)) {
/**
* here we check if another physical pin is already assigned to this logical output
*/
// todo: need to clear '&outputs' in io_pins.c
warning(CUSTOM_OBD_PIN_CONFLICT, "outputPin [%s] already assigned to %x%d", msg, this->port, this->pin);
engine->withError = true;
return;
}
this->port = port;
this->pin = pin;
2017-04-21 14:26:50 -07:00
}
#if (BOARD_EXT_GPIOCHIPS > 0)
else {
this->ext = true;
this->brainPin = brainPin;
}
#endif
2017-04-21 15:11:36 -07:00
this->currentLogicValue = INITIAL_PIN_STATE;
2017-04-21 14:26:50 -07:00
2019-07-03 00:18:04 -07:00
// The order of the next two calls may look strange, which is a good observation.
// We call them in this order so that the pin is set to a known state BEFORE
// it's enabled. Enabling the pin then setting it could result in a (brief)
// mystery state being driven on the pin (potentially dangerous).
2017-04-21 15:11:36 -07:00
setDefaultPinState(outputMode);
2019-07-03 00:18:04 -07:00
efiSetPadMode(msg, brainPin, mode);
2017-04-21 14:38:13 -07:00
#endif /* EFI_GPIO_HARDWARE */
}
void OutputPin::unregisterOutput(brain_pin_e oldPin) {
if (oldPin != GPIO_UNASSIGNED) {
scheduleMsg(logger, "unregistering %s", hwPortname(oldPin));
#if EFI_GPIO_HARDWARE && EFI_PROD_CODE
brain_pin_markUnused(oldPin);
port = nullptr;
#endif /* EFI_GPIO_HARDWARE */
}
}
2019-04-12 19:07:03 -07:00
#if EFI_GPIO_HARDWARE
2017-04-21 14:38:13 -07:00
// questionable trick: we avoid using 'getHwPort' and 'getHwPin' in case of errors in order to increase the changes of turning the LED
2019-09-19 18:41:52 -07:00
// by reducing stack requirement
ioportid_t criticalErrorLedPort;
ioportmask_t criticalErrorLedPin;
uint8_t criticalErrorLedState;
2020-09-09 15:18:59 -07:00
#ifndef LED_ERROR_BRAIN_PIN_MODE
#define LED_ERROR_BRAIN_PIN_MODE DEFAULT_OUTPUT
#endif /* LED_ERROR_BRAIN_PIN_MODE */
void initPrimaryPins(Logging *sharedLogger) {
logger = sharedLogger;
2019-09-19 18:41:52 -07:00
#if EFI_PROD_CODE
enginePins.errorLedPin.initPin("led: CRITICAL status", LED_CRITICAL_ERROR_BRAIN_PIN, &(LED_ERROR_BRAIN_PIN_MODE));
criticalErrorLedPort = getHwPort("CRITICAL", LED_CRITICAL_ERROR_BRAIN_PIN);
criticalErrorLedPin = getHwPin("CRITICAL", LED_CRITICAL_ERROR_BRAIN_PIN);
criticalErrorLedState = (LED_ERROR_BRAIN_PIN_MODE == INVERTED_OUTPUT) ? 0 : 1;
2019-09-19 18:41:52 -07:00
#endif /* EFI_PROD_CODE */
2017-04-21 12:14:37 -07:00
}
2017-04-21 12:28:47 -07:00
/**
* This method is part of fatal error handling.
* Please note that worst case scenario the pins might get re-enabled by some other code :(
* The whole method is pretty naive, but that's at least something.
*/
void turnAllPinsOff(void) {
for (int i = 0; i < INJECTION_PIN_COUNT; i++) {
enginePins.injectors[i].setValue(false);
}
for (int i = 0; i < IGNITION_PIN_COUNT; i++) {
enginePins.coils[i].setValue(false);
}
}
2017-04-21 12:14:37 -07:00
#endif /* EFI_GPIO_HARDWARE */