-
Notifications
You must be signed in to change notification settings - Fork 1
/
stringtable.go
95 lines (86 loc) · 2.29 KB
/
stringtable.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
// Copyright 2021 Cloud Privacy Labs, LLC
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package lpg
import (
// "fmt"
)
// stringTable keeps unique strings in a slice indexed by a map, so
// identical strings are not repeated, and integers can be used
// instead of strings.
type stringTable struct {
strs []rcstring
strmap map[string]int
firstFree int
}
type rcstring struct {
str string
c int
}
func (table *stringTable) init() {
table.strs = make([]rcstring, 0, 16)
table.strmap = make(map[string]int)
table.firstFree = -1
}
// allocate will allocate a string and return its index
func (table *stringTable) allocate(s string) int {
// defer func() {
// fmt.Printf("allocate %p %s %v\n", table, s, table)
// }()
// If string exists, increment reference count
i, exists := table.strmap[s]
if exists {
table.strs[i].c++
return i
}
if table.firstFree != -1 {
ret := table.firstFree
table.firstFree = table.strs[ret].c
table.strs[ret].c = 1
table.strs[ret].str = s
table.strmap[s] = ret
return ret
}
ret := len(table.strs)
table.strmap[s] = ret
table.strs = append(table.strs, rcstring{str: s, c: 1})
return ret
}
// lookup will lookup a string index without allocating
func (table *stringTable) lookup(s string) (int, bool) {
i, exists := table.strmap[s]
if exists {
return i, true
}
return 0, false
}
func (table *stringTable) str(i int) string {
// defer func() {
// if err := recover(); err != nil {
// fmt.Printf("***panic, table: %p %v i %d\n", table, table, i)
// }
// }()
return table.strs[i].str
}
func (table *stringTable) free(i int) {
// defer func() {
// fmt.Println("tree", i, table)
// }()
table.strs[i].c--
if table.strs[i].c > 0 {
return
}
table.strs[i].c = table.firstFree
table.firstFree = i
delete(table.strmap, table.strs[i].str)
}