X-Git-Url: http://nsz.repo.hu/git/?a=blobdiff_plain;f=src%2Fthread%2Fsem_timedwait.c;h=6d0d0114220890b1a6557cfe50a8c8327fc2dfb2;hb=f1c1a5ea8295a3f8e9ea2db8961c5a68e1a3f9ed;hp=4f45c17211bcd1d4da1142c73da44b10e4a7b518;hpb=a113434cd68ce30642c4995b1caadcd084be6f09;p=musl diff --git a/src/thread/sem_timedwait.c b/src/thread/sem_timedwait.c index 4f45c172..6d0d0114 100644 --- a/src/thread/sem_timedwait.c +++ b/src/thread/sem_timedwait.c @@ -6,36 +6,18 @@ static void cleanup(void *p) a_dec(p); } -int sem_timedwait(sem_t *sem, const struct timespec *at) +int sem_timedwait(sem_t *restrict sem, const struct timespec *restrict at) { - int r; - - if (a_fetch_add(sem->__val, -1) > 0) return 0; - a_inc(sem->__val); - - if (at && at->tv_nsec >= 1000000000UL) { - errno = EINVAL; - return -1; - } - - a_inc(sem->__val+1); - pthread_cleanup_push(cleanup, sem->__val+1) - - CANCELPT_BEGIN; - for (;;) { - r = 0; - if (!sem_trywait(sem)) break; - r = __timedwait(sem->__val, 0, CLOCK_REALTIME, at, 0); + while (sem_trywait(sem)) { + int r; + a_inc(sem->__val+1); + a_cas(sem->__val, 0, -1); + r = __timedwait(sem->__val, -1, CLOCK_REALTIME, at, cleanup, sem->__val+1, 0); + a_dec(sem->__val+1); if (r) { errno = r; - r = -1; - break; + return -1; } - CANCELPT_TRY; } - CANCELPT_END; - - pthread_cleanup_pop(1); - - return r; + return 0; }