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

#ifdef __GNUC__
__attribute__((__noinline__))
#endif
static int locking_getc(FILE *f, int tid)
{
	if (a_cas(&f->lock, 0, tid)) __lockfile(f);
	int c = getc_unlocked(f);
	if (a_swap(&f->lock, 0) & MAYBE_WAITERS)
		__wake(&f->lock, 1, 1);
	return c;
}

static inline int do_getc(FILE *f)
{
	int tid, l = f->lock;
	if (l < 0 || (l & ~MAYBE_WAITERS) == (tid=__pthread_self()->tid))
		return getc_unlocked(f);
	return locking_getc(f, tid);
}