mirror of
https://github.com/arduino/Arduino.git
synced 2025-01-22 11:52:14 +01:00
183 lines
4.3 KiB
C
183 lines
4.3 KiB
C
/*
|
|
Copyright (c) 2011-2012 Arduino. All right reserved.
|
|
|
|
This library is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU Lesser General Public
|
|
License as published by the Free Software Foundation; either
|
|
version 2.1 of the License, or (at your option) any later version.
|
|
|
|
This library is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
See the GNU Lesser General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Lesser General Public
|
|
License along with this library; if not, write to the Free Software
|
|
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*/
|
|
|
|
#include "WInterrupts.h"
|
|
|
|
typedef void (*interruptCB)(void);
|
|
|
|
static interruptCB callbacksPioA[32];
|
|
static interruptCB callbacksPioB[32];
|
|
static interruptCB callbacksPioC[32];
|
|
static interruptCB callbacksPioD[32];
|
|
|
|
/* Configure PIO interrupt sources */
|
|
static void __initialize() {
|
|
int i;
|
|
for (i=0; i<32; i++) {
|
|
callbacksPioA[i] = NULL;
|
|
callbacksPioB[i] = NULL;
|
|
callbacksPioC[i] = NULL;
|
|
callbacksPioD[i] = NULL;
|
|
}
|
|
|
|
pmc_enable_periph_clk(ID_PIOA);
|
|
NVIC_DisableIRQ(PIOA_IRQn);
|
|
NVIC_ClearPendingIRQ(PIOA_IRQn);
|
|
NVIC_SetPriority(PIOA_IRQn, 0);
|
|
NVIC_EnableIRQ(PIOA_IRQn);
|
|
|
|
pmc_enable_periph_clk(ID_PIOB);
|
|
NVIC_DisableIRQ(PIOB_IRQn);
|
|
NVIC_ClearPendingIRQ(PIOB_IRQn);
|
|
NVIC_SetPriority(PIOB_IRQn, 0);
|
|
NVIC_EnableIRQ(PIOB_IRQn);
|
|
|
|
pmc_enable_periph_clk(ID_PIOC);
|
|
NVIC_DisableIRQ(PIOC_IRQn);
|
|
NVIC_ClearPendingIRQ(PIOC_IRQn);
|
|
NVIC_SetPriority(PIOC_IRQn, 0);
|
|
NVIC_EnableIRQ(PIOC_IRQn);
|
|
|
|
pmc_enable_periph_clk(ID_PIOD);
|
|
NVIC_DisableIRQ(PIOD_IRQn);
|
|
NVIC_ClearPendingIRQ(PIOD_IRQn);
|
|
NVIC_SetPriority(PIOD_IRQn, 0);
|
|
NVIC_EnableIRQ(PIOD_IRQn);
|
|
}
|
|
|
|
|
|
void attachInterrupt(uint32_t pin, void (*callback)(void), uint32_t mode)
|
|
{
|
|
static int enabled = 0;
|
|
if (!enabled) {
|
|
__initialize();
|
|
enabled = 1;
|
|
}
|
|
|
|
// Retrieve pin information
|
|
Pio *pio = g_APinDescription[pin].pPort;
|
|
uint32_t mask = g_APinDescription[pin].ulPin;
|
|
uint32_t pos = 0;
|
|
|
|
uint32_t t;
|
|
for (t = mask; t>1; t>>=1, pos++)
|
|
;
|
|
|
|
// Set callback function
|
|
if (pio == PIOA)
|
|
callbacksPioA[pos] = callback;
|
|
if (pio == PIOB)
|
|
callbacksPioB[pos] = callback;
|
|
if (pio == PIOC)
|
|
callbacksPioC[pos] = callback;
|
|
if (pio == PIOD)
|
|
callbacksPioD[pos] = callback;
|
|
|
|
// Configure the interrupt mode
|
|
if (mode == CHANGE) {
|
|
// Disable additional interrupt mode (detects both rising and falling edges)
|
|
pio->PIO_AIMDR = mask;
|
|
} else {
|
|
// Enable additional interrupt mode
|
|
pio->PIO_AIMER = mask;
|
|
|
|
// Select mode of operation
|
|
if (mode == LOW) {
|
|
pio->PIO_LSR = mask; // "Level" Select Register
|
|
pio->PIO_FELLSR = mask; // "Falling Edge / Low Level" Select Register
|
|
}
|
|
if (mode == HIGH) {
|
|
pio->PIO_LSR = mask; // "Level" Select Register
|
|
pio->PIO_REHLSR = mask; // "Rising Edge / High Level" Select Register
|
|
}
|
|
if (mode == FALLING) {
|
|
pio->PIO_ESR = mask; // "Edge" Select Register
|
|
pio->PIO_FELLSR = mask; // "Falling Edge / Low Level" Select Register
|
|
}
|
|
if (mode == RISING) {
|
|
pio->PIO_ESR = mask; // "Edge" Select Register
|
|
pio->PIO_REHLSR = mask; // "Rising Edge / High Level" Select Register
|
|
}
|
|
}
|
|
|
|
// Enable interrupt
|
|
pio->PIO_IER = mask;
|
|
}
|
|
|
|
void detachInterrupt(uint32_t pin)
|
|
{
|
|
// Retrieve pin information
|
|
Pio *pio = g_APinDescription[pin].pPort;
|
|
uint32_t mask = g_APinDescription[pin].ulPin;
|
|
|
|
// Disable interrupt
|
|
pio->PIO_IDR = mask;
|
|
}
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
|
|
void PIOA_Handler(void) {
|
|
uint32_t isr = PIOA->PIO_ISR;
|
|
uint32_t i;
|
|
for (i=0; i<32; i++, isr>>=1) {
|
|
if ((isr & 0x1) == 0)
|
|
continue;
|
|
if (callbacksPioA[i])
|
|
callbacksPioA[i]();
|
|
}
|
|
}
|
|
|
|
void PIOB_Handler(void) {
|
|
uint32_t isr = PIOB->PIO_ISR;
|
|
uint32_t i;
|
|
for (i=0; i<32; i++, isr>>=1) {
|
|
if ((isr & 0x1) == 0)
|
|
continue;
|
|
if (callbacksPioB[i])
|
|
callbacksPioB[i]();
|
|
}
|
|
}
|
|
|
|
void PIOC_Handler(void) {
|
|
uint32_t isr = PIOC->PIO_ISR;
|
|
uint32_t i;
|
|
for (i=0; i<32; i++, isr>>=1) {
|
|
if ((isr & 0x1) == 0)
|
|
continue;
|
|
if (callbacksPioC[i])
|
|
callbacksPioC[i]();
|
|
}
|
|
}
|
|
|
|
void PIOD_Handler(void) {
|
|
uint32_t isr = PIOD->PIO_ISR;
|
|
uint32_t i;
|
|
for (i=0; i<32; i++, isr>>=1) {
|
|
if ((isr & 0x1) == 0)
|
|
continue;
|
|
if (callbacksPioD[i])
|
|
callbacksPioD[i]();
|
|
}
|
|
}
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|