mirror of
https://github.com/jart/cosmopolitan.git
synced 2025-02-07 23:13:34 +00:00
If threads are being used, then fork() will now acquire and release and runtime locks so that fork() may be safely used from threads. This also makes vfork() thread safe, because pthread mutexes will do nothing when the process is a child of vfork(). More torture tests have been written to confirm this all works like a charm. Additionally: - Invent hexpcpy() api - Rename nsync_malloc_() to kmalloc() - Complete posix named semaphore implementation - Make pthread_create() asynchronous signal safe - Add rm, rmdir, and touch to command interpreter builtins - Invent sigisprecious() and modify sigset functions to use it - Add unit tests for posix_spawn() attributes and fix its bugs One unresolved problem is the reclaiming of *NSYNC waiter memory in the forked child processes, within apps which have threads waiting on locks
109 lines
3.9 KiB
C
109 lines
3.9 KiB
C
#ifndef COSMOPOLITAN_LIBC_THREAD_POSIXTHREAD_INTERNAL_H_
|
|
#define COSMOPOLITAN_LIBC_THREAD_POSIXTHREAD_INTERNAL_H_
|
|
#include "libc/calls/struct/sched_param.h"
|
|
#include "libc/calls/struct/sigset.h"
|
|
#include "libc/runtime/runtime.h"
|
|
#include "libc/thread/thread.h"
|
|
#include "libc/thread/tls.h"
|
|
|
|
#define PT_OWNSTACK 1
|
|
#define PT_MAINTHREAD 2
|
|
|
|
#if !(__ASSEMBLER__ + __LINKER__ + 0)
|
|
COSMOPOLITAN_C_START_
|
|
|
|
/**
|
|
* @fileoverview Cosmopolitan POSIX Thread Internals
|
|
*/
|
|
|
|
typedef void (*atfork_f)(void);
|
|
|
|
// LEGAL TRANSITIONS ┌──> TERMINATED
|
|
// pthread_create ─┬─> JOINABLE ─┴┬─> DETACHED ──> ZOMBIE
|
|
// └──────────────┘
|
|
enum PosixThreadStatus {
|
|
|
|
// this is a running thread that needs pthread_join()
|
|
//
|
|
// the following transitions are possible:
|
|
//
|
|
// - kPosixThreadJoinable -> kPosixThreadTerminated if start_routine()
|
|
// returns, or is longjmp'd out of by pthread_exit(), and the thread
|
|
// is waiting to be joined.
|
|
//
|
|
// - kPosixThreadJoinable -> kPosixThreadDetached if pthread_detach()
|
|
// is called on this thread.
|
|
kPosixThreadJoinable,
|
|
|
|
// this is a managed thread that'll be cleaned up by the library.
|
|
//
|
|
// the following transitions are possible:
|
|
//
|
|
// - kPosixThreadDetached -> kPosixThreadZombie if start_routine()
|
|
// returns, or is longjmp'd out of by pthread_exit(), and the thread
|
|
// is waiting to be joined.
|
|
kPosixThreadDetached,
|
|
|
|
// this is a joinable thread that terminated.
|
|
//
|
|
// the following transitions are possible:
|
|
//
|
|
// - kPosixThreadTerminated -> _pthread_free() will happen when
|
|
// pthread_join() is called by the user.
|
|
kPosixThreadTerminated,
|
|
|
|
// this is a detached thread that terminated.
|
|
//
|
|
// the following transitions are possible:
|
|
//
|
|
// - kPosixThreadZombie -> _pthread_free() will happen whenever
|
|
// convenient, e.g. pthread_create() entry or atexit handler.
|
|
kPosixThreadZombie,
|
|
};
|
|
|
|
struct PosixThread {
|
|
_Atomic(enum PosixThreadStatus) status;
|
|
void *(*start_routine)(void *);
|
|
void *arg; // start_routine's parameter
|
|
void *rc; // start_routine's return value
|
|
int tid; // clone parent tid
|
|
int flags; // see PT_* constants
|
|
_Atomic(int) cancelled; // thread has bad beliefs
|
|
char cancelasync; // PTHREAD_CANCEL_DEFERRED/ASYNCHRONOUS
|
|
char canceldisable; // PTHREAD_CANCEL_ENABLE/DISABLE
|
|
char *altstack; // thread sigaltstack
|
|
char *tls; // bottom of tls allocation
|
|
struct CosmoTib *tib; // middle of tls allocation
|
|
jmp_buf exiter; // for pthread_exit
|
|
pthread_attr_t attr;
|
|
sigset_t sigmask;
|
|
struct _pthread_cleanup_buffer *cleanup;
|
|
};
|
|
|
|
extern struct PosixThread _pthread_main;
|
|
hidden extern pthread_spinlock_t _pthread_keys_lock;
|
|
hidden extern uint64_t _pthread_key_usage[(PTHREAD_KEYS_MAX + 63) / 64];
|
|
hidden extern pthread_key_dtor _pthread_key_dtor[PTHREAD_KEYS_MAX];
|
|
hidden extern _Thread_local void *_pthread_keys[PTHREAD_KEYS_MAX];
|
|
|
|
int _pthread_atfork(atfork_f, atfork_f, atfork_f) hidden;
|
|
int _pthread_reschedule(struct PosixThread *) hidden;
|
|
int _pthread_setschedparam_freebsd(int, int, const struct sched_param *) hidden;
|
|
void _pthread_free(struct PosixThread *) hidden;
|
|
void _pthread_cleanup(struct PosixThread *) hidden;
|
|
void _pthread_ungarbage(void) hidden;
|
|
void _pthread_wait(struct PosixThread *) hidden;
|
|
void _pthread_zombies_add(struct PosixThread *) hidden;
|
|
void _pthread_zombies_purge(void) hidden;
|
|
void _pthread_zombies_decimate(void) hidden;
|
|
void _pthread_zombies_harvest(void) hidden;
|
|
void _pthread_key_destruct(void *[PTHREAD_KEYS_MAX]) hidden;
|
|
void _pthread_key_lock(void) hidden;
|
|
void _pthread_key_unlock(void) hidden;
|
|
void _pthread_onfork_prepare(void) hidden;
|
|
void _pthread_onfork_parent(void) hidden;
|
|
void _pthread_onfork_child(void) hidden;
|
|
|
|
COSMOPOLITAN_C_END_
|
|
#endif /* !(__ASSEMBLER__ + __LINKER__ + 0) */
|
|
#endif /* COSMOPOLITAN_LIBC_THREAD_POSIXTHREAD_INTERNAL_H_ */
|