2010-09-22 15:10:42 +02:00
|
|
|
/**
|
|
|
|
* hardware timer abstraction
|
|
|
|
*
|
2013-06-18 17:21:38 +02:00
|
|
|
* Copyright (C) 2013 Freie Universität Berlin
|
2010-09-22 15:10:42 +02:00
|
|
|
*
|
2013-06-18 17:21:38 +02:00
|
|
|
* This file subject to the terms and conditions of the GNU Lesser General
|
|
|
|
* Public License. See the file LICENSE in the top level directory for more
|
|
|
|
* details.
|
2010-09-22 15:10:42 +02:00
|
|
|
*
|
|
|
|
* @ingroup kernel
|
|
|
|
* @{
|
|
|
|
* @file
|
|
|
|
* @author Heiko Will <hwill@inf.fu-berlin.de>
|
|
|
|
* @author Thomas Hillebrandt <hillebra@inf.fu-berlin.de>
|
|
|
|
* @author Kaspar Schleiser <kaspar.schleiser@fu-berlin.de>
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
2010-11-01 15:50:31 +01:00
|
|
|
#include <hwtimer.h>
|
|
|
|
#include <hwtimer_cpu.h>
|
|
|
|
#include <hwtimer_arch.h>
|
2010-09-22 15:10:42 +02:00
|
|
|
|
2010-11-01 15:50:31 +01:00
|
|
|
#include <kernel.h>
|
2010-11-04 16:21:39 +01:00
|
|
|
#include <thread.h>
|
2010-12-10 16:50:16 +01:00
|
|
|
#include <lifo.h>
|
2010-09-22 15:10:42 +02:00
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
typedef struct hwtimer_t {
|
|
|
|
void (*callback)(void*);
|
2013-06-20 18:18:29 +02:00
|
|
|
void *data;
|
2010-09-22 15:10:42 +02:00
|
|
|
} hwtimer_t;
|
|
|
|
|
2010-12-10 16:50:16 +01:00
|
|
|
static hwtimer_t timer[ARCH_MAXTIMERS];
|
2013-06-20 18:18:29 +02:00
|
|
|
static int lifo[ARCH_MAXTIMERS + 1];
|
2010-09-22 15:10:42 +02:00
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*/
|
|
|
|
|
2013-06-20 18:18:29 +02:00
|
|
|
static void multiplexer(int source)
|
|
|
|
{
|
2010-12-10 16:50:16 +01:00
|
|
|
lifo_insert(lifo, source);
|
|
|
|
lpm_prevent_sleep--;
|
|
|
|
|
2010-09-22 15:10:42 +02:00
|
|
|
timer[source].callback(timer[source].data);
|
|
|
|
}
|
|
|
|
|
2013-06-20 18:18:29 +02:00
|
|
|
static void hwtimer_wakeup(void *ptr)
|
|
|
|
{
|
2010-12-10 16:50:16 +01:00
|
|
|
int pid = (int)ptr;
|
|
|
|
thread_wakeup(pid);
|
2010-09-22 15:10:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void hwtimer_spin(unsigned long ticks)
|
|
|
|
{
|
|
|
|
unsigned long co = hwtimer_arch_now() + ticks;
|
2013-06-20 18:18:29 +02:00
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
while (hwtimer_arch_now() > co);
|
2013-06-20 18:18:29 +02:00
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
while (hwtimer_arch_now() < co);
|
2010-09-22 15:10:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*/
|
|
|
|
|
2013-06-20 18:18:29 +02:00
|
|
|
void hwtimer_init(void)
|
|
|
|
{
|
2010-09-22 15:10:42 +02:00
|
|
|
hwtimer_init_comp(F_CPU);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*/
|
|
|
|
|
2013-06-20 18:18:29 +02:00
|
|
|
void hwtimer_init_comp(uint32_t fcpu)
|
|
|
|
{
|
2010-09-22 15:10:42 +02:00
|
|
|
hwtimer_arch_init(multiplexer, fcpu);
|
2013-06-20 18:18:29 +02:00
|
|
|
|
2010-12-10 16:50:16 +01:00
|
|
|
lifo_init(lifo, ARCH_MAXTIMERS);
|
2013-06-20 18:18:29 +02:00
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
for (int i = 0; i < ARCH_MAXTIMERS; i++) {
|
2010-12-10 16:50:16 +01:00
|
|
|
lifo_insert(lifo, i);
|
2010-09-22 15:10:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*/
|
|
|
|
|
2013-06-20 18:18:29 +02:00
|
|
|
int hwtimer_active(void)
|
|
|
|
{
|
2013-06-24 22:37:35 +02:00
|
|
|
return (!lifo_empty(lifo));
|
2010-09-22 15:10:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
unsigned long hwtimer_now(void)
|
|
|
|
{
|
|
|
|
return hwtimer_arch_now();
|
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
void hwtimer_wait(unsigned long ticks)
|
|
|
|
{
|
2013-06-24 22:37:35 +02:00
|
|
|
if (ticks <= 6 || inISR()) {
|
2010-09-22 15:10:42 +02:00
|
|
|
hwtimer_spin(ticks);
|
|
|
|
return;
|
|
|
|
}
|
2013-06-20 18:18:29 +02:00
|
|
|
|
2010-11-04 16:21:39 +01:00
|
|
|
/* -2 is to adjust the real value */
|
2013-06-20 18:18:29 +02:00
|
|
|
int res = hwtimer_set(ticks - 2, hwtimer_wakeup, (void*)(unsigned int)(active_thread->pid));
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (res == -1) {
|
2010-09-22 15:10:42 +02:00
|
|
|
hwtimer_spin(ticks);
|
|
|
|
return;
|
|
|
|
}
|
2010-12-10 16:50:16 +01:00
|
|
|
|
|
|
|
thread_sleep();
|
2010-09-22 15:10:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*/
|
|
|
|
|
2012-03-08 18:58:39 +01:00
|
|
|
|
2010-09-22 15:10:42 +02:00
|
|
|
static int _hwtimer_set(unsigned long offset, void (*callback)(void*), void *ptr, bool absolute)
|
|
|
|
{
|
2013-06-24 22:37:35 +02:00
|
|
|
if (!inISR()) {
|
2010-11-04 16:21:39 +01:00
|
|
|
dINT();
|
|
|
|
}
|
2010-12-10 16:50:16 +01:00
|
|
|
|
|
|
|
int n = lifo_get(lifo);
|
2013-06-20 18:18:29 +02:00
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (n == -1) {
|
|
|
|
if (!inISR()) {
|
2010-11-04 16:21:39 +01:00
|
|
|
eINT();
|
|
|
|
}
|
2013-06-20 18:18:29 +02:00
|
|
|
|
2010-12-10 16:50:16 +01:00
|
|
|
puts("No hwtimer left.");
|
2010-09-22 15:10:42 +02:00
|
|
|
return -1;
|
|
|
|
}
|
2013-06-20 18:18:29 +02:00
|
|
|
|
2010-12-10 16:50:16 +01:00
|
|
|
timer[n].callback = callback;
|
|
|
|
timer[n].data = ptr;
|
2010-09-22 15:10:42 +02:00
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (absolute) {
|
2010-12-10 16:50:16 +01:00
|
|
|
hwtimer_arch_set_absolute(offset, n);
|
2010-11-04 16:21:39 +01:00
|
|
|
}
|
|
|
|
else {
|
2010-12-10 16:50:16 +01:00
|
|
|
hwtimer_arch_set(offset, n);
|
2010-11-04 16:21:39 +01:00
|
|
|
}
|
2010-09-22 15:10:42 +02:00
|
|
|
|
2010-12-10 16:50:16 +01:00
|
|
|
lpm_prevent_sleep++;
|
|
|
|
|
2013-06-24 22:37:35 +02:00
|
|
|
if (!inISR()) {
|
2010-11-04 16:21:39 +01:00
|
|
|
eINT();
|
|
|
|
}
|
2013-06-20 18:18:29 +02:00
|
|
|
|
2010-12-10 16:50:16 +01:00
|
|
|
return n;
|
2010-09-22 15:10:42 +02:00
|
|
|
}
|
|
|
|
|
2013-06-20 18:18:29 +02:00
|
|
|
int hwtimer_set(unsigned long offset, void (*callback)(void*), void *ptr)
|
|
|
|
{
|
2010-09-22 15:10:42 +02:00
|
|
|
return _hwtimer_set(offset, callback, ptr, false);
|
|
|
|
}
|
|
|
|
|
2013-06-20 18:18:29 +02:00
|
|
|
int hwtimer_set_absolute(unsigned long offset, void (*callback)(void*), void *ptr)
|
|
|
|
{
|
2010-09-22 15:10:42 +02:00
|
|
|
return _hwtimer_set(offset, callback, ptr, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*/
|
|
|
|
|
2010-12-10 16:50:16 +01:00
|
|
|
int hwtimer_remove(int n)
|
2010-09-22 15:10:42 +02:00
|
|
|
{
|
|
|
|
hwtimer_arch_disable_interrupt();
|
2010-12-10 16:50:16 +01:00
|
|
|
hwtimer_arch_unset(n);
|
|
|
|
|
|
|
|
lifo_insert(lifo, n);
|
|
|
|
timer[n].callback = NULL;
|
|
|
|
|
|
|
|
lpm_prevent_sleep--;
|
2013-06-20 18:18:29 +02:00
|
|
|
|
2010-09-22 15:10:42 +02:00
|
|
|
hwtimer_arch_enable_interrupt();
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|