sem_timedwait.c 628 B

1234567891011121314151617181920212223242526272829303132333435363738394041
  1. #include <semaphore.h>
  2. #include "pthread_impl.h"
  3. static void cleanup(void *p)
  4. {
  5. a_dec(p);
  6. }
  7. int sem_timedwait(sem_t *sem, const struct timespec *at)
  8. {
  9. int r;
  10. if (a_fetch_add(sem->__val, -1) > 0) return 0;
  11. a_inc(sem->__val);
  12. if (at && at->tv_nsec >= 1000000000UL) {
  13. errno = EINVAL;
  14. return -1;
  15. }
  16. a_inc(sem->__val+1);
  17. pthread_cleanup_push(cleanup, sem->__val+1)
  18. CANCELPT_BEGIN;
  19. for (;;) {
  20. r = 0;
  21. if (!sem_trywait(sem)) break;
  22. r = __timedwait(sem->__val, 0, CLOCK_REALTIME, at, 0);
  23. if (r) {
  24. errno = r;
  25. r = -1;
  26. break;
  27. }
  28. CANCELPT_TRY;
  29. }
  30. CANCELPT_END;
  31. pthread_cleanup_pop(1);
  32. return r;
  33. }