blob: f4438fde0f66525722766d9b4dbaaf7116446f3d (
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
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81== Goal
A memory-safe language with a simple translator to x86 that can be feasibly written in x86.
== Definitions of terms
Memory-safe: it should be impossible to:
a) create a pointer out of arbitrary data, or
b) to access heap memory after it's been freed.
Simple: do all the work in a 2-pass translator:
Pass 1: check each instruction's types in isolation.
Pass 2: emit code for each instruction in isolation.
== types
int
char
(address _ stack|heap|global)
(array _ n) # on stack or global
(ref _)
(ref array _) # by definition always on the heap
addresses to global can be saved and manipulated as usual
addresses on stack can't be saved to heap
addresses on heap can't be saved to global (use ref)
addresses to stack or heap can't be included in compound types
or used across a call
or used across a label
<reg x> : (address T stack|global) <- advance <reg/mem> : (array T), <reg offset> : (index T)
<reg x> : (address T heap) <- advance *<mem> : (ref array T), <reg offset> : (index T)
arrays require a size
(ref array _) may not include a size
Arguments of type 'address' are required to be on the stack or global. Can't
be on the heap.
So we need duplication for address and ref arguments?
Argv has type (array (address (array char) global))
variables on stack, heap and global are references. The name points at the
address. Use '*' to get at the value.
|