a2d/macros.inc

520 lines
14 KiB
PHP
Raw Normal View History

2018-03-04 21:36:00 -08:00
;;; ============================================================
2018-02-05 19:01:17 -08:00
;;; Generic Macros
2018-03-04 21:36:00 -08:00
;;; ============================================================
2018-02-05 19:01:17 -08:00
2018-03-05 08:41:22 -08:00
.define _is_immediate(arg) (.match (.mid (0, 1, {arg}), #))
.define _is_register(arg) (.match ({arg}, x) .or .match ({arg}, y))
.define _immediate_value(arg) (.right (.tcount ({arg})-1, {arg}))
.macro _op_lo op, arg
.if _is_immediate {arg}
op #<_immediate_value {arg}
.else
op arg
.endif
.endmacro
.macro _op_hi op, arg
.if _is_immediate {arg}
op #>_immediate_value {arg}
.else
op arg+1
.endif
.endmacro
2018-02-08 21:42:59 -08:00
2018-05-02 20:04:56 -07:00
;;; ============================================================
;;; Length-prefixed string
;;;
;;; Can include control chars by using:
;;;
;;; PASCAL_STRING {"abc",$0D,"def"}
.macro PASCAL_STRING str,res
.local data
.local end
.byte end - data
data: .byte str
end:
.if .paramcount > 1
.res res - (end - data), 0
.endif
.endmacro
2018-03-04 21:36:00 -08:00
;;; ============================================================
2018-02-05 21:49:39 -08:00
;;; Pad with zeros to the given address
.macro PAD_TO addr
.res addr - *, 0
.endmacro
2018-02-05 19:01:17 -08:00
2018-03-04 21:36:00 -08:00
;;; ============================================================
2018-02-15 19:29:05 -08:00
;;; Common patterns
2018-03-07 10:39:36 -08:00
.define AS_WORD(arg) arg & $FFFF
2018-04-04 19:16:50 -07:00
.macro return arg
2018-02-15 19:29:05 -08:00
lda arg
rts
.endmacro
2018-04-04 19:16:50 -07:00
.macro copy arg1, arg2, arg3, arg4
.if _is_register {arg2} && _is_register {arg4}
;; indexed load/indexed store
lda arg1,arg2
sta arg3,arg4
.elseif _is_register {arg2}
;; indexed load variant (arg2 is x or y)
lda arg1,arg2
sta arg3
.elseif _is_register {arg3}
;; indexed store variant (arg3 is x or y)
lda arg1
sta arg2,arg3
.else
2018-04-04 19:16:50 -07:00
lda arg1
sta arg2
.endif
2018-04-04 19:16:50 -07:00
.endmacro
2018-03-04 21:36:00 -08:00
;;; ============================================================
2018-02-05 19:01:17 -08:00
;;; Calls with one parameter (address in A,X)
.macro addr_call target, addr
lda #<addr
ldx #>addr
jsr target
.endmacro
2018-02-08 21:42:59 -08:00
.macro addr_call_indirect target, addr
lda addr
ldx addr+1
jsr target
.endmacro
2018-02-05 19:01:17 -08:00
.macro addr_jump target, addr
lda #<addr
ldx #>addr
jmp target
.endmacro
2018-03-04 21:36:00 -08:00
;;; ============================================================
2018-02-05 19:01:17 -08:00
;;; Calls with two paramters (call # in y, address in A,X)
;;; (various output orders to match original binary)
.macro axy_call target, yparam, addr
lda #<addr
ldx #>addr
ldy #yparam
jsr target
.endmacro
.macro yax_call target, yparam, addr
ldy #yparam
lda #<addr
ldx #>addr
jsr target
.endmacro
.macro yxa_call target, yparam, addr
ldy #yparam
ldx #>addr
lda #<addr
jsr target
.endmacro
.macro yxa_jump target, yparam, addr
ldy #yparam
ldx #>addr
lda #<addr
jmp target
.endmacro
2018-02-10 00:00:42 -08:00
2018-03-04 21:36:00 -08:00
;;; ============================================================
2018-02-05 19:01:17 -08:00
;;; 16-bit pseudo-ops
2018-02-06 15:36:39 -08:00
;;; Load A,X
;;; ldax #$1234 ; immediate
;;; ldax $1234 ; absolute
2018-02-05 19:01:17 -08:00
.macro ldax arg
2018-03-05 08:41:22 -08:00
_op_lo lda, {arg}
_op_hi ldx, {arg}
2018-02-05 19:01:17 -08:00
.endmacro
2018-04-15 15:41:35 -07:00
;;; Load A,Y
;;; lday #$1234 ; immediate
;;; lday $1234 ; absolute
.macro lday arg
_op_lo lda, {arg}
_op_hi ldy, {arg}
.endmacro
2018-02-06 18:42:00 -08:00
;;; Load X,Y
;;; ldxy #$1234 ; immediate
;;; ldxy $1234 ; absolute
.macro ldxy arg
2018-03-05 08:41:22 -08:00
_op_lo ldx, {arg}
_op_hi ldy, {arg}
2018-02-06 18:42:00 -08:00
.endmacro
2018-02-06 15:36:39 -08:00
;;; Store A,X
;;; stax $1234 ; absolute
2018-02-05 19:01:17 -08:00
.macro stax arg
sta arg
stx arg+1
.endmacro
;;; Core for add16/sub16
.macro _addsub16 op, opc, arg1, arg2, arg3, arg4, arg5, arg6
.if _is_register {arg2} && _is_register {arg4} && _is_register {arg6}
;; xxx16 $1111,x, $2222,x, $3333,x
lda arg1,arg2
opc
op arg3,arg4
sta arg5,arg6
lda arg1+1,arg2
op arg3+1,arg4
sta arg5+1,arg6
.elseif _is_register {arg2} && _is_register {arg4}
;; xxx16 $1111,x, $2222,x, $3333
lda arg1,arg2
opc
op arg3,arg4
sta arg5
lda arg1+1,arg2
op arg3+1,arg4
sta arg5+1
.elseif _is_register {arg2} && _is_register {arg5}
;; xxx16 $1111,x, $2222, $3333,x
;; xxx16 $1111,x, #$2222, $3333,x
lda arg1,arg2
opc
_op_lo op, {arg3}
sta arg4,arg5
lda arg1+1,arg2
_op_hi op, {arg3}
sta arg4+1,arg5
.elseif _is_register {arg2}
;; xxx16 $1111,x, $2222, $3333
;; xxx16 $1111,x, #$2222, $3333
lda arg1,arg2
opc
_op_lo op, {arg3}
sta arg4
lda arg1+1,arg2
_op_hi op, {arg3}
sta arg4+1
.elseif _is_register {arg3}
;; xxx16 $1111, $2222,x $3333
;; xxx16 #$1111, $2222,x $3333
_op_lo lda, {arg1}
opc
op arg2,arg3
sta arg4
_op_hi lda, {arg1}
op arg2+1,arg3
sta arg4+1
.elseif _is_register {arg4}
;; xxx16 $1111, $2222, $3333,x
;; xxx16 #$1111, $2222, $3333,x
;; xxx16 $1111, #$2222, $3333,x
;; xxx16 #$1111, #$2222, $3333,x
_op_lo lda, {arg1}
opc
_op_lo op, {arg2}
sta arg3,arg4
_op_hi lda, {arg1}
_op_hi op, {arg2}
sta arg3+1,arg4
.else
;; xxx16 $1111, $2222, $3333
;; xxx16 #$1111, $2222, $3333
;; xxx16 $1111, #$2222, $3333
;; xxx16 #$1111, #$2222, $3333
_op_lo lda, {arg1}
opc
_op_lo op, {arg2}
sta arg3
_op_hi lda, {arg1}
_op_hi op, {arg2}
sta arg3+1
.endif
.endmacro
2018-03-05 08:41:22 -08:00
;;; Core for add16/sub16, with leading carry operation
.macro _addsub16lc op, opc, arg1, arg2, arg3, arg4, arg5, arg6
opc
2018-03-05 08:41:22 -08:00
.if _is_register {arg2} && _is_register {arg4} && _is_register {arg6}
;; xxx16 $1111,x, $2222,x, $3333,x
2018-02-13 21:52:36 -08:00
lda arg1,arg2
op arg3,arg4
2018-02-13 21:52:36 -08:00
sta arg5,arg6
lda arg1+1,arg2
op arg3+1,arg4
2018-02-13 21:52:36 -08:00
sta arg5+1,arg6
.elseif _is_register {arg2} && _is_register {arg4}
;; xxx16 $1111,x, $2222,x, $3333
lda arg1,arg2
op arg3,arg4
sta arg5
lda arg1+1,arg2
op arg3+1,arg4
sta arg5+1
2018-03-05 08:41:22 -08:00
.elseif _is_register {arg2} && _is_register {arg5}
;; xxx16 $1111,x, $2222, $3333,x
;; xxx16 $1111,x, #$2222, $3333,x
2018-02-13 21:52:36 -08:00
lda arg1,arg2
_op_lo op, {arg3}
2018-02-13 21:52:36 -08:00
sta arg4,arg5
lda arg1+1,arg2
_op_hi op, {arg3}
2018-02-13 21:52:36 -08:00
sta arg4+1,arg5
2018-03-05 08:41:22 -08:00
.elseif _is_register {arg2}
;; xxx16 $1111,x, $2222, $3333
;; xxx16 $1111,x, #$2222, $3333
2018-02-09 23:33:54 -08:00
lda arg1,arg2
_op_lo op, {arg3}
2018-02-09 23:33:54 -08:00
sta arg4
lda arg1+1,arg2
_op_hi op, {arg3}
2018-02-09 23:33:54 -08:00
sta arg4+1
2018-03-05 08:41:22 -08:00
.elseif _is_register {arg3}
;; xxx16 $1111, $2222,x $3333
;; xxx16 #$1111, $2222,x $3333
2018-03-05 08:41:22 -08:00
_op_lo lda, {arg1}
op arg2,arg3
2018-02-09 23:33:54 -08:00
sta arg4
2018-03-05 08:41:22 -08:00
_op_hi lda, {arg1}
op arg2+1,arg3
2018-02-09 23:33:54 -08:00
sta arg4+1
2018-03-05 08:41:22 -08:00
.elseif _is_register {arg4}
;; xxx16 $1111, $2222, $3333,x
;; xxx16 #$1111, $2222, $3333,x
;; xxx16 $1111, #$2222, $3333,x
;; xxx16 #$1111, #$2222, $3333,x
2018-03-05 08:41:22 -08:00
_op_lo lda, {arg1}
_op_lo op, {arg2}
2018-02-09 23:33:54 -08:00
sta arg3,arg4
2018-03-05 08:41:22 -08:00
_op_hi lda, {arg1}
_op_hi op, {arg2}
2018-02-09 23:33:54 -08:00
sta arg3+1,arg4
2018-02-05 19:01:17 -08:00
.else
;; xxx16 $1111, $2222, $3333
;; xxx16 #$1111, $2222, $3333
;; xxx16 $1111, #$2222, $3333
;; xxx16 #$1111, #$2222, $3333
2018-03-05 08:41:22 -08:00
_op_lo lda, {arg1}
_op_lo op, {arg2}
2018-02-09 21:24:35 -08:00
sta arg3
2018-03-05 08:41:22 -08:00
_op_hi lda, {arg1}
_op_hi op, {arg2}
2018-02-09 21:24:35 -08:00
sta arg3+1
2018-02-05 19:01:17 -08:00
.endif
.endmacro
;;; Add arg1 to arg2, store to arg3
;;; add16 $1111, $2222, $3333 ; absolute, absolute, absolute
;;; add16 $1111, #$2222, $3333 ; absolute, immediate, absolute
;;; add16 $1111,x, $2222, $3333 ; indexed, absolute, absolute
;;; add16 $1111, $2222,x, $3333 ; absolute, indexed, absolute
;;; add16 $1111, $2222, $3333,x ; absolute, absolute, indexed
;;; add16 $1111,x, $2222, $3333,x ; indexed, absolute, indexed
;;; add16 $1111,x, $2222,x, $3333,x ; indexed, indexed, indexed
.macro add16 arg1, arg2, arg3, arg4, arg5, arg6
_addsub16 adc, clc, arg1, arg2, arg3, arg4, arg5, arg6
.endmacro
;;; (as above, but clc precedes first lda)
.macro add16lc arg1, arg2, arg3, arg4, arg5, arg6
_addsub16lc adc, clc, arg1, arg2, arg3, arg4, arg5, arg6
.endmacro
2018-03-05 08:41:22 -08:00
2018-02-05 19:01:17 -08:00
;;; Add arg1 (absolute) to arg2 (8-bit absolute), store to arg3
2018-02-06 15:36:39 -08:00
;;; add16_8 $1111, #$22, $3333 ; absolute, immediate, absolute
;;; add16_8 $1111, $22, $3333 ; absolute, absolute, absolute
2018-02-09 23:33:54 -08:00
.macro add16_8 arg1, arg2, arg3
2018-03-05 08:41:22 -08:00
_op_lo lda, {arg1}
2018-02-05 19:01:17 -08:00
clc
2018-02-09 21:24:35 -08:00
adc arg2
sta arg3
2018-03-05 08:41:22 -08:00
_op_hi lda, {arg1}
2018-02-05 19:01:17 -08:00
adc #0
2018-02-09 21:24:35 -08:00
sta arg3+1
2018-02-05 19:01:17 -08:00
.endmacro
2018-03-04 21:52:38 -08:00
;;; Add A,Z to arg1 (immediate or absolute), store to arg2
;;; addax #$1111, $3333 ; immediate, absolute
;;; addax $1111, $3333 ; absolute, absolute
.macro addax arg1, arg2
clc
2018-03-05 08:41:22 -08:00
_op_lo adc, {arg1}
2018-03-04 21:52:38 -08:00
sta arg2
txa
2018-03-05 08:41:22 -08:00
_op_hi adc, {arg1}
2018-03-04 21:52:38 -08:00
sta arg2+1
.endmacro
2018-02-06 15:36:39 -08:00
;;; Subtract arg2 from arg1, store to arg3
2018-03-05 08:41:22 -08:00
;;; sub16 #$1111, #$2222, $3333 ; immediate, immediate, absolute
2018-02-09 23:33:54 -08:00
;;; sub16 #$1111, $2222, $3333 ; immediate, absolute, absolute
2018-02-06 15:36:39 -08:00
;;; sub16 $1111, #$2222, $3333 ; absolute, immediate, absolute
;;; sub16 $1111, $2222, $3333 ; absolute, absolute, absolute
2018-02-09 23:33:54 -08:00
;;; sub16 $1111, $2222,x, $3333 ; absolute, indexed, absolute
;;; sub16 $1111, $2222, $3333,x ; absolute, absolute, indexed
2018-02-13 21:52:36 -08:00
;;; sub16 $1111,x, $2222,x, $3333 ; indexed, indexed, absolute
;;; sub16 $1111,x, $2222, $3333,x ; indexed, absolute, indexed
2018-03-25 20:18:43 -07:00
;;; sub16 $1111,x, $2222,x $3333,x ; indexed, indexed, indexed
.macro sub16 arg1, arg2, arg3, arg4, arg5, arg6
_addsub16 sbc, sec, arg1, arg2, arg3, arg4, arg5, arg6
.endmacro
;;; (as above, but sec precedes first lda)
.macro sub16lc arg1, arg2, arg3, arg4, arg5, arg6
_addsub16lc sbc, sec, arg1, arg2, arg3, arg4, arg5, arg6
2018-02-05 19:01:17 -08:00
.endmacro
2018-02-06 09:31:58 -08:00
2018-02-06 15:36:39 -08:00
;;; Subtract arg2 from arg1, store to arg3
2018-03-05 08:41:22 -08:00
;;; sub16_8 #$1111, #$22, $3333 ; immediate, immediate, absolute
;;; sub16_8 #$1111, $22, $3333 ; immediate, absolute, absolute
2018-02-06 15:36:39 -08:00
;;; sub16_8 $1111, #$22, $3333 ; absolute, immediate, absolute
;;; sub16_8 $1111, $22, $3333 ; absolute, absolute, absolute
2018-02-09 21:24:35 -08:00
.macro sub16_8 arg1, arg2, arg3
2018-03-05 08:41:22 -08:00
_op_lo lda, {arg1}
2018-02-06 15:36:39 -08:00
sec
2018-02-09 21:24:35 -08:00
sbc arg2
sta arg3
2018-03-05 08:41:22 -08:00
_op_hi lda, {arg1}
2018-02-06 15:36:39 -08:00
sbc #0
2018-02-09 21:24:35 -08:00
sta arg3+1
2018-02-06 15:36:39 -08:00
.endmacro
2018-02-06 09:31:58 -08:00
;;; Copy 16-bit value
2018-02-06 15:36:39 -08:00
;;; copy16 #$1111, $2222 ; immediate, absolute
;;; copy16 $1111, $2222 ; absolute, absolute
;;; copy16 $1111,x, $2222 ; indirect load, absolute store
;;; copy16 $1111, $2222,x ; absolute load, indirect store
2018-03-25 20:18:43 -07:00
;;; copy16 $1111,x $2222,x ; indirect load, indirect store
2018-03-05 08:41:22 -08:00
;;; copy16 #$1111, $2222,x ; immediate load, indirect store
2018-03-25 20:18:43 -07:00
.macro copy16 arg1, arg2, arg3, arg4
.if _is_register {arg2} && _is_register {arg4}
;; indexed load/indexed store
lda arg1,arg2
sta arg3,arg4
lda arg1+1,arg2
sta arg3+1,arg4
.elseif _is_register {arg2}
2018-02-06 09:31:58 -08:00
;; indexed load variant (arg2 is x or y)
lda arg1,arg2
sta arg3
lda arg1+1,arg2
sta arg3+1
2018-03-05 08:41:22 -08:00
.elseif _is_register {arg3}
2018-02-06 09:31:58 -08:00
;; indexed store variant (arg3 is x or y)
2018-03-05 08:41:22 -08:00
_op_lo lda, {arg1}
2018-02-06 09:31:58 -08:00
sta arg2,arg3
2018-03-05 08:41:22 -08:00
_op_hi lda, {arg1}
2018-02-06 09:31:58 -08:00
sta arg2+1,arg3
.else
2018-03-05 08:41:22 -08:00
_op_lo lda, {arg1}
2018-02-06 09:31:58 -08:00
sta arg2
2018-03-05 08:41:22 -08:00
_op_hi lda, {arg1}
2018-02-06 09:31:58 -08:00
sta arg2+1
.endif
.endmacro
2018-02-06 10:06:44 -08:00
2018-02-06 15:36:39 -08:00
;;; Compare 16-bit values
2018-03-05 08:41:22 -08:00
;;; cmp16 #$1111, #$2222 ; immediate, immediate (silly, but supported)
2018-02-06 15:36:39 -08:00
;;; cmp16 #$1111, $2222 ; immediate, absolute
;;; cmp16 $1111, #$2222 ; absolute, immediate
;;; cmp16 $1111, $2222 ; absolute, absolute
;;; cmp16 $1111,x, $2222 ; indirect, absolute
;;; cmp16 $1111, $2222,x ; absolute, indirect
2018-02-06 10:06:44 -08:00
.macro cmp16 arg1, arg2, arg3
2018-03-05 08:41:22 -08:00
.if _is_register {arg2}
2018-02-06 10:06:44 -08:00
;; indexed variant (arg2 is x or y)
lda arg1,arg2
cmp arg3
lda arg1+1,arg2
sbc arg3+1
2018-03-05 08:41:22 -08:00
.elseif _is_register {arg3}
2018-02-06 10:06:44 -08:00
;; indexed variant (arg3 is x or y)
lda arg1
cmp arg2,arg3
lda arg1+1
sbc arg2+1,arg3
.else
2018-03-05 08:41:22 -08:00
_op_lo lda, {arg1}
_op_lo cmp, {arg2}
_op_hi lda, {arg1}
_op_hi sbc, {arg2}
2018-02-06 10:06:44 -08:00
.endif
.endmacro
2018-02-09 22:10:11 -08:00
;;; Shift 16-bit values
;;; lsr16 $1111 ; absolute
.macro lsr16 arg1
lsr arg1+1
ror arg1
.endmacro
2018-02-25 16:06:17 -08:00
;;; asl16 $1111 ; absolute
.macro asl16 arg1
asl arg1
rol arg1+1
.endmacro
2018-02-25 16:06:17 -08:00
;;; Increment 16-bit value
.macro inc16 arg
.local skip
inc arg
bne skip
inc arg+1
skip:
.endmacro
;;; Decrement 16-bit value
.macro dec16 arg
.local skip
lda arg
bne skip
dec arg+1
skip: dec arg
.endmacro
;;; Helper macros to set up a scoped block of parameters at a pre-determined
;;; address.
;;;
;;; Note: to use this macro, your cfg must have a BSS segment:
;;; (BSS: load = BSS, type = bss)
;;;
;;; Example:
;;; .proc my_function
;;; PARAM_BLOCK params, $82
;;; arg1: .res 1
;;; arg2: .res 2
;;; arg3: .res 2
;;; END_PARAM_BLOCK
;;;
;;; lda params::arg1 ; equiv. to lda $82
;;; lda params::arg2 ; equiv. to lda $83
;;; lda params::arg3 ; equiv. to lda $85
;;;
.macro PARAM_BLOCK name, addr
2018-04-02 10:00:43 -05:00
name := addr
.scope name
saved_org := *
.pushseg
.bss
.org addr
start := *
.endmacro
.macro END_PARAM_BLOCK
size := * - start
.popseg
.org saved_org
.endscope
.endmacro