mirror of https://git.musl-libc.org/git/musl
Browse Source
instead of using a waiters count, add a bit to the lock field indicating that the lock may have waiters. threads which obtain the lock after contending for it will perform a potentially-spurious wake when they release the lock.master
3 changed files with 24 additions and 20 deletions
@ -1,28 +1,25 @@ |
|||
#include "stdio_impl.h" |
|||
#include "pthread_impl.h" |
|||
|
|||
#define MAYBE_WAITERS 0x40000000 |
|||
|
|||
int __lockfile(FILE *f) |
|||
{ |
|||
int owner, tid = __pthread_self()->tid; |
|||
if (f->lock == tid) |
|||
int owner = f->lock, tid = __pthread_self()->tid; |
|||
if ((owner & ~MAYBE_WAITERS) == tid) |
|||
return 0; |
|||
while ((owner = a_cas(&f->lock, 0, tid))) |
|||
__wait(&f->lock, &f->waiters, owner, 1); |
|||
for (;;) { |
|||
owner = a_cas(&f->lock, 0, tid); |
|||
if (!owner) return 1; |
|||
if (a_cas(&f->lock, owner, owner|MAYBE_WAITERS)==owner) break; |
|||
} |
|||
while ((owner = a_cas(&f->lock, 0, tid|MAYBE_WAITERS))) |
|||
__futexwait(&f->lock, owner, 1); |
|||
return 1; |
|||
} |
|||
|
|||
void __unlockfile(FILE *f) |
|||
{ |
|||
a_store(&f->lock, 0); |
|||
|
|||
/* The following read is technically invalid under situations
|
|||
* of self-synchronized destruction. Another thread may have |
|||
* called fclose as soon as the above store has completed. |
|||
* Nonetheless, since FILE objects always live in memory |
|||
* obtained by malloc from the heap, it's safe to assume |
|||
* the dereferences below will not fault. In the worst case, |
|||
* a spurious syscall will be made. If the implementation of |
|||
* malloc changes, this assumption needs revisiting. */ |
|||
|
|||
if (f->waiters) __wake(&f->lock, 1, 1); |
|||
if (a_swap(&f->lock, 0) & MAYBE_WAITERS) |
|||
__wake(&f->lock, 1, 1); |
|||
} |
|||
|
|||
@ -1,10 +1,14 @@ |
|||
#include "stdio_impl.h" |
|||
#include "pthread_impl.h" |
|||
|
|||
#define MAYBE_WAITERS 0x40000000 |
|||
|
|||
void flockfile(FILE *f) |
|||
{ |
|||
while (ftrylockfile(f)) { |
|||
int owner = f->lock; |
|||
if (owner) __wait(&f->lock, &f->waiters, owner, 1); |
|||
if (!owner) continue; |
|||
a_cas(&f->lock, owner, owner|MAYBE_WAITERS); |
|||
__futexwait(&f->lock, owner|MAYBE_WAITERS, 1); |
|||
} |
|||
} |
|||
|
|||
Loading…
Reference in new issue