123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243 |
- /*
- * Copyright (c) 2017 Intel Corporation
- *
- * SPDX-License-Identifier: Apache-2.0
- */
- #include <kernel.h>
- #include <ksched.h>
- #include <wait_q.h>
- #include <posix/pthread.h>
- struct k_spinlock z_pthread_spinlock;
- int64_t timespec_to_timeoutms(const struct timespec *abstime);
- #define MUTEX_MAX_REC_LOCK 32767
- /*
- * Default mutex attrs.
- */
- static const pthread_mutexattr_t def_attr = {
- .type = PTHREAD_MUTEX_DEFAULT,
- };
- static int acquire_mutex(pthread_mutex_t *m, k_timeout_t timeout)
- {
- int rc = 0;
- k_spinlock_key_t key = k_spin_lock(&z_pthread_spinlock);
- if (m->lock_count == 0U && m->owner == NULL) {
- m->lock_count++;
- m->owner = pthread_self();
- k_spin_unlock(&z_pthread_spinlock, key);
- return 0;
- } else if (m->owner == pthread_self()) {
- if (m->type == PTHREAD_MUTEX_RECURSIVE &&
- m->lock_count < MUTEX_MAX_REC_LOCK) {
- m->lock_count++;
- rc = 0;
- } else if (m->type == PTHREAD_MUTEX_ERRORCHECK) {
- rc = EDEADLK;
- } else {
- rc = EINVAL;
- }
- k_spin_unlock(&z_pthread_spinlock, key);
- return rc;
- }
- if (K_TIMEOUT_EQ(timeout, K_NO_WAIT)) {
- k_spin_unlock(&z_pthread_spinlock, key);
- return EINVAL;
- }
- rc = z_pend_curr(&z_pthread_spinlock, key, &m->wait_q, timeout);
- if (rc != 0) {
- rc = ETIMEDOUT;
- }
- return rc;
- }
- /**
- * @brief Intialize POSIX mutex.
- *
- * See IEEE 1003.1
- */
- int pthread_mutex_init(pthread_mutex_t *m,
- const pthread_mutexattr_t *attr)
- {
- const pthread_mutexattr_t *mattr;
- if (m == NULL)
- return EINVAL;
- m->owner = NULL;
- m->lock_count = 0U;
- mattr = (attr == NULL) ? &def_attr : attr;
- m->type = mattr->type;
- m->attr.type = mattr->type;
- z_waitq_init(&m->wait_q);
- return 0;
- }
- /**
- * @brief Lock POSIX mutex with non-blocking call.
- *
- * See IEEE 1003.1
- */
- int pthread_mutex_trylock(pthread_mutex_t *m)
- {
- if (m == NULL)
- return EINVAL;
- if (m->attr.type == -1)
- pthread_mutex_init(m, NULL);
- return acquire_mutex(m, K_NO_WAIT);
- }
- /**
- * @brief Lock POSIX mutex with timeout.
- *
- *
- * See IEEE 1003.1
- */
- int pthread_mutex_timedlock(pthread_mutex_t *m,
- const struct timespec *abstime)
- {
- int32_t timeout = (int32_t)timespec_to_timeoutms(abstime);
- if (m == NULL)
- return EINVAL;
- if (m->attr.type == -1)
- pthread_mutex_init(m, NULL);
- return acquire_mutex(m, K_MSEC(timeout));
- }
- /**
- * @brief Lock POSIX mutex with blocking call.
- *
- * See IEEE 1003.1
- */
- int pthread_mutex_lock(pthread_mutex_t *m)
- {
- if (m == NULL)
- return EINVAL;
- if (m->attr.type == -1)
- pthread_mutex_init(m, NULL);
- return acquire_mutex(m, K_FOREVER);
- }
- /**
- * @brief Unlock POSIX mutex.
- *
- * See IEEE 1003.1
- */
- int pthread_mutex_unlock(pthread_mutex_t *m)
- {
- k_spinlock_key_t key = k_spin_lock(&z_pthread_spinlock);
- k_tid_t thread;
- if (m == NULL)
- return EINVAL;
- if (m->attr.type == -1)
- pthread_mutex_init(m, NULL);
- if (m->owner != pthread_self()) {
- k_spin_unlock(&z_pthread_spinlock, key);
- return EPERM;
- }
- if (m->lock_count == 0U) {
- k_spin_unlock(&z_pthread_spinlock, key);
- return EINVAL;
- }
- m->lock_count--;
- if (m->lock_count == 0U) {
- thread = z_unpend_first_thread(&m->wait_q);
- if (thread) {
- m->owner = (pthread_t)thread;
- m->lock_count++;
- arch_thread_return_value_set(thread, 0);
- z_ready_thread(thread);
- z_reschedule(&z_pthread_spinlock, key);
- return 0;
- }
- m->owner = NULL;
- }
- k_spin_unlock(&z_pthread_spinlock, key);
- return 0;
- }
- /**
- * @brief Destroy POSIX mutex.
- *
- * See IEEE 1003.1
- */
- int pthread_mutex_destroy(pthread_mutex_t *m)
- {
- if ((m == NULL) || (m->attr.type == -1))
- return EINVAL;
- m->attr.type = -1;
- return 0;
- }
- /**
- * @brief Read protocol attribute for mutex.
- *
- * See IEEE 1003.1
- */
- int pthread_mutexattr_getprotocol(const pthread_mutexattr_t *attr,
- int *protocol)
- {
- *protocol = PTHREAD_PRIO_NONE;
- return 0;
- }
- /**
- * @brief Read type attribute for mutex.
- *
- * See IEEE 1003.1
- */
- int pthread_mutexattr_gettype(const pthread_mutexattr_t *attr, int *type)
- {
- *type = attr->type;
- return 0;
- }
- /**
- * @brief Set type attribute for mutex.
- *
- * See IEEE 1003.1
- */
- int pthread_mutexattr_settype(pthread_mutexattr_t *attr, int type)
- {
- int retc = EINVAL;
- if ((type == PTHREAD_MUTEX_NORMAL) ||
- (type == PTHREAD_MUTEX_RECURSIVE) ||
- (type == PTHREAD_MUTEX_ERRORCHECK)) {
- attr->type = type;
- retc = 0;
- }
- return retc;
- }
|