diff --git a/firmware/hw_layer/drivers/drivers.mk b/firmware/hw_layer/drivers/drivers.mk index d23c06ed1c..439d3d0455 100644 --- a/firmware/hw_layer/drivers/drivers.mk +++ b/firmware/hw_layer/drivers/drivers.mk @@ -6,4 +6,5 @@ HW_LAYER_DRIVERS_INC = \ HW_LAYER_DRIVERS_CORE = \ $(DRIVERS_DIR)/gpio/core.c \ -HW_LAYER_DRIVERS = +HW_LAYER_DRIVERS = \ + $(DRIVERS_DIR)/gpio/tle6240.c \ diff --git a/firmware/hw_layer/drivers/gpio/tle6240.c b/firmware/hw_layer/drivers/gpio/tle6240.c new file mode 100644 index 0000000000..115f45df17 --- /dev/null +++ b/firmware/hw_layer/drivers/gpio/tle6240.c @@ -0,0 +1,500 @@ +/* + * tle6240.c + * + * TLE6240GP Smart 16-Channel Low-Side Switch + * + * All 16 channels can be controlled via the serial interface (SPI). + * In addition to the serial control it is possible to control channel 1 to 4 + * and 9 to 12 direct in parallel with a separate input pin. + * + * Looks like 3.3v SI and SCLK are NOT possible (H above 0.7Vs required, that's 3.5v for 5.0Vs) + * 5 MHz SPI + * Update: looks like possible: + * DS page 3: "Compatible with 3 V Microcontrollers" + * DS page 12: "Input High Voltage 2.0 V min" + * + * @date Dec 29, 2018 + * @author Andrey Belomutskiy, (c) 2012-2018 + * + * @date Mar 06, 2019 + * @author Andrey Gusakov, (c) 2019 + */ + +#include +#include "gpio/gpio_ext.h" +#include "gpio/tle6240.h" +#include "pin_repository.h" + +#ifndef BOARD_TLE6240_COUNT + #define BOARD_TLE6240_COUNT 0 +#endif + +#if (BOARD_TLE6240_COUNT > 0) + +/* + * TODO list: + * - add irq support with fallback to polling mode (now polling mode only) + * - handle low-active inputs (set with PRG pin). Now driver assume high-active + * - add way to export native pin data of direct driven outputs. To avoid + * call to tle6240_writePad that will finaly call native gpio set/clear fn. + * In this case direct drive gpios should not be ocupaied by markUsed in init? + * - fill deinit function with some code? + * - support emergency shutdown using reset pin + * - convert diagnostic to some enum + * - use DMA (currently there is issue (?) with SPI+DMA on STM32F7xx) + */ + +/*==========================================================================*/ +/* Driver local definitions. */ +/*==========================================================================*/ + +#define DRIVER_NAME "tle6240" + +static bool drv_task_ready = false; + +typedef enum { + TLE6240_DISABLED = 0, + TLE6240_WAIT_INIT, + TLE6240_READY, + TLE6240_FAILED +} tle6240_drv_state; + +/* set 0000b for channes == 0..7 and 1111b for channels 8..15 */ +#define CMD_CHIP(ch) ((ch < 8) ? 0x00 : 0x0f) +/* Full Diagnoscit, data byte ignored */ +#define CMD_FULL_DIAG(ch) (((0x00 | CMD_CHIP(ch)) << 8) | 0x00) +/* Get state of 8 paralled inputs and 1-bit Diagnostic, data byte ignored */ +#define CMD_IO_SHORTDIAG(ch) (((0xc0 | CMD_CHIP(ch)) << 8) | 0x00) +/* Echo function test of SPI, SI will be connected to SO on next access */ +#define CMD_ECHO(ch) (((0xA0 | CMD_CHIP(ch)) << 8) | 0x00) +/* in data ORed, Full diagnostic output on next access */ +#define CMD_OR_DIAG(ch, data) (((0x30 | CMD_CHIP(ch)) << 8) | (data & 0xff)) +/* in data ANDed, Full diagnostic output on next access */ +#define CMD_AND_DIAG(ch, data) (((0xf0 | CMD_CHIP(ch)) << 8) | (data & 0xff)) + +/*==========================================================================*/ +/* Driver exported variables. */ +/*==========================================================================*/ + +/*==========================================================================*/ +/* Driver local variables and types. */ +/*==========================================================================*/ + +/* OS */ +SEMAPHORE_DECL(tle6240_wake, 10 /* or BOARD_TLE6240_COUNT ? */); +static THD_WORKING_AREA(tle6240_thread_1_wa, 256); + +/* Driver */ +struct tle6240_priv { + const struct tle6240_config *cfg; + /* cached output state - state last send to chip */ + uint16_t o_state_cached; + /* state to be sended to chip */ + uint16_t o_state; + /* direct driven output mask */ + uint16_t o_direct_mask; + /* full diagnostic status */ + uint16_t diag[2]; + /* diagnostic for ch 8..15 was requsted by last access + * can skip one transaction next time */ + bool diag_8_reguested; + + tle6240_drv_state drv_state; +}; + +static struct tle6240_priv chips[BOARD_TLE6240_COUNT]; + +/*==========================================================================*/ +/* Driver local functions. */ +/*==========================================================================*/ + +static SPIDriver *get_bus(struct tle6240_priv *chip) +{ + /* return non-const SPIDriver* from const struct cfg */ + return chip->cfg->spi_bus; +} + +/** + * @brief TLE6240 send and receive routine. + * @details Sends and receives 16 bits. CS asserted before and released + * after transaction. + */ + +static int tle6240_spi_rw(struct tle6240_priv *chip, uint16_t tx, uint16_t *rx) +{ + uint16_t rxb; + SPIDriver *spi = get_bus(chip); + + /* Acquire ownership of the bus. */ + spiAcquireBus(spi); + /* Setup transfer parameters. */ + spiStart(spi, &chip->cfg->spi_config); + /* Slave Select assertion. */ + spiSelect(spi); + /* Atomic transfer operations. */ + /* TODO: check why spiExchange transfers invalid data on STM32F7xx, DMA issue? */ + //spiExchange(spi, 2, &tx, &rxb); + rxb = spiPolledExchange(spi, tx); + /* Slave Select de-assertion. */ + spiUnselect(spi); + /* Ownership release. */ + spiReleaseBus(spi); + + if (rx) + *rx = rxb; + + /* no errors for now */ + return 0; +} + +/** + * @brief TLE6240 send output registers data. + * @details Sends ORed data to register, also receive 2-bit diagnostic. + */ + +static int tle6240_update_output_and_diag(struct tle6240_priv *chip) +{ + int ret; + uint16_t out_data; + + /* TODO: lock? */ + + /* atomic */ + /* set value only for non-direct driven pins */ + out_data = chip->o_state & (~chip->o_direct_mask); + if (chip->diag_8_reguested) { + /* diagnostic for OUT8..15 was requested on prev access */ + ret = tle6240_spi_rw(chip, CMD_OR_DIAG(0, (out_data >> 0) & 0xff), &chip->diag[1]); + ret |= tle6240_spi_rw(chip, CMD_OR_DIAG(8, (out_data >> 8) & 0xff), &chip->diag[0]); + } else { + ret = tle6240_spi_rw(chip, CMD_OR_DIAG(0, (out_data >> 0) & 0xff), NULL); + ret |= tle6240_spi_rw(chip, CMD_OR_DIAG(8, (out_data >> 8) & 0xff), &chip->diag[0]); + /* send same one more time to receive OUT8..15 diagnostic */ + ret |= tle6240_spi_rw(chip, CMD_OR_DIAG(8, (out_data >> 8) & 0xff), &chip->diag[1]); + } + + chip->diag_8_reguested = false; + if (ret == 0) { + /* atomic */ + chip->o_state_cached = out_data; + chip->diag_8_reguested = true; + } + + /* TODO: unlock? */ + + return ret; +} + +/** + * @brief TLE6240 chip init. + * @details Checks communication. Mark all used pins. + * Checks direct io signals integrity using test cmd. + * Reads initial diagnostic state. + */ + +static int tle6240_chip_init(struct tle6240_priv *chip) +{ + int n; + int ret; + uint16_t rx; + const struct tle6240_config *cfg = chip->cfg; + + /* mark pins used */ + ret = markUsed(cfg->spi_config.ssport, cfg->spi_config.sspad, DRIVER_NAME " CS"); + if (cfg->reset.port != NULL) + ret |= markUsed(cfg->reset.port, cfg->reset.pad, DRIVER_NAME " RST"); + for (n = 0; n < TLE6240_DIRECT_OUTPUTS; n++) + if (cfg->direct_io[n].port) + ret |= markUsed(cfg->direct_io[n].port, cfg->direct_io[n].pad, DRIVER_NAME " DIRECT IO"); + + if (ret) { + ret = -1; + goto err_gpios; + } + + /* release reset */ + if (cfg->reset.port != NULL) { + palClearPort(cfg->reset.port, + PAL_PORT_BIT(cfg->reset.pad)); + chThdSleepMilliseconds(1); + palSetPort(cfg->reset.port, + PAL_PORT_BIT(cfg->reset.pad)); + chThdSleepMilliseconds(10); + } + + /* check SPI communication */ + /* 0. set echo mode, chip number - don't care */ + ret = tle6240_spi_rw(chip, CMD_ECHO(0), NULL); + /* 1. check loopback */ + ret |= tle6240_spi_rw(chip, 0x5555, &rx); + if (ret || (rx != 0x5555)) { + //print(DRIVER_NAME " spi loopback test failed\n"); + ret = -2; + goto err_gpios; + } + + /* check direct io communication */ + /* 0. set all direct out to 0 */ + for (n = 0; n < TLE6240_DIRECT_OUTPUTS; n++) { + int i = (n < 4) ? n : (n + 4); + if (chip->o_direct_mask & (1 << i)) { + palClearPort(cfg->direct_io[n].port, + PAL_PORT_BIT(cfg->direct_io[n].pad)); + } + } + /* 1. disable IN0..7 outputs first (ADNed with 0x00) + * also will get full diag on next access */ + ret = tle6240_spi_rw(chip, CMD_AND_DIAG(0, 0x00), NULL); + /* 2. get diag for OUT0..7 and send disable OUT8..15 */ + ret |= tle6240_spi_rw(chip, CMD_AND_DIAG(8, 0x00), &chip->diag[0]); + /* 3. get diag for OUT8..15 and readback input status */ + ret |= tle6240_spi_rw(chip, CMD_IO_SHORTDIAG(0), &chip->diag[1]); + /* 4. send dummy short diag command and get 8 bit of input data and + * 8 bit of short diag */ + ret |= tle6240_spi_rw(chip, CMD_IO_SHORTDIAG(0), &rx); + rx = ((rx >> 4) & 0x0f00) | ((rx >> 8) & 0x000f); + if (ret || (rx & chip->o_direct_mask)) { + //print(DRIVER_NAME " direct io test #1 failed (invalid io mask %04x)\n", (rx & chip->o_direct_mask)); + ret = -3; + goto err_gpios; + } + + /* 5. set all direct io to 1 */ + for (n = 0; n < TLE6240_DIRECT_OUTPUTS; n++) { + int i = (n < 4) ? n : (n + 4); + if (chip->o_direct_mask & (1 << i)) { + palSetPort(cfg->direct_io[n].port, + PAL_PORT_BIT(cfg->direct_io[n].pad)); + } + } + /* 6. read chort diagnostic again */ + ret |= tle6240_spi_rw(chip, CMD_IO_SHORTDIAG(0), &rx); + rx = ((rx >> 4) & 0x0f00) | ((rx >> 8) & 0x000f); + rx &= chip->o_direct_mask; + if (ret || (rx != chip->o_direct_mask)) { + //print(DRIVER_NAME " direct io test #2 failed (invalid io mask %04x)\n", (rx ^ (~chip->o_direct_mask))); + ret = -4; + goto err_gpios; + } + + /* 7. set all all pins to OR mode, and upload pin states */ + ret = tle6240_update_output_and_diag(chip); + if (ret) { + //print(DRIVER_NAME " final setup error\n"); + ret = -5; + goto err_gpios; + } + + return 0; + +err_gpios: + /* unmark pins */ + markUnused(cfg->spi_config.ssport, cfg->spi_config.sspad); + if (cfg->reset.port != NULL) + markUnused(cfg->reset.port, cfg->reset.pad); + for (n = 0; n < TLE6240_DIRECT_OUTPUTS; n++) + if (cfg->direct_io[n].port) + markUnused(cfg->direct_io[n].port, cfg->direct_io[n].pad); + + return ret; +} + +/** + * @brief TLE6240 chip driver wakeup. + * @details Wake up driver. Will cause output register and + * diagnostic update. + */ + +static int tle6240_wake_driver(struct tle6240_priv *chip) +{ + (void)chip; + + chSemSignal(&tle6240_wake); + + return 0; +} + +/*==========================================================================*/ +/* Driver thread. */ +/*==========================================================================*/ + +static THD_FUNCTION(tle6240_driver_thread, p) +{ + int i; + msg_t msg; + + (void)p; + + chRegSetThreadName(DRIVER_NAME); + + while(1) { + msg = chSemWaitTimeout(&tle6240_wake, TIME_MS2I(TLE6240_POLL_INTERVAL_MS)); + + /* should we care about msg == MSG_TIMEOUT? */ + (void)msg; + + for (i = 0; i < BOARD_TLE6240_COUNT; i++) { + int ret; + struct tle6240_priv *chip; + + chip = &chips[i]; + if ((chip->cfg == NULL) || + (chip->drv_state == TLE6240_DISABLED) || + (chip->drv_state == TLE6240_FAILED)) + continue; + + ret = tle6240_update_output_and_diag(chip); + if (ret) { + /* set state to TLE6240_FAILED? */ + } + } + } +} + +/*==========================================================================*/ +/* Driver interrupt handlers. */ +/*==========================================================================*/ + +/* TODO: add IRQ support */ + +/*==========================================================================*/ +/* Driver exported functions. */ +/*==========================================================================*/ + +int tle6240_writePad(void *data, unsigned int pin, int value) +{ + struct tle6240_priv *chip; + + if ((pin >= TLE6240_DIRECT_OUTPUTS) || (data == NULL)) + return -1; + + chip = (struct tle6240_priv *)data; + + /* TODO: lock */ + if (value) + chip->o_state |= (1 << pin); + else + chip->o_state &= ~(1 << pin); + /* TODO: unlock */ + /* direct driven? */ + if (chip->o_direct_mask & (1 << pin)) { + int n = (pin < 8) ? pin : (pin - 4); + + /* TODO: ensure that TLE6240 configured in active high mode */ + if (value) + palSetPort(chip->cfg->direct_io[n].port, + PAL_PORT_BIT(chip->cfg->direct_io[n].pad)); + else + palClearPort(chip->cfg->direct_io[n].port, + PAL_PORT_BIT(chip->cfg->direct_io[n].pad)); + } else { + tle6240_wake_driver(chip); + } + + return 0; +} + +int tle6240_getDiag(void *data, unsigned int pin) +{ + int diag; + struct tle6240_priv *chip; + + if ((pin >= TLE6240_DIRECT_OUTPUTS) || (data == NULL)) + return -1; + + chip = (struct tle6240_priv *)data; + + diag = (chip->diag[(pin > 7) ? 1 : 0] >> ((pin % 8) * 2)) & 0x03; + + /* convert to some common enum? */ + return diag; +} + +int tle6240_init(void * data) +{ + int ret; + struct tle6240_priv *chip; + + chip = (struct tle6240_priv *)data; + + ret = tle6240_chip_init(chip); + if (ret) + return ret; + + chip->drv_state = TLE6240_READY; + + if (!drv_task_ready) { + chThdCreateStatic(tle6240_thread_1_wa, sizeof(tle6240_thread_1_wa), + NORMALPRIO + 1, tle6240_driver_thread, NULL); + drv_task_ready = true; + } + + return 0; +} + +int tle6240_deinit(void *data) +{ + (void)data; + + /* TODO: set all pins to inactive state, stop task? */ + return 0; +} + +struct gpiochip_ops tle6240_ops = { + .writePad = tle6240_writePad, + .readPad = NULL, /* chip outputs only */ + .getDiag = tle6240_getDiag, + .init = tle6240_init, + .deinit = tle6240_deinit, +}; + +/** + * @brief TLE6240 driver add. + * @details Checks for valid config + */ + +int tle6240_add(unsigned int n, const struct tle6240_config *cfg) +{ + int i; + struct tle6240_priv *chip; + + /* no config or no such chip */ + osalDbgCheck((cfg != NULL) && (cfg->spi_bus != NULL) && (n < BOARD_TLE6240_COUNT)); + + /* check for valid cs. + * DOTO: remove this check? CS can be driven by SPI */ + if (cfg->spi_config.ssport == NULL) + return -1; + + chip = &chips[n]; + + /* already initted? */ + if (chip->cfg != NULL) + return -1; + + chip->cfg = cfg; + chip->o_state = 0; + chip->o_state_cached = 0; + chip->o_direct_mask = 0; + chip->drv_state = TLE6240_WAIT_INIT; + for (i = 0; i < TLE6240_DIRECT_OUTPUTS; i++) { + if (cfg->direct_io[i].port != 0) + chip->o_direct_mask |= (1 << ((i < 4) ? i : (i + 4))); + } + + chip->drv_state = TLE6240_WAIT_INIT; + + /* register, return gpio chip base */ + return gpiochip_register(DRIVER_NAME, &tle6240_ops, TLE6240_OUTPUTS, chip); +} + +#else /* BOARD_TLE6240_COUNT > 0 */ + +int tle6240_add(unsigned int n, const struct tle6240_config *cfg) +{ + (void)n; (void)cfg; + + return -1; +} + +#endif /* BOARD_TLE6240_COUNT */ diff --git a/firmware/hw_layer/drivers/gpio/tle6240.h b/firmware/hw_layer/drivers/gpio/tle6240.h new file mode 100644 index 0000000000..bec72914be --- /dev/null +++ b/firmware/hw_layer/drivers/gpio/tle6240.h @@ -0,0 +1,45 @@ +/* + * tle6240.h + * + * TLE6240GP Smart 16-Channel Low-Side Switch + * + * @date Dec 29, 2018 + * @author Andrey Belomutskiy, (c) 2012-2018 + */ + +#ifndef HW_LAYER_TLE6240_H_ +#define HW_LAYER_TLE6240_H_ + +#include + +#define TLE6240_OUTPUTS 16 +#define TLE6240_DIRECT_OUTPUTS 8 + +/* DOTO: add irq support */ +#define TLE6240_POLL_INTERVAL_MS 100 + +struct tle6240_config { + SPIDriver *spi_bus; + const SPIConfig spi_config; + struct { + ioportid_t port; + uint_fast8_t pad; + } direct_io[TLE6240_DIRECT_OUTPUTS]; + struct { + ioportid_t port; + uint_fast8_t pad; + } reset; +}; + +#ifdef __cplusplus +extern "C" +{ +#endif /* __cplusplus */ + +int tle6240_add(unsigned int n, const struct tle6240_config *cfg); + +#ifdef __cplusplus +} +#endif /* __cplusplus */ + +#endif /* HW_LAYER_TLE6240_H_ */ diff --git a/firmware/hw_layer/tle6240.cpp b/firmware/hw_layer/tle6240.cpp deleted file mode 100644 index f3529d8d0f..0000000000 --- a/firmware/hw_layer/tle6240.cpp +++ /dev/null @@ -1,17 +0,0 @@ -/* - * tle6240.cpp - * - * TLE6240GP Smart 16-Channel Low-Side Switch - * - * All 16 channels can be controlled via the serial interface (SPI). In addition to the serial control it is possible to control channel 1 to 4 and 9 to 12 direct in parallel with a separate input pin. - * - * Looks like 3.3v SI and SCLK are NOT possible (H above 0.7Vs required, that's 3.5v for 5.0Vs) - * 5 MHz SPI - * - * @date Dec 29, 2018 - * @author Andrey Belomutskiy, (c) 2012-2018 - */ - - - - diff --git a/firmware/hw_layer/tle6240.h b/firmware/hw_layer/tle6240.h deleted file mode 100644 index 810638181d..0000000000 --- a/firmware/hw_layer/tle6240.h +++ /dev/null @@ -1,17 +0,0 @@ -/* - * tle6240.h - * - * TLE6240GP Smart 16-Channel Low-Side Switch - * - * @date Dec 29, 2018 - * @author Andrey Belomutskiy, (c) 2012-2018 - */ - -#ifndef HW_LAYER_TLE6240_H_ -#define HW_LAYER_TLE6240_H_ - - - - - -#endif /* HW_LAYER_TLE6240_H_ */