123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577 |
- /*
- * Copyright (c) 2017 Intel Corporation
- *
- * SPDX-License-Identifier: Apache-2.0
- */
- #ifndef ZEPHYR_INCLUDE_POSIX_PTHREAD_H_
- #define ZEPHYR_INCLUDE_POSIX_PTHREAD_H_
- #include <kernel.h>
- #include <wait_q.h>
- #include <posix/time.h>
- #include <posix/unistd.h>
- #include "posix_types.h"
- #include "posix_sched.h"
- #include <posix/pthread_key.h>
- #include <stdlib.h>
- #include <string.h>
- #ifdef __cplusplus
- extern "C" {
- #endif
- enum pthread_state {
- /* The thread structure is unallocated and available for reuse. */
- PTHREAD_TERMINATED = 0,
- /* The thread is running and joinable. */
- PTHREAD_JOINABLE,
- /* The thread is running and detached. */
- PTHREAD_DETACHED,
- /* A joinable thread exited and its return code is available. */
- PTHREAD_EXITED
- };
- struct posix_thread {
- struct k_thread thread;
- /* List of keys that thread has called pthread_setspecific() on */
- sys_slist_t key_list;
- /* Dynamic stack */
- void *dynamic_stack;
- /* Exit status */
- void *retval;
- /* Pthread cancellation */
- int cancel_state;
- int cancel_pending;
- pthread_mutex_t cancel_lock;
- /* Pthread State */
- enum pthread_state state;
- pthread_mutex_t state_lock;
- pthread_cond_t state_cond;
- };
- /* Pthread detach/joinable */
- #define PTHREAD_CREATE_JOINABLE PTHREAD_JOINABLE
- #define PTHREAD_CREATE_DETACHED PTHREAD_DETACHED
- /* Pthread cancellation */
- #define _PTHREAD_CANCEL_POS 0
- #define PTHREAD_CANCEL_ENABLE (0U << _PTHREAD_CANCEL_POS)
- #define PTHREAD_CANCEL_DISABLE BIT(_PTHREAD_CANCEL_POS)
- /* Passed to pthread_once */
- #define PTHREAD_ONCE_INIT 1
- /**
- * @brief Declare a pthread condition variable
- *
- * Declaration API for a pthread condition variable. This is not a
- * POSIX API, it's provided to better conform with Zephyr's allocation
- * strategies for kernel objects.
- *
- * @param name Symbol name of the condition variable
- */
- #define PTHREAD_COND_DEFINE(name) \
- struct pthread_cond name = { \
- .wait_q = Z_WAIT_Q_INIT(&name.wait_q), \
- }
- /**
- * @brief POSIX threading compatibility API
- *
- * See IEEE 1003.1
- */
- static inline int pthread_cond_init(pthread_cond_t *cv,
- const pthread_condattr_t *att)
- {
- ARG_UNUSED(att);
- if (cv == NULL)
- return EINVAL;
- cv->attr.type = PTHREAD_PROCESS_PRIVATE;
- z_waitq_init(&cv->wait_q);
- return 0;
- }
- /**
- * @brief POSIX threading compatibility API
- *
- * See IEEE 1003.1
- */
- static inline int pthread_cond_destroy(pthread_cond_t *cv)
- {
- if ((cv == NULL) || (cv->attr.type == -1))
- return EINVAL;
- cv->attr.type = -1;
- return 0;
- }
- /**
- * @brief POSIX threading compatibility API
- *
- * See IEEE 1003.1
- */
- int pthread_cond_signal(pthread_cond_t *cv);
- /**
- * @brief POSIX threading compatibility API
- *
- * See IEEE 1003.1
- */
- int pthread_cond_broadcast(pthread_cond_t *cv);
- /**
- * @brief POSIX threading compatibility API
- *
- * See IEEE 1003.1
- */
- int pthread_cond_wait(pthread_cond_t *cv, pthread_mutex_t *mut);
- /**
- * @brief POSIX threading compatibility API
- *
- * See IEEE 1003.1
- */
- int pthread_cond_timedwait(pthread_cond_t *cv, pthread_mutex_t *mut,
- const struct timespec *abstime);
- /**
- * @brief POSIX threading compatibility API
- *
- * See IEEE 1003.1.
- *
- * Note that pthread attribute structs are currently noops in Zephyr.
- */
- static inline int pthread_condattr_init(pthread_condattr_t *att)
- {
- return 0;
- }
- /**
- * @brief POSIX threading compatibility API
- *
- * See IEEE 1003.1
- *
- * Note that pthread attribute structs are currently noops in Zephyr.
- */
- static inline int pthread_condattr_destroy(pthread_condattr_t *att)
- {
- return 0;
- }
- /**
- * @brief Declare a pthread mutex
- *
- * Declaration API for a pthread mutex. This is not a POSIX API, it's
- * provided to better conform with Zephyr's allocation strategies for
- * kernel objects.
- *
- * @param name Symbol name of the mutex
- */
- #define PTHREAD_MUTEX_DEFINE(name) \
- struct pthread_mutex name = \
- { \
- .lock_count = 0, \
- .wait_q = Z_WAIT_Q_INIT(&name.wait_q), \
- .owner = NULL, \
- }
- /*
- * Mutex attributes - type
- *
- * PTHREAD_MUTEX_NORMAL: Owner of mutex cannot relock it. Attempting
- * to relock will cause deadlock.
- * PTHREAD_MUTEX_RECURSIVE: Owner can relock the mutex.
- * PTHREAD_MUTEX_ERRORCHECK: If owner attempts to relock the mutex, an
- * error is returned.
- *
- */
- #define PTHREAD_MUTEX_NORMAL 0
- #define PTHREAD_MUTEX_RECURSIVE 1
- #define PTHREAD_MUTEX_ERRORCHECK 2
- #define PTHREAD_MUTEX_DEFAULT PTHREAD_MUTEX_NORMAL
- /*
- * Mutex attributes - protocol
- *
- * PTHREAD_PRIO_NONE: Ownership of mutex does not affect priority.
- * PTHREAD_PRIO_INHERIT: Owner's priority is boosted to the priority of
- * highest priority thread blocked on the mutex.
- * PTHREAD_PRIO_PROTECT: Mutex has a priority ceiling. The owner's
- * priority is boosted to the highest priority ceiling of all mutexes
- * owned (regardless of whether or not other threads are blocked on
- * any of these mutexes).
- * FIXME: Only PRIO_NONE is supported. Implement other protocols.
- */
- #define PTHREAD_PRIO_NONE 0
- /**
- * @brief POSIX threading compatibility API
- *
- * See IEEE 1003.1
- */
- int pthread_mutex_destroy(pthread_mutex_t *m);
- /**
- * @brief POSIX threading compatibility API
- *
- * See IEEE 1003.1
- */
- int pthread_mutex_lock(pthread_mutex_t *m);
- /**
- * @brief POSIX threading compatibility API
- *
- * See IEEE 1003.1
- */
- int pthread_mutex_unlock(pthread_mutex_t *m);
- /**
- * @brief POSIX threading compatibility API
- *
- * See IEEE 1003.1
- */
- int pthread_mutex_timedlock(pthread_mutex_t *m,
- const struct timespec *abstime);
- /**
- * @brief POSIX threading compatibility API
- *
- * See IEEE 1003.1
- */
- int pthread_mutex_trylock(pthread_mutex_t *m);
- /**
- * @brief POSIX threading compatibility API
- *
- * See IEEE 1003.1
- */
- int pthread_mutex_init(pthread_mutex_t *m,
- const pthread_mutexattr_t *att);
- /**
- * @brief POSIX threading compatibility API
- *
- * See IEEE 1003.1
- */
- int pthread_mutexattr_setprotocol(pthread_mutexattr_t *attr, int protocol);
- /**
- * @brief POSIX threading compatibility API
- *
- * See IEEE 1003.1
- */
- int pthread_mutexattr_settype(pthread_mutexattr_t *attr, int type);
- /**
- * @brief POSIX threading compatibility API
- *
- * See IEEE 1003.1
- */
- int pthread_mutexattr_getprotocol(const pthread_mutexattr_t *attr,
- int *protocol);
- /**
- * @brief POSIX threading compatibility API
- *
- * See IEEE 1003.1
- */
- int pthread_mutexattr_gettype(const pthread_mutexattr_t *attr, int *type);
- /**
- * @brief POSIX threading compatibility API
- *
- * See IEEE 1003.1
- *
- * Note that pthread attribute structs are currently noops in Zephyr.
- */
- static inline int pthread_mutexattr_init(pthread_mutexattr_t *m)
- {
- ARG_UNUSED(m);
- return 0;
- }
- /**
- * @brief POSIX threading compatibility API
- *
- * See IEEE 1003.1
- *
- * Note that pthread attribute structs are currently noops in Zephyr.
- */
- static inline int pthread_mutexattr_destroy(pthread_mutexattr_t *m)
- {
- ARG_UNUSED(m);
- return 0;
- }
- /* FIXME: these are going to be tricky to implement. Zephyr has (for
- * good reason) deprecated its own "initializer" macros in favor of a
- * static "declaration" macros instead. Using such a macro inside a
- * gcc compound expression to declare and object then reference it
- * would work, but gcc limits such expressions to function context
- * (because they may need to generate code that runs at assignment
- * time) and much real-world use of these initializers is for static
- * variables. The best trick I can think of would be to declare it in
- * a special section and then initialize that section at runtime
- * startup, which sort of defeats the purpose of having these be
- * static...
- *
- * Instead, see the nonstandard PTHREAD_*_DEFINE macros instead, which
- * work similarly but conform to Zephyr's paradigms.
- */
- #define PTHREAD_MUTEX_INITIALIZER {.attr.type = -1}
- #define PTHREAD_COND_INITIALIZER {.attr.type = -1}
- /**
- * @brief Declare a pthread barrier
- *
- * Declaration API for a pthread barrier. This is not a
- * POSIX API, it's provided to better conform with Zephyr's allocation
- * strategies for kernel objects.
- *
- * @param name Symbol name of the barrier
- * @param count Thread count, same as the "count" argument to
- * pthread_barrier_init()
- */
- #define PTHREAD_BARRIER_DEFINE(name, count) \
- struct pthread_barrier name = { \
- .wait_q = Z_WAIT_Q_INIT(&name.wait_q), \
- .max = count, \
- }
- #define PTHREAD_BARRIER_SERIAL_THREAD 1
- /**
- * @brief POSIX threading compatibility API
- *
- * See IEEE 1003.1
- */
- int pthread_barrier_wait(pthread_barrier_t *b);
- /**
- * @brief POSIX threading compatibility API
- *
- * See IEEE 1003.1
- */
- static inline int pthread_barrier_init(pthread_barrier_t *b,
- const pthread_barrierattr_t *attr,
- unsigned int count)
- {
- ARG_UNUSED(attr);
- b->max = count;
- b->count = 0;
- z_waitq_init(&b->wait_q);
- return 0;
- }
- /**
- * @brief POSIX threading compatibility API
- *
- * See IEEE 1003.1
- */
- static inline int pthread_barrier_destroy(pthread_barrier_t *b)
- {
- ARG_UNUSED(b);
- return 0;
- }
- /**
- * @brief POSIX threading compatibility API
- *
- * See IEEE 1003.1
- *
- * Note that pthread attribute structs are currently noops in Zephyr.
- */
- static inline int pthread_barrierattr_init(pthread_barrierattr_t *b)
- {
- ARG_UNUSED(b);
- return 0;
- }
- /**
- * @brief POSIX threading compatibility API
- *
- * See IEEE 1003.1
- *
- * Note that pthread attribute structs are currently noops in Zephyr.
- */
- static inline int pthread_barrierattr_destroy(pthread_barrierattr_t *b)
- {
- ARG_UNUSED(b);
- return 0;
- }
- /* Predicates and setters for various pthread attribute values that we
- * don't support (or always support: the "process shared" attribute
- * can only be true given the way Zephyr implements these
- * objects). Leave these undefined for simplicity instead of defining
- * stubs to return an error that would have to be logged and
- * interpreted just to figure out that we didn't support it in the
- * first place. These APIs are very rarely used even in production
- * Unix code. Leave the declarations here so they can be easily
- * uncommented and implemented as needed.
- int pthread_condattr_getclock(const pthread_condattr_t * clockid_t *);
- int pthread_condattr_getpshared(const pthread_condattr_t * int *);
- int pthread_condattr_setclock(pthread_condattr_t *, clockid_t);
- int pthread_condattr_setpshared(pthread_condattr_t *, int);
- int pthread_mutex_consistent(pthread_mutex_t *);
- int pthread_mutex_getprioceiling(const pthread_mutex_t * int *);
- int pthread_mutex_setprioceiling(pthread_mutex_t *, int int *);
- int pthread_mutexattr_getprioceiling(const pthread_mutexattr_t *, int *);
- int pthread_mutexattr_getpshared(const pthread_mutexattr_t * int *);
- int pthread_mutexattr_getrobust(const pthread_mutexattr_t * int *);
- int pthread_mutexattr_setprioceiling(pthread_mutexattr_t *, int);
- int pthread_mutexattr_setpshared(pthread_mutexattr_t *, int);
- int pthread_mutexattr_setrobust(pthread_mutexattr_t *, int);
- int pthread_barrierattr_getpshared(const pthread_barrierattr_t *, int *);
- int pthread_barrierattr_setpshared(pthread_barrierattr_t *, int);
- */
- /* Base Pthread related APIs */
- /**
- * @brief Obtain ID of the calling thread.
- *
- * The results of calling this API from threads not created with
- * pthread_create() are undefined.
- *
- * See IEEE 1003.1
- */
- static inline pthread_t pthread_self(void)
- {
- return (pthread_t)k_current_get();
- }
- /**
- * @brief Compare thread IDs.
- *
- * See IEEE 1003.1
- */
- static inline int pthread_equal(pthread_t pt1, pthread_t pt2)
- {
- return (pt1 == pt2);
- }
- /**
- * @brief Destroy the read-write lock attributes object.
- *
- * See IEEE 1003.1
- */
- static inline int pthread_rwlockattr_destroy(pthread_rwlockattr_t *attr)
- {
- return 0;
- }
- /**
- * @brief initialize the read-write lock attributes object.
- *
- * See IEEE 1003.1
- */
- static inline int pthread_rwlockattr_init(pthread_rwlockattr_t *attr)
- {
- return 0;
- }
- int pthread_attr_getstacksize(const pthread_attr_t *attr, size_t *stacksize);
- int pthread_attr_setschedpolicy(pthread_attr_t *attr, int policy);
- int pthread_attr_getschedpolicy(const pthread_attr_t *attr, int *policy);
- int pthread_attr_setdetachstate(pthread_attr_t *attr, int detachstate);
- int pthread_attr_getdetachstate(const pthread_attr_t *attr, int *detachstate);
- int pthread_attr_init(pthread_attr_t *attr);
- int pthread_attr_destroy(pthread_attr_t *attr);
- int pthread_attr_getschedparam(const pthread_attr_t *attr,
- struct sched_param *schedparam);
- int pthread_getschedparam(pthread_t pthread, int *policy,
- struct sched_param *param);
- int pthread_attr_getstack(const pthread_attr_t *attr,
- void **stackaddr, size_t *stacksize);
- int pthread_attr_setstack(pthread_attr_t *attr, void *stackaddr,
- size_t stacksize);
- int pthread_attr_setstacksize(pthread_attr_t *attr, size_t stacksize);
- int pthread_once(pthread_once_t *once, void (*initFunc)(void));
- void pthread_exit(void *retval);
- int pthread_join(pthread_t thread, void **status);
- int pthread_cancel(pthread_t pthread);
- int pthread_detach(pthread_t thread);
- int pthread_create(pthread_t *newthread, const pthread_attr_t *attr,
- void *(*threadroutine)(void *), void *arg);
- int pthread_setcancelstate(int state, int *oldstate);
- int pthread_attr_setschedparam(pthread_attr_t *attr,
- const struct sched_param *schedparam);
- int pthread_setschedparam(pthread_t pthread, int policy,
- const struct sched_param *param);
- int pthread_rwlock_destroy(pthread_rwlock_t *rwlock);
- int pthread_rwlock_init(pthread_rwlock_t *rwlock,
- const pthread_rwlockattr_t *attr);
- int pthread_rwlock_rdlock(pthread_rwlock_t *rwlock);
- int pthread_rwlock_timedrdlock(pthread_rwlock_t *rwlock,
- const struct timespec *abstime);
- int pthread_rwlock_timedwrlock(pthread_rwlock_t *rwlock,
- const struct timespec *abstime);
- int pthread_rwlock_tryrdlock(pthread_rwlock_t *rwlock);
- int pthread_rwlock_trywrlock(pthread_rwlock_t *rwlock);
- int pthread_rwlock_unlock(pthread_rwlock_t *rwlock);
- int pthread_rwlock_wrlock(pthread_rwlock_t *rwlock);
- int pthread_key_create(pthread_key_t *key,
- void (*destructor)(void *));
- int pthread_key_delete(pthread_key_t key);
- int pthread_setspecific(pthread_key_t key, const void *value);
- void *pthread_getspecific(pthread_key_t key);
- /* Glibc / Oracle Extension Functions */
- /**
- * @brief Set name of POSIX thread.
- *
- * Non-portable, extension function that conforms with most
- * other definitions of this function.
- *
- * @param thread POSIX thread to set name
- * @param name Name string
- * @retval 0 Success
- * @retval ESRCH Thread does not exist
- * @retval EINVAL Name buffer is NULL
- * @retval Negative value if kernel function error
- *
- */
- int pthread_setname_np(pthread_t thread, const char *name);
- /**
- * @brief Get name of POSIX thread and store in name buffer
- * that is of size len.
- *
- * Non-portable, extension function that conforms with most
- * other definitions of this function.
- *
- * @param thread POSIX thread to obtain name information
- * @param name Destination buffer
- * @param len Destination buffer size
- * @retval 0 Success
- * @retval ESRCH Thread does not exist
- * @retval EINVAL Name buffer is NULL
- * @retval Negative value if kernel function error
- */
- int pthread_getname_np(pthread_t thread, char *name, size_t len);
- #ifdef __cplusplus
- }
- #endif
- #endif /* ZEPHYR_INCLUDE_POSIX_PTHREAD_H_ */
|