diff options
author | Julien Dessaux | 2021-09-23 14:59:14 +0200 |
---|---|---|
committer | Julien Dessaux | 2021-09-23 14:59:14 +0200 |
commit | f86b5724e530ec86eed04ebeb257293244f4be69 (patch) | |
tree | 0ceaddd9f08229899eb94f7134a048c5b36d9ba4 /pkg/pointer | |
parent | Each pointer needs its own stack, merging the two packages to avoid a circula... (diff) | |
download | gofunge98-f86b5724e530ec86eed04ebeb257293244f4be69.tar.gz gofunge98-f86b5724e530ec86eed04ebeb257293244f4be69.tar.bz2 gofunge98-f86b5724e530ec86eed04ebeb257293244f4be69.zip |
Moved the character execution's to the pointer in order to handle the k command
Diffstat (limited to 'pkg/pointer')
-rw-r--r-- | pkg/pointer/exec.go | 39 | ||||
-rw-r--r-- | pkg/pointer/pointer.go | 13 | ||||
-rw-r--r-- | pkg/pointer/pointer_test.go | 21 |
3 files changed, 64 insertions, 9 deletions
diff --git a/pkg/pointer/exec.go b/pkg/pointer/exec.go new file mode 100644 index 0000000..9d55136 --- /dev/null +++ b/pkg/pointer/exec.go @@ -0,0 +1,39 @@ +package pointer + +import ( + "log" + + "git.adyxax.org/adyxax/gofunge/pkg/field" +) + +func (p *Pointer) Exec(f *field.Field) (done bool, returnValue *int) { + c := p.Get(*f) + for jumpingMode := false; jumpingMode || c == ' ' || c == ';'; c = p.StepAndGet(*f) { + if jumpingMode { + if c == ';' { + jumpingMode = false + } + continue + } + } + switch c { + case '@': + return true, nil + case '#': + p.Step(*f) + case 'j': + n := p.Ss.Pop() + for j := 0; j < n; j++ { + p.Step(*f) + } + case 'q': + v := p.Ss.Pop() + return true, &v + default: + if !p.Redirect(c) { + log.Fatalf("Non implemented instruction code %d : %c", c, c) + } + } + p.Step(*f) + return +} diff --git a/pkg/pointer/pointer.go b/pkg/pointer/pointer.go index 4d847bf..902fe69 100644 --- a/pkg/pointer/pointer.go +++ b/pkg/pointer/pointer.go @@ -17,13 +17,13 @@ type Pointer struct { sox int soy int // The stack - ss *StackStack + Ss *StackStack // The next element for the multi-"threaded" b98 interpreter Next *Pointer } func NewPointer() *Pointer { - return &Pointer{dx: 1, ss: NewStackStack()} + return &Pointer{dx: 1, Ss: NewStackStack()} } func (p Pointer) Split() *Pointer { @@ -38,6 +38,11 @@ func (p Pointer) Get(f field.Field) int { return f.Get(p.x, p.y) } +func (p *Pointer) StepAndGet(f field.Field) int { + p.Step(f) + return p.Get(f) +} + func (p *Pointer) Set(x, y int) { p.x, p.y = x, y } @@ -71,8 +76,8 @@ func (p *Pointer) Redirect(c int) bool { case 'r': p.Reverse() case 'x': - dy := p.ss.Pop() - dx := p.ss.Pop() + dy := p.Ss.Pop() + dx := p.Ss.Pop() p.RedirectTo(dx, dy) default: return false diff --git a/pkg/pointer/pointer_test.go b/pkg/pointer/pointer_test.go index 4885d6b..aa007fe 100644 --- a/pkg/pointer/pointer_test.go +++ b/pkg/pointer/pointer_test.go @@ -9,7 +9,7 @@ import ( ) func TestNewPointer(t *testing.T) { - require.Equal(t, NewPointer(), &Pointer{dx: 1, ss: NewStackStack()}) + require.Equal(t, NewPointer(), &Pointer{dx: 1, Ss: NewStackStack()}) } func TestSplit(t *testing.T) { @@ -23,8 +23,8 @@ func TestSplit(t *testing.T) { // We check that p2 is a real copy p.Step(*f) p2.Step(*f) - require.Equal(t, &Pointer{x: 1, y: 0, dx: 1, ss: NewStackStack()}, p) - require.Equal(t, &Pointer{x: 1, y: 0, dx: 1, ss: NewStackStack()}, p2) + require.Equal(t, &Pointer{x: 1, y: 0, dx: 1, Ss: NewStackStack()}, p) + 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 @@ -51,6 +51,17 @@ func TestGet(t *testing.T) { require.Equal(t, int('@'), v) } +func TestStepAndGet(t *testing.T) { + // File of one char + file, err := os.Open("../field/test_data/minimal.b98") + require.NoError(t, err, "Failed to open file") + defer file.Close() + f, err := field.Load(file) + p := NewPointer() + v := p.StepAndGet(*f) + require.Equal(t, int('@'), v) +} + func TestSet(t *testing.T) { p := NewPointer() p.Set(3, 14) @@ -93,8 +104,8 @@ func TestRedirect(t *testing.T) { t.Run(tc.name, func(t *testing.T) { p := NewPointer() p.RedirectTo(3, 14) - p.ss.Push(2) - p.ss.Push(7) + 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") |