posix: use sys_sem instead of k_spinlock for pool synch

Based on Andy's talk at eoss 2024, use the sys/sem.h api instead
of the spinlock.h api to synchronize pooled elements since it
has minimal overhead like semaphores but also works from
userspace.

Signed-off-by: Chris Friedt <cfriedt@tenstorrent.com>
This commit is contained in:
Chris Friedt 2024-04-21 10:38:28 -04:00 committed by Chris Friedt
parent 36bbd3057e
commit 6e66aa1f7c
4 changed files with 154 additions and 155 deletions

View file

@ -11,6 +11,7 @@
#include <zephyr/posix/pthread.h>
#include <zephyr/sys/bitarray.h>
#include <zephyr/sys/__assert.h>
#include <zephyr/sys/sem.h>
struct pthread_key_data {
sys_snode_t node;
@ -19,7 +20,7 @@ struct pthread_key_data {
LOG_MODULE_REGISTER(pthread_key, CONFIG_PTHREAD_KEY_LOG_LEVEL);
static struct k_spinlock pthread_key_lock;
static SYS_SEM_DEFINE(pthread_key_lock, 1, 1);
/* This is non-standard (i.e. an implementation detail) */
#define PTHREAD_KEY_INITIALIZER (-1)
@ -128,42 +129,40 @@ int pthread_key_create(pthread_key_t *key,
int pthread_key_delete(pthread_key_t key)
{
size_t bit;
__unused int ret;
pthread_key_obj *key_obj;
int ret = 0;
pthread_key_obj *key_obj = NULL;
struct pthread_key_data *key_data;
sys_snode_t *node_l, *next_node_l;
k_spinlock_key_t key_key;
key_key = k_spin_lock(&pthread_key_lock);
SYS_SEM_LOCK(&pthread_key_lock) {
key_obj = get_posix_key(key);
if (key_obj == NULL) {
ret = EINVAL;
SYS_SEM_LOCK_BREAK;
}
key_obj = get_posix_key(key);
if (key_obj == NULL) {
k_spin_unlock(&pthread_key_lock, key_key);
return EINVAL;
/* Delete thread-specific elements associated with the key */
SYS_SLIST_FOR_EACH_NODE_SAFE(&(key_obj->key_data_l), node_l, next_node_l) {
/* Remove the object from the list key_data_l */
key_data = (struct pthread_key_data *)sys_slist_get(&(key_obj->key_data_l));
/* Deallocate the object's memory */
k_free((void *)key_data);
LOG_DBG("Freed key data %p for key %x in thread %x", key_data, key,
pthread_self());
}
bit = posix_key_to_offset(key_obj);
ret = sys_bitarray_free(&posix_key_bitarray, 1, bit);
__ASSERT_NO_MSG(ret == 0);
}
/* Delete thread-specific elements associated with the key */
SYS_SLIST_FOR_EACH_NODE_SAFE(&(key_obj->key_data_l),
node_l, next_node_l) {
/* Remove the object from the list key_data_l */
key_data = (struct pthread_key_data *)
sys_slist_get(&(key_obj->key_data_l));
/* Deallocate the object's memory */
k_free((void *)key_data);
LOG_DBG("Freed key data %p for key %x in thread %x", key_data, key, pthread_self());
if (ret == 0) {
LOG_DBG("Deleted key %p (%x)", key_obj, key);
}
bit = posix_key_to_offset(key_obj);
ret = sys_bitarray_free(&posix_key_bitarray, 1, bit);
__ASSERT_NO_MSG(ret == 0);
k_spin_unlock(&pthread_key_lock, key_key);
LOG_DBG("Deleted key %p (%x)", key_obj, key);
return 0;
return ret;
}
/**
@ -173,12 +172,10 @@ int pthread_key_delete(pthread_key_t key)
*/
int pthread_setspecific(pthread_key_t key, const void *value)
{
pthread_key_obj *key_obj;
pthread_key_obj *key_obj = NULL;
struct posix_thread *thread;
struct pthread_key_data *key_data;
pthread_thread_data *thread_spec_data;
k_spinlock_key_t key_key;
sys_snode_t *node_l;
sys_snode_t *node_l = NULL;
int retval = 0;
thread = to_posix_thread(pthread_self());
@ -190,37 +187,37 @@ int pthread_setspecific(pthread_key_t key, const void *value)
* If the key is already in the list, re-assign its value.
* Else add the key to the thread's list.
*/
key_key = k_spin_lock(&pthread_key_lock);
SYS_SEM_LOCK(&pthread_key_lock) {
key_obj = get_posix_key(key);
if (key_obj == NULL) {
retval = EINVAL;
SYS_SEM_LOCK_BREAK;
}
key_obj = get_posix_key(key);
if (key_obj == NULL) {
k_spin_unlock(&pthread_key_lock, key_key);
return EINVAL;
}
SYS_SLIST_FOR_EACH_NODE(&(thread->key_list), node_l) {
thread_spec_data = (pthread_thread_data *)node_l;
SYS_SLIST_FOR_EACH_NODE(&(thread->key_list), node_l) {
pthread_thread_data *thread_spec_data = (pthread_thread_data *)node_l;
if (thread_spec_data->key == key_obj) {
/* Key is already present so
* associate thread specific data
*/
/* Key is already present so associate thread specific data */
thread_spec_data->spec_data = (void *)value;
LOG_DBG("Paired key %x to value %p for thread %x", key, value,
pthread_self());
goto out;
break;
}
}
}
if (node_l == NULL) {
if (node_l != NULL) {
/* Key is already present, so we are done */
SYS_SEM_LOCK_BREAK;
}
/* Key and data need to be added */
key_data = k_malloc(sizeof(struct pthread_key_data));
if (key_data == NULL) {
LOG_DBG("Failed to allocate key data for key %x", key);
retval = ENOMEM;
goto out;
SYS_SEM_LOCK_BREAK;
}
LOG_DBG("Allocated key data %p for key %x in thread %x", key_data, key,
@ -239,9 +236,6 @@ int pthread_setspecific(pthread_key_t key, const void *value)
LOG_DBG("Paired key %x to value %p for thread %x", key, value, pthread_self());
}
out:
k_spin_unlock(&pthread_key_lock, key_key);
return retval;
}
@ -257,33 +251,30 @@ void *pthread_getspecific(pthread_key_t key)
pthread_thread_data *thread_spec_data;
void *value = NULL;
sys_snode_t *node_l;
k_spinlock_key_t key_key;
thread = to_posix_thread(pthread_self());
if (thread == NULL) {
return NULL;
}
key_key = k_spin_lock(&pthread_key_lock);
SYS_SEM_LOCK(&pthread_key_lock) {
key_obj = get_posix_key(key);
if (key_obj == NULL) {
value = NULL;
SYS_SEM_LOCK_BREAK;
}
key_obj = get_posix_key(key);
if (key_obj == NULL) {
k_spin_unlock(&pthread_key_lock, key_key);
return NULL;
}
/* Traverse the list of keys set by the thread, looking for key */
/* Traverse the list of keys set by the thread, looking for key */
SYS_SLIST_FOR_EACH_NODE(&(thread->key_list), node_l) {
thread_spec_data = (pthread_thread_data *)node_l;
if (thread_spec_data->key == key_obj) {
/* Key is present, so get the set thread data */
value = thread_spec_data->spec_data;
break;
SYS_SLIST_FOR_EACH_NODE(&(thread->key_list), node_l) {
thread_spec_data = (pthread_thread_data *)node_l;
if (thread_spec_data->key == key_obj) {
/* Key is present, so get the set thread data */
value = thread_spec_data->spec_data;
break;
}
}
}
k_spin_unlock(&pthread_key_lock, key_key);
return value;
}

View file

@ -12,10 +12,11 @@
#include <zephyr/logging/log.h>
#include <zephyr/posix/pthread.h>
#include <zephyr/sys/bitarray.h>
#include <zephyr/sys/sem.h>
LOG_MODULE_REGISTER(pthread_mutex, CONFIG_PTHREAD_MUTEX_LOG_LEVEL);
static struct k_spinlock pthread_mutex_spinlock;
static SYS_SEM_DEFINE(lock, 1, 1);
int64_t timespec_to_timeoutms(const struct timespec *abstime);
@ -106,35 +107,41 @@ struct k_mutex *to_posix_mutex(pthread_mutex_t *mu)
static int acquire_mutex(pthread_mutex_t *mu, k_timeout_t timeout)
{
int type;
size_t bit;
int ret = 0;
struct k_mutex *m;
k_spinlock_key_t key;
int type = -1;
size_t bit = -1;
size_t lock_count = -1;
struct k_mutex *m = NULL;
struct k_thread *owner = NULL;
key = k_spin_lock(&pthread_mutex_spinlock);
SYS_SEM_LOCK(&lock) {
m = to_posix_mutex(mu);
if (m == NULL) {
ret = EINVAL;
SYS_SEM_LOCK_BREAK;
}
m = to_posix_mutex(mu);
if (m == NULL) {
k_spin_unlock(&pthread_mutex_spinlock, key);
return EINVAL;
LOG_DBG("Locking mutex %p with timeout %llx", m, timeout.ticks);
bit = posix_mutex_to_offset(m);
type = posix_mutex_type[bit];
owner = m->owner;
lock_count = m->lock_count;
}
LOG_DBG("Locking mutex %p with timeout %llx", m, timeout.ticks);
if (ret != 0) {
goto handle_error;
}
bit = posix_mutex_to_offset(m);
type = posix_mutex_type[bit];
if (m->owner == k_current_get()) {
if (owner == k_current_get()) {
switch (type) {
case PTHREAD_MUTEX_NORMAL:
if (K_TIMEOUT_EQ(timeout, K_NO_WAIT)) {
k_spin_unlock(&pthread_mutex_spinlock, key);
LOG_DBG("Timeout locking mutex %p", m);
return EBUSY;
ret = EBUSY;
break;
}
/* On most POSIX systems, this usually results in an infinite loop */
k_spin_unlock(&pthread_mutex_spinlock, key);
LOG_DBG("Attempt to relock non-recursive mutex %p", m);
do {
(void)k_sleep(K_FOREVER);
@ -142,7 +149,7 @@ static int acquire_mutex(pthread_mutex_t *mu, k_timeout_t timeout)
CODE_UNREACHABLE;
break;
case PTHREAD_MUTEX_RECURSIVE:
if (m->lock_count >= MUTEX_MAX_REC_LOCK) {
if (lock_count >= MUTEX_MAX_REC_LOCK) {
LOG_DBG("Mutex %p locked recursively too many times", m);
ret = EAGAIN;
}
@ -157,7 +164,6 @@ static int acquire_mutex(pthread_mutex_t *mu, k_timeout_t timeout)
break;
}
}
k_spin_unlock(&pthread_mutex_spinlock, key);
if (ret == 0) {
ret = k_mutex_lock(m, timeout);
@ -171,6 +177,7 @@ static int acquire_mutex(pthread_mutex_t *mu, k_timeout_t timeout)
}
}
handle_error:
if (ret < 0) {
LOG_DBG("k_mutex_unlock() failed: %d", ret);
ret = -ret;

View file

@ -16,6 +16,7 @@
#include <zephyr/sys/atomic.h>
#include <zephyr/posix/pthread.h>
#include <zephyr/posix/unistd.h>
#include <zephyr/sys/sem.h>
#include <zephyr/sys/slist.h>
#include <zephyr/sys/util.h>
@ -87,7 +88,7 @@ static sys_dlist_t posix_thread_q[] = {
SYS_DLIST_STATIC_INIT(&posix_thread_q[POSIX_THREAD_DONE_Q]),
};
static struct posix_thread posix_thread_pool[CONFIG_MAX_PTHREAD_COUNT];
static struct k_spinlock pthread_pool_lock;
static SYS_SEM_DEFINE(pthread_pool_lock, 1, 1);
static int pthread_concurrency;
static inline void posix_thread_q_set(struct posix_thread *t, enum posix_thread_qid qid)
@ -219,7 +220,7 @@ void __z_pthread_cleanup_push(void *cleanup[3], void (*routine)(void *arg), void
struct posix_thread *t = NULL;
struct __pthread_cleanup *const c = (struct __pthread_cleanup *)cleanup;
K_SPINLOCK(&pthread_pool_lock) {
SYS_SEM_LOCK(&pthread_pool_lock) {
t = to_posix_thread(pthread_self());
BUILD_ASSERT(3 * sizeof(void *) == sizeof(*c));
__ASSERT_NO_MSG(t != NULL);
@ -236,7 +237,7 @@ void __z_pthread_cleanup_pop(int execute)
struct __pthread_cleanup *c = NULL;
struct posix_thread *t = NULL;
K_SPINLOCK(&pthread_pool_lock) {
SYS_SEM_LOCK(&pthread_pool_lock) {
t = to_posix_thread(pthread_self());
__ASSERT_NO_MSG(t != NULL);
node = sys_slist_get(&t->cleanup_list);
@ -476,7 +477,6 @@ static K_WORK_DELAYABLE_DEFINE(posix_thread_recycle_work, posix_thread_recycle_w
static void posix_thread_finalize(struct posix_thread *t, void *retval)
{
sys_snode_t *node_l;
k_spinlock_key_t key;
pthread_key_obj *key_obj;
pthread_thread_data *thread_spec_data;
@ -491,11 +491,11 @@ static void posix_thread_finalize(struct posix_thread *t, void *retval)
}
/* move thread from run_q to done_q */
key = k_spin_lock(&pthread_pool_lock);
sys_dlist_remove(&t->q_node);
posix_thread_q_set(t, POSIX_THREAD_DONE_Q);
t->retval = retval;
k_spin_unlock(&pthread_pool_lock, key);
SYS_SEM_LOCK(&pthread_pool_lock) {
sys_dlist_remove(&t->q_node);
posix_thread_q_set(t, POSIX_THREAD_DONE_Q);
t->retval = retval;
}
/* trigger recycle work */
(void)k_work_schedule(&posix_thread_recycle_work, K_MSEC(CONFIG_PTHREAD_RECYCLER_DELAY_MS));
@ -526,22 +526,22 @@ static void zephyr_thread_wrapper(void *arg1, void *arg2, void *arg3)
static void posix_thread_recycle(void)
{
k_spinlock_key_t key;
struct posix_thread *t;
struct posix_thread *safe_t;
sys_dlist_t recyclables = SYS_DLIST_STATIC_INIT(&recyclables);
key = k_spin_lock(&pthread_pool_lock);
SYS_DLIST_FOR_EACH_CONTAINER_SAFE(&posix_thread_q[POSIX_THREAD_DONE_Q], t, safe_t, q_node) {
if (t->attr.detachstate == PTHREAD_CREATE_JOINABLE) {
/* thread has not been joined yet */
continue;
}
SYS_SEM_LOCK(&pthread_pool_lock) {
SYS_DLIST_FOR_EACH_CONTAINER_SAFE(&posix_thread_q[POSIX_THREAD_DONE_Q], t, safe_t,
q_node) {
if (t->attr.detachstate == PTHREAD_CREATE_JOINABLE) {
/* thread has not been joined yet */
continue;
}
sys_dlist_remove(&t->q_node);
sys_dlist_append(&recyclables, &t->q_node);
sys_dlist_remove(&t->q_node);
sys_dlist_append(&recyclables, &t->q_node);
}
}
k_spin_unlock(&pthread_pool_lock, key);
if (sys_dlist_is_empty(&recyclables)) {
return;
@ -557,12 +557,12 @@ static void posix_thread_recycle(void)
}
}
key = k_spin_lock(&pthread_pool_lock);
while (!sys_dlist_is_empty(&recyclables)) {
t = CONTAINER_OF(sys_dlist_get(&recyclables), struct posix_thread, q_node);
posix_thread_q_set(t, POSIX_THREAD_READY_Q);
SYS_SEM_LOCK(&pthread_pool_lock) {
while (!sys_dlist_is_empty(&recyclables)) {
t = CONTAINER_OF(sys_dlist_get(&recyclables), struct posix_thread, q_node);
posix_thread_q_set(t, POSIX_THREAD_READY_Q);
}
}
k_spin_unlock(&pthread_pool_lock, key);
}
/**
@ -587,7 +587,7 @@ int pthread_create(pthread_t *th, const pthread_attr_t *_attr, void *(*threadrou
/* reclaim resources greedily */
posix_thread_recycle();
K_SPINLOCK(&pthread_pool_lock) {
SYS_SEM_LOCK(&pthread_pool_lock) {
if (!sys_dlist_is_empty(&posix_thread_q[POSIX_THREAD_READY_Q])) {
t = CONTAINER_OF(sys_dlist_get(&posix_thread_q[POSIX_THREAD_READY_Q]),
struct posix_thread, q_node);
@ -603,7 +603,7 @@ int pthread_create(pthread_t *th, const pthread_attr_t *_attr, void *(*threadrou
err = pthread_barrier_init(&barrier, NULL, 2);
if (err != 0) {
/* cannot allocate barrier. move thread back to ready_q */
K_SPINLOCK(&pthread_pool_lock) {
SYS_SEM_LOCK(&pthread_pool_lock) {
sys_dlist_remove(&t->q_node);
posix_thread_q_set(t, POSIX_THREAD_READY_Q);
}
@ -625,7 +625,7 @@ int pthread_create(pthread_t *th, const pthread_attr_t *_attr, void *(*threadrou
}
if (err != 0) {
/* cannot allocate pthread attributes (e.g. stack) */
K_SPINLOCK(&pthread_pool_lock) {
SYS_SEM_LOCK(&pthread_pool_lock) {
sys_dlist_remove(&t->q_node);
posix_thread_q_set(t, POSIX_THREAD_READY_Q);
}
@ -673,7 +673,7 @@ int pthread_getconcurrency(void)
{
int ret = 0;
K_SPINLOCK(&pthread_pool_lock) {
SYS_SEM_LOCK(&pthread_pool_lock) {
ret = pthread_concurrency;
}
@ -690,7 +690,7 @@ int pthread_setconcurrency(int new_level)
return EAGAIN;
}
K_SPINLOCK(&pthread_pool_lock) {
SYS_SEM_LOCK(&pthread_pool_lock) {
pthread_concurrency = new_level;
}
@ -705,8 +705,8 @@ int pthread_setconcurrency(int new_level)
int pthread_setcancelstate(int state, int *oldstate)
{
int ret = 0;
struct posix_thread *t;
bool cancel_pending = false;
struct posix_thread *t = NULL;
bool cancel_type = PTHREAD_CANCEL_ENABLE;
if (state != PTHREAD_CANCEL_ENABLE && state != PTHREAD_CANCEL_DISABLE) {
@ -714,11 +714,11 @@ int pthread_setcancelstate(int state, int *oldstate)
return EINVAL;
}
K_SPINLOCK(&pthread_pool_lock) {
SYS_SEM_LOCK(&pthread_pool_lock) {
t = to_posix_thread(pthread_self());
if (t == NULL) {
ret = EINVAL;
K_SPINLOCK_BREAK;
SYS_SEM_LOCK_BREAK;
}
if (oldstate != NULL) {
@ -753,11 +753,11 @@ int pthread_setcanceltype(int type, int *oldtype)
return EINVAL;
}
K_SPINLOCK(&pthread_pool_lock) {
SYS_SEM_LOCK(&pthread_pool_lock) {
t = to_posix_thread(pthread_self());
if (t == NULL) {
ret = EINVAL;
K_SPINLOCK_BREAK;
SYS_SEM_LOCK_BREAK;
}
if (oldtype != NULL) {
@ -776,16 +776,16 @@ int pthread_setcanceltype(int type, int *oldtype)
*/
void pthread_testcancel(void)
{
struct posix_thread *t;
bool cancel_pended = false;
struct posix_thread *t = NULL;
K_SPINLOCK(&pthread_pool_lock) {
SYS_SEM_LOCK(&pthread_pool_lock) {
t = to_posix_thread(pthread_self());
if (t == NULL) {
K_SPINLOCK_BREAK;
SYS_SEM_LOCK_BREAK;
}
if (t->attr.cancelstate != PTHREAD_CANCEL_ENABLE) {
K_SPINLOCK_BREAK;
SYS_SEM_LOCK_BREAK;
}
if (t->attr.cancelpending) {
cancel_pended = true;
@ -810,17 +810,17 @@ int pthread_cancel(pthread_t pthread)
bool cancel_type = PTHREAD_CANCEL_DEFERRED;
struct posix_thread *t = NULL;
K_SPINLOCK(&pthread_pool_lock) {
SYS_SEM_LOCK(&pthread_pool_lock) {
t = to_posix_thread(pthread);
if (t == NULL) {
ret = ESRCH;
K_SPINLOCK_BREAK;
SYS_SEM_LOCK_BREAK;
}
if (!__attr_is_initialized(&t->attr)) {
/* thread has already terminated */
ret = ESRCH;
K_SPINLOCK_BREAK;
SYS_SEM_LOCK_BREAK;
}
t->attr.cancelpending = true;
@ -852,11 +852,11 @@ int pthread_setschedparam(pthread_t pthread, int policy, const struct sched_para
return EINVAL;
}
K_SPINLOCK(&pthread_pool_lock) {
SYS_SEM_LOCK(&pthread_pool_lock) {
t = to_posix_thread(pthread);
if (t == NULL) {
ret = ESRCH;
K_SPINLOCK_BREAK;
SYS_SEM_LOCK_BREAK;
}
new_prio = posix_to_zephyr_priority(param->sched_priority, policy);
@ -892,11 +892,11 @@ int pthread_setschedprio(pthread_t thread, int prio)
return EINVAL;
}
K_SPINLOCK(&pthread_pool_lock) {
SYS_SEM_LOCK(&pthread_pool_lock) {
t = to_posix_thread(thread);
if (t == NULL) {
ret = ESRCH;
K_SPINLOCK_BREAK;
SYS_SEM_LOCK_BREAK;
}
new_prio = posix_to_zephyr_priority(prio, policy);
@ -965,16 +965,16 @@ int pthread_getschedparam(pthread_t pthread, int *policy, struct sched_param *pa
return EINVAL;
}
K_SPINLOCK(&pthread_pool_lock) {
SYS_SEM_LOCK(&pthread_pool_lock) {
t = to_posix_thread(pthread);
if (t == NULL) {
ret = ESRCH;
K_SPINLOCK_BREAK;
SYS_SEM_LOCK_BREAK;
}
if (!__attr_is_initialized(&t->attr)) {
ret = ESRCH;
K_SPINLOCK_BREAK;
SYS_SEM_LOCK_BREAK;
}
param->sched_priority =
@ -999,7 +999,7 @@ int pthread_once(pthread_once_t *once, void (*init_func)(void))
return EINVAL;
}
K_SPINLOCK(&pthread_pool_lock) {
SYS_SEM_LOCK(&pthread_pool_lock) {
if (!_once->flag) {
run_init_func = true;
_once->flag = true;
@ -1023,10 +1023,10 @@ void pthread_exit(void *retval)
{
struct posix_thread *self = NULL;
K_SPINLOCK(&pthread_pool_lock) {
SYS_SEM_LOCK(&pthread_pool_lock) {
self = to_posix_thread(pthread_self());
if (self == NULL) {
K_SPINLOCK_BREAK;
SYS_SEM_LOCK_BREAK;
}
/* Mark a thread as cancellable before exiting */
@ -1060,11 +1060,11 @@ int pthread_join(pthread_t pthread, void **status)
return EDEADLK;
}
K_SPINLOCK(&pthread_pool_lock) {
SYS_SEM_LOCK(&pthread_pool_lock) {
t = to_posix_thread(pthread);
if (t == NULL) {
ret = ESRCH;
K_SPINLOCK_BREAK;
SYS_SEM_LOCK_BREAK;
}
LOG_DBG("Pthread %p joining..", &t->thread);
@ -1072,12 +1072,12 @@ int pthread_join(pthread_t pthread, void **status)
if (t->attr.detachstate != PTHREAD_CREATE_JOINABLE) {
/* undefined behaviour */
ret = EINVAL;
K_SPINLOCK_BREAK;
SYS_SEM_LOCK_BREAK;
}
if (posix_thread_q_get(t) == POSIX_THREAD_READY_Q) {
ret = ESRCH;
K_SPINLOCK_BREAK;
SYS_SEM_LOCK_BREAK;
}
/*
@ -1124,18 +1124,18 @@ int pthread_detach(pthread_t pthread)
int ret = 0;
struct posix_thread *t;
K_SPINLOCK(&pthread_pool_lock) {
SYS_SEM_LOCK(&pthread_pool_lock) {
t = to_posix_thread(pthread);
if (t == NULL) {
ret = ESRCH;
K_SPINLOCK_BREAK;
SYS_SEM_LOCK_BREAK;
}
if (posix_thread_q_get(t) == POSIX_THREAD_READY_Q ||
t->attr.detachstate != PTHREAD_CREATE_JOINABLE) {
LOG_DBG("Pthread %p cannot be detached", &t->thread);
ret = EINVAL;
K_SPINLOCK_BREAK;
SYS_SEM_LOCK_BREAK;
}
t->attr.detachstate = PTHREAD_CREATE_DETACHED;
@ -1437,11 +1437,11 @@ int pthread_sigmask(int how, const sigset_t *ZRESTRICT set, sigset_t *ZRESTRICT
return EINVAL;
}
K_SPINLOCK(&pthread_pool_lock) {
SYS_SEM_LOCK(&pthread_pool_lock) {
t = to_posix_thread(pthread_self());
if (t == NULL) {
ret = ESRCH;
K_SPINLOCK_BREAK;
SYS_SEM_LOCK_BREAK;
}
if (oset != NULL) {
@ -1449,7 +1449,7 @@ int pthread_sigmask(int how, const sigset_t *ZRESTRICT set, sigset_t *ZRESTRICT
}
if (set == NULL) {
K_SPINLOCK_BREAK;
SYS_SEM_LOCK_BREAK;
}
switch (how) {

View file

@ -11,6 +11,7 @@
#include <zephyr/logging/log.h>
#include <zephyr/posix/pthread.h>
#include <zephyr/sys/bitarray.h>
#include <zephyr/sys/sem.h>
#define CONCURRENT_READER_LIMIT (CONFIG_MAX_PTHREAD_COUNT + 1)
@ -32,7 +33,7 @@ static uint32_t write_lock_acquire(struct posix_rwlock *rwl, int32_t timeout);
LOG_MODULE_REGISTER(pthread_rwlock, CONFIG_PTHREAD_RWLOCK_LOG_LEVEL);
static struct k_spinlock posix_rwlock_spinlock;
static SYS_SEM_DEFINE(posix_rwlock_lock, 1, 1);
static struct posix_rwlock posix_rwlock_pool[CONFIG_MAX_PTHREAD_RWLOCK_COUNT];
SYS_BITARRAY_DEFINE_STATIC(posix_rwlock_bitarray, CONFIG_MAX_PTHREAD_RWLOCK_COUNT);
@ -150,10 +151,10 @@ int pthread_rwlock_destroy(pthread_rwlock_t *rwlock)
return EINVAL;
}
K_SPINLOCK(&posix_rwlock_spinlock) {
SYS_SEM_LOCK(&posix_rwlock_lock) {
if (rwl->wr_owner != NULL) {
ret = EBUSY;
K_SPINLOCK_BREAK;
SYS_SEM_LOCK_BREAK;
}
bit = posix_rwlock_to_offset(rwl);