12345678910111213141516171819202122232425262728293031 |
- #include <semaphore.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] <= 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;
- }
|