pthread.h 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564
  1. /*
  2. * Copyright (c) 2017 Intel Corporation
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #ifndef ZEPHYR_INCLUDE_POSIX_PTHREAD_H_
  7. #define ZEPHYR_INCLUDE_POSIX_PTHREAD_H_
  8. #include <kernel.h>
  9. #include <wait_q.h>
  10. #include <posix/time.h>
  11. #include <posix/unistd.h>
  12. #include "posix_types.h"
  13. #include "posix_sched.h"
  14. #include <posix/pthread_key.h>
  15. #include <stdlib.h>
  16. #include <string.h>
  17. #ifdef __cplusplus
  18. extern "C" {
  19. #endif
  20. enum pthread_state {
  21. /* The thread structure is unallocated and available for reuse. */
  22. PTHREAD_TERMINATED = 0,
  23. /* The thread is running and joinable. */
  24. PTHREAD_JOINABLE,
  25. /* The thread is running and detached. */
  26. PTHREAD_DETACHED,
  27. /* A joinable thread exited and its return code is available. */
  28. PTHREAD_EXITED
  29. };
  30. struct posix_thread {
  31. struct k_thread thread;
  32. /* List of keys that thread has called pthread_setspecific() on */
  33. sys_slist_t key_list;
  34. /* Exit status */
  35. void *retval;
  36. /* Pthread cancellation */
  37. int cancel_state;
  38. int cancel_pending;
  39. pthread_mutex_t cancel_lock;
  40. /* Pthread State */
  41. enum pthread_state state;
  42. pthread_mutex_t state_lock;
  43. pthread_cond_t state_cond;
  44. };
  45. /* Pthread detach/joinable */
  46. #define PTHREAD_CREATE_JOINABLE PTHREAD_JOINABLE
  47. #define PTHREAD_CREATE_DETACHED PTHREAD_DETACHED
  48. /* Pthread cancellation */
  49. #define _PTHREAD_CANCEL_POS 0
  50. #define PTHREAD_CANCEL_ENABLE (0U << _PTHREAD_CANCEL_POS)
  51. #define PTHREAD_CANCEL_DISABLE BIT(_PTHREAD_CANCEL_POS)
  52. /* Passed to pthread_once */
  53. #define PTHREAD_ONCE_INIT 1
  54. /**
  55. * @brief Declare a pthread condition variable
  56. *
  57. * Declaration API for a pthread condition variable. This is not a
  58. * POSIX API, it's provided to better conform with Zephyr's allocation
  59. * strategies for kernel objects.
  60. *
  61. * @param name Symbol name of the condition variable
  62. */
  63. #define PTHREAD_COND_DEFINE(name) \
  64. struct pthread_cond name = { \
  65. .wait_q = Z_WAIT_Q_INIT(&name.wait_q), \
  66. }
  67. /**
  68. * @brief POSIX threading compatibility API
  69. *
  70. * See IEEE 1003.1
  71. */
  72. static inline int pthread_cond_init(pthread_cond_t *cv,
  73. const pthread_condattr_t *att)
  74. {
  75. ARG_UNUSED(att);
  76. z_waitq_init(&cv->wait_q);
  77. return 0;
  78. }
  79. /**
  80. * @brief POSIX threading compatibility API
  81. *
  82. * See IEEE 1003.1
  83. */
  84. static inline int pthread_cond_destroy(pthread_cond_t *cv)
  85. {
  86. return 0;
  87. }
  88. /**
  89. * @brief POSIX threading compatibility API
  90. *
  91. * See IEEE 1003.1
  92. */
  93. int pthread_cond_signal(pthread_cond_t *cv);
  94. /**
  95. * @brief POSIX threading compatibility API
  96. *
  97. * See IEEE 1003.1
  98. */
  99. int pthread_cond_broadcast(pthread_cond_t *cv);
  100. /**
  101. * @brief POSIX threading compatibility API
  102. *
  103. * See IEEE 1003.1
  104. */
  105. int pthread_cond_wait(pthread_cond_t *cv, pthread_mutex_t *mut);
  106. /**
  107. * @brief POSIX threading compatibility API
  108. *
  109. * See IEEE 1003.1
  110. */
  111. int pthread_cond_timedwait(pthread_cond_t *cv, pthread_mutex_t *mut,
  112. const struct timespec *abstime);
  113. /**
  114. * @brief POSIX threading compatibility API
  115. *
  116. * See IEEE 1003.1.
  117. *
  118. * Note that pthread attribute structs are currently noops in Zephyr.
  119. */
  120. static inline int pthread_condattr_init(pthread_condattr_t *att)
  121. {
  122. return 0;
  123. }
  124. /**
  125. * @brief POSIX threading compatibility API
  126. *
  127. * See IEEE 1003.1
  128. *
  129. * Note that pthread attribute structs are currently noops in Zephyr.
  130. */
  131. static inline int pthread_condattr_destroy(pthread_condattr_t *att)
  132. {
  133. return 0;
  134. }
  135. /**
  136. * @brief Declare a pthread mutex
  137. *
  138. * Declaration API for a pthread mutex. This is not a POSIX API, it's
  139. * provided to better conform with Zephyr's allocation strategies for
  140. * kernel objects.
  141. *
  142. * @param name Symbol name of the mutex
  143. */
  144. #define PTHREAD_MUTEX_DEFINE(name) \
  145. struct pthread_mutex name = \
  146. { \
  147. .lock_count = 0, \
  148. .wait_q = Z_WAIT_Q_INIT(&name.wait_q), \
  149. .owner = NULL, \
  150. }
  151. /*
  152. * Mutex attributes - type
  153. *
  154. * PTHREAD_MUTEX_NORMAL: Owner of mutex cannot relock it. Attempting
  155. * to relock will cause deadlock.
  156. * PTHREAD_MUTEX_RECURSIVE: Owner can relock the mutex.
  157. * PTHREAD_MUTEX_ERRORCHECK: If owner attempts to relock the mutex, an
  158. * error is returned.
  159. *
  160. */
  161. #define PTHREAD_MUTEX_NORMAL 0
  162. #define PTHREAD_MUTEX_RECURSIVE 1
  163. #define PTHREAD_MUTEX_ERRORCHECK 2
  164. #define PTHREAD_MUTEX_DEFAULT PTHREAD_MUTEX_NORMAL
  165. /*
  166. * Mutex attributes - protocol
  167. *
  168. * PTHREAD_PRIO_NONE: Ownership of mutex does not affect priority.
  169. * PTHREAD_PRIO_INHERIT: Owner's priority is boosted to the priority of
  170. * highest priority thread blocked on the mutex.
  171. * PTHREAD_PRIO_PROTECT: Mutex has a priority ceiling. The owner's
  172. * priority is boosted to the highest priority ceiling of all mutexes
  173. * owned (regardless of whether or not other threads are blocked on
  174. * any of these mutexes).
  175. * FIXME: Only PRIO_NONE is supported. Implement other protocols.
  176. */
  177. #define PTHREAD_PRIO_NONE 0
  178. /**
  179. * @brief POSIX threading compatibility API
  180. *
  181. * See IEEE 1003.1
  182. */
  183. int pthread_mutex_destroy(pthread_mutex_t *m);
  184. /**
  185. * @brief POSIX threading compatibility API
  186. *
  187. * See IEEE 1003.1
  188. */
  189. int pthread_mutex_lock(pthread_mutex_t *m);
  190. /**
  191. * @brief POSIX threading compatibility API
  192. *
  193. * See IEEE 1003.1
  194. */
  195. int pthread_mutex_unlock(pthread_mutex_t *m);
  196. /**
  197. * @brief POSIX threading compatibility API
  198. *
  199. * See IEEE 1003.1
  200. */
  201. int pthread_mutex_timedlock(pthread_mutex_t *m,
  202. const struct timespec *abstime);
  203. /**
  204. * @brief POSIX threading compatibility API
  205. *
  206. * See IEEE 1003.1
  207. */
  208. int pthread_mutex_trylock(pthread_mutex_t *m);
  209. /**
  210. * @brief POSIX threading compatibility API
  211. *
  212. * See IEEE 1003.1
  213. */
  214. int pthread_mutex_init(pthread_mutex_t *m,
  215. const pthread_mutexattr_t *att);
  216. /**
  217. * @brief POSIX threading compatibility API
  218. *
  219. * See IEEE 1003.1
  220. */
  221. int pthread_mutexattr_setprotocol(pthread_mutexattr_t *attr, int protocol);
  222. /**
  223. * @brief POSIX threading compatibility API
  224. *
  225. * See IEEE 1003.1
  226. */
  227. int pthread_mutexattr_settype(pthread_mutexattr_t *attr, int type);
  228. /**
  229. * @brief POSIX threading compatibility API
  230. *
  231. * See IEEE 1003.1
  232. */
  233. int pthread_mutexattr_getprotocol(const pthread_mutexattr_t *attr,
  234. int *protocol);
  235. /**
  236. * @brief POSIX threading compatibility API
  237. *
  238. * See IEEE 1003.1
  239. */
  240. int pthread_mutexattr_gettype(const pthread_mutexattr_t *attr, int *type);
  241. /**
  242. * @brief POSIX threading compatibility API
  243. *
  244. * See IEEE 1003.1
  245. *
  246. * Note that pthread attribute structs are currently noops in Zephyr.
  247. */
  248. static inline int pthread_mutexattr_init(pthread_mutexattr_t *m)
  249. {
  250. ARG_UNUSED(m);
  251. return 0;
  252. }
  253. /**
  254. * @brief POSIX threading compatibility API
  255. *
  256. * See IEEE 1003.1
  257. *
  258. * Note that pthread attribute structs are currently noops in Zephyr.
  259. */
  260. static inline int pthread_mutexattr_destroy(pthread_mutexattr_t *m)
  261. {
  262. ARG_UNUSED(m);
  263. return 0;
  264. }
  265. /* FIXME: these are going to be tricky to implement. Zephyr has (for
  266. * good reason) deprecated its own "initializer" macros in favor of a
  267. * static "declaration" macros instead. Using such a macro inside a
  268. * gcc compound expression to declare and object then reference it
  269. * would work, but gcc limits such expressions to function context
  270. * (because they may need to generate code that runs at assignment
  271. * time) and much real-world use of these initializers is for static
  272. * variables. The best trick I can think of would be to declare it in
  273. * a special section and then initialize that section at runtime
  274. * startup, which sort of defeats the purpose of having these be
  275. * static...
  276. *
  277. * Instead, see the nonstandard PTHREAD_*_DEFINE macros instead, which
  278. * work similarly but conform to Zephyr's paradigms.
  279. */
  280. /* #define PTHREAD_MUTEX_INITIALIZER */
  281. /* #define PTHREAD_COND_INITIALIZER */
  282. /**
  283. * @brief Declare a pthread barrier
  284. *
  285. * Declaration API for a pthread barrier. This is not a
  286. * POSIX API, it's provided to better conform with Zephyr's allocation
  287. * strategies for kernel objects.
  288. *
  289. * @param name Symbol name of the barrier
  290. * @param count Thread count, same as the "count" argument to
  291. * pthread_barrier_init()
  292. */
  293. #define PTHREAD_BARRIER_DEFINE(name, count) \
  294. struct pthread_barrier name = { \
  295. .wait_q = Z_WAIT_Q_INIT(&name.wait_q), \
  296. .max = count, \
  297. }
  298. #define PTHREAD_BARRIER_SERIAL_THREAD 1
  299. /**
  300. * @brief POSIX threading compatibility API
  301. *
  302. * See IEEE 1003.1
  303. */
  304. int pthread_barrier_wait(pthread_barrier_t *b);
  305. /**
  306. * @brief POSIX threading compatibility API
  307. *
  308. * See IEEE 1003.1
  309. */
  310. static inline int pthread_barrier_init(pthread_barrier_t *b,
  311. const pthread_barrierattr_t *attr,
  312. unsigned int count)
  313. {
  314. ARG_UNUSED(attr);
  315. b->max = count;
  316. b->count = 0;
  317. z_waitq_init(&b->wait_q);
  318. return 0;
  319. }
  320. /**
  321. * @brief POSIX threading compatibility API
  322. *
  323. * See IEEE 1003.1
  324. */
  325. static inline int pthread_barrier_destroy(pthread_barrier_t *b)
  326. {
  327. ARG_UNUSED(b);
  328. return 0;
  329. }
  330. /**
  331. * @brief POSIX threading compatibility API
  332. *
  333. * See IEEE 1003.1
  334. *
  335. * Note that pthread attribute structs are currently noops in Zephyr.
  336. */
  337. static inline int pthread_barrierattr_init(pthread_barrierattr_t *b)
  338. {
  339. ARG_UNUSED(b);
  340. return 0;
  341. }
  342. /**
  343. * @brief POSIX threading compatibility API
  344. *
  345. * See IEEE 1003.1
  346. *
  347. * Note that pthread attribute structs are currently noops in Zephyr.
  348. */
  349. static inline int pthread_barrierattr_destroy(pthread_barrierattr_t *b)
  350. {
  351. ARG_UNUSED(b);
  352. return 0;
  353. }
  354. /* Predicates and setters for various pthread attribute values that we
  355. * don't support (or always support: the "process shared" attribute
  356. * can only be true given the way Zephyr implements these
  357. * objects). Leave these undefined for simplicity instead of defining
  358. * stubs to return an error that would have to be logged and
  359. * interpreted just to figure out that we didn't support it in the
  360. * first place. These APIs are very rarely used even in production
  361. * Unix code. Leave the declarations here so they can be easily
  362. * uncommented and implemented as needed.
  363. int pthread_condattr_getclock(const pthread_condattr_t * clockid_t *);
  364. int pthread_condattr_getpshared(const pthread_condattr_t * int *);
  365. int pthread_condattr_setclock(pthread_condattr_t *, clockid_t);
  366. int pthread_condattr_setpshared(pthread_condattr_t *, int);
  367. int pthread_mutex_consistent(pthread_mutex_t *);
  368. int pthread_mutex_getprioceiling(const pthread_mutex_t * int *);
  369. int pthread_mutex_setprioceiling(pthread_mutex_t *, int int *);
  370. int pthread_mutexattr_getprioceiling(const pthread_mutexattr_t *, int *);
  371. int pthread_mutexattr_getpshared(const pthread_mutexattr_t * int *);
  372. int pthread_mutexattr_getrobust(const pthread_mutexattr_t * int *);
  373. int pthread_mutexattr_setprioceiling(pthread_mutexattr_t *, int);
  374. int pthread_mutexattr_setpshared(pthread_mutexattr_t *, int);
  375. int pthread_mutexattr_setrobust(pthread_mutexattr_t *, int);
  376. int pthread_barrierattr_getpshared(const pthread_barrierattr_t *, int *);
  377. int pthread_barrierattr_setpshared(pthread_barrierattr_t *, int);
  378. */
  379. /* Base Pthread related APIs */
  380. /**
  381. * @brief Obtain ID of the calling thread.
  382. *
  383. * The results of calling this API from threads not created with
  384. * pthread_create() are undefined.
  385. *
  386. * See IEEE 1003.1
  387. */
  388. static inline pthread_t pthread_self(void)
  389. {
  390. return (pthread_t)k_current_get();
  391. }
  392. /**
  393. * @brief Compare thread IDs.
  394. *
  395. * See IEEE 1003.1
  396. */
  397. static inline int pthread_equal(pthread_t pt1, pthread_t pt2)
  398. {
  399. return (pt1 == pt2);
  400. }
  401. /**
  402. * @brief Destroy the read-write lock attributes object.
  403. *
  404. * See IEEE 1003.1
  405. */
  406. static inline int pthread_rwlockattr_destroy(pthread_rwlockattr_t *attr)
  407. {
  408. return 0;
  409. }
  410. /**
  411. * @brief initialize the read-write lock attributes object.
  412. *
  413. * See IEEE 1003.1
  414. */
  415. static inline int pthread_rwlockattr_init(pthread_rwlockattr_t *attr)
  416. {
  417. return 0;
  418. }
  419. int pthread_attr_getstacksize(const pthread_attr_t *attr, size_t *stacksize);
  420. int pthread_attr_setschedpolicy(pthread_attr_t *attr, int policy);
  421. int pthread_attr_getschedpolicy(const pthread_attr_t *attr, int *policy);
  422. int pthread_attr_setdetachstate(pthread_attr_t *attr, int detachstate);
  423. int pthread_attr_getdetachstate(const pthread_attr_t *attr, int *detachstate);
  424. int pthread_attr_init(pthread_attr_t *attr);
  425. int pthread_attr_destroy(pthread_attr_t *attr);
  426. int pthread_attr_getschedparam(const pthread_attr_t *attr,
  427. struct sched_param *schedparam);
  428. int pthread_getschedparam(pthread_t pthread, int *policy,
  429. struct sched_param *param);
  430. int pthread_attr_getstack(const pthread_attr_t *attr,
  431. void **stackaddr, size_t *stacksize);
  432. int pthread_attr_setstack(pthread_attr_t *attr, void *stackaddr,
  433. size_t stacksize);
  434. int pthread_once(pthread_once_t *once, void (*initFunc)(void));
  435. void pthread_exit(void *retval);
  436. int pthread_join(pthread_t thread, void **status);
  437. int pthread_cancel(pthread_t pthread);
  438. int pthread_detach(pthread_t thread);
  439. int pthread_create(pthread_t *newthread, const pthread_attr_t *attr,
  440. void *(*threadroutine)(void *), void *arg);
  441. int pthread_setcancelstate(int state, int *oldstate);
  442. int pthread_attr_setschedparam(pthread_attr_t *attr,
  443. const struct sched_param *schedparam);
  444. int pthread_setschedparam(pthread_t pthread, int policy,
  445. const struct sched_param *param);
  446. int pthread_rwlock_destroy(pthread_rwlock_t *rwlock);
  447. int pthread_rwlock_init(pthread_rwlock_t *rwlock,
  448. const pthread_rwlockattr_t *attr);
  449. int pthread_rwlock_rdlock(pthread_rwlock_t *rwlock);
  450. int pthread_rwlock_timedrdlock(pthread_rwlock_t *rwlock,
  451. const struct timespec *abstime);
  452. int pthread_rwlock_timedwrlock(pthread_rwlock_t *rwlock,
  453. const struct timespec *abstime);
  454. int pthread_rwlock_tryrdlock(pthread_rwlock_t *rwlock);
  455. int pthread_rwlock_trywrlock(pthread_rwlock_t *rwlock);
  456. int pthread_rwlock_unlock(pthread_rwlock_t *rwlock);
  457. int pthread_rwlock_wrlock(pthread_rwlock_t *rwlock);
  458. int pthread_key_create(pthread_key_t *key,
  459. void (*destructor)(void *));
  460. int pthread_key_delete(pthread_key_t key);
  461. int pthread_setspecific(pthread_key_t key, const void *value);
  462. void *pthread_getspecific(pthread_key_t key);
  463. /* Glibc / Oracle Extension Functions */
  464. /**
  465. * @brief Set name of POSIX thread.
  466. *
  467. * Non-portable, extension function that conforms with most
  468. * other definitions of this function.
  469. *
  470. * @param thread POSIX thread to set name
  471. * @param name Name string
  472. * @retval 0 Success
  473. * @retval ESRCH Thread does not exist
  474. * @retval EINVAL Name buffer is NULL
  475. * @retval Negative value if kernel function error
  476. *
  477. */
  478. int pthread_setname_np(pthread_t thread, const char *name);
  479. /**
  480. * @brief Get name of POSIX thread and store in name buffer
  481. * that is of size len.
  482. *
  483. * Non-portable, extension function that conforms with most
  484. * other definitions of this function.
  485. *
  486. * @param thread POSIX thread to obtain name information
  487. * @param name Destination buffer
  488. * @param len Destination buffer size
  489. * @retval 0 Success
  490. * @retval ESRCH Thread does not exist
  491. * @retval EINVAL Name buffer is NULL
  492. * @retval Negative value if kernel function error
  493. */
  494. int pthread_getname_np(pthread_t thread, char *name, size_t len);
  495. #ifdef __cplusplus
  496. }
  497. #endif
  498. #endif /* ZEPHYR_INCLUDE_POSIX_PTHREAD_H_ */