2016-01-23 14:38:51 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2016 Leon George
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2018-06-18 20:34:08 +02:00
|
|
|
* @ingroup cpu_cc26xx_cc13xx
|
2017-06-22 15:43:17 +02:00
|
|
|
* @ingroup drivers_periph_gpio
|
2016-01-23 14:38:51 +01:00
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Low-level GPIO driver implementation
|
|
|
|
*
|
|
|
|
* @author Leon M. George <leon@georgemail.eu>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "cpu.h"
|
|
|
|
#include "periph/gpio.h"
|
|
|
|
|
|
|
|
#define DOE_SHIFT (29U)
|
|
|
|
|
2018-09-21 08:17:26 +02:00
|
|
|
#ifdef MODULE_PERIPH_GPIO_IRQ
|
|
|
|
#define GPIO_ISR_CHAN_NUMOF (32)
|
|
|
|
|
2016-01-23 14:38:51 +01:00
|
|
|
/**
|
|
|
|
* @brief static callback memory
|
|
|
|
*/
|
|
|
|
static gpio_isr_ctx_t gpio_chan[GPIO_ISR_CHAN_NUMOF];
|
2018-10-09 14:52:21 +02:00
|
|
|
#endif /* MODULE_PERIPH_GPIO_IRQ */
|
2016-01-23 14:38:51 +01:00
|
|
|
|
|
|
|
int gpio_init(gpio_t pin, gpio_mode_t mode)
|
|
|
|
{
|
2016-06-01 16:40:28 +02:00
|
|
|
if ((unsigned int)pin > 31)
|
2016-01-23 14:38:51 +01:00
|
|
|
return -1;
|
|
|
|
|
2020-03-13 07:23:25 +01:00
|
|
|
/* enable peripherals power domain */
|
|
|
|
if (!power_is_domain_enabled(POWER_DOMAIN_PERIPHERALS)) {
|
|
|
|
power_enable_domain(POWER_DOMAIN_PERIPHERALS);
|
|
|
|
}
|
|
|
|
|
2016-01-23 14:38:51 +01:00
|
|
|
/* enable GPIO clock */
|
|
|
|
PRCM->GPIOCLKGR |= 1;
|
|
|
|
PRCM->CLKLOADCTL |= CLKLOADCTL_LOAD;
|
|
|
|
while (!(PRCM->CLKLOADCTL & CLKLOADCTL_LOADDONE)) ;
|
|
|
|
|
|
|
|
/* configure the GPIO mode */
|
|
|
|
IOC->CFG[pin] = mode;
|
|
|
|
GPIO->DOE &= ~(1 << pin);
|
|
|
|
GPIO->DOE |= ((~(mode >> DOE_SHIFT) & 0x1) << pin);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int gpio_read(gpio_t pin)
|
|
|
|
{
|
|
|
|
if (GPIO->DOE & (1 << pin)) {
|
|
|
|
return (GPIO->DOUT >> pin) & 1;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return (GPIO->DIN >> pin) & 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void gpio_set(gpio_t pin)
|
|
|
|
{
|
|
|
|
GPIO->DOUTSET = (1 << pin);
|
|
|
|
}
|
|
|
|
|
|
|
|
void gpio_clear(gpio_t pin)
|
|
|
|
{
|
|
|
|
GPIO->DOUTCLR = (1 << pin);
|
|
|
|
}
|
|
|
|
|
|
|
|
void gpio_toggle(gpio_t pin)
|
|
|
|
{
|
|
|
|
GPIO->DOUTTGL = (1 << pin);
|
|
|
|
}
|
|
|
|
|
|
|
|
void gpio_write(gpio_t pin, int value)
|
|
|
|
{
|
|
|
|
if (value) {
|
|
|
|
GPIO->DOUTSET = (1 << pin);
|
|
|
|
} else {
|
|
|
|
GPIO->DOUTCLR = (1 << pin);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-21 08:17:26 +02:00
|
|
|
#ifdef MODULE_PERIPH_GPIO_IRQ
|
|
|
|
int gpio_init_int(gpio_t pin, gpio_mode_t mode, gpio_flank_t flank,
|
|
|
|
gpio_cb_t cb, void *arg)
|
|
|
|
{
|
|
|
|
int init = gpio_init(pin, mode);
|
|
|
|
if (init != 0)
|
|
|
|
return init;
|
|
|
|
|
|
|
|
NVIC_EnableIRQ(EDGE_DETECT_IRQN);
|
|
|
|
|
|
|
|
IOC->CFG[pin] |= flank;
|
|
|
|
|
|
|
|
gpio_chan[pin].cb = cb;
|
|
|
|
gpio_chan[pin].arg = arg;
|
|
|
|
|
|
|
|
/* clears the interrupt flag */
|
|
|
|
GPIO->EVFLAGS |= (1 << pin);
|
|
|
|
|
|
|
|
gpio_irq_enable(pin);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void gpio_irq_enable(gpio_t pin)
|
|
|
|
{
|
|
|
|
IOC->CFG[pin] |= IOCFG_EDGEIRQ_ENABLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
void gpio_irq_disable(gpio_t pin)
|
|
|
|
{
|
|
|
|
IOC->CFG[pin] &= ~IOCFG_EDGEIRQ_ENABLE;
|
|
|
|
}
|
|
|
|
|
2016-01-23 14:38:51 +01:00
|
|
|
void isr_edge(void)
|
|
|
|
{
|
|
|
|
for (unsigned pin = 0; pin < GPIO_ISR_CHAN_NUMOF; pin++) {
|
|
|
|
/* doc claims EVFLAGS will only be set for pins that have edge detection enabled */
|
2018-06-18 20:34:08 +02:00
|
|
|
if (GPIO->EVFLAGS & (1UL << pin)) {
|
|
|
|
GPIO->EVFLAGS |= (1UL << pin);
|
2016-01-23 14:38:51 +01:00
|
|
|
gpio_chan[pin].cb(gpio_chan[pin].arg);
|
|
|
|
}
|
|
|
|
}
|
2016-11-30 18:26:05 +01:00
|
|
|
cortexm_isr_end();
|
2016-01-23 14:38:51 +01:00
|
|
|
}
|
2018-10-09 14:52:21 +02:00
|
|
|
#endif /* MODULE_PERIPH_GPIO_IRQ */
|