2010-12-14 00:24:42 +01:00
|
|
|
/******************************************************************************
|
|
|
|
Copyright 2010, Freie Universität Berlin (FUB). All rights reserved.
|
|
|
|
|
|
|
|
These sources were developed at the Freie Universitaet Berlin, Computer Systems
|
|
|
|
and Telematics group (http://cst.mi.fu-berlin.de).
|
|
|
|
-------------------------------------------------------------------------------
|
2013-03-07 20:51:26 +01:00
|
|
|
This file is part of RIOT.
|
2010-12-14 00:24:42 +01:00
|
|
|
|
|
|
|
This program is free software: you can redistribute it and/or modify it under
|
|
|
|
the terms of the GNU General Public License as published by the Free Software
|
|
|
|
Foundation, either version 3 of the License, or (at your option) any later
|
|
|
|
version.
|
|
|
|
|
2013-03-07 20:51:26 +01:00
|
|
|
RIOT is distributed in the hope that it will be useful, but WITHOUT
|
2010-12-14 00:24:42 +01:00
|
|
|
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
|
|
|
|
FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License along with
|
|
|
|
this program. If not, see http://www.gnu.org/licenses/ .
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
For further information and questions please use the web site
|
|
|
|
http://scatterweb.mi.fu-berlin.de
|
|
|
|
and the mailinglist (subscription via web site)
|
|
|
|
scatterweb@lists.spline.inf.fu-berlin.de
|
|
|
|
*******************************************************************************/
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
2013-03-30 21:44:16 +01:00
|
|
|
#include <legacymsp430.h>
|
2010-12-14 00:24:42 +01:00
|
|
|
#include <gpioint.h>
|
2010-12-15 15:48:50 +01:00
|
|
|
#include <bitarithm.h>
|
2010-12-14 00:24:42 +01:00
|
|
|
#include <cpu.h>
|
|
|
|
#include <irq.h>
|
|
|
|
#include <hwtimer.h>
|
2013-03-30 21:44:16 +01:00
|
|
|
#include <cc430f6137.h>
|
2010-12-14 00:24:42 +01:00
|
|
|
|
|
|
|
/** min and max portnumber to generate interrupts */
|
|
|
|
#define PORTINT_MIN (1)
|
|
|
|
#define PORTINT_MAX (2)
|
|
|
|
|
|
|
|
/** amount of interrupt capable ports */
|
|
|
|
#define INT_PORTS (2)
|
|
|
|
|
|
|
|
/** number of bits per port */
|
|
|
|
#define BITMASK_SIZE (8)
|
|
|
|
|
|
|
|
/** debouncing port interrupts */
|
2010-12-16 18:09:40 +01:00
|
|
|
#define DEBOUNCE_TIMEOUT (250)
|
2010-12-14 00:24:42 +01:00
|
|
|
|
|
|
|
/** interrupt callbacks */
|
|
|
|
fp_irqcb cb[INT_PORTS][BITMASK_SIZE];
|
|
|
|
|
2010-12-15 15:48:50 +01:00
|
|
|
/** debounce interrupt flags */
|
|
|
|
uint8_t debounce_flags[INT_PORTS];
|
|
|
|
|
|
|
|
/** debounce interrupt times */
|
|
|
|
uint16_t debounce_time[INT_PORTS][BITMASK_SIZE];
|
2010-12-14 00:24:42 +01:00
|
|
|
|
2010-12-16 17:11:23 +01:00
|
|
|
uint16_t c1 = 0, c2 = 0;
|
|
|
|
|
2010-12-14 00:24:42 +01:00
|
|
|
void gpioint_init(void) {
|
|
|
|
uint8_t i, j;
|
|
|
|
for (i = 0; i < INT_PORTS; i++) {
|
|
|
|
for (j = 0; j < BITMASK_SIZE; j++) {
|
|
|
|
cb[i][j] = NULL;
|
2010-12-15 15:48:50 +01:00
|
|
|
debounce_time[i][j] = 0;
|
2010-12-14 00:24:42 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool gpioint_set(int port, uint32_t bitmask, int flags, fp_irqcb callback) {
|
|
|
|
int8_t base;
|
|
|
|
|
|
|
|
if ((port >= PORTINT_MIN) && (port <= PORTINT_MAX)) {
|
|
|
|
/* set the callback function */
|
2010-12-15 15:48:50 +01:00
|
|
|
base = number_of_highest_bit(bitmask);
|
2010-12-14 00:24:42 +01:00
|
|
|
if (base >= 0) {
|
2010-12-15 15:48:50 +01:00
|
|
|
cb[port - PORTINT_MIN][base] = callback;
|
2010-12-14 00:24:42 +01:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
return false;
|
|
|
|
}
|
2010-12-15 15:48:50 +01:00
|
|
|
if (flags & GPIOINT_DEBOUNCE) {
|
|
|
|
debounce_flags[port - PORTINT_MIN] |= bitmask;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
debounce_flags[port - PORTINT_MIN] &= ~bitmask;
|
|
|
|
}
|
2010-12-14 00:24:42 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
switch (port) {
|
|
|
|
case 1:
|
|
|
|
/* set port to input */
|
|
|
|
P1DIR &= ~bitmask;
|
|
|
|
/* enable internal pull-down */
|
|
|
|
P1OUT &= ~bitmask;
|
|
|
|
P1REN |= bitmask;
|
|
|
|
|
|
|
|
/* reset IRQ flag */
|
|
|
|
P1IFG &= ~bitmask;
|
|
|
|
|
|
|
|
/* trigger on rising... */
|
2010-12-15 15:48:50 +01:00
|
|
|
if (flags & GPIOINT_RISING_EDGE) {
|
2010-12-14 00:24:42 +01:00
|
|
|
P1IES &= bitmask;
|
|
|
|
}
|
|
|
|
/* ...or falling edge */
|
2010-12-15 15:48:50 +01:00
|
|
|
if (flags & GPIOINT_FALLING_EDGE) {
|
2010-12-14 00:24:42 +01:00
|
|
|
P1IES |= bitmask;
|
|
|
|
}
|
2010-12-15 15:48:50 +01:00
|
|
|
|
|
|
|
/* disable interrupt */
|
|
|
|
if (flags == GPIOINT_DISABLE) {
|
2010-12-14 00:24:42 +01:00
|
|
|
P1IE &= ~bitmask;
|
|
|
|
}
|
|
|
|
/* enable interrupt */
|
|
|
|
P1IE |= bitmask;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
/* set port to input */
|
|
|
|
P2DIR &= ~bitmask;
|
|
|
|
/* enable internal pull-down */
|
|
|
|
P2OUT &= ~bitmask;
|
|
|
|
P2REN |= bitmask;
|
|
|
|
|
|
|
|
/* reset IRQ flag */
|
|
|
|
P2IFG &= ~bitmask;
|
|
|
|
|
|
|
|
/* trigger on rising... */
|
|
|
|
if (flags == GPIOINT_RISING_EDGE) {
|
|
|
|
P2IES &= bitmask;
|
|
|
|
}
|
|
|
|
/* ...or falling edge */
|
|
|
|
else if (flags == GPIOINT_FALLING_EDGE) {
|
|
|
|
P2IES |= bitmask;
|
|
|
|
}
|
|
|
|
/* or disable interrupt */
|
|
|
|
else {
|
|
|
|
P2IE &= ~bitmask;
|
|
|
|
}
|
|
|
|
/* enable interrupt */
|
|
|
|
P2IE |= bitmask;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
interrupt (PORT1_VECTOR) __attribute__ ((naked)) port1_isr(void) {
|
2010-12-16 18:09:40 +01:00
|
|
|
uint8_t int_enable, ifg_num, p1ifg;
|
2010-12-14 00:24:42 +01:00
|
|
|
uint16_t p1iv;
|
2010-12-16 18:09:40 +01:00
|
|
|
uint16_t diff;
|
2010-12-14 00:24:42 +01:00
|
|
|
__enter_isr();
|
|
|
|
|
|
|
|
/* Debounce
|
|
|
|
* Disable PORT1 IRQ
|
|
|
|
*/
|
2010-12-16 18:09:40 +01:00
|
|
|
p1ifg = P1IFG;
|
2010-12-14 00:24:42 +01:00
|
|
|
p1iv = P1IV;
|
|
|
|
int_enable = P1IE;
|
|
|
|
P1IE = 0x00;
|
|
|
|
|
2010-12-16 17:11:23 +01:00
|
|
|
ifg_num = (p1iv >> 1) - 1;
|
2010-12-15 15:48:50 +01:00
|
|
|
/* check interrupt source */
|
2010-12-16 18:09:40 +01:00
|
|
|
if (debounce_flags[0] & p1ifg) {
|
2010-12-15 15:48:50 +01:00
|
|
|
/* check if bouncing */
|
2010-12-16 18:09:40 +01:00
|
|
|
diff = hwtimer_now() - debounce_time[0][ifg_num];
|
|
|
|
if (diff > DEBOUNCE_TIMEOUT) {
|
2010-12-15 15:48:50 +01:00
|
|
|
debounce_time[0][ifg_num] = hwtimer_now();
|
2010-12-16 18:09:40 +01:00
|
|
|
if (cb[0][ifg_num] != NULL) {
|
|
|
|
cb[0][ifg_num]();
|
|
|
|
}
|
2010-12-15 15:48:50 +01:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* TODO: check for long duration irq */
|
2010-12-16 18:09:40 +01:00
|
|
|
asm volatile (" nop ");
|
2010-12-15 15:48:50 +01:00
|
|
|
}
|
2010-12-14 00:24:42 +01:00
|
|
|
}
|
2010-12-16 17:11:23 +01:00
|
|
|
else {
|
2010-12-16 18:09:40 +01:00
|
|
|
if (cb[0][ifg_num] != NULL) {
|
|
|
|
cb[0][ifg_num]();
|
|
|
|
}
|
2010-12-16 17:11:23 +01:00
|
|
|
}
|
2010-12-14 00:24:42 +01:00
|
|
|
P1IFG = 0x00;
|
|
|
|
P1IE = int_enable;
|
|
|
|
__exit_isr();
|
|
|
|
}
|
|
|
|
|
|
|
|
interrupt (PORT2_VECTOR) __attribute__ ((naked)) port2_isr(void) {
|
2010-12-16 17:11:23 +01:00
|
|
|
uint8_t int_enable, ifg_num, p2ifg;
|
2010-12-14 00:24:42 +01:00
|
|
|
uint16_t p2iv;
|
2010-12-16 17:11:23 +01:00
|
|
|
uint16_t diff;
|
2010-12-14 00:24:42 +01:00
|
|
|
__enter_isr();
|
|
|
|
|
|
|
|
/* Debounce
|
|
|
|
* Disable PORT2 IRQ
|
|
|
|
*/
|
2010-12-16 17:11:23 +01:00
|
|
|
p2ifg = P2IFG;
|
2010-12-14 00:24:42 +01:00
|
|
|
p2iv = P2IV;
|
|
|
|
int_enable = P2IE;
|
|
|
|
P2IE = 0x00;
|
|
|
|
|
2010-12-16 17:11:23 +01:00
|
|
|
ifg_num = (p2iv >> 1) - 1;
|
2010-12-15 15:48:50 +01:00
|
|
|
/* check interrupt source */
|
2010-12-16 17:11:23 +01:00
|
|
|
if (debounce_flags[1] & p2ifg) {
|
2010-12-15 15:48:50 +01:00
|
|
|
/* check if bouncing */
|
2010-12-16 17:11:23 +01:00
|
|
|
diff = hwtimer_now() - debounce_time[1][ifg_num];
|
|
|
|
if (diff > DEBOUNCE_TIMEOUT) {
|
2010-12-15 15:48:50 +01:00
|
|
|
debounce_time[1][ifg_num] = hwtimer_now();
|
2010-12-16 17:11:23 +01:00
|
|
|
c1++;
|
|
|
|
if (cb[1][ifg_num] != NULL) {
|
|
|
|
cb[1][ifg_num]();
|
|
|
|
}
|
2010-12-15 15:48:50 +01:00
|
|
|
}
|
|
|
|
else {
|
2010-12-16 17:11:23 +01:00
|
|
|
c2++;
|
2010-12-15 15:48:50 +01:00
|
|
|
/* TODO: check for long duration irq */
|
2010-12-16 17:11:23 +01:00
|
|
|
asm volatile (" nop ");
|
2010-12-15 15:48:50 +01:00
|
|
|
}
|
2010-12-14 00:24:42 +01:00
|
|
|
}
|
2010-12-16 17:11:23 +01:00
|
|
|
else {
|
2010-12-16 18:09:40 +01:00
|
|
|
if (cb[1][ifg_num] != NULL) {
|
|
|
|
cb[1][ifg_num]();
|
|
|
|
}
|
2010-12-16 17:11:23 +01:00
|
|
|
}
|
2013-03-30 21:44:16 +01:00
|
|
|
|
|
|
|
//else {
|
|
|
|
// cb[1][ifg_num]();
|
|
|
|
//}
|
2010-12-15 15:48:50 +01:00
|
|
|
|
2010-12-14 00:24:42 +01:00
|
|
|
P2IFG = 0x00;
|
|
|
|
P2IE = int_enable;
|
|
|
|
__exit_isr();
|
|
|
|
}
|
|
|
|
|