1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-15 20:32:43 +01:00
RIOT/cpu/rpx0xx/periph/gpio.c
Marian Buschsieweke ea56dfc3ff
cpu/rpx0xx: add support for the RP2040 MCU
Co-authored-by: Fabian Hüßler <fabian.huessler@st.ovgu.de>
2021-07-14 12:41:20 +02:00

236 lines
6.9 KiB
C

/*
* Copyright (C) 2021 Otto-von-Guericke Universität Magdeburg
*
* 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 cpu_rpx0xx
* @ingroup drivers_periph_gpio
* @{
*
* @file
* @brief GPIO driver implementation for the RP2040
*
* @author Marian Buschsieweke <marian.buschsieweke@ovgu.de>
*
* @}
*/
#include <errno.h>
#include "bitarithm.h"
#include "board.h"
#include "irq.h"
#include "periph/gpio.h"
#include "periph_conf.h"
#include "periph_cpu.h"
#include "io_reg.h"
#define ENABLE_DEBUG 0
#include "debug.h"
#define GPIO_PIN_NUMOF 30U
#ifdef MODULE_PERIPH_GPIO_IRQ
static gpio_cb_t _cbs[GPIO_PIN_NUMOF];
static void *_args[GPIO_PIN_NUMOF];
#endif /* MODULE_PERIPH_GPIO_IRQ */
int gpio_init(gpio_t pin, gpio_mode_t mode)
{
assert(pin < GPIO_PIN_NUMOF);
SIO->GPIO_OE_CLR.reg = 1LU << pin;
SIO->GPIO_OUT_CLR.reg = 1LU << pin;
switch (mode) {
case GPIO_IN:
{
gpio_pad_ctrl_t pad_config = {
.input_enable = 1,
.schmitt_trig_enable = 1
};
gpio_io_ctrl_t io_config = {
.function_select = FUNCTION_SELECT_SIO,
};
gpio_set_pad_config(pin, pad_config);
gpio_set_io_config(pin, io_config);
}
break;
case GPIO_IN_PD:
{
gpio_pad_ctrl_t pad_config = {
.pull_down_enable = 1,
.input_enable = 1,
.schmitt_trig_enable = 1
};
gpio_io_ctrl_t io_config = {
.function_select = FUNCTION_SELECT_SIO,
};
gpio_set_pad_config(pin, pad_config);
gpio_set_io_config(pin, io_config);
}
break;
case GPIO_IN_PU:
{
gpio_pad_ctrl_t pad_config = {
.pull_up_enable = 1,
.input_enable = 1,
.schmitt_trig_enable = 1
};
gpio_io_ctrl_t io_config = {
.function_select = FUNCTION_SELECT_SIO,
};
gpio_set_pad_config(pin, pad_config);
gpio_set_io_config(pin, io_config);
}
break;
case GPIO_OUT:
{
gpio_pad_ctrl_t pad_config = {
.drive_strength = DRIVE_STRENGTH_12MA,
};
gpio_io_ctrl_t io_config = {
.function_select = FUNCTION_SELECT_SIO,
};
gpio_set_pad_config(pin, pad_config);
gpio_set_io_config(pin, io_config);
}
SIO->GPIO_OE_SET.reg = 1LU << pin;
break;
default:
return -ENOTSUP;
}
return 0;
}
int gpio_read(gpio_t pin)
{
if (SIO->GPIO_OE.reg & (1LU << pin)) {
/* pin is output: */
return SIO->GPIO_OUT.reg & (1LU << pin);
}
/* pin is input: */
return SIO->GPIO_IN.reg & (1LU << pin);
}
void gpio_set(gpio_t pin)
{
SIO->GPIO_OUT_SET.reg = 1LU << pin;
}
void gpio_clear(gpio_t pin)
{
SIO->GPIO_OUT_CLR.reg = 1LU << pin;
}
void gpio_toggle(gpio_t pin)
{
SIO->GPIO_OUT_XOR.reg = 1LU << pin;
}
void gpio_write(gpio_t pin, int value)
{
if (value) {
gpio_set(pin);
}
else {
gpio_clear(pin);
}
}
#ifdef MODULE_PERIPH_GPIO_IRQ
static void _irq_enable(gpio_t pin, unsigned flank)
{
volatile uint32_t *irq_enable_regs = &IO_BANK0->PROC0_INTE0.reg;
volatile uint32_t *irq_ack_regs = &IO_BANK0->INTR0.reg;
/* There are 4 bits to control IRQs per pin, hence the configuration is split across multiple
* I/O registers. The following calculates the position the four bits matching the given pin,
* where idx refers to the I/O register and shift_amount to the position in the I/O register.
*/
unsigned shift_amount = (pin & 0x7) << 2;
unsigned idx = pin >> 3;
/* make access atomic by disabling IRQs */
unsigned irq_state = irq_disable();
uint32_t value = irq_enable_regs[idx];
/* clear any stray IRQs */
io_reg_atomic_clear(&irq_ack_regs[idx], 0xFLU << shift_amount);
/* first, clear previous setting */
value &= ~(0xFLU << shift_amount);
/* then, apply new setting */
value |= flank << shift_amount;
irq_enable_regs[idx] = value;
irq_restore(irq_state);
NVIC_EnableIRQ(IO_IRQ_BANK0_IRQn);
}
void gpio_irq_enable(gpio_t pin)
{
io_reg_atomic_clear(gpio_io_register(pin), IO_BANK0_GPIO1_CTRL_IRQOVER_Msk);
}
void gpio_irq_disable(gpio_t pin)
{
/* Beware: The two-bit IRQOVER value needs to be set to 0b10 == IRQ_OVERRIDE_LOW. This
* implementation will set IRQOVER only to either 0b00 == IRQ_OVERRIDE_NORMAL or
* 0b10 == IRQ_OVERRIDE_LOW. If we just set the most significant bit, this will result in
* IRQOVER set to IRQ_OVERRIDE_LOW.
*
* IRQOVER must not be set by user code for this to work, though.
*/
io_reg_atomic_set(gpio_io_register(pin), IRQ_OVERRIDE_LOW << IO_BANK0_GPIO1_CTRL_IRQOVER_Pos);
}
int gpio_init_int(gpio_t pin, gpio_mode_t mode, gpio_flank_t flank, gpio_cb_t cb, void *arg)
{
assert(pin < GPIO_PIN_NUMOF);
int retval = gpio_init(pin, mode);
if (retval) {
return retval;
}
_cbs[pin] = cb;
_args[pin] = arg;
_irq_enable(pin, flank);
return 0;
}
void isr_io_bank0(void)
{
unsigned offset = 0;
volatile uint32_t *irq_status_regs = &IO_BANK0->PROC0_INTS0.reg;
volatile uint32_t *irq_ack_regs = &IO_BANK0->INTR0.reg;
DEBUG("[rp0x00] GPIO IRQ mask: %08x, %08x, %08x, %08x\n",
(unsigned)IO_BANK0->PROC0_INTE0.reg, (unsigned)IO_BANK0->PROC0_INTE1.reg,
(unsigned)IO_BANK0->PROC0_INTE2.reg, (unsigned)IO_BANK0->PROC0_INTE3.reg);
DEBUG("[rp0x00] GPIO IRQ status: %08x, %08x, %08x, %08x\n",
(unsigned)IO_BANK0->PROC0_INTS0.reg, (unsigned)IO_BANK0->PROC0_INTS1.reg,
(unsigned)IO_BANK0->PROC0_INTS2.reg, (unsigned)IO_BANK0->PROC0_INTS3.reg);
/* There are four IRQ status bits per pin, so there is info for 8 pins per I/O register.
* We will iterate over all IRQ status I/O registers in the outer loop, and over all 8 pins
* per register in the inner loop */
for (unsigned i = 0; i < (GPIO_PIN_NUMOF + 7) / 8; i++, offset += 8) {
unsigned status = irq_status_regs[i];
irq_ack_regs[i] = status;
for (unsigned pin = 0; pin < 8; pin++) {
/* Note: Not use bitarithm_test_and_clear() here, since two bits for a single pin could
* be set. This happens if both rising and falling flank (1 bit used to track each) have
* occurred until the ISR started to handle the first IRQ.
*/
if (status & (0xFLU << (pin << 2))) {
if (_cbs[pin + offset]) {
_cbs[pin + offset](_args[pin + offset]);
}
}
}
}
cortexm_isr_end();
}
#endif /* MODULE_PERIPH_GPIO_IRQ */