container duplex-list:_elem [
value:_elem
next:&:duplex-list:_elem
prev:&:duplex-list:_elem
]
def push x:_elem, in:&:duplex-list:_elem -> in:&:duplex-list:_elem [
local-scope
load-ingredients
result:&:duplex-list:_elem <- new {(duplex-list _elem): type}
*result <- merge x, in, 0
{
break-unless in
*in <- put *in, prev:offset, result
}
return result
]
def first in:&:duplex-list:_elem -> result:_elem [
local-scope
load-ingredients
return-unless in, 0
result <- get *in, value:offset
]
def next in:&:duplex-list:_elem -> result:&:duplex-list:_elem/contained-in:in [
local-scope
load-ingredients
return-unless in, 0
result <- get *in, next:offset
]
def prev in:&:duplex-list:_elem -> result:&:duplex-list:_elem/contained-in:in [
local-scope
load-ingredients
return-unless in, 0
result <- get *in, prev:offset
return result
]
scenario duplex-list-handling [
run [
local-scope
10:num/raw <- copy 34
11:num/raw <- copy 35
list:&:duplex-list:char <- push 3, 0
list <- push 4, list
list <- push 5, list
list2:&:duplex-list:char <- copy list
20:char/raw <- first list2
list2 <- next list2
21:char/raw <- first list2
list2 <- next list2
22:char/raw <- first list2
30:&:duplex-list:char/raw <- next list2
31:char/raw <- first 30:&:duplex-list:char/raw
32:&:duplex-list:char/raw <- next 30:&:duplex-list:char/raw
33:&:duplex-list:char/raw <- prev 30:&:duplex-list:char/raw
list2 <- prev list2
40:char/raw <- first list2
list2 <- prev list2
41:char/raw <- first list2
50:bool/raw <- equal list, list2
]
memory-should-contain [
0 <- 0
10 <- 34
11 <- 35
20 <- 5
21 <- 4
22 <- 3
30 <- 0
31 <- 0
32 <- 0
33 <- 0
40 <- 4
41 <- 5
50 <- 1
]
]
def insert x:_elem, in:&:duplex-list:_elem -> in:&:duplex-list:_elem [
local-scope
load-ingredients
new-node:&:duplex-list:_elem <- new {(duplex-list _elem): type}
*new-node <- put *new-node, value:offset, x
next-node:&:duplex-list:_elem <- get *in, next:offset
*in <- put *in, next:offset, new-node
*new-node <- put *new-node, prev:offset, in
*new-node <- put *new-node, next:offset, next-node
return-unless next-node
*next-node <- put *next-node, prev:offset, new-node
]
scenario inserting-into-duplex-list [
local-scope
list:&:duplex-list:char <- push 3, 0
list <- push 4, list
list <- push 5, list
run [
list2:&:duplex-list:char <- next list
list2 <- insert 6, list2
list2 <- copy list
10:char/raw <- first list2
list2 <- next list2
11:char/raw <- first list2
list2 <- next list2
12:char/raw <- first list2
list2 <- next list2
13:char/raw <- first list2
list2 <- prev list2
20:char/raw <- first list2
list2 <- prev list2
21:char/raw <- first list2
list2 <- prev list2
22:char/raw <- first list2
30:bool/raw <- equal list, list2
]
memory-should-contain [
10 <- 5
11 <- 4
12 <- 6
13 <- 3
20 <- 6
21 <- 4
22 <- 5
30 <- 1
]
]
scenario inserting-at-end-of-duplex-list [
local-scope
list:&:duplex-list:char <- push 3, 0
list <- push 4, list
list <- push 5, list
run [
list2:&:duplex-list:char <- next list
list2 <- next list2
list2 <- insert 6, list2
list2 <- copy list
10:char/raw <- first list2
list2 <- next list2
11:char/raw <- first list2
list2 <- next list2
12:char/raw <- first list2
list2 <- next list2
13:char/raw <- first list2
list2 <- prev list2
20:char/raw <- first list2
list2 <- prev list2
21:char/raw <- first list2
list2 <- prev list2
22:char/raw <- first list2
30:bool/raw <- equal list, list2
]
memory-should-contain [
10 <- 5
11 <- 4
12 <- 3
13 <- 6
20 <- 3
21 <- 4
22 <- 5
30 <- 1
]
]
scenario inserting-after-start-of-duplex-list [
local-scope
list:&:duplex-list:char <- push 3, 0
list <- push 4, list
list <- push 5, list
run [
list <- insert 6, list
list2:&:duplex-list:char <- copy list
10:char/raw <- first list2
list2 <- next list2
11:char/raw <- first list2
list2 <- next list2
12:char/raw <- first list2
list2 <- next list2
13:char/raw <- first list2
list2 <- prev list2
20:char/raw <- first list2
list2 <- prev list2
21:char/raw <- first list2
list2 <- prev list2
22:char/raw <- first list2
30:bool/raw <- equal list, list2
]
memory-should-contain [
10 <- 5
11 <- 6
12 <- 4
13 <- 3
20 <- 4
21 <- 6
22 <- 5
30 <- 1
]
]
def remove x:&:duplex-list:_elem/contained-in:in, in:&:duplex-list:_elem -> in:&:duplex-list:_elem [
local-scope
load-ingredients
return-unless x
next-node:&:duplex-list:_elem <- get *x, next:offset
prev-node:&:duplex-list:_elem <- get *x, prev:offset
*x <- put *x, next:offset, 0
*x <- put *x, prev:offset, 0
{
break-unless next-node
*next-node <- put *next-node, prev:offset, prev-node
}
{
break-unless prev-node
*prev-node <- put *prev-node, next:offset, next-node
return
}
return next-node
]
scenario removing-from-duplex-list [
local-scope
list:&:duplex-list:char <- push 3, 0
list <- push 4, list
list <- push 5, list
run [
list2:&:duplex-list:char <- next list
list <- remove list2, list
10:bool/raw <- equal list2, 0
list2 <- copy list
11:char/raw <- first list2
list2 <- next list2
12:char/raw <- first list2
20:&:duplex-list:char/raw <- next list2
list2 <- prev list2
30:char/raw <- first list2
40:bool/raw <- equal list, list2
]
memory-should-contain [
10 <- 0
11 <- 5
12 <- 3
20 <- 0
30 <- 5
40 <- 1
]
]
scenario removing-from-start-of-duplex-list [
local-scope
list:&:duplex-list:char <- push 3, 0
list <- push 4, list
list <- push 5, list
run [
list <- remove list, list
list2:&:duplex-list:char <- copy list
10:char/raw <- first list2
list2 <- next list2
11:char/raw <- first list2
20:&:duplex-list:char/raw <- next list2
list2 <- prev list2
30:char/raw <- first list2
40:bool/raw <- equal list, list2
]
memory-should-contain [
10 <- 4
11 <- 3
20 <- 0
30 <- 4
40 <- 1
]
]
scenario removing-from-end-of-duplex-list [
local-scope
list:&:duplex-list:char <- push 3, 0
list <- push 4, list
list <- push 5, list
run [
list2:&:duplex-list:char <- next list
list2 <- next list2
list <- remove list2, list
10:bool/raw <- equal list2, 0
list2 <- copy list
11:char/raw <- first list2
list2 <- next list2
12:char/raw <- first list2
20:&:duplex-list:char/raw <- next list2
list2 <- prev list2
30:char/raw <- first list2
40:bool/raw <- equal list, list2
]
memory-should-contain [
10 <- 0
11 <- 5
12 <- 4
20 <- 0
30 <- 5
40 <- 1
]
]
scenario removing-from-singleton-duplex-list [
local-scope
list:&:duplex-list:char <- push 3, 0
run [
list <- remove list, list
1:num/raw <- copy list
]
memory-should-contain [
1 <- 0
]
]
def remove-between start:&:duplex-list:_elem, end:&:duplex-list:_elem/contained-in:start -> start:&:duplex-list:_elem [
local-scope
load-ingredients
next:&:duplex-list:_elem <- get *start, next:offset
nothing-to-delete?:bool <- equal next, end
return-if nothing-to-delete?
assert next, [malformed duplex list]
*next <- put *next, prev:offset, 0
*start <- put *start, next:offset, end
return-unless end
prev:&:duplex-list:_elem <- get *end, prev:offset
assert prev, [malformed duplex list - 2]
*prev <- put *prev, next:offset, 0
*end <- put *end, prev:offset, start
]
scenario remove-range [
local-scope
list:&:duplex-list:char <- push 18, 0
list <- push 17, list
list <- push 16, list
list <- push 15, list
list <- push 14, list
list <- push 13, list
run [
list2:&:duplex-list:char <- next list
list2 <- next list2
list2 <- remove-between list2, 0
10:char/raw <- get *list, value:offset
list <- next list
11:char/raw <- get *list, value:offset
list <- next list
12:char/raw <- get *list, value:offset
20:&:duplex-list:char/raw <- next list
]
memory-should-contain [
10 <- 13
11 <- 14
12 <- 15
20 <- 0
]
]
scenario remove-range-to-final [
local-scope
list:&:duplex-list:char <- push 18, 0
list <- push 17, list
list <- push 16, list
list <- push 15, list
list <- push 14, list
list <- push 13, list
run [
list2:&:duplex-list:char <- next list
end:&:duplex-list:char <- next list2
end <- next end
end <- next end
end <- next end
remove-between list2, end
10:char/raw <- get *list, value:offset
list <- next list
11:char/raw <- next list
]
memory-should-contain [
10 <- 13
11 <- 14
12 <- 18
20 <- 0
]
]
scenario remove-range-empty [
local-scope
list:&:duplex-list:char <- push 15, 0
list <- push 14, list
list <- push 13, list
run [
list2:&:duplex-list:char <- next list
remove-between list, list2
10:char/raw <- get *list, value:offset
list <- next list
11:char/raw <- get *list, value:offset
list <- next list
12:char/raw <- get *list, value:offset
20:&:duplex-list:char/raw <- next list
]
memory-should-contain [
10 <- 13
11 <- 14
12 <- 15
20 <- 0
]
]
scenario remove-range-to-end [
local-scope
list:&:duplex-list:char <- push 18, 0
list <- push 17, list
list <- push 16, list
list <- push 15, list
list <- push 14, list
list <- push 13, list
run [
list2:&:duplex-list:char <- next list
remove-between list2, 0
10:char/raw <- get *list, value:offset
list <- next list
11:char/raw <- get *list, value:offset
20:&:duplex-list:char/raw <- next list
]
memory-should-contain [
10 <- 13
11 <- 14
20 <- 0
]
]
def insert-range in:&:duplex-list:_elem, start:&:duplex-list:_elem/contained-in:in -> in:&:duplex-list:_elem [
local-scope
load-ingredients
return-unless in
return-unless start
end:&:duplex-list:_elem <- copy start
{
next:&:duplex-list:_elem <- next end/insert-range
break-unless next
end <- copy next
loop
}
next:&:duplex-list:_elem <- next in
*end <- put *end, next:offset, next
{
break-unless next
*next <- put *next, prev:offset, end
}
*in <- put *in, next:offset, start
*start <- put *start, prev:offset, in
]
def append in:&:duplex-list:_elem, new:&:duplex-list:_elem/contained-in:in -> in:&:duplex-list:_elem [
local-scope
load-ingredients
last:&:duplex-list:_elem <- last in
*last <- put *last, next:offset, new
return-unless new
*new <- put *new, prev:offset, last
]
def last in:&:duplex-list:_elem -> result:&:duplex-list:_elem [
local-scope
load-ingredients
result <- copy in
{
next:&:duplex-list:_elem <- next result
break-unless next
result <- copy next
loop
}
]
def dump-from x:&:duplex-list:_elem [
local-scope
load-ingredients
$print x, [: ]
{
break-unless x
c:_elem <- get *x, value:offset
$print c, [ ]
x <- next x
{
is-newline?:bool <- equal c, 10/newline
break-unless is-newline?
$print 10/newline
$print x, [: ]
}
loop
}
$print 10/newline, [---], 10/newline
]