2014-08-29 07:52:33 -07:00
|
|
|
/**
|
|
|
|
* @file hardware.cpp
|
|
|
|
* @brief Hardware package entry point
|
|
|
|
*
|
|
|
|
* @date May 27, 2013
|
|
|
|
* @author Andrey Belomutskiy, (c) 2012-2014
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "main.h"
|
|
|
|
#include "hardware.h"
|
|
|
|
#include "pin_repository.h"
|
|
|
|
#include "io_pins.h"
|
|
|
|
#include "rtc_helper.h"
|
|
|
|
#include "rfiutil.h"
|
|
|
|
#include "console_io.h"
|
|
|
|
|
|
|
|
#include "adc_inputs.h"
|
|
|
|
|
|
|
|
#include "trigger_input.h"
|
|
|
|
#include "eficonsole.h"
|
|
|
|
#include "board_test.h"
|
|
|
|
|
|
|
|
#include "mcp3208.h"
|
|
|
|
#include "HIP9011.h"
|
2014-09-17 09:03:04 -07:00
|
|
|
#include "max31855.h"
|
2014-08-29 07:52:33 -07:00
|
|
|
#include "can_hw.h"
|
|
|
|
#include "histogram.h"
|
|
|
|
#include "mmc_card.h"
|
|
|
|
#include "neo6m.h"
|
|
|
|
#include "lcd_HD44780.h"
|
|
|
|
#include "settings.h"
|
|
|
|
#include "algo.h"
|
|
|
|
|
|
|
|
#if EFI_INTERNAL_FLASH
|
|
|
|
#include "flash_main.h"
|
|
|
|
#endif /* EFI_INTERNAL_FLASH */
|
|
|
|
|
|
|
|
#include "trigger_central.h"
|
|
|
|
#include "svnversion.h"
|
|
|
|
#include "engine_configuration.h"
|
|
|
|
#include "ec2.h"
|
|
|
|
|
|
|
|
extern engine_configuration2_s * engineConfiguration2;
|
|
|
|
|
|
|
|
static bool isSpiInitialized[5] = { false, false, false, false, false };
|
|
|
|
|
|
|
|
static void initSpiModule(SPIDriver *driver, ioportid_t sckPort, ioportmask_t sckPin, ioportid_t misoPort,
|
|
|
|
ioportmask_t misoPin, ioportid_t mosiPort, ioportmask_t mosiPin, int af) {
|
|
|
|
mySetPadMode("SPI clock", sckPort, sckPin, PAL_MODE_ALTERNATE(af));
|
|
|
|
|
|
|
|
mySetPadMode("SPI master out", mosiPort, mosiPin, PAL_MODE_ALTERNATE(af));
|
|
|
|
mySetPadMode("SPI master in ", misoPort, misoPin, PAL_MODE_ALTERNATE(af));
|
|
|
|
}
|
|
|
|
|
|
|
|
static Mutex spiMtx;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Only one consumer can use SPI bus at a given time
|
|
|
|
*/
|
|
|
|
void lockSpi(spi_device_e device) {
|
|
|
|
// todo: different locks for different SPI devices!
|
|
|
|
chMtxLock(&spiMtx);
|
|
|
|
}
|
|
|
|
|
|
|
|
void unlockSpi(void) {
|
|
|
|
chMtxUnlock();
|
|
|
|
}
|
|
|
|
|
|
|
|
void turnOnSpi(spi_device_e device) {
|
|
|
|
if (isSpiInitialized[device])
|
|
|
|
return; // already initialized
|
|
|
|
isSpiInitialized[device] = true;
|
|
|
|
if (device == SPI_DEVICE_1) {
|
|
|
|
#if STM32_SPI_USE_SPI1
|
|
|
|
// scheduleMsg(&logging, "Turning on SPI1 pins");
|
|
|
|
initSpiModule(&SPID1,
|
|
|
|
EFI_SPI1_SCK_PORT, EFI_SPI1_SCK_PIN,
|
|
|
|
EFI_SPI1_MISO_PORT, EFI_SPI1_MISO_PIN,
|
|
|
|
EFI_SPI1_MOSI_PORT, EFI_SPI1_MOSI_PIN,
|
|
|
|
EFI_SPI1_AF);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
if (device == SPI_DEVICE_2) {
|
|
|
|
#if STM32_SPI_USE_SPI2
|
|
|
|
// scheduleMsg(&logging, "Turning on SPI2 pins");
|
|
|
|
initSpiModule(&SPID2,
|
|
|
|
EFI_SPI2_SCK_PORT, EFI_SPI2_SCK_PIN,
|
|
|
|
EFI_SPI2_MISO_PORT, EFI_SPI2_MISO_PIN,
|
|
|
|
EFI_SPI2_MOSI_PORT, EFI_SPI2_MOSI_PIN,
|
|
|
|
EFI_SPI2_AF);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
if (device == SPI_DEVICE_3) {
|
|
|
|
#if STM32_SPI_USE_SPI3
|
|
|
|
// scheduleMsg(&logging, "Turning on SPI3 pins");
|
|
|
|
initSpiModule(&SPID3,
|
|
|
|
EFI_SPI3_SCK_PORT, EFI_SPI3_SCK_PIN,
|
|
|
|
EFI_SPI3_MISO_PORT, EFI_SPI3_MISO_PIN,
|
|
|
|
EFI_SPI3_MOSI_PORT, EFI_SPI3_MOSI_PIN,
|
|
|
|
EFI_SPI3_AF);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-08 15:02:52 -07:00
|
|
|
static void initSpiModules(board_configuration_s *boardConfiguration) {
|
2014-08-29 07:52:33 -07:00
|
|
|
if (boardConfiguration->is_enabled_spi_2) {
|
|
|
|
turnOnSpi(SPI_DEVICE_2);
|
|
|
|
}
|
|
|
|
if (boardConfiguration->is_enabled_spi_3) {
|
|
|
|
turnOnSpi(SPI_DEVICE_3);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static I2CConfig i2cfg = { OPMODE_I2C, 100000, STD_DUTY_CYCLE, };
|
|
|
|
|
|
|
|
void initI2Cmodule(void) {
|
|
|
|
print("Starting I2C module\r\n");
|
|
|
|
i2cInit();
|
|
|
|
i2cStart(&I2CD1, &i2cfg);
|
|
|
|
|
|
|
|
mySetPadMode("I2C clock", EFI_I2C_SCL_PORT, EFI_I2C_SCL_PIN,
|
|
|
|
PAL_MODE_ALTERNATE(EFI_I2C_AF) | PAL_STM32_OTYPE_OPENDRAIN);
|
|
|
|
mySetPadMode("I2C data", EFI_I2C_SDA_PORT, EFI_I2C_SDA_PIN,
|
|
|
|
PAL_MODE_ALTERNATE(EFI_I2C_AF) | PAL_STM32_OTYPE_OPENDRAIN);
|
|
|
|
}
|
|
|
|
|
|
|
|
//static char txbuf[1];
|
|
|
|
|
|
|
|
static void sendI2Cbyte(int addr, int data) {
|
|
|
|
// i2cAcquireBus(&I2CD1);
|
|
|
|
// txbuf[0] = data;
|
|
|
|
// i2cMasterTransmit(&I2CD1, addr, txbuf, 1, NULL, 0);
|
|
|
|
// i2cReleaseBus(&I2CD1);
|
|
|
|
}
|
|
|
|
|
2014-09-25 20:06:56 -07:00
|
|
|
// this is all very lame code, just playing with EXTI for now. TODO: refactor it competely!
|
|
|
|
static int joyTotal = 0;
|
|
|
|
static int joyA = 0;
|
|
|
|
static int joyB = 0;
|
|
|
|
static int joyC = 0;
|
|
|
|
|
2014-09-08 15:02:52 -07:00
|
|
|
void initHardware(Logging *logger, Engine *engine) {
|
|
|
|
engine_configuration_s *engineConfiguration = engine->engineConfiguration;
|
2014-09-08 21:02:56 -07:00
|
|
|
efiAssertVoid(engineConfiguration!=NULL, "engineConfiguration");
|
2014-09-08 15:02:52 -07:00
|
|
|
board_configuration_s *boardConfiguration = &engineConfiguration->bc;
|
|
|
|
|
2014-08-29 07:52:33 -07:00
|
|
|
printMsg(logger, "initHardware()");
|
|
|
|
// todo: enable protection. it's disabled because it takes
|
|
|
|
// 10 extra seconds to re-flash the chip
|
|
|
|
//flashProtect();
|
|
|
|
|
|
|
|
chMtxInit(&spiMtx);
|
|
|
|
|
|
|
|
#if EFI_HISTOGRAMS
|
|
|
|
/**
|
|
|
|
* histograms is a data structure for CPU monitor, it does not depend on configuration
|
|
|
|
*/
|
|
|
|
initHistogramsModule();
|
|
|
|
#endif /* EFI_HISTOGRAMS */
|
|
|
|
|
2014-11-02 11:04:37 -08:00
|
|
|
/**
|
|
|
|
* This is so early because we want to init logger
|
|
|
|
* which would be used while finding trigger synch index
|
|
|
|
* while config read
|
|
|
|
*/
|
|
|
|
initTriggerDecoder();
|
|
|
|
|
|
|
|
|
2014-08-29 07:52:33 -07:00
|
|
|
/**
|
|
|
|
* We need the LED_ERROR pin even before we read configuration
|
|
|
|
*/
|
|
|
|
initPrimaryPins();
|
|
|
|
|
2014-09-08 21:02:56 -07:00
|
|
|
if (hasFirmwareError()) {
|
2014-08-29 07:52:33 -07:00
|
|
|
return;
|
2014-09-08 21:02:56 -07:00
|
|
|
}
|
2014-08-29 07:52:33 -07:00
|
|
|
|
|
|
|
initDataStructures(engineConfiguration);
|
|
|
|
|
|
|
|
#if EFI_INTERNAL_FLASH
|
|
|
|
|
|
|
|
palSetPadMode(CONFIG_RESET_SWITCH_PORT, CONFIG_RESET_SWITCH_PIN, PAL_MODE_INPUT_PULLUP);
|
|
|
|
|
2014-11-02 10:03:07 -08:00
|
|
|
initFlash(engine);
|
2014-08-29 07:52:33 -07:00
|
|
|
/**
|
|
|
|
* this call reads configuration from flash memory or sets default configuration
|
|
|
|
* if flash state does not look right.
|
|
|
|
*/
|
|
|
|
if (SHOULD_INGORE_FLASH()) {
|
|
|
|
engineConfiguration->engineType = FORD_ASPIRE_1996;
|
2014-10-31 13:03:07 -07:00
|
|
|
resetConfigurationExt(logger, engineConfiguration->engineType, engine);
|
2014-08-29 07:52:33 -07:00
|
|
|
writeToFlash();
|
|
|
|
} else {
|
|
|
|
readFromFlash();
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
engineConfiguration->engineType = FORD_ASPIRE_1996;
|
|
|
|
resetConfigurationExt(logger, engineConfiguration->engineType, engineConfiguration, engineConfiguration2, boardConfiguration);
|
|
|
|
#endif /* EFI_INTERNAL_FLASH */
|
|
|
|
|
2014-09-08 21:02:56 -07:00
|
|
|
if (hasFirmwareError()) {
|
2014-08-29 07:52:33 -07:00
|
|
|
return;
|
2014-09-08 21:02:56 -07:00
|
|
|
}
|
2014-08-29 07:52:33 -07:00
|
|
|
|
2014-10-14 12:02:56 -07:00
|
|
|
mySetPadMode2("board test", boardConfiguration->boardTestModeJumperPin, PAL_MODE_INPUT_PULLUP);
|
2014-08-29 07:52:33 -07:00
|
|
|
bool isBoardTestMode_b = GET_BOARD_TEST_MODE_VALUE();
|
|
|
|
|
2014-10-22 19:03:06 -07:00
|
|
|
initAdcInputs(isBoardTestMode_b);
|
2014-08-29 07:52:33 -07:00
|
|
|
|
|
|
|
if (isBoardTestMode_b) {
|
|
|
|
initBoardTest();
|
|
|
|
efiAssertVoid(FALSE, "board test done");
|
|
|
|
}
|
|
|
|
|
|
|
|
initRtc();
|
|
|
|
|
|
|
|
initOutputPins();
|
|
|
|
|
|
|
|
#if EFI_HIP_9011
|
|
|
|
initHip9011();
|
|
|
|
#endif /* EFI_HIP_9011 */
|
|
|
|
|
2014-09-17 17:02:53 -07:00
|
|
|
#if EFI_MAX_31855
|
2014-09-17 09:03:04 -07:00
|
|
|
initMax31855(boardConfiguration);
|
2014-09-17 17:02:53 -07:00
|
|
|
#endif /* EFI_MAX_31855 */
|
2014-09-17 09:03:04 -07:00
|
|
|
|
2014-08-29 07:52:33 -07:00
|
|
|
#if EFI_CAN_SUPPORT
|
|
|
|
initCan();
|
|
|
|
#endif /* EFI_CAN_SUPPORT */
|
|
|
|
|
|
|
|
// init_adc_mcp3208(&adcState, &SPID2);
|
|
|
|
// requestAdcValue(&adcState, 0);
|
|
|
|
|
|
|
|
// todo: figure out better startup logic
|
2014-09-08 15:02:52 -07:00
|
|
|
initTriggerCentral(engine);
|
2014-08-29 07:52:33 -07:00
|
|
|
|
|
|
|
#if EFI_SHAFT_POSITION_INPUT
|
|
|
|
initShaftPositionInputCapture();
|
|
|
|
#endif /* EFI_SHAFT_POSITION_INPUT */
|
|
|
|
|
2014-09-08 15:02:52 -07:00
|
|
|
initSpiModules(boardConfiguration);
|
2014-08-29 07:52:33 -07:00
|
|
|
|
|
|
|
#if EFI_FILE_LOGGING
|
|
|
|
initMmcCard();
|
|
|
|
#endif /* EFI_FILE_LOGGING */
|
|
|
|
|
|
|
|
// initFixedLeds();
|
|
|
|
|
|
|
|
// initBooleanInputs();
|
|
|
|
|
|
|
|
#if EFI_UART_GPS
|
|
|
|
initGps();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if ADC_SNIFFER
|
|
|
|
initAdcDriver();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if EFI_HD44780_LCD
|
|
|
|
// initI2Cmodule();
|
|
|
|
lcd_HD44780_init();
|
|
|
|
if (hasFirmwareError())
|
|
|
|
return;
|
|
|
|
|
|
|
|
lcd_HD44780_print_string(VCS_VERSION);
|
|
|
|
|
|
|
|
#endif /* EFI_HD44780_LCD */
|
|
|
|
|
|
|
|
addConsoleActionII("i2c", sendI2Cbyte);
|
|
|
|
|
|
|
|
// while (true) {
|
|
|
|
// for (int addr = 0x20; addr < 0x28; addr++) {
|
|
|
|
// sendI2Cbyte(addr, 0);
|
|
|
|
// int err = i2cGetErrors(&I2CD1);
|
|
|
|
// print("I2C: err=%x from %d\r\n", err, addr);
|
|
|
|
// chThdSleepMilliseconds(5);
|
|
|
|
// sendI2Cbyte(addr, 255);
|
|
|
|
// chThdSleepMilliseconds(5);
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
|
|
|
|
printMsg(logger, "initHardware() OK!");
|
|
|
|
}
|
2014-09-20 12:03:00 -07:00
|
|
|
|
|
|
|
SPIDriver * getSpiDevice(spi_device_e spiDevice) {
|
2014-09-20 18:02:53 -07:00
|
|
|
if (spiDevice == SPI_NONE) {
|
|
|
|
return NULL;
|
|
|
|
}
|
2014-09-20 12:03:00 -07:00
|
|
|
#if STM32_SPI_USE_SPI1 || defined(__DOXYGEN__)
|
2014-09-20 18:02:53 -07:00
|
|
|
if (spiDevice == SPI_DEVICE_1) {
|
2014-09-20 12:03:00 -07:00
|
|
|
return &SPID1;
|
2014-09-20 18:02:53 -07:00
|
|
|
}
|
2014-09-20 12:03:00 -07:00
|
|
|
#endif
|
|
|
|
#if STM32_SPI_USE_SPI2 || defined(__DOXYGEN__)
|
2014-09-20 18:02:53 -07:00
|
|
|
if (spiDevic e== SPI_DEVICE_2) {
|
|
|
|
return &SPID2;
|
|
|
|
}
|
2014-09-20 12:03:00 -07:00
|
|
|
#endif
|
|
|
|
#if STM32_SPI_USE_SPI3 || defined(__DOXYGEN__)
|
2014-09-20 18:02:53 -07:00
|
|
|
if (spiDevice == SPI_DEVICE_3) {
|
2014-09-20 12:03:00 -07:00
|
|
|
return &SPID3;
|
2014-09-20 18:02:53 -07:00
|
|
|
}
|
2014-09-20 12:03:00 -07:00
|
|
|
#endif
|
|
|
|
firmwareError("Unexpected SPI device: %d", spiDevice);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2014-09-20 13:02:57 -07:00
|
|
|
void initSpiCs(SPIConfig *spiConfig, brain_pin_e csPin) {
|
|
|
|
spiConfig->end_cb = NULL;
|
|
|
|
ioportid_t port = getHwPort(csPin);
|
|
|
|
ioportmask_t pin = getHwPin(csPin);
|
|
|
|
spiConfig->ssport = port;
|
|
|
|
spiConfig->sspad = pin;
|
|
|
|
mySetPadMode("chip select", port, pin, PAL_STM32_MODE_OUTPUT);
|
|
|
|
}
|
|
|
|
|
2014-09-25 18:02:54 -07:00
|
|
|
// {EXT_CH_MODE_BOTH_EDGES | EXT_CH_MODE_AUTOSTART | EXT_MODE_GPIOA, extcb1},
|
|
|
|
|
|
|
|
static void extCallback(EXTDriver *extp, expchannel_t channel) {
|
2014-09-25 20:06:56 -07:00
|
|
|
joyTotal++;
|
|
|
|
if(channel==8) {
|
|
|
|
joyA++;
|
|
|
|
} else if (channel==9){
|
|
|
|
joyB++;
|
|
|
|
} else if (channel==10){
|
|
|
|
joyC++;
|
|
|
|
}
|
2014-09-25 18:02:54 -07:00
|
|
|
}
|
|
|
|
|
2014-09-25 20:06:56 -07:00
|
|
|
/**
|
|
|
|
* EXTI is a funny thing: you can only use same pin on one port. For example, you can use
|
|
|
|
* PA0 PB5 PE2 PD7
|
|
|
|
* but you cannot use
|
|
|
|
* PA0 PB0 PE2 PD7
|
|
|
|
* because pin '0' would be used on two different ports
|
|
|
|
*/
|
|
|
|
|
|
|
|
static EXTConfig extcfg = { { { EXT_CH_MODE_DISABLED, NULL }, { EXT_CH_MODE_DISABLED, NULL }, {
|
|
|
|
EXT_CH_MODE_DISABLED, NULL }, { EXT_CH_MODE_DISABLED, NULL }, { EXT_CH_MODE_DISABLED, NULL }, {
|
|
|
|
EXT_CH_MODE_DISABLED, NULL }, { EXT_CH_MODE_DISABLED, NULL }, { EXT_CH_MODE_DISABLED, NULL }, {
|
|
|
|
EXT_CH_MODE_DISABLED, NULL }, { EXT_CH_MODE_DISABLED, NULL }, { EXT_CH_MODE_DISABLED, NULL }, {
|
|
|
|
EXT_CH_MODE_DISABLED, NULL }, { EXT_CH_MODE_DISABLED, NULL }, { EXT_CH_MODE_DISABLED, NULL }, {
|
|
|
|
EXT_CH_MODE_DISABLED, NULL }, { EXT_CH_MODE_DISABLED, NULL }, { EXT_CH_MODE_DISABLED, NULL }, {
|
|
|
|
EXT_CH_MODE_DISABLED, NULL }, { EXT_CH_MODE_DISABLED, NULL }, { EXT_CH_MODE_DISABLED, NULL }, {
|
|
|
|
EXT_CH_MODE_DISABLED, NULL }, { EXT_CH_MODE_DISABLED, NULL }, { EXT_CH_MODE_DISABLED, NULL } } };
|
2014-09-25 18:02:54 -07:00
|
|
|
|
2014-09-25 20:06:56 -07:00
|
|
|
void initExt(void) {
|
2014-09-25 18:02:54 -07:00
|
|
|
|
2014-09-25 20:06:56 -07:00
|
|
|
// if (1 == 1) {
|
|
|
|
// return;
|
|
|
|
// }
|
|
|
|
|
|
|
|
extcfg.channels[8].mode = EXT_CH_MODE_BOTH_EDGES | EXT_CH_MODE_AUTOSTART | EXT_MODE_GPIOD; // PD8
|
|
|
|
extcfg.channels[8].cb = extCallback;
|
|
|
|
extcfg.channels[9].mode = EXT_CH_MODE_BOTH_EDGES | EXT_CH_MODE_AUTOSTART | EXT_MODE_GPIOD; // PD9
|
|
|
|
extcfg.channels[9].cb = extCallback;
|
|
|
|
extcfg.channels[10].mode = EXT_CH_MODE_BOTH_EDGES | EXT_CH_MODE_AUTOSTART | EXT_MODE_GPIOD; // PD10
|
|
|
|
extcfg.channels[10].cb = extCallback;
|
|
|
|
extcfg.channels[11].mode = EXT_CH_MODE_BOTH_EDGES | EXT_CH_MODE_AUTOSTART | EXT_MODE_GPIOD; // PD11
|
|
|
|
extcfg.channels[11].cb = extCallback;
|
|
|
|
|
|
|
|
|
|
|
|
mySetPadMode("joy A", GPIOD, 8, PAL_MODE_INPUT_PULLUP);
|
|
|
|
mySetPadMode("joy A", GPIOD, 9, PAL_MODE_INPUT_PULLUP);
|
|
|
|
mySetPadMode("joy A", GPIOD, 10, PAL_MODE_INPUT_PULLUP);
|
|
|
|
mySetPadMode("joy A", GPIOD, 11, PAL_MODE_INPUT_PULLUP);
|
|
|
|
}
|
2014-09-25 18:02:54 -07:00
|
|
|
|