pthread_once: g_lock may lead deadlock

For programs with the dependencies logic in pthread_once callback , using global locks may cause deadlock:

task A
pthread_once()
|
|-> nxrmutex_lock(&g_lock);
 -> init_routine(); // callback to wait task B
                                                  task B
                                                  pthread_once()
                                                  |
                                                   ->nxrmutex_lock(&g_lock); // Deadlock
                                                   ->init_routine(); // hold resource to wake up task A

Signed-off-by: hujun5 <hujun5@xiaomi.com>
This commit is contained in:
hujun5 2023-09-21 17:36:27 +08:00 committed by Xiang Xiao
parent f02ad03124
commit b6693065e7
2 changed files with 20 additions and 20 deletions

View File

@ -136,7 +136,7 @@
/* Used to initialize a pthread_once_t */ /* Used to initialize a pthread_once_t */
#define PTHREAD_ONCE_INIT (false) #define PTHREAD_ONCE_INIT {false, PTHREAD_MUTEX_INITIALIZER}
/* This is returned by pthread_barrier_wait. It must not match any errno /* This is returned by pthread_barrier_wait. It must not match any errno
* in errno.h * in errno.h
@ -371,8 +371,14 @@ typedef struct pthread_barrier_s pthread_barrier_t;
# define __PTHREAD_BARRIER_T_DEFINED 1 # define __PTHREAD_BARRIER_T_DEFINED 1
#endif #endif
struct pthread_once_s
{
bool done;
pthread_mutex_t mutex;
};
#ifndef __PTHREAD_ONCE_T_DEFINED #ifndef __PTHREAD_ONCE_T_DEFINED
typedef bool pthread_once_t; typedef struct pthread_once_s pthread_once_t;
# define __PTHREAD_ONCE_T_DEFINED 1 # define __PTHREAD_ONCE_T_DEFINED 1
#endif #endif
@ -839,7 +845,8 @@ typedef FAR struct pthread_spinlock_s pthread_spinlock_t;
#endif /* CONFIG_PTHREAD_SPINLOCKS */ #endif /* CONFIG_PTHREAD_SPINLOCKS */
#ifndef __PTHREAD_ONCE_T_DEFINED #ifndef __PTHREAD_ONCE_T_DEFINED
typedef bool pthread_once_t; struct pthread_once_s;
typedef struct pthread_once_s pthread_once_t;
# define __PTHREAD_ONCE_T_DEFINED 1 # define __PTHREAD_ONCE_T_DEFINED 1
#endif #endif

View File

@ -61,8 +61,6 @@
* *
****************************************************************************/ ****************************************************************************/
static rmutex_t g_lock = NXRMUTEX_INITIALIZER;
int pthread_once(FAR pthread_once_t *once_control, int pthread_once(FAR pthread_once_t *once_control,
CODE void (*init_routine)(void)) CODE void (*init_routine)(void))
{ {
@ -73,25 +71,20 @@ int pthread_once(FAR pthread_once_t *once_control,
return EINVAL; return EINVAL;
} }
/* Prohibit pre-emption while we test and set the once_control. */ if (!once_control->done)
nxrmutex_lock(&g_lock);
if (!*once_control)
{ {
*once_control = true; pthread_mutex_lock(&once_control->mutex);
if (!once_control->done)
{
/* Call the init_routine with pre-emption enabled. */ /* Call the init_routine with pre-emption enabled. */
init_routine(); init_routine();
nxrmutex_unlock(&g_lock); once_control->done = true;
return OK;
} }
/* The init_routine has already been called. pthread_mutex_unlock(&once_control->mutex);
* Restore pre-emption and return. }
*/
nxrmutex_unlock(&g_lock);
return OK; return OK;
} }