From 1d11df68fc011a023e19b4ec7db4df73bef331d7 Mon Sep 17 00:00:00 2001 From: Julien Dessaux Date: Mon, 4 Oct 2021 18:22:59 +0200 Subject: Implemented the stackstack --- src/stack.nim | 12 ++++++++++++ src/stackStack.nim | 57 ++++++++++++++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 69 insertions(+) create mode 100644 src/stackStack.nim (limited to 'src') diff --git a/src/stack.nim b/src/stack.nim index 2c6826d..eeef5cd 100644 --- a/src/stack.nim +++ b/src/stack.nim @@ -58,12 +58,24 @@ func Swap*(s: var Stack) = s.Push(b) func Transfert*(toss: var Stack, soss: var Stack, n: int) = + ## Implements a value transfert between two stacks, intended for use with the '{' + ## (aka begin) and '}' (aka end) stackstack commands toss.height += n + if toss.height > toss.size: + toss.size += 32 + toss.data.setlen(toss.size) for i in 1..min(soss.height, n): toss.data[toss.height-i] = soss.data[soss.height-i] + for i in min(soss.height, n)+1..n: + toss.data[toss.height-i] = 0 soss.height -= n if soss.height < 0: soss.height = 0 +func Discard*(s: var Stack, n: int) = + s.height -= n + if s.height < 0: + s.height = 0 + func Next*(s: Stack): ref Stack = return s.next diff --git a/src/stackStack.nim b/src/stackStack.nim new file mode 100644 index 0000000..6d32c87 --- /dev/null +++ b/src/stackStack.nim @@ -0,0 +1,57 @@ +import stack + +type + StackStack* = object + height: int + head: ref Stack + +func NewStackStack*(): ref StackStack = + result = new(StackStack) + result.head = NewStack() + result.height = 1 + +func Pop*(ss: StackStack): int = + return ss.head[].Pop() + +func Push*(ss: StackStack, v: int) = + ss.head[].Push(v) + +func PopVector*(ss: StackStack): (int, int) = + return ss.head[].PopVector() + +func PushVector*(ss: var StackStack, v: tuple[x, y: int]) = + ss.head[].PushVector(v) + +func Clear*(ss: var StackStack) = + ss.head[].Clear() + +func Begin*(ss: var StackStack, v: tuple[x, y: int]) = + inc ss.height + let soss = ss.head + let n = soss[].Pop() + ss.head = NewStack(size = abs(n), next = soss) + let toss = ss.head + if n > 0: + toss[].Transfert(soss[], n) + elif n < 0: + for i in 0 ..< -n: + soss[].Push(0) + soss[].PushVector(v) + +func End*(ss: var StackStack, v: var tuple[x, y: int]): bool = + ## Implements the '}' command behaviour which pops a stack from the stack stack + ## returns true if a reflect should happen + if ss.height == 1: + return true + let toss = ss.head + let soss = toss[].Next() + let n = toss[].Pop() + v.y = soss[].Pop() + v.x = soss[].Pop() + if n > 0: + soss[].Transfert(toss[], n) + else: + soss[].Discard(-n) + dec ss.height + ss.head = soss + return false -- cgit v1.2.3