2016-07-05 11:11:48 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2015 Kaspar Schleiser <kaspar@schleiser.de>
|
2018-04-03 23:18:11 +02:00
|
|
|
* 2016 Eistec AB
|
|
|
|
* 2018 Josua Arndt
|
2016-07-05 11:11:48 +02:00
|
|
|
*
|
|
|
|
* 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 sys_xtimer
|
|
|
|
|
|
|
|
* @{
|
|
|
|
* @file
|
|
|
|
* @brief xtimer implementation
|
2018-04-03 23:18:11 +02:00
|
|
|
*
|
2016-07-05 11:11:48 +02:00
|
|
|
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
* @author Joakim Nohlgård <joakim.nohlgard@eistec.se>
|
2018-04-03 23:18:11 +02:00
|
|
|
* @author Josua Arndt <jarndt@ias.rwth-aachen.de>
|
|
|
|
*
|
2016-07-05 11:11:48 +02:00
|
|
|
*/
|
2017-01-18 13:00:05 +01:00
|
|
|
#ifndef XTIMER_IMPLEMENTATION_H
|
|
|
|
#define XTIMER_IMPLEMENTATION_H
|
2016-07-05 11:11:48 +02:00
|
|
|
|
|
|
|
#ifndef XTIMER_H
|
|
|
|
#error "Do not include this file directly! Use xtimer.h instead"
|
|
|
|
#endif
|
|
|
|
|
2020-02-20 15:33:24 +01:00
|
|
|
#ifdef MODULE_XTIMER_ON_ZTIMER
|
|
|
|
#include "ztimer.h"
|
|
|
|
#else
|
2016-07-05 11:11:48 +02:00
|
|
|
#include "periph/timer.h"
|
2020-02-20 15:33:24 +01:00
|
|
|
#endif
|
|
|
|
|
2020-01-09 23:34:43 +01:00
|
|
|
#include "irq.h"
|
2016-07-05 11:11:48 +02:00
|
|
|
|
2020-02-20 15:33:24 +01:00
|
|
|
|
2016-07-05 11:11:48 +02:00
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
2020-01-09 23:34:43 +01:00
|
|
|
extern volatile uint64_t _xtimer_current_time;
|
2016-07-05 11:11:48 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief IPC message type for xtimer msg callback
|
|
|
|
*/
|
|
|
|
#define MSG_XTIMER 12345
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief returns the (masked) low-level timer counter value.
|
|
|
|
*/
|
2016-07-05 15:08:47 +02:00
|
|
|
static inline uint32_t _xtimer_lltimer_now(void)
|
2016-07-05 11:11:48 +02:00
|
|
|
{
|
2020-02-20 15:33:24 +01:00
|
|
|
#ifndef MODULE_XTIMER_ON_ZTIMER
|
2016-07-05 16:45:41 +02:00
|
|
|
return timer_read(XTIMER_DEV);
|
2020-02-20 15:33:24 +01:00
|
|
|
#else
|
|
|
|
return ztimer_now(ZTIMER_USEC);
|
|
|
|
#endif
|
|
|
|
|
2016-07-05 11:11:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief drop bits of a value that don't fit into the low-level timer.
|
|
|
|
*/
|
2016-07-05 15:08:47 +02:00
|
|
|
static inline uint32_t _xtimer_lltimer_mask(uint32_t val)
|
2016-07-05 11:11:48 +02:00
|
|
|
{
|
2017-03-07 16:43:17 +01:00
|
|
|
/* cppcheck-suppress shiftTooManyBits
|
|
|
|
* (reason: cppcheck bug. `XTIMER_MASK` is zero when `XTIMER_WIDTH` is 32) */
|
2016-07-05 16:00:48 +02:00
|
|
|
return val & ~XTIMER_MASK;
|
2016-07-05 11:11:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @{
|
|
|
|
* @brief xtimer internal stuff
|
|
|
|
* @internal
|
|
|
|
*/
|
2018-05-28 15:39:16 +02:00
|
|
|
|
2020-01-09 23:34:43 +01:00
|
|
|
uint32_t _xtimer_now(void);
|
2018-05-28 15:39:16 +02:00
|
|
|
|
2018-01-08 11:34:42 +01:00
|
|
|
void _xtimer_set64(xtimer_t *timer, uint32_t offset, uint32_t long_offset);
|
2016-07-05 16:00:48 +02:00
|
|
|
void _xtimer_periodic_wakeup(uint32_t *last_wakeup, uint32_t period);
|
|
|
|
void _xtimer_set_wakeup(xtimer_t *timer, uint32_t offset, kernel_pid_t pid);
|
|
|
|
void _xtimer_set_wakeup64(xtimer_t *timer, uint64_t offset, kernel_pid_t pid);
|
2019-03-21 15:05:21 +01:00
|
|
|
#ifdef MODULE_CORE_MSG
|
2016-07-05 16:00:48 +02:00
|
|
|
int _xtimer_msg_receive_timeout(msg_t *msg, uint32_t ticks);
|
|
|
|
int _xtimer_msg_receive_timeout64(msg_t *msg, uint64_t ticks);
|
2019-03-21 15:05:21 +01:00
|
|
|
void _xtimer_set_msg(xtimer_t *timer, uint32_t offset, msg_t *msg, kernel_pid_t target_pid);
|
|
|
|
void _xtimer_set_msg64(xtimer_t *timer, uint64_t offset, msg_t *msg, kernel_pid_t target_pid);
|
|
|
|
#endif /* MODULE_CORE_MSG */
|
2016-07-05 16:00:48 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Sleep for the given number of ticks
|
|
|
|
*/
|
|
|
|
void _xtimer_tsleep(uint32_t offset, uint32_t long_offset);
|
2016-07-05 11:11:48 +02:00
|
|
|
/** @} */
|
|
|
|
|
|
|
|
#ifndef XTIMER_MIN_SPIN
|
|
|
|
/**
|
|
|
|
* @brief Minimal value xtimer_spin() can spin
|
|
|
|
*/
|
2016-07-05 16:00:48 +02:00
|
|
|
#define XTIMER_MIN_SPIN _xtimer_usec_from_ticks(1)
|
2016-07-05 11:11:48 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef DOXYGEN
|
|
|
|
/* Doxygen warns that these are undocumented, but the documentation can be found in xtimer.h */
|
|
|
|
|
2020-01-09 23:34:43 +01:00
|
|
|
static inline uint64_t _xtimer_now64(void)
|
2016-07-05 11:11:48 +02:00
|
|
|
{
|
2020-01-09 23:34:43 +01:00
|
|
|
uint32_t now, elapsed;
|
2016-07-05 11:11:48 +02:00
|
|
|
|
2020-01-09 23:34:43 +01:00
|
|
|
/* time sensitive since _xtimer_current_time is updated here */
|
2020-01-22 12:46:17 +01:00
|
|
|
unsigned state = irq_disable();
|
2020-01-09 23:34:43 +01:00
|
|
|
now = _xtimer_lltimer_now();
|
|
|
|
#if XTIMER_MASK
|
|
|
|
elapsed = _xtimer_lltimer_mask(now - _xtimer_lltimer_mask((uint32_t)_xtimer_current_time));
|
|
|
|
_xtimer_current_time += (uint64_t)elapsed;
|
2016-07-05 11:11:48 +02:00
|
|
|
#else
|
2020-01-09 23:34:43 +01:00
|
|
|
elapsed = now - ((uint32_t)_xtimer_current_time & 0xFFFFFFFF);
|
|
|
|
_xtimer_current_time += (uint64_t)elapsed;
|
2016-07-05 11:11:48 +02:00
|
|
|
#endif
|
2020-01-09 23:34:43 +01:00
|
|
|
irq_restore(state);
|
|
|
|
|
|
|
|
return _xtimer_current_time;
|
2016-07-05 11:11:48 +02:00
|
|
|
}
|
|
|
|
|
2016-07-05 16:00:48 +02:00
|
|
|
static inline xtimer_ticks32_t xtimer_now(void)
|
|
|
|
{
|
2016-10-06 14:37:05 +02:00
|
|
|
xtimer_ticks32_t ret;
|
|
|
|
ret.ticks32 = _xtimer_now();
|
2016-07-05 16:00:48 +02:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline xtimer_ticks64_t xtimer_now64(void)
|
|
|
|
{
|
2016-10-06 14:37:05 +02:00
|
|
|
xtimer_ticks64_t ret;
|
|
|
|
ret.ticks64 = _xtimer_now64();
|
2016-07-05 16:00:48 +02:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2016-10-06 08:49:08 +02:00
|
|
|
static inline uint32_t xtimer_now_usec(void)
|
|
|
|
{
|
|
|
|
return xtimer_usec_from_ticks(xtimer_now());
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline uint64_t xtimer_now_usec64(void)
|
|
|
|
{
|
|
|
|
return xtimer_usec_from_ticks64(xtimer_now64());
|
|
|
|
}
|
|
|
|
|
2016-07-05 16:00:48 +02:00
|
|
|
static inline void _xtimer_spin(uint32_t offset) {
|
2016-07-05 15:08:47 +02:00
|
|
|
uint32_t start = _xtimer_lltimer_now();
|
2016-07-05 11:11:48 +02:00
|
|
|
#if XTIMER_MASK
|
2016-07-05 15:08:47 +02:00
|
|
|
offset = _xtimer_lltimer_mask(offset);
|
|
|
|
while (_xtimer_lltimer_mask(_xtimer_lltimer_now() - start) < offset);
|
2016-07-05 11:11:48 +02:00
|
|
|
#else
|
2016-07-05 15:08:47 +02:00
|
|
|
while ((_xtimer_lltimer_now() - start) < offset);
|
2016-07-05 11:11:48 +02:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2016-07-05 16:00:48 +02:00
|
|
|
static inline void _xtimer_tsleep32(uint32_t ticks)
|
|
|
|
{
|
|
|
|
_xtimer_tsleep(ticks, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void _xtimer_tsleep64(uint64_t ticks)
|
|
|
|
{
|
|
|
|
_xtimer_tsleep((uint32_t)ticks, (uint32_t)(ticks >> 32));
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void xtimer_spin(xtimer_ticks32_t ticks) {
|
|
|
|
_xtimer_spin(ticks.ticks32);
|
|
|
|
}
|
|
|
|
|
2020-09-24 12:07:50 +02:00
|
|
|
static inline void xtimer_msleep(uint32_t milliseconds)
|
|
|
|
{
|
|
|
|
_xtimer_tsleep64(_xtimer_ticks_from_usec64(milliseconds * US_PER_MS));
|
|
|
|
}
|
|
|
|
|
2016-07-05 11:11:48 +02:00
|
|
|
static inline void xtimer_usleep(uint32_t microseconds)
|
|
|
|
{
|
2016-07-05 16:00:48 +02:00
|
|
|
_xtimer_tsleep32(_xtimer_ticks_from_usec(microseconds));
|
2016-07-05 11:11:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline void xtimer_usleep64(uint64_t microseconds)
|
|
|
|
{
|
2016-07-05 16:00:48 +02:00
|
|
|
_xtimer_tsleep64(_xtimer_ticks_from_usec64(microseconds));
|
2016-07-05 11:11:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline void xtimer_sleep(uint32_t seconds)
|
|
|
|
{
|
2017-01-17 16:44:05 +01:00
|
|
|
_xtimer_tsleep64(_xtimer_ticks_from_usec64((uint64_t)seconds * US_PER_SEC));
|
2016-07-05 11:11:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline void xtimer_nanosleep(uint32_t nanoseconds)
|
|
|
|
{
|
2017-01-17 16:44:05 +01:00
|
|
|
_xtimer_tsleep32(_xtimer_ticks_from_usec(nanoseconds / NS_PER_US));
|
2016-07-05 16:00:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline void xtimer_tsleep32(xtimer_ticks32_t ticks)
|
|
|
|
{
|
|
|
|
_xtimer_tsleep32(ticks.ticks32);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void xtimer_tsleep64(xtimer_ticks64_t ticks)
|
|
|
|
{
|
|
|
|
_xtimer_tsleep64(ticks.ticks64);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void xtimer_periodic_wakeup(xtimer_ticks32_t *last_wakeup, uint32_t period)
|
|
|
|
{
|
|
|
|
_xtimer_periodic_wakeup(&last_wakeup->ticks32, _xtimer_ticks_from_usec(period));
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void xtimer_set_wakeup(xtimer_t *timer, uint32_t offset, kernel_pid_t pid)
|
|
|
|
{
|
|
|
|
_xtimer_set_wakeup(timer, _xtimer_ticks_from_usec(offset), pid);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void xtimer_set_wakeup64(xtimer_t *timer, uint64_t offset, kernel_pid_t pid)
|
|
|
|
{
|
|
|
|
_xtimer_set_wakeup64(timer, _xtimer_ticks_from_usec64(offset), pid);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void xtimer_set(xtimer_t *timer, uint32_t offset)
|
|
|
|
{
|
2020-01-09 23:34:43 +01:00
|
|
|
_xtimer_set64(timer, _xtimer_ticks_from_usec(offset), 0);
|
2016-07-05 16:00:48 +02:00
|
|
|
}
|
|
|
|
|
2018-04-03 23:18:11 +02:00
|
|
|
static inline void xtimer_set64(xtimer_t *timer, uint64_t period_us)
|
|
|
|
{
|
|
|
|
uint64_t ticks = _xtimer_ticks_from_usec64(period_us);
|
|
|
|
_xtimer_set64(timer, ticks, ticks >> 32);
|
|
|
|
}
|
|
|
|
|
2019-03-21 15:05:21 +01:00
|
|
|
#ifdef MODULE_CORE_MSG
|
2016-07-05 16:00:48 +02:00
|
|
|
static inline int xtimer_msg_receive_timeout(msg_t *msg, uint32_t timeout)
|
|
|
|
{
|
|
|
|
return _xtimer_msg_receive_timeout(msg, _xtimer_ticks_from_usec(timeout));
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int xtimer_msg_receive_timeout64(msg_t *msg, uint64_t timeout)
|
|
|
|
{
|
|
|
|
return _xtimer_msg_receive_timeout64(msg, _xtimer_ticks_from_usec64(timeout));
|
|
|
|
}
|
|
|
|
|
2019-03-21 15:05:21 +01:00
|
|
|
static inline void xtimer_set_msg(xtimer_t *timer, uint32_t offset, msg_t *msg, kernel_pid_t target_pid)
|
|
|
|
{
|
|
|
|
_xtimer_set_msg(timer, _xtimer_ticks_from_usec(offset), msg, target_pid);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void xtimer_set_msg64(xtimer_t *timer, uint64_t offset, msg_t *msg, kernel_pid_t target_pid)
|
|
|
|
{
|
|
|
|
_xtimer_set_msg64(timer, _xtimer_ticks_from_usec64(offset), msg, target_pid);
|
|
|
|
}
|
|
|
|
#endif /* MODULE_CORE_MSG */
|
|
|
|
|
2016-07-05 16:00:48 +02:00
|
|
|
static inline xtimer_ticks32_t xtimer_ticks_from_usec(uint32_t usec)
|
|
|
|
{
|
2016-10-06 14:37:05 +02:00
|
|
|
xtimer_ticks32_t ticks;
|
|
|
|
ticks.ticks32 = _xtimer_ticks_from_usec(usec);
|
2016-07-05 16:00:48 +02:00
|
|
|
return ticks;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline xtimer_ticks64_t xtimer_ticks_from_usec64(uint64_t usec)
|
|
|
|
{
|
2016-10-06 14:37:05 +02:00
|
|
|
xtimer_ticks64_t ticks;
|
|
|
|
ticks.ticks64 = _xtimer_ticks_from_usec64(usec);
|
2016-07-05 16:00:48 +02:00
|
|
|
return ticks;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline uint32_t xtimer_usec_from_ticks(xtimer_ticks32_t ticks)
|
|
|
|
{
|
|
|
|
return _xtimer_usec_from_ticks(ticks.ticks32);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline uint64_t xtimer_usec_from_ticks64(xtimer_ticks64_t ticks)
|
|
|
|
{
|
|
|
|
return _xtimer_usec_from_ticks64(ticks.ticks64);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline xtimer_ticks32_t xtimer_ticks(uint32_t ticks)
|
|
|
|
{
|
2016-10-06 14:37:05 +02:00
|
|
|
xtimer_ticks32_t ret;
|
|
|
|
ret.ticks32 = ticks;
|
2016-07-05 16:00:48 +02:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline xtimer_ticks64_t xtimer_ticks64(uint64_t ticks)
|
|
|
|
{
|
2016-10-06 14:37:05 +02:00
|
|
|
xtimer_ticks64_t ret;
|
|
|
|
ret.ticks64 = ticks;
|
2016-07-05 16:00:48 +02:00
|
|
|
return ret;
|
2016-07-05 11:11:48 +02:00
|
|
|
}
|
|
|
|
|
2016-10-06 08:54:02 +02:00
|
|
|
static inline xtimer_ticks32_t xtimer_diff(xtimer_ticks32_t a, xtimer_ticks32_t b)
|
|
|
|
{
|
2016-10-06 14:37:05 +02:00
|
|
|
xtimer_ticks32_t ret;
|
|
|
|
ret.ticks32 = a.ticks32 - b.ticks32;
|
2016-10-06 08:54:02 +02:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline xtimer_ticks64_t xtimer_diff64(xtimer_ticks64_t a, xtimer_ticks64_t b)
|
|
|
|
{
|
2016-10-06 14:37:05 +02:00
|
|
|
xtimer_ticks64_t ret;
|
|
|
|
ret.ticks64 = a.ticks64 - b.ticks64;
|
2016-10-06 08:54:02 +02:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline xtimer_ticks32_t xtimer_diff32_64(xtimer_ticks64_t a, xtimer_ticks64_t b)
|
|
|
|
{
|
2016-10-06 14:37:05 +02:00
|
|
|
uint64_t diff = a.ticks64 - b.ticks64;
|
|
|
|
xtimer_ticks32_t ret;
|
|
|
|
ret.ticks32 = diff;
|
2016-10-06 08:54:02 +02:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline bool xtimer_less(xtimer_ticks32_t a, xtimer_ticks32_t b)
|
|
|
|
{
|
|
|
|
return (a.ticks32 < b.ticks32);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline bool xtimer_less64(xtimer_ticks64_t a, xtimer_ticks64_t b)
|
|
|
|
{
|
|
|
|
return (a.ticks64 < b.ticks64);
|
|
|
|
}
|
|
|
|
|
2016-07-05 11:11:48 +02:00
|
|
|
#endif /* !defined(DOXYGEN) */
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2017-01-18 15:07:35 +01:00
|
|
|
#endif /* XTIMER_IMPLEMENTATION_H */
|