2014-09-09 11:12:39 +02:00
|
|
|
/*
|
2015-06-03 18:23:01 +02:00
|
|
|
* Copyright (C) 2014-2015 Freie Universität Berlin
|
2014-09-09 11:12:39 +02:00
|
|
|
*
|
|
|
|
* 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 driver_periph
|
|
|
|
* @{
|
|
|
|
*
|
2015-05-22 07:34:41 +02:00
|
|
|
* @file
|
2014-09-09 11:12:39 +02:00
|
|
|
* @brief Low-level GPIO driver implementation
|
|
|
|
*
|
|
|
|
* @author Troels Hoffmeyer <troels.d.hoffmeyer@gmail.com>
|
|
|
|
* @author Thomas Eichinger <thomas.eichinger@fu-berlin.de>
|
2015-06-03 18:23:01 +02:00
|
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
2014-09-09 11:12:39 +02:00
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "cpu.h"
|
2015-06-03 18:23:01 +02:00
|
|
|
#include "sched.h"
|
|
|
|
#include "thread.h"
|
2014-09-09 11:12:39 +02:00
|
|
|
#include "periph/gpio.h"
|
|
|
|
#include "periph_conf.h"
|
|
|
|
|
2014-11-11 16:24:26 +01:00
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
2014-09-09 11:12:39 +02:00
|
|
|
|
2015-06-03 18:23:01 +02:00
|
|
|
/**
|
|
|
|
* @brief Number of external interrupt lines
|
|
|
|
*/
|
|
|
|
#define NUMOF_IRQS (16U)
|
|
|
|
|
2016-02-20 15:41:04 +01:00
|
|
|
/**
|
|
|
|
* @brief Mask to get PINCFG reg value from mode value
|
|
|
|
*/
|
|
|
|
#define MODE_PINCFG_MASK (0x06)
|
|
|
|
|
2015-06-03 18:23:01 +02:00
|
|
|
/**
|
|
|
|
* @brief Mapping of pins to EXTI lines, -1 means not EXTI possible
|
|
|
|
*/
|
|
|
|
static const int8_t exti_config[2][32] = {
|
|
|
|
{-1, 1, -1, -1, 4, 5, 6, 7, -1, 9, 10, 11, 12, 13, 14, 15,
|
2016-04-14 14:23:02 +02:00
|
|
|
-1, 1, 2, 3, -1, -1, 6, 7, 12, 13, -1, 15, 8, -1, 10, 11},
|
2015-06-03 18:23:01 +02:00
|
|
|
{ 0, -1, 2, 3, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
2016-04-19 20:46:41 +02:00
|
|
|
0, 1, -1, -1, -1, -1, 6, 7, -1, -1, -1, -1, 8, -1, -1, -1},
|
2015-06-03 18:23:01 +02:00
|
|
|
};
|
2014-09-09 11:12:39 +02:00
|
|
|
|
2015-06-03 18:23:01 +02:00
|
|
|
/**
|
|
|
|
* @brief Hold one interrupt context per interrupt line
|
|
|
|
*/
|
2016-01-27 17:00:37 +01:00
|
|
|
static gpio_isr_ctx_t gpio_config[NUMOF_IRQS];
|
2014-09-09 11:12:39 +02:00
|
|
|
|
2015-06-03 18:23:01 +02:00
|
|
|
static inline PortGroup *_port(gpio_t pin)
|
2014-09-09 11:12:39 +02:00
|
|
|
{
|
2015-06-03 18:23:01 +02:00
|
|
|
return (PortGroup *)(pin & ~(0x1f));
|
|
|
|
}
|
2014-09-09 11:12:39 +02:00
|
|
|
|
2015-06-03 18:23:01 +02:00
|
|
|
static inline int _pin_pos(gpio_t pin)
|
|
|
|
{
|
|
|
|
return (pin & 0x1f);
|
|
|
|
}
|
2014-09-09 11:12:39 +02:00
|
|
|
|
2015-06-03 18:23:01 +02:00
|
|
|
static inline int _pin_mask(gpio_t pin)
|
|
|
|
{
|
|
|
|
return (1 << _pin_pos(pin));
|
2014-09-09 11:12:39 +02:00
|
|
|
}
|
|
|
|
|
2015-06-03 18:23:01 +02:00
|
|
|
static int _exti(gpio_t pin)
|
2014-09-09 11:12:39 +02:00
|
|
|
{
|
2015-06-03 18:23:01 +02:00
|
|
|
int port_num = ((pin >> 7) & 0x03);
|
2014-09-09 11:12:39 +02:00
|
|
|
|
2015-06-03 18:23:01 +02:00
|
|
|
if (port_num > 1) {
|
2014-10-17 09:18:36 +02:00
|
|
|
return -1;
|
|
|
|
}
|
2015-06-03 18:23:01 +02:00
|
|
|
return exti_config[port_num][_pin_pos(pin)];
|
|
|
|
}
|
|
|
|
|
2016-02-08 19:04:28 +01:00
|
|
|
void gpio_init_mux(gpio_t pin, gpio_mux_t mux)
|
2015-06-03 18:23:01 +02:00
|
|
|
{
|
|
|
|
PortGroup* port = _port(pin);
|
|
|
|
int pin_pos = _pin_pos(pin);
|
|
|
|
|
|
|
|
port->PINCFG[pin_pos].reg |= PORT_PINCFG_PMUXEN;
|
|
|
|
port->PMUX[pin_pos >> 1].reg &= ~(0xf << (4 * (pin_pos & 0x1)));
|
|
|
|
port->PMUX[pin_pos >> 1].reg |= (mux << (4 * (pin_pos & 0x1)));
|
|
|
|
}
|
2014-10-17 09:18:36 +02:00
|
|
|
|
2016-02-20 15:41:04 +01:00
|
|
|
int gpio_init(gpio_t pin, gpio_mode_t mode)
|
2015-06-03 18:23:01 +02:00
|
|
|
{
|
|
|
|
PortGroup* port = _port(pin);
|
|
|
|
int pin_pos = _pin_pos(pin);
|
|
|
|
int pin_mask = _pin_mask(pin);
|
2014-09-09 11:12:39 +02:00
|
|
|
|
2016-02-20 15:41:04 +01:00
|
|
|
/* make sure pin mode is applicable */
|
|
|
|
if (mode > 0x7) {
|
|
|
|
return -1;
|
2014-09-09 11:12:39 +02:00
|
|
|
}
|
2016-02-20 15:41:04 +01:00
|
|
|
|
|
|
|
/* set pin direction */
|
|
|
|
if (mode & 0x2) {
|
|
|
|
port->DIRCLR.reg = pin_mask;
|
2015-06-03 18:23:01 +02:00
|
|
|
}
|
|
|
|
else {
|
2016-02-20 15:41:04 +01:00
|
|
|
port->DIRSET.reg = pin_mask;
|
2015-06-03 18:23:01 +02:00
|
|
|
}
|
2016-02-20 15:41:04 +01:00
|
|
|
|
|
|
|
/* configure the pin cfg and clear output register */
|
|
|
|
port->PINCFG[pin_pos].reg = (mode & MODE_PINCFG_MASK);
|
|
|
|
port->OUTCLR.reg = pin_mask;
|
|
|
|
|
|
|
|
/* and set pull-up/pull-down if applicable */
|
|
|
|
if (mode == 0x7) {
|
|
|
|
port->OUTSET.reg = pin_mask;
|
|
|
|
}
|
|
|
|
|
2014-09-09 11:12:39 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-02-20 15:41:04 +01:00
|
|
|
int gpio_init_int(gpio_t pin, gpio_mode_t mode, gpio_flank_t flank,
|
2015-06-03 18:23:01 +02:00
|
|
|
gpio_cb_t cb, void *arg)
|
2014-09-09 11:12:39 +02:00
|
|
|
{
|
2015-06-03 18:23:01 +02:00
|
|
|
int exti = _exti(pin);
|
2014-09-09 11:12:39 +02:00
|
|
|
|
2015-06-03 18:23:01 +02:00
|
|
|
/* make sure EIC channel is valid */
|
|
|
|
if (exti == -1) {
|
2014-10-17 09:18:36 +02:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2015-06-03 18:23:01 +02:00
|
|
|
/* save callback */
|
|
|
|
gpio_config[exti].cb = cb;
|
|
|
|
gpio_config[exti].arg = arg;
|
2015-10-31 17:28:16 +01:00
|
|
|
/* configure pin as input and set MUX to peripheral function A */
|
2016-02-20 15:41:04 +01:00
|
|
|
gpio_init(pin, mode);
|
2015-06-03 18:23:01 +02:00
|
|
|
gpio_init_mux(pin, GPIO_MUX_A);
|
|
|
|
/* enable clocks for the EIC module */
|
2014-09-09 11:12:39 +02:00
|
|
|
PM->APBAMASK.reg |= PM_APBAMASK_EIC;
|
2015-09-17 15:57:51 +02:00
|
|
|
GCLK->CLKCTRL.reg = (EIC_GCLK_ID |
|
|
|
|
GCLK_CLKCTRL_CLKEN |
|
|
|
|
GCLK_CLKCTRL_GEN_GCLK0);
|
2016-02-11 14:25:02 +01:00
|
|
|
while (GCLK->STATUS.bit.SYNCBUSY) {}
|
2015-06-03 18:23:01 +02:00
|
|
|
/* configure the active flank */
|
|
|
|
EIC->CONFIG[exti >> 3].reg &= ~(0xf << ((exti & 0x7) * 4));
|
|
|
|
EIC->CONFIG[exti >> 3].reg |= (flank << ((exti & 0x7) * 4));
|
|
|
|
/* enable the global EIC interrupt */
|
2014-09-09 11:12:39 +02:00
|
|
|
NVIC_EnableIRQ(EIC_IRQn);
|
2015-06-03 18:23:01 +02:00
|
|
|
/* clear interrupt flag and enable the interrupt line and line wakeup */
|
|
|
|
EIC->INTFLAG.reg = (1 << exti);
|
|
|
|
EIC->WAKEUP.reg |= (1 << exti);
|
|
|
|
EIC->INTENSET.reg = (1 << exti);
|
|
|
|
/* enable the EIC module*/
|
|
|
|
EIC->CTRL.reg = EIC_CTRL_ENABLE;
|
2016-02-11 14:25:02 +01:00
|
|
|
while (EIC->STATUS.reg & EIC_STATUS_SYNCBUSY) {}
|
2014-09-09 11:12:39 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2015-06-03 18:23:01 +02:00
|
|
|
void gpio_irq_enable(gpio_t pin)
|
2014-09-09 11:12:39 +02:00
|
|
|
{
|
2015-06-03 18:23:01 +02:00
|
|
|
int exti = _exti(pin);
|
|
|
|
if (exti == -1) {
|
|
|
|
return;
|
2014-09-09 11:12:39 +02:00
|
|
|
}
|
2015-06-03 18:23:01 +02:00
|
|
|
EIC->INTENSET.reg = (1 << exti);
|
2014-09-09 11:12:39 +02:00
|
|
|
}
|
|
|
|
|
2015-06-03 18:23:01 +02:00
|
|
|
void gpio_irq_disable(gpio_t pin)
|
2014-09-09 11:12:39 +02:00
|
|
|
{
|
2015-06-03 18:23:01 +02:00
|
|
|
int exti = _exti(pin);
|
|
|
|
if (exti == -1) {
|
|
|
|
return;
|
2014-09-09 11:12:39 +02:00
|
|
|
}
|
2015-06-03 18:23:01 +02:00
|
|
|
EIC->INTENCLR.reg = (1 << exti);
|
2014-09-09 11:12:39 +02:00
|
|
|
}
|
|
|
|
|
2015-06-03 18:23:01 +02:00
|
|
|
int gpio_read(gpio_t pin)
|
2014-09-09 11:12:39 +02:00
|
|
|
{
|
2015-06-03 18:23:01 +02:00
|
|
|
PortGroup *port = _port(pin);
|
|
|
|
int mask = _pin_mask(pin);
|
2014-09-09 11:12:39 +02:00
|
|
|
|
2015-06-03 18:23:01 +02:00
|
|
|
if (port->DIR.reg & mask) {
|
|
|
|
return (port->OUT.reg & mask) ? 1 : 0;
|
2014-09-09 11:12:39 +02:00
|
|
|
}
|
2015-06-03 18:23:01 +02:00
|
|
|
else {
|
|
|
|
return (port->IN.reg & mask) ? 1 : 0;
|
2014-09-09 11:12:39 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-03 18:23:01 +02:00
|
|
|
void gpio_set(gpio_t pin)
|
2014-09-09 11:12:39 +02:00
|
|
|
{
|
2015-06-03 18:23:01 +02:00
|
|
|
_port(pin)->OUTSET.reg = _pin_mask(pin);
|
2014-09-09 11:12:39 +02:00
|
|
|
}
|
|
|
|
|
2015-06-03 18:23:01 +02:00
|
|
|
void gpio_clear(gpio_t pin)
|
2014-09-09 11:12:39 +02:00
|
|
|
{
|
2015-06-03 18:23:01 +02:00
|
|
|
_port(pin)->OUTCLR.reg = _pin_mask(pin);
|
2014-09-09 11:12:39 +02:00
|
|
|
}
|
|
|
|
|
2015-06-03 18:23:01 +02:00
|
|
|
void gpio_toggle(gpio_t pin)
|
2014-09-09 11:12:39 +02:00
|
|
|
{
|
2015-06-03 18:23:01 +02:00
|
|
|
_port(pin)->OUTTGL.reg = _pin_mask(pin);
|
2014-09-09 11:12:39 +02:00
|
|
|
}
|
|
|
|
|
2015-06-03 18:23:01 +02:00
|
|
|
void gpio_write(gpio_t pin, int value)
|
2014-09-09 11:12:39 +02:00
|
|
|
{
|
|
|
|
if (value) {
|
2015-06-03 18:23:01 +02:00
|
|
|
_port(pin)->OUTSET.reg = _pin_mask(pin);
|
2014-09-09 11:12:39 +02:00
|
|
|
} else {
|
2015-06-03 18:23:01 +02:00
|
|
|
_port(pin)->OUTCLR.reg = _pin_mask(pin);
|
2014-09-09 11:12:39 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-16 23:13:01 +02:00
|
|
|
void isr_eic(void)
|
2014-09-09 11:12:39 +02:00
|
|
|
{
|
2015-06-03 18:23:01 +02:00
|
|
|
for (int i = 0; i < NUMOF_IRQS; i++) {
|
|
|
|
if (EIC->INTFLAG.reg & (1 << i)) {
|
|
|
|
EIC->INTFLAG.reg = (1 << i);
|
2015-10-15 17:05:40 +02:00
|
|
|
if(EIC->INTENSET.reg & (1 << i)) {
|
|
|
|
gpio_config[i].cb(gpio_config[i].arg);
|
|
|
|
}
|
2015-04-24 15:32:20 +02:00
|
|
|
}
|
2014-09-09 11:12:39 +02:00
|
|
|
}
|
|
|
|
if (sched_context_switch_request) {
|
|
|
|
thread_yield();
|
|
|
|
}
|
|
|
|
}
|