Skip to content

Commit

Permalink
Remove Redundant Atomic Store from Recursive Unfair Lock in Recursive…
Browse files Browse the repository at this point in the history
… Case #trivial (TextureGroup#867)

* Optimize recursive unfair lock to remove a redundant set

* Simpler
  • Loading branch information
Adlai-Holler authored and bernieperez committed Apr 25, 2018
1 parent 55256a3 commit dc1d21f
Showing 1 changed file with 20 additions and 11 deletions.
31 changes: 20 additions & 11 deletions Source/Details/ASRecursiveUnfairLock.m
Original file line number Diff line number Diff line change
Expand Up @@ -28,33 +28,42 @@

void ASRecursiveUnfairLockLock(ASRecursiveUnfairLock *l)
{
// Just a cache for pthread_self so that we never call it twice.
pthread_t s = NULL;

// Try to lock without blocking. If we fail, check what thread owns it.
// Note that the owning thread CAN CHANGE freely, but it can't become `self`
// because only we are `self`. And if it's already `self` then we already have
// the lock, because we reset it to NULL before we unlock. So (thread == self) is
// invariant.

if (!os_unfair_lock_trylock(&l->_lock) && (rul_get_thread(l) != (s = pthread_self()))) {
// Owned by other thread. Possibly other threads are waiting too. Block.
const pthread_t s = pthread_self();
if (os_unfair_lock_trylock(&l->_lock)) {
// Owned by nobody. We now have the lock. Assign self.
rul_set_thread(l, s);
} else if (rul_get_thread(l) == s) {
// Owned by self (recursive lock). nop.
} else {
// Owned by other thread. Block and then set thread to self.
os_unfair_lock_lock(&l->_lock);
rul_set_thread(l, s);
}
// Now we've got the lock. Update the thread pointer and count.
rul_set_thread(l, s ?: pthread_self());

l->_count++;
}

BOOL ASRecursiveUnfairLockTryLock(ASRecursiveUnfairLock *l)
{
// Same logic as `Lock` function, see comments there.
pthread_t s = NULL;
// Same as Lock above. See comments there.

if (!os_unfair_lock_trylock(&l->_lock) && (rul_get_thread(l) != (s = pthread_self()))) {
const pthread_t s = pthread_self();
if (os_unfair_lock_trylock(&l->_lock)) {
// Owned by nobody. We now have the lock. Assign self.
rul_set_thread(l, s);
} else if (rul_get_thread(l) == s) {
// Owned by self (recursive lock). nop.
} else {
// Owned by other thread. Fail.
return NO;
}
rul_set_thread(l, s ?: pthread_self());

l->_count++;
return YES;
}
Expand Down

0 comments on commit dc1d21f

Please sign in to comment.