mirror of
https://github.com/fluencelabs/musl
synced 2025-06-22 19:21:58 +00:00
the goal is to be able to use pthread_setcancelstate internally in the implementation, whenever a function might want to use functions which are cancellation points but avoid becoming a cancellation point itself. i could have just used a separate internal function for temporarily inhibiting cancellation, but the solution in this commit is better because (1) it's one less implementation-specific detail in functions that need to use it, and (2) application code can also get the same benefit. previously, pthread_setcancelstate dependend on pthread_self, which would pull in unwanted thread setup overhead for non-threaded programs. now, it temporarily stores the state in the global libc struct if threads have not been initialized, and later moves it if needed. this way we can instead use __pthread_self, which has no dependencies and assumes that the thread register is already valid.
40 lines
870 B
C
40 lines
870 B
C
#include "pthread_impl.h"
|
|
|
|
static struct pthread main_thread;
|
|
|
|
/* pthread_key_create.c overrides this */
|
|
static const void *dummy[1] = { 0 };
|
|
weak_alias(dummy, __pthread_tsd_main);
|
|
|
|
#undef errno
|
|
static int *errno_location()
|
|
{
|
|
return __pthread_self()->errno_ptr;
|
|
}
|
|
|
|
static int init_main_thread()
|
|
{
|
|
main_thread.canceldisable = libc.canceldisable;
|
|
main_thread.tsd = (void **)__pthread_tsd_main;
|
|
main_thread.self = &main_thread;
|
|
if (__set_thread_area(&main_thread) < 0)
|
|
return -1;
|
|
main_thread.errno_ptr = __errno_location();
|
|
libc.errno_location = errno_location;
|
|
main_thread.tid = main_thread.pid =
|
|
syscall(SYS_set_tid_address, &main_thread.tid);
|
|
return 0;
|
|
}
|
|
|
|
pthread_t pthread_self()
|
|
{
|
|
static int init, failed;
|
|
if (!init) {
|
|
if (failed) return 0;
|
|
if (init_main_thread() < 0) failed = 1;
|
|
if (failed) return 0;
|
|
init = 1;
|
|
}
|
|
return __pthread_self();
|
|
}
|