rusefi/firmware/hw_layer/io_pins.cpp

195 lines
6.4 KiB
C++
Raw Normal View History

2014-08-29 07:52:33 -07:00
/**
* @file io_pins.c
* @brief It could be that the main purpose of this file is the status LED blinking
*
* @date Jan 24, 2013
* @author Andrey Belomutskiy, (c) 2012-2014
*/
#include <board.h>
#include "main.h"
#include "io_pins.h"
2014-11-08 08:09:38 -08:00
#include "efiGpio.h"
2014-08-29 07:52:33 -07:00
#include "pin_repository.h"
#include "gpio_helper.h"
#include "status_loop.h"
#include "engine_configuration.h"
#include "console_io.h"
2014-12-23 22:03:26 -08:00
#if EFI_ENGINE_CONTROL || defined(__DOXYGEN__)
2014-08-29 07:52:33 -07:00
#include "main_trigger_callback.h"
#endif /* EFI_ENGINE_CONTROL */
extern board_configuration_s *boardConfiguration;
2014-09-25 22:02:43 -07:00
static Logging logger;
2014-09-26 06:02:50 -07:00
extern OutputPin outputs[IO_PIN_COUNT];
2014-08-29 07:52:33 -07:00
2014-12-23 19:06:10 -08:00
#if defined(STM32F4XX)
2014-08-29 07:52:33 -07:00
static GPIO_TypeDef *PORTS[] = { GPIOA, GPIOB, GPIOC, GPIOD, GPIOE, GPIOF, GPIOG, GPIOH };
2014-12-23 19:06:10 -08:00
#else
static GPIO_TypeDef *PORTS[] = { GPIOA, GPIOB, GPIOC, GPIOD, GPIOF};
#endif
2014-08-29 07:52:33 -07:00
2014-12-18 10:05:24 -08:00
pin_output_mode_e DEFAULT_OUTPUT = OM_DEFAULT;
2014-08-29 07:52:33 -07:00
2015-01-07 15:04:40 -08:00
static void outputPinRegisterExt(const char *msg, OutputPin *output, GPIO_TypeDef *port, uint32_t pin,
2014-08-29 07:52:33 -07:00
pin_output_mode_e *outputMode) {
2014-12-24 11:05:19 -08:00
#if EFI_GPIO
2014-09-25 22:02:43 -07:00
if (port == GPIO_NULL) {
2014-08-29 07:52:33 -07:00
// that's for GRIO_NONE
2015-01-07 15:04:40 -08:00
output->port = port;
2014-08-29 07:52:33 -07:00
return;
}
assertOMode(*outputMode);
2014-09-25 22:02:43 -07:00
iomode_t mode = (*outputMode == OM_DEFAULT || *outputMode == OM_INVERTED) ?
PAL_MODE_OUTPUT_PUSHPULL :
PAL_MODE_OUTPUT_OPENDRAIN;
2014-08-29 07:52:33 -07:00
2015-01-07 15:04:40 -08:00
initOutputPinExt(msg, output, port, pin, mode);
2014-08-29 07:52:33 -07:00
2015-01-07 15:04:40 -08:00
output->setDefaultPinState(outputMode);
2014-12-24 11:05:19 -08:00
#endif
2014-08-29 07:52:33 -07:00
}
GPIO_TypeDef * getHwPort(brain_pin_e brainPin) {
2014-11-10 07:03:20 -08:00
if (brainPin == GPIO_UNASSIGNED)
2014-09-25 22:02:43 -07:00
return GPIO_NULL;
2014-11-10 07:03:20 -08:00
if (brainPin > GPIO_UNASSIGNED || brainPin < 0) {
2014-08-29 07:52:33 -07:00
firmwareError("Invalid brain_pin_e: %d", brainPin);
2014-09-25 22:02:43 -07:00
return GPIO_NULL;
2014-08-29 07:52:33 -07:00
}
return PORTS[brainPin / 16];
}
ioportmask_t getHwPin(brain_pin_e brainPin) {
2014-11-10 07:03:20 -08:00
if (brainPin == GPIO_UNASSIGNED)
2014-08-29 07:52:33 -07:00
return EFI_ERROR_CODE;
2014-11-10 07:03:20 -08:00
if (brainPin > GPIO_UNASSIGNED || brainPin < 0) {
2014-08-29 07:52:33 -07:00
firmwareError("Invalid brain_pin_e: %d", brainPin);
return EFI_ERROR_CODE;
}
return brainPin % 16;
}
2015-01-07 16:03:45 -08:00
void outputPinRegisterExt2(const char *msg, OutputPin *output, brain_pin_e brainPin, pin_output_mode_e *outputMode) {
2014-12-29 20:03:34 -08:00
if (brainPin == GPIO_UNASSIGNED)
return;
2014-08-29 07:52:33 -07:00
GPIO_TypeDef *hwPort = getHwPort(brainPin);
int hwPin = getHwPin(brainPin);
2015-01-07 16:03:45 -08:00
outputPinRegisterExt(msg, output, hwPort, hwPin, outputMode);
2014-08-29 07:52:33 -07:00
}
2015-01-07 15:04:40 -08:00
void outputPinRegister(const char *msg, OutputPin *output, GPIO_TypeDef *port, uint32_t pin) {
outputPinRegisterExt(msg, output, port, pin, &DEFAULT_OUTPUT);
2014-08-29 07:52:33 -07:00
}
2015-01-07 16:03:45 -08:00
OutputPin errorLedPin;
extern OutputPin checkEnginePin;
2014-08-29 07:52:33 -07:00
void initPrimaryPins(void) {
2015-01-07 16:03:45 -08:00
outputPinRegister("LED_ERROR", &errorLedPin, LED_ERROR_PORT, LED_ERROR_PIN);
2014-08-29 07:52:33 -07:00
}
2014-09-25 22:02:43 -07:00
static void getPinValue(const char *name) {
io_pin_e pin = getPinByName(name);
if (pin == IO_INVALID) {
return;
}
OutputPin * outputPin = &outputs[pin];
int value = getLogicPinValue(outputPin);
scheduleMsg(&logger, "pin_value %s %d", name, value);
}
2014-08-29 07:52:33 -07:00
void initOutputPins(void) {
2014-09-25 22:02:43 -07:00
initLogging(&logger, "io_pins");
2014-08-29 07:52:33 -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));
// outputPinRegister("ext led 1", LED_EXT_1, EXTRA_LED_1_PORT, EXTRA_LED_1_PIN);
// outputPinRegister("ext led 2", LED_EXT_2, EXTRA_LED_2_PORT, EXTRA_LED_2_PIN);
// outputPinRegister("ext led 3", LED_EXT_3, EXTRA_LED_2_PORT, EXTRA_LED_3_PIN);
// outputPinRegister("alive1", LED_DEBUG, GPIOD, 6);
2015-01-07 16:03:45 -08:00
outputPinRegisterExt2("MalfunctionIndicator", &checkEnginePin, boardConfiguration->malfunctionIndicatorPin, &DEFAULT_OUTPUT);
2014-08-29 07:52:33 -07:00
// todo: are these needed here? todo: make configurable
// outputPinRegister("spi CS1", SPI_CS_1, SPI_CS1_PORT, SPI_CS1_PIN);
// outputPinRegister("spi CS2", SPI_CS_2, SPI_CS2_PORT, SPI_CS2_PIN);
// outputPinRegister("spi CS3", SPI_CS_3, SPI_CS3_PORT, SPI_CS3_PIN);
// outputPinRegister("spi CS4", SPI_CS_4, SPI_CS4_PORT, SPI_CS4_PIN);
2014-12-23 19:06:10 -08:00
#if HAL_USE_SPI || defined(__DOXYGEN__)
2015-01-07 15:04:40 -08:00
outputPinRegister("spi CS5", &outputs[(int)SPI_CS_SD_MODULE], SPI_SD_MODULE_PORT, SPI_SD_MODULE_PIN);
2014-12-23 19:06:10 -08:00
#endif
2014-08-29 07:52:33 -07:00
// todo: should we move this code closer to the fuel pump logic?
2015-01-07 16:03:45 -08:00
outputPinRegisterExt2("fuel pump relay", &outputs[(int)FUEL_PUMP_RELAY], boardConfiguration->fuelPumpPin, &DEFAULT_OUTPUT);
2014-08-29 07:52:33 -07:00
2015-01-07 16:03:45 -08:00
outputPinRegisterExt2("main relay", &outputs[(int)MAIN_RELAY], boardConfiguration->mainRelayPin, &boardConfiguration->mainRelayPinMode);
2014-12-29 20:03:34 -08:00
2015-01-07 16:03:45 -08:00
outputPinRegisterExt2("fan relay", &outputs[(int)FAN_RELAY], boardConfiguration->fanPin, &DEFAULT_OUTPUT);
outputPinRegisterExt2("o2 heater", &outputs[(int)O2_HEATER], boardConfiguration->o2heaterPin, &DEFAULT_OUTPUT);
outputPinRegisterExt2("A/C relay", &outputs[(int)AC_RELAY], boardConfiguration->acRelayPin, &boardConfiguration->acRelayPinMode);
2014-08-29 07:52:33 -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);
*/
2014-09-25 22:02:43 -07:00
addConsoleActionS("get_pin_value", getPinValue);
2014-08-29 07:52:33 -07:00
}
2014-11-08 08:09:38 -08:00
2014-12-23 22:03:26 -08:00
#if EFI_GPIO
2014-11-08 08:09:38 -08:00
static io_pin_e TO_BE_TURNED_OFF_ON_ERROR[] = { SPARKOUT_1_OUTPUT, SPARKOUT_2_OUTPUT, SPARKOUT_3_OUTPUT,
SPARKOUT_4_OUTPUT, SPARKOUT_5_OUTPUT, SPARKOUT_6_OUTPUT, SPARKOUT_7_OUTPUT, SPARKOUT_8_OUTPUT,
SPARKOUT_9_OUTPUT, SPARKOUT_10_OUTPUT, SPARKOUT_11_OUTPUT, SPARKOUT_12_OUTPUT,
INJECTOR_1_OUTPUT, INJECTOR_2_OUTPUT, INJECTOR_3_OUTPUT, INJECTOR_4_OUTPUT, INJECTOR_5_OUTPUT,
INJECTOR_6_OUTPUT, INJECTOR_7_OUTPUT, INJECTOR_8_OUTPUT, INJECTOR_9_OUTPUT, INJECTOR_10_OUTPUT,
INJECTOR_11_OUTPUT, INJECTOR_12_OUTPUT, FUEL_PUMP_RELAY };
/**
* 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) {
int l = sizeof(TO_BE_TURNED_OFF_ON_ERROR) / sizeof(io_pin_e);
for (int i = 0; i < l; i++) {
2015-01-04 11:05:46 -08:00
turnOutputPinOff(TO_BE_TURNED_OFF_ON_ERROR[l]);
2014-11-08 08:09:38 -08:00
}
}
2014-12-23 22:03:26 -08:00
#endif