mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-17 11:12:46 +01:00
e8a4defafd
The driver only supports addressing GPIOs by their index, (ab)using `gpio_t` for that will break on platforms that encode GPIO values differently.
480 lines
13 KiB
C
480 lines
13 KiB
C
/*
|
|
* Copyright (C) 2018 Gunar Schorcht
|
|
*
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
* directory for more details.
|
|
*/
|
|
|
|
/**
|
|
* @ingroup drivers_pcf857x
|
|
* @brief Device driver for Texas Instruments PCF857X I2C I/O expanders
|
|
* @author Gunar Schorcht <gunar@schorcht.net>
|
|
* @file
|
|
* @{
|
|
*/
|
|
|
|
#include <errno.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#include "pcf857x.h"
|
|
|
|
#if IS_USED(MODULE_PCF857X_IRQ)
|
|
#include "event/thread.h"
|
|
#endif
|
|
|
|
#define ENABLE_DEBUG 0
|
|
#include "debug.h"
|
|
|
|
#if ENABLE_DEBUG
|
|
|
|
#define DEBUG_DEV(f, d, ...) \
|
|
DEBUG("[pcf857x] %s i2c dev=%d addr=%02x: " f "\n", \
|
|
__func__, d->params.dev, d->params.addr, ## __VA_ARGS__)
|
|
|
|
#else /* ENABLE_DEBUG */
|
|
|
|
#define DEBUG_DEV(f, d, ...)
|
|
|
|
#endif /* ENABLE_DEBUG */
|
|
|
|
#if !IS_USED(MODULE_PCF8574) && !IS_USED(MODULE_PCF8574A) && !IS_USED(MODULE_PCF8575)
|
|
#error "Please provide a list of pcf857x variants used by the application (pcf8574, pcf8574a or pcf8575)"
|
|
#endif
|
|
|
|
#if IS_USED(MODULE_PCF857X_IRQ_LOW)
|
|
#define PCF857X_EVENT_PRIO EVENT_PRIO_LOWEST
|
|
#elif IS_USED(MODULE_PCF857X_IRQ_MEDIUM)
|
|
#define PCF857X_EVENT_PRIO EVENT_PRIO_MEDIUM
|
|
#elif IS_USED(MODULE_PCF857X_IRQ_HIGHEST)
|
|
#define PCF857X_EVENT_PRIO EVENT_PRIO_HIGHEST
|
|
#endif
|
|
|
|
/** Forward declaration of functions for internal use */
|
|
|
|
static inline void _acquire(const pcf857x_t *dev);
|
|
static inline void _release(const pcf857x_t *dev);
|
|
static int _read(const pcf857x_t *dev, pcf857x_data_t *data);
|
|
static int _write(const pcf857x_t *dev, pcf857x_data_t data);
|
|
|
|
#if IS_USED(MODULE_PCF857X_IRQ)
|
|
|
|
/* interrutp service routine for IRQs */
|
|
static void _irq_isr(void *arg);
|
|
|
|
/* declaration of IRQ handler function */
|
|
static void _irq_handler(event_t *event);
|
|
|
|
/* internal update function */
|
|
static void _update_state(pcf857x_t* dev);
|
|
|
|
#endif /* MODULE_PCF857X_IRQ */
|
|
|
|
int pcf857x_init(pcf857x_t *dev, const pcf857x_params_t *params)
|
|
{
|
|
/* some parameter sanity checks */
|
|
assert(dev != NULL);
|
|
assert(params != NULL);
|
|
assert(params->exp < PCF857X_EXP_MAX);
|
|
#if IS_USED(MODULE_PCF857X_IRQ)
|
|
assert(gpio_is_valid(params->int_pin));
|
|
#endif
|
|
|
|
DEBUG_DEV("params=%p", dev, params);
|
|
|
|
/* init device data structure */
|
|
dev->params = *params;
|
|
|
|
switch (params->exp) {
|
|
#if IS_USED(MODULE_PCF8574)
|
|
/**< PCF8574 8 bit I/O expander used */
|
|
case PCF857X_EXP_PCF8574: dev->pin_num = PCF8574_GPIO_PIN_NUM;
|
|
dev->params.addr += PCF8574_BASE_ADDR;
|
|
break;
|
|
#endif
|
|
#if IS_USED(MODULE_PCF8574A)
|
|
/**< PCF8574A 8 bit I/O expander */
|
|
case PCF857X_EXP_PCF8574A: dev->pin_num = PCF8574A_GPIO_PIN_NUM;
|
|
dev->params.addr += PCF8574A_BASE_ADDR;
|
|
break;
|
|
#endif
|
|
#if IS_USED(MODULE_PCF8575)
|
|
/**< PCF8575 16 bit I/O expander */
|
|
case PCF857X_EXP_PCF8575: dev->pin_num = PCF8575_GPIO_PIN_NUM;
|
|
dev->params.addr += PCF8575_BASE_ADDR;
|
|
break;
|
|
#endif
|
|
default: return -ENOTSUP;
|
|
}
|
|
|
|
#if IS_USED(MODULE_PCF857X_IRQ)
|
|
/* initialize the IRQ event object used for delaying interrupts */
|
|
dev->irq_event.event.handler = _irq_handler;
|
|
dev->irq_event.dev = dev;
|
|
|
|
for (unsigned i = 0; i < dev->pin_num; i++) {
|
|
dev->isr[i].cb = NULL;
|
|
dev->isr[i].arg = NULL;
|
|
dev->enabled[i] = false;
|
|
}
|
|
|
|
/* initialize the interrupt pin */
|
|
if (gpio_init_int(dev->params.int_pin,
|
|
GPIO_IN_PU, GPIO_FALLING, _irq_isr, (void*)dev)) {
|
|
return -ENOSYS;
|
|
}
|
|
#endif /* MODULE_PCF857X_IRQ */
|
|
|
|
int res = PCF857X_OK;
|
|
|
|
_acquire(dev);
|
|
|
|
/* write 1 to all pins to switch them to INPUTS pulled up to HIGH */
|
|
dev->out = ~0;
|
|
res = _write(dev, dev->out);
|
|
|
|
if (!res) {
|
|
/* initial read all pins */
|
|
res = _read(dev, &dev->in);
|
|
|
|
/* set all pin modes to INPUT and set internal output data to 1 (HIGH) */
|
|
dev->modes = ~0;
|
|
}
|
|
|
|
_release(dev);
|
|
|
|
return res;
|
|
}
|
|
|
|
int pcf857x_gpio_init(pcf857x_t *dev, uint8_t pin, gpio_mode_t mode)
|
|
{
|
|
/* some parameter sanity checks */
|
|
assert(dev != NULL);
|
|
assert(pin < dev->pin_num);
|
|
|
|
DEBUG_DEV("pin=%u mode=%u", dev, pin, (unsigned)mode);
|
|
|
|
/*
|
|
* Since the LOW output is the only actively driven level possible with
|
|
* this expander, only in the case of GPIO_OUT we write a 0 to the pin
|
|
* to configure the pin as an output and actively drive it LOW. In all
|
|
* other modes, the pin is configured as an input and pulled-up to HIGH
|
|
* with the weak pull-up to emulate them.
|
|
*/
|
|
switch (mode) {
|
|
case GPIO_IN_PD: DEBUG_DEV("gpio mode GPIO_IN_PD not supported", dev);
|
|
return -EINVAL;
|
|
case GPIO_OUT: dev->modes &= ~(1 << pin); /* set mode bit to 0 */
|
|
dev->out &= ~(1 << pin); /* set output bit to 0 */
|
|
break;
|
|
default: dev->modes |= (1 << pin); /* set mode bit to 1 */
|
|
dev->out |= (1 << pin); /* set output bit to 1 */
|
|
break;
|
|
}
|
|
|
|
int res;
|
|
|
|
/* write the mode */
|
|
pcf857x_data_t data = dev->modes | dev->out;
|
|
_acquire(dev);
|
|
if ((res = _write(dev, data)) != PCF857X_OK) {
|
|
_release(dev);
|
|
return res;
|
|
}
|
|
|
|
#if IS_USED(MODULE_PCF857X_IRQ)
|
|
/* reset the callback in case the port used external interrupts before */
|
|
dev->isr[pin].cb = NULL;
|
|
dev->isr[pin].arg = NULL;
|
|
dev->enabled[pin] = false;
|
|
|
|
/*
|
|
* If an output of the expander is connected to an input of the same
|
|
* expander, there is no interrupt triggered by the input when the
|
|
* output changes.
|
|
* Therefore, we have to read input pins after the write operation to
|
|
* update the input pin state in the device data structure and to trigger
|
|
* an ISR if necessary.
|
|
*
|
|
* @note _update_state releases the bus.
|
|
*/
|
|
_update_state(dev);
|
|
#else
|
|
/* read to update the internal input state */
|
|
res = _read(dev, &dev->in);
|
|
_release(dev);
|
|
#endif
|
|
return res;
|
|
}
|
|
|
|
#if IS_USED(MODULE_PCF857X_IRQ)
|
|
int pcf857x_gpio_init_int(pcf857x_t *dev, uint8_t pin,
|
|
gpio_mode_t mode,
|
|
gpio_flank_t flank,
|
|
gpio_cb_t isr,
|
|
void *arg)
|
|
{
|
|
int res = PCF857X_OK;
|
|
|
|
/* initialize the pin */
|
|
if ((res = pcf857x_gpio_init(dev, pin, mode)) != PCF857X_OK) {
|
|
return res;
|
|
}
|
|
|
|
switch (flank) {
|
|
case GPIO_FALLING:
|
|
case GPIO_RISING:
|
|
case GPIO_BOTH: dev->isr[pin].cb = isr;
|
|
dev->isr[pin].arg = arg;
|
|
dev->flank[pin] = flank;
|
|
dev->enabled[pin] = true;
|
|
break;
|
|
default: DEBUG_DEV("invalid flank %d for pin %d", dev, flank, pin);
|
|
return -EINVAL;
|
|
}
|
|
|
|
return PCF857X_OK;
|
|
}
|
|
|
|
void pcf857x_gpio_irq_enable(pcf857x_t *dev, uint8_t pin)
|
|
{
|
|
/* some parameter sanity checks */
|
|
assert(dev != NULL);
|
|
assert(pin < dev->pin_num);
|
|
|
|
DEBUG_DEV("pin=%u", dev, pin);
|
|
dev->enabled[pin] = true;
|
|
}
|
|
|
|
void pcf857x_gpio_irq_disable(pcf857x_t *dev, uint8_t pin)
|
|
{
|
|
/* some parameter sanity checks */
|
|
assert(dev != NULL);
|
|
assert(pin < dev->pin_num);
|
|
|
|
DEBUG_DEV("pin=%u", dev, pin);
|
|
dev->enabled[pin] = false;
|
|
}
|
|
#endif
|
|
|
|
int pcf857x_gpio_read(pcf857x_t *dev, uint8_t pin)
|
|
{
|
|
/* some parameter sanity checks */
|
|
assert(dev != NULL);
|
|
assert(pin < dev->pin_num);
|
|
|
|
DEBUG_DEV("pin=%u", dev, pin);
|
|
|
|
/*
|
|
* If we use the interrupt, we always have an up-to-date input snapshot
|
|
* stored in the device data structure and which can be used directly.
|
|
* Otherwise we have to read the pins first.
|
|
*/
|
|
#if !IS_USED(MODULE_PCF857X_IRQ)
|
|
_acquire(dev);
|
|
_read(dev, &dev->in);
|
|
_release(dev);
|
|
#endif
|
|
return (dev->in &(1 << pin)) ? 1 : 0;
|
|
}
|
|
|
|
void pcf857x_gpio_write(pcf857x_t *dev, uint8_t pin, int value)
|
|
{
|
|
/* some parameter sanity checks */
|
|
assert(dev != NULL);
|
|
assert(pin < dev->pin_num);
|
|
|
|
DEBUG_DEV("pin=%u value=%d", dev, pin, value);
|
|
|
|
/* set pin bit value */
|
|
if (value) {
|
|
dev->out |= (1 << pin);
|
|
}
|
|
else {
|
|
dev->out &= ~(1 << pin);
|
|
}
|
|
|
|
/* update pin values */
|
|
pcf857x_data_t data = dev->modes | dev->out;
|
|
_acquire(dev);
|
|
_write(dev, data);
|
|
#if IS_USED(MODULE_PCF857X_IRQ)
|
|
/*
|
|
* If an output of the expander is connected to an input of the same
|
|
* expander, there is no interrupt triggered by the input when the
|
|
* output changes.
|
|
* Therefore, we have to read input pins after the write operation to
|
|
* update the input pin state in the device data structure and to trigger
|
|
* an ISR if necessary.
|
|
*
|
|
* @note _update_state releases the bus.
|
|
*/
|
|
_update_state(dev);
|
|
#else
|
|
_release(dev);
|
|
#endif
|
|
}
|
|
|
|
void pcf857x_gpio_clear(pcf857x_t *dev, uint8_t pin)
|
|
{
|
|
DEBUG_DEV("pin=%u", dev, pin);
|
|
return pcf857x_gpio_write(dev, pin, 0);
|
|
}
|
|
|
|
void pcf857x_gpio_set(pcf857x_t *dev, uint8_t pin)
|
|
{
|
|
DEBUG_DEV("pin=%u", dev, pin);
|
|
return pcf857x_gpio_write(dev, pin, 1);
|
|
}
|
|
|
|
void pcf857x_gpio_toggle(pcf857x_t *dev, uint8_t pin)
|
|
{
|
|
DEBUG_DEV("pin=%u", dev, pin);
|
|
return pcf857x_gpio_write(dev, pin, (dev->out & (1 << pin)) ? 0 : 1);
|
|
}
|
|
|
|
/** Functions for internal use only */
|
|
|
|
#if IS_USED(MODULE_PCF857X_IRQ)
|
|
|
|
/* interrupt service routine for IRQs */
|
|
static void _irq_isr(void *arg)
|
|
{
|
|
assert(arg != NULL);
|
|
|
|
/* just indicate that an interrupt occurred and return */
|
|
event_post(PCF857X_EVENT_PRIO, (event_t*)&((pcf857x_t*)arg)->irq_event);
|
|
}
|
|
|
|
/* handle one IRQ event of device referenced by the event */
|
|
static void _irq_handler(event_t* event)
|
|
{
|
|
pcf857x_irq_event_t* irq_event = (pcf857x_irq_event_t*)event;
|
|
|
|
assert(irq_event != NULL);
|
|
_acquire(irq_event->dev);
|
|
/* _update_state releases the bus */
|
|
_update_state(irq_event->dev);
|
|
}
|
|
|
|
/*
|
|
* @warning: It is expected that the I2C bus is already acquired when the
|
|
* function is called. However, it is released by this function
|
|
* before the function returns.
|
|
*/
|
|
static void _update_state(pcf857x_t* dev)
|
|
{
|
|
assert(dev != NULL);
|
|
DEBUG_DEV("", dev);
|
|
|
|
/* save old input values */
|
|
pcf857x_data_t old_in = dev->in;
|
|
pcf857x_data_t new_in;
|
|
|
|
/* read in new input values and release the bus */
|
|
if (_read(dev, &dev->in)) {
|
|
_release(dev);
|
|
return;
|
|
}
|
|
_release(dev);
|
|
|
|
new_in = dev->in;
|
|
|
|
/* iterate over all pins to check whether ISR has to be called */
|
|
for (unsigned i = 0; i < dev->pin_num; i++) {
|
|
pcf857x_data_t mask = 1 << i;
|
|
|
|
/*
|
|
* if pin is input, interrupt is enabled, has an ISR registered
|
|
* and the input value changed
|
|
*/
|
|
if (((dev->modes & mask) != 0) && dev->enabled[i] &&
|
|
(dev->isr[i].cb != NULL) && ((old_in ^ new_in) & mask)) {
|
|
/* check for the flank and the activated flank mode */
|
|
if ((dev->flank[i] == GPIO_BOTH) || /* no matter what flank */
|
|
((new_in & mask) == 0 && /* falling flank */
|
|
(dev->flank[i] == GPIO_FALLING)) ||
|
|
((new_in & mask) == mask && /* rising flank */
|
|
(dev->flank[i] == GPIO_RISING))) {
|
|
|
|
/* call the ISR */
|
|
dev->isr[i].cb(dev->isr[i].arg);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
#endif /* MODULE_PCF857X_IRQ */
|
|
|
|
static inline void _acquire(const pcf857x_t *dev)
|
|
{
|
|
assert(dev != NULL);
|
|
i2c_acquire(dev->params.dev);
|
|
}
|
|
|
|
static inline void _release(const pcf857x_t *dev)
|
|
{
|
|
assert(dev != NULL);
|
|
i2c_release(dev->params.dev);
|
|
}
|
|
|
|
static int _read(const pcf857x_t *dev, pcf857x_data_t *data)
|
|
{
|
|
assert(dev != NULL);
|
|
assert(data != NULL);
|
|
|
|
uint8_t bytes[2];
|
|
size_t len = (dev->pin_num == 8) ? 1 : 2;
|
|
|
|
int res = i2c_read_bytes(dev->params.dev, dev->params.addr, bytes, len, 0);
|
|
|
|
if (res != 0) {
|
|
DEBUG_DEV("could not read data, reason %d (%s)",
|
|
dev, res, strerror(res * -1));
|
|
return res;
|
|
}
|
|
|
|
if (dev->pin_num == 8) {
|
|
*data = bytes[0];
|
|
DEBUG_DEV("data=%02x", dev, *data);
|
|
}
|
|
else {
|
|
*data = (bytes[1] << 8) | bytes[0];
|
|
DEBUG_DEV("data=%04x", dev, *data);
|
|
}
|
|
|
|
return PCF857X_OK;
|
|
}
|
|
|
|
static int _write(const pcf857x_t *dev, pcf857x_data_t data)
|
|
{
|
|
assert(dev != NULL);
|
|
|
|
uint8_t bytes[2];
|
|
size_t len;
|
|
|
|
if (dev->pin_num == 8) {
|
|
DEBUG_DEV("data=%02x", dev, data & 0xff);
|
|
|
|
bytes[0] = data & 0xff;
|
|
len = 1;
|
|
}
|
|
else {
|
|
DEBUG_DEV("data=%04x", dev, data);
|
|
|
|
bytes[0] = data & 0xff;
|
|
bytes[1] = data >> 8;
|
|
len = 2;
|
|
}
|
|
|
|
int res = i2c_write_bytes(dev->params.dev, dev->params.addr, bytes, len, 0);
|
|
|
|
if (res != 0) {
|
|
DEBUG_DEV("could not write data, reason %d (%s)",
|
|
dev, res, strerror(res * -1));
|
|
return res;
|
|
}
|
|
|
|
return PCF857X_OK;
|
|
}
|