2015-08-14 15:56:47 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2015 Kaspar Schleiser <kaspar@schleiser.de>
|
2016-07-04 21:14:13 +02:00
|
|
|
* Copyright (C) 2016 Eistec AB
|
2015-08-14 15:56:47 +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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @defgroup sys_xtimer Timers
|
|
|
|
* @ingroup sys
|
|
|
|
* @brief Provides a high level timer module to register
|
|
|
|
* timers, get current system time, and let a thread sleep for
|
|
|
|
* a certain amount of time.
|
|
|
|
*
|
2016-07-04 21:14:13 +02:00
|
|
|
* The implementation takes one low-level timer and multiplexes it.
|
2015-08-14 15:56:47 +02:00
|
|
|
*
|
|
|
|
* Insertion and removal of timers has O(n) complexity with (n) being the
|
|
|
|
* number of active timers. The reason for this is that multiplexing is
|
|
|
|
* realized by next-first singly linked lists.
|
|
|
|
*
|
|
|
|
* @{
|
|
|
|
* @file
|
|
|
|
* @brief xtimer interface definitions
|
|
|
|
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
2016-07-04 21:14:13 +02:00
|
|
|
* @author Joakim Nohlgård <joakim.nohlgard@eistec.se>
|
2015-08-14 15:56:47 +02:00
|
|
|
*/
|
|
|
|
#ifndef XTIMER_H
|
|
|
|
#define XTIMER_H
|
|
|
|
|
2019-03-21 15:05:21 +01:00
|
|
|
#include <stdbool.h>
|
2015-08-14 15:56:47 +02:00
|
|
|
#include <stdint.h>
|
|
|
|
#include "timex.h"
|
2019-03-21 15:05:21 +01:00
|
|
|
#ifdef MODULE_CORE_MSG
|
2016-07-05 11:11:48 +02:00
|
|
|
#include "msg.h"
|
2019-03-21 15:05:21 +01:00
|
|
|
#endif /* MODULE_CORE_MSG */
|
2016-11-22 18:21:36 +01:00
|
|
|
#include "mutex.h"
|
2019-03-21 15:05:21 +01:00
|
|
|
#include "kernel_types.h"
|
2019-07-12 16:23:20 +02:00
|
|
|
#include "rmutex.h"
|
2015-08-14 15:56:47 +02:00
|
|
|
|
2019-12-10 11:04:29 +01:00
|
|
|
#ifdef MODULE_ZTIMER_XTIMER_COMPAT
|
|
|
|
#include "ztimer/xtimer_compat.h"
|
|
|
|
#else
|
|
|
|
|
2015-08-14 15:56:47 +02:00
|
|
|
#include "board.h"
|
2020-04-30 11:46:11 +02:00
|
|
|
#ifndef MODULE_XTIMER_ON_ZTIMER
|
2015-08-14 15:56:47 +02:00
|
|
|
#include "periph_conf.h"
|
2020-02-20 15:33:24 +01:00
|
|
|
#endif
|
2015-08-14 15:56:47 +02:00
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
2016-07-05 16:00:48 +02:00
|
|
|
/**
|
|
|
|
* @brief xtimer timestamp (64 bit)
|
|
|
|
*
|
|
|
|
* @note This is a struct in order to make the xtimer API type strict
|
|
|
|
*/
|
|
|
|
typedef struct {
|
2017-03-02 18:13:22 +01:00
|
|
|
uint64_t ticks64; /**< Tick count */
|
2016-07-05 16:00:48 +02:00
|
|
|
} xtimer_ticks64_t;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief xtimer timestamp (32 bit)
|
|
|
|
*
|
|
|
|
* @note This is a struct in order to make the xtimer API type strict
|
|
|
|
*/
|
|
|
|
typedef struct {
|
2017-03-02 18:13:22 +01:00
|
|
|
uint32_t ticks32; /**< Tick count */
|
2016-07-05 16:00:48 +02:00
|
|
|
} xtimer_ticks32_t;
|
|
|
|
|
2015-08-14 15:56:47 +02:00
|
|
|
/**
|
|
|
|
* @brief xtimer callback type
|
|
|
|
*/
|
2016-07-05 23:13:03 +02:00
|
|
|
typedef void (*xtimer_callback_t)(void*);
|
2015-08-14 15:56:47 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief xtimer timer structure
|
|
|
|
*/
|
|
|
|
typedef struct xtimer {
|
2017-03-02 18:13:22 +01:00
|
|
|
struct xtimer *next; /**< reference to next timer in timer lists */
|
2020-01-09 23:34:43 +01:00
|
|
|
uint32_t offset; /**< lower 32bit offset time */
|
|
|
|
uint32_t long_offset; /**< upper 32bit offset time */
|
|
|
|
uint32_t start_time; /**< lower 32bit absolute start time */
|
|
|
|
uint32_t long_start_time; /**< upper 32bit absolute start time */
|
2016-07-05 23:13:03 +02:00
|
|
|
xtimer_callback_t callback; /**< callback function to call when timer
|
2015-08-14 15:56:47 +02:00
|
|
|
expires */
|
2017-03-02 18:13:22 +01:00
|
|
|
void *arg; /**< argument to pass to callback function */
|
2015-08-14 15:56:47 +02:00
|
|
|
} xtimer_t;
|
|
|
|
|
|
|
|
/**
|
2016-07-05 16:00:48 +02:00
|
|
|
* @brief get the current system time as 32bit time stamp value
|
2015-08-14 15:56:47 +02:00
|
|
|
*
|
2016-07-05 16:00:48 +02:00
|
|
|
* @note Overflows 2**32 ticks, thus returns xtimer_now64() % 32,
|
|
|
|
* but is cheaper.
|
2015-08-14 15:56:47 +02:00
|
|
|
*
|
2016-07-05 16:00:48 +02:00
|
|
|
* @return current time as 32bit time stamp
|
2015-08-14 15:56:47 +02:00
|
|
|
*/
|
2016-07-05 16:00:48 +02:00
|
|
|
static inline xtimer_ticks32_t xtimer_now(void);
|
2015-08-14 15:56:47 +02:00
|
|
|
|
|
|
|
/**
|
2016-07-05 16:00:48 +02:00
|
|
|
* @brief get the current system time as 64bit time stamp
|
2015-08-14 15:56:47 +02:00
|
|
|
*
|
2016-07-05 16:00:48 +02:00
|
|
|
* @return current time as 64bit time stamp
|
2015-08-14 15:56:47 +02:00
|
|
|
*/
|
2016-07-05 16:00:48 +02:00
|
|
|
static inline xtimer_ticks64_t xtimer_now64(void);
|
2015-08-14 15:56:47 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief get the current system time into a timex_t
|
|
|
|
*
|
|
|
|
* @param[out] out pointer to timex_t the time will be written to
|
|
|
|
*/
|
|
|
|
void xtimer_now_timex(timex_t *out);
|
|
|
|
|
2016-10-06 08:49:08 +02:00
|
|
|
/**
|
|
|
|
* @brief get the current system time in microseconds since start
|
|
|
|
*
|
|
|
|
* This is a convenience function for @c xtimer_usec_from_ticks(xtimer_now())
|
|
|
|
*/
|
|
|
|
static inline uint32_t xtimer_now_usec(void);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief get the current system time in microseconds since start
|
|
|
|
*
|
|
|
|
* This is a convenience function for @c xtimer_usec_from_ticks64(xtimer_now64())
|
|
|
|
*/
|
|
|
|
static inline uint64_t xtimer_now_usec64(void);
|
|
|
|
|
2015-08-14 15:56:47 +02:00
|
|
|
/**
|
|
|
|
* @brief xtimer initialization function
|
|
|
|
*
|
|
|
|
* This sets up xtimer. Has to be called once at system boot.
|
|
|
|
* If @ref auto_init is enabled, it will call this for you.
|
|
|
|
*/
|
|
|
|
void xtimer_init(void);
|
|
|
|
|
|
|
|
/**
|
2016-04-26 13:55:54 +02:00
|
|
|
* @brief Pause the execution of a thread for some seconds
|
2015-08-14 15:56:47 +02:00
|
|
|
*
|
|
|
|
* When called from an ISR, this function will spin and thus block the MCU in
|
|
|
|
* interrupt context for the specified amount in *seconds*, so don't *ever* use
|
|
|
|
* it there.
|
|
|
|
*
|
|
|
|
* @param[in] seconds the amount of seconds the thread should sleep
|
|
|
|
*/
|
2016-07-05 16:00:48 +02:00
|
|
|
static inline void xtimer_sleep(uint32_t seconds);
|
2015-08-14 15:56:47 +02:00
|
|
|
|
2020-09-24 12:07:50 +02:00
|
|
|
/**
|
|
|
|
* @brief Pause the execution of a thread for some milliseconds
|
|
|
|
*
|
|
|
|
* @param[in] milliseconds the amount of milliseconds the thread should sleep
|
|
|
|
*/
|
|
|
|
static inline void xtimer_msleep(uint32_t milliseconds);
|
|
|
|
|
2015-08-14 15:56:47 +02:00
|
|
|
/**
|
2016-04-26 13:55:54 +02:00
|
|
|
* @brief Pause the execution of a thread for some microseconds
|
2015-08-14 15:56:47 +02:00
|
|
|
*
|
|
|
|
* When called from an ISR, this function will spin and thus block the MCU for
|
|
|
|
* the specified amount in microseconds, so only use it there for *very* short
|
2019-12-04 02:49:18 +01:00
|
|
|
* periods, e.g., less than XTIMER_BACKOFF converted to µs.
|
2015-08-14 15:56:47 +02:00
|
|
|
*
|
|
|
|
* @param[in] microseconds the amount of microseconds the thread should sleep
|
|
|
|
*/
|
2016-07-05 16:00:48 +02:00
|
|
|
static inline void xtimer_usleep(uint32_t microseconds);
|
2015-08-14 15:56:47 +02:00
|
|
|
|
2020-02-13 15:07:06 +01:00
|
|
|
/**
|
|
|
|
* @brief Pause the execution of a thread for some microseconds
|
|
|
|
*
|
|
|
|
* See xtimer_usleep() for more information.
|
|
|
|
*
|
|
|
|
* @param[in] microseconds the amount of microseconds the thread should sleep
|
|
|
|
*/
|
|
|
|
static inline void xtimer_usleep64(uint64_t microseconds);
|
|
|
|
|
2015-08-14 15:56:47 +02:00
|
|
|
/**
|
|
|
|
* @brief Stop execution of a thread for some time
|
|
|
|
*
|
|
|
|
* Don't expect nanosecond accuracy. As of now, this function just calls
|
|
|
|
* xtimer_usleep(nanoseconds/1000).
|
|
|
|
*
|
|
|
|
* When called from an ISR, this function will spin-block, so only use it there
|
|
|
|
* for *very* short periods.
|
|
|
|
*
|
|
|
|
* @param[in] nanoseconds the amount of nanoseconds the thread should sleep
|
|
|
|
*/
|
2016-07-05 16:00:48 +02:00
|
|
|
static inline void xtimer_nanosleep(uint32_t nanoseconds);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Stop execution of a thread for some time, 32bit version
|
|
|
|
*
|
|
|
|
* When called from an ISR, this function will spin and thus block the MCU for
|
|
|
|
* the specified amount, so only use it there for *very* short periods,
|
|
|
|
* e.g. less than XTIMER_BACKOFF.
|
|
|
|
*
|
|
|
|
* @param[in] ticks number of ticks the thread should sleep
|
|
|
|
*/
|
|
|
|
static inline void xtimer_tsleep32(xtimer_ticks32_t ticks);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Stop execution of a thread for some time, 64bit version
|
|
|
|
*
|
|
|
|
* When called from an ISR, this function will spin and thus block the MCU for
|
|
|
|
* the specified amount, so only use it there for *very* short periods,
|
|
|
|
* e.g. less than XTIMER_BACKOFF.
|
|
|
|
*
|
|
|
|
* @param[in] ticks number of ticks the thread should sleep
|
|
|
|
*/
|
|
|
|
static inline void xtimer_tsleep64(xtimer_ticks64_t ticks);
|
2015-08-14 15:56:47 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Stop execution of a thread for some time, blocking
|
|
|
|
*
|
|
|
|
* This function will spin-block, so only use it *very* short periods.
|
|
|
|
*
|
2016-07-05 16:00:48 +02:00
|
|
|
* @param[in] ticks the number of xtimer ticks the thread should spin for
|
2015-08-14 15:56:47 +02:00
|
|
|
*/
|
2016-07-05 16:00:48 +02:00
|
|
|
static inline void xtimer_spin(xtimer_ticks32_t ticks);
|
2015-08-14 15:56:47 +02:00
|
|
|
|
2016-07-05 23:01:26 +02:00
|
|
|
/**
|
2015-08-14 15:56:47 +02:00
|
|
|
* @brief will cause the calling thread to be suspended until the absolute
|
2016-07-05 23:01:26 +02:00
|
|
|
* time (@p last_wakeup + @p period).
|
2015-08-14 15:56:47 +02:00
|
|
|
*
|
2015-09-07 14:23:01 +02:00
|
|
|
* When the function returns, @p last_wakeup is set to
|
2016-07-05 23:01:26 +02:00
|
|
|
* (@p last_wakeup + @p period).
|
2015-08-14 15:56:47 +02:00
|
|
|
*
|
|
|
|
* This function can be used to create periodic wakeups.
|
|
|
|
* @c last_wakeup should be set to xtimer_now() before first call of the
|
|
|
|
* function.
|
|
|
|
*
|
2016-07-05 23:01:26 +02:00
|
|
|
* If the result of (@p last_wakeup + @p period) would be in the past, the function
|
|
|
|
* sets @p last_wakeup to @p last_wakeup + @p period and returns immediately.
|
2015-08-14 15:56:47 +02:00
|
|
|
*
|
2016-07-05 23:01:26 +02:00
|
|
|
* @param[in] last_wakeup base time stamp for the wakeup
|
|
|
|
* @param[in] period time in microseconds that will be added to last_wakeup
|
2015-08-14 15:56:47 +02:00
|
|
|
*/
|
2016-07-05 16:00:48 +02:00
|
|
|
static inline void xtimer_periodic_wakeup(xtimer_ticks32_t *last_wakeup, uint32_t period);
|
2015-08-14 15:56:47 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Set a timer that wakes up a thread
|
|
|
|
*
|
|
|
|
* This function sets a timer that will wake up a thread when the timer has
|
|
|
|
* expired.
|
|
|
|
*
|
2016-03-31 19:08:53 +02:00
|
|
|
* @param[in] timer timer struct to work with.
|
2015-08-14 15:56:47 +02:00
|
|
|
* @param[in] offset microseconds from now
|
|
|
|
* @param[in] pid pid of the thread that will be woken up
|
|
|
|
*/
|
2016-07-05 16:00:48 +02:00
|
|
|
static inline void xtimer_set_wakeup(xtimer_t *timer, uint32_t offset, kernel_pid_t pid);
|
2015-08-14 15:56:47 +02:00
|
|
|
|
2015-09-03 20:00:35 +02:00
|
|
|
/**
|
|
|
|
* @brief Set a timer that wakes up a thread, 64bit version
|
|
|
|
*
|
|
|
|
* This function sets a timer that will wake up a thread when the timer has
|
|
|
|
* expired.
|
|
|
|
*
|
2016-03-31 19:08:53 +02:00
|
|
|
* @param[in] timer timer struct to work with.
|
2015-09-03 20:00:35 +02:00
|
|
|
* @param[in] offset microseconds from now
|
|
|
|
* @param[in] pid pid of the thread that will be woken up
|
|
|
|
*/
|
2016-07-05 16:00:48 +02:00
|
|
|
static inline void xtimer_set_wakeup64(xtimer_t *timer, uint64_t offset, kernel_pid_t pid);
|
2015-09-03 20:00:35 +02:00
|
|
|
|
2015-08-14 15:56:47 +02:00
|
|
|
/**
|
|
|
|
* @brief Set a timer to execute a callback at some time in the future
|
|
|
|
*
|
|
|
|
* Expects timer->callback to be set.
|
|
|
|
*
|
|
|
|
* The callback specified in the timer struct will be executed @p offset
|
2016-07-05 16:00:48 +02:00
|
|
|
* ticks in the future.
|
2015-08-14 15:56:47 +02:00
|
|
|
*
|
|
|
|
* @warning BEWARE! Callbacks from xtimer_set() are being executed in interrupt
|
2019-12-04 02:49:18 +01:00
|
|
|
* context (unless offset < XTIMER_BACKOFF converted to µs).
|
|
|
|
* DON'T USE THIS FUNCTION unless you know *exactly* what that means.
|
2015-08-14 15:56:47 +02:00
|
|
|
*
|
2016-03-31 19:08:53 +02:00
|
|
|
* @param[in] timer the timer structure to use.
|
2015-08-14 15:56:47 +02:00
|
|
|
* @param[in] offset time in microseconds from now specifying that timer's
|
|
|
|
* callback's execution time
|
|
|
|
*/
|
2016-07-05 16:00:48 +02:00
|
|
|
static inline void xtimer_set(xtimer_t *timer, uint32_t offset);
|
2015-08-14 15:56:47 +02:00
|
|
|
|
2018-04-03 23:18:11 +02:00
|
|
|
/**
|
|
|
|
* @brief Set a timer to execute a callback at some time in the future, 64bit
|
|
|
|
* version
|
|
|
|
*
|
|
|
|
* Expects timer->callback to be set.
|
|
|
|
*
|
|
|
|
* The callback specified in the timer struct will be executed @p offset_usec
|
|
|
|
* microseconds in the future.
|
|
|
|
*
|
|
|
|
* @warning BEWARE! Callbacks from xtimer_set() are being executed in interrupt
|
2019-12-04 02:49:18 +01:00
|
|
|
* context (unless offset < XTIMER_BACKOFF converted to µs).
|
|
|
|
* DON'T USE THIS FUNCTION unless you know *exactly* what that means.
|
2018-04-03 23:18:11 +02:00
|
|
|
*
|
|
|
|
* @param[in] timer the timer structure to use.
|
|
|
|
* @param[in] offset_us time in microseconds from now specifying that timer's
|
|
|
|
* callback's execution time
|
|
|
|
*/
|
|
|
|
static inline void xtimer_set64(xtimer_t *timer, uint64_t offset_us);
|
|
|
|
|
2015-08-14 15:56:47 +02:00
|
|
|
/**
|
|
|
|
* @brief remove a timer
|
|
|
|
*
|
|
|
|
* @note this function runs in O(n) with n being the number of active timers
|
|
|
|
*
|
|
|
|
* @param[in] timer ptr to timer structure that will be removed
|
|
|
|
*/
|
2016-02-25 19:56:15 +01:00
|
|
|
void xtimer_remove(xtimer_t *timer);
|
2015-08-14 15:56:47 +02:00
|
|
|
|
2016-07-05 16:00:48 +02:00
|
|
|
/**
|
|
|
|
* @brief Convert microseconds to xtimer ticks
|
|
|
|
*
|
|
|
|
* @param[in] usec microseconds
|
2015-08-14 15:56:47 +02:00
|
|
|
*
|
2016-07-05 16:00:48 +02:00
|
|
|
* @return xtimer time stamp
|
2015-08-14 15:56:47 +02:00
|
|
|
*/
|
2016-07-05 16:00:48 +02:00
|
|
|
static inline xtimer_ticks32_t xtimer_ticks_from_usec(uint32_t usec);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Convert microseconds to xtimer ticks, 64 bit version
|
|
|
|
*
|
|
|
|
* @param[in] usec microseconds
|
|
|
|
*
|
|
|
|
* @return xtimer time stamp
|
|
|
|
*/
|
|
|
|
static inline xtimer_ticks64_t xtimer_ticks_from_usec64(uint64_t usec);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Convert xtimer ticks to microseconds
|
|
|
|
*
|
|
|
|
* @param[in] ticks xtimer time stamp
|
|
|
|
*
|
|
|
|
* @return microseconds
|
|
|
|
*/
|
|
|
|
static inline uint32_t xtimer_usec_from_ticks(xtimer_ticks32_t ticks);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Convert xtimer ticks to microseconds, 64 bit version
|
|
|
|
*
|
|
|
|
* @param[in] ticks xtimer time stamp
|
|
|
|
*
|
|
|
|
* @return microseconds
|
|
|
|
*/
|
|
|
|
static inline uint64_t xtimer_usec_from_ticks64(xtimer_ticks64_t ticks);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Create an xtimer time stamp
|
|
|
|
*
|
|
|
|
* @param[in] ticks number of xtimer ticks
|
|
|
|
*
|
|
|
|
* @return xtimer time stamp
|
|
|
|
*/
|
|
|
|
static inline xtimer_ticks32_t xtimer_ticks(uint32_t ticks);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Create an xtimer time stamp, 64 bit version
|
|
|
|
*
|
|
|
|
* @param[in] ticks number of xtimer ticks
|
|
|
|
*
|
|
|
|
* @return xtimer time stamp
|
|
|
|
*/
|
|
|
|
static inline xtimer_ticks64_t xtimer_ticks64(uint64_t ticks);
|
2015-08-14 15:56:47 +02:00
|
|
|
|
2016-10-06 08:54:02 +02:00
|
|
|
/**
|
|
|
|
* @brief Compute difference between two xtimer time stamps
|
|
|
|
*
|
|
|
|
* @param[in] a left operand
|
|
|
|
* @param[in] b right operand
|
|
|
|
*
|
|
|
|
* @return @p a - @p b
|
|
|
|
*/
|
|
|
|
static inline xtimer_ticks32_t xtimer_diff(xtimer_ticks32_t a, xtimer_ticks32_t b);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Compute difference between two xtimer time stamps, 64 bit version
|
|
|
|
*
|
|
|
|
* @param[in] a left operand
|
|
|
|
* @param[in] b right operand
|
|
|
|
*
|
|
|
|
* @return @p a - @p b
|
|
|
|
*/
|
|
|
|
static inline xtimer_ticks64_t xtimer_diff64(xtimer_ticks64_t a, xtimer_ticks64_t b);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Compute 32 bit difference between two 64 bit xtimer time stamps
|
|
|
|
*
|
|
|
|
* @param[in] a left operand
|
|
|
|
* @param[in] b right operand
|
|
|
|
*
|
|
|
|
* @return @p a - @p b cast truncated to 32 bit
|
|
|
|
*/
|
|
|
|
static inline xtimer_ticks32_t xtimer_diff32_64(xtimer_ticks64_t a, xtimer_ticks64_t b);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Compare two xtimer time stamps
|
|
|
|
*
|
|
|
|
* @param[in] a left operand
|
|
|
|
* @param[in] b right operand
|
|
|
|
*
|
|
|
|
* @return @p a < @p b
|
|
|
|
*/
|
|
|
|
static inline bool xtimer_less(xtimer_ticks32_t a, xtimer_ticks32_t b);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Compare two xtimer time stamps, 64 bit version
|
|
|
|
*
|
|
|
|
* @param[in] a left operand
|
|
|
|
* @param[in] b right operand
|
|
|
|
*
|
|
|
|
* @return @p a < @p b
|
|
|
|
*/
|
|
|
|
static inline bool xtimer_less64(xtimer_ticks64_t a, xtimer_ticks64_t b);
|
|
|
|
|
2016-11-22 18:21:36 +01:00
|
|
|
/**
|
|
|
|
* @brief lock a mutex but with timeout
|
|
|
|
*
|
|
|
|
* @param[in] mutex mutex to lock
|
|
|
|
* @param[in] us timeout in microseconds relative
|
|
|
|
*
|
|
|
|
* @return 0, when returned after mutex was locked
|
2020-10-02 08:14:02 +02:00
|
|
|
* @return -1, when the timeout occurred
|
2016-11-22 18:21:36 +01:00
|
|
|
*/
|
|
|
|
int xtimer_mutex_lock_timeout(mutex_t *mutex, uint64_t us);
|
|
|
|
|
2019-07-12 16:23:20 +02:00
|
|
|
/**
|
|
|
|
* @brief lock a rmutex but with timeout
|
|
|
|
*
|
|
|
|
* @param[in] rmutex rmutex to lock
|
|
|
|
* @param[in] us timeout in microseconds relative
|
|
|
|
*
|
|
|
|
* @return 0, when returned after rmutex was locked
|
2020-10-02 08:14:02 +02:00
|
|
|
* @return -1, when the timeout occurred
|
2019-07-12 16:23:20 +02:00
|
|
|
*/
|
|
|
|
int xtimer_rmutex_lock_timeout(rmutex_t *rmutex, uint64_t us);
|
|
|
|
|
2020-02-13 15:37:14 +01:00
|
|
|
#if defined(MODULE_CORE_THREAD_FLAGS) || defined(DOXYGEN)
|
2019-07-12 16:23:20 +02:00
|
|
|
|
2017-09-01 23:52:56 +02:00
|
|
|
/**
|
|
|
|
* @brief Set timeout thread flag after @p timeout
|
|
|
|
*
|
|
|
|
* This function will set THREAD_FLAG_TIMEOUT on the current thread after @p
|
|
|
|
* timeout usec have passed.
|
|
|
|
*
|
|
|
|
* @param[in] t timer struct to use
|
|
|
|
* @param[in] timeout timeout in usec
|
|
|
|
*/
|
|
|
|
void xtimer_set_timeout_flag(xtimer_t *t, uint32_t timeout);
|
|
|
|
|
2020-02-13 15:37:14 +01:00
|
|
|
/**
|
|
|
|
* @brief Set timeout thread flag after @p timeout
|
|
|
|
*
|
|
|
|
* See xtimer_set_timeout_flag() for more information.
|
|
|
|
*
|
|
|
|
* @param[in] t timer struct to use
|
|
|
|
* @param[in] timeout timeout in usec
|
|
|
|
*/
|
|
|
|
void xtimer_set_timeout_flag64(xtimer_t *t, uint64_t timeout);
|
|
|
|
#endif
|
|
|
|
|
2020-01-27 15:28:15 +01:00
|
|
|
/**
|
|
|
|
* @brief Get remaining time of timer
|
|
|
|
*
|
|
|
|
* @param[in] timer timer struct to use
|
|
|
|
*
|
|
|
|
* @returns time in usec until timer triggers
|
|
|
|
* @returns 0 if timer is not set (or has already passed)
|
|
|
|
*/
|
|
|
|
uint64_t xtimer_left_usec(const xtimer_t *timer);
|
|
|
|
|
2019-11-07 13:34:27 +01:00
|
|
|
#if defined(MODULE_CORE_MSG) || defined(DOXYGEN)
|
2019-03-21 15:05:21 +01:00
|
|
|
/**
|
|
|
|
* @brief Set a timer that sends a message
|
|
|
|
*
|
|
|
|
* This function sets a timer that will send a message @p offset ticks
|
|
|
|
* from now.
|
|
|
|
*
|
|
|
|
* The message struct specified by msg parameter will not be copied, e.g., it
|
|
|
|
* needs to point to valid memory until the message has been delivered.
|
|
|
|
*
|
|
|
|
* @param[in] timer timer struct to work with.
|
|
|
|
* @param[in] offset microseconds from now
|
|
|
|
* @param[in] msg ptr to msg that will be sent
|
|
|
|
* @param[in] target_pid pid the message will be sent to
|
|
|
|
*/
|
|
|
|
static inline void xtimer_set_msg(xtimer_t *timer, uint32_t offset, msg_t *msg, kernel_pid_t target_pid);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Set a timer that sends a message, 64bit version
|
|
|
|
*
|
|
|
|
* This function sets a timer that will send a message @p offset microseconds
|
|
|
|
* from now.
|
|
|
|
*
|
|
|
|
* The message struct specified by msg parameter will not be copied, e.g., it
|
|
|
|
* needs to point to valid memory until the message has been delivered.
|
|
|
|
*
|
|
|
|
* @param[in] timer timer struct to work with.
|
|
|
|
* @param[in] offset microseconds from now
|
|
|
|
* @param[in] msg ptr to msg that will be sent
|
|
|
|
* @param[in] target_pid pid the message will be sent to
|
|
|
|
*/
|
|
|
|
static inline void xtimer_set_msg64(xtimer_t *timer, uint64_t offset, msg_t *msg, kernel_pid_t target_pid);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief receive a message blocking but with timeout
|
|
|
|
*
|
|
|
|
* @param[out] msg pointer to a msg_t which will be filled in case of
|
|
|
|
* no timeout
|
|
|
|
* @param[in] timeout timeout in microseconds relative
|
|
|
|
*
|
|
|
|
* @return < 0 on error, other value otherwise
|
|
|
|
*/
|
|
|
|
static inline int xtimer_msg_receive_timeout(msg_t *msg, uint32_t timeout);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief receive a message blocking but with timeout, 64bit version
|
|
|
|
*
|
|
|
|
* @param[out] msg pointer to a msg_t which will be filled in case of no
|
|
|
|
* timeout
|
|
|
|
* @param[in] timeout timeout in microseconds relative
|
|
|
|
*
|
|
|
|
* @return < 0 on error, other value otherwise
|
|
|
|
*/
|
|
|
|
static inline int xtimer_msg_receive_timeout64(msg_t *msg, uint64_t timeout);
|
|
|
|
#endif
|
|
|
|
|
2015-08-14 15:56:47 +02:00
|
|
|
/**
|
|
|
|
* @brief xtimer backoff value
|
|
|
|
*
|
2019-12-04 02:49:18 +01:00
|
|
|
* All timers that are less than XTIMER_BACKOFF ticks in the future will
|
2015-08-14 15:56:47 +02:00
|
|
|
* just spin.
|
|
|
|
*
|
|
|
|
* This is supposed to be defined per-device in e.g., periph_conf.h.
|
|
|
|
*/
|
|
|
|
#ifndef XTIMER_BACKOFF
|
|
|
|
#define XTIMER_BACKOFF 30
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef XTIMER_ISR_BACKOFF
|
|
|
|
/**
|
2016-07-05 16:00:48 +02:00
|
|
|
* @brief xtimer IRQ backoff time, in hardware ticks
|
2015-08-14 15:56:47 +02:00
|
|
|
*
|
2016-07-05 16:00:48 +02:00
|
|
|
* When scheduling the next IRQ, if it is less than the backoff time
|
2015-08-14 15:56:47 +02:00
|
|
|
* in the future, just spin.
|
|
|
|
*
|
|
|
|
* This is supposed to be defined per-device in e.g., periph_conf.h.
|
|
|
|
*/
|
|
|
|
#define XTIMER_ISR_BACKOFF 20
|
|
|
|
#endif
|
|
|
|
|
2016-05-06 22:10:52 +02:00
|
|
|
/*
|
|
|
|
* Default xtimer configuration
|
2015-08-14 15:56:47 +02:00
|
|
|
*/
|
2016-07-05 16:45:41 +02:00
|
|
|
#ifndef XTIMER_DEV
|
2016-05-06 22:10:52 +02:00
|
|
|
/**
|
|
|
|
* @brief Underlying hardware timer device to assign to xtimer
|
|
|
|
*/
|
2016-07-05 16:45:41 +02:00
|
|
|
#define XTIMER_DEV TIMER_DEV(0)
|
2016-05-06 22:10:52 +02:00
|
|
|
/**
|
|
|
|
* @brief Underlying hardware timer channel to assign to xtimer
|
|
|
|
*/
|
2015-08-14 15:56:47 +02:00
|
|
|
#define XTIMER_CHAN (0)
|
|
|
|
|
2016-05-06 22:10:52 +02:00
|
|
|
#if (TIMER_0_MAX_VALUE) == 0xfffffful
|
|
|
|
#define XTIMER_WIDTH (24)
|
|
|
|
#elif (TIMER_0_MAX_VALUE) == 0xffff
|
|
|
|
#define XTIMER_WIDTH (16)
|
2017-10-06 21:48:51 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef XTIMER_WIDTH
|
2015-08-14 15:56:47 +02:00
|
|
|
/**
|
2016-05-06 22:10:52 +02:00
|
|
|
* @brief xtimer timer width
|
|
|
|
*
|
|
|
|
* This value specifies the width (in bits) of the hardware timer used by xtimer.
|
|
|
|
* Default is 32.
|
2015-08-14 15:56:47 +02:00
|
|
|
*/
|
2016-05-06 22:10:52 +02:00
|
|
|
#define XTIMER_WIDTH (32)
|
|
|
|
#endif
|
2015-08-14 15:56:47 +02:00
|
|
|
|
2016-09-29 08:57:16 +02:00
|
|
|
#if (XTIMER_WIDTH != 32) || DOXYGEN
|
2015-08-14 15:56:47 +02:00
|
|
|
/**
|
|
|
|
* @brief xtimer timer mask
|
|
|
|
*
|
|
|
|
* This value specifies the mask relative to 0xffffffff that the used timer
|
|
|
|
* counts to, e.g., 0xffffffff & ~TIMER_MAXVALUE.
|
|
|
|
*
|
|
|
|
* For a 16bit timer, the mask would be 0xFFFF0000, for a 24bit timer, the mask
|
2016-05-06 22:10:52 +02:00
|
|
|
* would be 0xFF000000.
|
2015-08-14 15:56:47 +02:00
|
|
|
*/
|
2016-05-06 22:10:52 +02:00
|
|
|
#define XTIMER_MASK ((0xffffffff >> XTIMER_WIDTH) << XTIMER_WIDTH)
|
|
|
|
#else
|
2015-09-11 16:01:26 +02:00
|
|
|
#define XTIMER_MASK (0)
|
2015-08-14 15:56:47 +02:00
|
|
|
#endif
|
2016-05-06 22:10:52 +02:00
|
|
|
|
2017-03-06 20:34:13 +01:00
|
|
|
/**
|
|
|
|
* @brief Base frequency of xtimer is 1 MHz
|
|
|
|
*/
|
|
|
|
#define XTIMER_HZ_BASE (1000000ul)
|
|
|
|
|
2020-08-19 07:05:32 +02:00
|
|
|
#if !defined(XTIMER_HZ) && !defined(MODULE_XTIMER_ON_ZTIMER)
|
2016-07-05 16:00:48 +02:00
|
|
|
/**
|
|
|
|
* @brief Frequency of the underlying hardware timer
|
|
|
|
*/
|
2017-03-06 20:34:13 +01:00
|
|
|
#define XTIMER_HZ XTIMER_HZ_BASE
|
|
|
|
#endif
|
|
|
|
|
2020-08-19 07:05:32 +02:00
|
|
|
#if !defined(XTIMER_SHIFT) && !defined(MODULE_XTIMER_ON_ZTIMER)
|
2017-10-10 12:13:25 +02:00
|
|
|
#if (XTIMER_HZ == 32768ul)
|
|
|
|
/* No shift necessary, the conversion is not a power of two and is handled by
|
|
|
|
* functions in tick_conversion.h */
|
|
|
|
#define XTIMER_SHIFT (0)
|
|
|
|
#elif (XTIMER_HZ == XTIMER_HZ_BASE)
|
2017-03-06 20:34:13 +01:00
|
|
|
/**
|
|
|
|
* @brief xtimer prescaler value
|
|
|
|
*
|
|
|
|
* If the underlying hardware timer is running at a power of two multiple of
|
|
|
|
* 15625, XTIMER_SHIFT can be used to adjust the difference.
|
|
|
|
*
|
|
|
|
* For a 1 MHz hardware timer, set XTIMER_SHIFT to 0.
|
|
|
|
* For a 2 MHz or 500 kHz, set XTIMER_SHIFT to 1.
|
|
|
|
* For a 4 MHz or 250 kHz, set XTIMER_SHIFT to 2.
|
|
|
|
* For a 8 MHz or 125 kHz, set XTIMER_SHIFT to 3.
|
|
|
|
* For a 16 MHz or 62.5 kHz, set XTIMER_SHIFT to 4.
|
|
|
|
* and for 32 MHz, set XTIMER_SHIFT to 5.
|
|
|
|
*
|
|
|
|
* The direction of the shift is handled by the macros in tick_conversion.h
|
|
|
|
*/
|
|
|
|
#define XTIMER_SHIFT (0)
|
|
|
|
#elif (XTIMER_HZ >> 1 == XTIMER_HZ_BASE) || (XTIMER_HZ << 1 == XTIMER_HZ_BASE)
|
|
|
|
#define XTIMER_SHIFT (1)
|
|
|
|
#elif (XTIMER_HZ >> 2 == XTIMER_HZ_BASE) || (XTIMER_HZ << 2 == XTIMER_HZ_BASE)
|
|
|
|
#define XTIMER_SHIFT (2)
|
|
|
|
#elif (XTIMER_HZ >> 3 == XTIMER_HZ_BASE) || (XTIMER_HZ << 3 == XTIMER_HZ_BASE)
|
|
|
|
#define XTIMER_SHIFT (3)
|
|
|
|
#elif (XTIMER_HZ >> 4 == XTIMER_HZ_BASE) || (XTIMER_HZ << 4 == XTIMER_HZ_BASE)
|
|
|
|
#define XTIMER_SHIFT (4)
|
|
|
|
#elif (XTIMER_HZ >> 5 == XTIMER_HZ_BASE) || (XTIMER_HZ << 5 == XTIMER_HZ_BASE)
|
|
|
|
#define XTIMER_SHIFT (5)
|
2018-03-30 10:12:37 +02:00
|
|
|
#elif (XTIMER_HZ >> 6 == XTIMER_HZ_BASE) || (XTIMER_HZ << 6 == XTIMER_HZ_BASE)
|
|
|
|
#define XTIMER_SHIFT (6)
|
2017-03-06 20:34:13 +01:00
|
|
|
#else
|
|
|
|
#error "XTIMER_SHIFT cannot be derived for given XTIMER_HZ, verify settings!"
|
|
|
|
#endif
|
2020-08-19 07:05:32 +02:00
|
|
|
#elif !defined(MODULE_XTIMER_ON_ZTIMER)
|
2017-03-06 20:34:13 +01:00
|
|
|
#error "XTIMER_SHIFT is set relative to XTIMER_HZ, no manual define required!"
|
2016-07-05 16:00:48 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "xtimer/tick_conversion.h"
|
2015-08-14 15:56:47 +02:00
|
|
|
|
2016-07-05 11:11:48 +02:00
|
|
|
#include "xtimer/implementation.h"
|
2015-08-14 15:56:47 +02:00
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
2015-09-15 00:11:28 +02:00
|
|
|
|
2020-04-30 16:45:34 +02:00
|
|
|
#endif /* MODULE_ZTIMER_XTIMER_COMPAT */
|
2019-12-10 11:04:29 +01:00
|
|
|
|
2015-09-15 00:11:28 +02:00
|
|
|
/** @} */
|
2015-08-14 15:56:47 +02:00
|
|
|
#endif /* XTIMER_H */
|