1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/cpu/kinetis_common/gpio.c

458 lines
11 KiB
C
Raw Normal View History

/*
* Copyright (C) 2014 Freie Universität Berlin
* Copyright (C) 2014 PHYTEC Messtechnik GmbH
* Copyright (C) 2014 Eistec AB
*
* 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_kinetis_common_gpio
*
* @{
*
* @file
* @brief Low-level GPIO driver implementation
*
* @author Hauke Petersen <mail@haukepetersen.de>
* @author Johann Fischer <j.fischer@phytec.de>
* @author Jonas Remmert <j.remmert@phytec.de>
* @author Joakim Nohlgård <joakim.nohlgard@eistec.se>
*
* @}
*/
#include "cpu.h"
#include "sched.h"
#include "thread.h"
#include "utlist.h"
#include "mutex.h"
#include "periph/gpio.h"
#include "periph_conf.h"
#define ENABLE_DEBUG (0)
#include "debug.h"
#ifndef PIN_MUX_FUNCTION_ANALOG
#define PIN_MUX_FUNCTION_ANALOG 0
#endif
#ifndef PIN_MUX_FUNCTION_GPIO
#define PIN_MUX_FUNCTION_GPIO 1
#endif
/**
* @brief Linked list entry for interrupt configurations.
*/
typedef struct gpio_int_config_entry {
struct gpio_int_config_entry* next; /**< pointer to next entry */
gpio_cb_t cb; /**< callback called from GPIO interrupt */
void *arg; /**< argument passed to the callback */
uint32_t irqc; /**< remember interrupt configuration between disable/enable */
uint8_t pin; /**< pin number within the port */
} gpio_int_config_entry_t;
/* Linked list of interrupt handlers for each port.
* Using a linked list saves memory when less than 80% of all GPIO pins on the
* CPU are configured for interrupts, which is true for (almost) all real world
* applications. */
static gpio_int_config_entry_t* gpio_interrupts[PORT_NUMOF];
static mutex_t int_config_lock = MUTEX_INIT;
/* Maximum number of simultaneously enabled GPIO interrupts. Each pool entry
* uses 20 bytes of RAM.
*/
#ifndef GPIO_INT_POOL_SIZE
#define GPIO_INT_POOL_SIZE 16
#endif
/* Pool of linked list entries which can be used by any port configuration.
* Rationale: Avoid dynamic memory inside low level periph drivers. */
gpio_int_config_entry_t config_pool[GPIO_INT_POOL_SIZE];
static PORT_Type * const _port_ptrs[] = PORT_BASE_PTRS;
static GPIO_Type * const _gpio_ptrs[] = GPIO_BASE_PTRS;
static inline uint32_t _port_num(gpio_t dev) {
return (uint32_t)((dev & GPIO_PORT_MASK) >> GPIO_PORT_SHIFT);
}
static inline uint8_t _pin_num(gpio_t dev) {
return (uint8_t)((dev & GPIO_PIN_MASK) >> GPIO_PIN_SHIFT);
}
static inline PORT_Type *_port(gpio_t dev) {
return _port_ptrs[_port_num(dev)];
}
static inline GPIO_Type *_gpio(gpio_t dev) {
return _gpio_ptrs[_port_num(dev)];
}
static void _clear_interrupt_config(gpio_t dev) {
gpio_int_config_entry_t* entry = NULL;
uint8_t pin_number = _pin_num(dev);
mutex_lock(&int_config_lock);
/* Search for the given pin in the port's interrupt configuration */
LL_SEARCH_SCALAR(gpio_interrupts[_port_num(dev)], entry, pin, pin_number);
if (entry != NULL) {
LL_DELETE(gpio_interrupts[_port_num(dev)], entry);
/* pin == 0 means the entry is available */
entry->pin = 0;
}
mutex_unlock(&int_config_lock);
}
static gpio_int_config_entry_t* _allocate_interrupt_config(uint8_t port) {
gpio_int_config_entry_t* ret = NULL;
mutex_lock(&int_config_lock);
for (uint8_t i = 0; i < GPIO_INT_POOL_SIZE; ++i) {
if (config_pool[i].pin == 0) {
/* temporarily set pin to something non-zero until the proper pin
* number is set by the init code */
config_pool[i].pin = 200;
ret = &config_pool[i];
LL_PREPEND(gpio_interrupts[port], ret);
break;
}
}
mutex_unlock(&int_config_lock);
return ret;
}
int gpio_init(gpio_t dev, gpio_dir_t dir, gpio_pp_t pushpull)
{
switch (_port_num(dev)) {
#ifdef PORTA_BASE
case PORT_A:
PORTA_CLOCK_GATE = 1;
break;
#endif
#ifdef PORTB_BASE
case PORT_B:
PORTB_CLOCK_GATE = 1;
break;
#endif
#ifdef PORTC_BASE
case PORT_C:
PORTC_CLOCK_GATE = 1;
break;
#endif
#ifdef PORTD_BASE
case PORT_D:
PORTD_CLOCK_GATE = 1;
break;
#endif
#ifdef PORTE_BASE
case PORT_E:
PORTE_CLOCK_GATE = 1;
break;
#endif
#ifdef PORTF_BASE
case PORT_F:
PORTF_CLOCK_GATE = 1;
break;
#endif
#ifdef PORTG_BASE
case PORT_G:
PORTG_CLOCK_GATE = 1;
break;
#endif
default:
return -1;
}
uint8_t pin = _pin_num(dev);
PORT_Type *port = _port(dev);
GPIO_Type *gpio = _gpio(dev);
_clear_interrupt_config(dev);
/* Reset all pin control settings for the pin */
/* Switch to analog input function while fiddling with the settings, to be safe. */
port->PCR[pin] = PORT_PCR_MUX(PIN_MUX_FUNCTION_ANALOG);
/* set to push-pull configuration */
switch (pushpull) {
case GPIO_PULLUP:
port->PCR[pin] |= PORT_PCR_PE_MASK | PORT_PCR_PS_MASK; /* Pull enable, pull up */
break;
case GPIO_PULLDOWN:
port->PCR[pin] |= PORT_PCR_PE_MASK; /* Pull enable, !pull up */
break;
default:
break;
}
if (dir == GPIO_DIR_OUT) {
BITBAND_REG32(gpio->PDDR, pin) = 1; /* set pin to output mode */
gpio->PCOR = GPIO_PCOR_PTCO(1 << pin); /* set output to low */
}
else {
BITBAND_REG32(gpio->PDDR, pin) = 0; /* set pin to input mode */
}
/* Select GPIO function for the pin */
port->PCR[pin] |= PORT_PCR_MUX(PIN_MUX_FUNCTION_GPIO);
return 0;
}
int gpio_init_int(gpio_t dev, gpio_pp_t pushpull, gpio_flank_t flank, gpio_cb_t cb, void *arg)
{
int res;
IRQn_Type irqn;
res = gpio_init(dev, GPIO_DIR_IN, pushpull);
if (res < 0) {
return res;
}
switch (_port_num(dev)) {
#ifdef PORTA_BASE
case PORT_A:
irqn = PORTA_IRQn;
break;
#endif
#ifdef PORTB_BASE
case PORT_B:
irqn = PORTB_IRQn;
break;
#endif
#ifdef PORTC_BASE
case PORT_C:
irqn = PORTC_IRQn;
break;
#endif
#ifdef PORTD_BASE
case PORT_D:
irqn = PORTD_IRQn;
break;
#endif
#ifdef PORTE_BASE
case PORT_E:
irqn = PORTE_IRQn;
break;
#endif
#ifdef PORTF_BASE
case PORT_F:
irqn = PORTF_IRQn;
break;
#endif
#ifdef PORTG_BASE
case PORT_G:
irqn = PORTG_IRQn;
break;
#endif
default:
return -1;
}
uint32_t irqc;
/* configure the active edges */
switch (flank) {
case GPIO_RISING:
irqc = PORT_PCR_IRQC(PIN_INTERRUPT_RISING);
break;
case GPIO_FALLING:
irqc = PORT_PCR_IRQC(PIN_INTERRUPT_FALLING);
break;
case GPIO_BOTH:
irqc = PORT_PCR_IRQC(PIN_INTERRUPT_EDGE);
break;
default:
/* Unknown setting */
return -1;
}
gpio_int_config_entry_t* config = _allocate_interrupt_config(_port_num(dev));
if (config == NULL) {
/* No free interrupt config entries */
return -1;
}
/* Enable port interrupts in the NVIC */
NVIC_SetPriority(irqn, GPIO_IRQ_PRIO);
NVIC_EnableIRQ(irqn);
uint8_t pin = _pin_num(dev);
PORT_Type *port = _port(dev);
config->cb = cb;
config->arg = arg;
config->irqc = irqc;
/* Allow the callback to be found by the IRQ handler by setting the proper
* pin number */
config->pin = pin;
port->PCR[pin] &= ~(PORT_PCR_IRQC_MASK); /* Disable interrupt */
BITBAND_REG32(port->PCR[pin], PORT_PCR_ISF_SHIFT) = 1; /* Clear interrupt flag */
port->PCR[pin] |= config->irqc; /* Enable interrupt */
return 0;
}
void gpio_irq_enable(gpio_t dev)
{
/* Restore saved state */
PORT_Type *port = _port(dev);
gpio_int_config_entry_t* entry = NULL;
uint8_t pin_number = _pin_num(dev);
mutex_lock(&int_config_lock);
/* Search for the given pin in the port's interrupt configuration */
LL_SEARCH_SCALAR(gpio_interrupts[_port_num(dev)], entry, pin, pin_number);
uint32_t irqc = entry->irqc;
mutex_unlock(&int_config_lock);
if (entry == NULL) {
/* Pin has not been configured for interrupts */
return;
}
port->PCR[pin_number] &= ~(PORT_PCR_IRQC_MASK);
port->PCR[pin_number] |= irqc;
}
void gpio_irq_disable(gpio_t dev)
{
/* Save irqc state before disabling to allow enabling with the same trigger
* settings later. */
PORT_Type *port = _port(dev);
uint8_t pin_number = _pin_num(dev);
uint32_t irqc = PORT_PCR_IRQC_MASK & port->PCR[pin_number];
gpio_int_config_entry_t* entry = NULL;
mutex_lock(&int_config_lock);
/* Search for the given pin in the port's interrupt configuration */
LL_SEARCH_SCALAR(gpio_interrupts[_port_num(dev)], entry, pin, pin_number);
if (entry == NULL) {
/* Pin has not been configured for interrupts */
mutex_unlock(&int_config_lock);
return;
}
entry->irqc = irqc;
mutex_unlock(&int_config_lock);
port->PCR[pin_number] &= ~(PORT_PCR_IRQC_MASK);
}
int gpio_read(gpio_t dev)
{
return ((_gpio(dev)->PDIR & GPIO_PDIR_PDI(1 << _pin_num(dev))) ? 1 : 0);
}
void gpio_set(gpio_t dev)
{
_gpio(dev)->PSOR = (1 << _pin_num(dev));
}
void gpio_clear(gpio_t dev)
{
_gpio(dev)->PCOR = (1 << _pin_num(dev));
}
void gpio_toggle(gpio_t dev)
{
_gpio(dev)->PTOR = (1 << _pin_num(dev));
}
void gpio_write(gpio_t dev, int value)
{
if (value) {
_gpio(dev)->PSOR = (1 << _pin_num(dev));
}
else {
_gpio(dev)->PCOR = (1 << _pin_num(dev));
}
}
static inline void irq_handler(uint8_t port_num)
{
gpio_int_config_entry_t *entry;
PORT_Type *port = _port_ptrs[port_num];
uint32_t isf = port->ISFR; /* Interrupt status flags */
LL_FOREACH(gpio_interrupts[port_num], entry) {
if (isf & (1 << entry->pin)) {
if (entry->cb != NULL) {
entry->cb(entry->arg);
}
}
}
/* Clear interrupt flags */
port->ISFR = isf;
if (sched_context_switch_request) {
thread_yield();
}
}
#ifdef PORTA_BASE
void isr_porta(void)
{
irq_handler(PORT_A);
}
#endif /* PORTA_BASE */
#ifdef PORTB_BASE
void isr_portb(void)
{
irq_handler(PORT_B);
}
#endif /* ISR_PORT_B */
#ifdef PORTC_BASE
void isr_portc(void)
{
irq_handler(PORT_C);
}
#endif /* ISR_PORT_C */
#ifdef PORTD_BASE
void isr_portd(void)
{
irq_handler(PORT_D);
}
#endif /* ISR_PORT_D */
#ifdef PORTE_BASE
void isr_porte(void)
{
irq_handler(PORT_E);
}
#endif /* ISR_PORT_E */
#ifdef PORTF_BASE
void isr_portf(void)
{
irq_handler(PORT_F);
}
#endif /* ISR_PORT_F */
#ifdef PORTG_BASE
void isr_portg(void)
{
irq_handler(PORT_G);
}
#endif /* ISR_PORT_G */