2015-08-14 15:56:47 +02:00
|
|
|
/**
|
|
|
|
* Copyright (C) 2015 Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
*
|
|
|
|
* 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 xtimer
|
|
|
|
* @{
|
|
|
|
* @file
|
|
|
|
* @brief xtimer convenience functionality
|
|
|
|
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <assert.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "xtimer.h"
|
|
|
|
#include "mutex.h"
|
|
|
|
#include "thread.h"
|
|
|
|
#include "irq.h"
|
|
|
|
|
|
|
|
#include "timex.h"
|
|
|
|
|
|
|
|
#define ENABLE_DEBUG 0
|
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
static void _callback_unlock_mutex(void* arg)
|
|
|
|
{
|
|
|
|
mutex_t *mutex = (mutex_t *) arg;
|
|
|
|
mutex_unlock(mutex);
|
|
|
|
}
|
|
|
|
|
|
|
|
void _xtimer_sleep(uint32_t offset, uint32_t long_offset)
|
|
|
|
{
|
|
|
|
if (inISR()) {
|
|
|
|
assert(!long_offset);
|
|
|
|
xtimer_spin(offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
xtimer_t timer;
|
|
|
|
mutex_t mutex = MUTEX_INIT;
|
|
|
|
|
|
|
|
timer.callback = _callback_unlock_mutex;
|
|
|
|
timer.arg = (void*) &mutex;
|
2015-09-20 02:00:55 +02:00
|
|
|
timer.target = timer.long_target = 0;
|
2015-08-14 15:56:47 +02:00
|
|
|
|
|
|
|
mutex_lock(&mutex);
|
|
|
|
_xtimer_set64(&timer, offset, long_offset);
|
|
|
|
mutex_lock(&mutex);
|
|
|
|
}
|
|
|
|
|
|
|
|
void xtimer_usleep_until(uint32_t *last_wakeup, uint32_t interval) {
|
|
|
|
xtimer_t timer;
|
|
|
|
mutex_t mutex = MUTEX_INIT;
|
|
|
|
|
|
|
|
timer.callback = _callback_unlock_mutex;
|
|
|
|
timer.arg = (void*) &mutex;
|
|
|
|
|
|
|
|
uint32_t target = *last_wakeup + interval;
|
|
|
|
|
|
|
|
uint32_t now = xtimer_now();
|
|
|
|
/* make sure we're not setting a value in the past */
|
|
|
|
if (now < *last_wakeup) {
|
|
|
|
/* base timer overflowed */
|
|
|
|
if (!((target < *last_wakeup) && (target > now))) {
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (! ((target < *last_wakeup) || (target > now))) {
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2015-09-15 00:11:28 +02:00
|
|
|
/* For large offsets, set an absolute target time, as
|
|
|
|
* it is more exact.
|
|
|
|
*
|
|
|
|
* As that might cause an underflow, for small offsets,
|
|
|
|
* use a relative offset, as that can never underflow.
|
|
|
|
*
|
|
|
|
* For very small offsets, spin.
|
|
|
|
*/
|
2015-08-14 15:56:47 +02:00
|
|
|
uint32_t offset = target - now;
|
|
|
|
|
2015-09-15 00:11:28 +02:00
|
|
|
if (offset > (XTIMER_BACKOFF * 2)) {
|
2015-08-14 15:56:47 +02:00
|
|
|
mutex_lock(&mutex);
|
2015-09-15 00:11:28 +02:00
|
|
|
if (offset >> 9) { /* >= 512 */
|
|
|
|
offset = target;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
offset += xtimer_now();
|
|
|
|
}
|
|
|
|
_xtimer_set_absolute(&timer, offset);
|
2015-08-14 15:56:47 +02:00
|
|
|
mutex_lock(&mutex);
|
|
|
|
}
|
|
|
|
else {
|
2015-09-14 17:06:46 +02:00
|
|
|
xtimer_spin(offset);
|
2015-08-14 15:56:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
*last_wakeup = target;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _callback_msg(void* arg)
|
|
|
|
{
|
|
|
|
msg_t *msg = (msg_t*)arg;
|
|
|
|
msg_send_int(msg, msg->sender_pid);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void _setup_msg(xtimer_t *timer, msg_t *msg, kernel_pid_t target_pid)
|
|
|
|
{
|
|
|
|
timer->callback = _callback_msg;
|
|
|
|
timer->arg = (void*) msg;
|
|
|
|
|
|
|
|
/* use sender_pid field to get target_pid into callback function */
|
|
|
|
msg->sender_pid = target_pid;
|
|
|
|
}
|
|
|
|
|
|
|
|
void xtimer_set_msg(xtimer_t *timer, uint32_t offset, msg_t *msg, kernel_pid_t target_pid)
|
|
|
|
{
|
|
|
|
_setup_msg(timer, msg, target_pid);
|
|
|
|
xtimer_set(timer, offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
void xtimer_set_msg64(xtimer_t *timer, uint64_t offset, msg_t *msg, kernel_pid_t target_pid)
|
|
|
|
{
|
|
|
|
_setup_msg(timer, msg, target_pid);
|
|
|
|
_xtimer_set64(timer, offset, offset >> 32);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _callback_wakeup(void* arg)
|
|
|
|
{
|
|
|
|
thread_wakeup((kernel_pid_t)((intptr_t)arg));
|
|
|
|
}
|
|
|
|
|
|
|
|
void xtimer_set_wakeup(xtimer_t *timer, uint32_t offset, kernel_pid_t pid)
|
|
|
|
{
|
|
|
|
timer->callback = _callback_wakeup;
|
|
|
|
timer->arg = (void*) ((intptr_t)pid);
|
|
|
|
|
|
|
|
xtimer_set(timer, offset);
|
|
|
|
}
|
|
|
|
|
2015-09-03 20:00:35 +02:00
|
|
|
void xtimer_set_wakeup64(xtimer_t *timer, uint64_t offset, kernel_pid_t pid)
|
|
|
|
{
|
|
|
|
timer->callback = _callback_wakeup;
|
|
|
|
timer->arg = (void*) ((intptr_t)pid);
|
|
|
|
|
|
|
|
_xtimer_set64(timer, offset, offset >> 32);
|
|
|
|
}
|
|
|
|
|
2015-08-14 15:56:47 +02:00
|
|
|
/**
|
|
|
|
* see http://www.hackersdelight.org/magic.htm.
|
|
|
|
* This is to avoid using long integer division functions
|
|
|
|
* the compiler otherwise links in.
|
|
|
|
*/
|
2015-09-27 10:33:39 +02:00
|
|
|
static inline uint64_t _us_to_sec(uint64_t us)
|
2015-08-14 15:56:47 +02:00
|
|
|
{
|
2015-09-27 10:33:39 +02:00
|
|
|
return (unsigned long long)(us * 0x431bde83) >> (0x12 + 32);
|
2015-08-14 15:56:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void xtimer_now_timex(timex_t *out)
|
|
|
|
{
|
|
|
|
uint64_t now = xtimer_now64();
|
|
|
|
|
2015-09-27 10:33:39 +02:00
|
|
|
out->seconds = _us_to_sec(now);
|
2015-08-14 15:56:47 +02:00
|
|
|
out->microseconds = now - (out->seconds * SEC_IN_USEC);
|
|
|
|
}
|
|
|
|
|
|
|
|
int xtimer_msg_receive_timeout64(msg_t *m, uint64_t timeout) {
|
|
|
|
msg_t tmsg;
|
|
|
|
tmsg.type = MSG_XTIMER;
|
|
|
|
tmsg.content.ptr = (char *) &tmsg;
|
|
|
|
|
|
|
|
xtimer_t t;
|
2015-09-20 02:00:55 +02:00
|
|
|
t.target = t.long_target = 0;
|
2015-08-14 15:56:47 +02:00
|
|
|
xtimer_set_msg64(&t, timeout, &tmsg, sched_active_pid);
|
|
|
|
|
|
|
|
msg_receive(m);
|
|
|
|
if (m->type == MSG_XTIMER && m->content.ptr == (char *) &tmsg) {
|
|
|
|
/* we hit the timeout */
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
xtimer_remove(&t);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|