summaryrefslogtreecommitdiff
path: root/src/stdio/__lockfile.c
blob: 2ff75d8add3cea9da761d7a5290e2c20bf0d900a (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
#include "stdio_impl.h"
#include "pthread_impl.h"

#define MAYBE_WAITERS 0x40000000

int __lockfile(FILE *f)
{
	int owner = f->lock, tid = __pthread_self()->tid;
	if ((owner & ~MAYBE_WAITERS) == tid)
		return 0;
	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)
{
	if (a_swap(&f->lock, 0) & MAYBE_WAITERS)
		__wake(&f->lock, 1, 1);
}