scenario channel [
run [
1:address:shared:source:number, 2:address:shared:sink:number <- new-channel 3/capacity
2:address:shared:sink:number <- write 2:address:shared:sink:number, 34
3:number, 1:address:shared:source:number <- read 1:address:shared:source:number
]
memory-should-contain [
3 <- 34
]
]
container channel:_elem [
first-full:number
first-free:number
data:address:shared:array:_elem
]
container source:_elem [
chan:address:shared:channel:_elem
]
container sink:_elem [
chan:address:shared:channel:_elem
]
def new-channel capacity:number -> in:address:shared:source:_elem, out:address:shared:sink:_elem [
local-scope
load-ingredients
result:address:shared:channel:_elem <- new {(channel _elem): type}
full:address:number <- get-address *result, first-full:offset
*full <- copy 0
free:address:number <- get-address *result, first-free:offset
*free <- copy 0
capacity <- add capacity, 1
dest:address:address:shared:array:_elem <- get-address *result, data:offset
*dest <- new _elem:type, capacity
in <- new {(source _elem): type}
chan:address:address:shared:channel:_elem <- get-address *in, chan:offset
*chan <- copy result
out <- new {(sink _elem): type}
chan:address:address:shared:channel:_elem <- get-address *out, chan:offset
*chan <- copy result
]
def write out:address:shared:sink:_elem, val:_elem -> out:address:shared:sink:_elem [
local-scope
load-ingredients
chan:address:shared:channel:_elem <- get *out, chan:offset
{
full:boolean <- channel-full? chan
break-unless full
full-address:address:number <- get-address *chan, first-full:offset
wait-for-location *full-address
}
circular-buffer:address:shared:array:_elem <- get *chan, data:offset
free:address:number <- get-address *chan, first-free:offset
dest:address:_elem <- index-address *circular-buffer, *free
*dest <- copy val
*free <- add *free, 1
{
len:number <- length *circular-buffer
at-end?:boolean <- greater-or-equal *free, len
break-unless at-end?
*free <- copy 0
}
]
def read in:address:shared:source:_elem -> result:_elem, in:address:shared:source:_elem [
local-scope
load-ingredients
chan:address:shared:channel:_elem <- get *in, chan:offset
{
empty?:boolean <- channel-empty? chan
break-unless empty?
free-address:address:number <- get-address *chan, first-free:offset
wait-for-location *free-address
}
full:address:number <- get-address *chan, first-full:offset
circular-buffer:address:shared:array:_elem <- get *chan, data:offset
result <- index *circular-buffer, *full
*full <- add *full, 1
{
len:number <- length *circular-buffer
at-end?:boolean <- greater-or-equal *full, len
break-unless at-end?
*full <- copy 0
}
]
def clear in:address:shared:source:_elem -> in:address:shared:source:_elem [
local-scope
load-ingredients
chan:address:shared:channel:_elem <- get *in, chan:offset
{
empty?:boolean <- channel-empty? chan
break-if empty?
_, in <- read in
}
]
scenario channel-initialization [
run [
1:address:shared:source:number <- new-channel 3/capacity
2:address:shared:channel:number <- get *1:address:shared:source:number, chan:offset
3:number <- get *2:address:shared:channel:number, first-full:offset
4:number <- get *2:address:shared:channel:number, first-free:offset
]
memory-should-contain [
3 <- 0
4 <- 0
]
]
scenario channel-write-increments-free [
run [
_, 1:address:shared:sink:number <- new-channel 3/capacity
1:address:shared:sink:number <- write 1:address:shared:sink:number, 34
2:address:shared:channel:number <- get *1:address:shared:sink:number, chan:offset
3:number <- get *2:address:shared:channel:character, first-full:offset
4:number <- get *2:address:shared:channel:character, first-free:offset
]
memory-should-contain [
3 <- 0
4 <- 1
]
]
scenario channel-read-increments-full [
run [
1:address:shared:source:number, 2:address:shared:sink:number <- new-channel 3/capacity
2:address:shared:sink:number <- write 2:address:shared:sink:number, 34
_, 1:address:shared:source:number <- read 1:address:shared:source:number
3:address:shared:channel:number <- get *1:address:shared:source:number, chan:offset
4:number <- get *3:address:shared:channel:number, first-full:offset
5:number <- get *3:address:shared:channel:number, first-free:offset
]
memory-should-contain [
4 <- 1
5 <- 1
]
]
scenario channel-wrap [
run [
1:address:shared:source:number, 2:address:shared:sink:number <- new-channel 1/capacity
3:address:shared:channel:number <- get *1:address:shared:source:number, chan:offset
2:address:shared:sink:number <- write 2:address:shared:sink:number, 34
_, 1:address:shared:source:number <- read 1:address:shared:source:number
4:number <- get *3:address:shared:channel:number, first-free:offset
5:number <- get *3:address:shared:channel:number, first-free:offset
2:address:shared:sink:number <- write 2:address:shared:sink:number, 34
6:number <- get *3:address:shared:channel:number, first-free:offset
_, 1:address:shared:source:number <- read 1:address:shared:source:number
7:number <- get *3:address:shared:channel:number, first-full:offset
]
memory-should-contain [
4 <- 1
5 <- 1
6 <- 0
7 <- 0
]
]
def channel-empty? chan:address:shared:channel:_elem -> result:boolean [
local-scope
load-ingredients
full:number <- get *chan, first-full:offset
free:number <- get *chan, first-free:offset
result <- equal full, free
]
def channel-full? chan:address:shared:channel:_elem -> result:boolean [
local-scope
load-ingredients
tmp:number <- get *chan, first-free:offset
tmp <- add tmp, 1
{
len:number <- capacity chan
at-end?:boolean <- greater-or-equal tmp, len
break-unless at-end?
tmp <- copy 0
}
full:number <- get *chan, first-full:offset
result <- equal full, tmp
]
def capacity chan:address:shared:channel:_elem -> result:number [
local-scope
load-ingredients
q:address:shared:array:_elem <- get *chan, data:offset
result <- length *q
]
scenario channel-new-empty-not-full [
run [
1:address:shared:source:number, 2:address:shared:sink:number <- new-channel 3/capacity
3:address:shared:channel:number <- get *1:address:shared:source:number, chan:offset
4:boolean <- channel-empty? 3:address:shared:channel:number
5:boolean <- channel-full? 3:address:shared:channel:number
]
memory-should-contain [
4 <- 1
5 <- 0
]
]
scenario channel-write-not-empty [
run [
1:address:shared:source:number, 2:address:shared:sink:number <- new-channel 3/capacity
3:address:shared:channel:number <- get *1:address:shared:source:number, chan:offset
2:address:shared:sink:number <- write 2:address:shared:sink:number, 34
4:boolean <- channel-empty? 3:address:shared:channel:number
5:boolean <- channel-full? 3:address:shared:channel:number
]
memory-should-contain [
4 <- 0
5 <- 0
]
]
scenario channel-write-full [
run [
1:address:shared:source:number, 2:address:shared:sink:number <- new-channel 1/capacity
3:address:shared:channel:number <- get *1:address:shared:source:number, chan:offset
2:address:shared:sink:number <- write 2:address:shared:sink:number, 34
4:boolean <- channel-empty? 3:address:shared:channel:number
5:boolean <- channel-full? 3:address:shared:channel:number
]
memory-should-contain [
4 <- 0
5 <- 1
]
]
scenario channel-read-not-full [
run [
1:address:shared:source:number, 2:address:shared:sink:number <- new-channel 1/capacity
3:address:shared:channel:number <- get *1:address:shared:source:number, chan:offset
2:address:shared:sink:number <- write 2:address:shared:sink:number, 34
_, 1:address:shared:source:number <- read 1:address:shared:source:number
4:boolean <- channel-empty? 3:address:shared:channel:number
5:boolean <- channel-full? 3:address:shared:channel:number
]
memory-should-contain [
4 <- 1
5 <- 0
]
]
def buffer-lines in:address:shared:source:character, buffered-out:address:shared:sink:character -> buffered-out:address:shared:sink:character, in:address:shared:source:character [
local-scope
load-ingredients
{
line:address:shared:buffer <- new-buffer 30
{
+next-character
c:character, in <- read in
{
backspace?:boolean <- equal c, 8
break-unless backspace?
{
buffer-length:address:number <- get-address *line, length:offset
buffer-empty?:boolean <- equal *buffer-length, 0
break-if buffer-empty?
*buffer-length <- subtract *buffer-length, 1
}
loop +next-character:label
}
line <- append line, c
line-done?:boolean <- equal c, 10/newline
break-if line-done?
eof?:boolean <- equal c, 0/eof
break-if eof?
loop
}
i:number <- copy 0
line-contents:address:shared:array:character <- get *line, data:offset
max:number <- get *line, length:offset
{
done?:boolean <- greater-or-equal i, max
break-if done?
c:character <- index *line-contents, i
buffered-out <- write buffered-out, c
i <- add i, 1
loop
}
loop
}
]
scenario buffer-lines-blocks-until-newline [
run [
1:address:shared:source:number, 2:address:shared:sink:number <- new-channel 10/capacity
_, 3:address:shared:sink:number/buffered-stdin <- new-channel 10/capacity
4:address:shared:channel:number/buffered-stdin <- get *3:address:shared:source:number, chan:offset
5:boolean <- channel-empty? 4:address:shared:channel:character/buffered-stdin
assert 5:boolean, [
F buffer-lines-blocks-until-newline: channel should be empty after init]
6:number/buffer-routine <- start-running buffer-lines, 1:address:shared:source:character/stdin, 3:address:shared:sink:character/buffered-stdin
wait-for-routine 6:number/buffer-routine
7:boolean <- channel-empty? 4:address:shared:channel:character/buffered-stdin
assert 7:boolean, [
F buffer-lines-blocks-until-newline: channel should be empty after buffer-lines bring-up]
2:address:shared:sink:character <- write 2:address:shared:sink:character, 97/a
restart 6:number/buffer-routine
wait-for-routine 6:number/buffer-routine
8:boolean <- channel-empty? 4:address:shared:channel:character/buffered-stdin
assert 8:boolean, [
F buffer-lines-blocks-until-newline: channel should be empty after writing 'a']
2:address:shared:sink:character <- write 2:address:shared:sink:character, 98/b
restart 6:number/buffer-routine
wait-for-routine 6:number/buffer-routine
9:boolean <- channel-empty? 4:address:shared:channel:character/buffered-stdin
assert 9:boolean, [
F buffer-lines-blocks-until-newline: channel should be empty after writing 'b']
2:address:shared:sink:character <- write 2:address:shared:sink:character, 10/newline
restart 6:number/buffer-routine
wait-for-routine 6:number/buffer-routine
10:boolean <- channel-empty? 4:address:shared:channel:character/buffered-stdin
11:boolean/completed? <- not 10:boolean
assert 11:boolean/completed?, [
F buffer-lines-blocks-until-newline: channel should contain data after writing newline]
trace 1, [test], [reached end]
]
trace-should-contain [
test: reached end
]
]