pthread.h 15 KB

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