532 lines
15 KiB
C
532 lines
15 KiB
C
/*
|
|
* 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-2020
|
|
*
|
|
* @date Mar 06, 2019
|
|
* @author Andrey Gusakov, (c) 2019
|
|
*/
|
|
|
|
#include "global.h"
|
|
#include "gpio/gpio_ext.h"
|
|
#include "gpio/tle6240.h"
|
|
#include "pin_repository.h"
|
|
#include "os_util.h"
|
|
|
|
#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 finally call native gpio set/clear fn.
|
|
* In this case direct drive gpios should not be occupied 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];
|
|
|
|
static const char* tle6240_pin_names[TLE6240_OUTPUTS] = {
|
|
"tle6240.OUT1", "tle6240.OUT2", "tle6240.OUT3", "tle6240.OUT4",
|
|
"tle6240.OUT5", "tle6240.OUT6", "tle6240.OUT7", "tle6240.OUT8",
|
|
"tle6240.OUT9", "tle6240.OUT10", "tle6240.OUT11", "tle6240.OUT12",
|
|
"tle6240.OUT13", "tle6240.OUT14", "tle6240.OUT15", "tle6240.OUT16",
|
|
};
|
|
|
|
/*==========================================================================*/
|
|
/* 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 = gpio_pin_markUsed(cfg->spi_config.ssport, cfg->spi_config.sspad, DRIVER_NAME " CS");
|
|
ret = 0;
|
|
if (cfg->reset.port != NULL)
|
|
ret |= gpio_pin_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 |= gpio_pin_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 */
|
|
//gpio_pin_markUnused(cfg->spi_config.ssport, cfg->spi_config.sspad);
|
|
if (cfg->reset.port != NULL)
|
|
gpio_pin_markUnused(cfg->reset.port, cfg->reset.pad);
|
|
for (n = 0; n < TLE6240_DIRECT_OUTPUTS; n++)
|
|
if (cfg->direct_io[n].port)
|
|
gpio_pin_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;
|
|
|
|
if (isIsrContext()) {
|
|
// this is for normal runtime
|
|
int wasLocked = lockAnyContext();
|
|
chSemSignalI(&tle6240_wake);
|
|
if (!wasLocked) {
|
|
unlockAnyContext();
|
|
}
|
|
} else {
|
|
// this is for start-up to not hang up
|
|
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. */
|
|
/*==========================================================================*/
|
|
|
|
static int tle6240_writePad(void *data, unsigned int pin, int value)
|
|
{
|
|
struct tle6240_priv *chip;
|
|
|
|
if ((pin >= TLE6240_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;
|
|
}
|
|
|
|
static brain_pin_diag_e tle6240_getDiag(void *data, unsigned int pin)
|
|
{
|
|
int val;
|
|
brain_pin_diag_e diag;
|
|
struct tle6240_priv *chip;
|
|
|
|
if ((pin >= TLE6240_OUTPUTS) || (data == NULL))
|
|
return PIN_INVALID;
|
|
|
|
chip = (struct tle6240_priv *)data;
|
|
|
|
val = (chip->diag[(pin > 7) ? 1 : 0] >> ((pin % 8) * 2)) & 0x03;
|
|
if (val == 0x3)
|
|
diag = PIN_OK;
|
|
else if (val == 0x2)
|
|
/* Overload, shorted load or overtemperature */
|
|
diag = PIN_OVERLOAD | PIN_DRIVER_OVERTEMP;
|
|
else if (val == 0x1)
|
|
diag = PIN_OPEN;
|
|
else if (val == 0x0)
|
|
diag = PIN_SHORT_TO_GND;
|
|
|
|
return diag;
|
|
}
|
|
|
|
static 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;
|
|
}
|
|
|
|
static 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 index, const struct tle6240_config *cfg)
|
|
{
|
|
int i;
|
|
int ret;
|
|
struct tle6240_priv *chip;
|
|
|
|
/* no config or no such chip */
|
|
if ((!cfg) || (!cfg->spi_bus) || (index >= BOARD_TLE6240_COUNT))
|
|
return -1;
|
|
|
|
/* check for valid cs.
|
|
* TODO: remove this check? CS can be driven by SPI */
|
|
//if (cfg->spi_config.ssport == NULL)
|
|
// return -1;
|
|
|
|
chip = &chips[index];
|
|
|
|
/* 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 */
|
|
ret = gpiochip_register(DRIVER_NAME, &tle6240_ops, TLE6240_OUTPUTS, chip);
|
|
|
|
/* set default pin names, board init code can rewrite */
|
|
gpiochips_setPinNames(ret, tle6240_pin_names);
|
|
|
|
return ret;
|
|
}
|
|
|
|
#else /* BOARD_TLE6240_COUNT > 0 */
|
|
|
|
int tle6240_add(unsigned int index, const struct tle6240_config *cfg)
|
|
{
|
|
(void)index; (void)cfg;
|
|
|
|
return -1;
|
|
}
|
|
|
|
#endif /* BOARD_TLE6240_COUNT */
|