/* * Copyright (C) 2015 Rakendra Thapa * * This file is subject to the terms and conditions of the GNU Lesser General * Public License v2.1. See the file LICENSE in the top level directory for more * details. */ /** * @ingroup cpu_lm4f120 * @ingroup drivers_periph_timer * @{ * * @file timer.c * @brief Implementation of the low-level timer driver for the LM4F120 * * @author Rakendra Thapa * Marc Poulhiès */ #include #include "cpu.h" #include "periph_conf.h" #include "periph/timer.h" #include "mutex.h" #define ENABLE_DEBUG (0) #include "debug.h" /** * @brief Interrupt context for each configured timer * @{ */ static timer_isr_ctx_t isr_ctx[TIMER_NUMOF]; static uint32_t isr_divisor[TIMER_NUMOF]; /**@}*/ #include "hw_timer.h" /* enable timer interrupts */ static inline void _irq_enable(tim_t tim); /* Missing from driverlib */ static inline unsigned long PRIV_TimerPrescaleSnapshotGet(unsigned long ulbase, unsigned long ultimer) { return ((ultimer == TIMER_A) ? HWREG(ulbase + TIMER_O_TAPS) : HWREG(ulbase + TIMER_O_TBPS)); } static inline uint64_t _scaled_to_ll_value(uint32_t uncorrected, uint32_t divisor) { const uint64_t scaledv = (uint64_t) uncorrected * divisor; return scaledv; } static inline uint32_t _llvalue_to_scaled_value(uint64_t corrected, uint32_t divisor) { const uint64_t scaledv = corrected / divisor; return scaledv; } int timer_init(tim_t tim, unsigned long freq, timer_cb_t cb, void *arg) { if (tim >= TIMER_NUMOF){ return -1; } const timer_conf_t *cfg = &timer_config[tim]; isr_ctx[tim].cb = cb; isr_ctx[tim].arg = arg; isr_divisor[tim] = ROM_SysCtlClockGet() / freq; unsigned int timer_side = TIMER_A; unsigned int timer_cfg = TIMER_CFG_SPLIT_PAIR | TIMER_CFG_A_PERIODIC_UP | TIMER_TAMR_TAMIE; unsigned int timer_intbit = TIMER_TIMA_TIMEOUT | TIMER_TIMA_MATCH; ROM_SysCtlPeripheralEnable(cfg->sysctl); ROM_TimerDisable(cfg->dev, timer_side); ROM_TimerConfigure(cfg->dev, timer_cfg); uint64_t val_max = _scaled_to_ll_value(cfg->max, isr_divisor[tim]); ROM_TimerPrescaleSet(cfg->dev, timer_side, val_max >> 32); ROM_TimerLoadSet(cfg->dev, timer_side, val_max & 0xFFFFFFFF); ROM_TimerIntClear(cfg->dev, timer_intbit); ROM_TimerIntEnable(cfg->dev, timer_intbit); _irq_enable(tim); timer_start(tim); return 0; } int timer_set_absolute(tim_t tim, int channel, unsigned int value) { if (tim >= TIMER_NUMOF || channel >= timer_config[tim].channels) { return -1; } const timer_conf_t *cfg = &timer_config[tim]; unsigned int timer_side = TIMER_A; unsigned long long scaledv; ROM_TimerDisable(cfg->dev, timer_side); scaledv = _scaled_to_ll_value(value, isr_divisor[tim]); if (scaledv>>32){ ROM_TimerPrescaleMatchSet(cfg->dev, timer_side, scaledv >> 32); } else { ROM_TimerPrescaleMatchSet(cfg->dev, timer_side, 0); } ROM_TimerMatchSet(cfg->dev, timer_side, (uint32_t) (scaledv & 0xFFFFFFFF)); ROM_TimerEnable(cfg->dev, timer_side); return 0; } int timer_clear(tim_t tim, int channel) { if (tim >= TIMER_NUMOF || channel >= timer_config[tim].channels) { return -1; } ROM_TimerIntClear(timer_config[tim].dev, TIMER_TIMA_TIMEOUT); return 0; } unsigned int timer_read(tim_t tim) { unsigned int timer_side = TIMER_A; unsigned long long high_bits, high_bits_dup; unsigned long long low_bits; unsigned long long total; unsigned int scaled_value; if (tim >= TIMER_NUMOF){ return -1; } const timer_conf_t *cfg = &timer_config[tim]; /* handle overflow happening between the 2 register reads */ do { high_bits = ((uint64_t)PRIV_TimerPrescaleSnapshotGet(cfg->dev, timer_side)) << 32; low_bits = (uint64_t)ROM_TimerValueGet(cfg->dev, timer_side); high_bits_dup = ((uint64_t)PRIV_TimerPrescaleSnapshotGet(cfg->dev, timer_side)) << 32; } while (high_bits != high_bits_dup); total = high_bits + low_bits; DEBUG("Combined %lx:%lx\n", (uint32_t)(total>>32), (uint32_t)(total & 0xFFFFFFFF)); scaled_value = _llvalue_to_scaled_value(total, isr_divisor[tim]); return scaled_value; } void timer_start(tim_t tim) { if (tim >= TIMER_NUMOF){ return; } ROM_TimerEnable(timer_config[tim].dev, TIMER_A); } void timer_stop(tim_t tim) { if (tim >= TIMER_NUMOF){ return; } ROM_TimerDisable(timer_config[tim].dev, TIMER_A); } static inline void _irq_enable(tim_t tim) { if (tim >= TIMER_NUMOF){ return; } ROM_IntPrioritySet(timer_config[tim].intbase, 32); ROM_IntEnable(timer_config[tim].intbase); } void _isr_timer(tim_t tim) { /* Clears both IT */ ROM_TimerIntClear(timer_config[tim].dev, TIMER_TIMA_TIMEOUT | TIMER_TIMA_MATCH); isr_ctx[tim].cb(isr_ctx[tim].arg, 0); cortexm_isr_end(); } #ifdef TIMER_0_ISR void TIMER_0_ISR(void) { _isr_timer(0); } #endif /* TIMER_0_ISR */ #ifdef TIMER_1_ISR void TIMER_1_ISR(void) { _isr_timer(1); } #endif /* TIMER_1_ISR */ /** @} */