fix wide printf forms ignoring width for %lc format specifier
[musl] / src / thread / sem_timedwait.c
index df5f3a6..aa67376 100644 (file)
@@ -1,4 +1,5 @@
 #include <semaphore.h>
+#include <limits.h>
 #include "pthread_impl.h"
 
 static void cleanup(void *p)
@@ -8,17 +9,21 @@ static void cleanup(void *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) a_spin();
+       while (spins-- && !(sem->__val[0] & SEM_VALUE_MAX) && !sem->__val[1])
+               a_spin();
 
        while (sem_trywait(sem)) {
-               int r;
+               int r, priv = sem->__val[2];
                a_inc(sem->__val+1);
-               a_cas(sem->__val, 0, -1);
-               r = __timedwait(sem->__val, -1, CLOCK_REALTIME, at, cleanup, sem->__val+1, sem->__val[2]);
-               a_dec(sem->__val+1);
+               a_cas(sem->__val, 0, 0x80000000);
+               pthread_cleanup_push(cleanup, (void *)(sem->__val+1));
+               r = __timedwait_cp(sem->__val, 0x80000000, CLOCK_REALTIME, at, priv);
+               pthread_cleanup_pop(1);
                if (r) {
                        errno = r;
                        return -1;