X-Git-Url: http://nsz.repo.hu/git/?a=blobdiff_plain;f=src%2Fthread%2Fpthread_rwlock_unlock.c;h=a6d20854c8a5001b3d8739636619d4d76c73694b;hb=697acde67e0da4d73b46445ed536fe9923d515c7;hp=f39117e743ee75c20a420f8ecfec1d336adb8d42;hpb=0b44a0315b47dd8eced9f3b7f31580cf14bbfc01;p=musl diff --git a/src/thread/pthread_rwlock_unlock.c b/src/thread/pthread_rwlock_unlock.c index f39117e7..a6d20854 100644 --- a/src/thread/pthread_rwlock_unlock.c +++ b/src/thread/pthread_rwlock_unlock.c @@ -2,16 +2,17 @@ int pthread_rwlock_unlock(pthread_rwlock_t *rw) { - struct pthread *self = pthread_self(); - if (rw->__owner == self->tid) { - rw->__owner = 0; - a_store(&rw->__wrlock, 0); - if (rw->__waiters) - __wake(&rw->__wrlock, -1, 0); - return 0; - } - a_dec(&rw->__readers); - if (rw->__waiters && !rw->__readers) - __wake(&rw->__readers, 1, 0); + int val, cnt, waiters, new; + + do { + val = rw->_rw_lock; + cnt = val & 0x7fffffff; + waiters = rw->_rw_waiters; + new = (cnt == 0x7fffffff || cnt == 1) ? 0 : val-1; + } while (a_cas(&rw->_rw_lock, val, new) != val); + + if (!new && (waiters || val<0)) + __wake(&rw->_rw_lock, cnt, 0); + return 0; }