-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathsymbol_table.go
105 lines (86 loc) · 1.57 KB
/
symbol_table.go
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
package expr
type symbolTable struct {
tokens map[int]string
items []*item
}
func (st *symbolTable) addSymbol(symbol []byte, token int) {
st.tokens[token] = string(symbol)
for _, item := range st.items {
if item.add(symbol, token) {
return
}
}
st.items = append(st.items, newItem(symbol, token))
}
func (st *symbolTable) findToken(value []byte) (int, bool) {
maybe := false
for _, item := range st.items {
token, ok := item.find(value)
if token > 0 {
return token, ok
}
if ok {
maybe = true
}
}
return 0, maybe
}
type item struct {
char byte
token int
items []*item
}
func newItem(symbol []byte, token int) *item {
var v *item
var root *item
for idx, c := range symbol {
if v == nil {
v = &item{char: c}
root = v
} else {
v.items = append(v.items, &item{char: c})
v = v.items[0]
}
if idx == len(symbol)-1 {
v.token = token
}
}
return root
}
func (i *item) add(symbol []byte, token int) bool {
if i.char != symbol[0] {
return false
}
if len(symbol) == 1 {
i.token = token
return true
}
symbol = symbol[1:]
for _, item := range i.items {
if item.add(symbol, token) {
return true
}
}
i.items = append(i.items, newItem(symbol, token))
return true
}
func (i *item) find(symbol []byte) (int, bool) {
if i.char != symbol[0] {
return 0, false
}
if len(symbol) == 1 {
return i.token, len(i.items) > 0
}
maybe := false
symbol = symbol[1:]
for _, item := range i.items {
token, ok := item.find(symbol)
if token > 0 {
return token, ok
}
if ok {
maybe = ok
}
}
return 0, maybe
}