2015-07-04 09:37:54 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2015 Rakendra Thapa <rakendrathapa@gmail.com
|
|
|
|
*
|
|
|
|
* 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
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Implementation of the low-level timer driver for the LM4F120
|
|
|
|
*
|
|
|
|
* @author Rakendra Thapa <rakendrathapa@gmail.com>
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdint.h>
|
|
|
|
|
|
|
|
#include "cpu.h"
|
|
|
|
#include "sched.h"
|
|
|
|
#include "thread.h"
|
|
|
|
#include "periph_conf.h"
|
|
|
|
#include "periph/timer.h"
|
2015-07-10 04:06:50 +02:00
|
|
|
#include "mutex.h"
|
2015-07-04 09:37:54 +02:00
|
|
|
|
2015-07-10 04:06:50 +02:00
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
2015-07-04 09:37:54 +02:00
|
|
|
/* 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) {
|
2015-07-15 11:48:31 +02:00
|
|
|
config[dev].cb = callback; // User Function
|
2015-07-14 18:11:25 +02:00
|
|
|
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;
|
2015-07-04 09:37:54 +02:00
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int timer_set(tim_t dev, int channel, unsigned int timeout)
|
|
|
|
{
|
|
|
|
if (dev == TIMER_0) {
|
|
|
|
unsigned int now = timer_read(dev);
|
2015-07-14 18:11:25 +02:00
|
|
|
DEBUG("timer_set now=0x%x\n",now);
|
|
|
|
DEBUG("timer_set timeout=0x%x\n", timeout);
|
2015-07-10 04:06:50 +02:00
|
|
|
return timer_set_absolute(dev, channel, now+timeout);
|
2015-07-04 09:37:54 +02:00
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int timer_set_absolute(tim_t dev, int channel, unsigned int value)
|
|
|
|
{
|
|
|
|
if (dev == TIMER_0) {
|
2015-07-14 18:11:25 +02:00
|
|
|
WTIMER0_TAILR_R = 0x00000000 | value; // period; Reload value
|
|
|
|
DEBUG("Setting timer absolute value=0x%x\n", value);
|
2015-07-04 09:37:54 +02:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int timer_clear(tim_t dev, int channel)
|
|
|
|
{
|
|
|
|
if (dev == TIMER_0){
|
2015-07-14 18:11:25 +02:00
|
|
|
WTIMER0_ICR_R = TIMER_ICR_TATOCINT; // clear timer0A timeout flag
|
2015-07-04 09:37:54 +02:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned int timer_read(tim_t dev)
|
|
|
|
{
|
|
|
|
if (dev == TIMER_0) {
|
2015-07-14 18:11:25 +02:00
|
|
|
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);
|
2015-07-10 04:06:50 +02:00
|
|
|
return (loadTimer0Val - currTimer0Val);
|
2015-07-04 09:37:54 +02:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void timer_start(tim_t dev)
|
|
|
|
{
|
|
|
|
if (dev == TIMER_0) {
|
2015-07-14 18:11:25 +02:00
|
|
|
ROM_TimerEnable(WTIMER0_BASE, TIMER_A);
|
2015-07-04 09:37:54 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void timer_stop(tim_t dev)
|
|
|
|
{
|
|
|
|
if (dev == TIMER_0) {
|
2015-07-14 18:11:25 +02:00
|
|
|
ROM_TimerDisable(WTIMER0_BASE, TIMER_A);
|
2015-07-04 09:37:54 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void timer_irq_enable(tim_t dev)
|
|
|
|
{
|
|
|
|
if (dev == TIMER_0) {
|
2015-07-14 18:11:25 +02:00
|
|
|
ROM_IntEnable(INT_WTIMER0A);
|
|
|
|
ROM_TimerIntEnable(WTIMER0_BASE, TIMER_TIMA_TIMEOUT);
|
2015-07-04 09:37:54 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void timer_irq_disable(tim_t dev)
|
|
|
|
{
|
|
|
|
if (dev == TIMER_0) {
|
2015-07-14 18:11:25 +02:00
|
|
|
ROM_IntDisable(INT_WTIMER0A);
|
2015-07-04 09:37:54 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void timer_reset(tim_t dev)
|
|
|
|
{
|
|
|
|
if (dev == TIMER_0) {
|
2015-07-14 18:11:25 +02:00
|
|
|
// Performs a software reset of a peripheral
|
|
|
|
ROM_SysCtlPeripheralReset(SYSCTL_PERIPH_WTIMER0);
|
2015-07-04 09:37:54 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#if TIMER_0_EN
|
2015-07-14 00:57:52 +02:00
|
|
|
void isr_timer0(void)
|
2015-07-04 09:37:54 +02:00
|
|
|
{
|
2015-07-14 18:11:25 +02:00
|
|
|
TIMER0_ICR_R = TIMER_ICR_TATOCINT; // acknowledge timer0A timeout
|
|
|
|
config[TIMER_0].cb(0);
|
2015-07-10 04:06:50 +02:00
|
|
|
|
2015-07-14 18:11:25 +02:00
|
|
|
if (sched_context_switch_request){
|
|
|
|
thread_yield();
|
|
|
|
}
|
2015-07-04 09:37:54 +02:00
|
|
|
}
|
2015-07-14 00:57:52 +02:00
|
|
|
void isr_wtimer0(void)
|
2015-07-04 09:37:54 +02:00
|
|
|
{
|
2015-07-14 18:11:25 +02:00
|
|
|
WTIMER0_ICR_R = TIMER_ICR_TATOCINT; // acknowledge timer0A timeout
|
2015-07-15 11:48:31 +02:00
|
|
|
|
2015-07-14 18:11:25 +02:00
|
|
|
config[TIMER_0].cb(0);
|
|
|
|
if (sched_context_switch_request){
|
|
|
|
thread_yield();
|
|
|
|
}
|
2015-07-04 09:37:54 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#endif /* TIMER_0_EN */
|