2013-04-01 02:11:43 +02:00
|
|
|
/*
|
|
|
|
* hwtimer_cpu.c - architecture dependent hardware timer functionality
|
|
|
|
* Copyright (C) 2013 Oliver Hahm <oliver.hahm@inria.fr>
|
2013-04-25 13:30:09 +02:00
|
|
|
* Copyright (C) 2013 Thomas Eichinger <thomas.eichinger@fu-berlin.de>
|
2013-04-01 02:11:43 +02:00
|
|
|
*
|
2013-08-16 10:20:23 +02:00
|
|
|
* This source code is licensed under the GNU Lesser General Public License,
|
|
|
|
* Version 2. See the file LICENSE for more details.
|
2013-04-01 02:11:43 +02:00
|
|
|
*
|
|
|
|
* This file is part of RIOT.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdint.h>
|
|
|
|
#include "mc1322x.h"
|
2013-10-24 13:53:09 +02:00
|
|
|
#include "cpu.h"
|
2014-07-09 21:08:13 +02:00
|
|
|
#include "arch/hwtimer_arch.h"
|
2013-04-25 13:30:09 +02:00
|
|
|
#include "irq.h"
|
2013-04-01 02:11:43 +02:00
|
|
|
|
2013-11-11 12:30:06 +01:00
|
|
|
#include <stdio.h>
|
|
|
|
|
2013-04-02 02:06:24 +02:00
|
|
|
/* High level interrupt handler */
|
|
|
|
static void (*int_handler)(int);
|
|
|
|
|
2013-10-24 13:53:09 +02:00
|
|
|
#define TMRx_ANY_INTERRUPT 0xa800
|
|
|
|
|
|
|
|
void tmr_isr(void) {
|
|
|
|
/* detemine which timer caused the interrupt */
|
2013-11-11 12:30:06 +01:00
|
|
|
if (TMR0->SCTRLbits.TCF && TMR0->SCTRLbits.TCFIE) {
|
|
|
|
TMR0->SCTRLbits.TCF = 0;
|
2013-11-14 12:05:06 +01:00
|
|
|
TMR0->CSCTRLbits.TCF1 = 0;
|
2013-11-11 12:30:06 +01:00
|
|
|
TMR0->ENBL &= ~(1<<0);
|
|
|
|
int_handler(0);
|
2013-10-24 13:53:09 +02:00
|
|
|
}
|
2013-11-11 12:30:06 +01:00
|
|
|
else if (TMR1->SCTRLbits.TCF && TMR1->SCTRLbits.TCFIE) {
|
|
|
|
TMR1->SCTRLbits.TCF = 0;
|
2013-11-14 12:05:06 +01:00
|
|
|
TMR1->CSCTRLbits.TCF1 = 0;
|
2013-11-11 12:30:06 +01:00
|
|
|
TMR0->ENBL &= ~(1<<1);
|
|
|
|
int_handler(1);
|
2013-10-24 13:53:09 +02:00
|
|
|
}
|
2013-11-11 12:30:06 +01:00
|
|
|
else if (TMR2->SCTRLbits.TCF && TMR2->SCTRLbits.TCFIE) {
|
|
|
|
TMR2->SCTRLbits.TCF = 0;
|
2013-11-14 12:05:06 +01:00
|
|
|
TMR2->CSCTRLbits.TCF1 = 0;
|
2013-11-11 12:30:06 +01:00
|
|
|
TMR0->ENBL &= ~(1<<2);
|
|
|
|
int_handler(2);
|
2013-10-24 13:53:09 +02:00
|
|
|
}
|
2013-11-11 12:30:06 +01:00
|
|
|
else if (TMR3->SCTRLbits.TCF && TMR3->SCTRLbits.TCFIE) {
|
|
|
|
TMR3->SCTRLbits.TCF = 0;
|
2013-11-14 12:05:06 +01:00
|
|
|
TMR3->CSCTRLbits.TCF1 = 0;
|
2013-11-11 12:30:06 +01:00
|
|
|
TMR0->ENBL &= ~(1<<3);
|
|
|
|
int_handler(3);
|
2013-10-24 13:53:09 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-07-23 10:36:16 +02:00
|
|
|
void timer_x_init(volatile struct TMR_struct* const TMRx) {
|
2013-04-01 02:11:43 +02:00
|
|
|
/* Reset the timer */
|
2013-11-11 12:30:06 +01:00
|
|
|
|
2013-04-01 02:11:43 +02:00
|
|
|
/* Clear status */
|
2013-04-25 13:30:09 +02:00
|
|
|
TMRx->SCTRL = 0;
|
2013-04-01 02:11:43 +02:00
|
|
|
/* disable interrupt */
|
2013-04-25 13:30:09 +02:00
|
|
|
TMRx->CSCTRL =0x0000;
|
2013-04-01 02:11:43 +02:00
|
|
|
/* Reload/initialize to zero */
|
2013-04-25 13:30:09 +02:00
|
|
|
TMRx->LOAD = 0;
|
2013-04-01 02:11:43 +02:00
|
|
|
|
|
|
|
/* disable comparison */
|
2013-04-25 13:30:09 +02:00
|
|
|
TMRx->COMP1 = 0;
|
|
|
|
TMRx->CMPLD1 = 0;
|
2014-02-11 18:15:43 +01:00
|
|
|
|
2013-04-01 02:11:43 +02:00
|
|
|
/* set counter to zero */
|
2013-11-11 12:30:06 +01:00
|
|
|
TMRx->CNTR = TMRx->LOAD;
|
2014-02-11 18:15:43 +01:00
|
|
|
|
2013-04-25 13:30:09 +02:00
|
|
|
/* set timer control bits */
|
|
|
|
TMRx->CTRLbits.COUNT_MODE = 1; /* use rising edge of primary source */
|
|
|
|
TMRx->CTRLbits.PRIMARY_CNT_SOURCE = 0x0f; /* Perip. clock with 128 prescale (for 24MHz = 187500Hz) */
|
|
|
|
TMRx->CTRLbits.SECONDARY_CNT_SOURCE = 0x00; /* don't need this */
|
2013-11-14 12:05:06 +01:00
|
|
|
TMRx->CTRLbits.ONCE = 0x01; /* don't keep counting */
|
2013-04-25 13:30:09 +02:00
|
|
|
TMRx->CTRLbits.LENGTH = 0x00; /* continue counting */
|
|
|
|
TMRx->CTRLbits.DIR = 0x00; /* count up */
|
|
|
|
TMRx->CTRLbits.CO_INIT = 0x00; /* other counters cannot force a reinitialization of this counter*/
|
|
|
|
TMRx->CTRLbits.OUTPUT_MODE = 0x00; /* OFLAG is asserted while counter is active */
|
2013-07-23 10:36:16 +02:00
|
|
|
}
|
2013-04-01 02:11:43 +02:00
|
|
|
|
2013-04-25 13:30:09 +02:00
|
|
|
void hwtimer_arch_init(void (*handler)(int), uint32_t fcpu) {
|
|
|
|
int_handler = handler;
|
2014-02-11 18:15:43 +01:00
|
|
|
|
2013-04-02 02:06:24 +02:00
|
|
|
/* TODO: do scaling voodoo */
|
|
|
|
(void) fcpu;
|
2014-02-11 18:15:43 +01:00
|
|
|
|
2013-11-11 12:30:06 +01:00
|
|
|
/* disable all timers */
|
|
|
|
TMR0->ENBL = 0;
|
2013-04-25 13:30:09 +02:00
|
|
|
timer_x_init(TMR0);
|
|
|
|
timer_x_init(TMR1);
|
|
|
|
timer_x_init(TMR2);
|
|
|
|
timer_x_init(TMR3);
|
2013-10-24 13:53:09 +02:00
|
|
|
|
2013-11-11 12:30:06 +01:00
|
|
|
register_isr(INT_NUM_TMR, &tmr_isr);
|
|
|
|
hwtimer_arch_enable_interrupt();
|
2013-07-23 10:36:16 +02:00
|
|
|
}
|
2013-04-25 13:30:09 +02:00
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*/
|
|
|
|
void hwtimer_arch_enable_interrupt(void) {
|
2013-04-25 14:56:32 +02:00
|
|
|
/* this enables timer interrupts in general by using the ITC.
|
|
|
|
* Timer specific interrupt control is given by the TMRx structs. */
|
2013-07-23 10:36:16 +02:00
|
|
|
//enable_irq(INT_NUM_TMR);
|
|
|
|
ITC->INTENABLEbits.TMR = 1;
|
2013-04-25 13:30:09 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*/
|
|
|
|
void hwtimer_arch_disable_interrupt(void) {
|
2013-04-25 14:56:32 +02:00
|
|
|
/* this disables timer interrupts in general by using the ITC.
|
|
|
|
* Timer specific interrupt control is given by the TMRx structs. */
|
2013-07-23 10:36:16 +02:00
|
|
|
//disable_irq(INT_NUM_TMR);
|
2013-11-11 12:30:06 +01:00
|
|
|
ITC->INTENABLEbits.TMR = 0;
|
2013-04-25 13:30:09 +02:00
|
|
|
}
|
2013-04-01 02:11:43 +02:00
|
|
|
|
2013-04-25 13:30:09 +02:00
|
|
|
/*---------------------------------------------------------------------------*/
|
|
|
|
void hwtimer_arch_set(unsigned long offset, short timer) {
|
|
|
|
/* get corresponding struct for the given ::timer parameter */
|
2013-11-11 12:30:06 +01:00
|
|
|
struct TMR_struct* tmr = (void *) TMR_BASE + (timer * TMR_OFFSET);
|
2014-02-11 18:15:43 +01:00
|
|
|
|
2013-04-25 13:30:09 +02:00
|
|
|
/* disable IRQs and save the status register */
|
2013-11-11 12:30:06 +01:00
|
|
|
uint32_t cpsr = disableIRQ();
|
|
|
|
|
|
|
|
TMR0->ENBL &= ~(1<<timer); /* disable timer */
|
2013-04-25 13:30:09 +02:00
|
|
|
tmr->COMP1 = tmr->CNTR + offset; /* load the current value + offset into the compare register */
|
|
|
|
tmr->CSCTRLbits.TCF1 = 0; /* reset compare flag */
|
|
|
|
tmr->CSCTRLbits.TCF1EN = 1; /* enable intterupts when TCF1 is set \ */
|
2013-11-11 12:30:06 +01:00
|
|
|
TMR0->ENBL |= (1<<timer); /* enable timer */
|
2014-02-11 18:15:43 +01:00
|
|
|
tmr->SCTRLbits.TCFIE = 1; /* enable interrupts when TCF is one - do we need both?*/
|
|
|
|
|
2013-11-11 12:30:06 +01:00
|
|
|
/* restore status register */
|
2013-04-25 13:30:09 +02:00
|
|
|
restoreIRQ(cpsr);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*/
|
|
|
|
void hwtimer_arch_set_absolute(unsigned long value, short timer) {
|
2013-04-25 13:32:50 +02:00
|
|
|
/* get corresponding struct for the given ::timer parameter */
|
2013-11-11 12:30:06 +01:00
|
|
|
struct TMR_struct* tmr = (void *) TMR_BASE + (timer * TMR_OFFSET);
|
2014-02-11 18:15:43 +01:00
|
|
|
|
2013-04-25 13:32:50 +02:00
|
|
|
/* disable IRQs and save the status register */
|
2013-11-11 12:30:06 +01:00
|
|
|
uint32_t cpsr = disableIRQ();
|
2014-02-11 18:15:43 +01:00
|
|
|
|
2013-11-11 12:30:06 +01:00
|
|
|
TMR0->ENBL &= ~(1<<timer); /* disable timer */
|
2013-04-25 13:32:50 +02:00
|
|
|
tmr->COMP1 = value; /* load the value into the compare register */
|
|
|
|
tmr->CSCTRLbits.TCF1 = 0; /* reset compare flag */
|
2013-04-25 13:38:55 +02:00
|
|
|
tmr->CSCTRLbits.TCF1EN = 1; /* enable interrupts when TCF1 is set \ */
|
2013-11-11 12:30:06 +01:00
|
|
|
TMR0->ENBL |= (1<<timer); /* enable timer */
|
2014-02-11 18:15:43 +01:00
|
|
|
tmr->SCTRLbits.TCFIE = 1; /* enable interrupts when TCF is one - do we need both?*/
|
|
|
|
|
2013-11-11 12:30:06 +01:00
|
|
|
/* restore status register */
|
2013-04-25 13:32:50 +02:00
|
|
|
restoreIRQ(cpsr);
|
2013-04-25 13:30:09 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*/
|
|
|
|
void hwtimer_arch_unset(short timer) {
|
2013-04-25 13:38:55 +02:00
|
|
|
/* get corresponding struct for the given ::timer parameter */
|
2013-07-23 10:36:16 +02:00
|
|
|
struct TMR_struct* tmr = (void *) TMR_BASE + (timer + TMR_OFFSET);
|
2014-02-11 18:15:43 +01:00
|
|
|
|
2013-04-25 13:38:55 +02:00
|
|
|
tmr->CSCTRLbits.TCF1 = 0; /* reset compare flag */
|
|
|
|
tmr->CSCTRLbits.TCF1EN = 0; /* disable interrupts for TCF1 */
|
|
|
|
tmr->SCTRLbits.TCFIE = 0; /* disable interrupts for TCF */
|
2013-04-25 13:30:09 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*/
|
|
|
|
unsigned long hwtimer_arch_now(void) {
|
2013-04-25 13:38:55 +02:00
|
|
|
return TMR0->CNTR;
|
2013-04-01 02:11:43 +02:00
|
|
|
}
|