#include #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] <= 0 && !sem->__val[1]) a_spin(); while (sem_trywait(sem)) { int r; a_inc(sem->__val+1); a_cas(sem->__val, 0, -1); pthread_cleanup_push(cleanup, (void *)(sem->__val+1)); r = __timedwait_cp(sem->__val, -1, CLOCK_REALTIME, at, sem->__val[2]); pthread_cleanup_pop(1); if (r && r != EINTR) { errno = r; return -1; } } return 0; }