Expressions
Ophis permits a reasonably rich set of arithmetic operations to be
done at assemble time. So far, all of our arguments and values
have either been constants or label names. In this chapter, we
will modify the print macro so that it calls a
subroutine to do the actual printing. This will shrink the final
code size a fair bit.
Here's our printing routine. It's fairly straightforward.
; PRINTSTR routine. Accumulator stores the low byte of the address,
; X register stores the high byte. Destroys the values of $10 and
; $11.
.scope
printstr:
sta $10
stx $11
ldy #$00
_lp: lda ($10), y
beq _done
jsr chrout
iny
bne _lp
_done: rts
.scend
However, now we are faced with the problem of what to do with
the print macro. We need to take a 16-bit
value and store it in two 8-bit registers. We can use
the < and > operators
to take the low or high byte of a word, respectively.
The print macro becomes:
.macro print
lda #<_1
ldx #>_1
jsr printstr
.macend
Also, since BASIC uses the locations $10 and $11, we should really
cache them at the start of the program and restore them at the
end:
.data
.org $C000
.space cache 2
.text
; Save the zero page locations that printstr uses.
lda $10
sta cache
lda $11
sta cache+1
; ... main program goes here ...
; Restore the zero page values printstr uses.
lda cache
sta $10
lda cache+1
sta $11
Note that we only have to name cache once, but
can use addition to refer to any offset from it.We
could spare ourselves some trouble here and use $fb instead of
$10, which BASIC does not use, but the
example is more thorough this way.
Ophis supports following operations, with the following precedence
levels (higher entries bind more tightly):
Ophis Operators
Operators
Description
[ ]Parenthesized expressions
< >Byte selection (low, high)
* /Multiply, divide
+ -Add, subtract
| & ^Bitwise OR, AND, XOR
Note that brackets, not parentheses, are used to group arithmetic
operations. Parentheses are reserved for the indirect addressing
modes.
The code for this version of the code is
in .