xgo/compiler/symbol_table.go

146 lines
2.9 KiB
Go
Raw Normal View History

2019-01-09 10:17:42 +03:00
package compiler
// SymbolTable represents a symbol table.
2019-01-09 10:17:42 +03:00
type SymbolTable struct {
parent *SymbolTable
block bool
2019-02-02 10:27:29 +03:00
store map[string]*Symbol
2019-01-09 10:17:42 +03:00
numDefinition int
maxDefinition int
2019-02-02 10:27:29 +03:00
freeSymbols []*Symbol
2019-01-09 10:17:42 +03:00
}
// NewSymbolTable creates a SymbolTable.
2019-01-09 10:17:42 +03:00
func NewSymbolTable() *SymbolTable {
return &SymbolTable{
2019-02-02 10:27:29 +03:00
store: make(map[string]*Symbol),
2019-01-09 10:17:42 +03:00
}
}
// Define adds a new symbol in the current scope.
2019-02-02 10:27:29 +03:00
func (t *SymbolTable) Define(name string) *Symbol {
symbol := &Symbol{Name: name, Index: t.nextIndex()}
2019-01-09 10:17:42 +03:00
t.numDefinition++
if t.Parent(true) == nil {
symbol.Scope = ScopeGlobal
} else {
symbol.Scope = ScopeLocal
}
t.store[name] = symbol
t.updateMaxDefs(symbol.Index + 1)
return symbol
}
// DefineBuiltin adds a symbol for builtin function.
2019-02-02 10:27:29 +03:00
func (t *SymbolTable) DefineBuiltin(index int, name string) *Symbol {
symbol := &Symbol{
2019-01-09 10:17:42 +03:00
Name: name,
Index: index,
Scope: ScopeBuiltin,
}
t.store[name] = symbol
return symbol
}
// Resolve resolves a symbol with a given name.
2019-02-02 10:27:29 +03:00
func (t *SymbolTable) Resolve(name string) (symbol *Symbol, depth int, ok bool) {
2019-01-09 10:17:42 +03:00
symbol, ok = t.store[name]
if !ok && t.parent != nil {
symbol, depth, ok = t.parent.Resolve(name)
if !ok {
return
}
if !t.block {
depth++
2019-01-09 10:17:42 +03:00
}
// if symbol is defined in parent table and if it's not global/builtin
// then it's free variable.
if depth > 0 && symbol.Scope != ScopeGlobal && symbol.Scope != ScopeBuiltin {
return t.defineFree(symbol), depth, true
}
return
}
return
}
// Fork creates a new symbol table for a new scope.
2019-01-09 10:17:42 +03:00
func (t *SymbolTable) Fork(block bool) *SymbolTable {
return &SymbolTable{
2019-02-02 10:27:29 +03:00
store: make(map[string]*Symbol),
2019-01-09 10:17:42 +03:00
parent: t,
block: block,
}
}
// Parent returns the outer scope of the current symbol table.
2019-01-09 10:17:42 +03:00
func (t *SymbolTable) Parent(skipBlock bool) *SymbolTable {
if skipBlock && t.block {
return t.parent.Parent(skipBlock)
}
return t.parent
}
// MaxSymbols returns the total number of symbols defined in the scope.
2019-01-09 10:17:42 +03:00
func (t *SymbolTable) MaxSymbols() int {
return t.maxDefinition
}
// FreeSymbols returns free symbols for the scope.
2019-02-02 10:27:29 +03:00
func (t *SymbolTable) FreeSymbols() []*Symbol {
2019-01-09 10:17:42 +03:00
return t.freeSymbols
}
// Names returns the name of all the symbols.
2019-01-09 10:17:42 +03:00
func (t *SymbolTable) Names() []string {
var names []string
for name := range t.store {
names = append(names, name)
}
return names
}
func (t *SymbolTable) nextIndex() int {
if t.block {
return t.parent.nextIndex() + t.numDefinition
}
return t.numDefinition
}
func (t *SymbolTable) updateMaxDefs(numDefs int) {
if numDefs > t.maxDefinition {
t.maxDefinition = numDefs
}
if t.block {
t.parent.updateMaxDefs(numDefs)
}
}
2019-02-02 10:27:29 +03:00
func (t *SymbolTable) defineFree(original *Symbol) *Symbol {
2019-01-09 10:17:42 +03:00
// TODO: should we check duplicates?
t.freeSymbols = append(t.freeSymbols, original)
2019-02-02 10:27:29 +03:00
symbol := &Symbol{
2019-01-09 10:17:42 +03:00
Name: original.Name,
Index: len(t.freeSymbols) - 1,
Scope: ScopeFree,
}
t.store[original.Name] = symbol
return symbol
}