package vm import ( "fmt" "log" "reflect" "testing" ) func init() { log.SetFlags(log.Lshortfile) } func TestVM(t *testing.T) { for _, test := range tests { test := test t.Run("", func(t *testing.T) { m := &Machine{} for _, v := range test.sym { m.Push(v) } m.PushCode(test.code...) if err := m.Run(); err != nil { t.Errorf("run error: %v", err) } t.Log(Vstring(m.mem)) if r := Vstring(m.mem[test.start:test.end]); r != test.mem { t.Errorf("got %v, want %v", r, test.mem) } }) } } func BenchmarkVM(b *testing.B) { for _, test := range tests { test := test b.Run("", func(b *testing.B) { for i := 0; i < b.N; i++ { b.StopTimer() m := &Machine{} m.PushCode(test.code...) b.StartTimer() if err := m.Run(); err != nil { b.Errorf("run error: %v", err) } } }) } } var tests = []struct { sym []Value // initial memory values code []Instruction // bytecode to execute start, end int // mem string // expected memory content }{{ // #00 -- A simple addition. code: []Instruction{ {Op: Push, Arg: []int{1}}, {Op: Push, Arg: []int{2}}, {Op: Add}, {Op: Exit}, }, start: 0, end: 1, mem: "[3]", }, { // #01 -- A simple subtraction. code: []Instruction{ {Op: Push, Arg: []int{2}}, {Op: Push, Arg: []int{3}}, {Op: Sub}, {Op: Exit}, }, start: 0, end: 1, mem: "[1]", }, { // #02 -- A simple multiplication. code: []Instruction{ {Op: Push, Arg: []int{3}}, {Op: Push, Arg: []int{2}}, {Op: Mul}, {Op: Exit}, }, start: 0, end: 1, mem: "[6]", }, { // #03 -- lower. code: []Instruction{ {Op: Push, Arg: []int{3}}, {Op: Push, Arg: []int{2}}, {Op: Lower}, {Op: Exit}, }, start: 0, end: 1, mem: "[true]", }, { // #04 -- greater. code: []Instruction{ {Op: Push, Arg: []int{2}}, {Op: Push, Arg: []int{3}}, {Op: Greater}, {Op: Exit}, }, start: 0, end: 1, mem: "[true]", }, { // #05 -- equal. code: []Instruction{ {Op: Push, Arg: []int{2}}, {Op: Push, Arg: []int{3}}, {Op: Equal}, {Op: Exit}, }, start: 0, end: 1, mem: "[false]", }, { // #06 -- equalSet. code: []Instruction{ {Op: Push, Arg: []int{2}}, {Op: Push, Arg: []int{3}}, {Op: EqualSet}, {Op: Exit}, }, start: 0, end: 2, mem: "[2 false]", }, { // #07 -- equalSet. code: []Instruction{ {Op: Push, Arg: []int{3}}, {Op: Push, Arg: []int{3}}, {Op: EqualSet}, {Op: Exit}, }, start: 0, end: 1, mem: "[true]", }, { // #08 not. code: []Instruction{ {Op: Push, Arg: []int{3}}, {Op: Push, Arg: []int{3}}, {Op: Equal}, {Op: Not}, {Op: Exit}, }, start: 0, end: 1, mem: "[false]", }, { // #09 pop. code: []Instruction{ {Op: Push, Arg: []int{3}}, {Op: Push, Arg: []int{2}}, {Op: Pop, Arg: []int{1}}, {Op: Exit}, }, start: 0, end: 1, mem: "[3]", }, { // #10 -- Assign a variable. sym: []Value{{Type: TypeOf(0), Value: reflect.ValueOf(0)}}, code: []Instruction{ {Op: Grow, Arg: []int{1}}, {Op: New, Arg: []int{2, 0}}, {Op: Push, Arg: []int{2}}, {Op: Assign, Arg: []int{1}}, {Op: Exit}, }, start: 1, end: 2, mem: "[2]", }, { // #11 -- Calling a function defined outside the VM. sym: []Value{ValueOf(fmt.Println), ValueOf("Hello")}, code: []Instruction{ {Op: Dup, Arg: []int{0}}, {Op: CallX, Arg: []int{1}}, {Op: Exit}, }, start: 1, end: 3, mem: "[6 ]", }, { // #12 -- Defining and calling a function in VM. code: []Instruction{ {Op: Jump, Arg: []int{3}}, // 0 {Op: Push, Arg: []int{3}}, // 1 {Op: Return, Arg: []int{1, 1}}, // 2 {Op: Push, Arg: []int{1}}, // 3 {Op: Calli, Arg: []int{1}}, // 4 {Op: Exit}, // 5 }, start: 0, end: 1, mem: "[3]", }, { // #13 -- Defining and calling a function in VM. code: []Instruction{ {Op: Jump, Arg: []int{3}}, // 0 {Op: Push, Arg: []int{3}}, // 1 {Op: Return, Arg: []int{1, 1}}, // 2 {Op: Push, Arg: []int{1}}, // 3 {Op: Push, Arg: []int{1}}, // 4 {Op: Call}, // 5 {Op: Exit}, // 6 }, start: 0, end: 1, mem: "[3]", }, { // #14 -- Defining and calling a function in VM. code: []Instruction{ {Op: Jump, Arg: []int{5}}, // 0 {Op: Push, Arg: []int{3}}, // 1 {Op: Fassign, Arg: []int{-2}}, // 2 {Op: Fdup, Arg: []int{-2}}, // 3 {Op: Return, Arg: []int{1, 1}}, // 4 {Op: Push, Arg: []int{1}}, // 5 {Op: Push, Arg: []int{1}}, // 6 {Op: Call}, // 7 {Op: Exit}, // 8 }, start: 0, end: 1, mem: "[3]", }, { // #15 -- Fibonacci numbers, hand written. Showcase recursivity. code: []Instruction{ {Op: Jump, Arg: []int{19}}, // 0 {Op: Push, Arg: []int{2}}, // 2 [2] {Op: Fdup, Arg: []int{-2}}, // 1 [2 i] {Op: Lower}, // 3 [true/false] {Op: JumpTrue, Arg: []int{13}}, // 4 [], goto 17 {Op: Push, Arg: []int{2}}, // 5 [i 2] {Op: Fdup, Arg: []int{-2}}, // 6 [i] {Op: Sub}, // 7 [(i-2)] {Op: Push, Arg: []int{1}}, // 8 {Op: Call}, // 9 [fib(i-2)] {Op: Push, Arg: []int{1}}, // 10 [(i-2) i 1] {Op: Fdup, Arg: []int{-2}}, // 11 [fib(i-2) i] {Op: Sub}, // 12 [(i-2) (i-1)] {Op: Push, Arg: []int{1}}, // 13 {Op: Call}, // 14 [fib(i-2) fib(i-1)] {Op: Add}, // 15 [fib(i-2)+fib(i-1)] {Op: Return, Arg: []int{1, 1}}, // 16 return i {Op: Fdup, Arg: []int{-2}}, // 17 [i] {Op: Return, Arg: []int{1, 1}}, // 18 return i {Op: Push, Arg: []int{6}}, // 19 [1] {Op: Push, Arg: []int{1}}, // 20 {Op: Call}, // 21 [fib(*1)] {Op: Exit}, // 22 }, start: 0, end: 1, mem: "[8]", }, { // #16 -- Fibonacci with some immediate instructions. code: []Instruction{ {Op: Jump, Arg: []int{14}}, // 0 {Op: Fdup, Arg: []int{-2}}, // 1 [i] {Op: Loweri, Arg: []int{2}}, // 2 [true/false] {Op: JumpTrue, Arg: []int{9}}, // 3 [], goto 12 {Op: Fdup, Arg: []int{-2}}, // 4 [i] {Op: Subi, Arg: []int{2}}, // 5 [(i-2)] {Op: Calli, Arg: []int{1}}, // 6 [fib(i-2)] {Op: Fdup, Arg: []int{-2}}, // 7 [fib(i-2) i] {Op: Subi, Arg: []int{1}}, // 8 [(i-2) (i-1)] {Op: Calli, Arg: []int{1}}, // 9 [fib(i-2) fib(i-1)], call 1 {Op: Add}, // 10 [fib(i-2)+fib(i-1)] {Op: Return, Arg: []int{1, 1}}, // 11 return i {Op: Fdup, Arg: []int{-2}}, // 12 [i] {Op: Return, Arg: []int{1, 1}}, // 13 return i {Op: Push, Arg: []int{6}}, // 14 [1] {Op: Calli, Arg: []int{1}}, // 15 [fib(*1)], call 1 {Op: Exit}, // 16 }, start: 0, end: 1, mem: "[8]", }}