mu/074list.mu

110 lines
2.8 KiB
Plaintext
Raw Normal View History

2015-05-19 17:36:46 +00:00
# A list links up multiple objects together to make them easier to manage.
#
# The objects must be of the same type. If you want to store multiple types in
# a single list, use an exclusive-container.
2015-05-19 17:36:46 +00:00
container list:_elem [
value:_elem
next:address:list:_elem
2015-05-16 06:00:16 +00:00
]
def push x:_elem, in:address:list:_elem -> result:address:list:_elem [
local-scope
load-ingredients
result <- new {(list _elem): type}
*result <- merge x, in
2015-05-16 06:00:16 +00:00
]
def first in:address:list:_elem -> result:_elem [
local-scope
load-ingredients
result <- get *in, value:offset
2015-05-16 06:00:16 +00:00
]
def rest in:address:list:_elem -> result:address:list:_elem/contained-in:in [
local-scope
load-ingredients
result <- get *in, next:offset
]
2015-05-16 06:00:16 +00:00
scenario list-handling [
run [
1:address:list:number <- push 3, 0
1:address:list:number <- push 4, 1:address:list:number
1:address:list:number <- push 5, 1:address:list:number
2:number <- first 1:address:list:number
1:address:list:number <- rest 1:address:list:number
3:number <- first 1:address:list:number
1:address:list:number <- rest 1:address:list:number
4:number <- first 1:address:list:number
1:address:list:number <- rest 1:address:list:number
2015-05-16 06:00:16 +00:00
]
memory-should-contain [
1 <- 0 # empty to empty, dust to dust..
2 <- 5
3 <- 4
4 <- 3
]
]
def length l:address:list:_elem -> result:number [
local-scope
load-ingredients
return-unless l, 0
rest:address:list:_elem <- rest l
length-of-rest:number <- length rest
result <- add length-of-rest, 1
]
def to-text in:address:list:_elem -> result:address:array:character [
local-scope
load-ingredients
buf:address:buffer <- new-buffer 80
buf <- to-buffer in, buf
result <- buffer-to-array buf
]
# variant of 'to-text' which stops printing after a few elements (and so is robust to cycles)
def to-text-line in:address:list:_elem -> result:address:array:character [
local-scope
load-ingredients
buf:address:buffer <- new-buffer 80
buf <- to-buffer in, buf, 6 # max elements to display
result <- buffer-to-array buf
]
def to-buffer in:address:list:_elem, buf:address:buffer -> buf:address:buffer [
local-scope
load-ingredients
{
break-if in
buf <- append buf, 48/0
return
}
# append in.value to buf
val:_elem <- get *in, value:offset
buf <- append buf, val
# now prepare next
next:address:list:_elem <- rest in
nextn:number <- copy next
return-unless next
buf <- append buf, [ -> ]
# and recurse
remaining:number, optional-ingredient-found?:boolean <- next-ingredient
{
break-if optional-ingredient-found?
# unlimited recursion
buf <- to-buffer next, buf
return
}
{
break-unless remaining
# limited recursion
remaining <- subtract remaining, 1
buf <- to-buffer next, buf, remaining
return
}
# past recursion depth; insert ellipses and stop
append buf, [...]
]