2019-01-09 10:17:42 +03:00
|
|
|
package compiler
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/gob"
|
2019-02-11 08:44:22 +03:00
|
|
|
"fmt"
|
2019-01-09 10:17:42 +03:00
|
|
|
"io"
|
2019-02-11 08:44:22 +03:00
|
|
|
"reflect"
|
2019-01-09 10:17:42 +03:00
|
|
|
|
2019-02-21 03:26:11 +03:00
|
|
|
"github.com/d5/tengo/compiler/source"
|
2019-01-09 10:17:42 +03:00
|
|
|
"github.com/d5/tengo/objects"
|
|
|
|
)
|
|
|
|
|
2019-01-15 09:24:33 +03:00
|
|
|
// Bytecode is a compiled instructions and constants.
|
2019-01-09 10:17:42 +03:00
|
|
|
type Bytecode struct {
|
2019-02-21 03:26:11 +03:00
|
|
|
FileSet *source.FileSet
|
|
|
|
MainFunction *objects.CompiledFunction
|
2019-01-09 10:17:42 +03:00
|
|
|
Constants []objects.Object
|
|
|
|
}
|
|
|
|
|
2019-01-15 09:24:33 +03:00
|
|
|
// Encode writes Bytecode data to the writer.
|
2019-01-09 10:17:42 +03:00
|
|
|
func (b *Bytecode) Encode(w io.Writer) error {
|
|
|
|
enc := gob.NewEncoder(w)
|
|
|
|
|
2019-02-21 03:26:11 +03:00
|
|
|
if err := enc.Encode(b.FileSet); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := enc.Encode(b.MainFunction); err != nil {
|
2019-01-09 10:17:42 +03:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-01-29 00:09:40 +03:00
|
|
|
// constants
|
2019-01-15 21:14:16 +03:00
|
|
|
return enc.Encode(b.Constants)
|
2019-01-09 10:17:42 +03:00
|
|
|
}
|
|
|
|
|
2019-03-07 04:20:05 +03:00
|
|
|
// CountObjects returns the number of objects found in Constants.
|
|
|
|
func (b *Bytecode) CountObjects() int {
|
|
|
|
n := 0
|
|
|
|
|
|
|
|
for _, c := range b.Constants {
|
|
|
|
n += objects.CountObjects(c)
|
|
|
|
}
|
|
|
|
|
|
|
|
return n
|
|
|
|
}
|
|
|
|
|
2019-02-11 08:44:22 +03:00
|
|
|
// FormatInstructions returns human readable string representations of
|
|
|
|
// compiled instructions.
|
|
|
|
func (b *Bytecode) FormatInstructions() []string {
|
2019-02-21 03:26:11 +03:00
|
|
|
return FormatInstructions(b.MainFunction.Instructions, 0)
|
2019-02-11 08:44:22 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// FormatConstants returns human readable string representations of
|
|
|
|
// compiled constants.
|
|
|
|
func (b *Bytecode) FormatConstants() (output []string) {
|
|
|
|
for cidx, cn := range b.Constants {
|
|
|
|
switch cn := cn.(type) {
|
|
|
|
case *objects.CompiledFunction:
|
|
|
|
output = append(output, fmt.Sprintf("[% 3d] (Compiled Function|%p)", cidx, &cn))
|
|
|
|
for _, l := range FormatInstructions(cn.Instructions, 0) {
|
|
|
|
output = append(output, fmt.Sprintf(" %s", l))
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
output = append(output, fmt.Sprintf("[% 3d] %s (%s|%p)", cidx, cn, reflect.TypeOf(cn).Elem().Name(), &cn))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-01-09 10:17:42 +03:00
|
|
|
func init() {
|
2019-02-21 03:26:11 +03:00
|
|
|
gob.Register(&source.FileSet{})
|
|
|
|
gob.Register(&source.File{})
|
2019-03-19 19:43:03 +03:00
|
|
|
gob.Register(&objects.Array{})
|
|
|
|
gob.Register(&objects.Bool{})
|
|
|
|
gob.Register(&objects.Bytes{})
|
2019-01-09 10:17:42 +03:00
|
|
|
gob.Register(&objects.Char{})
|
2019-03-19 19:43:03 +03:00
|
|
|
gob.Register(&objects.Closure{})
|
2019-01-09 10:17:42 +03:00
|
|
|
gob.Register(&objects.CompiledFunction{})
|
2019-03-19 19:43:03 +03:00
|
|
|
gob.Register(&objects.Error{})
|
2019-02-21 03:26:11 +03:00
|
|
|
gob.Register(&objects.Float{})
|
2019-03-19 19:43:03 +03:00
|
|
|
gob.Register(&objects.ImmutableArray{})
|
|
|
|
gob.Register(&objects.ImmutableMap{})
|
2019-02-21 03:26:11 +03:00
|
|
|
gob.Register(&objects.Int{})
|
2019-03-19 19:43:03 +03:00
|
|
|
gob.Register(&objects.Map{})
|
2019-02-21 03:26:11 +03:00
|
|
|
gob.Register(&objects.String{})
|
2019-03-19 19:43:03 +03:00
|
|
|
gob.Register(&objects.Time{})
|
|
|
|
gob.Register(&objects.Undefined{})
|
|
|
|
gob.Register(&objects.UserFunction{})
|
2019-01-09 10:17:42 +03:00
|
|
|
}
|