summaryrefslogtreecommitdiff
path: root/src/stdio/fgets.c
blob: 3135a69a2b9ebfe7b8475af708295c9e7e626056 (plain)
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
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
#include "stdio_impl.h"

#define MIN(a,b) ((a)<(b) ? (a) : (b))

char *fgets(char *s, int n, FILE *f)
{
	char *p = s;
	unsigned char *z;
	size_t k;
	int c;

	if (n--<=1) {
		if (n) return 0;
		*s = 0;
		return s;
	}

	FLOCK(f);

	while (n) {
		z = memchr(f->rpos, '\n', f->rend - f->rpos);
		k = z ? z - f->rpos + 1 : f->rend - f->rpos;
		k = MIN(k, n);
		memcpy(p, f->rpos, k);
		f->rpos += k;
		p += k;
		n -= k;
		if (z || !n) break;
		if ((c = getc_unlocked(f)) < 0) {
			if (p==s || !feof(f)) s = 0;
			break;
		}
		n--;
		if ((*p++ = c) == '\n') break;
	}
	*p = 0;

	FUNLOCK(f);

	return s;
}

weak_alias(fgets, fgets_unlocked);