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:
parent
f02ad03124
commit
b6693065e7
@ -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
|
||||||
|
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user