2017-02-16 17:53:27 +00:00
|
|
|
package vm
|
|
|
|
|
|
|
|
import (
|
|
|
|
. "testing"
|
|
|
|
|
|
|
|
"github.com/mediocregopher/ginger/lang"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestCompiler(t *T) {
|
|
|
|
mkcmd := func(a lang.Atom, args ...lang.Term) lang.Tuple {
|
2017-02-16 18:37:31 +00:00
|
|
|
// TODO a 1-tuple should be the same as its element?
|
2017-02-16 17:53:27 +00:00
|
|
|
if len(args) == 1 {
|
|
|
|
return lang.Tuple{a, args[0]}
|
|
|
|
}
|
|
|
|
return lang.Tuple{a, lang.Tuple(args)}
|
|
|
|
}
|
|
|
|
mkint := func(i string) lang.Tuple {
|
|
|
|
return lang.Tuple{Int, lang.Const(i)}
|
|
|
|
}
|
|
|
|
|
|
|
|
type test struct {
|
|
|
|
in []lang.Term
|
|
|
|
exp uint64
|
|
|
|
}
|
|
|
|
|
|
|
|
one := mkint("1")
|
|
|
|
two := mkint("2")
|
2017-02-16 18:37:31 +00:00
|
|
|
foo := mkcmd(Var, lang.Atom("foo"))
|
|
|
|
bar := mkcmd(Var, lang.Atom("bar"))
|
|
|
|
baz := mkcmd(Var, lang.Atom("baz"))
|
2017-02-16 17:53:27 +00:00
|
|
|
|
|
|
|
tests := []test{
|
|
|
|
{
|
|
|
|
in: []lang.Term{one},
|
|
|
|
exp: 1,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
in: []lang.Term{
|
|
|
|
mkcmd(Add, mkcmd(Tuple, one, two)),
|
|
|
|
},
|
|
|
|
exp: 3,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
in: []lang.Term{
|
|
|
|
mkcmd(Assign, foo, one),
|
2017-02-16 18:37:31 +00:00
|
|
|
mkcmd(Add, mkcmd(Tuple, foo, two)),
|
2017-02-16 17:53:27 +00:00
|
|
|
},
|
|
|
|
exp: 3,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
in: []lang.Term{
|
|
|
|
mkcmd(Assign, foo, mkcmd(Tuple, one, two)),
|
2017-02-16 18:37:31 +00:00
|
|
|
mkcmd(Add, foo),
|
2017-02-16 17:53:27 +00:00
|
|
|
},
|
|
|
|
exp: 3,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
in: []lang.Term{
|
|
|
|
mkcmd(Assign, foo, mkcmd(Tuple, one, two)),
|
2017-02-16 18:37:31 +00:00
|
|
|
mkcmd(Assign, bar, mkcmd(Add, foo)),
|
|
|
|
mkcmd(Assign, baz, mkcmd(Add, foo)),
|
|
|
|
mkcmd(Add, mkcmd(Tuple, bar, baz)),
|
2017-02-16 17:53:27 +00:00
|
|
|
},
|
|
|
|
exp: 6,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range tests {
|
|
|
|
t.Logf("testing program: %v", test.in)
|
|
|
|
mod, err := Build(test.in...)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("building failed: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
out, err := mod.Run()
|
|
|
|
if err != nil {
|
|
|
|
mod.Dump()
|
|
|
|
t.Fatalf("running failed: %s", err)
|
|
|
|
} else if out != test.exp {
|
|
|
|
mod.Dump()
|
|
|
|
t.Fatalf("expected result %T:%v, got %T:%v", test.exp, test.exp, out, out)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|