2015-10-16 15:36:56 +01:00
|
|
|
SixtyPical
|
2015-10-16 09:30:24 +01:00
|
|
|
==========
|
|
|
|
|
2017-11-17 16:12:59 +00:00
|
|
|
This document describes the SixtyPical programming language version 0.8-PRE,
|
2015-10-16 15:36:56 +01:00
|
|
|
both its execution aspect and its static analysis aspect (even though
|
|
|
|
these are, technically speaking, separate concepts.)
|
2015-10-16 09:30:24 +01:00
|
|
|
|
2015-10-16 15:36:56 +01:00
|
|
|
This document is nominally normative, but the tests in the `tests` directory
|
|
|
|
are even more normative.
|
2015-10-16 09:30:24 +01:00
|
|
|
|
2015-10-16 19:15:01 +01:00
|
|
|
Refer to the bottom of this document for an EBNF grammar of the syntax of
|
|
|
|
the language.
|
|
|
|
|
2015-10-16 09:30:24 +01:00
|
|
|
Types
|
|
|
|
-----
|
|
|
|
|
2017-11-20 16:14:17 +00:00
|
|
|
There are five *primitive types* in SixtyPical:
|
2015-10-16 09:30:24 +01:00
|
|
|
|
|
|
|
* bit (2 possible values)
|
|
|
|
* byte (256 possible values)
|
2017-11-17 16:56:52 +00:00
|
|
|
* word (65536 possible values)
|
2015-10-18 19:41:26 +01:00
|
|
|
* routine (code stored somewhere in memory, read-only)
|
|
|
|
* vector (address of a routine)
|
2015-10-16 09:30:24 +01:00
|
|
|
|
2017-11-20 16:14:17 +00:00
|
|
|
There is also one *type constructor*:
|
|
|
|
|
|
|
|
* X table (256 entries, each holding a value of type X)
|
|
|
|
|
2017-11-24 11:30:20 +00:00
|
|
|
This constructor can only be applied to one type, `byte`.
|
2017-11-20 16:14:17 +00:00
|
|
|
|
2015-10-16 09:30:24 +01:00
|
|
|
Memory locations
|
|
|
|
----------------
|
|
|
|
|
2017-11-17 16:56:52 +00:00
|
|
|
A primary concept in SixtyPical is the *memory location*. At any given point
|
|
|
|
in time during execution, each memory location is either *uninitialized* or
|
|
|
|
*initialized*. At any given point in the program text, too, each memory
|
2015-10-16 09:30:24 +01:00
|
|
|
location is either uninitialized or initialized. Where-ever it is one or
|
|
|
|
the other during execution, it is the same in the corresponding place in
|
|
|
|
the program text; thus, it is a static property.
|
|
|
|
|
|
|
|
There are four general kinds of memory location. The first three are
|
|
|
|
pre-defined and built-in.
|
|
|
|
|
|
|
|
### Registers ###
|
|
|
|
|
|
|
|
Each of these hold a byte. They are initially uninitialized.
|
|
|
|
|
|
|
|
a
|
|
|
|
x
|
|
|
|
y
|
|
|
|
|
|
|
|
### Flags ###
|
|
|
|
|
|
|
|
Each of these hold a bit. They are initially uninitialized.
|
|
|
|
|
|
|
|
c (carry)
|
|
|
|
z (zero)
|
|
|
|
v (overflow)
|
|
|
|
n (negative)
|
|
|
|
|
|
|
|
### Constants ###
|
|
|
|
|
|
|
|
It may be strange to think of constants as memory locations, but keep in mind
|
2015-10-16 15:36:56 +01:00
|
|
|
that a memory location in SixtyPical need not map to a memory location in the
|
|
|
|
underlying hardware. All constants are read-only. Each is initially
|
|
|
|
initialized with the value that corresponds with its name.
|
2015-10-16 09:30:24 +01:00
|
|
|
|
|
|
|
They come in bit and byte types. There are two bit constants,
|
|
|
|
|
|
|
|
off
|
|
|
|
on
|
|
|
|
|
2017-11-17 16:56:52 +00:00
|
|
|
two hundred and fifty-six byte constants,
|
2015-10-16 09:30:24 +01:00
|
|
|
|
|
|
|
0
|
|
|
|
1
|
|
|
|
...
|
|
|
|
255
|
|
|
|
|
2017-11-17 16:56:52 +00:00
|
|
|
and sixty-five thousand five hundred and thirty-six word constants,
|
|
|
|
|
2017-11-20 15:18:21 +00:00
|
|
|
word 0
|
|
|
|
word 1
|
2017-11-17 16:56:52 +00:00
|
|
|
...
|
2017-11-20 15:18:21 +00:00
|
|
|
word 65535
|
2017-11-17 16:56:52 +00:00
|
|
|
|
2017-11-20 15:18:21 +00:00
|
|
|
Note that if a word constant is between 256 and 65535, the leading `word`
|
|
|
|
token can be omitted.
|
2017-11-17 16:56:52 +00:00
|
|
|
|
2015-10-16 09:30:24 +01:00
|
|
|
### User-defined ###
|
|
|
|
|
|
|
|
There may be any number of user-defined memory locations. They are defined
|
2017-11-17 16:56:52 +00:00
|
|
|
by giving the type (which may be any type except `bit` and `routine`) and the
|
2015-10-18 19:41:26 +01:00
|
|
|
name.
|
2015-10-16 09:30:24 +01:00
|
|
|
|
|
|
|
byte pos
|
|
|
|
|
2016-06-16 11:02:13 -05:00
|
|
|
An address in memory may be given explicitly on a user-defined memory location.
|
2015-10-18 16:22:36 +01:00
|
|
|
|
2015-10-18 19:41:26 +01:00
|
|
|
byte table screen @ 1024
|
2015-10-18 16:22:36 +01:00
|
|
|
|
2016-06-16 11:02:13 -05:00
|
|
|
Or, a user-defined memory location may be given an initial value. But in this
|
|
|
|
case, an explicit address in memory cannot be given.
|
|
|
|
|
2016-06-16 11:03:31 -05:00
|
|
|
byte pos : 0
|
2016-06-16 11:02:13 -05:00
|
|
|
|
2017-11-17 16:56:52 +00:00
|
|
|
A user-defined vector memory location is decorated with `inputs`, `outputs`
|
|
|
|
and `trashes` lists like a routine (see below), and it may only hold addresses
|
|
|
|
of routines which are compatible. (Meaning, the routine's inputs (resp. outputs,
|
|
|
|
trashes) must be a subset of the vector's inputs (resp. outputs, trashes.))
|
2015-10-19 13:04:08 +01:00
|
|
|
|
|
|
|
vector actor_logic
|
|
|
|
inputs a, score
|
|
|
|
outputs x
|
|
|
|
trashes y
|
|
|
|
@ $c000
|
|
|
|
|
2017-11-20 16:14:17 +00:00
|
|
|
Note that in the code of a routine, if a memory location is named by a
|
|
|
|
user-defined symbol, it is an address in memory, and can be read and written.
|
|
|
|
But if it is named by a literal integer, either decimal or hexadecimal, it
|
|
|
|
is a constant and can only be read (and when read always yields that constant
|
|
|
|
value. So, for instance, to read the value at `screen` above, in the code,
|
|
|
|
you would need to reference the symbol `screen`; attempting to read 1024
|
|
|
|
would not work.
|
|
|
|
|
|
|
|
This is actually useful, at least at this point, as you can rely on the fact
|
|
|
|
that literal integers in the code are always immediate values. (But this
|
|
|
|
may change at some point.)
|
2017-11-20 14:10:43 +00:00
|
|
|
|
2015-10-16 09:30:24 +01:00
|
|
|
Routines
|
|
|
|
--------
|
|
|
|
|
2017-11-17 16:56:52 +00:00
|
|
|
Every routine must list all the memory locations it *reads from*, which we
|
|
|
|
call its `inputs`, and all the memory locations it *writes to*. The latter
|
|
|
|
we divide into two groups: its `outputs` which it intentionally initializes,
|
|
|
|
and its `trashes`, which it does not care about, and leaves uninitialized.
|
|
|
|
For example, if it uses a register to temporarily store an intermediate
|
|
|
|
value used in a multiplication, that register has no meaning outside of
|
|
|
|
the multiplication, and is one of the routine's `trashes`.
|
|
|
|
|
|
|
|
It is common to say that the `trashes` are the memory locations that are
|
|
|
|
*not preserved* by the routine.
|
2015-10-16 09:30:24 +01:00
|
|
|
|
|
|
|
routine foo
|
|
|
|
inputs a, score
|
|
|
|
outputs x
|
|
|
|
trashes y {
|
|
|
|
...
|
|
|
|
}
|
|
|
|
|
2017-11-20 13:25:09 +00:00
|
|
|
The union of the `outputs` and `trashes` is sometimes collectively called
|
|
|
|
"the WRITES" of the routine, for historical reasons and as shorthand.
|
2017-11-17 16:56:52 +00:00
|
|
|
|
2015-10-16 09:30:24 +01:00
|
|
|
Routines may call only routines previously defined in the program source.
|
2015-10-20 09:33:30 +01:00
|
|
|
Thus, directly recursive routines are not allowed. (However, routines may
|
|
|
|
also call routines via vectors, which are dynamically assigned. In this
|
|
|
|
case, there is, for the time being, no check for recursive calls.)
|
2015-10-16 09:30:24 +01:00
|
|
|
|
2015-10-16 15:36:56 +01:00
|
|
|
For a SixtyPical program to be run, there must be one routine called `main`.
|
|
|
|
This routine is executed when the program is run.
|
|
|
|
|
2017-11-17 16:56:52 +00:00
|
|
|
The memory locations given as inputs to a routine are considered to be initialized
|
2015-10-16 15:36:56 +01:00
|
|
|
at the beginning of the routine. Various instructions cause memory locations
|
|
|
|
to be initialized after they are executed. Calling a routine which trashes
|
|
|
|
some memory locations causes those memory locations to be uninitialized after
|
|
|
|
that routine is called. At the end of a routine, all memory locations listed
|
2017-11-17 16:56:52 +00:00
|
|
|
as outputs must be initialized.
|
2015-10-16 09:30:24 +01:00
|
|
|
|
2017-11-20 13:25:09 +00:00
|
|
|
A literal word can given instead of the body of the routine. This word is the
|
|
|
|
absolute address of an "external" routine located in memory but not defined by
|
|
|
|
the SixtyPical program.
|
2015-10-17 13:50:21 +01:00
|
|
|
|
|
|
|
routine chrout
|
|
|
|
inputs a
|
|
|
|
trashes a
|
|
|
|
@ 65490
|
|
|
|
|
2015-10-16 09:30:24 +01:00
|
|
|
Instructions
|
|
|
|
------------
|
|
|
|
|
2017-11-17 16:56:52 +00:00
|
|
|
Instructions are inspired by, and in many cases closely resemble, the 6502
|
|
|
|
instruction set. However, in many cases they do not map 1:1 to 6502 instructions.
|
|
|
|
If a SixtyPical instruction cannot be translated validly to one more more 6502
|
|
|
|
instructions while retaining all the stated constraints, that's a static error
|
|
|
|
in a SixtyPical program, and technically any implementation of SixtyPical, even
|
|
|
|
an interpreter, should flag it up.
|
|
|
|
|
2015-10-16 09:30:24 +01:00
|
|
|
### ld ###
|
|
|
|
|
2015-10-18 19:02:07 +01:00
|
|
|
ld <dest-memory-location>, <src-memory-location> [+ <index-memory-location>]
|
2015-10-16 09:30:24 +01:00
|
|
|
|
|
|
|
Reads from src and writes to dest.
|
|
|
|
|
|
|
|
* It is illegal if dest is not a register.
|
2017-11-17 16:56:52 +00:00
|
|
|
* It is illegal if dest does not occur in the WRITES of the current routine.
|
2015-10-16 09:30:24 +01:00
|
|
|
* It is illegal if src is not of same type as dest (i.e., is not a byte.)
|
|
|
|
* It is illegal if src is uninitialized.
|
|
|
|
|
|
|
|
After execution, dest is considered initialized. The flags `z` and `n` may be
|
2017-11-17 16:56:52 +00:00
|
|
|
changed by this instruction; they must be named in the WRITES, and they
|
2015-10-16 15:36:56 +01:00
|
|
|
are considered initialized after it has executed.
|
2015-10-16 09:30:24 +01:00
|
|
|
|
2015-10-18 19:02:07 +01:00
|
|
|
If and only if src is a byte table, the index-memory-location must be given.
|
|
|
|
|
2015-10-16 09:30:24 +01:00
|
|
|
Some combinations, such as `ld x, y`, are illegal because they do not map to
|
|
|
|
underlying opcodes.
|
|
|
|
|
|
|
|
### st ###
|
|
|
|
|
2015-10-18 19:02:07 +01:00
|
|
|
st <src-memory-location>, <dest-memory-location> [+ <index-memory-location>]
|
2015-10-16 09:30:24 +01:00
|
|
|
|
|
|
|
Reads from src and writes to dest.
|
|
|
|
|
|
|
|
* It is illegal if dest is a register or if dest is read-only.
|
2017-11-17 16:56:52 +00:00
|
|
|
* It is illegal if dest does not occur in the WRITES of the current routine.
|
2015-10-16 09:30:24 +01:00
|
|
|
* It is illegal if src is not of same type as dest.
|
|
|
|
* It is illegal if src is uninitialized.
|
|
|
|
|
|
|
|
After execution, dest is considered initialized. No flags are
|
|
|
|
changed by this instruction (unless of course dest is a flag.)
|
|
|
|
|
2015-10-18 19:02:07 +01:00
|
|
|
If and only if dest is a byte table, the index-memory-location must be given.
|
|
|
|
|
2017-11-20 13:25:09 +00:00
|
|
|
### copy ###
|
|
|
|
|
|
|
|
copy <src-memory-location>, <dest-memory-location>
|
|
|
|
|
|
|
|
Reads from src and writes to dest. Differs from `st` in that is able to
|
|
|
|
copy more general types of data (for example, vectors,) and it trashes the
|
|
|
|
`z` and `n` flags and the `a` register.
|
|
|
|
|
|
|
|
* It is illegal if dest is read-only.
|
|
|
|
* It is illegal if dest does not occur in the WRITES of the current routine.
|
|
|
|
* It is illegal if src is not of same type as dest.
|
|
|
|
* It is illegal if src is uninitialized.
|
|
|
|
|
|
|
|
After execution, dest is considered initialized, and `z` and `n`, and
|
|
|
|
`a` are considered uninitialized.
|
|
|
|
|
2015-10-16 09:30:24 +01:00
|
|
|
### add dest, src ###
|
|
|
|
|
|
|
|
add <dest-memory-location>, <src-memory-location>
|
|
|
|
|
|
|
|
Adds the contents of src to dest and stores the result in dest.
|
|
|
|
|
|
|
|
* It is illegal if src OR dest OR c is uninitialized.
|
|
|
|
* It is illegal if dest is read-only.
|
2017-11-17 16:56:52 +00:00
|
|
|
* It is illegal if dest does not occur in the WRITES of the current routine.
|
2015-10-16 09:30:24 +01:00
|
|
|
|
2017-11-17 16:56:52 +00:00
|
|
|
Affects n, z, c, and v flags, requiring that they be in the WRITES,
|
2015-10-16 15:36:56 +01:00
|
|
|
and initializing them afterwards.
|
2015-10-16 09:30:24 +01:00
|
|
|
|
2015-10-16 15:36:56 +01:00
|
|
|
dest and src continue to be initialized afterwards.
|
2015-10-16 09:30:24 +01:00
|
|
|
|
|
|
|
### inc ###
|
|
|
|
|
2015-10-16 15:36:56 +01:00
|
|
|
inc <dest-memory-location>
|
|
|
|
|
|
|
|
Increments the value in dest. Does not honour carry.
|
|
|
|
|
|
|
|
* It is illegal if dest is uninitialized.
|
|
|
|
* It is illegal if dest is read-only.
|
2017-11-17 16:56:52 +00:00
|
|
|
* It is illegal if dest does not occur in the WRITES of the current routine.
|
2015-10-16 15:36:56 +01:00
|
|
|
|
2017-11-17 16:56:52 +00:00
|
|
|
Affects n and z flags, requiring that they be in the WRITES,
|
2015-10-16 15:36:56 +01:00
|
|
|
and initializing them afterwards.
|
|
|
|
|
2015-10-16 09:30:24 +01:00
|
|
|
### sub ###
|
|
|
|
|
|
|
|
sub <dest-memory-location>, <src-memory-location>
|
|
|
|
|
|
|
|
Subtracts the contents of src from dest and stores the result in dest.
|
|
|
|
|
2015-10-16 15:36:56 +01:00
|
|
|
* It is illegal if src OR dest OR c is uninitialized.
|
|
|
|
* It is illegal if dest is read-only.
|
2017-11-17 16:56:52 +00:00
|
|
|
* It is illegal if dest does not occur in the WRITES of the current routine.
|
2015-10-16 15:36:56 +01:00
|
|
|
|
2017-11-17 16:56:52 +00:00
|
|
|
Affects n, z, c, and v flags, requiring that they be in the WRITES,
|
2015-10-16 15:36:56 +01:00
|
|
|
and initializing them afterwards.
|
|
|
|
|
|
|
|
dest and src continue to be initialized afterwards.
|
2015-10-16 09:30:24 +01:00
|
|
|
|
|
|
|
### dec ###
|
|
|
|
|
2015-10-16 19:15:01 +01:00
|
|
|
dec <dest-memory-location>
|
2015-10-16 15:36:56 +01:00
|
|
|
|
|
|
|
Decrements the value in dest. Does not honour carry.
|
|
|
|
|
|
|
|
* It is illegal if dest is uninitialized.
|
|
|
|
* It is illegal if dest is read-only.
|
2017-11-17 16:56:52 +00:00
|
|
|
* It is illegal if dest does not occur in the WRITES of the current routine.
|
2015-10-16 15:36:56 +01:00
|
|
|
|
2017-11-17 16:56:52 +00:00
|
|
|
Affects n and z flags, requiring that they be in the WRITES,
|
2015-10-16 15:36:56 +01:00
|
|
|
and initializing them afterwards.
|
|
|
|
|
2015-10-16 09:30:24 +01:00
|
|
|
### cmp ###
|
|
|
|
|
|
|
|
cmp <dest-memory-location>, <src-memory-location>
|
|
|
|
|
2015-10-16 19:15:01 +01:00
|
|
|
Subtracts the contents of src from dest (without considering carry) but
|
|
|
|
does not store the result anywhere, only sets the resulting flags.
|
2015-10-16 09:30:24 +01:00
|
|
|
|
2015-10-16 15:36:56 +01:00
|
|
|
* It is illegal if src OR dest is uninitialized.
|
|
|
|
|
2017-11-17 16:56:52 +00:00
|
|
|
Affects n, z, and c flags, requiring that they be in the WRITES,
|
2015-10-16 15:36:56 +01:00
|
|
|
and initializing them afterwards.
|
2015-10-16 09:30:24 +01:00
|
|
|
|
2015-10-16 18:39:38 +01:00
|
|
|
### and, or, xor ###
|
2015-10-16 09:30:24 +01:00
|
|
|
|
|
|
|
and <dest-memory-location>, <src-memory-location>
|
|
|
|
or <dest-memory-location>, <src-memory-location>
|
2015-10-16 18:39:38 +01:00
|
|
|
xor <dest-memory-location>, <src-memory-location>
|
2015-10-16 09:30:24 +01:00
|
|
|
|
2015-10-16 18:39:38 +01:00
|
|
|
Applies the given bitwise Boolean operation to src and dest and stores
|
|
|
|
the result in dest.
|
2015-10-16 09:30:24 +01:00
|
|
|
|
2015-10-16 18:39:38 +01:00
|
|
|
* It is illegal if src OR dest OR is uninitialized.
|
|
|
|
* It is illegal if dest is read-only.
|
2017-11-17 16:56:52 +00:00
|
|
|
* It is illegal if dest does not occur in the WRITES of the current routine.
|
2015-10-16 09:30:24 +01:00
|
|
|
|
2017-11-17 16:56:52 +00:00
|
|
|
Affects n and z flags, requiring that they be in the WRITES of the
|
2015-10-16 18:39:38 +01:00
|
|
|
current routine, and sets them as initialized afterwards.
|
2015-10-16 09:30:24 +01:00
|
|
|
|
2015-10-16 18:39:38 +01:00
|
|
|
dest and src continue to be initialized afterwards.
|
2015-10-16 09:30:24 +01:00
|
|
|
|
2015-10-16 18:39:38 +01:00
|
|
|
### shl, shr ###
|
2015-10-16 09:30:24 +01:00
|
|
|
|
|
|
|
shl <dest-memory-location>
|
2015-10-16 18:39:38 +01:00
|
|
|
shr <dest-memory-location>
|
2015-10-16 09:30:24 +01:00
|
|
|
|
2015-10-16 18:39:38 +01:00
|
|
|
`shl` shifts the dest left one bit position. The rightmost position becomes `c`,
|
2015-10-16 09:30:24 +01:00
|
|
|
and `c` becomes the bit that was shifted off the left.
|
|
|
|
|
2015-10-16 18:39:38 +01:00
|
|
|
`shr` shifts the dest right one bit position. The leftmost position becomes `c`,
|
|
|
|
and `c` becomes the bit that was shifted off the right.
|
|
|
|
|
2015-10-16 09:30:24 +01:00
|
|
|
* It is illegal if dest is a register besides `a`.
|
|
|
|
* It is illegal if dest is read-only.
|
|
|
|
* It is illegal if dest OR c is uninitialized.
|
2017-11-17 16:56:52 +00:00
|
|
|
* It is illegal if dest does not occur in the WRITES of the current routine.
|
2015-10-16 09:30:24 +01:00
|
|
|
|
2017-11-17 16:56:52 +00:00
|
|
|
Affects the c flag, requiring that it be in the WRITES of the
|
2015-10-16 18:39:38 +01:00
|
|
|
current routine, and it continues to be initialized afterwards.
|
|
|
|
|
2015-10-16 09:30:24 +01:00
|
|
|
### call ###
|
|
|
|
|
2015-10-20 09:33:30 +01:00
|
|
|
call <executable-name>
|
|
|
|
|
|
|
|
Transfers execution to the given executable, whether that is a previously-
|
|
|
|
defined routine, or a vector location which contains the address of a routine
|
2015-10-21 19:14:59 +01:00
|
|
|
which will be called indirectly. Execution will be transferred back to the
|
|
|
|
current routine, when execution of the executable is finished.
|
2015-10-16 09:30:24 +01:00
|
|
|
|
2017-11-17 16:56:52 +00:00
|
|
|
* It is illegal if any of the memory locations listed in the called routine's
|
|
|
|
`inputs` are uninitialized immediately before the call.
|
2015-10-16 09:30:24 +01:00
|
|
|
|
|
|
|
Just after the call,
|
|
|
|
|
2017-11-17 16:56:52 +00:00
|
|
|
* All memory locations listed in the called routine's `trashes` are considered
|
|
|
|
to now be uninitialized.
|
|
|
|
* All memory locations listed in the called routine's `outputs` are considered
|
|
|
|
to not be initialized.
|
2015-10-16 09:30:24 +01:00
|
|
|
|
2015-10-21 19:14:59 +01:00
|
|
|
### goto ###
|
|
|
|
|
|
|
|
goto <executable-name>
|
|
|
|
|
|
|
|
Unilaterally transfers execution to the given executable. Execution will not
|
|
|
|
be transferred back to the current routine when execution of the executable is
|
|
|
|
finished; rather, it will be transferred back to the caller of the current
|
|
|
|
routine.
|
|
|
|
|
|
|
|
If `goto` is used in a routine, it must be in tail position. That is, it
|
|
|
|
must be the final instruction in the routine.
|
|
|
|
|
|
|
|
Just before the goto,
|
|
|
|
|
2017-11-20 13:25:09 +00:00
|
|
|
* It is illegal if any of the memory locations in the target routine's
|
|
|
|
`inputs` list is uninitialized.
|
2015-10-21 19:14:59 +01:00
|
|
|
|
|
|
|
In addition,
|
|
|
|
|
2017-11-17 16:56:52 +00:00
|
|
|
* The target executable's WRITES must not include any locations
|
|
|
|
that are not already included in the current routine's WRITES.
|
2015-10-21 19:14:59 +01:00
|
|
|
|
2015-10-16 09:30:24 +01:00
|
|
|
### if ###
|
|
|
|
|
2015-10-16 19:15:01 +01:00
|
|
|
if <src-memory-location> {
|
|
|
|
<true-branch>
|
2015-10-16 09:30:24 +01:00
|
|
|
} else {
|
2015-10-16 19:15:01 +01:00
|
|
|
<false-branch>
|
2015-10-16 09:30:24 +01:00
|
|
|
}
|
|
|
|
|
2015-10-16 19:15:01 +01:00
|
|
|
Executes the true-branch if the value in src is nonzero, otherwise executes
|
|
|
|
the false-branch. The false-branch is optional may be omitted; in this case
|
|
|
|
it is treated like an empty block.
|
2015-10-16 09:30:24 +01:00
|
|
|
|
2015-10-16 19:15:01 +01:00
|
|
|
* It is illegal if src is not z, c, n, or v.
|
|
|
|
* It is illegal if src is not initialized.
|
|
|
|
* It is illegal if any location initialized at the end of the true-branch
|
|
|
|
is not initialized at the end of the false-branch, and vice versa.
|
2015-10-16 09:30:24 +01:00
|
|
|
|
2017-11-20 14:10:43 +00:00
|
|
|
The sense of the test can be inverted with `not`.
|
|
|
|
|
2015-10-18 13:37:35 +01:00
|
|
|
### repeat ###
|
|
|
|
|
|
|
|
repeat {
|
|
|
|
<block>
|
|
|
|
} until <src-memory-location>
|
|
|
|
|
|
|
|
Executes the block repeatedly until the src (observed at the end of the
|
|
|
|
execution of the block) is non-zero. The block is always executed as least
|
|
|
|
once.
|
|
|
|
|
|
|
|
* It is illegal if any memory location is uninitialized at the exit of
|
|
|
|
the loop when that memory location is initialized at the start of
|
|
|
|
the loop.
|
|
|
|
|
|
|
|
To simulate a "while" loop, use an `if` internal to the block, like
|
|
|
|
|
|
|
|
repeat {
|
|
|
|
cmp y, 25
|
|
|
|
if z {
|
|
|
|
}
|
|
|
|
} until z
|
|
|
|
|
2017-11-20 13:25:09 +00:00
|
|
|
"until" is optional, but if omitted, must be replaced with "forever":
|
2015-10-18 19:41:26 +01:00
|
|
|
|
2017-11-20 13:25:09 +00:00
|
|
|
repeat {
|
|
|
|
cmp y, 25
|
|
|
|
if z {
|
|
|
|
}
|
|
|
|
} forever
|
2015-10-18 19:41:26 +01:00
|
|
|
|
2017-11-20 13:25:09 +00:00
|
|
|
The sense of the test can be inverted with `not`.
|
2015-10-18 19:41:26 +01:00
|
|
|
|
2017-11-20 13:25:09 +00:00
|
|
|
repeat {
|
|
|
|
cmp y, 25
|
|
|
|
if z {
|
|
|
|
}
|
|
|
|
} until not z
|
2015-10-18 19:41:26 +01:00
|
|
|
|
2015-10-16 09:30:24 +01:00
|
|
|
Grammar
|
|
|
|
-------
|
|
|
|
|
|
|
|
Program ::= {Defn} {Routine}.
|
2016-06-16 11:10:43 -05:00
|
|
|
Defn ::= Type Ident<new> [Constraints] (":" Literal | "@" LitWord).
|
2015-10-19 13:04:08 +01:00
|
|
|
Type ::= "byte" ["table"] | "vector"
|
|
|
|
Constrnt::= ["inputs" LocExprs] ["outputs" LocExprs] ["trashes" LocExprs].
|
2016-06-16 11:10:43 -05:00
|
|
|
Routine ::= "routine" Ident<new> Constraints (Block | "@" LitWord).
|
2015-10-16 09:30:24 +01:00
|
|
|
LocExprs::= LocExpr {"," LocExpr}.
|
2017-11-17 15:54:50 +00:00
|
|
|
LocExpr ::= Register | Flag | Literal | Ident.
|
2015-10-16 09:30:24 +01:00
|
|
|
Register::= "a" | "x" | "y".
|
|
|
|
Flag ::= "c" | "z" | "n" | "v".
|
2016-06-16 11:10:43 -05:00
|
|
|
Literal ::= LitByte | LitWord.
|
2015-10-17 13:50:21 +01:00
|
|
|
LitByte ::= "0" ... "255".
|
|
|
|
LitWord ::= "0" ... "65535".
|
2015-10-16 09:30:24 +01:00
|
|
|
Block ::= "{" {Instr} "}".
|
2015-10-18 17:40:53 +01:00
|
|
|
Instr ::= "ld" LocExpr "," LocExpr ["+" LocExpr]
|
|
|
|
| "st" LocExpr "," LocExpr ["+" LocExpr]
|
2015-10-16 09:30:24 +01:00
|
|
|
| "add" LocExpr "," LocExpr
|
|
|
|
| "sub" LocExpr "," LocExpr
|
|
|
|
| "cmp" LocExpr "," LocExpr
|
|
|
|
| "and" LocExpr "," LocExpr
|
|
|
|
| "or" LocExpr "," LocExpr
|
|
|
|
| "xor" LocExpr "," LocExpr
|
|
|
|
| "shl" LocExpr
|
|
|
|
| "shr" LocExpr
|
|
|
|
| "inc" LocExpr
|
|
|
|
| "dec" LocExpr
|
2015-10-20 14:10:33 +01:00
|
|
|
| "call" Ident<routine>
|
|
|
|
| "goto" Ident<executable>
|
2015-10-18 13:37:35 +01:00
|
|
|
| "if" ["not"] LocExpr Block ["else" Block]
|
2015-10-18 15:32:28 +01:00
|
|
|
| "repeat" Block ("until" ["not"] LocExpr | "forever")
|
2015-10-18 19:41:26 +01:00
|
|
|
| "copy" LocExpr "," LocExpr ["+" LocExpr]
|
2015-10-18 13:37:35 +01:00
|
|
|
.
|