2014-03-31 23:31:30 +01:00
|
|
|
SixtyPical
|
|
|
|
==========
|
|
|
|
|
2014-04-01 13:01:27 +01:00
|
|
|
SixtyPical is a very low-level programming language, similar to 6502 assembly,
|
|
|
|
with block structure and static analysis through abstract interpretation.
|
|
|
|
|
|
|
|
It is a work in progress, currently at the proof-of-concept stage.
|
|
|
|
|
|
|
|
It is expected that a common use case for SixtyPical would be retroprogramming
|
2014-04-01 21:06:35 +01:00
|
|
|
for the Commodore 64 and other 6502-based computers such as the VIC-20.
|
2014-04-01 13:01:27 +01:00
|
|
|
|
|
|
|
Many SixtyPical instructions map precisely to 6502 opcodes. However, SixtyPical
|
|
|
|
is not an assembly language. The programmer does not have total control over
|
|
|
|
the layout of code and data in memory. The language has a type system which
|
|
|
|
distinguishes addresses from non-addresses (16-bit values for which it does
|
|
|
|
not make sense to treat them as addresses.) Some 6502 opcodes have no
|
|
|
|
SixtyPical equivalent. Some SixtyPical instructions are named after 6502
|
|
|
|
opcodes, but generate slightly different (safer, but intuitively related)
|
|
|
|
sequences of opcodes. Et cetera.
|
|
|
|
|
|
|
|
`sixtypical` is the reference implementation of SixtyPical. It is written in
|
|
|
|
Haskell. It can currently parse and analyze a SixtyPical program, and will
|
|
|
|
eventually be able to compile it to an Ophis assembler listing.
|
|
|
|
|
|
|
|
Concepts
|
|
|
|
--------
|
|
|
|
|
|
|
|
### Routines ###
|
|
|
|
|
|
|
|
Instead of the assembly-language subroutine, SixtyPical provides the _routine_
|
|
|
|
as the abstraction for a reusable sequence of code.
|
|
|
|
|
|
|
|
A routine may be called, or may be included inline, by another routine.
|
|
|
|
|
|
|
|
There is one top-level routine called `main` which represents the entire
|
|
|
|
program.
|
|
|
|
|
|
|
|
The instructions of a routine are analyzed using abstract interpretation.
|
|
|
|
One thing we specifically do is determine which registers and memory locations
|
|
|
|
are *not* affected by the routine.
|
|
|
|
|
|
|
|
If a register is not affected by a routine, then a caller of that routine may
|
|
|
|
assume that the value in that register is retained.
|
|
|
|
|
|
|
|
Of course, a routine may intentionally affect a register or memory location,
|
|
|
|
as an output. It must declare this. We're not there yet.
|
|
|
|
|
|
|
|
### Addresses ###
|
|
|
|
|
|
|
|
The body of a routine may not refer to an address literally. It must use
|
|
|
|
a symbol that was declared previously.
|
|
|
|
|
|
|
|
An address may be declared with `reserve`, which is like `.data` or `.bss`
|
|
|
|
in an assembler. This is an address into the program's data. It is global
|
|
|
|
to all routines.
|
|
|
|
|
|
|
|
An address may be declared with `locate`, which is like `.alias` in an
|
|
|
|
assembler, with the understanding that the value will be treated "like an
|
|
|
|
address." This is generally an address into the operating system or hardware
|
|
|
|
(e.g. kernal routine, I/O port, etc.)
|
|
|
|
|
2014-04-01 20:34:16 +01:00
|
|
|
Not there yet:
|
|
|
|
|
|
|
|
> Inside a routine, an address may be declared with `temporary`. This is like
|
|
|
|
> `static` in C, except the value at that address is not guaranteed to be
|
|
|
|
> retained between invokations of the routine. Such addresses may only be used
|
|
|
|
> within the routine where they are declared. If analysis indicates that two
|
|
|
|
> temporary addresses are never used simultaneously, they may be merged
|
|
|
|
> to the same address.
|
|
|
|
|
|
|
|
An address knows what kind of data is stored at the address:
|
|
|
|
|
|
|
|
* `byte`: an 8-bit byte. not part of a word. not to be used as an address.
|
|
|
|
(could be an index though.)
|
|
|
|
* `word`: a 16-bit word. not to be used as an address.
|
2014-04-01 21:06:35 +01:00
|
|
|
* `vector`: a 16-bit address of a routine. Only a handful of operations
|
|
|
|
are supported on vectors:
|
|
|
|
|
|
|
|
* copying the contents of one vector to another
|
|
|
|
* copying the address of a routine into a vector
|
|
|
|
* jumping indirectly to a vector (i.e. to the code at the address
|
|
|
|
contained in the vector (and this can only happen at the end of a
|
2014-04-01 23:22:38 +01:00
|
|
|
routine (NYI))
|
2014-04-01 21:06:35 +01:00
|
|
|
* `jsr`'ing indirectly to a vector (which is done with a fun
|
2014-04-01 23:22:38 +01:00
|
|
|
generated trick (NYI))
|
2014-04-01 21:06:35 +01:00
|
|
|
|
2014-04-01 23:22:38 +01:00
|
|
|
* `byte table`: a series of `byte`s contiguous in memory starting from the
|
|
|
|
address. This is the only kind of address that can be used in
|
|
|
|
indexed addressing.
|
2014-04-01 13:01:27 +01:00
|
|
|
|
|
|
|
### Blocks ###
|
|
|
|
|
2014-04-01 21:06:35 +01:00
|
|
|
Each routine is a block. It may be composed of inner blocks, if those
|
|
|
|
inner blocks are attached to certain instructions.
|
2014-04-01 13:01:27 +01:00
|
|
|
|
|
|
|
SixtyPical does not have instructions that map literally to the 6502 branch
|
2014-04-01 17:00:16 +01:00
|
|
|
instructions. Instead, it has an `if` construct, with two blocks (for the
|
|
|
|
"then" and `else` parts), and the branch instructions map to conditions for
|
|
|
|
this construct.
|
2014-04-01 13:01:27 +01:00
|
|
|
|
2014-04-01 17:55:59 +01:00
|
|
|
Similarly, there is a `repeat` construct. The same branch instructions can
|
|
|
|
be used in the condition to this construct. In this case, they branch back
|
|
|
|
to the top of the `repeat` loop.
|
|
|
|
|
2014-04-01 13:01:27 +01:00
|
|
|
The abstract states of the machine at each of the different block exits are
|
|
|
|
merged during analysis. If any register or memory location is treated
|
|
|
|
inconsistently (e.g. updated in one branch of the test, but not the other,)
|
|
|
|
that register cannot subsequently be used without a declaration to the effect
|
|
|
|
that we know what's going on. (This is all a bit fuzzy right now.)
|
|
|
|
|
2014-04-01 14:33:57 +01:00
|
|
|
There is also no `rts` instruction. It is included at the end of a routine,
|
2014-04-01 17:00:16 +01:00
|
|
|
but only when the routine is used as a subroutine. Also, if the routine
|
|
|
|
ends by `jsr`ing another routine, it reserves the right to do a tail-call
|
|
|
|
or even a fallthrough.
|
2014-04-01 14:33:57 +01:00
|
|
|
|
|
|
|
There are also _with_ instructions, which are associated with an opcode
|
|
|
|
that has a natural symmetrical opcode (e.g. `pha`, `sei`). These instructions
|
|
|
|
take a block. The natural symmetrical opcode is inserted at the end of the
|
|
|
|
block.
|
|
|
|
|
2014-04-01 18:45:17 +01:00
|
|
|
Unsupported Opcodes
|
|
|
|
-------------------
|
|
|
|
|
|
|
|
6502 opcodes with no language-level equivalent instructions in SixtyPical
|
2014-04-01 21:06:35 +01:00
|
|
|
are `brk`, `cli`, `pla`, `plp`, `rti`, and `rts`. These may be
|
2014-04-01 18:45:17 +01:00
|
|
|
inserted into the output program as a SixtyPical → 6502 compiler sees fit,
|
|
|
|
however.
|
|
|
|
|
|
|
|
Note to self, the `pl` opcodes *do* change flags.
|
|
|
|
|
2014-04-01 17:00:16 +01:00
|
|
|
Instruction Support so far
|
|
|
|
--------------------------
|
|
|
|
|
|
|
|
A `X` indicates unsupported. A `!` indicates will-not-support.
|
|
|
|
|
|
|
|
Funny syntax indicates use of a special form.
|
|
|
|
|
|
|
|
In these, `absolute` must be a `reserve`d or `locate`d address.
|
|
|
|
|
2014-04-02 09:05:30 +01:00
|
|
|
.
|
|
|
|
adc #immediate
|
|
|
|
adc absolute
|
2014-04-01 17:00:16 +01:00
|
|
|
|
2014-04-02 09:05:30 +01:00
|
|
|
and #immediate
|
|
|
|
and absolute
|
2014-04-01 17:00:16 +01:00
|
|
|
|
|
|
|
X asl
|
|
|
|
X asl absolute
|
|
|
|
|
|
|
|
if bcc { block } else { block }
|
|
|
|
|
|
|
|
if bcs { block } else { block }
|
|
|
|
|
|
|
|
if beq { block } else { block }
|
|
|
|
|
|
|
|
X bit absolute
|
|
|
|
|
|
|
|
if bmi { block } else { block }
|
|
|
|
|
|
|
|
if bne { block } else { block }
|
|
|
|
|
|
|
|
if bpl { block } else { block }
|
|
|
|
|
|
|
|
if bvc { block } else { block }
|
|
|
|
|
|
|
|
if bvs { block } else { block }
|
|
|
|
|
2014-04-01 18:04:43 +01:00
|
|
|
clc
|
2014-04-01 17:00:16 +01:00
|
|
|
|
2014-04-01 18:04:43 +01:00
|
|
|
cld
|
2014-04-01 17:00:16 +01:00
|
|
|
|
2014-04-01 18:04:43 +01:00
|
|
|
clv
|
2014-04-01 17:00:16 +01:00
|
|
|
|
2014-04-01 18:34:41 +01:00
|
|
|
cmp #immediate
|
2014-04-01 17:00:16 +01:00
|
|
|
cmp absolute
|
|
|
|
|
2014-04-01 18:34:41 +01:00
|
|
|
cpx #immediate
|
2014-04-01 17:04:14 +01:00
|
|
|
cpx absolute
|
2014-04-01 17:00:16 +01:00
|
|
|
|
2014-04-01 18:34:41 +01:00
|
|
|
cpy #immediate
|
2014-04-01 17:04:14 +01:00
|
|
|
cpy absolute
|
2014-04-01 17:00:16 +01:00
|
|
|
|
2014-04-01 17:32:33 +01:00
|
|
|
dec absolute
|
2014-04-01 17:00:16 +01:00
|
|
|
|
2014-04-01 17:32:33 +01:00
|
|
|
dex
|
2014-04-01 17:00:16 +01:00
|
|
|
|
2014-04-01 17:32:33 +01:00
|
|
|
dey
|
2014-04-01 17:00:16 +01:00
|
|
|
|
|
|
|
X eor #immediate
|
|
|
|
X eor absolute
|
|
|
|
|
2014-04-01 17:32:33 +01:00
|
|
|
inc absolute
|
2014-04-01 17:00:16 +01:00
|
|
|
|
2014-04-01 17:32:33 +01:00
|
|
|
inx
|
2014-04-01 17:00:16 +01:00
|
|
|
|
2014-04-01 17:32:33 +01:00
|
|
|
iny
|
2014-04-01 17:00:16 +01:00
|
|
|
|
|
|
|
* jsr routine
|
2014-04-01 21:06:35 +01:00
|
|
|
X jsr vector
|
|
|
|
|
|
|
|
X jmp routine
|
|
|
|
* jmp vector
|
2014-04-01 17:00:16 +01:00
|
|
|
|
2014-04-01 18:34:41 +01:00
|
|
|
lda #immediate
|
2014-04-01 17:00:16 +01:00
|
|
|
lda absolute
|
2014-04-01 23:47:10 +01:00
|
|
|
lda absolute, x
|
2014-04-02 09:05:30 +01:00
|
|
|
lda absolute, y
|
2014-04-01 17:00:16 +01:00
|
|
|
|
2014-04-01 18:34:41 +01:00
|
|
|
ldx #immediate
|
2014-04-01 17:00:16 +01:00
|
|
|
ldx absolute
|
|
|
|
|
2014-04-01 18:34:41 +01:00
|
|
|
ldy #immediate
|
2014-04-01 17:00:16 +01:00
|
|
|
ldy absolute
|
|
|
|
|
|
|
|
X lsr
|
|
|
|
X lsr absolute
|
|
|
|
|
|
|
|
nop
|
|
|
|
|
2014-04-02 09:11:39 +01:00
|
|
|
ora #immediate
|
|
|
|
ora absolute
|
2014-04-01 17:00:16 +01:00
|
|
|
|
|
|
|
X pha { block }
|
|
|
|
|
|
|
|
X php { block }
|
|
|
|
|
|
|
|
X rol
|
|
|
|
X rol absolute
|
|
|
|
|
|
|
|
X ror
|
|
|
|
X ror absolute
|
|
|
|
|
2014-04-02 09:11:39 +01:00
|
|
|
sbc #immediate
|
|
|
|
sbc absolute
|
2014-04-01 17:00:16 +01:00
|
|
|
|
2014-04-01 18:04:43 +01:00
|
|
|
sec
|
2014-04-01 17:00:16 +01:00
|
|
|
|
2014-04-01 18:04:43 +01:00
|
|
|
sed
|
2014-04-01 17:00:16 +01:00
|
|
|
|
2014-04-02 09:05:30 +01:00
|
|
|
sei { block }
|
2014-04-01 17:00:16 +01:00
|
|
|
|
|
|
|
sta absolute
|
2014-04-01 23:47:10 +01:00
|
|
|
sta absolute, x
|
2014-04-02 09:05:30 +01:00
|
|
|
sta absolute, y
|
2014-04-01 17:00:16 +01:00
|
|
|
|
|
|
|
stx absolute
|
|
|
|
|
|
|
|
sty absolute
|
|
|
|
|
|
|
|
tax
|
|
|
|
|
|
|
|
tay
|
|
|
|
|
|
|
|
X tsx
|
|
|
|
|
|
|
|
txa
|
|
|
|
|
|
|
|
X txs
|
|
|
|
|
|
|
|
tya
|
|
|
|
|
2014-04-01 14:33:57 +01:00
|
|
|
TODO
|
|
|
|
----
|
|
|
|
|
2014-04-02 14:40:01 +01:00
|
|
|
* comments
|
2014-04-01 23:47:10 +01:00
|
|
|
* Initial values for reserved, incl. tables
|
2014-04-02 14:40:01 +01:00
|
|
|
* give length for tables, must be there for reserved
|
2014-04-01 17:15:21 +01:00
|
|
|
* Character tables ("strings" to everybody else)
|
|
|
|
* Work out the analyses again and document them
|
2014-04-02 14:40:01 +01:00
|
|
|
* lda wordaddress --> is not legal. use lda <wordaddr or lda >wordaddr
|
2014-04-01 14:33:57 +01:00
|
|
|
* Addressing modes; rename instructions to match
|
|
|
|
|
2014-04-01 13:01:27 +01:00
|
|
|
Tests
|
|
|
|
-----
|
|
|
|
|
2014-03-31 23:31:30 +01:00
|
|
|
-> Tests for functionality "Parse SixtyPical program"
|
2014-04-01 13:01:27 +01:00
|
|
|
|
2014-03-31 23:31:30 +01:00
|
|
|
-> Functionality "Parse SixtyPical program" is implemented by
|
|
|
|
-> shell command "bin/sixtypical parse %(test-file)"
|
|
|
|
|
2014-04-01 13:01:27 +01:00
|
|
|
-> Tests for functionality "Check SixtyPical program"
|
|
|
|
|
|
|
|
-> Functionality "Check SixtyPical program" is implemented by
|
|
|
|
-> shell command "bin/sixtypical check %(test-file)"
|
|
|
|
|
|
|
|
`main` must be present.
|
|
|
|
|
2014-03-31 23:31:30 +01:00
|
|
|
| routine main {
|
|
|
|
| nop
|
|
|
|
| }
|
2014-04-01 13:01:27 +01:00
|
|
|
= True
|
|
|
|
|
|
|
|
| routine frog {
|
|
|
|
| nop
|
|
|
|
| }
|
|
|
|
? missing 'main' routine
|
|
|
|
|
2014-04-01 23:22:38 +01:00
|
|
|
A program may `reserve` and `assign`.
|
2014-03-31 23:31:30 +01:00
|
|
|
|
2014-04-01 23:22:38 +01:00
|
|
|
| reserve byte lives
|
|
|
|
| assign byte gdcol 647
|
2014-03-31 23:31:30 +01:00
|
|
|
| reserve word score
|
2014-04-01 23:22:38 +01:00
|
|
|
| assign word memstr 641
|
|
|
|
| reserve vector v
|
|
|
|
| assign vector cinv 788
|
|
|
|
| reserve byte table frequencies
|
|
|
|
| assign byte table screen 1024
|
2014-03-31 23:31:30 +01:00
|
|
|
| routine main {
|
2014-04-01 23:22:38 +01:00
|
|
|
| nop
|
2014-03-31 23:31:30 +01:00
|
|
|
| }
|
2014-04-01 13:01:27 +01:00
|
|
|
= True
|
2014-03-31 23:31:30 +01:00
|
|
|
|
2014-04-02 09:19:53 +01:00
|
|
|
A program may declare an `external`.
|
|
|
|
|
|
|
|
| external blastoff 49152
|
|
|
|
| routine main {
|
|
|
|
| jsr blastoff
|
|
|
|
| }
|
|
|
|
= True
|
|
|
|
|
2014-03-31 23:31:30 +01:00
|
|
|
All declarations (`reserve`s and `assign`s) must come before any `routines`.
|
|
|
|
|
|
|
|
| routine main {
|
2014-04-01 13:29:45 +01:00
|
|
|
| lda score
|
2014-03-31 23:31:30 +01:00
|
|
|
| }
|
|
|
|
| reserve word score
|
|
|
|
? expecting "routine"
|
2014-04-01 13:29:45 +01:00
|
|
|
|
|
|
|
All locations used in all routines must be declared first.
|
|
|
|
|
|
|
|
| reserve word score
|
|
|
|
| routine main {
|
|
|
|
| lda score
|
|
|
|
| cmp screen
|
|
|
|
| }
|
|
|
|
? undeclared location
|
|
|
|
|
|
|
|
Even in inner blocks.
|
|
|
|
|
|
|
|
| reserve word score
|
2014-04-01 18:23:04 +01:00
|
|
|
| assign word screen 1024
|
2014-04-01 13:29:45 +01:00
|
|
|
| routine main {
|
|
|
|
| lda score
|
|
|
|
| cmp screen
|
2014-04-01 17:00:16 +01:00
|
|
|
| if beq {
|
2014-04-01 13:29:45 +01:00
|
|
|
| lda score
|
|
|
|
| } else {
|
|
|
|
| lda fnord
|
|
|
|
| }
|
|
|
|
| }
|
|
|
|
? undeclared location
|
2014-04-01 13:38:42 +01:00
|
|
|
|
2014-04-02 09:19:53 +01:00
|
|
|
All routines jsr'ed to must be defined, or external.
|
|
|
|
|
|
|
|
| routine main {
|
|
|
|
| jsr blastoff
|
|
|
|
| }
|
|
|
|
? undeclared routine
|
|
|
|
|
2014-04-01 22:29:50 +01:00
|
|
|
No duplicate location names in declarations.
|
2014-04-01 13:38:42 +01:00
|
|
|
|
|
|
|
| reserve word score
|
|
|
|
| assign word score 4000
|
|
|
|
| routine main {
|
|
|
|
| nop
|
|
|
|
| }
|
2014-04-01 22:29:50 +01:00
|
|
|
? duplicate location name
|
|
|
|
|
2014-04-02 09:19:53 +01:00
|
|
|
No duplicate routine names.
|
2014-04-01 22:29:50 +01:00
|
|
|
|
|
|
|
| routine main {
|
|
|
|
| nop
|
|
|
|
| }
|
|
|
|
| routine main {
|
|
|
|
| txa
|
|
|
|
| }
|
|
|
|
? duplicate routine name
|
2014-04-01 14:48:07 +01:00
|
|
|
|
2014-04-02 09:19:53 +01:00
|
|
|
No duplicate routine names, including externals.
|
|
|
|
|
|
|
|
| external main 7000
|
|
|
|
| routine main {
|
|
|
|
| nop
|
|
|
|
| }
|
|
|
|
? duplicate routine name
|
|
|
|
|
2014-04-01 21:06:35 +01:00
|
|
|
We can jump to a vector.
|
|
|
|
|
|
|
|
| reserve vector blah
|
|
|
|
| routine main {
|
|
|
|
| jmp blah
|
|
|
|
| }
|
|
|
|
= True
|
|
|
|
|
|
|
|
We can't jump to a word.
|
|
|
|
|
2014-04-01 22:29:50 +01:00
|
|
|
| reserve word blah
|
|
|
|
| routine main {
|
|
|
|
| jmp blah
|
|
|
|
| }
|
|
|
|
? jmp to non-vector
|
|
|
|
|
|
|
|
We can't jump to a byte.
|
|
|
|
|
|
|
|
| assign byte screen 1024
|
|
|
|
| routine main {
|
|
|
|
| jmp screen
|
|
|
|
| }
|
|
|
|
? jmp to non-vector
|
2014-04-01 21:06:35 +01:00
|
|
|
|
2014-04-01 23:22:38 +01:00
|
|
|
We can absolute-indexed a byte table.
|
|
|
|
|
|
|
|
| assign byte table screen 1024
|
|
|
|
| routine main {
|
|
|
|
| sta screen, x
|
|
|
|
| }
|
|
|
|
= True
|
|
|
|
|
|
|
|
We cannot absolute-indexed a byte.
|
|
|
|
|
|
|
|
| assign byte screen 1024
|
|
|
|
| routine main {
|
|
|
|
| sta screen, x
|
|
|
|
| }
|
|
|
|
? indexed access of non-table
|
|
|
|
|
|
|
|
We cannot absolute-indexed a word.
|
|
|
|
|
|
|
|
| assign word screen 1024
|
|
|
|
| routine main {
|
|
|
|
| sta screen, x
|
|
|
|
| }
|
|
|
|
? indexed access of non-table
|
|
|
|
|
2014-04-01 14:48:07 +01:00
|
|
|
-> Tests for functionality "Emit ASM for SixtyPical program"
|
|
|
|
|
|
|
|
-> Functionality "Emit ASM for SixtyPical program" is implemented by
|
|
|
|
-> shell command "bin/sixtypical emit %(test-file)"
|
|
|
|
|
|
|
|
| reserve word score
|
2014-04-01 23:22:38 +01:00
|
|
|
| assign byte table screen 1024
|
2014-04-01 14:48:07 +01:00
|
|
|
| routine main {
|
2014-04-01 18:23:04 +01:00
|
|
|
| lda #4
|
2014-04-01 18:34:41 +01:00
|
|
|
| ldx #0
|
2014-04-02 14:40:01 +01:00
|
|
|
| ldy #$FF
|
2014-04-01 14:48:07 +01:00
|
|
|
| lda screen
|
2014-04-01 23:47:10 +01:00
|
|
|
| lda screen, x
|
2014-04-02 09:05:30 +01:00
|
|
|
| lda screen, y
|
2014-04-01 17:32:33 +01:00
|
|
|
| inc screen
|
2014-04-01 14:48:07 +01:00
|
|
|
| tax
|
2014-04-01 17:32:33 +01:00
|
|
|
| inx
|
|
|
|
| dex
|
2014-04-01 15:49:54 +01:00
|
|
|
| stx score
|
2014-04-01 14:48:07 +01:00
|
|
|
| tay
|
2014-04-01 17:32:33 +01:00
|
|
|
| iny
|
|
|
|
| dey
|
2014-04-01 15:49:54 +01:00
|
|
|
| sty score
|
2014-04-01 14:48:07 +01:00
|
|
|
| cmp score
|
2014-04-01 18:34:41 +01:00
|
|
|
| cmp #30
|
2014-04-01 14:48:07 +01:00
|
|
|
| ldx score
|
2014-04-01 17:04:14 +01:00
|
|
|
| cpx screen
|
2014-04-01 18:34:41 +01:00
|
|
|
| cpx #31
|
2014-04-01 14:48:07 +01:00
|
|
|
| txa
|
|
|
|
| ldy score
|
2014-04-01 17:04:14 +01:00
|
|
|
| cpy screen
|
2014-04-01 18:34:41 +01:00
|
|
|
| cpy #32
|
2014-04-01 14:48:07 +01:00
|
|
|
| tya
|
2014-04-01 15:49:54 +01:00
|
|
|
| sta screen
|
2014-04-01 23:22:38 +01:00
|
|
|
| sta screen, x
|
2014-04-02 09:05:30 +01:00
|
|
|
| sta screen, y
|
2014-04-01 17:32:33 +01:00
|
|
|
| dec screen
|
2014-04-01 18:04:43 +01:00
|
|
|
| clc
|
|
|
|
| cld
|
|
|
|
| clv
|
|
|
|
| sec
|
|
|
|
| sed
|
2014-04-02 09:05:30 +01:00
|
|
|
| adc #8
|
|
|
|
| adc screen
|
|
|
|
| and #8
|
|
|
|
| and screen
|
2014-04-02 09:11:39 +01:00
|
|
|
| sbc #8
|
|
|
|
| sbc screen
|
|
|
|
| ora #8
|
|
|
|
| ora screen
|
2014-04-01 14:48:07 +01:00
|
|
|
| }
|
|
|
|
= main:
|
2014-04-01 18:23:04 +01:00
|
|
|
= lda #4
|
2014-04-01 18:34:41 +01:00
|
|
|
= ldx #0
|
|
|
|
= ldy #255
|
2014-04-01 14:48:07 +01:00
|
|
|
= lda screen
|
2014-04-01 23:47:10 +01:00
|
|
|
= lda screen, x
|
2014-04-02 09:05:30 +01:00
|
|
|
= lda screen, y
|
2014-04-01 17:32:33 +01:00
|
|
|
= inc screen
|
2014-04-01 14:48:07 +01:00
|
|
|
= tax
|
2014-04-01 17:32:33 +01:00
|
|
|
= inx
|
|
|
|
= dex
|
2014-04-01 15:49:54 +01:00
|
|
|
= stx score
|
2014-04-01 14:48:07 +01:00
|
|
|
= tay
|
2014-04-01 17:32:33 +01:00
|
|
|
= iny
|
|
|
|
= dey
|
2014-04-01 15:49:54 +01:00
|
|
|
= sty score
|
2014-04-01 14:48:07 +01:00
|
|
|
= cmp score
|
2014-04-01 18:34:41 +01:00
|
|
|
= cmp #30
|
2014-04-01 14:59:28 +01:00
|
|
|
= ldx score
|
2014-04-01 17:04:14 +01:00
|
|
|
= cpx screen
|
2014-04-01 18:34:41 +01:00
|
|
|
= cpx #31
|
2014-04-01 14:48:07 +01:00
|
|
|
= txa
|
2014-04-01 14:59:28 +01:00
|
|
|
= ldy score
|
2014-04-01 17:04:14 +01:00
|
|
|
= cpy screen
|
2014-04-01 18:34:41 +01:00
|
|
|
= cpy #32
|
2014-04-01 14:48:07 +01:00
|
|
|
= tya
|
2014-04-01 15:49:54 +01:00
|
|
|
= sta screen
|
2014-04-01 23:22:38 +01:00
|
|
|
= sta screen, x
|
2014-04-02 09:05:30 +01:00
|
|
|
= sta screen, y
|
2014-04-01 17:32:33 +01:00
|
|
|
= dec screen
|
2014-04-01 18:04:43 +01:00
|
|
|
= clc
|
|
|
|
= cld
|
|
|
|
= clv
|
|
|
|
= sec
|
|
|
|
= sed
|
2014-04-02 09:05:30 +01:00
|
|
|
= adc #8
|
|
|
|
= adc screen
|
|
|
|
= and #8
|
|
|
|
= and screen
|
2014-04-02 09:11:39 +01:00
|
|
|
= sbc #8
|
|
|
|
= sbc screen
|
|
|
|
= ora #8
|
|
|
|
= ora screen
|
2014-04-01 14:48:07 +01:00
|
|
|
= rts
|
2014-04-02 13:53:56 +01:00
|
|
|
=
|
|
|
|
= score: .word 0
|
|
|
|
= .alias screen 1024
|
2014-04-01 17:15:21 +01:00
|
|
|
|
2014-04-02 14:40:01 +01:00
|
|
|
| assign word screen $0400
|
2014-04-01 17:15:21 +01:00
|
|
|
| routine main {
|
|
|
|
| lda screen
|
|
|
|
| cmp screen
|
|
|
|
| if beq {
|
|
|
|
| tax
|
|
|
|
| } else {
|
|
|
|
| tay
|
|
|
|
| }
|
|
|
|
| sta screen
|
|
|
|
| }
|
|
|
|
= main:
|
|
|
|
= lda screen
|
|
|
|
= cmp screen
|
2014-04-01 19:44:45 +01:00
|
|
|
= BEQ _label_1
|
2014-04-01 17:15:21 +01:00
|
|
|
= tay
|
2014-04-01 19:44:45 +01:00
|
|
|
= jmp _past_1
|
|
|
|
= _label_1:
|
2014-04-01 17:15:21 +01:00
|
|
|
= tax
|
2014-04-01 19:44:45 +01:00
|
|
|
= _past_1:
|
2014-04-01 17:15:21 +01:00
|
|
|
= sta screen
|
|
|
|
= rts
|
2014-04-02 13:53:56 +01:00
|
|
|
=
|
|
|
|
= .alias screen 1024
|
2014-04-01 17:55:59 +01:00
|
|
|
|
2014-04-01 18:23:04 +01:00
|
|
|
| assign byte screen 1024
|
2014-04-01 17:55:59 +01:00
|
|
|
| reserve byte zero
|
|
|
|
| routine main {
|
|
|
|
| ldy zero
|
|
|
|
| repeat bne {
|
|
|
|
| inc screen
|
|
|
|
| dey
|
|
|
|
| cpy zero
|
|
|
|
| }
|
|
|
|
| sty screen
|
|
|
|
| }
|
|
|
|
= main:
|
|
|
|
= ldy zero
|
|
|
|
=
|
2014-04-01 19:44:45 +01:00
|
|
|
= _repeat_1:
|
2014-04-01 17:55:59 +01:00
|
|
|
= inc screen
|
|
|
|
= dey
|
|
|
|
= cpy zero
|
2014-04-01 19:44:45 +01:00
|
|
|
= BNE _repeat_1
|
2014-04-01 17:55:59 +01:00
|
|
|
= sty screen
|
|
|
|
= rts
|
2014-04-02 13:53:56 +01:00
|
|
|
=
|
|
|
|
= .alias screen 1024
|
|
|
|
= zero: .byte 0
|
2014-04-01 19:44:45 +01:00
|
|
|
|
|
|
|
Nested ifs.
|
|
|
|
|
|
|
|
| routine main {
|
|
|
|
| if beq {
|
|
|
|
| if bcc {
|
|
|
|
| lda #0
|
|
|
|
| } else {
|
|
|
|
| if bvs {
|
|
|
|
| lda #1
|
|
|
|
| } else {
|
|
|
|
| lda #2
|
|
|
|
| }
|
|
|
|
| }
|
|
|
|
| } else {
|
|
|
|
| lda #3
|
|
|
|
| }
|
|
|
|
| }
|
|
|
|
= main:
|
|
|
|
= BEQ _label_3
|
|
|
|
= lda #3
|
|
|
|
= jmp _past_3
|
|
|
|
= _label_3:
|
|
|
|
= BCC _label_2
|
|
|
|
= BVS _label_1
|
|
|
|
= lda #2
|
|
|
|
= jmp _past_1
|
|
|
|
= _label_1:
|
|
|
|
= lda #1
|
|
|
|
= _past_1:
|
|
|
|
= jmp _past_2
|
|
|
|
= _label_2:
|
|
|
|
= lda #0
|
|
|
|
= _past_2:
|
|
|
|
= _past_3:
|
|
|
|
= rts
|
2014-04-01 20:34:16 +01:00
|
|
|
|
2014-04-01 21:06:35 +01:00
|
|
|
Installing an interrupt handler (at the Kernal level, i.e. with CINV)
|
2014-04-01 20:34:16 +01:00
|
|
|
|
|
|
|
| assign byte screen 1024
|
|
|
|
| assign vector cinv 788
|
|
|
|
| reserve vector save_cinv
|
|
|
|
|
|
2014-04-01 21:06:35 +01:00
|
|
|
| routine main {
|
2014-04-01 20:34:16 +01:00
|
|
|
| sei {
|
|
|
|
| copy vector cinv to save_cinv
|
|
|
|
| copy routine our_cinv to cinv
|
|
|
|
| }
|
|
|
|
| }
|
|
|
|
|
|
|
|
|
| routine our_cinv {
|
|
|
|
| inc screen
|
|
|
|
| jmp save_cinv
|
|
|
|
| }
|
2014-04-01 22:05:02 +01:00
|
|
|
= main:
|
|
|
|
= sei
|
|
|
|
= lda cinv
|
|
|
|
= sta save_cinv
|
|
|
|
= lda cinv+1
|
|
|
|
= sta save_cinv+1
|
|
|
|
= lda #<our_cinv
|
|
|
|
= sta cinv
|
|
|
|
= lda #>our_cinv
|
|
|
|
= sta cinv+1
|
|
|
|
= cli
|
|
|
|
= rts
|
|
|
|
=
|
|
|
|
= our_cinv:
|
|
|
|
= inc screen
|
|
|
|
= jmp (save_cinv)
|
|
|
|
= rts
|
2014-04-02 13:53:56 +01:00
|
|
|
=
|
|
|
|
= .alias screen 1024
|
|
|
|
= .alias cinv 788
|
|
|
|
= save_cinv: .word 0
|