pthread.c 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668
  1. /*
  2. * Copyright (c) 2018 Intel Corporation
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <kernel.h>
  7. #include <stdio.h>
  8. #include <sys/atomic.h>
  9. #include <ksched.h>
  10. #include <wait_q.h>
  11. #include <posix/pthread.h>
  12. #include <sys/slist.h>
  13. #define PTHREAD_INIT_FLAGS PTHREAD_CANCEL_ENABLE
  14. #define PTHREAD_CANCELED ((void *) -1)
  15. #define LOWEST_POSIX_THREAD_PRIORITY 1
  16. PTHREAD_MUTEX_DEFINE(pthread_key_lock);
  17. static const pthread_attr_t init_pthread_attrs = {
  18. .priority = LOWEST_POSIX_THREAD_PRIORITY,
  19. .stack = NULL,
  20. .stacksize = 0,
  21. .flags = PTHREAD_INIT_FLAGS,
  22. .delayedstart = 0,
  23. #if defined(CONFIG_PREEMPT_ENABLED)
  24. .schedpolicy = SCHED_RR,
  25. #else
  26. .schedpolicy = SCHED_FIFO,
  27. #endif
  28. .detachstate = PTHREAD_CREATE_JOINABLE,
  29. .initialized = true,
  30. };
  31. static struct posix_thread posix_thread_pool[CONFIG_MAX_PTHREAD_COUNT];
  32. PTHREAD_MUTEX_DEFINE(pthread_pool_lock);
  33. static bool is_posix_prio_valid(uint32_t priority, int policy)
  34. {
  35. if (priority >= sched_get_priority_min(policy) &&
  36. priority <= sched_get_priority_max(policy)) {
  37. return true;
  38. }
  39. return false;
  40. }
  41. static uint32_t zephyr_to_posix_priority(int32_t z_prio, int *policy)
  42. {
  43. uint32_t prio;
  44. if (z_prio < 0) {
  45. *policy = SCHED_FIFO;
  46. prio = -1 * (z_prio + 1);
  47. } else {
  48. *policy = SCHED_RR;
  49. prio = (CONFIG_NUM_PREEMPT_PRIORITIES - z_prio);
  50. }
  51. return prio;
  52. }
  53. static int32_t posix_to_zephyr_priority(uint32_t priority, int policy)
  54. {
  55. int32_t prio;
  56. if (policy == SCHED_FIFO) {
  57. /* Zephyr COOP priority starts from -1 */
  58. prio = -1 * (priority + 1);
  59. } else {
  60. prio = (CONFIG_NUM_PREEMPT_PRIORITIES - priority);
  61. }
  62. return prio;
  63. }
  64. /**
  65. * @brief Set scheduling parameter attributes in thread attributes object.
  66. *
  67. * See IEEE 1003.1
  68. */
  69. int pthread_attr_setschedparam(pthread_attr_t *attr,
  70. const struct sched_param *schedparam)
  71. {
  72. int priority = schedparam->sched_priority;
  73. if ((attr == NULL) || (attr->initialized == 0U) ||
  74. (is_posix_prio_valid(priority, attr->schedpolicy) == false)) {
  75. return EINVAL;
  76. }
  77. attr->priority = priority;
  78. return 0;
  79. }
  80. /**
  81. * @brief Set stack attributes in thread attributes object.
  82. *
  83. * See IEEE 1003.1
  84. */
  85. int pthread_attr_setstack(pthread_attr_t *attr, void *stackaddr,
  86. size_t stacksize)
  87. {
  88. if (stackaddr == NULL) {
  89. return EACCES;
  90. }
  91. attr->stack = stackaddr;
  92. attr->stacksize = stacksize;
  93. return 0;
  94. }
  95. int pthread_attr_setstacksize(pthread_attr_t *attr, size_t stacksize)
  96. {
  97. attr->stacksize = stacksize;
  98. return 0;
  99. }
  100. static void zephyr_thread_wrapper(void *arg1, void *arg2, void *arg3)
  101. {
  102. void * (*fun_ptr)(void *) = arg3;
  103. fun_ptr(arg1);
  104. pthread_exit(NULL);
  105. }
  106. /**
  107. * @brief Create a new thread.
  108. *
  109. * Pthread attribute should not be NULL. API will return Error on NULL
  110. * attribute value.
  111. *
  112. * See IEEE 1003.1
  113. */
  114. int pthread_create(pthread_t *newthread, const pthread_attr_t *attr,
  115. void *(*threadroutine)(void *), void *arg)
  116. {
  117. int32_t prio;
  118. uint32_t pthread_num;
  119. pthread_condattr_t cond_attr;
  120. struct posix_thread *thread;
  121. void *stack;
  122. /*
  123. * FIXME: Pthread attribute must be non-null and it provides stack
  124. * pointer and stack size. So even though POSIX 1003.1 spec accepts
  125. * attrib as NULL but zephyr needs it initialized with valid stack.
  126. */
  127. if ((attr == NULL) || (attr->initialized == 0U)
  128. || (attr->stacksize == 0)) {
  129. return EINVAL;
  130. }
  131. pthread_mutex_lock(&pthread_pool_lock);
  132. for (pthread_num = 0;
  133. pthread_num < CONFIG_MAX_PTHREAD_COUNT; pthread_num++) {
  134. thread = &posix_thread_pool[pthread_num];
  135. if (thread->state == PTHREAD_TERMINATED) {
  136. thread->state = PTHREAD_JOINABLE;
  137. break;
  138. }
  139. }
  140. pthread_mutex_unlock(&pthread_pool_lock);
  141. if (pthread_num >= CONFIG_MAX_PTHREAD_COUNT) {
  142. return EAGAIN;
  143. }
  144. prio = posix_to_zephyr_priority(attr->priority, attr->schedpolicy);
  145. thread = &posix_thread_pool[pthread_num];
  146. /*
  147. * Ignore return value, as we know that Zephyr implementation
  148. * cannot fail.
  149. */
  150. (void)pthread_mutex_init(&thread->state_lock, NULL);
  151. (void)pthread_mutex_init(&thread->cancel_lock, NULL);
  152. pthread_mutex_lock(&thread->cancel_lock);
  153. thread->cancel_state = (1 << _PTHREAD_CANCEL_POS) & attr->flags;
  154. thread->cancel_pending = 0;
  155. pthread_mutex_unlock(&thread->cancel_lock);
  156. pthread_mutex_lock(&thread->state_lock);
  157. thread->state = attr->detachstate;
  158. pthread_mutex_unlock(&thread->state_lock);
  159. pthread_cond_init(&thread->state_cond, &cond_attr);
  160. sys_slist_init(&thread->key_list);
  161. if (attr->stack == NULL) {
  162. stack = k_aligned_alloc(ARCH_STACK_PTR_ALIGN, attr->stacksize);
  163. if (!stack) {
  164. return ENOMEM;
  165. }
  166. thread->dynamic_stack = stack;
  167. } else {
  168. stack = attr->stack;
  169. thread->dynamic_stack = NULL;
  170. }
  171. *newthread = (pthread_t) k_thread_create(&thread->thread, stack,
  172. attr->stacksize,
  173. (k_thread_entry_t)
  174. zephyr_thread_wrapper,
  175. (void *)arg, NULL,
  176. threadroutine, prio,
  177. (~K_ESSENTIAL & attr->flags),
  178. K_MSEC(attr->delayedstart));
  179. return 0;
  180. }
  181. /**
  182. * @brief Set cancelability State.
  183. *
  184. * See IEEE 1003.1
  185. */
  186. int pthread_setcancelstate(int state, int *oldstate)
  187. {
  188. struct posix_thread *pthread = (struct posix_thread *) pthread_self();
  189. if (state != PTHREAD_CANCEL_ENABLE &&
  190. state != PTHREAD_CANCEL_DISABLE) {
  191. return EINVAL;
  192. }
  193. *oldstate = pthread->cancel_state;
  194. pthread_mutex_lock(&pthread->cancel_lock);
  195. pthread->cancel_state = state;
  196. pthread_mutex_unlock(&pthread->cancel_lock);
  197. if (state == PTHREAD_CANCEL_ENABLE && pthread->cancel_pending) {
  198. pthread_exit((void *)PTHREAD_CANCELED);
  199. }
  200. return 0;
  201. }
  202. /**
  203. * @brief Cancel execution of a thread.
  204. *
  205. * See IEEE 1003.1
  206. */
  207. int pthread_cancel(pthread_t pthread)
  208. {
  209. struct posix_thread *thread = (struct posix_thread *) pthread;
  210. int cancel_state;
  211. if ((thread == NULL) || (thread->state == PTHREAD_TERMINATED)) {
  212. return ESRCH;
  213. }
  214. pthread_mutex_lock(&thread->cancel_lock);
  215. thread->cancel_pending = 1;
  216. cancel_state = thread->cancel_state;
  217. pthread_mutex_unlock(&thread->cancel_lock);
  218. if (cancel_state == PTHREAD_CANCEL_ENABLE) {
  219. pthread_mutex_lock(&thread->state_lock);
  220. if (thread->state == PTHREAD_DETACHED) {
  221. thread->state = PTHREAD_TERMINATED;
  222. } else {
  223. thread->retval = PTHREAD_CANCELED;
  224. thread->state = PTHREAD_EXITED;
  225. pthread_cond_broadcast(&thread->state_cond);
  226. }
  227. pthread_mutex_unlock(&thread->state_lock);
  228. k_thread_abort((k_tid_t) thread);
  229. }
  230. return 0;
  231. }
  232. /**
  233. * @brief Set thread scheduling policy and parameters.
  234. *
  235. * See IEEE 1003.1
  236. */
  237. int pthread_setschedparam(pthread_t pthread, int policy,
  238. const struct sched_param *param)
  239. {
  240. k_tid_t thread = (k_tid_t)pthread;
  241. int new_prio;
  242. if (thread == NULL) {
  243. return ESRCH;
  244. }
  245. if (policy != SCHED_RR && policy != SCHED_FIFO) {
  246. return EINVAL;
  247. }
  248. if (is_posix_prio_valid(param->sched_priority, policy) == false) {
  249. return EINVAL;
  250. }
  251. new_prio = posix_to_zephyr_priority(param->sched_priority, policy);
  252. k_thread_priority_set(thread, new_prio);
  253. return 0;
  254. }
  255. /**
  256. * @brief Initialise threads attribute object
  257. *
  258. * See IEEE 1003.1
  259. */
  260. int pthread_attr_init(pthread_attr_t *attr)
  261. {
  262. if (attr == NULL) {
  263. return ENOMEM;
  264. }
  265. (void)memcpy(attr, &init_pthread_attrs, sizeof(pthread_attr_t));
  266. return 0;
  267. }
  268. /**
  269. * @brief Get thread scheduling policy and parameters
  270. *
  271. * See IEEE 1003.1
  272. */
  273. int pthread_getschedparam(pthread_t pthread, int *policy,
  274. struct sched_param *param)
  275. {
  276. struct posix_thread *thread = (struct posix_thread *) pthread;
  277. uint32_t priority;
  278. if ((thread == NULL) || (thread->state == PTHREAD_TERMINATED)) {
  279. return ESRCH;
  280. }
  281. priority = k_thread_priority_get((k_tid_t) thread);
  282. param->sched_priority = zephyr_to_posix_priority(priority, policy);
  283. return 0;
  284. }
  285. /**
  286. * @brief Dynamic package initialization
  287. *
  288. * See IEEE 1003.1
  289. */
  290. int pthread_once(pthread_once_t *once, void (*init_func)(void))
  291. {
  292. pthread_mutex_lock(&pthread_key_lock);
  293. if (*once == PTHREAD_ONCE_INIT) {
  294. pthread_mutex_unlock(&pthread_key_lock);
  295. return 0;
  296. }
  297. init_func();
  298. *once = PTHREAD_ONCE_INIT;
  299. pthread_mutex_unlock(&pthread_key_lock);
  300. return 0;
  301. }
  302. /**
  303. * @brief Terminate calling thread.
  304. *
  305. * See IEEE 1003.1
  306. */
  307. void pthread_exit(void *retval)
  308. {
  309. struct posix_thread *self = (struct posix_thread *)pthread_self();
  310. pthread_key_obj *key_obj;
  311. pthread_thread_data *thread_spec_data;
  312. sys_snode_t *node_l;
  313. /* Make a thread as cancelable before exiting */
  314. pthread_mutex_lock(&self->cancel_lock);
  315. if (self->cancel_state == PTHREAD_CANCEL_DISABLE) {
  316. self->cancel_state = PTHREAD_CANCEL_ENABLE;
  317. }
  318. pthread_mutex_unlock(&self->cancel_lock);
  319. pthread_mutex_lock(&self->state_lock);
  320. if (self->state == PTHREAD_JOINABLE) {
  321. self->retval = retval;
  322. self->state = PTHREAD_EXITED;
  323. self->retval = retval;
  324. pthread_cond_broadcast(&self->state_cond);
  325. } else {
  326. self->state = PTHREAD_TERMINATED;
  327. }
  328. SYS_SLIST_FOR_EACH_NODE(&self->key_list, node_l) {
  329. thread_spec_data = (pthread_thread_data *)node_l;
  330. if (thread_spec_data != NULL) {
  331. key_obj = thread_spec_data->key;
  332. if (key_obj->destructor != NULL) {
  333. (key_obj->destructor)(thread_spec_data->spec_data);
  334. }
  335. }
  336. }
  337. if (self->dynamic_stack) {
  338. k_free(self->dynamic_stack);
  339. self->dynamic_stack = NULL;
  340. }
  341. pthread_mutex_unlock(&self->state_lock);
  342. k_thread_abort((k_tid_t)self);
  343. }
  344. /**
  345. * @brief Wait for a thread termination.
  346. *
  347. * See IEEE 1003.1
  348. */
  349. int pthread_join(pthread_t thread, void **status)
  350. {
  351. struct posix_thread *pthread = (struct posix_thread *) thread;
  352. int ret = 0;
  353. if (pthread == NULL) {
  354. return ESRCH;
  355. }
  356. if (pthread == pthread_self()) {
  357. return EDEADLK;
  358. }
  359. pthread_mutex_lock(&pthread->state_lock);
  360. if (pthread->state == PTHREAD_JOINABLE) {
  361. pthread_cond_wait(&pthread->state_cond, &pthread->state_lock);
  362. }
  363. if (pthread->state == PTHREAD_EXITED) {
  364. if (status != NULL) {
  365. *status = pthread->retval;
  366. }
  367. } else if (pthread->state == PTHREAD_DETACHED) {
  368. ret = EINVAL;
  369. } else {
  370. ret = ESRCH;
  371. }
  372. pthread_mutex_unlock(&pthread->state_lock);
  373. return ret;
  374. }
  375. /**
  376. * @brief Detach a thread.
  377. *
  378. * See IEEE 1003.1
  379. */
  380. int pthread_detach(pthread_t thread)
  381. {
  382. struct posix_thread *pthread = (struct posix_thread *) thread;
  383. int ret = 0;
  384. if (pthread == NULL) {
  385. return ESRCH;
  386. }
  387. pthread_mutex_lock(&pthread->state_lock);
  388. switch (pthread->state) {
  389. case PTHREAD_JOINABLE:
  390. pthread->state = PTHREAD_DETACHED;
  391. /* Broadcast the condition.
  392. * This will make threads waiting to join this thread continue.
  393. */
  394. pthread_cond_broadcast(&pthread->state_cond);
  395. break;
  396. case PTHREAD_EXITED:
  397. pthread->state = PTHREAD_TERMINATED;
  398. /* THREAD has already exited.
  399. * Pthread remained to provide exit status.
  400. */
  401. break;
  402. case PTHREAD_TERMINATED:
  403. ret = ESRCH;
  404. break;
  405. default:
  406. ret = EINVAL;
  407. break;
  408. }
  409. pthread_mutex_unlock(&pthread->state_lock);
  410. return ret;
  411. }
  412. /**
  413. * @brief Get detach state attribute in thread attributes object.
  414. *
  415. * See IEEE 1003.1
  416. */
  417. int pthread_attr_getdetachstate(const pthread_attr_t *attr, int *detachstate)
  418. {
  419. if ((attr == NULL) || (attr->initialized == 0U)) {
  420. return EINVAL;
  421. }
  422. *detachstate = attr->detachstate;
  423. return 0;
  424. }
  425. /**
  426. * @brief Set detach state attribute in thread attributes object.
  427. *
  428. * See IEEE 1003.1
  429. */
  430. int pthread_attr_setdetachstate(pthread_attr_t *attr, int detachstate)
  431. {
  432. if ((attr == NULL) || (attr->initialized == 0U) ||
  433. (detachstate != PTHREAD_CREATE_DETACHED &&
  434. detachstate != PTHREAD_CREATE_JOINABLE)) {
  435. return EINVAL;
  436. }
  437. attr->detachstate = detachstate;
  438. return 0;
  439. }
  440. /**
  441. * @brief Get scheduling policy attribute in Thread attributes.
  442. *
  443. * See IEEE 1003.1
  444. */
  445. int pthread_attr_getschedpolicy(const pthread_attr_t *attr, int *policy)
  446. {
  447. if ((attr == NULL) || (attr->initialized == 0U)) {
  448. return EINVAL;
  449. }
  450. *policy = attr->schedpolicy;
  451. return 0;
  452. }
  453. /**
  454. * @brief Set scheduling policy attribute in Thread attributes object.
  455. *
  456. * See IEEE 1003.1
  457. */
  458. int pthread_attr_setschedpolicy(pthread_attr_t *attr, int policy)
  459. {
  460. if ((attr == NULL) || (attr->initialized == 0U) ||
  461. (policy != SCHED_RR && policy != SCHED_FIFO)) {
  462. return EINVAL;
  463. }
  464. attr->schedpolicy = policy;
  465. return 0;
  466. }
  467. /**
  468. * @brief Get stack size attribute in thread attributes object.
  469. *
  470. * See IEEE 1003.1
  471. */
  472. int pthread_attr_getstacksize(const pthread_attr_t *attr, size_t *stacksize)
  473. {
  474. if ((attr == NULL) || (attr->initialized == 0U)) {
  475. return EINVAL;
  476. }
  477. *stacksize = attr->stacksize;
  478. return 0;
  479. }
  480. /**
  481. * @brief Get stack attributes in thread attributes object.
  482. *
  483. * See IEEE 1003.1
  484. */
  485. int pthread_attr_getstack(const pthread_attr_t *attr,
  486. void **stackaddr, size_t *stacksize)
  487. {
  488. if ((attr == NULL) || (attr->initialized == 0U)) {
  489. return EINVAL;
  490. }
  491. *stackaddr = attr->stack;
  492. *stacksize = attr->stacksize;
  493. return 0;
  494. }
  495. /**
  496. * @brief Get thread attributes object scheduling parameters.
  497. *
  498. * See IEEE 1003.1
  499. */
  500. int pthread_attr_getschedparam(const pthread_attr_t *attr,
  501. struct sched_param *schedparam)
  502. {
  503. if ((attr == NULL) || (attr->initialized == 0U)) {
  504. return EINVAL;
  505. }
  506. schedparam->sched_priority = attr->priority;
  507. return 0;
  508. }
  509. /**
  510. * @brief Destroy thread attributes object.
  511. *
  512. * See IEEE 1003.1
  513. */
  514. int pthread_attr_destroy(pthread_attr_t *attr)
  515. {
  516. if ((attr != NULL) && (attr->initialized != 0U)) {
  517. attr->initialized = false;
  518. return 0;
  519. }
  520. return EINVAL;
  521. }
  522. int pthread_setname_np(pthread_t thread, const char *name)
  523. {
  524. #ifdef CONFIG_THREAD_NAME
  525. k_tid_t kthread = (k_tid_t)thread;
  526. if (kthread == NULL) {
  527. return ESRCH;
  528. }
  529. if (name == NULL) {
  530. return EINVAL;
  531. }
  532. return k_thread_name_set(kthread, name);
  533. #else
  534. ARG_UNUSED(thread);
  535. ARG_UNUSED(name);
  536. return 0;
  537. #endif
  538. }
  539. int pthread_getname_np(pthread_t thread, char *name, size_t len)
  540. {
  541. #ifdef CONFIG_THREAD_NAME
  542. k_tid_t kthread = (k_tid_t)thread;
  543. if (kthread == NULL) {
  544. return ESRCH;
  545. }
  546. if (name == NULL) {
  547. return EINVAL;
  548. }
  549. memset(name, '\0', len);
  550. return k_thread_name_copy(kthread, name, len-1);
  551. #else
  552. ARG_UNUSED(thread);
  553. ARG_UNUSED(name);
  554. ARG_UNUSED(len);
  555. return 0;
  556. #endif
  557. }