-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathenvironment.c
238 lines (193 loc) · 4.88 KB
/
environment.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
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
#include <stdio.h>
#include <math.h>
#include <stdlib.h>
#include <string.h>
#include "sexpr.h"
#include "environment.h"
#include "util.h"
sym* sym_new(char *name, sexpr* e) {
sym *b = malloc(sizeof(sym));
b->val = e;
b->next = NULL;
b->name = n_strcpy(b->name, name);
return b;
}
void sym_free(sym* b) {
free(b->name);
/* I don't think I'll want to do this once garbage collection is a thing.
All the sym table entries should be pure references to things on the
heap. */
sym *n, *p = b->next;
while (p) {
free(p->name);
n = p->next;
free(p);
p = n;
}
free(b);
}
scope* scope_new(unsigned int size) {
scope *e = malloc(sizeof(scope));
e->sym_table = calloc(size, sizeof(sym*));
e->size = size;
e->parent = NULL;
return e;
}
void scope_free(scope *sc) {
for (unsigned int j = 0; j < sc->size; j++) {
if (sc->sym_table[j])
sym_free(sc->sym_table[j]);
}
free(sc->sym_table);
free(sc);
}
int bt_hash(unsigned int size, char *s) {
long h = 0;
int len = strlen(s);
for (int j = 0; j < len; j++) {
h += (long)pow(163, len - (j+1)) * s[j];
h %= size;
}
return h;
}
void scope_insert_var(scope* sc, char *name, sexpr *exp) {
unsigned int h = bt_hash(sc->size, name);
sym *s = sym_new(name, exp);
if (sc->sym_table[h] == NULL) {
/* Easy case! Just create the new entry and stick it there */
sc->sym_table[h] = s;
}
else {
/* A rebinding of the same name in the same scope should free up the
previous binding */
sym *existing = sc->sym_table[h];
while (existing) {
if (strcmp(existing->name, name) == 0) {
existing->val = s->val;
return;
}
existing = existing->next;
}
s->next = sc->sym_table[h];
sc->sym_table[h] = s;
}
}
void scope_insert_global_var(scope *sc, char *name, sexpr *exp) {
while (sc->parent)
sc = sc->parent;
scope_insert_var(sc, name, exp);
}
sexpr* scope_fetch_var(vm_heap *vm, scope *sc, char* key) {
int h = bt_hash(sc->size, key);
if (!sc->sym_table[h]) {
char msg[256];
snprintf(msg, sizeof msg, "%s'%s'", "Unbound symbol: ", key);
sexpr *r = CHECK_PARENT_SCOPE(vm, sc, key, msg);
return r;
}
sym *b = sc->sym_table[h];
/* Assuming a variable called is probably going to be
called again soon, it might be worth moving it to
the top of the chain. But that's more complicated
code */
while (b && strcmp(b->name, key) != 0)
b = b->next;
if (!b) {
char msg[256];
snprintf(msg, sizeof msg, "%s%s", "Unbound symbol: ", key);
return CHECK_PARENT_SCOPE(vm, sc, key, msg);
}
b->val->global_scope = sc->parent ? 0 : 1;
return b->val;
}
void env_dump(vm_heap *vm, scope* env) {
sym *b;
puts("Binding for current scope:");
for (unsigned int j = 0; j < env->size; j++) {
b = env->sym_table[j];
if (b) {
printf(" %s ", b->name);
print_sexpr_type(b->val);
putchar(' ');
sexpr_pprint(b->val);
putchar('\n');
}
}
printf("\nHeap items:\n");
printf("Heap count: %lu\n", vm->count);
}
vm_heap* vm_new(void) {
vm_heap *vm = malloc(sizeof(vm_heap));
vm->count = 0;
vm->gc_generation = 0;
vm->heap = NULL;
return vm;
}
void vm_add(vm_heap* vm, sexpr* expr) {
expr->neighbour = vm->heap;
vm->heap = expr;
vm->count++;
}
void vm_free(vm_heap *vm) {
sexpr *node;
while (vm->heap) {
node = vm->heap;
vm->heap = vm->heap->neighbour;
sexpr_free(node);
}
}
void mark_chain(vm_heap* vm, sexpr *chain) {
/* Bailing out if it has been marked avoids cycles in the graph of
connection objects */
if (chain->gen == vm->gc_generation)
return;
if (chain->type == LVAL_LIST && chain->count > 0) {
for (int j = 0; j < chain->count; j++)
mark_chain(vm, chain->children[j]);
}
else if (chain->type == LVAL_FUN && !chain->builtin) {
mark_chain(vm, chain->params);
mark_chain(vm, chain->body);
}
chain->gen = vm->gc_generation;
}
/* The garbage collector is a simple mark-and-sweep algorithm. Loop through
the symbol table and mark off any s-expressions that are still in use,
then loop over the VM's heap linked list and prune any items whose
generation count is less than current generation.
Note -- built-in functions are stored in the symbol table but they aren't
in the heap so they won't be deleted by the garbage collector */
void gc_run(vm_heap* vm, scope* env) {
vm->gc_generation++;
/* Need to pass over the entire symbol table and mark which objects
are still referenced. Don't bother marking built-ins because we are
never going to recycle them. */
for (unsigned int j = 0; j < env->size; j++) {
sym *s = env->sym_table[j];
while (s) {
mark_chain(vm, s->val);
s = s->next;
}
}
sexpr *dead, *prev = NULL;
sexpr *h = vm->heap;
int swept = 0;
while (h) {
if (h->gen < vm->gc_generation) {
dead = h;
if (!prev)
vm->heap = h->neighbour;
else
prev->neighbour = h->neighbour;
h = h->neighbour;
sexpr_free(dead);
--vm->count;
++swept;
}
else {
prev = h;
h = h->neighbour;
}
}
printf("%d s-exprs deleted.\n", swept);
}