mirror of
https://github.com/ziglang/zig.git
synced 2026-01-18 05:15:18 +00:00
34 lines
685 B
C
34 lines
685 B
C
#include <semaphore.h>
|
|
#include <limits.h>
|
|
#include "pthread_impl.h"
|
|
|
|
static void cleanup(void *p)
|
|
{
|
|
a_dec(p);
|
|
}
|
|
|
|
int sem_timedwait(sem_t *restrict sem, const struct timespec *restrict at)
|
|
{
|
|
pthread_testcancel();
|
|
|
|
if (!sem_trywait(sem)) return 0;
|
|
|
|
int spins = 100;
|
|
while (spins-- && !(sem->__val[0] & SEM_VALUE_MAX) && !sem->__val[1])
|
|
a_spin();
|
|
|
|
while (sem_trywait(sem)) {
|
|
int r, priv = sem->__val[2];
|
|
a_inc(sem->__val+1);
|
|
a_cas(sem->__val, 0, 0x80000000);
|
|
pthread_cleanup_push(cleanup, (void *)(sem->__val+1));
|
|
r = __timedwait_cp(sem->__val, 0x80000000, CLOCK_REALTIME, at, priv);
|
|
pthread_cleanup_pop(1);
|
|
if (r) {
|
|
errno = r;
|
|
return -1;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|