610 lines
19 KiB
C++
610 lines
19 KiB
C++
/**
|
|
* @file efi_gpio.cpp
|
|
* @brief EFI-related GPIO code
|
|
*
|
|
* @date Sep 26, 2014
|
|
* @author Andrey Belomutskiy, (c) 2012-2020
|
|
*/
|
|
|
|
#include "global.h"
|
|
#include "engine.h"
|
|
#include "efi_gpio.h"
|
|
#include "drivers/gpio/gpio_ext.h"
|
|
#include "perf_trace.h"
|
|
#include "engine_controller.h"
|
|
|
|
#if EFI_GPIO_HARDWARE
|
|
#include "pin_repository.h"
|
|
#include "io_pins.h"
|
|
#endif /* EFI_GPIO_HARDWARE */
|
|
|
|
#if EFI_ELECTRONIC_THROTTLE_BODY
|
|
#include "electronic_throttle.h"
|
|
#endif /* EFI_ELECTRONIC_THROTTLE_BODY */
|
|
|
|
EXTERN_ENGINE;
|
|
|
|
#if EFI_ENGINE_SNIFFER
|
|
#include "engine_sniffer.h"
|
|
extern WaveChart waveChart;
|
|
#endif /* EFI_ENGINE_SNIFFER */
|
|
|
|
// todo: clean this mess, this should become 'static'/private
|
|
EnginePins enginePins;
|
|
static Logging* logger;
|
|
|
|
static pin_output_mode_e DEFAULT_OUTPUT = OM_DEFAULT;
|
|
pin_output_mode_e INVERTED_OUTPUT = OM_INVERTED;
|
|
|
|
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",
|
|
"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",
|
|
"j9", "iA", "iB", "iC"};
|
|
|
|
static const char *auxValveShortNames[] = { "a1", "a2"};
|
|
|
|
static RegisteredOutputPin * registeredOutputHead = nullptr;
|
|
|
|
RegisteredNamedOutputPin::RegisteredNamedOutputPin(const char *name, short pinOffset,
|
|
short pinModeOffset) : RegisteredOutputPin(name, pinOffset, pinModeOffset) {
|
|
}
|
|
|
|
RegisteredOutputPin::RegisteredOutputPin(const char *registrationName, short pinOffset,
|
|
short pinModeOffset) {
|
|
this->registrationName = registrationName;
|
|
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;
|
|
}
|
|
|
|
bool RegisteredOutputPin::isPinConfigurationChanged() {
|
|
#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]));
|
|
return curPin != newPin || curMode != newMode;
|
|
#else
|
|
return true;
|
|
#endif // EFI_PROD_CODE
|
|
}
|
|
|
|
void RegisteredOutputPin::init(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
|
|
brain_pin_e newPin = *(brain_pin_e *) ((void *) (&((char*) engineConfiguration)[pinOffset]));
|
|
pin_output_mode_e *newMode = (pin_output_mode_e *) ((void *) (&((char*) engineConfiguration)[pinModeOffset]));
|
|
|
|
if (isPinConfigurationChanged()) {
|
|
this->initPin(registrationName, newPin, newMode);
|
|
}
|
|
}
|
|
|
|
void RegisteredOutputPin::unregister() {
|
|
#if EFI_PROD_CODE
|
|
brain_pin_e curPin = *(brain_pin_e *) ((void *) (&((char*)&activeConfiguration)[pinOffset]));
|
|
if (isPinConfigurationChanged()) {
|
|
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("Main relay", CONFIG_PIN_OFFSETS(mainRelay)),
|
|
hpfpValve("hpfpValve", CONFIG_PIN_OFFSETS(hpfpValve)),
|
|
starterControl("Starter control", CONFIG_PIN_OFFSETS(starterControl)),
|
|
starterRelayDisable("Starter disable", CONFIG_PIN_OFFSETS(starterRelayDisable)),
|
|
fanRelay("Fan", CONFIG_PIN_OFFSETS(fan)),
|
|
acRelay("A/C relay", CONFIG_PIN_OFFSETS(acRelay)),
|
|
fuelPumpRelay("Fuel pump", CONFIG_PIN_OFFSETS(fuelPump)),
|
|
boostPin("Boost", CONFIG_PIN_OFFSETS(boostControl)),
|
|
idleSolenoidPin("Idle Valve", idle_solenoidPin_offset, idle_solenoidPinMode_offset),
|
|
secondIdleSolenoidPin("Idle Valve#2", CONFIG_OFFSET(secondSolenoidPin), idle_solenoidPinMode_offset),
|
|
alternatorPin("Alternator control", CONFIG_PIN_OFFSETS(alternatorControl)),
|
|
checkEnginePin("checkEnginePin", CONFIG_PIN_OFFSETS(malfunctionIndicator)),
|
|
tachOut("tachOut", CONFIG_PIN_OFFSETS(tachOutput)),
|
|
triggerDecoderErrorPin("led: trigger debug", CONFIG_PIN_OFFSETS(triggerError)),
|
|
hipCs("hipCs", CONFIG_PIN_OFFSETS(hip9011Cs))
|
|
{
|
|
tachOut.name = PROTOCOL_TACH_NAME;
|
|
hpfpValve.name = PROTOCOL_HPFP_NAME;
|
|
|
|
static_assert(efi::size(sparkNames) >= IGNITION_PIN_COUNT, "Too many ignition pins");
|
|
for (int i = 0; i < IGNITION_PIN_COUNT;i++) {
|
|
enginePins.coils[i].name = sparkNames[i];
|
|
enginePins.coils[i].shortName = sparkShortNames[i];
|
|
}
|
|
|
|
static_assert(efi::size(injectorNames) >= INJECTION_PIN_COUNT, "Too many injection pins");
|
|
for (int i = 0; i < INJECTION_PIN_COUNT;i++) {
|
|
enginePins.injectors[i].injectorIndex = i;
|
|
enginePins.injectors[i].name = injectorNames[i];
|
|
enginePins.injectors[i].shortName = injectorShortNames[i];
|
|
}
|
|
|
|
static_assert(efi::size(auxValveShortNames) >= AUX_DIGITAL_VALVE_COUNT, "Too many aux valve pins");
|
|
for (int i = 0; i < AUX_DIGITAL_VALVE_COUNT;i++) {
|
|
enginePins.auxValve[i].name = auxValveShortNames[i];
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Sets the value of the pin. On this layer the value is assigned as is, without any conversion.
|
|
*/
|
|
|
|
#if EFI_PROD_CODE
|
|
#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); \
|
|
} \
|
|
}
|
|
|
|
#else /* EFI_PROD_CODE */
|
|
|
|
#define setPinValue(outputPin, electricalValue, logicValue) \
|
|
{ \
|
|
if ((outputPin)->currentLogicValue != (logicValue)) { \
|
|
(outputPin)->currentLogicValue = (logicValue); \
|
|
} \
|
|
setMockState((outputPin)->brainPin, logicValue); \
|
|
}
|
|
#endif /* EFI_PROD_CODE */
|
|
|
|
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();
|
|
}
|
|
for (int i = 0; i < AUX_DIGITAL_VALVE_COUNT; i++) {
|
|
result |= auxValve[i].stop();
|
|
}
|
|
return result;
|
|
}
|
|
|
|
void EnginePins::unregisterPins() {
|
|
stopInjectionPins();
|
|
stopIgnitionPins();
|
|
|
|
#if EFI_ELECTRONIC_THROTTLE_BODY
|
|
unregisterEtbPins();
|
|
#endif /* EFI_ELECTRONIC_THROTTLE_BODY */
|
|
#if EFI_PROD_CODE
|
|
// todo: add pinMode
|
|
unregisterOutputIfPinChanged(sdCsPin, sdCardCsPin);
|
|
unregisterOutputIfPinChanged(accelerometerCs, LIS302DLCsPin);
|
|
|
|
for (int i = 0;i < FSIO_COMMAND_COUNT;i++) {
|
|
unregisterOutputIfPinChanged(fsioOutputs[i], fsioOutputPins[i]);
|
|
}
|
|
|
|
|
|
RegisteredOutputPin * pin = registeredOutputHead;
|
|
while (pin != nullptr) {
|
|
pin->unregister();
|
|
pin = pin->next;
|
|
}
|
|
|
|
#endif /* EFI_PROD_CODE */
|
|
}
|
|
|
|
void EnginePins::debug() {
|
|
#if EFI_PROD_CODE
|
|
RegisteredOutputPin * pin = registeredOutputHead;
|
|
while (pin != nullptr) {
|
|
scheduleMsg(logger, "%s %d", pin->registrationName, pin->currentLogicValue);
|
|
pin = pin->next;
|
|
}
|
|
#endif // EFI_PROD_CODE
|
|
}
|
|
|
|
void EnginePins::startPins(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
|
|
#if EFI_ENGINE_CONTROL
|
|
startInjectionPins();
|
|
startIgnitionPins();
|
|
startAuxValves();
|
|
#endif /* EFI_ENGINE_CONTROL */
|
|
|
|
RegisteredOutputPin * pin = registeredOutputHead;
|
|
while (pin != nullptr) {
|
|
pin->init(PASS_ENGINE_PARAMETER_SIGNATURE);
|
|
pin = pin->next;
|
|
}
|
|
}
|
|
|
|
void EnginePins::reset() {
|
|
for (int i = 0; i < INJECTION_PIN_COUNT;i++) {
|
|
injectors[i].reset();
|
|
}
|
|
for (int i = 0; i < IGNITION_PIN_COUNT;i++) {
|
|
coils[i].reset();
|
|
}
|
|
}
|
|
|
|
void EnginePins::stopIgnitionPins(void) {
|
|
#if EFI_PROD_CODE
|
|
for (int i = 0; i < IGNITION_PIN_COUNT; i++) {
|
|
unregisterOutputIfPinOrModeChanged(enginePins.coils[i], ignitionPins[i], ignitionPinMode);
|
|
}
|
|
#endif /* EFI_PROD_CODE */
|
|
}
|
|
|
|
void EnginePins::stopInjectionPins(void) {
|
|
#if EFI_PROD_CODE
|
|
for (int i = 0; i < INJECTION_PIN_COUNT; i++) {
|
|
unregisterOutputIfPinOrModeChanged(enginePins.injectors[i], injectionPins[i], injectionPinMode);
|
|
}
|
|
#endif /* EFI_PROD_CODE */
|
|
}
|
|
|
|
void EnginePins::startAuxValves(void) {
|
|
#if EFI_PROD_CODE
|
|
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 */
|
|
}
|
|
|
|
void EnginePins::startIgnitionPins(void) {
|
|
#if EFI_PROD_CODE
|
|
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));
|
|
}
|
|
}
|
|
#endif /* EFI_PROD_CODE */
|
|
}
|
|
|
|
void EnginePins::startInjectionPins(void) {
|
|
#if EFI_PROD_CODE
|
|
// 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));
|
|
}
|
|
}
|
|
#endif /* EFI_PROD_CODE */
|
|
}
|
|
|
|
NamedOutputPin::NamedOutputPin() : OutputPin() {
|
|
}
|
|
|
|
NamedOutputPin::NamedOutputPin(const char *name) : OutputPin() {
|
|
this->name = name;
|
|
}
|
|
|
|
const char *NamedOutputPin::getName() const {
|
|
return name;
|
|
}
|
|
|
|
const char *NamedOutputPin::getShortName() const {
|
|
return shortName == nullptr ? name : shortName;
|
|
}
|
|
|
|
void NamedOutputPin::setHigh() {
|
|
#if EFI_DEFAILED_LOGGING
|
|
// signal->hi_time = hTimeNow();
|
|
#endif /* EFI_DEFAILED_LOGGING */
|
|
|
|
// turn the output level ACTIVE
|
|
setValue(true);
|
|
|
|
#if EFI_ENGINE_SNIFFER
|
|
addEngineSnifferEvent(getShortName(), PROTOCOL_ES_UP);
|
|
#endif /* EFI_ENGINE_SNIFFER */
|
|
}
|
|
|
|
void NamedOutputPin::setLow() {
|
|
// turn off the output
|
|
setValue(false);
|
|
|
|
#if EFI_ENGINE_SNIFFER
|
|
addEngineSnifferEvent(getShortName(), PROTOCOL_ES_DOWN);
|
|
#endif /* EFI_ENGINE_SNIFFER */
|
|
}
|
|
|
|
InjectorOutputPin::InjectorOutputPin() : NamedOutputPin() {
|
|
reset();
|
|
injectorIndex = -1;
|
|
}
|
|
|
|
bool NamedOutputPin::stop() {
|
|
#if EFI_GPIO_HARDWARE
|
|
if (isInitialized() && getLogicValue()) {
|
|
setValue(false);
|
|
scheduleMsg(logger, "turning off %s", name);
|
|
return true;
|
|
}
|
|
#endif /* EFI_GPIO_HARDWARE */
|
|
return false;
|
|
}
|
|
|
|
void InjectorOutputPin::reset() {
|
|
// If this injector was open, close it and reset state
|
|
if (overlappingCounter != 0) {
|
|
overlappingCounter = 0;
|
|
setValue(0);
|
|
}
|
|
|
|
// todo: this could be refactored by calling some super-reset method
|
|
currentLogicValue = 0;
|
|
}
|
|
|
|
IgnitionOutputPin::IgnitionOutputPin() {
|
|
reset();
|
|
}
|
|
|
|
void IgnitionOutputPin::reset() {
|
|
outOfOrder = false;
|
|
signalFallSparkId = 0;
|
|
}
|
|
|
|
OutputPin::OutputPin() {
|
|
modePtr = &DEFAULT_OUTPUT;
|
|
}
|
|
|
|
bool OutputPin::isInitialized() {
|
|
#if EFI_GPIO_HARDWARE && EFI_PROD_CODE
|
|
#if (BOARD_EXT_GPIOCHIPS > 0)
|
|
if (ext)
|
|
return true;
|
|
#endif /* (BOARD_EXT_GPIOCHIPS > 0) */
|
|
return port != NULL;
|
|
#else /* EFI_GPIO_HARDWARE */
|
|
return true;
|
|
#endif /* EFI_GPIO_HARDWARE */
|
|
}
|
|
|
|
void OutputPin::toggle() {
|
|
setValue(!getLogicValue());
|
|
}
|
|
|
|
bool OutputPin::getAndSet(int logicValue) {
|
|
bool oldValue = currentLogicValue;
|
|
setValue(logicValue);
|
|
return oldValue;
|
|
}
|
|
|
|
void OutputPin::setOnchipValue(int electricalValue, int logicValue) {
|
|
#if EFI_PROD_CODE
|
|
if (port != GPIO_NULL) {
|
|
setPinValue(this, electricalValue, logicValue);
|
|
} else {
|
|
// even without physical pin sometimes it's nice to track logic pin value
|
|
currentLogicValue = logicValue;
|
|
}
|
|
#endif // EFI_PROD_CODE
|
|
}
|
|
|
|
void OutputPin::setValue(int logicValue) {
|
|
#if ENABLE_PERF_TRACE
|
|
// todo: https://github.com/rusefi/rusefi/issues/1638
|
|
// ScopePerf perf(PE::OutputPinSetValue);
|
|
#endif // ENABLE_PERF_TRACE
|
|
|
|
#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 electricalValue = getElectricalValue(logicValue, mode);
|
|
|
|
#if (BOARD_EXT_GPIOCHIPS > 0)
|
|
if (!this->ext) {
|
|
setOnchipValue(electricalValue, logicValue);
|
|
} else {
|
|
/* external pin */
|
|
gpiochips_writePad(this->brainPin, logicValue);
|
|
/* TODO: check return value */
|
|
currentLogicValue = logicValue;
|
|
}
|
|
#else
|
|
setOnchipValue(electricalValue, logicValue);
|
|
#endif
|
|
|
|
#else /* EFI_PROD_CODE */
|
|
setPinValue(this, eValue, logicValue);
|
|
#endif /* EFI_PROD_CODE */
|
|
}
|
|
|
|
bool OutputPin::getLogicValue() const {
|
|
return currentLogicValue;
|
|
}
|
|
|
|
void OutputPin::setDefaultPinState(const pin_output_mode_e *outputMode) {
|
|
pin_output_mode_e mode = *outputMode;
|
|
/* may be*/UNUSED(mode);
|
|
assertOMode(mode);
|
|
this->modePtr = outputMode;
|
|
setValue(false); // initial state
|
|
}
|
|
|
|
void initOutputPins(DECLARE_ENGINE_PARAMETER_SIGNATURE) {
|
|
#if EFI_GPIO_HARDWARE
|
|
/**
|
|
* 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));
|
|
|
|
#if HAL_USE_SPI
|
|
enginePins.sdCsPin.initPin("SD CS", CONFIG(sdCardCsPin));
|
|
#endif /* HAL_USE_SPI */
|
|
|
|
#if EFI_SHAFT_POSITION_INPUT
|
|
// todo: migrate remaining OutputPin to RegisteredOutputPin in order to get consistent dynamic pin init/deinit
|
|
enginePins.debugTriggerSync.initPin("debug: sync", CONFIG(debugTriggerSync));
|
|
#endif // EFI_SHAFT_POSITION_INPUT
|
|
|
|
enginePins.o2heater.initPin("O2 heater", CONFIG(o2heaterPin));
|
|
|
|
#endif /* EFI_GPIO_HARDWARE */
|
|
}
|
|
|
|
void OutputPin::initPin(const char *msg, brain_pin_e brainPin) {
|
|
initPin(msg, brainPin, &DEFAULT_OUTPUT);
|
|
}
|
|
|
|
void OutputPin::initPin(const char *msg, brain_pin_e brainPin, const pin_output_mode_e *outputMode) {
|
|
#if EFI_UNIT_TEST
|
|
this->brainPin = brainPin;
|
|
#endif
|
|
|
|
#if EFI_GPIO_HARDWARE && EFI_PROD_CODE
|
|
if (brainPin == GPIO_UNASSIGNED)
|
|
return;
|
|
|
|
if (*outputMode > OM_OPENDRAIN_INVERTED) {
|
|
firmwareError(CUSTOM_INVALID_MODE_SETTING, "%s invalid pin_output_mode_e", msg);
|
|
return;
|
|
}
|
|
iomode_t mode = (*outputMode == OM_DEFAULT || *outputMode == OM_INVERTED) ?
|
|
PAL_MODE_OUTPUT_PUSHPULL : PAL_MODE_OUTPUT_OPENDRAIN;
|
|
|
|
#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);
|
|
|
|
/**
|
|
* This method is used for digital GPIO pins only, for peripheral pins see mySetPadMode
|
|
*/
|
|
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;
|
|
}
|
|
#if (BOARD_EXT_GPIOCHIPS > 0)
|
|
else {
|
|
this->ext = true;
|
|
this->brainPin = brainPin;
|
|
}
|
|
#endif
|
|
|
|
this->currentLogicValue = 0;
|
|
|
|
// 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).
|
|
setDefaultPinState(outputMode);
|
|
efiSetPadMode(msg, brainPin, mode);
|
|
if (brain_pin_is_onchip(brainPin)) {
|
|
int actualValue = palReadPad(port, pin);
|
|
// we had enough drama with pin configuration in board.h and else that we shall self-check
|
|
// todo: handle OM_OPENDRAIN and OM_OPENDRAIN_INVERTED as well
|
|
if (*outputMode == OM_DEFAULT || *outputMode == OM_INVERTED) {
|
|
if (*outputMode == OM_INVERTED) {
|
|
actualValue = !actualValue;
|
|
}
|
|
if (actualValue) {
|
|
firmwareError(OBD_PCM_Processor_Fault, "%s: startup pin state %s value=%d mode=%s", msg, hwPortname(brainPin), actualValue, getPin_output_mode_e(*outputMode));
|
|
}
|
|
}
|
|
}
|
|
|
|
#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
|
|
efiSetPadUnused(oldPin);
|
|
port = nullptr;
|
|
#endif /* EFI_GPIO_HARDWARE */
|
|
}
|
|
}
|
|
|
|
#if EFI_GPIO_HARDWARE
|
|
|
|
// questionable trick: we avoid using 'getHwPort' and 'getHwPin' in case of errors in order to increase the changes of turning the LED
|
|
// by reducing stack requirement
|
|
ioportid_t criticalErrorLedPort;
|
|
ioportmask_t criticalErrorLedPin;
|
|
uint8_t criticalErrorLedState;
|
|
|
|
#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;
|
|
#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;
|
|
|
|
addConsoleAction("gpio_pins", EnginePins::debug);
|
|
#endif /* EFI_PROD_CODE */
|
|
}
|
|
|
|
/**
|
|
* 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);
|
|
}
|
|
}
|
|
#endif /* EFI_GPIO_HARDWARE */
|