1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2024-12-29 04:50:03 +01:00

cpu/nrf5x: allow multiple I2C and SPI buses on a shared periph

This commit is contained in:
Benjamin Valentin 2022-08-19 15:32:09 +02:00
parent 14b5eca22d
commit 7b7eb9d17b
6 changed files with 84 additions and 23 deletions

View File

@ -268,8 +268,10 @@ void spi_twi_irq_register_i2c(NRF_TWIM_Type *bus,
* @brief Acquire the shared I2C/SPI peripheral in I2C mode * @brief Acquire the shared I2C/SPI peripheral in I2C mode
* *
* @param bus bus to acquire exclusive access on * @param bus bus to acquire exclusive access on
* @param cb ISR handler to call on IRQ
* @param arg ISR handler argument
*/ */
void nrf5x_i2c_acquire(NRF_TWIM_Type *bus); void nrf5x_i2c_acquire(NRF_TWIM_Type *bus, spi_twi_irq_cb_t cb, void *arg);
/** /**
* @brief Release the shared I2C/SPI peripheral in I2C mode * @brief Release the shared I2C/SPI peripheral in I2C mode
@ -281,16 +283,18 @@ void nrf5x_i2c_release(NRF_TWIM_Type *bus);
/** /**
* @brief Acquire the shared I2C/SPI peripheral in SPI mode * @brief Acquire the shared I2C/SPI peripheral in SPI mode
* *
* @param bus bus to acquire exclusive access on * @param bus bus to release exclusive access on
* @param cb ISR handler to call on IRQ
* @param arg ISR handler argument
*/ */
void nrf5x_spi_release(NRF_SPIM_Type *bus); void nrf5x_spi_acquire(NRF_SPIM_Type *bus, spi_twi_irq_cb_t cb, void *arg);
/** /**
* @brief Acquire the shared I2C/SPI peripheral in SPI mode * @brief Acquire the shared I2C/SPI peripheral in SPI mode
* *
* @param bus bus to release exclusive access on * @param bus bus to release exclusive access on
*/ */
void nrf5x_spi_acquire(NRF_SPIM_Type *bus); void nrf5x_spi_release(NRF_SPIM_Type *bus);
/** /**
* @brief USBDEV buffers must be word aligned because of DMA restrictions * @brief USBDEV buffers must be word aligned because of DMA restrictions

View File

@ -140,19 +140,26 @@ void spi_twi_irq_register_i2c(NRF_TWIM_Type *bus,
_irq[num] = cb; _irq[num] = cb;
_irq_arg[num] = arg; _irq_arg[num] = arg;
NVIC_EnableIRQ(_isr[num]); NVIC_EnableIRQ(_isr[num]);
} }
void nrf5x_i2c_acquire(NRF_TWIM_Type *bus) void nrf5x_i2c_acquire(NRF_TWIM_Type *bus,
spi_twi_irq_cb_t cb, void *arg)
{ {
size_t num = _i2c_dev2num(bus); size_t num = _i2c_dev2num(bus);
mutex_lock(&_locks[num]); mutex_lock(&_locks[num]);
_irq[num] = cb;
_irq_arg[num] = arg;
} }
void nrf5x_spi_acquire(NRF_SPIM_Type *bus) void nrf5x_spi_acquire(NRF_SPIM_Type *bus,
spi_twi_irq_cb_t cb, void *arg)
{ {
size_t num = _spi_dev2num(bus); size_t num = _spi_dev2num(bus);
mutex_lock(&_locks[num]); mutex_lock(&_locks[num]);
_irq[num] = cb;
_irq_arg[num] = arg;
} }
void nrf5x_i2c_release(NRF_TWIM_Type *bus) void nrf5x_i2c_release(NRF_TWIM_Type *bus)

View File

@ -102,8 +102,18 @@ static void _init_pins(i2c_t dev)
{ {
gpio_init(i2c_config[dev].scl, GPIO_IN_OD_PU); gpio_init(i2c_config[dev].scl, GPIO_IN_OD_PU);
gpio_init(i2c_config[dev].sda, GPIO_IN_OD_PU); gpio_init(i2c_config[dev].sda, GPIO_IN_OD_PU);
}
/* Beware: This needs to be kept in sync with the SPI version of this.
* Specifically, when registers are configured that are valid to the peripheral
* in both SPI and I2C mode, the register needs to be configured in both the I2C
* and the SPI variant of _setup_shared_peripheral() to avoid from parameters
* leaking from one bus into the other */
static void _setup_shared_peripheral(i2c_t dev)
{
bus(dev)->PSEL.SCL = i2c_config[dev].scl; bus(dev)->PSEL.SCL = i2c_config[dev].scl;
bus(dev)->PSEL.SDA = i2c_config[dev].sda; bus(dev)->PSEL.SDA = i2c_config[dev].sda;
bus(dev)->FREQUENCY = i2c_config[dev].speed;
} }
void i2c_init(i2c_t dev) void i2c_init(i2c_t dev)
@ -111,7 +121,6 @@ void i2c_init(i2c_t dev)
assert(dev < I2C_NUMOF); assert(dev < I2C_NUMOF);
/* Initialize mutex */ /* Initialize mutex */
mutex_init(&locks[dev]);
mutex_init(&busy[dev]); mutex_init(&busy[dev]);
mutex_lock(&busy[dev]); mutex_lock(&busy[dev]);
@ -122,8 +131,8 @@ void i2c_init(i2c_t dev)
/* configure pins */ /* configure pins */
_init_pins(dev); _init_pins(dev);
/* configure dev clock speed */ /* configure shared periphal speed */
bus(dev)->FREQUENCY = i2c_config[dev].speed; _setup_shared_peripheral(dev);
spi_twi_irq_register_i2c(bus(dev), i2c_isr_handler, (void *)(uintptr_t)dev); spi_twi_irq_register_i2c(bus(dev), i2c_isr_handler, (void *)(uintptr_t)dev);
@ -158,7 +167,13 @@ void i2c_acquire(i2c_t dev)
{ {
assert(dev < I2C_NUMOF); assert(dev < I2C_NUMOF);
mutex_lock(&locks[dev]); if (IS_USED(MODULE_PERIPH_I2C_RECONFIGURE)) {
mutex_lock(&locks[dev]);
}
nrf5x_i2c_acquire(bus(dev), i2c_isr_handler, (void *)(uintptr_t)dev);
_setup_shared_peripheral(dev);
bus(dev)->ENABLE = TWIM_ENABLE_ENABLE_Enabled; bus(dev)->ENABLE = TWIM_ENABLE_ENABLE_Enabled;
DEBUG("[i2c] acquired dev %i\n", (int)dev); DEBUG("[i2c] acquired dev %i\n", (int)dev);
@ -169,7 +184,12 @@ void i2c_release(i2c_t dev)
assert(dev < I2C_NUMOF); assert(dev < I2C_NUMOF);
bus(dev)->ENABLE = TWIM_ENABLE_ENABLE_Disabled; bus(dev)->ENABLE = TWIM_ENABLE_ENABLE_Disabled;
mutex_unlock(&locks[dev]);
if (IS_USED(MODULE_PERIPH_I2C_RECONFIGURE)) {
mutex_unlock(&locks[dev]);
}
nrf5x_i2c_release(bus(dev));
DEBUG("[i2c] released dev %i\n", (int)dev); DEBUG("[i2c] released dev %i\n", (int)dev);
} }

View File

@ -123,16 +123,28 @@ static void _clear_workaround(spi_t bus)
#endif #endif
} }
/* Beware: This needs to be kept in sync with the I2C version of this.
* Specifically, when registers are configured that are valid to the peripheral
* in both SPI and I2C mode, the register needs to be configured in both the I2C
* and the SPI variant of _setup_shared_peripheral() to avoid from parameters
* leaking from one bus into the other */
static void _setup_shared_peripheral(spi_t bus)
{
SPI_SCKSEL = spi_config[bus].sclk;
SPI_MOSISEL = spi_config[bus].mosi;
SPI_MISOSEL = spi_config[bus].miso;
}
void spi_init(spi_t bus) void spi_init(spi_t bus)
{ {
assert(bus < SPI_NUMOF); assert(bus < SPI_NUMOF);
/* initialize mutex */ /* initialize mutex */
mutex_init(&locks[bus]);
mutex_init(&busy[bus]); mutex_init(&busy[bus]);
mutex_lock(&busy[bus]); mutex_lock(&busy[bus]);
/* initialize pins */ /* initialize pins */
spi_init_pins(bus); spi_init_pins(bus);
_setup_shared_peripheral(bus);
} }
int spi_init_with_gpio_mode(spi_t bus, const spi_gpio_mode_t* mode) int spi_init_with_gpio_mode(spi_t bus, const spi_gpio_mode_t* mode)
@ -165,9 +177,6 @@ void spi_init_pins(spi_t bus)
spi_init_with_gpio_mode(bus, &gpio_modes); spi_init_with_gpio_mode(bus, &gpio_modes);
/* select pins for the SPI device */ /* select pins for the SPI device */
SPI_SCKSEL = spi_config[bus].sclk;
SPI_MOSISEL = spi_config[bus].mosi;
SPI_MISOSEL = spi_config[bus].miso;
_setup_workaround_for_ftpan_58(bus); _setup_workaround_for_ftpan_58(bus);
spi_twi_irq_register_spi(dev(bus), spi_isr_handler, (void *)(uintptr_t)bus); spi_twi_irq_register_spi(dev(bus), spi_isr_handler, (void *)(uintptr_t)bus);
} }
@ -177,7 +186,13 @@ void spi_acquire(spi_t bus, spi_cs_t cs, spi_mode_t mode, spi_clk_t clk)
(void)cs; (void)cs;
assert((unsigned)bus < SPI_NUMOF); assert((unsigned)bus < SPI_NUMOF);
mutex_lock(&locks[bus]); if (IS_USED(MODULE_PERIPH_SPI_RECONFIGURE)) {
mutex_lock(&locks[bus]);
}
nrf5x_spi_acquire(dev(bus), spi_isr_handler, (void *)(uintptr_t)bus);
_setup_shared_peripheral(bus);
/* configure bus */ /* configure bus */
dev(bus)->CONFIG = mode; dev(bus)->CONFIG = mode;
dev(bus)->FREQUENCY = clk; dev(bus)->FREQUENCY = clk;
@ -189,7 +204,12 @@ void spi_release(spi_t bus)
{ {
/* power off everything */ /* power off everything */
dev(bus)->ENABLE = 0; dev(bus)->ENABLE = 0;
mutex_unlock(&locks[bus]);
if (IS_USED(MODULE_PERIPH_SPI_RECONFIGURE)) {
mutex_unlock(&locks[bus]);
}
nrf5x_spi_release(dev(bus));
} }
static size_t _transfer(spi_t bus, const uint8_t *out_buf, uint8_t *in_buf, static size_t _transfer(spi_t bus, const uint8_t *out_buf, uint8_t *in_buf,

View File

@ -212,8 +212,10 @@ void spi_twi_irq_register_i2c(NRF_TWIM_Type *bus,
* @brief Acquire the shared I2C/SPI peripheral in I2C mode * @brief Acquire the shared I2C/SPI peripheral in I2C mode
* *
* @param bus bus to acquire exclusive access on * @param bus bus to acquire exclusive access on
* @param cb ISR handler to call on IRQ
* @param arg ISR handler argument
*/ */
void nrf5x_i2c_acquire(NRF_TWIM_Type *bus); void nrf5x_i2c_acquire(NRF_TWIM_Type *bus, spi_twi_irq_cb_t cb, void *arg);
/** /**
* @brief Release the shared I2C/SPI peripheral in I2C mode * @brief Release the shared I2C/SPI peripheral in I2C mode
@ -225,16 +227,18 @@ void nrf5x_i2c_release(NRF_TWIM_Type *bus);
/** /**
* @brief Acquire the shared I2C/SPI peripheral in SPI mode * @brief Acquire the shared I2C/SPI peripheral in SPI mode
* *
* @param bus bus to acquire exclusive access on * @param bus bus to release exclusive access on
* @param cb ISR handler to call on IRQ
* @param arg ISR handler argument
*/ */
void nrf5x_spi_release(NRF_SPIM_Type *bus); void nrf5x_spi_acquire(NRF_SPIM_Type *bus, spi_twi_irq_cb_t cb, void *arg);
/** /**
* @brief Acquire the shared I2C/SPI peripheral in SPI mode * @brief Acquire the shared I2C/SPI peripheral in SPI mode
* *
* @param bus bus to release exclusive access on * @param bus bus to release exclusive access on
*/ */
void nrf5x_spi_acquire(NRF_SPIM_Type *bus); void nrf5x_spi_release(NRF_SPIM_Type *bus);
#ifdef __cplusplus #ifdef __cplusplus
} }

View File

@ -91,16 +91,22 @@ void spi_twi_irq_register_i2c(NRF_TWIM_Type *bus,
NVIC_EnableIRQ(_isr[num]); NVIC_EnableIRQ(_isr[num]);
} }
void nrf5x_i2c_acquire(NRF_TWIM_Type *bus) void nrf5x_i2c_acquire(NRF_TWIM_Type *bus,
spi_twi_irq_cb_t cb, void *arg)
{ {
size_t num = _i2c_dev2num(bus); size_t num = _i2c_dev2num(bus);
mutex_lock(&_locks[num]); mutex_lock(&_locks[num]);
_irq[num] = cb;
_irq_arg[num] = arg;
} }
void nrf5x_spi_acquire(NRF_SPIM_Type *bus) void nrf5x_spi_acquire(NRF_SPIM_Type *bus,
spi_twi_irq_cb_t cb, void *arg)
{ {
size_t num = _spi_dev2num(bus); size_t num = _spi_dev2num(bus);
mutex_lock(&_locks[num]); mutex_lock(&_locks[num]);
_irq[num] = cb;
_irq_arg[num] = arg;
} }
void nrf5x_i2c_release(NRF_TWIM_Type *bus) void nrf5x_i2c_release(NRF_TWIM_Type *bus)