2014-02-02 16:48:18 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2014 INRIA
|
|
|
|
*
|
|
|
|
* The source code is licensed under the LGPLv2 license,
|
|
|
|
* See the file LICENSE for more details.
|
|
|
|
*/
|
|
|
|
|
2010-09-22 15:10:42 +02:00
|
|
|
/**
|
2014-02-02 16:48:18 +01:00
|
|
|
* @ingroup cpu
|
2010-09-22 15:10:42 +02:00
|
|
|
* @file
|
|
|
|
* @internal
|
|
|
|
* @brief ARM kernel timer CPU dependent functions implementation
|
|
|
|
*
|
|
|
|
* @author Thomas Hillebrandt <hillebra@inf.fu-berlin.de>
|
|
|
|
* @author Heiko Will <hwill@inf.fu-berlin.de>
|
2013-06-21 03:52:57 +02:00
|
|
|
* @author Oliver Hahm <oliver.hahm@inria.fr>
|
2010-09-22 15:10:42 +02:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "cpu.h"
|
|
|
|
#include "bitarithm.h"
|
|
|
|
#include "hwtimer_cpu.h"
|
2014-07-09 21:08:13 +02:00
|
|
|
#include "arch/hwtimer_arch.h"
|
2012-11-08 17:37:29 +01:00
|
|
|
#include "irq.h"
|
2010-09-22 15:10:42 +02:00
|
|
|
|
|
|
|
#define VULP(x) ((volatile unsigned long*) (x))
|
|
|
|
|
2013-04-02 02:04:19 +02:00
|
|
|
/* High level interrupt handler */
|
2010-09-22 15:10:42 +02:00
|
|
|
static void (*int_handler)(int);
|
|
|
|
|
2013-04-02 02:04:19 +02:00
|
|
|
/* Timer 0-3 interrupt handler */
|
2010-09-22 15:10:42 +02:00
|
|
|
static void timer_irq(void) __attribute__((interrupt("IRQ")));
|
|
|
|
|
2013-06-21 03:52:57 +02:00
|
|
|
inline static unsigned long get_base_address(short timer)
|
|
|
|
{
|
|
|
|
return (volatile unsigned long)(TMR0_BASE_ADDR + (timer / 8) * 0x6C000 + (timer / 4 - (timer / 8) * 2) * 0x4000);
|
2010-09-22 15:10:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void timer_irq(void)
|
|
|
|
{
|
|
|
|
short timer = 0;
|
2013-06-21 03:52:57 +02:00
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (T0IR) {
|
2010-09-22 15:10:42 +02:00
|
|
|
timer = 0;
|
2013-06-21 03:52:57 +02:00
|
|
|
}
|
2013-06-24 22:37:35 +02:00
|
|
|
else if (T1IR) {
|
2010-09-22 15:10:42 +02:00
|
|
|
timer = 4;
|
2013-06-21 03:52:57 +02:00
|
|
|
}
|
2013-06-24 22:37:35 +02:00
|
|
|
else if (T2IR) {
|
2010-09-22 15:10:42 +02:00
|
|
|
timer = 8;
|
|
|
|
}
|
|
|
|
|
|
|
|
volatile unsigned long base = get_base_address(timer);
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (*VULP(base + TXIR) & BIT0) {
|
2013-06-21 03:52:57 +02:00
|
|
|
*VULP(base + TXMCR) &= ~BIT0;
|
|
|
|
*VULP(base + TXIR) = BIT0;
|
2010-09-22 15:10:42 +02:00
|
|
|
int_handler(timer);
|
|
|
|
}
|
2013-06-21 03:52:57 +02:00
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (*VULP(base + TXIR) & BIT1) {
|
2013-06-21 03:52:57 +02:00
|
|
|
*VULP(base + TXMCR) &= ~BIT3;
|
|
|
|
*VULP(base + TXIR) = BIT1;
|
2010-09-22 15:10:42 +02:00
|
|
|
int_handler(timer + 1);
|
|
|
|
}
|
2013-06-21 03:52:57 +02:00
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (*VULP(base + TXIR) & BIT2) {
|
2013-06-21 03:52:57 +02:00
|
|
|
*VULP(base + TXMCR) &= ~BIT6;
|
|
|
|
*VULP(base + TXIR) = BIT2;
|
2010-09-22 15:10:42 +02:00
|
|
|
int_handler(timer + 2);
|
|
|
|
}
|
2013-06-21 03:52:57 +02:00
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (*VULP(base + TXIR) & BIT3) {
|
2013-06-21 03:52:57 +02:00
|
|
|
*VULP(base + TXMCR) &= ~BIT9;
|
|
|
|
*VULP(base + TXIR) = BIT3;
|
2010-09-22 15:10:42 +02:00
|
|
|
int_handler(timer + 3);
|
|
|
|
}
|
|
|
|
|
2013-06-21 03:52:57 +02:00
|
|
|
VICVectAddr = 0; /* acknowledge interrupt (if using VIC IRQ) */
|
2010-09-22 15:10:42 +02:00
|
|
|
}
|
|
|
|
|
2013-06-21 03:52:57 +02:00
|
|
|
static void timer0_init(uint32_t cpsr)
|
|
|
|
{
|
|
|
|
PCONP |= PCTIM0; /* power up timer */
|
|
|
|
T0TCR = 2; /* disable and reset timer */
|
|
|
|
T0MCR = 0; /* disable compare */
|
|
|
|
T0CCR = 0; /* capture is disabled */
|
|
|
|
T0EMR = 0; /* no external match output */
|
|
|
|
T0PR = cpsr; /* set prescaler */
|
2010-09-22 15:10:42 +02:00
|
|
|
install_irq(TIMER0_INT, &timer_irq, 1);
|
2013-06-21 03:52:57 +02:00
|
|
|
T0TCR = 1; /* reset counter */
|
2010-09-22 15:10:42 +02:00
|
|
|
}
|
|
|
|
|
2013-06-21 03:52:57 +02:00
|
|
|
static void timer1_init(uint32_t cpsr)
|
|
|
|
{
|
|
|
|
PCONP |= PCTIM1; /* power up timer */
|
|
|
|
T1TCR = 2; /* disable and reset timer */
|
|
|
|
T1MCR = 0; /* disable compare */
|
|
|
|
T1CCR = 0; /* capture is disabled */
|
|
|
|
T1EMR = 0; /* no external match output */
|
|
|
|
T1PR = cpsr; /* set prescaler */
|
2010-09-22 15:10:42 +02:00
|
|
|
install_irq(TIMER1_INT, &timer_irq, 1);
|
2013-06-21 03:52:57 +02:00
|
|
|
T1TCR = 1; /* reset counter */
|
2010-09-22 15:10:42 +02:00
|
|
|
}
|
|
|
|
|
2013-06-21 03:52:57 +02:00
|
|
|
static void timer2_init(uint32_t cpsr)
|
|
|
|
{
|
|
|
|
PCONP |= PCTIM2; /* power up timer */
|
|
|
|
T2TCR = 2; /* disable and reset timer */
|
|
|
|
T2MCR = 0; /* disable compare */
|
|
|
|
T2CCR = 0; /* capture is disabled */
|
|
|
|
T2EMR = 0; /* no external match output */
|
|
|
|
T2PR = cpsr; /* set prescaler */
|
2010-09-22 15:10:42 +02:00
|
|
|
install_irq(TIMER2_INT, &timer_irq, 1);
|
2013-06-21 03:52:57 +02:00
|
|
|
T2TCR = 1; /* reset counter */
|
2010-09-22 15:10:42 +02:00
|
|
|
}
|
|
|
|
|
2013-06-21 03:52:57 +02:00
|
|
|
void hwtimer_arch_init(void (*handler)(int), uint32_t fcpu)
|
|
|
|
{
|
2010-09-22 15:10:42 +02:00
|
|
|
uint32_t cpsr;
|
|
|
|
int_handler = handler;
|
|
|
|
cpu_clock_scale(fcpu, HWTIMER_SPEED, &cpsr);
|
|
|
|
timer0_init(cpsr);
|
|
|
|
timer1_init(cpsr);
|
|
|
|
timer2_init(cpsr);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*/
|
|
|
|
|
2013-06-21 03:52:57 +02:00
|
|
|
void hwtimer_arch_enable_interrupt(void)
|
|
|
|
{
|
2010-09-22 15:10:42 +02:00
|
|
|
VICIntEnable = 1 << TIMER0_INT; /* Enable Interrupt */
|
|
|
|
VICIntEnable = 1 << TIMER1_INT; /* Enable Interrupt */
|
|
|
|
VICIntEnable = 1 << TIMER2_INT; /* Enable Interrupt */
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*/
|
|
|
|
|
2013-06-21 03:52:57 +02:00
|
|
|
void hwtimer_arch_disable_interrupt(void)
|
|
|
|
{
|
2010-09-22 15:10:42 +02:00
|
|
|
VICIntEnClr = 1 << TIMER0_INT; /* Disable Interrupt */
|
|
|
|
VICIntEnClr = 1 << TIMER1_INT; /* Disable Interrupt */
|
|
|
|
VICIntEnClr = 1 << TIMER2_INT; /* Disable Interrupt */
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*/
|
|
|
|
|
2013-06-21 03:52:57 +02:00
|
|
|
void hwtimer_arch_set(unsigned long offset, short timer)
|
|
|
|
{
|
|
|
|
/* Calculate base address of timer register */
|
|
|
|
/* Timer 0-3 are matched to TIMER0 */
|
|
|
|
/* Timer 4-7 are matched to TIMER1 */
|
|
|
|
/* Timer 8-11 are matched to TIMER2 */
|
2010-09-22 15:10:42 +02:00
|
|
|
volatile unsigned long base = get_base_address(timer);
|
2013-06-21 03:52:57 +02:00
|
|
|
/* Calculate match register address of corresponding timer */
|
2010-09-22 15:10:42 +02:00
|
|
|
timer %= 4;
|
2012-03-08 18:58:39 +01:00
|
|
|
unsigned long cpsr = disableIRQ();
|
2013-06-21 03:52:57 +02:00
|
|
|
volatile unsigned long *addr = VULP(base + TXMR0 + 4 * timer);
|
|
|
|
/* Calculate match register value */
|
2010-09-22 15:10:42 +02:00
|
|
|
unsigned long value = *VULP(base + TXTC) + offset;
|
2013-06-21 03:52:57 +02:00
|
|
|
*addr = value; /* set match register */
|
|
|
|
*VULP(base + TXIR) = 0x01 << timer; /* reset interrupt register value for corresponding match register */
|
|
|
|
*VULP(base + TXMCR) &= ~(7 << (3 * timer)); /* Clear all bits */
|
|
|
|
*VULP(base + TXMCR) |= (MR0I << (3 * timer)); /* enable interrupt for match register */
|
2012-03-08 18:58:39 +01:00
|
|
|
restoreIRQ(cpsr);
|
2010-09-22 15:10:42 +02:00
|
|
|
}
|
|
|
|
|
2013-06-21 03:52:57 +02:00
|
|
|
void hwtimer_arch_set_absolute(unsigned long value, short timer)
|
|
|
|
{
|
|
|
|
/* Calculate base address of timer register */
|
|
|
|
/* Timer 0-3 are matched to TIMER0 */
|
|
|
|
/* Timer 4-7 are matched to TIMER1 */
|
|
|
|
/* Timer 8-11 are matched to TIMER2 */
|
2010-09-22 15:10:42 +02:00
|
|
|
volatile unsigned long base = get_base_address(timer);
|
2013-06-21 03:52:57 +02:00
|
|
|
/* Calculate match register address of corresponding timer */
|
2010-09-22 15:10:42 +02:00
|
|
|
timer %= 4;
|
2013-06-21 03:52:57 +02:00
|
|
|
volatile unsigned long *addr = VULP(base + TXMR0 + 4 * timer);
|
|
|
|
/* Calculate match register value */
|
|
|
|
*addr = value; /* set match register */
|
|
|
|
*VULP(base + TXIR) = 0x01 << timer; /* reset interrupt register value for corresponding match register */
|
|
|
|
*VULP(base + TXMCR) &= ~(7 << (3 * timer)); /* Clear all bits */
|
|
|
|
*VULP(base + TXMCR) |= (MR0I << (3 * timer)); /* enable interrupt for match register */
|
2010-09-22 15:10:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*/
|
|
|
|
|
2013-06-21 03:52:57 +02:00
|
|
|
void hwtimer_arch_unset(short timer)
|
|
|
|
{
|
2010-09-22 15:10:42 +02:00
|
|
|
volatile unsigned long base = get_base_address(timer);
|
|
|
|
timer %= 4;
|
2013-06-21 03:52:57 +02:00
|
|
|
*VULP(base + TXMCR) &= ~(MR0I << (3 * timer)); /* disable interrupt for match register */
|
|
|
|
*VULP(base + TXIR) = 0x01 << timer; /* reset interrupt register value for corresponding match register */
|
2010-09-22 15:10:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*/
|
|
|
|
|
2013-06-21 03:52:57 +02:00
|
|
|
unsigned long hwtimer_arch_now(void)
|
|
|
|
{
|
2010-09-22 15:10:42 +02:00
|
|
|
return T0TC;
|
|
|
|
}
|
2010-09-24 16:27:10 +02:00
|
|
|
|
2013-06-21 03:52:57 +02:00
|
|
|
void hwtimer_arch_setcounter(unsigned int val)
|
|
|
|
{
|
2010-09-24 16:27:10 +02:00
|
|
|
T0TC = val;
|
|
|
|
}
|