2015-06-09 12:45:35 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2015 Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2017-06-22 15:43:17 +02:00
|
|
|
* @ingroup cpu_lpc2387
|
|
|
|
* @ingroup drivers_periph_gpio
|
2015-06-09 12:45:35 +02:00
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief CPU specific low-level GPIO driver implementation for the LPC2387
|
|
|
|
*
|
|
|
|
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <assert.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "irq.h"
|
|
|
|
#include "periph/gpio.h"
|
|
|
|
#include "bitfield.h"
|
|
|
|
|
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
|
|
|
|
2018-09-21 08:19:15 +02:00
|
|
|
#ifdef MODULE_PERIPH_GPIO_IRQ
|
2015-06-09 12:45:35 +02:00
|
|
|
#define GPIO_NUM_ISR (16)
|
|
|
|
|
|
|
|
static BITFIELD(_gpio_config_bitfield, GPIO_NUM_ISR);
|
|
|
|
|
2016-01-27 17:00:37 +01:00
|
|
|
static gpio_isr_ctx_t _gpio_states[GPIO_NUM_ISR];
|
2015-06-09 12:45:35 +02:00
|
|
|
static BITFIELD(_gpio_rising, GPIO_NUM_ISR);
|
|
|
|
static BITFIELD(_gpio_falling, GPIO_NUM_ISR);
|
|
|
|
static uint8_t _gpio_isr_map[64]; /* only ports 0+2 can have ISRs */
|
|
|
|
|
|
|
|
static void _gpio_configure(gpio_t pin, unsigned rising, unsigned falling);
|
|
|
|
|
2020-04-12 21:01:46 +02:00
|
|
|
static inline int _isr_map_entry2(unsigned port, unsigned pin)
|
|
|
|
{
|
|
|
|
return pin + (port ? 32 : 0);
|
|
|
|
}
|
|
|
|
|
2015-06-09 12:45:35 +02:00
|
|
|
static int _isr_map_entry(gpio_t pin) {
|
|
|
|
unsigned _pin = pin & 31;
|
|
|
|
unsigned port = pin >> 5;
|
|
|
|
|
|
|
|
/* lpc2387 can only interrupt in pins of port 0 and 2 */
|
|
|
|
if (port && port != 2) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2020-04-12 21:01:46 +02:00
|
|
|
return _isr_map_entry2(port, _pin);
|
2015-06-09 12:45:35 +02:00
|
|
|
}
|
2018-10-09 14:58:26 +02:00
|
|
|
#endif /* MODULE_PERIPH_GPIO_IRQ */
|
2018-09-21 08:19:15 +02:00
|
|
|
|
|
|
|
void gpio_init_ports(void) {
|
|
|
|
SCS |= 0x1; /* set GPIO ports 0 and 1 to FIO mode (3.7.2) */
|
|
|
|
#ifdef MODULE_PERIPH_GPIO_IRQ
|
|
|
|
memset(&_gpio_isr_map[0], 0xff, 64);
|
2018-10-09 14:58:26 +02:00
|
|
|
#endif /* MODULE_PERIPH_GPIO_IRQ */
|
2018-09-21 08:19:15 +02:00
|
|
|
}
|
2015-06-09 12:45:35 +02:00
|
|
|
|
2016-02-20 14:11:08 +01:00
|
|
|
int gpio_init(gpio_t pin, gpio_mode_t mode)
|
2015-06-09 12:45:35 +02:00
|
|
|
{
|
|
|
|
unsigned _pin = pin & 31;
|
|
|
|
unsigned port = pin >> 5;
|
|
|
|
FIO_PORT_t *_port = &FIO_PORTS[port];
|
|
|
|
|
|
|
|
/* set direction */
|
2018-06-18 09:10:25 +02:00
|
|
|
switch (mode){
|
|
|
|
case GPIO_OUT:
|
|
|
|
_port->DIR |= 1<<_pin;
|
|
|
|
break;
|
|
|
|
case GPIO_IN:
|
|
|
|
_port->DIR &= ~(1<<_pin);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return -1;
|
|
|
|
}
|
2015-06-09 12:45:35 +02:00
|
|
|
|
|
|
|
gpio_init_mux(pin, 0);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2015-11-28 14:18:29 +01:00
|
|
|
int gpio_init_mux(unsigned pin, unsigned mux)
|
|
|
|
{
|
|
|
|
(void) mux;
|
2018-06-18 09:10:25 +02:00
|
|
|
unsigned pos = (pin & 0xf) << 1;
|
|
|
|
PINSEL[pin>>4] &= ~(0x3 << pos);
|
2015-06-09 12:45:35 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-09-21 08:19:15 +02:00
|
|
|
int gpio_read(gpio_t pin)
|
|
|
|
{
|
|
|
|
unsigned _pin = pin & 31;
|
|
|
|
unsigned port = pin >> 5;
|
|
|
|
FIO_PORT_t *_port = &FIO_PORTS[port];
|
|
|
|
return (_port->PIN & (1 << _pin)) != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void gpio_set(gpio_t pin)
|
|
|
|
{
|
|
|
|
unsigned _pin = pin & 31;
|
|
|
|
unsigned port = pin >> 5;
|
|
|
|
FIO_PORT_t *_port = &FIO_PORTS[port];
|
|
|
|
_port->SET = 1 << _pin;
|
|
|
|
}
|
|
|
|
|
|
|
|
void gpio_clear(gpio_t pin)
|
|
|
|
{
|
|
|
|
unsigned _pin = pin & 31;
|
|
|
|
unsigned port = pin >> 5;
|
|
|
|
FIO_PORT_t *_port = &FIO_PORTS[port];
|
|
|
|
_port->CLR = 1 << _pin;
|
|
|
|
}
|
|
|
|
|
|
|
|
void gpio_toggle(gpio_t dev)
|
|
|
|
{
|
|
|
|
if (gpio_read(dev)) {
|
|
|
|
gpio_clear(dev);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
gpio_set(dev);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void gpio_write(gpio_t dev, int value)
|
|
|
|
{
|
|
|
|
if (value) {
|
|
|
|
gpio_set(dev);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
gpio_clear(dev);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef MODULE_PERIPH_GPIO_IRQ
|
|
|
|
static void _gpio_configure(gpio_t pin, unsigned rising, unsigned falling)
|
|
|
|
{
|
|
|
|
unsigned _pin = pin & 31;
|
|
|
|
unsigned port = pin >> 5;
|
|
|
|
|
|
|
|
/* set irq settings */
|
|
|
|
volatile unsigned long *en_f;
|
|
|
|
volatile unsigned long *en_r;
|
|
|
|
volatile unsigned long *en_clr;
|
|
|
|
|
|
|
|
if (!port) {
|
|
|
|
en_f = &IO0_INT_EN_F;
|
|
|
|
en_r = &IO0_INT_EN_R;
|
|
|
|
en_clr = &IO0_INT_CLR;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
en_f = &IO2_INT_EN_F;
|
|
|
|
en_r = &IO2_INT_EN_R;
|
|
|
|
en_clr = &IO2_INT_CLR;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* configure irq */
|
|
|
|
unsigned int bit = 0x1 << _pin;
|
|
|
|
|
|
|
|
unsigned state = irq_disable();
|
|
|
|
|
|
|
|
*en_clr |= bit; /* clear interrupt */
|
|
|
|
|
|
|
|
if (falling) {
|
|
|
|
*en_f |= bit; /* enable falling edge */
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
*en_f &= ~bit; /* disable falling edge */
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rising) {
|
|
|
|
*en_r |= bit; /* enable rising edge */
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
*en_r &= ~bit; /* disable rising edge */
|
|
|
|
}
|
|
|
|
|
|
|
|
irq_restore(state);
|
|
|
|
}
|
|
|
|
|
2016-02-20 14:11:08 +01:00
|
|
|
int gpio_init_int(gpio_t pin, gpio_mode_t mode, gpio_flank_t flank,
|
2018-06-18 09:10:25 +02:00
|
|
|
gpio_cb_t cb, void *arg)
|
2015-06-09 12:45:35 +02:00
|
|
|
{
|
2016-02-20 14:11:08 +01:00
|
|
|
(void)mode;
|
|
|
|
|
2015-06-09 12:45:35 +02:00
|
|
|
DEBUG("gpio_init_int(): pin %u\n", pin);
|
|
|
|
int isr_map_entry;
|
|
|
|
|
|
|
|
/* check if interrupt is possible for this pin */
|
|
|
|
if ((isr_map_entry = _isr_map_entry(pin)) == -1) {
|
|
|
|
DEBUG("gpio_init_int(): pin %u cannot be used to generate interrupts.\n", pin);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* find free isr state entry */
|
|
|
|
int _state_index = _gpio_isr_map[isr_map_entry];
|
|
|
|
|
|
|
|
if (_state_index == 0xff) {
|
|
|
|
_state_index = bf_get_unset(_gpio_config_bitfield, GPIO_NUM_ISR);
|
|
|
|
_gpio_isr_map[isr_map_entry] = _state_index;
|
|
|
|
DEBUG("gpio_init_int(): pin has state_index=%i isr_map_entry=%i\n", _state_index, isr_map_entry);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_state_index == 0xff) {
|
2015-09-12 12:43:15 +02:00
|
|
|
#ifdef DEVELHELP
|
2015-06-09 12:45:35 +02:00
|
|
|
puts("lpc2387: gpio: warning: no free gpio callback state!");
|
|
|
|
#endif
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* store callback */
|
|
|
|
_gpio_states[_state_index].cb = cb;
|
|
|
|
_gpio_states[_state_index].arg = arg;
|
|
|
|
|
|
|
|
extern void GPIO_IRQHandler(void);
|
|
|
|
|
|
|
|
if (flank & GPIO_FALLING) {
|
|
|
|
bf_set(_gpio_falling, _state_index);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
bf_unset(_gpio_falling, _state_index);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (flank & GPIO_RISING) {
|
|
|
|
bf_set(_gpio_rising, _state_index);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
bf_unset(_gpio_rising, _state_index);
|
|
|
|
}
|
|
|
|
|
|
|
|
_gpio_configure(pin, flank & GPIO_RISING, flank & GPIO_FALLING);
|
|
|
|
|
|
|
|
/* activate irq */
|
|
|
|
INTWAKE |= GPIO0WAKE | GPIO2WAKE; /* allow GPIO to wake up from power down */
|
|
|
|
install_irq(GPIO_INT, &GPIO_IRQHandler, IRQP_GPIO); /* install irq handler */
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void gpio_irq_enable(gpio_t pin)
|
|
|
|
{
|
|
|
|
int isr_map_entry =_isr_map_entry(pin);
|
|
|
|
int _state_index = _gpio_isr_map[isr_map_entry];
|
|
|
|
|
|
|
|
if (_state_index == 0xff) {
|
|
|
|
DEBUG("gpio_irq_enable(): trying to enable unconfigured pin.\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
_gpio_configure(pin,
|
|
|
|
bf_isset(_gpio_rising, _state_index),
|
|
|
|
bf_isset(_gpio_falling, _state_index));
|
|
|
|
}
|
|
|
|
|
|
|
|
void gpio_irq_disable(gpio_t dev)
|
|
|
|
{
|
|
|
|
_gpio_configure(dev, 0, 0);
|
|
|
|
}
|
|
|
|
|
2020-04-12 21:06:56 +02:00
|
|
|
static void test_irq(int port, unsigned long active_pins)
|
2015-06-09 12:45:35 +02:00
|
|
|
{
|
|
|
|
/* Test each bit of rising and falling masks, if set trigger interrupt
|
|
|
|
* on corresponding device */
|
2020-04-12 21:01:46 +02:00
|
|
|
|
|
|
|
while (active_pins) {
|
|
|
|
/* we want the position of the first one bit, so N_bits - (N_leading_zeros + 1) */
|
|
|
|
unsigned pin = 32 - __builtin_clz(active_pins) - 1;
|
|
|
|
|
|
|
|
/* get the index of the configured interrupt */
|
|
|
|
int _state_index = _gpio_isr_map[_isr_map_entry2(port, pin)];
|
|
|
|
|
|
|
|
/* check if interrupt is configured */
|
|
|
|
if (_state_index != 0xff) {
|
|
|
|
_gpio_states[_state_index].cb(_gpio_states[_state_index].arg);
|
2015-06-09 12:45:35 +02:00
|
|
|
}
|
|
|
|
|
2020-04-12 21:01:46 +02:00
|
|
|
/* clear bit */
|
|
|
|
active_pins &= ~(1 << pin);
|
2015-06-09 12:45:35 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void GPIO_IRQHandler(void) __attribute__((interrupt("IRQ")));
|
|
|
|
|
|
|
|
void GPIO_IRQHandler(void)
|
|
|
|
{
|
2020-04-12 21:06:56 +02:00
|
|
|
unsigned long int_stat;
|
2015-06-09 12:45:35 +02:00
|
|
|
|
2020-04-12 21:06:56 +02:00
|
|
|
if (IO_INT_STAT & BIT0) { /* interrupt(s) on PORT0 pending */
|
|
|
|
int_stat = IO0_INT_STAT_F | IO0_INT_STAT_R; /* get risen & fallen pin IRQs */
|
|
|
|
IO0_INT_CLR = int_stat; /* clear IRQ flags */
|
|
|
|
test_irq(0, int_stat);
|
2015-06-09 12:45:35 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (IO_INT_STAT & BIT2) { /* interrupt(s) on PORT2 pending */
|
2020-04-12 21:06:56 +02:00
|
|
|
int_stat = IO2_INT_STAT_F | IO2_INT_STAT_R; /* get risen & fallen pin IRQs */
|
|
|
|
IO2_INT_CLR = int_stat; /* clear IRQ flags */
|
|
|
|
test_irq(2, int_stat);
|
2015-06-09 12:45:35 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
VICVectAddr = 0; /* Acknowledge Interrupt */
|
|
|
|
}
|
2018-10-09 14:58:26 +02:00
|
|
|
#endif /* MODULE_PERIPH_GPIO_IRQ */
|