summaryrefslogtreecommitdiff
path: root/symbol.c
blob: 294d9a41e795355119f150f321d9fb04dfac6f0f (plain) (blame)
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
#include <u.h>
#include <libc.h>
#include <bio.h>

#include "apl9.h"

Symtab *globalsymtab;
Symtab *currentsymtab;

Symbol *
getsym(Symtab *tab, Rune *name)
{
	for(int i = 0; i < tab->nsyms; i++)
		if(runestrcmp(tab->syms[i]->name, name) == 0)
			return tab->syms[i];
	
	tab->nsyms++;
	tab->syms = realloc(tab->syms, sizeof(Symbol *) * tab->nsyms);
	tab->syms[tab->nsyms-1] = malloc(sizeof(Symbol));
	tab->syms[tab->nsyms-1]->name = runestrdup(name);
	tab->syms[tab->nsyms-1]->undefined = 1;
	return tab->syms[tab->nsyms-1];
}

Symtab *
newsymtab(void)
{
	Symtab *tab = malloc(sizeof(Symtab));
	tab->nsyms = 0;
	tab->syms = nil;

	Symbol *io = getsym(tab, L"⎕IO");
	io->value.tag = ArrayTag;
	io->value.array = mkscalarint(currentsymtab ? globalIO() : 1);
	io->value.shy = 0;
	io->undefined = 0;

	return tab;
}

void
freesymtab(Symtab *tab)
{
	print("Freeing symtab\n");
	int i;
	for(i = 0; i < tab->nsyms; i++){
		Symbol *s = tab->syms[i];
		if(s->undefined == 0 && s->value.tag == ArrayTag)
			freearray(s->value.array);
	}
	free(tab->syms);
	free(tab);
	print("Done freeing symtab\n");
}

vlong
globalIO(void)
{
	Symbol *s = getsym(currentsymtab, L"⎕IO");
	return s->value.array->intdata[0];
}