xref: /aosp_15_r20/external/musl/src/thread/sem_timedwait.c (revision c9945492fdd68bbe62686c5b452b4dc1be3f8453)
1*c9945492SAndroid Build Coastguard Worker #include <semaphore.h>
2*c9945492SAndroid Build Coastguard Worker #include <limits.h>
3*c9945492SAndroid Build Coastguard Worker #include "pthread_impl.h"
4*c9945492SAndroid Build Coastguard Worker 
cleanup(void * p)5*c9945492SAndroid Build Coastguard Worker static void cleanup(void *p)
6*c9945492SAndroid Build Coastguard Worker {
7*c9945492SAndroid Build Coastguard Worker 	a_dec(p);
8*c9945492SAndroid Build Coastguard Worker }
9*c9945492SAndroid Build Coastguard Worker 
sem_timedwait(sem_t * restrict sem,const struct timespec * restrict at)10*c9945492SAndroid Build Coastguard Worker int sem_timedwait(sem_t *restrict sem, const struct timespec *restrict at)
11*c9945492SAndroid Build Coastguard Worker {
12*c9945492SAndroid Build Coastguard Worker 	pthread_testcancel();
13*c9945492SAndroid Build Coastguard Worker 
14*c9945492SAndroid Build Coastguard Worker 	if (!sem_trywait(sem)) return 0;
15*c9945492SAndroid Build Coastguard Worker 
16*c9945492SAndroid Build Coastguard Worker 	int spins = 100;
17*c9945492SAndroid Build Coastguard Worker 	while (spins-- && !(sem->__val[0] & SEM_VALUE_MAX) && !sem->__val[1])
18*c9945492SAndroid Build Coastguard Worker 		a_spin();
19*c9945492SAndroid Build Coastguard Worker 
20*c9945492SAndroid Build Coastguard Worker 	while (sem_trywait(sem)) {
21*c9945492SAndroid Build Coastguard Worker 		int r, priv = sem->__val[2];
22*c9945492SAndroid Build Coastguard Worker 		a_inc(sem->__val+1);
23*c9945492SAndroid Build Coastguard Worker 		a_cas(sem->__val, 0, 0x80000000);
24*c9945492SAndroid Build Coastguard Worker 		pthread_cleanup_push(cleanup, (void *)(sem->__val+1));
25*c9945492SAndroid Build Coastguard Worker 		r = __timedwait_cp(sem->__val, 0x80000000, CLOCK_REALTIME, at, priv);
26*c9945492SAndroid Build Coastguard Worker 		pthread_cleanup_pop(1);
27*c9945492SAndroid Build Coastguard Worker 		if (r) {
28*c9945492SAndroid Build Coastguard Worker 			errno = r;
29*c9945492SAndroid Build Coastguard Worker 			return -1;
30*c9945492SAndroid Build Coastguard Worker 		}
31*c9945492SAndroid Build Coastguard Worker 	}
32*c9945492SAndroid Build Coastguard Worker 	return 0;
33*c9945492SAndroid Build Coastguard Worker }
34