/* ChibiOS - Copyright (C) 2006..2015 Giovanni Di Sirio. This file is part of ChibiOS. ChibiOS is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 3 of the License, or (at your option) any later version. ChibiOS is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . */ /** * @file chthreads.h * @brief Threads module macros and structures. * * @addtogroup threads * @{ */ #ifndef _CHTHREADS_H_ #define _CHTHREADS_H_ /*lint -sem(chThdExit, r_no) -sem(chThdExitS, r_no)*/ /*===========================================================================*/ /* Module constants. */ /*===========================================================================*/ /*===========================================================================*/ /* Module pre-compile time settings. */ /*===========================================================================*/ #if !defined(CH_CFG_THREAD_EXTRA_FIELDS) #error "CH_CFG_THREAD_EXTRA_FIELDS not defined in chconf.h" #endif #if !defined(CH_CFG_THREAD_INIT_HOOK) #error "CH_CFG_THREAD_INIT_HOOK not defined in chconf.h" #endif #if !defined(CH_CFG_THREAD_EXIT_HOOK) #error "CH_CFG_THREAD_EXIT_HOOK not defined in chconf.h" #endif /*===========================================================================*/ /* Derived constants and error checks. */ /*===========================================================================*/ /*===========================================================================*/ /* Module data structures and types. */ /*===========================================================================*/ /** * @brief Thread function. */ typedef void (*tfunc_t)(void *p); /** * @brief Type of a thread descriptor. */ typedef struct { /** * @brief Thread name. */ const char *name; /** * @brief Pointer to the working area base. */ stkalign_t *wbase; /** * @brief End of the working area. */ stkalign_t *wend; /** * @brief Thread priority. */ tprio_t prio; /** * @brief Thread function pointer. */ tfunc_t funcp; /** * @brief Thread argument. */ void *arg; } thread_descriptor_t; /*===========================================================================*/ /* Module macros. */ /*===========================================================================*/ /** * @name Threads queues */ /** * @brief Data part of a static threads queue object initializer. * @details This macro should be used when statically initializing a threads * queue that is part of a bigger structure. * * @param[in] name the name of the threads queue variable */ #define _THREADS_QUEUE_DATA(name) {(thread_t *)&name, (thread_t *)&name} /** * @brief Static threads queue object initializer. * @details Statically initialized threads queues require no explicit * initialization using @p queue_init(). * * @param[in] name the name of the threads queue variable */ #define _THREADS_QUEUE_DECL(name) \ threads_queue_t name = _THREADS_QUEUE_DATA(name) /** @} */ /** * @name Macro Functions * @{ */ /** * @brief Delays the invoking thread for the specified number of seconds. * @note The specified time is rounded up to a value allowed by the real * system tick clock. * @note The maximum specifiable value is implementation dependent. * * @param[in] sec time in seconds, must be different from zero * * @api */ #define chThdSleepSeconds(sec) chThdSleep(S2ST(sec)) /** * @brief Delays the invoking thread for the specified number of * milliseconds. * @note The specified time is rounded up to a value allowed by the real * system tick clock. * @note The maximum specifiable value is implementation dependent. * * @param[in] msec time in milliseconds, must be different from zero * * @api */ #define chThdSleepMilliseconds(msec) chThdSleep(MS2ST(msec)) /** * @brief Delays the invoking thread for the specified number of * microseconds. * @note The specified time is rounded up to a value allowed by the real * system tick clock. * @note The maximum specifiable value is implementation dependent. * * @param[in] usec time in microseconds, must be different from zero * * @api */ #define chThdSleepMicroseconds(usec) chThdSleep(US2ST(usec)) /** @} */ /*===========================================================================*/ /* External declarations. */ /*===========================================================================*/ #ifdef __cplusplus extern "C" { #endif thread_t *_thread_init(thread_t *tp, const char *name, tprio_t prio); #if CH_DBG_FILL_THREADS == TRUE void _thread_memfill(uint8_t *startp, uint8_t *endp, uint8_t v); #endif thread_t *chThdCreateSuspendedI(const thread_descriptor_t *tdp); thread_t *chThdCreateSuspended(const thread_descriptor_t *tdp); thread_t *chThdCreateI(const thread_descriptor_t *tdp); thread_t *chThdCreate(const thread_descriptor_t *tdp); thread_t *chThdCreateStatic(void *wsp, size_t size, tprio_t prio, tfunc_t pf, void *arg); thread_t *chThdStart(thread_t *tp); tprio_t chThdSetPriority(tprio_t newprio); msg_t chThdSuspendS(thread_reference_t *trp); msg_t chThdSuspendTimeoutS(thread_reference_t *trp, systime_t timeout); void chThdResumeI(thread_reference_t *trp, msg_t msg); void chThdResumeS(thread_reference_t *trp, msg_t msg); void chThdResume(thread_reference_t *trp, msg_t msg); msg_t chThdEnqueueTimeoutS(threads_queue_t *tqp, systime_t timeout); void chThdDequeueNextI(threads_queue_t *tqp, msg_t msg); void chThdDequeueAllI(threads_queue_t *tqp, msg_t msg); void chThdSleep(systime_t time); void chThdSleepUntil(systime_t time); systime_t chThdSleepUntilWindowed(systime_t prev, systime_t next); void chThdYield(void); void chThdExit(msg_t msg); void chThdExitS(msg_t msg); #if CH_CFG_USE_WAITEXIT == TRUE msg_t chThdWait(thread_t *tp); #endif #ifdef __cplusplus } #endif /*===========================================================================*/ /* Module inline functions. */ /*===========================================================================*/ /** * @brief Returns a pointer to the current @p thread_t. * * @return A pointer to the current thread. * * @xclass */ static inline thread_t *chThdGetSelfX(void) { return ch.rlist.current; } /** * @brief Returns the current thread priority. * @note Can be invoked in any context. * * @return The current thread priority. * * @xclass */ static inline tprio_t chThdGetPriorityX(void) { return chThdGetSelfX()->prio; } /** * @brief Returns the number of ticks consumed by the specified thread. * @note This function is only available when the * @p CH_DBG_THREADS_PROFILING configuration option is enabled. * * @param[in] tp pointer to the thread * @return The number of consumed system ticks. * * @xclass */ #if (CH_DBG_THREADS_PROFILING == TRUE) || defined(__DOXYGEN__) static inline systime_t chThdGetTicksX(thread_t *tp) { return tp->time; } #endif /** * @brief Returns the stack limit of the specified thread. * * @param[in] tp pointer to the thread * @return The stack limit pointer. * * @xclass */ static inline stkalign_t *chthdGetStackLimitX(thread_t *tp) { return tp->stklimit; } /** * @brief Verifies if the specified thread is in the @p CH_STATE_FINAL state. * * @param[in] tp pointer to the thread * @retval true thread terminated. * @retval false thread not terminated. * * @xclass */ static inline bool chThdTerminatedX(thread_t *tp) { return (bool)(tp->state == CH_STATE_FINAL); } /** * @brief Resumes a thread created with @p chThdCreateI(). * * @param[in] tp pointer to the thread * @return The pointer to the @p thread_t structure allocated for * the thread into the working space area. * * @iclass */ static inline thread_t *chThdStartI(thread_t *tp) { chDbgAssert(tp->state == CH_STATE_WTSTART, "wrong state"); return chSchReadyI(tp); } /** * @brief Suspends the invoking thread for the specified time. * * @param[in] time the delay in system ticks, the special values are * handled as follow: * - @a TIME_INFINITE the thread enters an infinite sleep * state. * - @a TIME_IMMEDIATE this value is not allowed. * . * * @sclass */ static inline void chThdSleepS(systime_t time) { chDbgCheck(time != TIME_IMMEDIATE); (void) chSchGoSleepTimeoutS(CH_STATE_SLEEPING, time); } /** * @brief Initializes a threads queue object. * * @param[out] tqp pointer to the threads queue object * * @init */ static inline void chThdQueueObjectInit(threads_queue_t *tqp) { queue_init(tqp); } /** * @brief Evaluates to @p true if the specified queue is empty. * * @param[out] tqp pointer to the threads queue object * @return The queue status. * @retval false if the queue is not empty. * @retval true if the queue is empty. * * @iclass */ static inline bool chThdQueueIsEmptyI(threads_queue_t *tqp) { chDbgCheckClassI(); return queue_isempty(tqp); } /** * @brief Dequeues and wakes up one thread from the threads queue object. * @details Dequeues one thread from the queue without checking if the queue * is empty. * @pre The queue must contain at least an object. * * @param[in] tqp pointer to the threads queue object * @param[in] msg the message code * * @iclass */ static inline void chThdDoDequeueNextI(threads_queue_t *tqp, msg_t msg) { thread_t *tp; chDbgAssert(queue_notempty(tqp), "empty queue"); tp = queue_fifo_remove(tqp); chDbgAssert(tp->state == CH_STATE_QUEUED, "invalid state"); tp->u.rdymsg = msg; (void) chSchReadyI(tp); } #endif /* _CHTHREADS_H_ */ /** @} */