2015-10-16 08:38:38 +00:00
|
|
|
|
SixtyPical Analysis
|
2015-10-16 08:30:24 +00:00
|
|
|
|
===================
|
|
|
|
|
|
2015-10-16 08:38:38 +00:00
|
|
|
|
This is a test suite, written in [Falderal][] format, for the SixtyPical
|
2015-10-16 08:30:24 +00:00
|
|
|
|
static analysis rules.
|
|
|
|
|
|
|
|
|
|
[Falderal]: http://catseye.tc/node/Falderal
|
|
|
|
|
|
2015-10-16 08:38:38 +00:00
|
|
|
|
-> Functionality "Analyze SixtyPical program" is implemented by
|
2018-02-06 16:14:44 +00:00
|
|
|
|
-> shell command "bin/sixtypical --analyze-only --traceback %(test-body-file) && echo ok"
|
2015-10-16 08:30:24 +00:00
|
|
|
|
|
2015-10-16 08:38:38 +00:00
|
|
|
|
-> Tests for functionality "Analyze SixtyPical program"
|
2015-10-16 08:30:24 +00:00
|
|
|
|
|
|
|
|
|
### Rudiments ###
|
|
|
|
|
|
|
|
|
|
Routines must declare their inputs, outputs, and memory locations they trash.
|
|
|
|
|
|
|
|
|
|
| routine up
|
|
|
|
|
| inputs a
|
|
|
|
|
| outputs a
|
|
|
|
|
| trashes c, z, v, n
|
|
|
|
|
| {
|
|
|
|
|
| st off, c
|
|
|
|
|
| add a, 1
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
Routines may not declare a memory location to be both an output and trashed.
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs a
|
|
|
|
|
| trashes a
|
|
|
|
|
| {
|
|
|
|
|
| ld a, 0
|
|
|
|
|
| }
|
2015-10-22 08:48:26 +00:00
|
|
|
|
? InconsistentConstraintsError: a
|
2015-10-16 08:30:24 +00:00
|
|
|
|
|
|
|
|
|
If a routine declares it outputs a location, that location should be initialized.
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs a, x, z, n
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulOutputError: a
|
2015-10-16 08:30:24 +00:00
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs a
|
|
|
|
|
| outputs a
|
|
|
|
|
| {
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
If a routine declares it outputs a location, that location may or may not have
|
|
|
|
|
been initialized. Trashing is mainly a signal to the caller.
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| trashes x, z, n
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| trashes x, z, n
|
|
|
|
|
| {
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
If a routine modifies a location, it needs to either output it or trash it.
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? ForbiddenWriteError: x
|
2015-10-16 08:30:24 +00:00
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs x, z, n
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| trashes x, z, n
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
2018-02-08 13:34:03 +00:00
|
|
|
|
This is true regardless of whether it's an input or not.
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs x
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? ForbiddenWriteError: x
|
2018-02-08 13:34:03 +00:00
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs x
|
|
|
|
|
| outputs x, z, n
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs x
|
|
|
|
|
| trashes x, z, n
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
If a routine trashes a location, this must be declared.
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| trashes x
|
|
|
|
|
| {
|
|
|
|
|
| trash x
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| {
|
|
|
|
|
| trash x
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? ForbiddenWriteError: x
|
2018-02-08 13:34:03 +00:00
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| outputs x
|
|
|
|
|
| {
|
|
|
|
|
| trash x
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulOutputError: x
|
2018-02-08 13:34:03 +00:00
|
|
|
|
|
|
|
|
|
If a routine causes a location to be trashed, this must be declared in the caller.
|
|
|
|
|
|
|
|
|
|
| routine trash_x
|
|
|
|
|
| trashes x, z, n
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| trashes x, z, n
|
|
|
|
|
| {
|
|
|
|
|
| call trash_x
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| routine trash_x
|
|
|
|
|
| trashes x, z, n
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| call trash_x
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? ForbiddenWriteError: x
|
2018-02-08 13:34:03 +00:00
|
|
|
|
|
|
|
|
|
| routine trash_x
|
|
|
|
|
| trashes x, z, n
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| call trash_x
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulOutputError: x (in foo, line 12)
|
2018-02-08 13:34:03 +00:00
|
|
|
|
|
2017-12-12 15:34:51 +00:00
|
|
|
|
If a routine reads or writes a user-define memory location, it needs to declare that too.
|
|
|
|
|
|
|
|
|
|
| byte b1 @ 60000
|
|
|
|
|
| byte b2 : 3
|
|
|
|
|
| word w1 @ 60001
|
|
|
|
|
| word w2 : 2000
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs b1, w1
|
|
|
|
|
| outputs b2, w2
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| ld a, b1
|
|
|
|
|
| st a, b2
|
|
|
|
|
| copy w1, w2
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
2015-10-16 08:30:24 +00:00
|
|
|
|
### ld ###
|
|
|
|
|
|
|
|
|
|
Can't `ld` from a memory location that isn't initialized.
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs a, x
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| ld a, x
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs a
|
|
|
|
|
| trashes a
|
|
|
|
|
| {
|
|
|
|
|
| ld a, x
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulReadError: x
|
2015-10-16 08:30:24 +00:00
|
|
|
|
|
|
|
|
|
Can't `ld` to a memory location that doesn't appear in (outputs ∪ trashes).
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| ld a, 0
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs a
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| ld a, 0
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs z, n
|
|
|
|
|
| trashes a
|
|
|
|
|
| {
|
|
|
|
|
| ld a, 0
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| ld a, 0
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? ForbiddenWriteError: a
|
2015-10-16 08:30:24 +00:00
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| trashes a, n
|
|
|
|
|
| {
|
|
|
|
|
| ld a, 0
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? ForbiddenWriteError: z
|
2015-10-16 08:30:24 +00:00
|
|
|
|
|
2016-06-16 16:08:57 +00:00
|
|
|
|
Can't `ld` a `word` type.
|
|
|
|
|
|
|
|
|
|
| word foo
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs foo
|
|
|
|
|
| trashes a, n, z
|
|
|
|
|
| {
|
|
|
|
|
| ld a, foo
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? TypeMismatchError: foo and a
|
2016-06-16 16:08:57 +00:00
|
|
|
|
|
2015-10-16 08:30:24 +00:00
|
|
|
|
### st ###
|
|
|
|
|
|
|
|
|
|
Can't `st` from a memory location that isn't initialized.
|
|
|
|
|
|
|
|
|
|
| byte lives
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs x
|
|
|
|
|
| trashes lives
|
|
|
|
|
| {
|
|
|
|
|
| st x, lives
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| byte lives
|
|
|
|
|
| routine main
|
|
|
|
|
| trashes x, lives
|
|
|
|
|
| {
|
|
|
|
|
| st x, lives
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulReadError: x
|
2015-10-16 08:30:24 +00:00
|
|
|
|
|
|
|
|
|
Can't `st` to a memory location that doesn't appear in (outputs ∪ trashes).
|
|
|
|
|
|
|
|
|
|
| byte lives
|
|
|
|
|
| routine main
|
|
|
|
|
| trashes lives
|
|
|
|
|
| {
|
|
|
|
|
| st 0, lives
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| byte lives
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs lives
|
|
|
|
|
| {
|
|
|
|
|
| st 0, lives
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| byte lives
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs lives
|
|
|
|
|
| {
|
|
|
|
|
| st 0, lives
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? ForbiddenWriteError: lives
|
2015-10-16 08:30:24 +00:00
|
|
|
|
|
2017-12-08 13:41:48 +00:00
|
|
|
|
Can't `st` a `word` type.
|
|
|
|
|
|
|
|
|
|
| word foo
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs foo
|
|
|
|
|
| trashes a, n, z
|
|
|
|
|
| {
|
|
|
|
|
| ld a, 0
|
|
|
|
|
| st a, foo
|
|
|
|
|
| }
|
2018-02-06 16:14:44 +00:00
|
|
|
|
? TypeMismatchError
|
2017-12-08 13:41:48 +00:00
|
|
|
|
|
|
|
|
|
### tables ###
|
|
|
|
|
|
2018-02-05 12:35:19 +00:00
|
|
|
|
Storing to a table, you must use an index.
|
2015-10-18 17:12:47 +00:00
|
|
|
|
|
|
|
|
|
| byte one
|
2018-02-06 11:34:21 +00:00
|
|
|
|
| byte table[256] many
|
2015-10-18 17:12:47 +00:00
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs one
|
|
|
|
|
| trashes a, x, n, z
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| ld a, 0
|
|
|
|
|
| st a, one
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| byte one
|
2018-02-06 11:34:21 +00:00
|
|
|
|
| byte table[256] many
|
2015-10-18 17:12:47 +00:00
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs many
|
|
|
|
|
| trashes a, x, n, z
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| ld a, 0
|
|
|
|
|
| st a, many
|
|
|
|
|
| }
|
|
|
|
|
? TypeMismatchError
|
|
|
|
|
|
|
|
|
|
| byte one
|
2018-02-06 11:34:21 +00:00
|
|
|
|
| byte table[256] many
|
2015-10-18 17:12:47 +00:00
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs one
|
|
|
|
|
| trashes a, x, n, z
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| ld a, 0
|
|
|
|
|
| st a, one + x
|
|
|
|
|
| }
|
|
|
|
|
? TypeMismatchError
|
|
|
|
|
|
|
|
|
|
| byte one
|
2018-02-06 11:34:21 +00:00
|
|
|
|
| byte table[256] many
|
2015-10-18 17:12:47 +00:00
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs many
|
|
|
|
|
| trashes a, x, n, z
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| ld a, 0
|
|
|
|
|
| st a, many + x
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
2018-02-05 12:35:19 +00:00
|
|
|
|
The index must be initialized.
|
|
|
|
|
|
|
|
|
|
| byte one
|
2018-02-06 11:34:21 +00:00
|
|
|
|
| byte table[256] many
|
2018-02-05 12:35:19 +00:00
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs many
|
|
|
|
|
| trashes a, x, n, z
|
|
|
|
|
| {
|
|
|
|
|
| ld a, 0
|
|
|
|
|
| st a, many + x
|
|
|
|
|
| }
|
|
|
|
|
? UnmeaningfulReadError: x
|
|
|
|
|
|
|
|
|
|
Reading from a table, you must use an index.
|
2015-10-18 17:12:47 +00:00
|
|
|
|
|
|
|
|
|
| byte one
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs one
|
|
|
|
|
| trashes a, x, n, z
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| st x, one
|
|
|
|
|
| ld a, one
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| byte one
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs one
|
|
|
|
|
| trashes a, x, n, z
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| st x, one
|
|
|
|
|
| ld a, one + x
|
|
|
|
|
| }
|
|
|
|
|
? TypeMismatchError
|
|
|
|
|
|
2018-02-06 11:34:21 +00:00
|
|
|
|
| byte table[256] many
|
2015-10-18 17:12:47 +00:00
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs many
|
|
|
|
|
| trashes a, x, n, z
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| ld a, 0
|
|
|
|
|
| st a, many + x
|
|
|
|
|
| ld a, many
|
|
|
|
|
| }
|
|
|
|
|
? TypeMismatchError
|
|
|
|
|
|
2018-02-06 11:34:21 +00:00
|
|
|
|
| byte table[256] many
|
2015-10-18 17:12:47 +00:00
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs many
|
|
|
|
|
| trashes a, x, n, z
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| ld a, 0
|
|
|
|
|
| st a, many + x
|
|
|
|
|
| ld a, many + x
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
2018-02-06 11:34:21 +00:00
|
|
|
|
| byte table[256] many
|
2018-02-05 12:35:19 +00:00
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs many
|
|
|
|
|
| outputs many
|
|
|
|
|
| trashes a, x, n, z
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| ld a, many + x
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
The index must be initialized.
|
|
|
|
|
|
2018-02-06 11:34:21 +00:00
|
|
|
|
| byte table[256] many
|
2018-02-05 12:35:19 +00:00
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs many
|
|
|
|
|
| outputs many
|
|
|
|
|
| trashes a, x, n, z
|
|
|
|
|
| {
|
|
|
|
|
| ld a, many + x
|
|
|
|
|
| }
|
|
|
|
|
? UnmeaningfulReadError: x
|
|
|
|
|
|
2018-04-18 14:45:47 +00:00
|
|
|
|
There are other operations you can do on tables. (1/3)
|
2018-04-18 13:21:03 +00:00
|
|
|
|
|
|
|
|
|
| byte table[256] many
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs many
|
|
|
|
|
| outputs many
|
2018-04-18 14:45:47 +00:00
|
|
|
|
| trashes a, x, c, n, z, v
|
2018-04-18 13:21:03 +00:00
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| ld a, 0
|
|
|
|
|
| st off, c
|
|
|
|
|
| add a, many + x
|
|
|
|
|
| sub a, many + x
|
|
|
|
|
| cmp a, many + x
|
2018-04-18 14:45:47 +00:00
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
There are other operations you can do on tables. (2/3)
|
|
|
|
|
|
|
|
|
|
| byte table[256] many
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs many
|
|
|
|
|
| outputs many
|
|
|
|
|
| trashes a, x, c, n, z
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| ld a, 0
|
2018-04-18 13:21:03 +00:00
|
|
|
|
| and a, many + x
|
|
|
|
|
| or a, many + x
|
|
|
|
|
| xor a, many + x
|
2018-04-18 14:45:47 +00:00
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
There are other operations you can do on tables. (3/3)
|
|
|
|
|
|
|
|
|
|
| byte table[256] many
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs many
|
|
|
|
|
| outputs many
|
|
|
|
|
| trashes a, x, c, n, z
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| ld a, 0
|
|
|
|
|
| st off, c
|
2018-04-18 13:21:03 +00:00
|
|
|
|
| shl many + x
|
|
|
|
|
| shr many + x
|
|
|
|
|
| inc many + x
|
|
|
|
|
| dec many + x
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
2017-12-08 13:41:48 +00:00
|
|
|
|
Copying to and from a word table.
|
2016-06-16 16:08:57 +00:00
|
|
|
|
|
2017-12-08 13:41:48 +00:00
|
|
|
|
| word one
|
2018-02-06 11:34:21 +00:00
|
|
|
|
| word table[256] many
|
2016-06-16 16:08:57 +00:00
|
|
|
|
|
|
|
|
|
|
| routine main
|
2017-12-08 13:41:48 +00:00
|
|
|
|
| inputs one, many
|
|
|
|
|
| outputs one, many
|
|
|
|
|
| trashes a, x, n, z
|
2016-06-16 16:08:57 +00:00
|
|
|
|
| {
|
2017-12-08 13:41:48 +00:00
|
|
|
|
| ld x, 0
|
|
|
|
|
| copy one, many + x
|
|
|
|
|
| copy many + x, one
|
2016-06-16 16:08:57 +00:00
|
|
|
|
| }
|
2017-12-08 13:41:48 +00:00
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| word one
|
2018-02-06 11:52:50 +00:00
|
|
|
|
| word table[256] many
|
2017-12-08 13:41:48 +00:00
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs one, many
|
|
|
|
|
| outputs one, many
|
|
|
|
|
| trashes a, x, n, z
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| copy one, many
|
|
|
|
|
| }
|
|
|
|
|
? TypeMismatchError
|
|
|
|
|
|
|
|
|
|
| word one
|
2018-02-06 11:52:50 +00:00
|
|
|
|
| word table[256] many
|
2017-12-08 13:41:48 +00:00
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs one, many
|
|
|
|
|
| outputs one, many
|
|
|
|
|
| trashes a, x, n, z
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| copy one + x, many
|
|
|
|
|
| }
|
|
|
|
|
? TypeMismatchError
|
2016-06-16 16:08:57 +00:00
|
|
|
|
|
2017-12-12 16:04:59 +00:00
|
|
|
|
You can also copy a literal word to a word table.
|
2018-02-14 13:41:48 +00:00
|
|
|
|
(Even if the table has fewer than 256 entries.)
|
2017-12-12 16:04:59 +00:00
|
|
|
|
|
2018-02-14 13:41:48 +00:00
|
|
|
|
| word table[32] many
|
2017-12-12 16:04:59 +00:00
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs many
|
|
|
|
|
| outputs many
|
|
|
|
|
| trashes a, x, n, z
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| copy 9999, many + x
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
2018-02-14 13:41:48 +00:00
|
|
|
|
#### tables: range checking ####
|
|
|
|
|
|
2018-02-16 12:28:11 +00:00
|
|
|
|
It is a static analysis error if it cannot be proven that a read or write
|
|
|
|
|
to a table falls within the defined size of that table.
|
|
|
|
|
|
|
|
|
|
(If a table has 256 entries, then there is never a problem, because a byte
|
|
|
|
|
cannot index any entry outside of 0..255.)
|
|
|
|
|
|
|
|
|
|
A SixtyPical implementation must be able to prove that the index is inside
|
|
|
|
|
the range of the table in various ways. The simplest is to show that a
|
|
|
|
|
constant value falls inside or outside the range of the table.
|
2018-02-14 13:41:48 +00:00
|
|
|
|
|
|
|
|
|
| byte table[32] many
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs many
|
|
|
|
|
| outputs many
|
|
|
|
|
| trashes a, x, n, z
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 31
|
|
|
|
|
| ld a, many + x
|
|
|
|
|
| st a, many + x
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| byte table[32] many
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs many
|
|
|
|
|
| outputs many
|
|
|
|
|
| trashes a, x, n, z
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 32
|
|
|
|
|
| ld a, many + x
|
|
|
|
|
| }
|
2018-02-14 15:54:57 +00:00
|
|
|
|
? RangeExceededError
|
2018-02-14 13:41:48 +00:00
|
|
|
|
|
|
|
|
|
| byte table[32] many
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs many
|
|
|
|
|
| outputs many
|
|
|
|
|
| trashes a, x, n, z
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 32
|
|
|
|
|
| ld a, 0
|
|
|
|
|
| st a, many + x
|
|
|
|
|
| }
|
2018-02-14 15:54:57 +00:00
|
|
|
|
? RangeExceededError
|
2018-02-14 13:41:48 +00:00
|
|
|
|
|
2018-02-16 12:28:11 +00:00
|
|
|
|
This applies to `copy` as well.
|
|
|
|
|
|
|
|
|
|
| word one: 77
|
|
|
|
|
| word table[32] many
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs many, one
|
|
|
|
|
| outputs many, one
|
|
|
|
|
| trashes a, x, n, z
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 31
|
|
|
|
|
| copy one, many + x
|
|
|
|
|
| copy many + x, one
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| word one: 77
|
|
|
|
|
| word table[32] many
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs many, one
|
|
|
|
|
| outputs many, one
|
|
|
|
|
| trashes a, x, n, z
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 32
|
|
|
|
|
| copy many + x, one
|
|
|
|
|
| }
|
|
|
|
|
? RangeExceededError
|
|
|
|
|
|
|
|
|
|
| word one: 77
|
|
|
|
|
| word table[32] many
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs many, one
|
|
|
|
|
| outputs many, one
|
|
|
|
|
| trashes a, x, n, z
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 32
|
|
|
|
|
| copy one, many + x
|
|
|
|
|
| }
|
|
|
|
|
? RangeExceededError
|
|
|
|
|
|
2018-03-06 15:42:12 +00:00
|
|
|
|
`AND`'ing a register with a value ensures the range of the
|
|
|
|
|
register will not exceed the range of the value. This can
|
|
|
|
|
be used to "clip" the range of an index so that it fits in
|
|
|
|
|
a table.
|
|
|
|
|
|
|
|
|
|
| word one: 77
|
|
|
|
|
| word table[32] many
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs a, many, one
|
|
|
|
|
| outputs many, one
|
|
|
|
|
| trashes a, x, n, z
|
|
|
|
|
| {
|
|
|
|
|
| and a, 31
|
|
|
|
|
| ld x, a
|
|
|
|
|
| copy one, many + x
|
|
|
|
|
| copy many + x, one
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
Test for "clipping", but not enough.
|
|
|
|
|
|
|
|
|
|
| word one: 77
|
|
|
|
|
| word table[32] many
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs a, many, one
|
|
|
|
|
| outputs many, one
|
|
|
|
|
| trashes a, x, n, z
|
|
|
|
|
| {
|
|
|
|
|
| and a, 63
|
|
|
|
|
| ld x, a
|
|
|
|
|
| copy one, many + x
|
|
|
|
|
| copy many + x, one
|
|
|
|
|
| }
|
|
|
|
|
? RangeExceededError
|
|
|
|
|
|
|
|
|
|
If you alter the value after "clipping" it, the range can
|
|
|
|
|
no longer be guaranteed.
|
|
|
|
|
|
|
|
|
|
| word one: 77
|
|
|
|
|
| word table[32] many
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs a, many, one
|
|
|
|
|
| outputs many, one
|
|
|
|
|
| trashes a, x, n, z
|
|
|
|
|
| {
|
|
|
|
|
| and a, 31
|
|
|
|
|
| ld x, a
|
|
|
|
|
| inc x
|
|
|
|
|
| copy one, many + x
|
|
|
|
|
| copy many + x, one
|
|
|
|
|
| }
|
|
|
|
|
? RangeExceededError
|
|
|
|
|
|
2015-10-16 08:30:24 +00:00
|
|
|
|
### add ###
|
|
|
|
|
|
|
|
|
|
Can't `add` from or to a memory location that isn't initialized.
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs a
|
|
|
|
|
| outputs a
|
|
|
|
|
| trashes c, z, v, n
|
|
|
|
|
| {
|
|
|
|
|
| st off, c
|
|
|
|
|
| add a, 0
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| byte lives
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs a
|
|
|
|
|
| outputs a
|
|
|
|
|
| trashes c, z, v, n
|
|
|
|
|
| {
|
|
|
|
|
| st off, c
|
|
|
|
|
| add a, lives
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulReadError: lives
|
2015-10-16 08:30:24 +00:00
|
|
|
|
|
|
|
|
|
| byte lives
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs lives
|
|
|
|
|
| outputs a
|
|
|
|
|
| trashes c, z, v, n
|
|
|
|
|
| {
|
|
|
|
|
| st off, c
|
|
|
|
|
| add a, lives
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulReadError: a
|
2015-10-16 08:30:24 +00:00
|
|
|
|
|
|
|
|
|
Can't `add` to a memory location that isn't writeable.
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs a
|
|
|
|
|
| trashes c
|
|
|
|
|
| {
|
|
|
|
|
| st off, c
|
|
|
|
|
| add a, 0
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? ForbiddenWriteError: a
|
2015-10-16 08:30:24 +00:00
|
|
|
|
|
2017-12-07 11:31:46 +00:00
|
|
|
|
You can `add` a word constant to a word memory location.
|
|
|
|
|
|
|
|
|
|
| word score
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs a, score
|
|
|
|
|
| outputs score
|
|
|
|
|
| trashes a, c, z, v, n
|
|
|
|
|
| {
|
|
|
|
|
| st off, c
|
|
|
|
|
| add score, 1999
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
`add`ing a word constant to a word memory location trashes `a`.
|
|
|
|
|
|
|
|
|
|
| word score
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs a, score
|
|
|
|
|
| outputs score, a
|
|
|
|
|
| trashes c, z, v, n
|
|
|
|
|
| {
|
|
|
|
|
| st off, c
|
|
|
|
|
| add score, 1999
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulOutputError: a
|
2017-12-07 11:31:46 +00:00
|
|
|
|
|
2017-12-07 11:49:15 +00:00
|
|
|
|
To be sure, `add`ing a word constant to a word memory location trashes `a`.
|
2017-12-07 11:31:46 +00:00
|
|
|
|
|
|
|
|
|
| word score
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs score
|
|
|
|
|
| outputs score
|
|
|
|
|
| trashes c, z, v, n
|
|
|
|
|
| {
|
|
|
|
|
| st off, c
|
|
|
|
|
| add score, 1999
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? ForbiddenWriteError: a
|
2017-12-07 11:31:46 +00:00
|
|
|
|
|
2017-12-07 13:15:21 +00:00
|
|
|
|
You can `add` a word memory location to another word memory location.
|
|
|
|
|
|
|
|
|
|
| word score
|
|
|
|
|
| word delta
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs score, delta
|
|
|
|
|
| outputs score
|
|
|
|
|
| trashes a, c, z, v, n
|
|
|
|
|
| {
|
|
|
|
|
| st off, c
|
|
|
|
|
| add score, delta
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
`add`ing a word memory location to a word memory location trashes `a`.
|
|
|
|
|
|
|
|
|
|
| word score
|
|
|
|
|
| word delta
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs score, delta
|
|
|
|
|
| outputs score
|
|
|
|
|
| trashes c, z, v, n
|
|
|
|
|
| {
|
|
|
|
|
| st off, c
|
|
|
|
|
| add score, delta
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? ForbiddenWriteError: a
|
2017-12-07 13:15:21 +00:00
|
|
|
|
|
2017-12-07 14:14:43 +00:00
|
|
|
|
You can `add` a word memory location, or a constant, to a pointer.
|
|
|
|
|
|
|
|
|
|
| pointer ptr
|
|
|
|
|
| word delta
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs ptr, delta
|
|
|
|
|
| outputs ptr
|
|
|
|
|
| trashes a, c, z, v, n
|
|
|
|
|
| {
|
|
|
|
|
| st off, c
|
|
|
|
|
| add ptr, delta
|
|
|
|
|
| add ptr, word 1
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
`add`ing a word memory location, or a constant, to a pointer, trashes `a`.
|
|
|
|
|
|
|
|
|
|
| pointer ptr
|
|
|
|
|
| word delta
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs ptr, delta
|
|
|
|
|
| outputs ptr
|
|
|
|
|
| trashes c, z, v, n
|
|
|
|
|
| {
|
|
|
|
|
| st off, c
|
|
|
|
|
| add ptr, delta
|
|
|
|
|
| add ptr, word 1
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? ForbiddenWriteError: a
|
2017-12-07 14:14:43 +00:00
|
|
|
|
|
2015-10-16 18:15:01 +00:00
|
|
|
|
### sub ###
|
|
|
|
|
|
|
|
|
|
Can't `sub` from or to a memory location that isn't initialized.
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs a
|
|
|
|
|
| outputs a
|
|
|
|
|
| trashes c, z, v, n
|
|
|
|
|
| {
|
|
|
|
|
| st off, c
|
|
|
|
|
| sub a, 0
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| byte lives
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs a
|
|
|
|
|
| outputs a
|
|
|
|
|
| trashes c, z, v, n
|
|
|
|
|
| {
|
|
|
|
|
| st off, c
|
|
|
|
|
| sub a, lives
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulReadError: lives
|
2015-10-16 18:15:01 +00:00
|
|
|
|
|
|
|
|
|
| byte lives
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs lives
|
|
|
|
|
| outputs a
|
|
|
|
|
| trashes c, z, v, n
|
|
|
|
|
| {
|
|
|
|
|
| st off, c
|
|
|
|
|
| sub a, lives
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulReadError: a
|
2015-10-16 18:15:01 +00:00
|
|
|
|
|
|
|
|
|
Can't `sub` to a memory location that isn't writeable.
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs a
|
|
|
|
|
| trashes c
|
|
|
|
|
| {
|
|
|
|
|
| st off, c
|
|
|
|
|
| sub a, 0
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? ForbiddenWriteError: a
|
2015-10-16 18:15:01 +00:00
|
|
|
|
|
2017-12-13 14:07:59 +00:00
|
|
|
|
You can `sub` a word constant from a word memory location.
|
|
|
|
|
|
|
|
|
|
| word score
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs a, score
|
|
|
|
|
| outputs score
|
|
|
|
|
| trashes a, c, z, v, n
|
|
|
|
|
| {
|
|
|
|
|
| st on, c
|
|
|
|
|
| sub score, 1999
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
`sub`ing a word constant from a word memory location trashes `a`.
|
|
|
|
|
|
|
|
|
|
| word score
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs a, score
|
|
|
|
|
| outputs score, a
|
|
|
|
|
| trashes c, z, v, n
|
|
|
|
|
| {
|
|
|
|
|
| st on, c
|
|
|
|
|
| sub score, 1999
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulOutputError: a
|
2017-12-13 14:07:59 +00:00
|
|
|
|
|
|
|
|
|
You can `sub` a word memory location from another word memory location.
|
|
|
|
|
|
|
|
|
|
| word score
|
|
|
|
|
| word delta
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs score, delta
|
|
|
|
|
| outputs score
|
|
|
|
|
| trashes a, c, z, v, n
|
|
|
|
|
| {
|
|
|
|
|
| st off, c
|
|
|
|
|
| sub score, delta
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
`sub`ing a word memory location from a word memory location trashes `a`.
|
|
|
|
|
|
|
|
|
|
| word score
|
|
|
|
|
| word delta
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs score, delta
|
|
|
|
|
| outputs score
|
|
|
|
|
| trashes c, z, v, n
|
|
|
|
|
| {
|
|
|
|
|
| st off, c
|
|
|
|
|
| sub score, delta
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? ForbiddenWriteError: a
|
2017-12-13 14:07:59 +00:00
|
|
|
|
|
2015-10-16 18:15:01 +00:00
|
|
|
|
### inc ###
|
|
|
|
|
|
|
|
|
|
Location must be initialized and writeable.
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| inc x
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulReadError: x
|
2015-10-16 18:15:01 +00:00
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs x
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| inc x
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? ForbiddenWriteError: x
|
2015-10-16 18:15:01 +00:00
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs x
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| inc x
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
2016-06-16 16:08:57 +00:00
|
|
|
|
Can't `inc` a `word` type.
|
|
|
|
|
|
|
|
|
|
| word foo
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs foo
|
|
|
|
|
| outputs foo
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| inc foo
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? TypeMismatchError: foo
|
2016-06-16 16:08:57 +00:00
|
|
|
|
|
2015-10-16 18:15:01 +00:00
|
|
|
|
### dec ###
|
|
|
|
|
|
|
|
|
|
Location must be initialized and writeable.
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| dec x
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulReadError: x
|
2015-10-16 18:15:01 +00:00
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs x
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| dec x
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? ForbiddenWriteError: x
|
2015-10-16 18:15:01 +00:00
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs x
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| dec x
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
2016-06-16 16:08:57 +00:00
|
|
|
|
Can't `dec` a `word` type.
|
|
|
|
|
|
|
|
|
|
| word foo
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs foo
|
|
|
|
|
| outputs foo
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| dec foo
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? TypeMismatchError: foo
|
2016-06-16 16:08:57 +00:00
|
|
|
|
|
2015-10-16 18:15:01 +00:00
|
|
|
|
### cmp ###
|
|
|
|
|
|
2018-03-28 13:52:16 +00:00
|
|
|
|
Some rudimentary tests for `cmp`.
|
2015-10-16 18:32:18 +00:00
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs a
|
|
|
|
|
| trashes z, c, n
|
|
|
|
|
| {
|
|
|
|
|
| cmp a, 4
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs a
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| cmp a, 4
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? ForbiddenWriteError: c
|
2015-10-16 18:32:18 +00:00
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| trashes z, c, n
|
|
|
|
|
| {
|
|
|
|
|
| cmp a, 4
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulReadError: a
|
2015-10-16 18:15:01 +00:00
|
|
|
|
|
|
|
|
|
### and ###
|
|
|
|
|
|
2018-03-28 13:52:16 +00:00
|
|
|
|
Some rudimentary tests for `and`.
|
2015-10-16 18:32:18 +00:00
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs a
|
|
|
|
|
| outputs a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| and a, 4
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs a
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| and a, 4
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? ForbiddenWriteError: a
|
2015-10-16 18:32:18 +00:00
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| and a, 4
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulReadError: a
|
2015-10-16 18:15:01 +00:00
|
|
|
|
|
|
|
|
|
### or ###
|
|
|
|
|
|
2018-03-28 13:52:16 +00:00
|
|
|
|
Some rudimentary tests for `or`.
|
2015-10-16 18:32:18 +00:00
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs a
|
|
|
|
|
| outputs a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| or a, 4
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs a
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| or a, 4
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? ForbiddenWriteError: a
|
2015-10-16 18:32:18 +00:00
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| or a, 4
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulReadError: a
|
2015-10-16 18:15:01 +00:00
|
|
|
|
|
|
|
|
|
### xor ###
|
|
|
|
|
|
2018-03-28 13:52:16 +00:00
|
|
|
|
Some rudimentary tests for `xor`.
|
2015-10-16 18:32:18 +00:00
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs a
|
|
|
|
|
| outputs a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| xor a, 4
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs a
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| xor a, 4
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? ForbiddenWriteError: a
|
2015-10-16 18:32:18 +00:00
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| xor a, 4
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulReadError: a
|
2015-10-16 18:15:01 +00:00
|
|
|
|
|
|
|
|
|
### shl ###
|
|
|
|
|
|
2018-03-28 13:52:16 +00:00
|
|
|
|
Some rudimentary tests for `shl`.
|
2015-10-16 18:32:18 +00:00
|
|
|
|
|
2018-04-18 11:01:53 +00:00
|
|
|
|
| byte foo
|
2015-10-16 18:32:18 +00:00
|
|
|
|
| routine main
|
2018-04-18 11:01:53 +00:00
|
|
|
|
| inputs foo, a, c
|
|
|
|
|
| outputs foo, a, c, z, n
|
2015-10-16 18:32:18 +00:00
|
|
|
|
| {
|
|
|
|
|
| shl a
|
2018-04-18 11:01:53 +00:00
|
|
|
|
| shl foo
|
2015-10-16 18:32:18 +00:00
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs a, c
|
|
|
|
|
| outputs c, z, n
|
|
|
|
|
| {
|
|
|
|
|
| shl a
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? ForbiddenWriteError: a
|
2015-10-16 18:32:18 +00:00
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs a
|
|
|
|
|
| outputs a, c, z, n
|
|
|
|
|
| {
|
|
|
|
|
| shl a
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulReadError: c
|
2015-10-16 18:15:01 +00:00
|
|
|
|
|
|
|
|
|
### shr ###
|
|
|
|
|
|
2018-03-28 13:52:16 +00:00
|
|
|
|
Some rudimentary tests for `shr`.
|
2015-10-16 18:32:18 +00:00
|
|
|
|
|
2018-04-18 11:01:53 +00:00
|
|
|
|
| byte foo
|
2015-10-16 18:32:18 +00:00
|
|
|
|
| routine main
|
2018-04-18 11:01:53 +00:00
|
|
|
|
| inputs foo, a, c
|
|
|
|
|
| outputs foo, a, c, z, n
|
2015-10-16 18:32:18 +00:00
|
|
|
|
| {
|
|
|
|
|
| shr a
|
2018-04-18 11:01:53 +00:00
|
|
|
|
| shr foo
|
2015-10-16 18:32:18 +00:00
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs a, c
|
|
|
|
|
| outputs c, z, n
|
|
|
|
|
| {
|
|
|
|
|
| shr a
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? ForbiddenWriteError: a
|
2015-10-16 18:32:18 +00:00
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs a
|
|
|
|
|
| outputs a, c, z, n
|
|
|
|
|
| {
|
|
|
|
|
| shr a
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulReadError: c
|
2015-10-16 08:30:24 +00:00
|
|
|
|
|
2018-03-28 13:52:16 +00:00
|
|
|
|
### nop ###
|
|
|
|
|
|
|
|
|
|
Some rudimentary tests for `nop`.
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| {
|
|
|
|
|
| nop
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
2015-10-16 08:30:24 +00:00
|
|
|
|
### call ###
|
|
|
|
|
|
|
|
|
|
When calling a routine, all of the locations it lists as inputs must be
|
|
|
|
|
initialized.
|
|
|
|
|
|
|
|
|
|
| byte lives
|
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| inputs x
|
|
|
|
|
| trashes lives
|
|
|
|
|
| {
|
|
|
|
|
| st x, lives
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| {
|
|
|
|
|
| call foo
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulReadError: x
|
2015-10-16 08:30:24 +00:00
|
|
|
|
|
|
|
|
|
Note that if you call a routine that trashes a location, you also trash it.
|
|
|
|
|
|
|
|
|
|
| byte lives
|
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| inputs x
|
|
|
|
|
| trashes lives
|
|
|
|
|
| {
|
|
|
|
|
| st x, lives
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs x, z, n
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| call foo
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? ForbiddenWriteError: lives
|
2015-10-16 08:30:24 +00:00
|
|
|
|
|
|
|
|
|
| byte lives
|
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| inputs x
|
|
|
|
|
| trashes lives
|
|
|
|
|
| {
|
|
|
|
|
| st x, lives
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs x, z, n
|
|
|
|
|
| trashes lives
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| call foo
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
You can't output a value that the thing you called trashed.
|
|
|
|
|
|
|
|
|
|
| byte lives
|
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| inputs x
|
|
|
|
|
| trashes lives
|
|
|
|
|
| {
|
|
|
|
|
| st x, lives
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs x, z, n, lives
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| call foo
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulOutputError: lives
|
2015-10-16 08:30:24 +00:00
|
|
|
|
|
|
|
|
|
...unless you write to it yourself afterwards.
|
|
|
|
|
|
|
|
|
|
| byte lives
|
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| inputs x
|
|
|
|
|
| trashes lives
|
|
|
|
|
| {
|
|
|
|
|
| st x, lives
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs x, z, n, lives
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| call foo
|
|
|
|
|
| st x, lives
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
If a routine declares outputs, they are initialized in the caller after
|
|
|
|
|
calling it.
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| outputs x, z, n
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs a
|
|
|
|
|
| trashes x, z, n
|
|
|
|
|
| {
|
|
|
|
|
| call foo
|
|
|
|
|
| ld a, x
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| {
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs a
|
|
|
|
|
| trashes x
|
|
|
|
|
| {
|
|
|
|
|
| call foo
|
|
|
|
|
| ld a, x
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulReadError: x
|
2015-10-16 08:30:24 +00:00
|
|
|
|
|
|
|
|
|
If a routine trashes locations, they are uninitialized in the caller after
|
|
|
|
|
calling it.
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| trashes x, z, n
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| trashes x, z, n
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs a
|
|
|
|
|
| trashes x, z, n
|
|
|
|
|
| {
|
|
|
|
|
| call foo
|
|
|
|
|
| ld a, x
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulReadError: x
|
2015-10-16 08:30:24 +00:00
|
|
|
|
|
2015-10-17 13:54:28 +00:00
|
|
|
|
Calling an extern is just the same as calling a defined routine with the
|
|
|
|
|
same constraints.
|
|
|
|
|
|
|
|
|
|
| routine chrout
|
|
|
|
|
| inputs a
|
|
|
|
|
| trashes a
|
|
|
|
|
| @ 65490
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| ld a, 65
|
|
|
|
|
| call chrout
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| routine chrout
|
|
|
|
|
| inputs a
|
|
|
|
|
| trashes a
|
|
|
|
|
| @ 65490
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| call chrout
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulReadError: a
|
2015-10-17 13:54:28 +00:00
|
|
|
|
|
|
|
|
|
| routine chrout
|
|
|
|
|
| inputs a
|
|
|
|
|
| trashes a
|
|
|
|
|
| @ 65490
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| trashes a, x, z, n
|
|
|
|
|
| {
|
|
|
|
|
| ld a, 65
|
|
|
|
|
| call chrout
|
|
|
|
|
| ld x, a
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulReadError: a
|
2015-10-17 13:54:28 +00:00
|
|
|
|
|
2017-12-13 15:23:06 +00:00
|
|
|
|
### trash ###
|
|
|
|
|
|
|
|
|
|
Trash does nothing except indicate that we do not care about the value anymore.
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| inputs a
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| st a, x
|
|
|
|
|
| ld a, 0
|
|
|
|
|
| trash a
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| inputs a
|
|
|
|
|
| outputs a, x
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| st a, x
|
|
|
|
|
| ld a, 0
|
|
|
|
|
| trash a
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulOutputError: a
|
2017-12-13 15:23:06 +00:00
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| inputs a
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| st a, x
|
|
|
|
|
| trash a
|
|
|
|
|
| st a, x
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulReadError: a
|
2017-12-13 15:23:06 +00:00
|
|
|
|
|
2015-10-16 08:30:24 +00:00
|
|
|
|
### if ###
|
|
|
|
|
|
|
|
|
|
Both blocks of an `if` are analyzed.
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| inputs a
|
2015-10-16 08:38:38 +00:00
|
|
|
|
| outputs x
|
|
|
|
|
| trashes a, z, n, c
|
2015-10-16 08:30:24 +00:00
|
|
|
|
| {
|
|
|
|
|
| cmp a, 42
|
|
|
|
|
| if z {
|
2015-10-16 08:38:38 +00:00
|
|
|
|
| ld x, 7
|
2015-10-16 08:30:24 +00:00
|
|
|
|
| } else {
|
2015-10-16 08:38:38 +00:00
|
|
|
|
| ld x, 23
|
2015-10-16 08:30:24 +00:00
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
2015-10-16 13:01:45 +00:00
|
|
|
|
|
|
|
|
|
If a location is initialized in one block, is must be initialized in the other as well.
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| inputs a
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes a, z, n, c
|
|
|
|
|
| {
|
|
|
|
|
| cmp a, 42
|
|
|
|
|
| if z {
|
|
|
|
|
| ld x, 7
|
|
|
|
|
| } else {
|
|
|
|
|
| ld a, 23
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
? InconsistentInitializationError: x
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| inputs a
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes a, z, n, c
|
|
|
|
|
| {
|
|
|
|
|
| cmp a, 42
|
|
|
|
|
| if z {
|
|
|
|
|
| ld a, 6
|
|
|
|
|
| } else {
|
|
|
|
|
| ld x, 7
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
? InconsistentInitializationError: x
|
|
|
|
|
|
2015-10-18 14:32:28 +00:00
|
|
|
|
| routine foo
|
|
|
|
|
| inputs a
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes a, z, n, c
|
|
|
|
|
| {
|
|
|
|
|
| cmp a, 42
|
|
|
|
|
| if not z {
|
|
|
|
|
| ld a, 6
|
|
|
|
|
| } else {
|
|
|
|
|
| ld x, 7
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
? InconsistentInitializationError: x
|
|
|
|
|
|
2017-12-12 14:55:57 +00:00
|
|
|
|
However, this only pertains to initialization. If a value is already
|
|
|
|
|
initialized, either because it was set previous to the `if`, or is an
|
|
|
|
|
input to the routine, and it is initialized in one branch, it need not
|
|
|
|
|
be initialized in the other.
|
|
|
|
|
|
2018-02-08 12:46:28 +00:00
|
|
|
|
| routine foo
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes a, z, n, c
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| ld a, 0
|
|
|
|
|
| cmp a, 42
|
|
|
|
|
| if z {
|
|
|
|
|
| ld x, 7
|
|
|
|
|
| } else {
|
|
|
|
|
| ld a, 23
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
2017-12-12 14:55:57 +00:00
|
|
|
|
| routine foo
|
|
|
|
|
| inputs x
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes a, z, n, c
|
|
|
|
|
| {
|
|
|
|
|
| ld a, 0
|
|
|
|
|
| cmp a, 42
|
|
|
|
|
| if z {
|
|
|
|
|
| ld x, 7
|
|
|
|
|
| } else {
|
|
|
|
|
| ld a, 23
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
2015-10-16 13:01:45 +00:00
|
|
|
|
An `if` with a single block is analyzed as if it had an empty `else` block.
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| inputs a
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes a, z, n, c
|
|
|
|
|
| {
|
|
|
|
|
| cmp a, 42
|
|
|
|
|
| if z {
|
|
|
|
|
| ld x, 7
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
? InconsistentInitializationError: x
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| inputs a
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes a, z, n, c
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| cmp a, 42
|
|
|
|
|
| if z {
|
|
|
|
|
| ld x, 7
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
2015-10-18 12:37:35 +00:00
|
|
|
|
|
2015-10-18 14:32:28 +00:00
|
|
|
|
| routine foo
|
|
|
|
|
| inputs a
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes a, z, n, c
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| cmp a, 42
|
|
|
|
|
| if not z {
|
|
|
|
|
| ld x, 7
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
2018-02-08 12:46:28 +00:00
|
|
|
|
The cardinal rule for trashes in an `if` is the "union rule": if one branch
|
|
|
|
|
trashes {`a`} and the other branch trashes {`b`} then the whole `if` statement
|
|
|
|
|
trashes {`a`, `b`}.
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| inputs a, x, z
|
|
|
|
|
| trashes a, x
|
|
|
|
|
| {
|
|
|
|
|
| if z {
|
|
|
|
|
| trash a
|
|
|
|
|
| } else {
|
|
|
|
|
| trash x
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| inputs a, x, z
|
|
|
|
|
| trashes a
|
|
|
|
|
| {
|
|
|
|
|
| if z {
|
|
|
|
|
| trash a
|
|
|
|
|
| } else {
|
|
|
|
|
| trash x
|
|
|
|
|
| }
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? ForbiddenWriteError: x (in foo, line 10)
|
2018-02-08 12:46:28 +00:00
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| inputs a, x, z
|
|
|
|
|
| trashes x
|
|
|
|
|
| {
|
|
|
|
|
| if z {
|
|
|
|
|
| trash a
|
|
|
|
|
| } else {
|
|
|
|
|
| trash x
|
|
|
|
|
| }
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? ForbiddenWriteError: a (in foo, line 10)
|
2018-02-08 12:46:28 +00:00
|
|
|
|
|
2015-10-18 12:37:35 +00:00
|
|
|
|
### repeat ###
|
|
|
|
|
|
|
|
|
|
Repeat loop.
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs x, y, n, z, c
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| ld y, 15
|
|
|
|
|
| repeat {
|
|
|
|
|
| inc x
|
|
|
|
|
| inc y
|
|
|
|
|
| cmp x, 10
|
|
|
|
|
| } until z
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
You can initialize something inside the loop that was uninitialized outside.
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs x, y, n, z, c
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| repeat {
|
|
|
|
|
| ld y, 15
|
|
|
|
|
| inc x
|
|
|
|
|
| cmp x, 10
|
|
|
|
|
| } until z
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
But you can't UNinitialize something at the end of the loop that you need
|
|
|
|
|
initialized at the start.
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| trashes y
|
|
|
|
|
| {
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs x, y, n, z, c
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| ld y, 15
|
|
|
|
|
| repeat {
|
|
|
|
|
| inc x
|
|
|
|
|
| inc y
|
|
|
|
|
| call foo
|
|
|
|
|
| cmp x, 10
|
|
|
|
|
| } until z
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulReadError: y
|
2015-10-18 19:16:14 +00:00
|
|
|
|
|
2017-11-20 16:39:39 +00:00
|
|
|
|
And if you trash the test expression (i.e. `z` in the below) inside the loop,
|
|
|
|
|
this is an error too.
|
|
|
|
|
|
|
|
|
|
| word one : 0
|
|
|
|
|
| word two : 0
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs one, two
|
|
|
|
|
| outputs two
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| repeat {
|
|
|
|
|
| copy one, two
|
|
|
|
|
| } until z
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulReadError: z
|
2017-11-20 16:39:39 +00:00
|
|
|
|
|
2017-12-12 15:18:59 +00:00
|
|
|
|
The body of `repeat forever` can be empty.
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| {
|
|
|
|
|
| repeat {
|
|
|
|
|
| } forever
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
2018-03-29 12:33:06 +00:00
|
|
|
|
While `repeat` is most often used with `z`, it can also be used with `n`.
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs y, n, z
|
|
|
|
|
| {
|
|
|
|
|
| ld y, 15
|
|
|
|
|
| repeat {
|
|
|
|
|
| dec y
|
|
|
|
|
| } until n
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
2018-03-08 17:17:49 +00:00
|
|
|
|
### for ###
|
|
|
|
|
|
|
|
|
|
Basic "open-faced for" loop. We'll start with the "upto" variant.
|
|
|
|
|
|
|
|
|
|
In a "for" loop, we know the exact range the loop variable takes on.
|
|
|
|
|
|
|
|
|
|
| byte table[16] tab
|
|
|
|
|
|
|
|
|
|
|
| define foo routine inputs tab trashes a, x, c, z, v, n {
|
|
|
|
|
| ld x, 0
|
2018-03-14 09:51:24 +00:00
|
|
|
|
| for x up to 15 {
|
2018-03-08 17:17:49 +00:00
|
|
|
|
| ld a, tab + x
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| byte table[15] tab
|
|
|
|
|
|
|
|
|
|
|
| define foo routine inputs tab trashes a, x, c, z, v, n {
|
|
|
|
|
| ld x, 0
|
2018-03-14 09:51:24 +00:00
|
|
|
|
| for x up to 15 {
|
2018-03-08 17:17:49 +00:00
|
|
|
|
| ld a, tab + x
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
? RangeExceededError
|
|
|
|
|
|
2018-03-26 11:48:57 +00:00
|
|
|
|
You need to initialize the loop variable before the loop.
|
|
|
|
|
|
|
|
|
|
| byte table[16] tab
|
|
|
|
|
|
|
|
|
|
|
| define foo routine inputs tab trashes a, x, c, z, v, n {
|
|
|
|
|
| for x up to 15 {
|
|
|
|
|
| ld a, 0
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
? UnmeaningfulReadError
|
|
|
|
|
|
2018-03-08 17:17:49 +00:00
|
|
|
|
You cannot modify the loop variable in a "for" loop.
|
|
|
|
|
|
|
|
|
|
| byte table[16] tab
|
|
|
|
|
|
|
|
|
|
|
| define foo routine inputs tab trashes a, x, c, z, v, n {
|
|
|
|
|
| ld x, 0
|
2018-03-14 09:51:24 +00:00
|
|
|
|
| for x up to 15 {
|
2018-03-08 17:17:49 +00:00
|
|
|
|
| ld x, 0
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
? ForbiddenWriteError
|
|
|
|
|
|
2018-03-26 16:40:43 +00:00
|
|
|
|
This includes nesting a loop on the same variable.
|
|
|
|
|
|
|
|
|
|
| byte table[16] tab
|
|
|
|
|
|
|
|
|
|
|
| define foo routine inputs tab trashes a, x, c, z, v, n {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| for x up to 8 {
|
|
|
|
|
| for x up to 15 {
|
|
|
|
|
| ld a, 0
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
? ForbiddenWriteError
|
|
|
|
|
|
|
|
|
|
But nesting with two different variables is okay.
|
|
|
|
|
|
|
|
|
|
| byte table[16] tab
|
|
|
|
|
|
|
|
|
|
|
| define foo routine inputs tab trashes a, x, y, c, z, v, n {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| for x up to 8 {
|
|
|
|
|
| ld a, x
|
|
|
|
|
| ld y, a
|
|
|
|
|
| for y up to 15 {
|
|
|
|
|
| ld a, 0
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
Inside the inner loop, the outer variable is still not writeable.
|
|
|
|
|
|
|
|
|
|
| byte table[16] tab
|
|
|
|
|
|
|
|
|
|
|
| define foo routine inputs tab trashes a, x, y, c, z, v, n {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| for x up to 8 {
|
|
|
|
|
| ld a, x
|
|
|
|
|
| ld y, a
|
|
|
|
|
| for y up to 15 {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
? ForbiddenWriteError
|
|
|
|
|
|
2018-03-08 17:17:49 +00:00
|
|
|
|
If the range isn't known to be smaller than the final value, you can't go up to it.
|
|
|
|
|
|
|
|
|
|
| byte table[32] tab
|
|
|
|
|
|
|
|
|
|
|
| define foo routine inputs tab trashes a, x, c, z, v, n {
|
|
|
|
|
| ld x, 16
|
2018-03-14 09:51:24 +00:00
|
|
|
|
| for x up to 15 {
|
2018-03-08 17:17:49 +00:00
|
|
|
|
| ld a, tab + x
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
? RangeExceededError
|
|
|
|
|
|
2018-03-14 09:51:24 +00:00
|
|
|
|
In a "for" loop (downward-counting variant), we know the exact range the loop variable takes on.
|
2018-03-13 17:49:48 +00:00
|
|
|
|
|
|
|
|
|
| byte table[16] tab
|
|
|
|
|
|
|
|
|
|
|
| define foo routine inputs tab trashes a, x, c, z, v, n {
|
|
|
|
|
| ld x, 15
|
2018-03-14 09:51:24 +00:00
|
|
|
|
| for x down to 0 {
|
2018-03-13 17:49:48 +00:00
|
|
|
|
| ld a, tab + x
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| byte table[15] tab
|
|
|
|
|
|
|
|
|
|
|
| define foo routine inputs tab trashes a, x, c, z, v, n {
|
|
|
|
|
| ld x, 15
|
2018-03-14 09:51:24 +00:00
|
|
|
|
| for x down to 0 {
|
2018-03-13 17:49:48 +00:00
|
|
|
|
| ld a, tab + x
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
? RangeExceededError
|
|
|
|
|
|
2018-03-26 11:48:57 +00:00
|
|
|
|
You need to initialize the loop variable before a "for" loop (downward variant).
|
|
|
|
|
|
|
|
|
|
| byte table[16] tab
|
|
|
|
|
|
|
|
|
|
|
| define foo routine inputs tab trashes a, x, c, z, v, n {
|
|
|
|
|
| for x down to 15 {
|
|
|
|
|
| ld a, 0
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
? UnmeaningfulReadError
|
|
|
|
|
|
2018-03-14 09:51:24 +00:00
|
|
|
|
You cannot modify the loop variable in a "for" loop (downward variant).
|
2018-03-13 17:49:48 +00:00
|
|
|
|
|
|
|
|
|
| byte table[16] tab
|
|
|
|
|
|
|
|
|
|
|
| define foo routine inputs tab trashes a, x, c, z, v, n {
|
|
|
|
|
| ld x, 15
|
2018-03-14 09:51:24 +00:00
|
|
|
|
| for x down to 0 {
|
2018-03-13 17:49:48 +00:00
|
|
|
|
| ld x, 0
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
? ForbiddenWriteError
|
|
|
|
|
|
|
|
|
|
If the range isn't known to be larger than the final value, you can't go down to it.
|
|
|
|
|
|
|
|
|
|
| byte table[32] tab
|
|
|
|
|
|
|
|
|
|
|
| define foo routine inputs tab trashes a, x, c, z, v, n {
|
|
|
|
|
| ld x, 0
|
2018-03-14 09:51:24 +00:00
|
|
|
|
| for x down to 0 {
|
2018-03-13 17:49:48 +00:00
|
|
|
|
| ld a, tab + x
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
? RangeExceededError
|
|
|
|
|
|
2018-03-23 17:13:46 +00:00
|
|
|
|
You can initialize something inside the loop that was uninitialized outside.
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs x, y, n, z
|
|
|
|
|
| trashes c
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| for x up to 15 {
|
|
|
|
|
| ld y, 15
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
But you can't UNinitialize something at the end of the loop that you need
|
|
|
|
|
initialized at the start of that loop.
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| trashes y
|
|
|
|
|
| {
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs x, y, n, z
|
|
|
|
|
| trashes c
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| ld y, 15
|
|
|
|
|
| for x up to 15 {
|
|
|
|
|
| inc y
|
|
|
|
|
| call foo
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
? UnmeaningfulReadError: y
|
|
|
|
|
|
2018-04-19 12:18:52 +00:00
|
|
|
|
### save ###
|
|
|
|
|
|
2018-04-20 12:36:08 +00:00
|
|
|
|
Basic neutral test, where the `save` makes no difference.
|
2018-04-19 12:18:52 +00:00
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs a, x
|
|
|
|
|
| outputs a, x
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
2018-04-20 12:36:08 +00:00
|
|
|
|
| ld a, 1
|
|
|
|
|
| save x {
|
|
|
|
|
| ld a, 2
|
|
|
|
|
| }
|
|
|
|
|
| ld a, 3
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
2018-04-25 12:06:11 +00:00
|
|
|
|
Saving any location (other than `a`) will trash `a`.
|
2018-04-20 12:36:08 +00:00
|
|
|
|
|
|
|
|
|
| routine main
|
2018-04-25 12:06:11 +00:00
|
|
|
|
| inputs a, x
|
2018-04-20 12:36:08 +00:00
|
|
|
|
| outputs a, x
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
2018-04-25 12:06:11 +00:00
|
|
|
|
| ld a, 1
|
|
|
|
|
| save x {
|
|
|
|
|
| ld a, 2
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
? UnmeaningfulOutputError
|
|
|
|
|
|
|
|
|
|
Saving `a` does not trash anything.
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs a, x
|
|
|
|
|
| outputs a, x
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 1
|
|
|
|
|
| save a {
|
|
|
|
|
| ld x, 2
|
|
|
|
|
| }
|
|
|
|
|
| ld x, 3
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
A defined value that has been saved can be trashed inside the block.
|
|
|
|
|
It will continue to be defined outside the block.
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs x, y
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
2018-04-20 12:36:08 +00:00
|
|
|
|
| ld x, 0
|
2018-04-19 12:18:52 +00:00
|
|
|
|
| save x {
|
2018-04-25 12:06:11 +00:00
|
|
|
|
| ld y, 0
|
2018-04-20 12:36:08 +00:00
|
|
|
|
| trash x
|
2018-04-19 12:18:52 +00:00
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
2018-04-20 12:36:08 +00:00
|
|
|
|
A trashed value that has been saved can be used inside the block.
|
|
|
|
|
It will continue to be trashed outside the block.
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs a
|
|
|
|
|
| outputs a, x
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| trash x
|
|
|
|
|
| save x {
|
|
|
|
|
| ld a, 0
|
|
|
|
|
| ld x, 1
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
? UnmeaningfulOutputError: x
|
|
|
|
|
|
2018-04-20 13:35:41 +00:00
|
|
|
|
A value which is not output from the routine, is preserved by the
|
|
|
|
|
routine; and can appear in a `save` exactly because a `save` preserves it.
|
|
|
|
|
|
2018-04-25 12:06:11 +00:00
|
|
|
|
| routine main
|
|
|
|
|
| outputs y
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| save x {
|
|
|
|
|
| ld y, 0
|
|
|
|
|
| ld x, 1
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
Because saving anything except `a` trashes `a`, a common idiom is to save `a`
|
|
|
|
|
first in a nested series of `save`s.
|
|
|
|
|
|
2018-04-20 13:35:41 +00:00
|
|
|
|
| routine main
|
|
|
|
|
| inputs a
|
|
|
|
|
| outputs a
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
2018-04-25 12:06:11 +00:00
|
|
|
|
| save a {
|
|
|
|
|
| save x {
|
|
|
|
|
| ld a, 0
|
|
|
|
|
| ld x, 1
|
|
|
|
|
| }
|
2018-04-20 13:35:41 +00:00
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
2018-04-25 12:27:47 +00:00
|
|
|
|
Not just registers, but also user-defined locations can be saved.
|
|
|
|
|
|
|
|
|
|
| byte foo
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| save foo {
|
|
|
|
|
| st 5, foo
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
But only if they are bytes.
|
|
|
|
|
|
|
|
|
|
| word foo
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| save foo {
|
|
|
|
|
| copy 555, foo
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
? TypeMismatchError
|
|
|
|
|
|
|
|
|
|
| byte table[16] tab
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| trashes a, y, z, n
|
|
|
|
|
| {
|
|
|
|
|
| save tab {
|
|
|
|
|
| ld y, 0
|
|
|
|
|
| st 5, tab + y
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
? TypeMismatchError
|
|
|
|
|
|
2018-04-25 12:38:54 +00:00
|
|
|
|
A `goto` cannot appear within a `save` block, even if it is otherwise in tail position.
|
|
|
|
|
|
|
|
|
|
| routine other
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| ld a, 0
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| ld a, 1
|
|
|
|
|
| save x {
|
|
|
|
|
| ld x, 2
|
|
|
|
|
| goto other
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
? IllegalJumpError
|
|
|
|
|
|
|
|
|
|
### with interrupts ###
|
|
|
|
|
|
|
|
|
|
| vector routine
|
|
|
|
|
| inputs x
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| bar
|
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| inputs x
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| inc x
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs bar
|
|
|
|
|
| trashes a, n, z
|
|
|
|
|
| {
|
|
|
|
|
| with interrupts off {
|
|
|
|
|
| copy foo, bar
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
A `goto` cannot appear within a `with interrupts` block, even if it is
|
|
|
|
|
otherwise in tail position.
|
|
|
|
|
|
|
|
|
|
| vector routine
|
|
|
|
|
| inputs x
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| bar
|
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| inputs x
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| inc x
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine other
|
|
|
|
|
| trashes bar, a, n, z
|
|
|
|
|
| {
|
|
|
|
|
| ld a, 0
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| trashes bar, a, n, z
|
|
|
|
|
| {
|
|
|
|
|
| with interrupts off {
|
|
|
|
|
| copy foo, bar
|
|
|
|
|
| goto other
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
? IllegalJumpError
|
|
|
|
|
|
2015-10-18 19:16:14 +00:00
|
|
|
|
### copy ###
|
|
|
|
|
|
|
|
|
|
Can't `copy` from a memory location that isn't initialized.
|
|
|
|
|
|
|
|
|
|
| byte lives
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs x
|
|
|
|
|
| outputs lives
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| copy x, lives
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| byte lives
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs lives
|
|
|
|
|
| trashes x, a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| copy x, lives
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulReadError: x
|
2015-10-18 19:16:14 +00:00
|
|
|
|
|
2015-10-19 12:04:08 +00:00
|
|
|
|
Can't `copy` to a memory location that doesn't appear in (outputs ∪ trashes).
|
2015-10-18 19:16:14 +00:00
|
|
|
|
|
|
|
|
|
| byte lives
|
|
|
|
|
| routine main
|
|
|
|
|
| trashes lives, a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| copy 0, lives
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| byte lives
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs lives
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| copy 0, lives
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| byte lives
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs lives
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| copy 0, lives
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? ForbiddenWriteError: lives
|
2015-10-18 19:16:14 +00:00
|
|
|
|
|
|
|
|
|
a, z, and n are trashed, and must be declared as such
|
|
|
|
|
|
|
|
|
|
| byte lives
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs lives
|
|
|
|
|
| {
|
|
|
|
|
| copy 0, lives
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? ForbiddenWriteError: n
|
2015-10-18 19:16:14 +00:00
|
|
|
|
|
|
|
|
|
a, z, and n are trashed, and must not be declared as outputs.
|
|
|
|
|
|
|
|
|
|
| byte lives
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs lives, a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| copy 0, lives
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulOutputError: n
|
2015-10-19 12:04:08 +00:00
|
|
|
|
|
|
|
|
|
Unless of course you subsequently initialize them.
|
|
|
|
|
|
|
|
|
|
| byte lives
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs lives, a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| copy 0, lives
|
|
|
|
|
| ld a, 0
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
2016-06-16 16:08:57 +00:00
|
|
|
|
Can `copy` from a `byte` to a `byte`.
|
|
|
|
|
|
|
|
|
|
| byte source : 0
|
|
|
|
|
| byte dest
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs source
|
|
|
|
|
| outputs dest
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| copy source, dest
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
2018-02-08 11:04:11 +00:00
|
|
|
|
The understanding is that, because `copy` trashes `a`, `a` cannot be used
|
|
|
|
|
as the destination of a `copy`.
|
|
|
|
|
|
|
|
|
|
| byte source : 0
|
|
|
|
|
| byte dest
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs source
|
|
|
|
|
| outputs dest
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| copy source, a
|
|
|
|
|
| }
|
|
|
|
|
? ForbiddenWriteError
|
|
|
|
|
|
2016-06-16 16:08:57 +00:00
|
|
|
|
Can `copy` from a `word` to a `word`.
|
|
|
|
|
|
|
|
|
|
| word source : 0
|
|
|
|
|
| word dest
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs source
|
|
|
|
|
| outputs dest
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| copy source, dest
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
Can't `copy` from a `byte` to a `word`.
|
|
|
|
|
|
|
|
|
|
| byte source : 0
|
|
|
|
|
| word dest
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs source
|
|
|
|
|
| outputs dest
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| copy source, dest
|
|
|
|
|
| }
|
|
|
|
|
? TypeMismatchError
|
|
|
|
|
|
|
|
|
|
Can't `copy` from a `word` to a `byte`.
|
|
|
|
|
|
|
|
|
|
| word source : 0
|
|
|
|
|
| byte dest
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs source
|
|
|
|
|
| outputs dest
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| copy source, dest
|
|
|
|
|
| }
|
|
|
|
|
? TypeMismatchError
|
|
|
|
|
|
2018-02-08 11:04:11 +00:00
|
|
|
|
### Buffers and pointers ###
|
2017-11-24 12:35:36 +00:00
|
|
|
|
|
2017-12-01 15:10:16 +00:00
|
|
|
|
Note that `^buf` is a constant value, so it by itself does not require `buf` to be
|
|
|
|
|
listed in any input/output sets.
|
|
|
|
|
|
|
|
|
|
However, if the code reads from it through a pointer, it *should* be in `inputs`.
|
|
|
|
|
|
|
|
|
|
Likewise, if the code writes to it through a pointer, it *should* be in `outputs`.
|
|
|
|
|
|
|
|
|
|
Of course, unless you write to *all* the bytes in a buffer, some of those bytes
|
|
|
|
|
might not be meaningful. So how meaningful is this check?
|
|
|
|
|
|
|
|
|
|
This is an open problem.
|
|
|
|
|
|
|
|
|
|
For now, convention says: if it is being read, list it in `inputs`, and if it is
|
|
|
|
|
being modified, list it in both `inputs` and `outputs`.
|
2017-12-01 13:52:56 +00:00
|
|
|
|
|
|
|
|
|
Write literal through a pointer.
|
2017-11-24 12:35:36 +00:00
|
|
|
|
|
|
|
|
|
| buffer[2048] buf
|
|
|
|
|
| pointer ptr
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
2017-12-01 15:10:16 +00:00
|
|
|
|
| inputs buf
|
|
|
|
|
| outputs y, buf
|
2017-11-24 12:35:36 +00:00
|
|
|
|
| trashes a, z, n, ptr
|
|
|
|
|
| {
|
|
|
|
|
| ld y, 0
|
2017-12-01 12:36:58 +00:00
|
|
|
|
| copy ^buf, ptr
|
2017-11-24 12:35:36 +00:00
|
|
|
|
| copy 123, [ptr] + y
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
It does use `y`.
|
|
|
|
|
|
|
|
|
|
| buffer[2048] buf
|
|
|
|
|
| pointer ptr
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
2017-12-01 15:10:16 +00:00
|
|
|
|
| inputs buf
|
|
|
|
|
| outputs buf
|
2017-11-24 12:35:36 +00:00
|
|
|
|
| trashes a, z, n, ptr
|
|
|
|
|
| {
|
2017-12-01 12:36:58 +00:00
|
|
|
|
| copy ^buf, ptr
|
2017-11-24 12:35:36 +00:00
|
|
|
|
| copy 123, [ptr] + y
|
|
|
|
|
| }
|
|
|
|
|
? UnmeaningfulReadError
|
|
|
|
|
|
2017-12-01 13:52:56 +00:00
|
|
|
|
Write stored value through a pointer.
|
|
|
|
|
|
|
|
|
|
| buffer[2048] buf
|
|
|
|
|
| pointer ptr
|
|
|
|
|
| byte foo
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
2017-12-01 15:10:16 +00:00
|
|
|
|
| inputs foo, buf
|
|
|
|
|
| outputs y, buf
|
2017-12-01 13:52:56 +00:00
|
|
|
|
| trashes a, z, n, ptr
|
|
|
|
|
| {
|
|
|
|
|
| ld y, 0
|
|
|
|
|
| copy ^buf, ptr
|
|
|
|
|
| copy foo, [ptr] + y
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
Read through a pointer.
|
|
|
|
|
|
|
|
|
|
| buffer[2048] buf
|
|
|
|
|
| pointer ptr
|
|
|
|
|
| byte foo
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
2017-12-01 15:10:16 +00:00
|
|
|
|
| inputs buf
|
2017-12-01 13:52:56 +00:00
|
|
|
|
| outputs foo
|
|
|
|
|
| trashes a, y, z, n, ptr
|
|
|
|
|
| {
|
|
|
|
|
| ld y, 0
|
|
|
|
|
| copy ^buf, ptr
|
|
|
|
|
| copy [ptr] + y, foo
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
2018-04-18 09:57:57 +00:00
|
|
|
|
Read and write through two pointers.
|
|
|
|
|
|
|
|
|
|
| buffer[2048] buf
|
|
|
|
|
| pointer ptra
|
|
|
|
|
| pointer ptrb
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs buf
|
|
|
|
|
| outputs buf
|
|
|
|
|
| trashes a, y, z, n, ptra, ptrb
|
|
|
|
|
| {
|
|
|
|
|
| ld y, 0
|
|
|
|
|
| copy ^buf, ptra
|
|
|
|
|
| copy ^buf, ptrb
|
|
|
|
|
| copy [ptra] + y, [ptrb] + y
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
2018-02-08 11:58:24 +00:00
|
|
|
|
Read through a pointer to the `a` register. Note that this is done with `ld`,
|
|
|
|
|
not `copy`.
|
2018-02-08 11:04:11 +00:00
|
|
|
|
|
|
|
|
|
| buffer[2048] buf
|
|
|
|
|
| pointer ptr
|
|
|
|
|
| byte foo
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs buf
|
2018-02-08 11:58:24 +00:00
|
|
|
|
| outputs a
|
|
|
|
|
| trashes y, z, n, ptr
|
2018-02-08 11:04:11 +00:00
|
|
|
|
| {
|
|
|
|
|
| ld y, 0
|
|
|
|
|
| copy ^buf, ptr
|
|
|
|
|
| ld a, [ptr] + y
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
2018-02-09 11:23:18 +00:00
|
|
|
|
Write the `a` register through a pointer. Note that this is done with `st`,
|
|
|
|
|
not `copy`.
|
|
|
|
|
|
|
|
|
|
| buffer[2048] buf
|
|
|
|
|
| pointer ptr
|
|
|
|
|
| byte foo
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs buf
|
|
|
|
|
| outputs buf
|
|
|
|
|
| trashes a, y, z, n, ptr
|
|
|
|
|
| {
|
|
|
|
|
| ld y, 0
|
|
|
|
|
| copy ^buf, ptr
|
|
|
|
|
| ld a, 255
|
|
|
|
|
| st a, [ptr] + y
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
2016-06-16 16:08:57 +00:00
|
|
|
|
### routines ###
|
|
|
|
|
|
2015-10-22 19:01:02 +00:00
|
|
|
|
Routines are constants. You need not, and in fact cannot, specify a constant
|
|
|
|
|
as an input to, an output of, or as a trashed value of a routine.
|
2015-10-19 12:04:08 +00:00
|
|
|
|
|
2018-02-05 17:01:25 +00:00
|
|
|
|
| vector routine
|
2015-10-19 12:04:08 +00:00
|
|
|
|
| inputs x
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes z, n
|
2018-02-05 17:01:25 +00:00
|
|
|
|
| vec
|
2015-10-19 12:04:08 +00:00
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| inputs x
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| inc x
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs foo
|
|
|
|
|
| outputs vec
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| copy foo, vec
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? ConstantConstraintError: foo
|
2015-10-22 19:01:02 +00:00
|
|
|
|
|
2018-02-05 17:01:25 +00:00
|
|
|
|
| vector routine
|
2015-10-22 19:01:02 +00:00
|
|
|
|
| inputs x
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes z, n
|
2018-02-05 17:01:25 +00:00
|
|
|
|
| vec
|
2015-10-22 19:01:02 +00:00
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| inputs x
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| inc x
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs vec, foo
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| copy foo, vec
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? ConstantConstraintError: foo
|
2015-10-22 19:01:02 +00:00
|
|
|
|
|
2018-02-05 17:01:25 +00:00
|
|
|
|
| vector routine
|
2015-10-22 19:01:02 +00:00
|
|
|
|
| inputs x
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes z, n
|
2018-02-05 17:01:25 +00:00
|
|
|
|
| vec
|
2015-10-22 19:01:02 +00:00
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| inputs x
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| inc x
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs vec
|
|
|
|
|
| trashes a, z, n, foo
|
|
|
|
|
| {
|
|
|
|
|
| copy foo, vec
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? ConstantConstraintError: foo
|
2015-10-22 19:01:02 +00:00
|
|
|
|
|
2018-04-18 09:13:15 +00:00
|
|
|
|
#### routine-vector type compatibility
|
|
|
|
|
|
|
|
|
|
You can copy the address of a routine into a vector, if that vector type
|
|
|
|
|
is at least as "wide" as the type of the routine. More specifically,
|
|
|
|
|
|
|
|
|
|
- the vector must take _exactly_ the same inputs as the routine
|
|
|
|
|
- the vector must make _exactly_ the same outputs as the routine
|
|
|
|
|
- the vector must trash _at least_ what the routine trashes
|
|
|
|
|
|
|
|
|
|
If the vector and the routine have the very same signature, that's not an error.
|
2015-10-22 19:01:02 +00:00
|
|
|
|
|
2018-02-05 17:01:25 +00:00
|
|
|
|
| vector routine
|
2018-04-18 09:13:15 +00:00
|
|
|
|
| inputs x, y
|
|
|
|
|
| outputs x, y
|
2015-10-22 19:01:02 +00:00
|
|
|
|
| trashes z, n
|
2018-02-05 17:01:25 +00:00
|
|
|
|
| vec
|
2015-10-22 19:01:02 +00:00
|
|
|
|
|
|
|
|
|
|
| routine foo
|
2018-04-18 09:13:15 +00:00
|
|
|
|
| inputs x, y
|
|
|
|
|
| outputs x, y
|
2015-10-22 19:01:02 +00:00
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| inc x
|
2018-04-18 09:13:15 +00:00
|
|
|
|
| inc y
|
2015-10-22 19:01:02 +00:00
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs vec
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| copy foo, vec
|
|
|
|
|
| }
|
2015-10-19 12:04:08 +00:00
|
|
|
|
= ok
|
|
|
|
|
|
2018-04-18 09:13:15 +00:00
|
|
|
|
If the vector takes an input that the routine doesn't take, that's not an error.
|
|
|
|
|
(The interface requires that a parameter be specified before calling, but the
|
|
|
|
|
implementation doesn't actually read it.)
|
2015-10-19 12:04:08 +00:00
|
|
|
|
|
2018-02-05 17:01:25 +00:00
|
|
|
|
| vector routine
|
2018-04-18 09:13:15 +00:00
|
|
|
|
| inputs x, y, a
|
|
|
|
|
| outputs x, y
|
2015-10-19 12:04:08 +00:00
|
|
|
|
| trashes z, n
|
2018-02-05 17:01:25 +00:00
|
|
|
|
| vec
|
2015-10-19 12:04:08 +00:00
|
|
|
|
|
|
|
|
|
|
| routine foo
|
2018-04-18 09:13:15 +00:00
|
|
|
|
| inputs x, y
|
|
|
|
|
| outputs x, y
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| inc x
|
|
|
|
|
| inc y
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs vec
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| copy foo, vec
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
If the vector fails to take an input that the routine takes, that's an error.
|
|
|
|
|
|
|
|
|
|
| vector routine
|
2015-10-19 12:04:08 +00:00
|
|
|
|
| inputs x
|
2018-04-18 09:13:15 +00:00
|
|
|
|
| outputs x, y
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| vec
|
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| inputs x, y
|
|
|
|
|
| outputs x, y
|
2015-10-19 12:04:08 +00:00
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| inc x
|
2018-04-18 09:13:15 +00:00
|
|
|
|
| inc y
|
2015-10-19 12:04:08 +00:00
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs vec
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| copy foo, vec
|
|
|
|
|
| }
|
2015-10-19 18:17:27 +00:00
|
|
|
|
? IncompatibleConstraintsError
|
|
|
|
|
|
2018-04-18 09:25:23 +00:00
|
|
|
|
If the vector produces an output that the routine doesn't produce, that's not an error.
|
|
|
|
|
(The interface claims the result of calling the routine is defined, but the implementation
|
|
|
|
|
actually preserves it instead of changing it; the caller can still treat it as a defined
|
|
|
|
|
output.)
|
2017-12-12 10:10:58 +00:00
|
|
|
|
|
2018-02-05 17:01:25 +00:00
|
|
|
|
| vector routine
|
2018-04-18 09:13:15 +00:00
|
|
|
|
| inputs x, y
|
|
|
|
|
| outputs x, y, a
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| vec
|
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| inputs x, y
|
|
|
|
|
| outputs x, y
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| inc x
|
|
|
|
|
| inc y
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs vec
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| copy foo, vec
|
|
|
|
|
| }
|
2018-04-18 09:25:23 +00:00
|
|
|
|
= ok
|
2018-04-18 09:13:15 +00:00
|
|
|
|
|
|
|
|
|
If the vector fails to produce an output that the routine produces, that's an error.
|
|
|
|
|
|
|
|
|
|
| vector routine
|
|
|
|
|
| inputs x, y
|
2017-12-12 10:10:58 +00:00
|
|
|
|
| outputs x
|
2018-04-18 09:13:15 +00:00
|
|
|
|
| trashes z, n
|
|
|
|
|
| vec
|
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| inputs x, y
|
|
|
|
|
| outputs x, y
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| inc x
|
|
|
|
|
| inc y
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs vec
|
2017-12-12 10:10:58 +00:00
|
|
|
|
| trashes a, z, n
|
2018-04-18 09:13:15 +00:00
|
|
|
|
| {
|
|
|
|
|
| copy foo, vec
|
|
|
|
|
| }
|
|
|
|
|
? IncompatibleConstraintsError
|
|
|
|
|
|
|
|
|
|
If the vector fails to trash something the routine trashes, that's an error.
|
|
|
|
|
|
|
|
|
|
| vector routine
|
|
|
|
|
| inputs x, y
|
|
|
|
|
| outputs x, y
|
|
|
|
|
| trashes z
|
2018-02-05 17:01:25 +00:00
|
|
|
|
| vec
|
2017-12-12 10:10:58 +00:00
|
|
|
|
|
|
|
|
|
|
| routine foo
|
2018-04-18 09:13:15 +00:00
|
|
|
|
| inputs x, y
|
|
|
|
|
| outputs x, y
|
2017-12-12 10:10:58 +00:00
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| inc x
|
2018-04-18 09:13:15 +00:00
|
|
|
|
| inc y
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs vec
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| copy foo, vec
|
|
|
|
|
| }
|
|
|
|
|
? IncompatibleConstraintsError
|
|
|
|
|
|
|
|
|
|
If the vector trashes something the routine doesn't trash, that's not an error.
|
|
|
|
|
(The implementation preserves something the interface doesn't guarantee is
|
|
|
|
|
preserved. The caller gets no guarantee that it's preserved. It actually is,
|
|
|
|
|
but it doesn't know that.)
|
|
|
|
|
|
|
|
|
|
| vector routine
|
|
|
|
|
| inputs x, y
|
|
|
|
|
| outputs x, y
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| vec
|
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| inputs x, y
|
|
|
|
|
| outputs x, y
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| inc x
|
|
|
|
|
| inc y
|
2017-12-12 10:10:58 +00:00
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs vec
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| copy foo, vec
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
2018-04-18 09:13:15 +00:00
|
|
|
|
#### other properties of routines
|
|
|
|
|
|
2015-10-19 18:17:27 +00:00
|
|
|
|
Routines are read-only.
|
|
|
|
|
|
2018-02-05 17:01:25 +00:00
|
|
|
|
| vector routine
|
2015-10-19 18:17:27 +00:00
|
|
|
|
| inputs x
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes z, n
|
2018-02-05 17:01:25 +00:00
|
|
|
|
| vec
|
2015-10-19 18:17:27 +00:00
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| inputs x
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| inc x
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs vec
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| copy vec, foo
|
|
|
|
|
| }
|
|
|
|
|
? TypeMismatchError
|
2015-10-20 12:15:21 +00:00
|
|
|
|
|
|
|
|
|
Indirect call.
|
|
|
|
|
|
2018-02-05 17:01:25 +00:00
|
|
|
|
| vector routine
|
|
|
|
|
| outputs x trashes z, n
|
|
|
|
|
| foo
|
2015-10-20 12:15:21 +00:00
|
|
|
|
|
|
|
|
|
|
| routine bar outputs x trashes z, n {
|
|
|
|
|
| ld x, 200
|
|
|
|
|
| }
|
|
|
|
|
|
|
2015-10-22 19:01:02 +00:00
|
|
|
|
| routine main outputs x, foo trashes a, z, n {
|
2015-10-20 12:15:21 +00:00
|
|
|
|
| copy bar, foo
|
|
|
|
|
| call foo
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
2015-10-21 16:05:49 +00:00
|
|
|
|
Calling the vector does indeed trash the things the vector says it does.
|
2015-10-20 12:15:21 +00:00
|
|
|
|
|
2018-02-05 17:01:25 +00:00
|
|
|
|
| vector routine trashes x, z, n foo
|
2015-10-20 12:15:21 +00:00
|
|
|
|
|
|
|
|
|
|
| routine bar trashes x, z, n {
|
|
|
|
|
| ld x, 200
|
|
|
|
|
| }
|
|
|
|
|
|
|
2015-10-22 19:01:02 +00:00
|
|
|
|
| routine main outputs x, foo trashes z, n {
|
2015-10-20 12:15:21 +00:00
|
|
|
|
| ld x, 0
|
|
|
|
|
| copy bar, foo
|
|
|
|
|
| call foo
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulOutputError: x
|
2015-10-21 10:41:52 +00:00
|
|
|
|
|
2015-10-21 14:45:14 +00:00
|
|
|
|
`goto`, if present, must be in tail position (the final instruction in a routine.)
|
2015-10-21 10:41:52 +00:00
|
|
|
|
|
|
|
|
|
| routine bar trashes x, z, n {
|
|
|
|
|
| ld x, 200
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main trashes x, z, n {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| goto bar
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| routine bar trashes x, z, n {
|
|
|
|
|
| ld x, 200
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main trashes x, z, n {
|
|
|
|
|
| goto bar
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| }
|
|
|
|
|
? IllegalJumpError
|
|
|
|
|
|
|
|
|
|
| routine bar trashes x, z, n {
|
|
|
|
|
| ld x, 200
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main trashes x, z, n {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| if z {
|
|
|
|
|
| ld x, 1
|
|
|
|
|
| goto bar
|
|
|
|
|
| }
|
|
|
|
|
| }
|
2015-10-21 14:45:14 +00:00
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
| routine bar trashes x, z, n {
|
|
|
|
|
| ld x, 200
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main trashes x, z, n {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| if z {
|
|
|
|
|
| ld x, 1
|
|
|
|
|
| goto bar
|
|
|
|
|
| }
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| }
|
2015-10-21 10:41:52 +00:00
|
|
|
|
? IllegalJumpError
|
|
|
|
|
|
2018-03-13 13:33:30 +00:00
|
|
|
|
| routine bar trashes x, z, n {
|
|
|
|
|
| ld x, 200
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main trashes x, z, n {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| if z {
|
|
|
|
|
| ld x, 1
|
|
|
|
|
| goto bar
|
|
|
|
|
| } else {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| goto bar
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
2018-03-13 17:00:49 +00:00
|
|
|
|
| routine bar trashes x, z, n {
|
|
|
|
|
| ld x, 200
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main trashes x, z, n {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| if z {
|
|
|
|
|
| ld x, 1
|
|
|
|
|
| goto bar
|
|
|
|
|
| } else {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| }
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
For the purposes of `goto`, the end of a loop is never tail position.
|
|
|
|
|
|
|
|
|
|
| routine bar trashes x, z, n {
|
|
|
|
|
| ld x, 200
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main trashes x, z, n {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| repeat {
|
|
|
|
|
| inc x
|
|
|
|
|
| goto bar
|
|
|
|
|
| } until z
|
|
|
|
|
| }
|
|
|
|
|
? IllegalJumpError
|
|
|
|
|
|
2015-10-21 10:41:52 +00:00
|
|
|
|
Can't `goto` a routine that outputs or trashes more than the current routine.
|
|
|
|
|
|
|
|
|
|
| routine bar trashes x, y, z, n {
|
|
|
|
|
| ld x, 200
|
|
|
|
|
| ld y, 200
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main trashes x, z, n {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| goto bar
|
|
|
|
|
| }
|
2015-10-21 14:45:14 +00:00
|
|
|
|
? IncompatibleConstraintsError
|
2015-10-21 10:41:52 +00:00
|
|
|
|
|
|
|
|
|
| routine bar outputs y trashes z, n {
|
|
|
|
|
| ld y, 200
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main trashes x, z, n {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| goto bar
|
|
|
|
|
| }
|
2015-10-21 14:45:14 +00:00
|
|
|
|
? IncompatibleConstraintsError
|
2015-10-21 16:05:49 +00:00
|
|
|
|
|
|
|
|
|
Can `goto` a routine that outputs or trashes less than the current routine.
|
|
|
|
|
|
|
|
|
|
| routine bar trashes x, z, n {
|
|
|
|
|
| ld x, 1
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main trashes a, x, z, n {
|
|
|
|
|
| ld a, 0
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| goto bar
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
Indirect goto.
|
|
|
|
|
|
2018-02-05 17:01:25 +00:00
|
|
|
|
| vector routine outputs x trashes a, z, n foo
|
2015-10-21 16:05:49 +00:00
|
|
|
|
|
|
|
|
|
|
| routine bar outputs x trashes a, z, n {
|
|
|
|
|
| ld x, 200
|
|
|
|
|
| }
|
|
|
|
|
|
|
2015-10-22 19:01:02 +00:00
|
|
|
|
| routine main outputs x trashes foo, a, z, n {
|
2015-10-21 16:05:49 +00:00
|
|
|
|
| copy bar, foo
|
|
|
|
|
| goto foo
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
2015-10-22 08:54:11 +00:00
|
|
|
|
Jumping through the vector does indeed trash, or output, the things the
|
|
|
|
|
vector says it does.
|
2015-10-21 16:05:49 +00:00
|
|
|
|
|
2018-02-05 17:01:25 +00:00
|
|
|
|
| vector routine
|
2015-10-22 08:54:11 +00:00
|
|
|
|
| trashes a, x, z, n
|
2018-02-05 17:01:25 +00:00
|
|
|
|
| foo
|
2015-10-22 08:48:26 +00:00
|
|
|
|
|
|
2015-10-22 08:54:11 +00:00
|
|
|
|
| routine bar
|
|
|
|
|
| trashes a, x, z, n {
|
2015-10-22 08:48:26 +00:00
|
|
|
|
| ld x, 200
|
|
|
|
|
| }
|
|
|
|
|
|
|
2015-10-22 08:54:11 +00:00
|
|
|
|
| routine sub
|
|
|
|
|
| trashes foo, a, x, z, n {
|
2015-10-22 08:48:26 +00:00
|
|
|
|
| ld x, 0
|
|
|
|
|
| copy bar, foo
|
|
|
|
|
| goto foo
|
|
|
|
|
| }
|
|
|
|
|
|
|
2015-10-22 19:01:02 +00:00
|
|
|
|
| routine main
|
2015-10-22 08:54:11 +00:00
|
|
|
|
| outputs a
|
|
|
|
|
| trashes foo, x, z, n {
|
2015-10-22 08:48:26 +00:00
|
|
|
|
| call sub
|
|
|
|
|
| ld a, x
|
|
|
|
|
| }
|
2018-03-06 14:39:47 +00:00
|
|
|
|
? UnmeaningfulReadError: x
|
2015-10-21 16:05:49 +00:00
|
|
|
|
|
2018-02-05 17:01:25 +00:00
|
|
|
|
| vector routine
|
2015-10-22 08:54:11 +00:00
|
|
|
|
| outputs x
|
2018-02-05 17:01:25 +00:00
|
|
|
|
| trashes a, z, n foo
|
2015-10-22 08:54:11 +00:00
|
|
|
|
|
|
|
|
|
|
| routine bar
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes a, z, n {
|
|
|
|
|
| ld x, 200
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine sub
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes foo, a, z, n {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| copy bar, foo
|
|
|
|
|
| goto foo
|
|
|
|
|
| }
|
|
|
|
|
|
|
2015-10-22 19:01:02 +00:00
|
|
|
|
| routine main
|
2015-10-22 08:54:11 +00:00
|
|
|
|
| outputs a
|
|
|
|
|
| trashes foo, x, z, n {
|
|
|
|
|
| call sub
|
|
|
|
|
| ld a, x
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
2018-02-02 17:45:07 +00:00
|
|
|
|
|
2018-02-06 10:05:56 +00:00
|
|
|
|
### vector tables ###
|
2018-02-02 17:45:07 +00:00
|
|
|
|
|
2018-02-05 12:04:28 +00:00
|
|
|
|
A vector can be copied into a vector table.
|
2018-02-02 17:45:07 +00:00
|
|
|
|
|
2018-02-05 17:01:25 +00:00
|
|
|
|
| vector routine
|
2018-02-02 17:45:07 +00:00
|
|
|
|
| outputs x
|
|
|
|
|
| trashes a, z, n
|
2018-02-05 17:01:25 +00:00
|
|
|
|
| one
|
2018-02-06 11:34:21 +00:00
|
|
|
|
| vector (routine
|
2018-02-02 17:45:07 +00:00
|
|
|
|
| outputs x
|
2018-02-06 11:34:21 +00:00
|
|
|
|
| trashes a, z, n)
|
|
|
|
|
| table[256] many
|
2018-02-02 17:45:07 +00:00
|
|
|
|
|
|
|
|
|
|
| routine bar outputs x trashes a, z, n {
|
|
|
|
|
| ld x, 200
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs one, many
|
|
|
|
|
| outputs one, many
|
|
|
|
|
| trashes a, x, n, z
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| copy bar, one
|
|
|
|
|
| copy one, many + x
|
2018-02-05 12:04:28 +00:00
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
A vector can be copied out of a vector table.
|
|
|
|
|
|
2018-02-05 17:01:25 +00:00
|
|
|
|
| vector routine
|
2018-02-05 12:04:28 +00:00
|
|
|
|
| outputs x
|
|
|
|
|
| trashes a, z, n
|
2018-02-05 17:01:25 +00:00
|
|
|
|
| one
|
2018-02-06 11:34:21 +00:00
|
|
|
|
| vector (routine
|
2018-02-05 12:04:28 +00:00
|
|
|
|
| outputs x
|
2018-02-06 11:34:21 +00:00
|
|
|
|
| trashes a, z, n)
|
|
|
|
|
| table[256] many
|
2018-02-05 12:04:28 +00:00
|
|
|
|
|
|
|
|
|
|
| routine bar outputs x trashes a, z, n {
|
|
|
|
|
| ld x, 200
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs one, many
|
|
|
|
|
| outputs one, many
|
|
|
|
|
| trashes a, x, n, z
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
2018-02-02 17:56:50 +00:00
|
|
|
|
| copy many + x, one
|
|
|
|
|
| call one
|
2018-02-02 17:45:07 +00:00
|
|
|
|
| }
|
|
|
|
|
= ok
|
2018-02-05 12:04:28 +00:00
|
|
|
|
|
|
|
|
|
A routine can be copied into a vector table.
|
|
|
|
|
|
2018-02-06 11:34:21 +00:00
|
|
|
|
| vector (routine
|
2018-02-05 17:01:25 +00:00
|
|
|
|
| outputs x
|
2018-02-06 11:34:21 +00:00
|
|
|
|
| trashes a, z, n)
|
|
|
|
|
| table[256] many
|
2018-02-05 12:04:28 +00:00
|
|
|
|
|
|
|
|
|
|
| routine bar outputs x trashes a, z, n {
|
|
|
|
|
| ld x, 200
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs many
|
|
|
|
|
| outputs many
|
|
|
|
|
| trashes a, x, n, z
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| copy bar, many + x
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
A vector in a vector table cannot be directly called.
|
|
|
|
|
|
2018-02-06 11:34:21 +00:00
|
|
|
|
| vector (routine
|
2018-02-05 17:01:25 +00:00
|
|
|
|
| outputs x
|
2018-02-06 11:34:21 +00:00
|
|
|
|
| trashes a, z, n)
|
|
|
|
|
| table[256] many
|
2018-02-05 12:04:28 +00:00
|
|
|
|
|
|
|
|
|
|
| routine bar outputs x trashes a, z, n {
|
|
|
|
|
| ld x, 200
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| inputs many
|
|
|
|
|
| outputs many
|
|
|
|
|
| trashes a, x, n, z
|
|
|
|
|
| {
|
|
|
|
|
| ld x, 0
|
|
|
|
|
| copy bar, many + x
|
|
|
|
|
| call many + x
|
|
|
|
|
| }
|
2018-03-06 12:23:57 +00:00
|
|
|
|
? SyntaxError
|
2018-02-06 10:05:56 +00:00
|
|
|
|
|
|
|
|
|
### typedef ###
|
|
|
|
|
|
|
|
|
|
A typedef is a more-readable alias for a type. "Alias" means
|
|
|
|
|
that types have structural equivalence, not name equivalence.
|
|
|
|
|
|
|
|
|
|
| typedef routine
|
|
|
|
|
| inputs x
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| routine_type
|
|
|
|
|
|
|
|
|
|
|
| vector routine_type vec
|
|
|
|
|
|
|
|
|
|
|
| routine foo
|
|
|
|
|
| inputs x
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| {
|
|
|
|
|
| inc x
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs vec
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| copy foo, vec
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
|
|
|
|
|
|
|
|
|
The new style routine definitions support typedefs.
|
|
|
|
|
|
|
|
|
|
| typedef routine
|
|
|
|
|
| inputs x
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| routine_type
|
|
|
|
|
|
|
|
|
|
|
| vector routine_type vec
|
|
|
|
|
|
|
|
|
|
|
| define foo routine_type
|
|
|
|
|
| {
|
|
|
|
|
| inc x
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| routine main
|
|
|
|
|
| outputs vec
|
|
|
|
|
| trashes a, z, n
|
|
|
|
|
| {
|
|
|
|
|
| copy foo, vec
|
|
|
|
|
| }
|
|
|
|
|
= ok
|
2018-02-09 13:35:59 +00:00
|
|
|
|
|
|
|
|
|
### static ###
|
|
|
|
|
|
|
|
|
|
When memory locations are defined static to a routine, they cannot be
|
|
|
|
|
directly input, nor directly output; and since they are always initialized,
|
|
|
|
|
they cannot be trashed. Thus, they really don't participate in the analysis.
|
|
|
|
|
|
|
|
|
|
| define foo routine
|
|
|
|
|
| inputs x
|
|
|
|
|
| outputs x
|
|
|
|
|
| trashes z, n
|
|
|
|
|
| static byte t : 0
|
|
|
|
|
| {
|
|
|
|
|
| st x, t
|
|
|
|
|
| inc t
|
|
|
|
|
| ld x, t
|
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
|
|
| define main routine
|
|
|
|
|
| trashes a, x, z, n
|
|
|
|
|
| static byte t : 0
|
|
|
|
|
| {
|
|
|
|
|
| ld x, t
|
|
|
|
|
| call foo
|
|
|
|
|
| }
|
|
|
|
|
= ok
|