//+build linux package unix import "core:c" // TODO(tetra): For robustness, I'd like to mark this with align 16. // I cannot currently do this. // And at the time of writing there is a bug with putting it // as the only field in a struct. pthread_t :: distinct u64 // pthread_t :: struct #align(16) { x: u64 }; // NOTE(tetra): Got all the size constants from pthreadtypes-arch.h on my // Linux machine. PTHREAD_COND_T_SIZE :: 48 PTHREAD_MUTEXATTR_T_SIZE :: 4 PTHREAD_CONDATTR_T_SIZE :: 4 PTHREAD_RWLOCKATTR_T_SIZE :: 8 PTHREAD_BARRIERATTR_T_SIZE :: 4 // WARNING: The sizes of these things are different yet again // on non-X86! when size_of(int) == 8 { PTHREAD_ATTR_T_SIZE :: 56 PTHREAD_MUTEX_T_SIZE :: 40 PTHREAD_RWLOCK_T_SIZE :: 56 PTHREAD_BARRIER_T_SIZE :: 32 } else when size_of(int) == 4 { PTHREAD_ATTR_T_SIZE :: 32 PTHREAD_MUTEX_T_SIZE :: 32 PTHREAD_RWLOCK_T_SIZE :: 44 PTHREAD_BARRIER_T_SIZE :: 20 } pthread_cond_t :: struct #align(16) { _: [PTHREAD_COND_T_SIZE] c.char, } pthread_mutex_t :: struct #align(16) { _: [PTHREAD_MUTEX_T_SIZE] c.char, } pthread_rwlock_t :: struct #align(16) { _: [PTHREAD_RWLOCK_T_SIZE] c.char, } pthread_barrier_t :: struct #align(16) { _: [PTHREAD_BARRIER_T_SIZE] c.char, } pthread_attr_t :: struct #align(16) { _: [PTHREAD_ATTR_T_SIZE] c.char, } pthread_condattr_t :: struct #align(16) { _: [PTHREAD_CONDATTR_T_SIZE] c.char, } pthread_mutexattr_t :: struct #align(16) { _: [PTHREAD_MUTEXATTR_T_SIZE] c.char, } pthread_rwlockattr_t :: struct #align(16) { _: [PTHREAD_RWLOCKATTR_T_SIZE] c.char, } pthread_barrierattr_t :: struct #align(16) { _: [PTHREAD_BARRIERATTR_T_SIZE] c.char, } PTHREAD_MUTEX_NORMAL :: 0 PTHREAD_MUTEX_RECURSIVE :: 1 PTHREAD_MUTEX_ERRORCHECK :: 2 // TODO(tetra, 2019-11-01): Maybe make `enum c.int`s for these? PTHREAD_CREATE_JOINABLE :: 0 PTHREAD_CREATE_DETACHED :: 1 PTHREAD_INHERIT_SCHED :: 0 PTHREAD_EXPLICIT_SCHED :: 1 PTHREAD_PROCESS_PRIVATE :: 0 PTHREAD_PROCESS_SHARED :: 1 SCHED_OTHER :: 0 SCHED_FIFO :: 1 SCHED_RR :: 2 // Round robin. sched_param :: struct { sched_priority: c.int, } sem_t :: struct #align(16) { _: [SEM_T_SIZE] c.char, } when size_of(int) == 8 { SEM_T_SIZE :: 32 } else when size_of(int) == 4 { SEM_T_SIZE :: 16 } PTHREAD_CANCEL_ENABLE :: 0 PTHREAD_CANCEL_DISABLE :: 1 PTHREAD_CANCEL_DEFERRED :: 0 PTHREAD_CANCEL_ASYNCHRONOUS :: 1 foreign import "system:pthread" @(default_calling_convention="c") foreign pthread { // create named semaphore. // used in process-shared semaphores. sem_open :: proc(name: cstring, flags: c.int) -> ^sem_t --- sem_init :: proc(sem: ^sem_t, pshared: c.int, initial_value: c.uint) -> c.int --- sem_destroy :: proc(sem: ^sem_t) -> c.int --- sem_post :: proc(sem: ^sem_t) -> c.int --- sem_wait :: proc(sem: ^sem_t) -> c.int --- sem_trywait :: proc(sem: ^sem_t) -> c.int --- // sem_timedwait :: proc(sem: ^sem_t, timeout: time.TimeSpec) -> c.int ---; // NOTE: unclear whether pthread_yield is well-supported on Linux systems, // see https://linux.die.net/man/3/pthread_yield pthread_yield :: proc() -> c.int --- pthread_setcancelstate :: proc (state: c.int, old_state: ^c.int) -> c.int --- pthread_setcanceltype :: proc (type: c.int, old_type: ^c.int) -> c.int --- pthread_cancel :: proc (thread: pthread_t) -> c.int --- }