-
Notifications
You must be signed in to change notification settings - Fork 11
/
Copy pathtail2.c
84 lines (75 loc) · 1.63 KB
/
tail2.c
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
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
/*
tail2 -- simple tail command with no line length limitation
*/
#include <stdio.h>
#include <stdlib.h>
static void tail(FILE *f, int nlines);
static unsigned char *readline(FILE *f);
int
main(int argc, char *argv[])
{
if (argc != 2) {
fprintf(stderr, "Usage: %s n\n", argv[0]);
exit(1);
}
tail(stdin, atoi(argv[1]));
exit(0);
}
static void
tail(FILE *f, int nlines)
{
unsigned char **ringbuf;
unsigned char **p;
unsigned char *line;
int n;
#define INC(ptrvar) do { \
ptrvar++; \
if (ptrvar >= ringbuf + nlines) \
ptrvar = ringbuf; \
} while (0)
if (nlines == 0) return;
ringbuf = calloc(nlines, sizeof(char*));
if (ringbuf == NULL) exit(1);
p = ringbuf;
while (line = readline(f)) {
if (*p)
free(*p);
*p = line;
INC(p);
}
if (*p == NULL) p = ringbuf;
for (n = nlines; n > 0 && *p; n--) {
printf("%s", *p);
free(*p);
INC(p);
}
free(ringbuf);
}
static unsigned char *
readline(FILE *f)
{
unsigned char *buf, *p;
size_t buflen = BUFSIZ;
int c;
buf = p = malloc(sizeof(char) * buflen);
if (buf == NULL) exit(1);
for (;;) {
c = getc(f);
if (c == EOF) {
if (buf == p) {
free(buf);
return NULL;
}
break;
}
*p++ = c;
if (p >= buf + buflen - 1) {
buflen *= 2;
buf = realloc(buf, buflen);
if (buf == NULL) exit(1);
}
if (c == '\n') break;
}
*p++ = '\0';
return buf;
}