/****************************************************************************** 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). ------------------------------------------------------------------------------- This file is part of µkleos. 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. FeuerWare 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 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 #include #include #include #include #include #include #include /** 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 */ #define DEBOUNCE_TIMEOUT (250) /** interrupt callbacks */ fp_irqcb cb[INT_PORTS][BITMASK_SIZE]; /** debounce interrupt flags */ uint8_t debounce_flags[INT_PORTS]; /** debounce interrupt times */ uint16_t debounce_time[INT_PORTS][BITMASK_SIZE]; uint16_t c1 = 0, c2 = 0; 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; debounce_time[i][j] = 0; } } } 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 */ base = number_of_highest_bit(bitmask); if (base >= 0) { cb[port - PORTINT_MIN][base] = callback; } else { return false; } if (flags & GPIOINT_DEBOUNCE) { debounce_flags[port - PORTINT_MIN] |= bitmask; } else { debounce_flags[port - PORTINT_MIN] &= ~bitmask; } } 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... */ if (flags & GPIOINT_RISING_EDGE) { P1IES &= bitmask; } /* ...or falling edge */ if (flags & GPIOINT_FALLING_EDGE) { P1IES |= bitmask; } /* disable interrupt */ if (flags == GPIOINT_DISABLE) { 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) { uint8_t int_enable, ifg_num, p1ifg; uint16_t p1iv; uint16_t diff; __enter_isr(); /* Debounce * Disable PORT1 IRQ */ p1ifg = P1IFG; p1iv = P1IV; int_enable = P1IE; P1IE = 0x00; ifg_num = (p1iv >> 1) - 1; /* check interrupt source */ if (debounce_flags[0] & p1ifg) { /* check if bouncing */ diff = hwtimer_now() - debounce_time[0][ifg_num]; if (diff > DEBOUNCE_TIMEOUT) { debounce_time[0][ifg_num] = hwtimer_now(); if (cb[0][ifg_num] != NULL) { cb[0][ifg_num](); } } else { /* TODO: check for long duration irq */ asm volatile (" nop "); } } else { if (cb[0][ifg_num] != NULL) { cb[0][ifg_num](); } } P1IFG = 0x00; P1IE = int_enable; __exit_isr(); } interrupt (PORT2_VECTOR) __attribute__ ((naked)) port2_isr(void) { uint8_t int_enable, ifg_num, p2ifg; uint16_t p2iv; uint16_t diff; __enter_isr(); /* Debounce * Disable PORT2 IRQ */ p2ifg = P2IFG; p2iv = P2IV; int_enable = P2IE; P2IE = 0x00; ifg_num = (p2iv >> 1) - 1; /* check interrupt source */ if (debounce_flags[1] & p2ifg) { /* check if bouncing */ diff = hwtimer_now() - debounce_time[1][ifg_num]; if (diff > DEBOUNCE_TIMEOUT) { debounce_time[1][ifg_num] = hwtimer_now(); c1++; if (cb[1][ifg_num] != NULL) { cb[1][ifg_num](); } } else { c2++; /* TODO: check for long duration irq */ asm volatile (" nop "); } } else { if (cb[1][ifg_num] != NULL) { cb[1][ifg_num](); } } P2IFG = 0x00; P2IE = int_enable; __exit_isr(); }