mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
135 lines
2.6 KiB
C
135 lines
2.6 KiB
C
/*
|
|
* 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.
|
|
*/
|
|
|
|
/**
|
|
* @ingroup cpu_cc26x0
|
|
* @{
|
|
*
|
|
* @file
|
|
* @brief Low-level GPIO driver implementation
|
|
*
|
|
* @author Leon M. George <leon@georgemail.eu>
|
|
*
|
|
* @}
|
|
*/
|
|
|
|
#include "cpu.h"
|
|
#include "sched.h"
|
|
#include "thread.h"
|
|
#include "periph/gpio.h"
|
|
|
|
#define GPIO_ISR_CHAN_NUMOF (32)
|
|
|
|
#define DOE_SHIFT (29U)
|
|
|
|
/**
|
|
* @brief static callback memory
|
|
*/
|
|
static gpio_isr_ctx_t gpio_chan[GPIO_ISR_CHAN_NUMOF];
|
|
|
|
int gpio_init(gpio_t pin, gpio_mode_t mode)
|
|
{
|
|
if ((pin < 0) || (pin > 31))
|
|
return -1;
|
|
|
|
/* enable GPIO clock */
|
|
PRCM->PDCTL0 |= PDCTL0_PERIPH_ON;
|
|
while(!(PRCM->PDSTAT0 & PDSTAT0_PERIPH_ON)) ;
|
|
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);
|
|
GPIO->DOUTCLR = (1 << pin);
|
|
|
|
return 0;
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
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);
|
|
}
|
|
}
|
|
|
|
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 */
|
|
if (GPIO->EVFLAGS & (1 << pin)) {
|
|
GPIO->EVFLAGS |= (1 << pin);
|
|
gpio_chan[pin].cb(gpio_chan[pin].arg);
|
|
}
|
|
}
|
|
if (sched_context_switch_request) {
|
|
thread_yield();
|
|
}
|
|
}
|