2019-01-09 10:17:42 +03:00
|
|
|
package compiler
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/gob"
|
|
|
|
"io"
|
|
|
|
|
|
|
|
"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 {
|
|
|
|
Instructions []byte
|
|
|
|
Constants []objects.Object
|
|
|
|
}
|
|
|
|
|
2019-01-15 09:24:33 +03:00
|
|
|
// Decode reads Bytecode data from the reader.
|
2019-01-09 10:17:42 +03:00
|
|
|
func (b *Bytecode) Decode(r io.Reader) error {
|
|
|
|
dec := gob.NewDecoder(r)
|
|
|
|
|
|
|
|
if err := dec.Decode(&b.Instructions); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-01-29 00:09:40 +03:00
|
|
|
if err := dec.Decode(&b.Constants); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// replace Bool and Undefined with known value
|
|
|
|
for i, v := range b.Constants {
|
|
|
|
b.Constants[i] = cleanupObjects(v)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
2019-01-09 10:17:42 +03:00
|
|
|
}
|
|
|
|
|
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)
|
|
|
|
|
|
|
|
if err := enc.Encode(b.Instructions); err != nil {
|
|
|
|
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-01-29 00:09:40 +03:00
|
|
|
func cleanupObjects(o objects.Object) objects.Object {
|
|
|
|
switch o := o.(type) {
|
|
|
|
case *objects.Bool:
|
|
|
|
if o.IsFalsy() {
|
|
|
|
return objects.FalseValue
|
|
|
|
}
|
|
|
|
return objects.TrueValue
|
|
|
|
case *objects.Undefined:
|
|
|
|
return objects.UndefinedValue
|
|
|
|
case *objects.Array:
|
|
|
|
for i, v := range o.Value {
|
|
|
|
o.Value[i] = cleanupObjects(v)
|
|
|
|
}
|
|
|
|
case *objects.Map:
|
|
|
|
for k, v := range o.Value {
|
|
|
|
o.Value[k] = cleanupObjects(v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return o
|
|
|
|
}
|
|
|
|
|
2019-01-09 10:17:42 +03:00
|
|
|
func init() {
|
|
|
|
gob.Register(&objects.Int{})
|
|
|
|
gob.Register(&objects.Float{})
|
|
|
|
gob.Register(&objects.String{})
|
|
|
|
gob.Register(&objects.Bool{})
|
|
|
|
gob.Register(&objects.Char{})
|
|
|
|
gob.Register(&objects.Array{})
|
2019-01-26 02:08:17 +03:00
|
|
|
gob.Register(&objects.ImmutableArray{})
|
2019-01-09 10:17:42 +03:00
|
|
|
gob.Register(&objects.Map{})
|
2019-01-26 02:08:17 +03:00
|
|
|
gob.Register(&objects.ImmutableMap{})
|
2019-01-09 10:17:42 +03:00
|
|
|
gob.Register(&objects.CompiledFunction{})
|
|
|
|
gob.Register(&objects.Undefined{})
|
2019-01-18 12:43:46 +03:00
|
|
|
gob.Register(&objects.Error{})
|
|
|
|
gob.Register(&objects.Bytes{})
|
|
|
|
gob.Register(&objects.StringIterator{})
|
|
|
|
gob.Register(&objects.MapIterator{})
|
|
|
|
gob.Register(&objects.ArrayIterator{})
|
2019-01-09 10:17:42 +03:00
|
|
|
}
|