/* * Copyright (C) 2015 Rakendra Thapa * @} */ #include #include "cpu.h" #include "sched.h" #include "thread.h" #include "periph_conf.h" #include "periph/timer.h" #include "mutex.h" #define ENABLE_DEBUG (0) #include "debug.h" /* guard file in case no timers are defined */ #if TIMER_0_EN /** * @brief Struct holding the configuration data */ typedef struct { void (*cb)(int); /**< timeout callback */ } timer_conf_t; static timer_conf_t config[TIMER_NUMOF]; int timer_init(tim_t dev, unsigned int us_per_tick, void (*callback)(int)) { if (dev == TIMER_0) { config[dev].cb = callback; // User Function ROM_SysCtlPeripheralEnable(SYSCTL_PERIPH_WTIMER0); // Activate Timer0 WTIMER0_CTL_R &= ~0x00000001; // Disable timer0A during setup WTIMER0_CFG_R = TIMER_CFG_16_BIT; WTIMER0_TAMR_R = TIMER_TAMR_TAMR_PERIOD; // Configure for periodic mode WTIMER0_TAPR_R = TIMER_0_PRESCALER; // 1us timer0A WTIMER0_ICR_R = 0x00000001; // clear timer0A timeout flag WTIMER0_IMR_R |= 0x00000001; // arm timeout interrupt ROM_IntPrioritySet(INT_WTIMER0A, 32); timer_irq_enable(dev); timer_start(dev); DEBUG("startTimeout Value=0x%lx\n", ROM_TimerValueGet(WTIMER0_BASE, TIMER_A)); return 1; } return -1; } int timer_set(tim_t dev, int channel, unsigned int timeout) { if (dev == TIMER_0) { unsigned int now = timer_read(dev); DEBUG("timer_set now=0x%x\n",now); DEBUG("timer_set timeout=0x%x\n", timeout); return timer_set_absolute(dev, channel, now+timeout); } return -1; } int timer_set_absolute(tim_t dev, int channel, unsigned int value) { if (dev == TIMER_0) { WTIMER0_TAILR_R = 0x00000000 | value; // period; Reload value DEBUG("Setting timer absolute value=0x%x\n", value); return 1; } return -1; } int timer_clear(tim_t dev, int channel) { if (dev == TIMER_0){ WTIMER0_ICR_R = TIMER_ICR_TATOCINT; // clear timer0A timeout flag return 1; } return -1; } unsigned int timer_read(tim_t dev) { if (dev == TIMER_0) { unsigned int currTimer0Val=0; unsigned int loadTimer0Val=0; currTimer0Val = (unsigned int)ROM_TimerValueGet(WTIMER0_BASE, TIMER_A); loadTimer0Val = (unsigned int)ROM_TimerLoadGet(WTIMER0_BASE, TIMER_A); DEBUG("WTIMER0_TAILR_R=0x%lx\t currTimer0Val=0x%x\t loadTimer0Val=0x%x\n", WTIMER0_TAILR_R, currTimer0Val, loadTimer0Val); return (loadTimer0Val - currTimer0Val); } return 0; } void timer_start(tim_t dev) { if (dev == TIMER_0) { ROM_TimerEnable(WTIMER0_BASE, TIMER_A); } } void timer_stop(tim_t dev) { if (dev == TIMER_0) { ROM_TimerDisable(WTIMER0_BASE, TIMER_A); } } void timer_irq_enable(tim_t dev) { if (dev == TIMER_0) { ROM_IntEnable(INT_WTIMER0A); ROM_TimerIntEnable(WTIMER0_BASE, TIMER_TIMA_TIMEOUT); } } void timer_irq_disable(tim_t dev) { if (dev == TIMER_0) { ROM_IntDisable(INT_WTIMER0A); } } void timer_reset(tim_t dev) { if (dev == TIMER_0) { // Performs a software reset of a peripheral ROM_SysCtlPeripheralReset(SYSCTL_PERIPH_WTIMER0); } } #if TIMER_0_EN void TIMER0IntHandler(void) { TIMER0_ICR_R = TIMER_ICR_TATOCINT; // acknowledge timer0A timeout config[TIMER_0].cb(0); if (sched_context_switch_request){ thread_yield(); } } void WTIMER0IntHandler(void) { WTIMER0_ICR_R = TIMER_ICR_TATOCINT; // acknowledge timer0A timeout config[TIMER_0].cb(0); if (sched_context_switch_request){ thread_yield(); } } #endif #endif /* TIMER_0_EN */