Merge pull request #8951 from ZetaR60/RIOT_atmega_ext_int_clarity
cpu/atmega_common: external interrupt fix and refactor
This commit is contained in:
commit
1aed925ca8
@ -42,6 +42,20 @@ enum {
|
|||||||
PORT_G = 6, /**< port G */
|
PORT_G = 6, /**< port G */
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Available external interrupt pins on the ATmega1281 family
|
||||||
|
*
|
||||||
|
* In order of their interrupt number.
|
||||||
|
*/
|
||||||
|
#define CPU_ATMEGA_EXT_INTS { GPIO_PIN(PORT_D, 0), \
|
||||||
|
GPIO_PIN(PORT_D, 1), \
|
||||||
|
GPIO_PIN(PORT_D, 2), \
|
||||||
|
GPIO_PIN(PORT_D, 3), \
|
||||||
|
GPIO_PIN(PORT_E, 4), \
|
||||||
|
GPIO_PIN(PORT_E, 5), \
|
||||||
|
GPIO_PIN(PORT_E, 6), \
|
||||||
|
GPIO_PIN(PORT_E, 7) }
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @name Defines for the I2C interface
|
* @name Defines for the I2C interface
|
||||||
* @{
|
* @{
|
||||||
|
|||||||
@ -35,7 +35,7 @@ extern "C" {
|
|||||||
#define GPIO_PIN(x, y) ((x << 4) | y)
|
#define GPIO_PIN(x, y) ((x << 4) | y)
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Available ports on the ATmega328p family
|
* @brief Available ports on the ATmega1284p family
|
||||||
*/
|
*/
|
||||||
enum {
|
enum {
|
||||||
PORT_A = 0, /**< port A */
|
PORT_A = 0, /**< port A */
|
||||||
@ -44,6 +44,15 @@ enum {
|
|||||||
PORT_D = 3 /**< port D */
|
PORT_D = 3 /**< port D */
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Available external interrupt pins on the ATmega1284p family
|
||||||
|
*
|
||||||
|
* In order of their interrupt number
|
||||||
|
*/
|
||||||
|
#define CPU_ATMEGA_EXT_INTS { GPIO_PIN(PORT_D, 2), \
|
||||||
|
GPIO_PIN(PORT_D, 3), \
|
||||||
|
GPIO_PIN(PORT_B, 2) }
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @name Defines for the I2C interface
|
* @name Defines for the I2C interface
|
||||||
* @{
|
* @{
|
||||||
|
|||||||
@ -44,6 +44,20 @@ enum {
|
|||||||
PORT_L = 10 /**< port L */
|
PORT_L = 10 /**< port L */
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Available external interrupt pins on the ATmega2560 family
|
||||||
|
*
|
||||||
|
* In order of their interrupt number.
|
||||||
|
*/
|
||||||
|
#define CPU_ATMEGA_EXT_INTS { GPIO_PIN(PORT_D, 0), \
|
||||||
|
GPIO_PIN(PORT_D, 1), \
|
||||||
|
GPIO_PIN(PORT_D, 2), \
|
||||||
|
GPIO_PIN(PORT_D, 3), \
|
||||||
|
GPIO_PIN(PORT_E, 4), \
|
||||||
|
GPIO_PIN(PORT_E, 5), \
|
||||||
|
GPIO_PIN(PORT_E, 6), \
|
||||||
|
GPIO_PIN(PORT_E, 7) }
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @name Defines for the I2C interface
|
* @name Defines for the I2C interface
|
||||||
* @{
|
* @{
|
||||||
|
|||||||
@ -46,6 +46,20 @@ enum {
|
|||||||
PORT_G = 6, /**< port G */
|
PORT_G = 6, /**< port G */
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Available external interrupt pins on the ATmega256rfr family
|
||||||
|
*
|
||||||
|
* In order of their interrupt number.
|
||||||
|
*/
|
||||||
|
#define CPU_ATMEGA_EXT_INTS { GPIO_PIN(PORT_D, 0), \
|
||||||
|
GPIO_PIN(PORT_D, 1), \
|
||||||
|
GPIO_PIN(PORT_D, 2), \
|
||||||
|
GPIO_PIN(PORT_D, 3), \
|
||||||
|
GPIO_PIN(PORT_E, 4), \
|
||||||
|
GPIO_PIN(PORT_E, 5), \
|
||||||
|
GPIO_PIN(PORT_E, 6), \
|
||||||
|
GPIO_PIN(PORT_E, 7) }
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @name Defines for the I2C interface
|
* @name Defines for the I2C interface
|
||||||
* @{
|
* @{
|
||||||
|
|||||||
@ -41,6 +41,14 @@ enum {
|
|||||||
PORT_D = 3 /**< port D */
|
PORT_D = 3 /**< port D */
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Available external interrupt pins on the ATmega328p family
|
||||||
|
*
|
||||||
|
* In order of their interrupt number.
|
||||||
|
*/
|
||||||
|
#define CPU_ATMEGA_EXT_INTS { GPIO_PIN(PORT_D, 2), \
|
||||||
|
GPIO_PIN(PORT_D, 3) }
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @name Defines for the I2C interface
|
* @name Defines for the I2C interface
|
||||||
* @{
|
* @{
|
||||||
|
|||||||
@ -30,6 +30,7 @@
|
|||||||
#include "cpu.h"
|
#include "cpu.h"
|
||||||
#include "periph/gpio.h"
|
#include "periph/gpio.h"
|
||||||
#include "periph_conf.h"
|
#include "periph_conf.h"
|
||||||
|
#include "periph_cpu.h"
|
||||||
|
|
||||||
#define GPIO_BASE_PORT_A (0x20)
|
#define GPIO_BASE_PORT_A (0x20)
|
||||||
#define GPIO_OFFSET_PORT_H (0xCB)
|
#define GPIO_OFFSET_PORT_H (0xCB)
|
||||||
@ -40,18 +41,18 @@
|
|||||||
* @brief Define GPIO interruptions for an specific atmega CPU, by default
|
* @brief Define GPIO interruptions for an specific atmega CPU, by default
|
||||||
* 2 (for small atmega CPUs)
|
* 2 (for small atmega CPUs)
|
||||||
*/
|
*/
|
||||||
#if defined(INT2_vect)
|
#if defined(INT7_vect)
|
||||||
#define GPIO_EXT_INT_NUMOF (3U)
|
#define GPIO_EXT_INT_NUMOF (8U)
|
||||||
|
#elif defined(INT6_vect)
|
||||||
|
#define GPIO_EXT_INT_NUMOF (7U)
|
||||||
|
#elif defined(INT5_vect)
|
||||||
|
#define GPIO_EXT_INT_NUMOF (6U)
|
||||||
|
#elif defined(INT4_vect)
|
||||||
|
#define GPIO_EXT_INT_NUMOF (5U)
|
||||||
#elif defined(INT3_vect)
|
#elif defined(INT3_vect)
|
||||||
#define GPIO_EXT_INT_NUMOF (4U)
|
#define GPIO_EXT_INT_NUMOF (4U)
|
||||||
#elif defined(INT4_vect)
|
#elif defined(INT2_vect)
|
||||||
#define GPIO_EXT_INT_NUMOF (4U)
|
#define GPIO_EXT_INT_NUMOF (3U)
|
||||||
#elif defined(INT5_vect)
|
|
||||||
#define GPIO_EXT_INT_NUMOF (4U)
|
|
||||||
#elif defined(INT6_vect)
|
|
||||||
#define GPIO_EXT_INT_NUMOF (4U)
|
|
||||||
#elif defined(INT7_vect)
|
|
||||||
#define GPIO_EXT_INT_NUMOF (4U)
|
|
||||||
#else
|
#else
|
||||||
#define GPIO_EXT_INT_NUMOF (2U)
|
#define GPIO_EXT_INT_NUMOF (2U)
|
||||||
#endif
|
#endif
|
||||||
@ -109,6 +110,21 @@ static inline uint16_t _pin_addr(gpio_t pin)
|
|||||||
return (_port_addr(pin) - 0x02);
|
return (_port_addr(pin) - 0x02);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline int8_t _int_num(gpio_t pin)
|
||||||
|
{
|
||||||
|
uint8_t num;
|
||||||
|
const gpio_t ext_ints[GPIO_EXT_INT_NUMOF] = CPU_ATMEGA_EXT_INTS;
|
||||||
|
|
||||||
|
/* find pin in ext_ints array to get the interrupt number */
|
||||||
|
for (num = 0; num < GPIO_EXT_INT_NUMOF; num++) {
|
||||||
|
if (pin == ext_ints[num]) {
|
||||||
|
return num;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
int gpio_init(gpio_t pin, gpio_mode_t mode)
|
int gpio_init(gpio_t pin, gpio_mode_t mode)
|
||||||
{
|
{
|
||||||
switch (mode) {
|
switch (mode) {
|
||||||
@ -132,20 +148,14 @@ int gpio_init(gpio_t pin, gpio_mode_t mode)
|
|||||||
int gpio_init_int(gpio_t pin, gpio_mode_t mode, gpio_flank_t flank,
|
int gpio_init_int(gpio_t pin, gpio_mode_t mode, gpio_flank_t flank,
|
||||||
gpio_cb_t cb, void *arg)
|
gpio_cb_t cb, void *arg)
|
||||||
{
|
{
|
||||||
uint8_t pin_num = _pin_num(pin);
|
int8_t int_num = _int_num(pin);
|
||||||
|
|
||||||
if ((_port_num(pin) == PORT_D && pin_num > 3)
|
if ((mode != GPIO_IN) && (mode != GPIO_IN_PU)) {
|
||||||
#if defined (PORTE)
|
return -1;
|
||||||
|| (_port_num(pin) == PORT_E && pin_num < 4)
|
}
|
||||||
|| (_port_num(pin) != PORT_D && _port_num(pin) != PORT_E)
|
|
||||||
#elif defined(CPU_ATMEGA328P)
|
/* not a valid interrupt pin */
|
||||||
|| (pin_num < 2) || (_port_num(pin) != PORT_D)
|
if (int_num < 0) {
|
||||||
#elif defined(CPU_ATMEGA1284P)
|
|
||||||
|| (_port_num(pin) == PORT_B && pin_num != 2)
|
|
||||||
|| (_port_num(pin) == PORT_D && pin_num < 2)
|
|
||||||
|| (_port_num(pin) != PORT_D && _port_num(pin) != PORT_D)
|
|
||||||
#endif
|
|
||||||
|| ((mode != GPIO_IN) && (mode != GPIO_IN_PU))) {
|
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -154,35 +164,27 @@ int gpio_init_int(gpio_t pin, gpio_mode_t mode, gpio_flank_t flank,
|
|||||||
/* clear global interrupt flag */
|
/* clear global interrupt flag */
|
||||||
cli();
|
cli();
|
||||||
|
|
||||||
#if defined(CPU_ATMEGA328P)
|
/* enable interrupt number int_num */
|
||||||
/* INT pins start at PD2 instead of at PD0 */
|
EIMSK |= (1 << int_num);
|
||||||
pin_num -= 2;
|
|
||||||
#elif defined(CPU_ATMEGA1284P)
|
|
||||||
/* INT pins start at PD2 instead of at PD0 on PORT_D */
|
|
||||||
if (_port_num(pin) == PORT_D) {
|
|
||||||
pin_num -= 2;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
EIMSK |= (1 << pin_num);
|
|
||||||
|
|
||||||
/* configure the flank */
|
/* configure the flank */
|
||||||
if (flank > GPIO_RISING) {
|
if (flank > GPIO_RISING) {
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (pin_num < 4) {
|
/* apply flank to interrupt number int_num */
|
||||||
EICRA |= (flank << (pin_num * 2));
|
if (int_num < 4) {
|
||||||
|
EICRA |= (flank << (int_num * 2));
|
||||||
}
|
}
|
||||||
#if defined(EICRB)
|
#if defined(EICRB)
|
||||||
else {
|
else {
|
||||||
EICRB |= (flank << (pin_num * 2) % 4);
|
EICRB |= (flank << ((int_num % 4) * 2));
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* set callback */
|
/* set callback */
|
||||||
config[pin_num].cb = cb;
|
config[int_num].cb = cb;
|
||||||
config[pin_num].arg = arg;
|
config[int_num].arg = arg;
|
||||||
|
|
||||||
/* set global interrupt flag */
|
/* set global interrupt flag */
|
||||||
sei();
|
sei();
|
||||||
@ -192,38 +194,12 @@ int gpio_init_int(gpio_t pin, gpio_mode_t mode, gpio_flank_t flank,
|
|||||||
|
|
||||||
void gpio_irq_enable(gpio_t pin)
|
void gpio_irq_enable(gpio_t pin)
|
||||||
{
|
{
|
||||||
#if defined(CPU_ATMEGA328P)
|
EIMSK |= (1 << _int_num(pin));
|
||||||
/* INT pins start at PD2 instead of at PD0 */
|
|
||||||
EIMSK |= (1 << (_pin_num(pin) - 2));
|
|
||||||
#elif defined(CPU_ATMEGA1284P)
|
|
||||||
/* INT pins start at PD2 instead of at PD0 on PORT_D */
|
|
||||||
if (_port_num(pin) == PORT_D) {
|
|
||||||
EIMSK |= (1 << (_pin_num(pin) - 2));
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
EIMSK |= (1 << _pin_num(pin));
|
|
||||||
}
|
|
||||||
#else
|
|
||||||
EIMSK |= (1 << _pin_num(pin));
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void gpio_irq_disable(gpio_t pin)
|
void gpio_irq_disable(gpio_t pin)
|
||||||
{
|
{
|
||||||
#if defined(CPU_ATMEGA328P)
|
EIMSK &= ~(1 << _int_num(pin));
|
||||||
/* INT pins start at PD2 instead of at PD0 */
|
|
||||||
EIMSK &= ~(1 << (_pin_num(pin) - 2));
|
|
||||||
#elif defined (CPU_ATMEGA1284P)
|
|
||||||
/* INT pins start at PD2 instead of at PD0 on PORT_D */
|
|
||||||
if (_port_num(pin) == PORT_D) {
|
|
||||||
EIMSK &= ~(1 << (_pin_num(pin) - 2));
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
EIMSK &= ~(1 << _pin_num(pin));
|
|
||||||
}
|
|
||||||
#else
|
|
||||||
EIMSK &= ~(1 << _pin_num(pin));
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int gpio_read(gpio_t pin)
|
int gpio_read(gpio_t pin)
|
||||||
@ -261,10 +237,10 @@ void gpio_write(gpio_t pin, int value)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void irq_handler(uint8_t pin_num)
|
static inline void irq_handler(uint8_t int_num)
|
||||||
{
|
{
|
||||||
__enter_isr();
|
__enter_isr();
|
||||||
config[pin_num].cb(config[pin_num].arg);
|
config[int_num].cb(config[int_num].arg);
|
||||||
__exit_isr();
|
__exit_isr();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user