mirror of
https://github.com/fluencelabs/musl
synced 2025-06-24 20:21:59 +00:00
if a multithreaded program became non-multithreaded (i.e. all other threads exited) while one thread held an internal lock, the remaining thread would fail to release the lock. the the program then became multithreaded again at a later time, any further attempts to obtain the lock would deadlock permanently. the underlying cause is that the value of libc.threads_minus_1 at unlock time might not match the value at lock time. one solution would be returning a flag to the caller indicating whether the lock was taken and needs to be unlocked, but there is a simpler solution: using the lock itself as such a flag. note that this flag is not needed anyway for correctness; if the lock is not held, the unlock code is harmless. however, the memory synchronization properties associated with a_store are costly on some archs, so it's best to avoid executing the unlock code when it is unnecessary.
75 lines
1.5 KiB
C
75 lines
1.5 KiB
C
#ifndef LIBC_H
|
|
#define LIBC_H
|
|
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <limits.h>
|
|
|
|
struct __libc {
|
|
void *main_thread;
|
|
int threaded;
|
|
int secure;
|
|
size_t *auxv;
|
|
volatile int threads_minus_1;
|
|
int canceldisable;
|
|
FILE *ofl_head;
|
|
int ofl_lock[2];
|
|
size_t tls_size;
|
|
size_t page_size;
|
|
};
|
|
|
|
extern size_t __hwcap;
|
|
|
|
#ifndef PAGE_SIZE
|
|
#define PAGE_SIZE libc.page_size
|
|
#endif
|
|
|
|
#if !defined(__PIC__) || (100*__GNUC__+__GNUC_MINOR__ >= 303 && !defined(__PCC__))
|
|
|
|
#ifdef __PIC__
|
|
#if __GNUC__ < 4
|
|
#define BROKEN_VISIBILITY 1
|
|
#endif
|
|
#define ATTR_LIBC_VISIBILITY __attribute__((visibility("hidden")))
|
|
#else
|
|
#define ATTR_LIBC_VISIBILITY
|
|
#endif
|
|
|
|
extern struct __libc __libc ATTR_LIBC_VISIBILITY;
|
|
#define libc __libc
|
|
|
|
#else
|
|
|
|
#define USE_LIBC_ACCESSOR
|
|
#define ATTR_LIBC_VISIBILITY
|
|
extern struct __libc *__libc_loc(void) __attribute__((const));
|
|
#define libc (*__libc_loc())
|
|
|
|
#endif
|
|
|
|
|
|
/* Designed to avoid any overhead in non-threaded processes */
|
|
void __lock(volatile int *) ATTR_LIBC_VISIBILITY;
|
|
void __unlock(volatile int *) ATTR_LIBC_VISIBILITY;
|
|
int __lockfile(FILE *) ATTR_LIBC_VISIBILITY;
|
|
void __unlockfile(FILE *) ATTR_LIBC_VISIBILITY;
|
|
#define LOCK(x) __lock(x)
|
|
#define UNLOCK(x) __unlock(x)
|
|
|
|
void __synccall(void (*)(void *), void *);
|
|
int __setxid(int, int, int, int);
|
|
|
|
extern char **__environ;
|
|
|
|
#undef weak_alias
|
|
#define weak_alias(old, new) \
|
|
extern __typeof(old) new __attribute__((weak, alias(#old)))
|
|
|
|
#undef LFS64_2
|
|
#define LFS64_2(x, y) weak_alias(x, y)
|
|
|
|
#undef LFS64
|
|
#define LFS64(x) LFS64_2(x, x##64)
|
|
|
|
#endif
|