123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645 |
- /*
- * Copyright (c) 2018 Intel Corporation
- *
- * SPDX-License-Identifier: Apache-2.0
- */
- #include <kernel.h>
- #include <stdio.h>
- #include <sys/atomic.h>
- #include <ksched.h>
- #include <wait_q.h>
- #include <posix/pthread.h>
- #include <sys/slist.h>
- #define PTHREAD_INIT_FLAGS PTHREAD_CANCEL_ENABLE
- #define PTHREAD_CANCELED ((void *) -1)
- #define LOWEST_POSIX_THREAD_PRIORITY 1
- PTHREAD_MUTEX_DEFINE(pthread_key_lock);
- static const pthread_attr_t init_pthread_attrs = {
- .priority = LOWEST_POSIX_THREAD_PRIORITY,
- .stack = NULL,
- .stacksize = 0,
- .flags = PTHREAD_INIT_FLAGS,
- .delayedstart = 0,
- #if defined(CONFIG_PREEMPT_ENABLED)
- .schedpolicy = SCHED_RR,
- #else
- .schedpolicy = SCHED_FIFO,
- #endif
- .detachstate = PTHREAD_CREATE_JOINABLE,
- .initialized = true,
- };
- static struct posix_thread posix_thread_pool[CONFIG_MAX_PTHREAD_COUNT];
- PTHREAD_MUTEX_DEFINE(pthread_pool_lock);
- static bool is_posix_prio_valid(uint32_t priority, int policy)
- {
- if (priority >= sched_get_priority_min(policy) &&
- priority <= sched_get_priority_max(policy)) {
- return true;
- }
- return false;
- }
- static uint32_t zephyr_to_posix_priority(int32_t z_prio, int *policy)
- {
- uint32_t prio;
- if (z_prio < 0) {
- *policy = SCHED_FIFO;
- prio = -1 * (z_prio + 1);
- } else {
- *policy = SCHED_RR;
- prio = (CONFIG_NUM_PREEMPT_PRIORITIES - z_prio);
- }
- return prio;
- }
- static int32_t posix_to_zephyr_priority(uint32_t priority, int policy)
- {
- int32_t prio;
- if (policy == SCHED_FIFO) {
- /* Zephyr COOP priority starts from -1 */
- prio = -1 * (priority + 1);
- } else {
- prio = (CONFIG_NUM_PREEMPT_PRIORITIES - priority);
- }
- return prio;
- }
- /**
- * @brief Set scheduling parameter attributes in thread attributes object.
- *
- * See IEEE 1003.1
- */
- int pthread_attr_setschedparam(pthread_attr_t *attr,
- const struct sched_param *schedparam)
- {
- int priority = schedparam->sched_priority;
- if ((attr == NULL) || (attr->initialized == 0U) ||
- (is_posix_prio_valid(priority, attr->schedpolicy) == false)) {
- return EINVAL;
- }
- attr->priority = priority;
- return 0;
- }
- /**
- * @brief Set stack attributes in thread attributes object.
- *
- * See IEEE 1003.1
- */
- int pthread_attr_setstack(pthread_attr_t *attr, void *stackaddr,
- size_t stacksize)
- {
- if (stackaddr == NULL) {
- return EACCES;
- }
- attr->stack = stackaddr;
- attr->stacksize = stacksize;
- return 0;
- }
- static void zephyr_thread_wrapper(void *arg1, void *arg2, void *arg3)
- {
- void * (*fun_ptr)(void *) = arg3;
- fun_ptr(arg1);
- pthread_exit(NULL);
- }
- /**
- * @brief Create a new thread.
- *
- * Pthread attribute should not be NULL. API will return Error on NULL
- * attribute value.
- *
- * See IEEE 1003.1
- */
- int pthread_create(pthread_t *newthread, const pthread_attr_t *attr,
- void *(*threadroutine)(void *), void *arg)
- {
- int32_t prio;
- uint32_t pthread_num;
- pthread_condattr_t cond_attr;
- struct posix_thread *thread;
- /*
- * FIXME: Pthread attribute must be non-null and it provides stack
- * pointer and stack size. So even though POSIX 1003.1 spec accepts
- * attrib as NULL but zephyr needs it initialized with valid stack.
- */
- if ((attr == NULL) || (attr->initialized == 0U)
- || (attr->stack == NULL) || (attr->stacksize == 0)) {
- return EINVAL;
- }
- pthread_mutex_lock(&pthread_pool_lock);
- for (pthread_num = 0;
- pthread_num < CONFIG_MAX_PTHREAD_COUNT; pthread_num++) {
- thread = &posix_thread_pool[pthread_num];
- if (thread->state == PTHREAD_TERMINATED) {
- thread->state = PTHREAD_JOINABLE;
- break;
- }
- }
- pthread_mutex_unlock(&pthread_pool_lock);
- if (pthread_num >= CONFIG_MAX_PTHREAD_COUNT) {
- return EAGAIN;
- }
- prio = posix_to_zephyr_priority(attr->priority, attr->schedpolicy);
- thread = &posix_thread_pool[pthread_num];
- /*
- * Ignore return value, as we know that Zephyr implementation
- * cannot fail.
- */
- (void)pthread_mutex_init(&thread->state_lock, NULL);
- (void)pthread_mutex_init(&thread->cancel_lock, NULL);
- pthread_mutex_lock(&thread->cancel_lock);
- thread->cancel_state = (1 << _PTHREAD_CANCEL_POS) & attr->flags;
- thread->cancel_pending = 0;
- pthread_mutex_unlock(&thread->cancel_lock);
- pthread_mutex_lock(&thread->state_lock);
- thread->state = attr->detachstate;
- pthread_mutex_unlock(&thread->state_lock);
- pthread_cond_init(&thread->state_cond, &cond_attr);
- sys_slist_init(&thread->key_list);
- *newthread = (pthread_t) k_thread_create(&thread->thread, attr->stack,
- attr->stacksize,
- (k_thread_entry_t)
- zephyr_thread_wrapper,
- (void *)arg, NULL,
- threadroutine, prio,
- (~K_ESSENTIAL & attr->flags),
- K_MSEC(attr->delayedstart));
- return 0;
- }
- /**
- * @brief Set cancelability State.
- *
- * See IEEE 1003.1
- */
- int pthread_setcancelstate(int state, int *oldstate)
- {
- struct posix_thread *pthread = (struct posix_thread *) pthread_self();
- if (state != PTHREAD_CANCEL_ENABLE &&
- state != PTHREAD_CANCEL_DISABLE) {
- return EINVAL;
- }
- *oldstate = pthread->cancel_state;
- pthread_mutex_lock(&pthread->cancel_lock);
- pthread->cancel_state = state;
- pthread_mutex_unlock(&pthread->cancel_lock);
- if (state == PTHREAD_CANCEL_ENABLE && pthread->cancel_pending) {
- pthread_exit((void *)PTHREAD_CANCELED);
- }
- return 0;
- }
- /**
- * @brief Cancel execution of a thread.
- *
- * See IEEE 1003.1
- */
- int pthread_cancel(pthread_t pthread)
- {
- struct posix_thread *thread = (struct posix_thread *) pthread;
- int cancel_state;
- if ((thread == NULL) || (thread->state == PTHREAD_TERMINATED)) {
- return ESRCH;
- }
- pthread_mutex_lock(&thread->cancel_lock);
- thread->cancel_pending = 1;
- cancel_state = thread->cancel_state;
- pthread_mutex_unlock(&thread->cancel_lock);
- if (cancel_state == PTHREAD_CANCEL_ENABLE) {
- pthread_mutex_lock(&thread->state_lock);
- if (thread->state == PTHREAD_DETACHED) {
- thread->state = PTHREAD_TERMINATED;
- } else {
- thread->retval = PTHREAD_CANCELED;
- thread->state = PTHREAD_EXITED;
- pthread_cond_broadcast(&thread->state_cond);
- }
- pthread_mutex_unlock(&thread->state_lock);
- k_thread_abort((k_tid_t) thread);
- }
- return 0;
- }
- /**
- * @brief Set thread scheduling policy and parameters.
- *
- * See IEEE 1003.1
- */
- int pthread_setschedparam(pthread_t pthread, int policy,
- const struct sched_param *param)
- {
- k_tid_t thread = (k_tid_t)pthread;
- int new_prio;
- if (thread == NULL) {
- return ESRCH;
- }
- if (policy != SCHED_RR && policy != SCHED_FIFO) {
- return EINVAL;
- }
- if (is_posix_prio_valid(param->sched_priority, policy) == false) {
- return EINVAL;
- }
- new_prio = posix_to_zephyr_priority(param->sched_priority, policy);
- k_thread_priority_set(thread, new_prio);
- return 0;
- }
- /**
- * @brief Initialise threads attribute object
- *
- * See IEEE 1003.1
- */
- int pthread_attr_init(pthread_attr_t *attr)
- {
- if (attr == NULL) {
- return ENOMEM;
- }
- (void)memcpy(attr, &init_pthread_attrs, sizeof(pthread_attr_t));
- return 0;
- }
- /**
- * @brief Get thread scheduling policy and parameters
- *
- * See IEEE 1003.1
- */
- int pthread_getschedparam(pthread_t pthread, int *policy,
- struct sched_param *param)
- {
- struct posix_thread *thread = (struct posix_thread *) pthread;
- uint32_t priority;
- if ((thread == NULL) || (thread->state == PTHREAD_TERMINATED)) {
- return ESRCH;
- }
- priority = k_thread_priority_get((k_tid_t) thread);
- param->sched_priority = zephyr_to_posix_priority(priority, policy);
- return 0;
- }
- /**
- * @brief Dynamic package initialization
- *
- * See IEEE 1003.1
- */
- int pthread_once(pthread_once_t *once, void (*init_func)(void))
- {
- pthread_mutex_lock(&pthread_key_lock);
- if (*once == PTHREAD_ONCE_INIT) {
- pthread_mutex_unlock(&pthread_key_lock);
- return 0;
- }
- init_func();
- *once = PTHREAD_ONCE_INIT;
- pthread_mutex_unlock(&pthread_key_lock);
- return 0;
- }
- /**
- * @brief Terminate calling thread.
- *
- * See IEEE 1003.1
- */
- void pthread_exit(void *retval)
- {
- struct posix_thread *self = (struct posix_thread *)pthread_self();
- pthread_key_obj *key_obj;
- pthread_thread_data *thread_spec_data;
- sys_snode_t *node_l;
- /* Make a thread as cancelable before exiting */
- pthread_mutex_lock(&self->cancel_lock);
- if (self->cancel_state == PTHREAD_CANCEL_DISABLE) {
- self->cancel_state = PTHREAD_CANCEL_ENABLE;
- }
- pthread_mutex_unlock(&self->cancel_lock);
- pthread_mutex_lock(&self->state_lock);
- if (self->state == PTHREAD_JOINABLE) {
- self->retval = retval;
- self->state = PTHREAD_EXITED;
- self->retval = retval;
- pthread_cond_broadcast(&self->state_cond);
- } else {
- self->state = PTHREAD_TERMINATED;
- }
- SYS_SLIST_FOR_EACH_NODE(&self->key_list, node_l) {
- thread_spec_data = (pthread_thread_data *)node_l;
- if (thread_spec_data != NULL) {
- key_obj = thread_spec_data->key;
- if (key_obj->destructor != NULL) {
- (key_obj->destructor)(thread_spec_data->spec_data);
- }
- }
- }
- pthread_mutex_unlock(&self->state_lock);
- k_thread_abort((k_tid_t)self);
- }
- /**
- * @brief Wait for a thread termination.
- *
- * See IEEE 1003.1
- */
- int pthread_join(pthread_t thread, void **status)
- {
- struct posix_thread *pthread = (struct posix_thread *) thread;
- int ret = 0;
- if (pthread == NULL) {
- return ESRCH;
- }
- if (pthread == pthread_self()) {
- return EDEADLK;
- }
- pthread_mutex_lock(&pthread->state_lock);
- if (pthread->state == PTHREAD_JOINABLE) {
- pthread_cond_wait(&pthread->state_cond, &pthread->state_lock);
- }
- if (pthread->state == PTHREAD_EXITED) {
- if (status != NULL) {
- *status = pthread->retval;
- }
- } else if (pthread->state == PTHREAD_DETACHED) {
- ret = EINVAL;
- } else {
- ret = ESRCH;
- }
- pthread_mutex_unlock(&pthread->state_lock);
- return ret;
- }
- /**
- * @brief Detach a thread.
- *
- * See IEEE 1003.1
- */
- int pthread_detach(pthread_t thread)
- {
- struct posix_thread *pthread = (struct posix_thread *) thread;
- int ret = 0;
- if (pthread == NULL) {
- return ESRCH;
- }
- pthread_mutex_lock(&pthread->state_lock);
- switch (pthread->state) {
- case PTHREAD_JOINABLE:
- pthread->state = PTHREAD_DETACHED;
- /* Broadcast the condition.
- * This will make threads waiting to join this thread continue.
- */
- pthread_cond_broadcast(&pthread->state_cond);
- break;
- case PTHREAD_EXITED:
- pthread->state = PTHREAD_TERMINATED;
- /* THREAD has already exited.
- * Pthread remained to provide exit status.
- */
- break;
- case PTHREAD_TERMINATED:
- ret = ESRCH;
- break;
- default:
- ret = EINVAL;
- break;
- }
- pthread_mutex_unlock(&pthread->state_lock);
- return ret;
- }
- /**
- * @brief Get detach state attribute in thread attributes object.
- *
- * See IEEE 1003.1
- */
- int pthread_attr_getdetachstate(const pthread_attr_t *attr, int *detachstate)
- {
- if ((attr == NULL) || (attr->initialized == 0U)) {
- return EINVAL;
- }
- *detachstate = attr->detachstate;
- return 0;
- }
- /**
- * @brief Set detach state attribute in thread attributes object.
- *
- * See IEEE 1003.1
- */
- int pthread_attr_setdetachstate(pthread_attr_t *attr, int detachstate)
- {
- if ((attr == NULL) || (attr->initialized == 0U) ||
- (detachstate != PTHREAD_CREATE_DETACHED &&
- detachstate != PTHREAD_CREATE_JOINABLE)) {
- return EINVAL;
- }
- attr->detachstate = detachstate;
- return 0;
- }
- /**
- * @brief Get scheduling policy attribute in Thread attributes.
- *
- * See IEEE 1003.1
- */
- int pthread_attr_getschedpolicy(const pthread_attr_t *attr, int *policy)
- {
- if ((attr == NULL) || (attr->initialized == 0U)) {
- return EINVAL;
- }
- *policy = attr->schedpolicy;
- return 0;
- }
- /**
- * @brief Set scheduling policy attribute in Thread attributes object.
- *
- * See IEEE 1003.1
- */
- int pthread_attr_setschedpolicy(pthread_attr_t *attr, int policy)
- {
- if ((attr == NULL) || (attr->initialized == 0U) ||
- (policy != SCHED_RR && policy != SCHED_FIFO)) {
- return EINVAL;
- }
- attr->schedpolicy = policy;
- return 0;
- }
- /**
- * @brief Get stack size attribute in thread attributes object.
- *
- * See IEEE 1003.1
- */
- int pthread_attr_getstacksize(const pthread_attr_t *attr, size_t *stacksize)
- {
- if ((attr == NULL) || (attr->initialized == 0U)) {
- return EINVAL;
- }
- *stacksize = attr->stacksize;
- return 0;
- }
- /**
- * @brief Get stack attributes in thread attributes object.
- *
- * See IEEE 1003.1
- */
- int pthread_attr_getstack(const pthread_attr_t *attr,
- void **stackaddr, size_t *stacksize)
- {
- if ((attr == NULL) || (attr->initialized == 0U)) {
- return EINVAL;
- }
- *stackaddr = attr->stack;
- *stacksize = attr->stacksize;
- return 0;
- }
- /**
- * @brief Get thread attributes object scheduling parameters.
- *
- * See IEEE 1003.1
- */
- int pthread_attr_getschedparam(const pthread_attr_t *attr,
- struct sched_param *schedparam)
- {
- if ((attr == NULL) || (attr->initialized == 0U)) {
- return EINVAL;
- }
- schedparam->sched_priority = attr->priority;
- return 0;
- }
- /**
- * @brief Destroy thread attributes object.
- *
- * See IEEE 1003.1
- */
- int pthread_attr_destroy(pthread_attr_t *attr)
- {
- if ((attr != NULL) && (attr->initialized != 0U)) {
- attr->initialized = false;
- return 0;
- }
- return EINVAL;
- }
- int pthread_setname_np(pthread_t thread, const char *name)
- {
- #ifdef CONFIG_THREAD_NAME
- k_tid_t kthread = (k_tid_t)thread;
- if (kthread == NULL) {
- return ESRCH;
- }
- if (name == NULL) {
- return EINVAL;
- }
- return k_thread_name_set(kthread, name);
- #else
- ARG_UNUSED(thread);
- ARG_UNUSED(name);
- return 0;
- #endif
- }
- int pthread_getname_np(pthread_t thread, char *name, size_t len)
- {
- #ifdef CONFIG_THREAD_NAME
- k_tid_t kthread = (k_tid_t)thread;
- if (kthread == NULL) {
- return ESRCH;
- }
- if (name == NULL) {
- return EINVAL;
- }
- memset(name, '\0', len);
- return k_thread_name_copy(kthread, name, len-1);
- #else
- ARG_UNUSED(thread);
- ARG_UNUSED(name);
- ARG_UNUSED(len);
- return 0;
- #endif
- }
|