-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathslist.c
103 lines (88 loc) · 1.5 KB
/
slist.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
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
#include <string.h>
#include <stdlib.h>
#include <alloca.h>
#include "slist.h"
slist*
s_cons(const char* text, slist* rest)
{
slist* xs = malloc(sizeof(slist));
xs->data = strdup(text);
xs->refs = 1;
xs->next = rest;
return xs;
}
void
s_free(slist* xs)
{
if (xs == 0) {
return;
}
xs->refs -= 1;
if (xs->refs == 0) {
s_free(xs->next);
free(xs->data);
free(xs);
}
}
slist*
s_split(const char* text, char delim)
{
if (*text == 0) {
return 0;
}
int plen = 0;
while (text[plen] != 0 && text[plen] != delim) {
plen += 1;
}
int skip = 0;
if (text[plen] == delim) {
skip = 1;
}
slist* rest = s_split(text + plen + skip, delim);
char* part = alloca(plen + 2);
memcpy(part, text, plen);
part[plen] = 0;
return s_cons(part, rest);
}
slist*
s_concat(slist* xs, slist* ys)
{
if (xs == 0) {
if (ys == 0) {
return 0;
}
else {
return s_concat(ys, 0);
}
}
return s_cons(xs->data, s_concat(xs->next, ys));
}
slist*
s_cat_free(slist* xs, slist* ys)
{
slist* zs = s_concat(xs, ys);
s_free(xs);
s_free(ys);
return zs;
}
slist*
s_copy(slist* xs)
{
return s_concat(xs, 0);
}
slist*
s_reverse(slist* xs)
{
slist* ys = 0;
for (; xs; xs = xs->next) {
ys = s_cons(xs->data, ys);
}
return ys;
}
slist*
s_rev_free(slist* xs)
{
slist* ys = s_reverse(xs);
s_free(xs);
return ys;
}