Each pointer needs its own stack, merging the two packages to avoid a circular dependency
This commit is contained in:
parent
e3bc1251e8
commit
759ee2aa10
8 changed files with 206 additions and 31 deletions
|
@ -25,7 +25,8 @@ func (i *Interpreter) Run() {
|
||||||
func (i *Interpreter) Step() {
|
func (i *Interpreter) Step() {
|
||||||
var prev *pointer.Pointer = nil
|
var prev *pointer.Pointer = nil
|
||||||
for p := i.p; p != nil; p = p.Next {
|
for p := i.p; p != nil; p = p.Next {
|
||||||
switch p.Get(*i.f) {
|
c := p.Get(*i.f)
|
||||||
|
switch c {
|
||||||
case '@':
|
case '@':
|
||||||
if prev == nil {
|
if prev == nil {
|
||||||
i.p = p.Next
|
i.p = p.Next
|
||||||
|
@ -33,8 +34,13 @@ func (i *Interpreter) Step() {
|
||||||
prev.Next = p.Next
|
prev.Next = p.Next
|
||||||
}
|
}
|
||||||
break
|
break
|
||||||
|
case '#':
|
||||||
|
p.Step(*i.f)
|
||||||
default:
|
default:
|
||||||
log.Fatalf("Non implemented instruction code %d : %c", p.Get(*i.f), p.Get(*i.f))
|
if !p.Redirect(c) {
|
||||||
|
log.Fatalf("Non implemented instruction code %d : %c", c, c)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
p.Step(*i.f)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
50
pkg/interpreter/interpreter_test.go
Normal file
50
pkg/interpreter/interpreter_test.go
Normal file
|
@ -0,0 +1,50 @@
|
||||||
|
package interpreter
|
||||||
|
|
||||||
|
import (
|
||||||
|
"os"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"git.adyxax.org/adyxax/gofunge/pkg/field"
|
||||||
|
"git.adyxax.org/adyxax/gofunge/pkg/pointer"
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestRun(t *testing.T) {
|
||||||
|
file, err := os.Open("../field/test_data/minimal.b98")
|
||||||
|
require.NoError(t, err)
|
||||||
|
defer file.Close()
|
||||||
|
f, err := field.Load(file)
|
||||||
|
require.NoError(t, err)
|
||||||
|
NewInterpreter(f, pointer.NewPointer()).Run()
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestStep(t *testing.T) {
|
||||||
|
testCases := []struct {
|
||||||
|
name string
|
||||||
|
filename string
|
||||||
|
pointer *pointer.Pointer
|
||||||
|
expectedField *field.Field
|
||||||
|
expectedPointer *pointer.Pointer
|
||||||
|
}{
|
||||||
|
{"minimal", "../field/test_data/minimal.b98", nil, nil, nil},
|
||||||
|
}
|
||||||
|
for _, tc := range testCases {
|
||||||
|
t.Run(tc.name, func(t *testing.T) {
|
||||||
|
file, err := os.Open(tc.filename)
|
||||||
|
require.NoError(t, err)
|
||||||
|
defer file.Close()
|
||||||
|
f, err := field.Load(file)
|
||||||
|
require.NoError(t, err)
|
||||||
|
if tc.pointer == nil {
|
||||||
|
tc.pointer = pointer.NewPointer()
|
||||||
|
}
|
||||||
|
NewInterpreter(f, tc.pointer).Step()
|
||||||
|
if tc.expectedField != nil {
|
||||||
|
require.Equal(t, tc.expectedField, f)
|
||||||
|
}
|
||||||
|
if tc.expectedPointer != nil {
|
||||||
|
require.Equal(t, tc.expectedPointer, tc.pointer)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,6 +1,10 @@
|
||||||
package pointer
|
package pointer
|
||||||
|
|
||||||
import "git.adyxax.org/adyxax/gofunge/pkg/field"
|
import (
|
||||||
|
"math/rand"
|
||||||
|
|
||||||
|
"git.adyxax.org/adyxax/gofunge/pkg/field"
|
||||||
|
)
|
||||||
|
|
||||||
type Pointer struct {
|
type Pointer struct {
|
||||||
// the position
|
// the position
|
||||||
|
@ -12,12 +16,14 @@ type Pointer struct {
|
||||||
// The Storage offset
|
// The Storage offset
|
||||||
sox int
|
sox int
|
||||||
soy int
|
soy int
|
||||||
|
// The stack
|
||||||
|
ss *StackStack
|
||||||
// The next element for the multi-"threaded" b98 interpreter
|
// The next element for the multi-"threaded" b98 interpreter
|
||||||
Next *Pointer
|
Next *Pointer
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewPointer() *Pointer {
|
func NewPointer() *Pointer {
|
||||||
return &Pointer{dx: 1}
|
return &Pointer{dx: 1, ss: NewStackStack()}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p Pointer) Split() *Pointer {
|
func (p Pointer) Split() *Pointer {
|
||||||
|
@ -31,3 +37,45 @@ func (p *Pointer) Step(f field.Field) {
|
||||||
func (p Pointer) Get(f field.Field) int {
|
func (p Pointer) Get(f field.Field) int {
|
||||||
return f.Get(p.x, p.y)
|
return f.Get(p.x, p.y)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (p *Pointer) Set(x, y int) {
|
||||||
|
p.x, p.y = x, y
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *Pointer) RedirectTo(dx, dy int) {
|
||||||
|
p.dx, p.dy = dx, dy
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *Pointer) Reverse() {
|
||||||
|
p.dx, p.dy = -p.dx, -p.dy
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *Pointer) Redirect(c int) bool {
|
||||||
|
switch c {
|
||||||
|
case '^':
|
||||||
|
p.dx, p.dy = 0, -1
|
||||||
|
case '>':
|
||||||
|
p.dx, p.dy = 1, 0
|
||||||
|
case 'v':
|
||||||
|
p.dx, p.dy = 0, 1
|
||||||
|
case '<':
|
||||||
|
p.dx, p.dy = -1, 0
|
||||||
|
case '?':
|
||||||
|
directions := []int{0, -1, 1, 0, 0, 1, -1, 0}
|
||||||
|
r := 2 * rand.Intn(4)
|
||||||
|
p.dx, p.dy = directions[r], directions[r+1]
|
||||||
|
case '[':
|
||||||
|
p.dx, p.dy = p.dy, -p.dx
|
||||||
|
case ']':
|
||||||
|
p.dx, p.dy = -p.dy, p.dx
|
||||||
|
case 'r':
|
||||||
|
p.Reverse()
|
||||||
|
case 'x':
|
||||||
|
dy := p.ss.Pop()
|
||||||
|
dx := p.ss.Pop()
|
||||||
|
p.RedirectTo(dx, dy)
|
||||||
|
default:
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
|
@ -9,7 +9,7 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestNewPointer(t *testing.T) {
|
func TestNewPointer(t *testing.T) {
|
||||||
require.Equal(t, NewPointer(), &Pointer{dx: 1})
|
require.Equal(t, NewPointer(), &Pointer{dx: 1, ss: NewStackStack()})
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestSplit(t *testing.T) {
|
func TestSplit(t *testing.T) {
|
||||||
|
@ -23,8 +23,8 @@ func TestSplit(t *testing.T) {
|
||||||
// We check that p2 is a real copy
|
// We check that p2 is a real copy
|
||||||
p.Step(*f)
|
p.Step(*f)
|
||||||
p2.Step(*f)
|
p2.Step(*f)
|
||||||
require.Equal(t, &Pointer{x: 1, y: 0, dx: 1}, p)
|
require.Equal(t, &Pointer{x: 1, y: 0, dx: 1, ss: NewStackStack()}, p)
|
||||||
require.Equal(t, &Pointer{x: 1, y: 0, dx: 1}, p2)
|
require.Equal(t, &Pointer{x: 1, y: 0, dx: 1, ss: NewStackStack()}, p2)
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestStep(t *testing.T) { // Step is thoroughly tested in the field package
|
func TestStep(t *testing.T) { // Step is thoroughly tested in the field package
|
||||||
|
@ -50,3 +50,61 @@ func TestGet(t *testing.T) {
|
||||||
v := p.Get(*f)
|
v := p.Get(*f)
|
||||||
require.Equal(t, int('@'), v)
|
require.Equal(t, int('@'), v)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestSet(t *testing.T) {
|
||||||
|
p := NewPointer()
|
||||||
|
p.Set(3, 14)
|
||||||
|
require.Equal(t, 3, p.x)
|
||||||
|
require.Equal(t, 14, p.y)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestRedirectTo(t *testing.T) {
|
||||||
|
p := NewPointer()
|
||||||
|
p.RedirectTo(3, 14)
|
||||||
|
require.Equal(t, 3, p.dx)
|
||||||
|
require.Equal(t, 14, p.dy)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestReverse(t *testing.T) {
|
||||||
|
p := NewPointer()
|
||||||
|
p.RedirectTo(3, 14)
|
||||||
|
p.Reverse()
|
||||||
|
require.Equal(t, -3, p.dx)
|
||||||
|
require.Equal(t, -14, p.dy)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestRedirect(t *testing.T) {
|
||||||
|
testCases := []struct {
|
||||||
|
name string
|
||||||
|
input byte
|
||||||
|
expectedDx int
|
||||||
|
expectedDy int
|
||||||
|
}{
|
||||||
|
{"up", '^', 0, -1},
|
||||||
|
{"right", '>', 1, 0},
|
||||||
|
{"down", 'v', 0, 1},
|
||||||
|
{"left", '<', -1, 0},
|
||||||
|
{"turn left", '[', 14, -3},
|
||||||
|
{"turn right", ']', -14, 3},
|
||||||
|
{"reverse", 'r', -3, -14},
|
||||||
|
{"redirectTo", 'x', 2, 7},
|
||||||
|
}
|
||||||
|
for _, tc := range testCases {
|
||||||
|
t.Run(tc.name, func(t *testing.T) {
|
||||||
|
p := NewPointer()
|
||||||
|
p.RedirectTo(3, 14)
|
||||||
|
p.ss.Push(2)
|
||||||
|
p.ss.Push(7)
|
||||||
|
v := p.Redirect(int(tc.input))
|
||||||
|
require.Equal(t, true, v)
|
||||||
|
require.Equal(t, tc.expectedDx, p.dx, "Invalid dx value")
|
||||||
|
require.Equal(t, tc.expectedDy, p.dy, "Invalid dy value")
|
||||||
|
})
|
||||||
|
}
|
||||||
|
// We cannot really test random, can we? This just gives coverage
|
||||||
|
p := NewPointer()
|
||||||
|
v := p.Redirect(int('?'))
|
||||||
|
require.Equal(t, true, v)
|
||||||
|
// A character that does not redirect should return false
|
||||||
|
require.Equal(t, false, p.Redirect(int('@')))
|
||||||
|
}
|
||||||
|
|
|
@ -1,8 +1,4 @@
|
||||||
package stack
|
package pointer
|
||||||
|
|
||||||
import (
|
|
||||||
"git.adyxax.org/adyxax/gofunge/pkg/pointer"
|
|
||||||
)
|
|
||||||
|
|
||||||
type StackStack struct {
|
type StackStack struct {
|
||||||
head *Stack
|
head *Stack
|
||||||
|
@ -16,7 +12,7 @@ func NewStackStack() *StackStack {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ss *StackStack) Begin(p *pointer.Pointer) {
|
func (ss *StackStack) Begin(p *Pointer) {
|
||||||
ss.height++
|
ss.height++
|
||||||
soss := ss.head
|
soss := ss.head
|
||||||
n := soss.Pop()
|
n := soss.Pop()
|
||||||
|
@ -44,7 +40,7 @@ func (ss *StackStack) Begin(p *pointer.Pointer) {
|
||||||
p.CalculateNewStorageOffset()
|
p.CalculateNewStorageOffset()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ss *StackStack) End(p *pointer.Pointer) (reflect bool) {
|
func (ss *StackStack) End(p *Pointer) (reflect bool) {
|
||||||
if ss.height == 1 {
|
if ss.height == 1 {
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
@ -89,3 +85,11 @@ func (ss *StackStack) Under() (reflect bool) {
|
||||||
}
|
}
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (ss StackStack) Pop() int {
|
||||||
|
return ss.head.Pop()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ss StackStack) Push(v int) {
|
||||||
|
ss.head.Push(v)
|
||||||
|
}
|
|
@ -1,11 +1,10 @@
|
||||||
package stack
|
package pointer
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"os"
|
"os"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"git.adyxax.org/adyxax/gofunge/pkg/field"
|
"git.adyxax.org/adyxax/gofunge/pkg/field"
|
||||||
"git.adyxax.org/adyxax/gofunge/pkg/pointer"
|
|
||||||
"github.com/stretchr/testify/require"
|
"github.com/stretchr/testify/require"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -17,7 +16,7 @@ func TestBegin(t *testing.T) {
|
||||||
expected.head.next.Push(0)
|
expected.head.next.Push(0)
|
||||||
expected.height++
|
expected.height++
|
||||||
ss := NewStackStack()
|
ss := NewStackStack()
|
||||||
p := pointer.NewPointer()
|
p := NewPointer()
|
||||||
ss.Begin(p)
|
ss.Begin(p)
|
||||||
require.Equal(t, expected, ss)
|
require.Equal(t, expected, ss)
|
||||||
x, y := p.GetStorageOffset()
|
x, y := p.GetStorageOffset()
|
||||||
|
@ -40,7 +39,7 @@ func TestBegin(t *testing.T) {
|
||||||
expected.head.next.Push(0)
|
expected.head.next.Push(0)
|
||||||
expected.head.next.Push(0)
|
expected.head.next.Push(0)
|
||||||
expected.height++
|
expected.height++
|
||||||
p := pointer.NewPointer()
|
p := NewPointer()
|
||||||
file, err := os.Open("../field/test_data/hello.b98")
|
file, err := os.Open("../field/test_data/hello.b98")
|
||||||
require.NoError(t, err, "Failed to open file")
|
require.NoError(t, err, "Failed to open file")
|
||||||
f, err := field.Load(file)
|
f, err := field.Load(file)
|
||||||
|
@ -61,7 +60,7 @@ func TestBegin(t *testing.T) {
|
||||||
expected.head.next.Push(2)
|
expected.head.next.Push(2)
|
||||||
expected.head.next.Push(3)
|
expected.head.next.Push(3)
|
||||||
expected.height++
|
expected.height++
|
||||||
p := pointer.NewPointer()
|
p := NewPointer()
|
||||||
p.SetStorageOffset(2, 3)
|
p.SetStorageOffset(2, 3)
|
||||||
file, err := os.Open("../field/test_data/hello.b98")
|
file, err := os.Open("../field/test_data/hello.b98")
|
||||||
require.NoError(t, err, "Failed to open file")
|
require.NoError(t, err, "Failed to open file")
|
||||||
|
@ -87,7 +86,7 @@ func TestBegin(t *testing.T) {
|
||||||
expected.head.next.Push(36)
|
expected.head.next.Push(36)
|
||||||
expected.head.next.Push(42)
|
expected.head.next.Push(42)
|
||||||
expected.height++
|
expected.height++
|
||||||
p := pointer.NewPointer()
|
p := NewPointer()
|
||||||
p.SetStorageOffset(36, 42)
|
p.SetStorageOffset(36, 42)
|
||||||
ss := NewStackStack()
|
ss := NewStackStack()
|
||||||
ss.head.Push(7)
|
ss.head.Push(7)
|
||||||
|
@ -104,7 +103,7 @@ func TestBegin(t *testing.T) {
|
||||||
func TestEnd(t *testing.T) {
|
func TestEnd(t *testing.T) {
|
||||||
t.Run("empty", func(t *testing.T) {
|
t.Run("empty", func(t *testing.T) {
|
||||||
expected := NewStackStack()
|
expected := NewStackStack()
|
||||||
p := pointer.NewPointer()
|
p := NewPointer()
|
||||||
ss := NewStackStack()
|
ss := NewStackStack()
|
||||||
ss.Begin(p)
|
ss.Begin(p)
|
||||||
reflect := ss.End(p)
|
reflect := ss.End(p)
|
||||||
|
@ -121,7 +120,7 @@ func TestEnd(t *testing.T) {
|
||||||
expected.head.Pop()
|
expected.head.Pop()
|
||||||
expected.head.Pop()
|
expected.head.Pop()
|
||||||
expected.head.Pop()
|
expected.head.Pop()
|
||||||
p := pointer.NewPointer()
|
p := NewPointer()
|
||||||
ss := NewStackStack()
|
ss := NewStackStack()
|
||||||
ss.head.Push(7)
|
ss.head.Push(7)
|
||||||
ss.head.Push(12)
|
ss.head.Push(12)
|
||||||
|
@ -139,7 +138,7 @@ func TestEnd(t *testing.T) {
|
||||||
})
|
})
|
||||||
t.Run("drop too much", func(t *testing.T) {
|
t.Run("drop too much", func(t *testing.T) {
|
||||||
expected := NewStackStack()
|
expected := NewStackStack()
|
||||||
p := pointer.NewPointer()
|
p := NewPointer()
|
||||||
ss := NewStackStack()
|
ss := NewStackStack()
|
||||||
ss.Begin(p)
|
ss.Begin(p)
|
||||||
ss.head.Push(-3)
|
ss.head.Push(-3)
|
||||||
|
@ -149,7 +148,7 @@ func TestEnd(t *testing.T) {
|
||||||
})
|
})
|
||||||
t.Run("reflect", func(t *testing.T) {
|
t.Run("reflect", func(t *testing.T) {
|
||||||
expected := NewStackStack()
|
expected := NewStackStack()
|
||||||
p := pointer.NewPointer()
|
p := NewPointer()
|
||||||
ss := NewStackStack()
|
ss := NewStackStack()
|
||||||
reflect := ss.End(p)
|
reflect := ss.End(p)
|
||||||
require.Equal(t, true, reflect)
|
require.Equal(t, true, reflect)
|
||||||
|
@ -164,7 +163,7 @@ func TestEnd(t *testing.T) {
|
||||||
expected.head.Push(14)
|
expected.head.Push(14)
|
||||||
expected.head.Push(-2)
|
expected.head.Push(-2)
|
||||||
expected.head.Push(5)
|
expected.head.Push(5)
|
||||||
p := pointer.NewPointer()
|
p := NewPointer()
|
||||||
ss := NewStackStack()
|
ss := NewStackStack()
|
||||||
ss.head.size = 4
|
ss.head.size = 4
|
||||||
ss.head.data = make([]int, 4)
|
ss.head.data = make([]int, 4)
|
||||||
|
@ -195,7 +194,7 @@ func TestUnder(t *testing.T) {
|
||||||
require.Equal(t, expected, ss)
|
require.Equal(t, expected, ss)
|
||||||
})
|
})
|
||||||
t.Run("positive", func(t *testing.T) {
|
t.Run("positive", func(t *testing.T) {
|
||||||
pe := pointer.NewPointer()
|
pe := NewPointer()
|
||||||
expected := NewStackStack()
|
expected := NewStackStack()
|
||||||
expected.head.Push(1)
|
expected.head.Push(1)
|
||||||
expected.head.Push(2)
|
expected.head.Push(2)
|
||||||
|
@ -209,7 +208,7 @@ func TestUnder(t *testing.T) {
|
||||||
expected.head.next.Pop()
|
expected.head.next.Pop()
|
||||||
expected.head.next.Pop()
|
expected.head.next.Pop()
|
||||||
expected.head.next.Pop()
|
expected.head.next.Pop()
|
||||||
p := pointer.NewPointer()
|
p := NewPointer()
|
||||||
ss := NewStackStack()
|
ss := NewStackStack()
|
||||||
ss.head.Push(1)
|
ss.head.Push(1)
|
||||||
ss.head.Push(2)
|
ss.head.Push(2)
|
||||||
|
@ -223,7 +222,7 @@ func TestUnder(t *testing.T) {
|
||||||
require.Equal(t, expected, ss)
|
require.Equal(t, expected, ss)
|
||||||
})
|
})
|
||||||
t.Run("negative", func(t *testing.T) {
|
t.Run("negative", func(t *testing.T) {
|
||||||
pe := pointer.NewPointer()
|
pe := NewPointer()
|
||||||
expected := NewStackStack()
|
expected := NewStackStack()
|
||||||
expected.Begin(pe)
|
expected.Begin(pe)
|
||||||
expected.head.next.Push(12)
|
expected.head.next.Push(12)
|
||||||
|
@ -237,7 +236,7 @@ func TestUnder(t *testing.T) {
|
||||||
expected.head.Pop()
|
expected.head.Pop()
|
||||||
expected.head.Pop()
|
expected.head.Pop()
|
||||||
expected.head.Pop()
|
expected.head.Pop()
|
||||||
p := pointer.NewPointer()
|
p := NewPointer()
|
||||||
ss := NewStackStack()
|
ss := NewStackStack()
|
||||||
ss.Begin(p)
|
ss.Begin(p)
|
||||||
ss.head.Push(8)
|
ss.head.Push(8)
|
||||||
|
@ -249,3 +248,13 @@ func TestUnder(t *testing.T) {
|
||||||
require.Equal(t, expected, ss)
|
require.Equal(t, expected, ss)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestPushPop(t *testing.T) {
|
||||||
|
ss := NewStackStack()
|
||||||
|
ss.Push(12)
|
||||||
|
ss.Push(5)
|
||||||
|
v := ss.Pop()
|
||||||
|
require.Equal(t, 5, v)
|
||||||
|
v = ss.Pop()
|
||||||
|
require.Equal(t, 12, v)
|
||||||
|
}
|
|
@ -1,4 +1,4 @@
|
||||||
package stack
|
package pointer
|
||||||
|
|
||||||
type Stack struct {
|
type Stack struct {
|
||||||
size int
|
size int
|
|
@ -1,4 +1,4 @@
|
||||||
package stack
|
package pointer
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"testing"
|
"testing"
|
Loading…
Add table
Add a link
Reference in a new issue