mirror of
https://github.com/freewilll/apple2-go.git
synced 2025-01-17 22:32:28 +00:00
1375 lines
72 KiB
Plaintext
1375 lines
72 KiB
Plaintext
AS65 Assembler for R6502 [1.42]. Copyright 1994-2007, Frank A. Kingswood Page 1
|
||
---------------------------------------------------- 6502_interrupt_test.a65 -----------------------------------------------------
|
||
|
||
1025 lines read, no errors in pass 1.
|
||
;
|
||
; 6 5 0 2 I N T E R R U P T T E S T
|
||
;
|
||
; Copyright (C) 2013 Klaus Dormann
|
||
;
|
||
; This program is free software: you can redistribute it and/or modify
|
||
; it under the terms of the GNU General Public License as published by
|
||
; the Free Software Foundation, either version 3 of the License, or
|
||
; (at your option) any later version.
|
||
;
|
||
; This program is distributed in the hope that it will be useful,
|
||
; but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
; GNU General Public License for more details.
|
||
;
|
||
; You should have received a copy of the GNU General Public License
|
||
; along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||
|
||
|
||
; This program is designed to test IRQ and NMI of a 6502 emulator. It requires
|
||
; an internal or external feedback register to the IRQ & NMI inputs
|
||
;
|
||
; version 15-aug-2014
|
||
; contact info at http://2m5.de or email K@2m5.de
|
||
;
|
||
; assembled with AS65 from http://www.kingswood-consulting.co.uk/assemblers/
|
||
; command line switches: -l -m -s2 -w -h0
|
||
; | | | | no page headers in listing
|
||
; | | | wide listing (133 char/col)
|
||
; | | write intel hex file instead of binary
|
||
; | expand macros in listing
|
||
; generate pass2 listing
|
||
;
|
||
; No IO - should be run from a monitor with access to registers.
|
||
; To run load intel hex image with a load command, than alter PC to 400 hex and
|
||
; enter a go command.
|
||
; Loop on program counter determines error or successful completion of test.
|
||
; Check listing for relevant traps (jump/branch *).
|
||
;
|
||
; Debugging hints:
|
||
; Most of the code is written sequentially. if you hit a trap, check the
|
||
; immediately preceeding code for the instruction to be tested. Results are
|
||
; tested first, flags are checked second by pushing them onto the stack and
|
||
; pulling them to the accumulator after the result was checked. The "real"
|
||
; flags are no longer valid for the tested instruction at this time!
|
||
; If the tested instruction was indexed, the relevant index (X or Y) must
|
||
; also be checked. Opposed to the flags, X and Y registers are still valid.
|
||
;
|
||
; versions:
|
||
; 19-jul-2013 1st version distributed for testing
|
||
; 16-aug-2013 added error report to standard output option
|
||
; 15-aug-2014 added filter to feedback (bit 7 will cause diag stop in emu)
|
||
|
||
|
||
; C O N F I G U R A T I O N
|
||
;
|
||
;ROM_vectors MUST be writable & the I_flag MUST be alterable
|
||
|
||
;load_data_direct (0=move from code segment, 1=load directly)
|
||
;loading directly is preferred but may not be supported by your platform
|
||
;0 produces only consecutive object code, 1 is not suitable for a binary image
|
||
0001 = load_data_direct = 1
|
||
|
||
;NMI & IRQ are tested with a feedback register
|
||
;emulators diag register - set i_drive = 0 for a latch (74HC573)
|
||
bffc = I_port = $bffc ;feedback port address
|
||
0000 = I_ddr = 0 ;feedback DDR address, 0 = no DDR
|
||
0001 = I_drive = 1 ;0 = totem pole, 1 = open collector
|
||
0000 = IRQ_bit = 0 ;bit number of feedback to IRQ
|
||
0001 = NMI_bit = 1 ;bit number of feedback to NMI, -1 if not available
|
||
007f = I_filter = $7f ;filtering bit 7 = diag stop
|
||
|
||
;typical IO chip port B - set i_drive = 0 to avoid pullup resistors
|
||
;I_port = $bfb2 ;feedback port address
|
||
;I_ddr = $bfb3 ;feedback DDR address, 0 = no DDR
|
||
;I_drive = 1 ;0 = totem pole, 1 = open collector
|
||
;IRQ_bit = 0 ;bit number of feedback to IRQ
|
||
;NMI_bit = 1 ;bit number of feedback to NMI, -1 if not available
|
||
;I_filter = $ff ;no bits filtered
|
||
|
||
;decimal mode flag during IRQ, NMI & BRK
|
||
0000 = D_clear = 0 ;0 = not cleared (NMOS), 1 = cleared (CMOS)
|
||
|
||
;configure memory - try to stay away from memory used by the system
|
||
;zero_page memory start address, 6 consecutive Bytes required
|
||
0000 = zero_page = $0
|
||
|
||
;data_segment memory start address, 4 consecutive Bytes required
|
||
0200 = data_segment = $200
|
||
|
||
;code_segment memory start address
|
||
0800 = code_segment = $800
|
||
|
||
;report errors through I/O channel (0=use standard self trap loops, 1=include
|
||
;report.i65 as I/O channel)
|
||
0000 = report = 0
|
||
|
||
noopt ;do not take shortcuts
|
||
|
||
;macros for error & success traps to allow user modification
|
||
;example:
|
||
;trap macro
|
||
; jsr my_error_handler
|
||
; endm
|
||
;trap_eq macro
|
||
; bne skip\?
|
||
; trap ;failed equal (zero)
|
||
;skip\?
|
||
; endm
|
||
;
|
||
; my_error_handler should pop the calling address from the stack and report it.
|
||
; putting larger portions of code (more than 3 bytes) inside the trap macro
|
||
; may lead to branch range problems for some tests.
|
||
if report = 0
|
||
trap macro
|
||
jmp * ;failed anyway
|
||
endm
|
||
trap_eq macro
|
||
beq * ;failed equal (zero)
|
||
endm
|
||
trap_ne macro
|
||
bne * ;failed not equal (non zero)
|
||
endm
|
||
; please observe that during the test the stack gets invalidated
|
||
; therefore a RTS inside the success macro is not possible
|
||
success macro
|
||
jmp * ;test passed, no errors
|
||
endm
|
||
endif
|
||
if report = 1
|
||
trap macro
|
||
jsr report_error
|
||
endm
|
||
trap_eq macro
|
||
bne skip\?
|
||
trap ;failed equal (zero)
|
||
skip\?
|
||
endm
|
||
trap_ne macro
|
||
beq skip\?
|
||
trap ;failed not equal (non zero)
|
||
skip\?
|
||
endm
|
||
; please observe that during the test the stack gets invalidated
|
||
; therefore a RTS inside the success macro is not possible
|
||
success macro
|
||
jsr report_success
|
||
endm
|
||
endif
|
||
|
||
|
||
0001 = carry equ %00000001 ;flag bits in status
|
||
0002 = zero equ %00000010
|
||
0004 = intdis equ %00000100
|
||
0008 = decmode equ %00001000
|
||
0010 = break equ %00010000
|
||
0020 = reserv equ %00100000
|
||
0040 = overfl equ %01000000
|
||
0080 = minus equ %10000000
|
||
|
||
0001 = fc equ carry
|
||
0002 = fz equ zero
|
||
0003 = fzc equ carry+zero
|
||
0040 = fv equ overfl
|
||
0042 = fvz equ overfl+zero
|
||
0080 = fn equ minus
|
||
0081 = fnc equ minus+carry
|
||
0082 = fnz equ minus+zero
|
||
0083 = fnzc equ minus+zero+carry
|
||
00c0 = fnv equ minus+overfl
|
||
|
||
0030 = fao equ break+reserv ;bits always on after PHP, BRK
|
||
0034 = fai equ fao+intdis ;+ forced interrupt disable
|
||
00ff = m8 equ $ff ;8 bit mask
|
||
00fb = m8i equ $ff&~intdis ;8 bit mask - interrupt disable
|
||
|
||
;macros to set status
|
||
push_stat macro ;setting flags in the processor status register
|
||
lda #\1
|
||
pha ;use stack to load status
|
||
endm
|
||
|
||
set_stat macro ;setting flags in the processor status register
|
||
lda #\1
|
||
pha ;use stack to load status
|
||
plp
|
||
endm
|
||
|
||
if load_data_direct = 1
|
||
data
|
||
else
|
||
bss ;uninitialized segment, copy of data at end of code!
|
||
endif
|
||
0000 = org zero_page
|
||
;BRK, IRQ, NMI test interrupt save
|
||
0000 : zpt
|
||
0000 : 00 irq_a ds 1 ;a register
|
||
0001 : 00 irq_x ds 1 ;x register
|
||
0002 : 00 irq_f ds 1 ;flags
|
||
0003 : 00 nmi_a ds 1 ;a register
|
||
0004 : 00 nmi_x ds 1 ;x register
|
||
0005 : 00 nmi_f ds 1 ;flags
|
||
0006 : zp_bss
|
||
|
||
;fixed stack locations
|
||
01fe = lst_f equ $1fe ;last flags before interrupt
|
||
01ff = lst_a equ $1ff ;last accumulator before interrupt
|
||
|
||
0200 = org data_segment
|
||
;concurrent NMI, IRQ & BRK test result
|
||
0200 : 00 nmi_count ds 1 ;lowest number handled first, $ff = never
|
||
0201 : 00 irq_count ds 1 ;separation-1 = instructions between interrupts
|
||
0202 : 00 brk_count ds 1
|
||
;expected interrupt mask
|
||
0203 : 00 I_src ds 1 ;bit: 0=BRK, 1=IRQ, 2=NMI
|
||
0204 : data_bss
|
||
|
||
code
|
||
0800 = org code_segment
|
||
0800 : d8 start cld
|
||
0801 : a900 lda #0 ;clear expected interrupts for 2nd run
|
||
0803 : 8d0302 sta I_src
|
||
0806 : a2ff ldx #$ff
|
||
0808 : 9a txs
|
||
|
||
;initialize I/O for report channel
|
||
if report = 1
|
||
jsr report_init
|
||
endif
|
||
|
||
; load system vectors
|
||
if load_data_direct != 1
|
||
ldx #5
|
||
ld_vect lda vec_init,x
|
||
sta vec_bss,x
|
||
dex
|
||
bpl ld_vect
|
||
endif
|
||
|
||
; IRQ & NMI test - requires a feedback register
|
||
if I_drive > 1
|
||
ERROR ;invalid interrupt drive!
|
||
endif
|
||
if NMI_bit < 0
|
||
if I_drive = 0 ;totem pole (push/pull, 0 -> I_port to force interrupt)
|
||
I_set macro ibit ;ibit = interrupt bit
|
||
lda I_port ;turn on interrupt by bit
|
||
and #I_filter-(1<<\1)
|
||
plp ;set flags
|
||
pha ;save to verify
|
||
php
|
||
sta I_port ;interrupt next instruction plus outbound delay
|
||
endm
|
||
I_clr macro ibit ;ibit = interrupt bit
|
||
lda I_port ;turn off interrupt by bit
|
||
and #I_filter
|
||
ora #(1<<ibit)
|
||
sta I_port
|
||
endm
|
||
I_clr IRQ_bit ;turn off IRQ
|
||
if I_ddr != 0 ;with DDR
|
||
lda I_ddr ;set DDR for IRQ to enabled
|
||
and #I_filter
|
||
ora #(1<<IRQ_bit)
|
||
sta I_ddr
|
||
endif
|
||
else ;open collector, 0 -> I_DDR or I_port to force interrupt
|
||
if I_ddr != 0 ;with DDR
|
||
I_set macro ibit ;ibit = interrupt bit
|
||
lda I_ddr ;turn on interrupt by bit
|
||
and #I_filter
|
||
ora #(1<<\1)
|
||
plp ;set flags
|
||
pha ;save to verify
|
||
php
|
||
sta I_ddr ;interrupt next instruction plus outbound delay
|
||
endm
|
||
I_clr macro ibit ;ibit = interrupt bit
|
||
lda I_ddr ;turn off interrupt by bit
|
||
and #I_filter-(1<<ibit)
|
||
sta I_ddr
|
||
endm
|
||
I_clr IRQ_bit ;turn off IRQ
|
||
lda I_port ;precharge IRQ
|
||
and #I_filter-(1<<IRQ_bit)
|
||
sta I_port
|
||
else ;no DDR
|
||
I_set macro ibit ;ibit = interrupt bit
|
||
lda I_port ;turn on interrupt by bit
|
||
and #I_filter
|
||
ora #(1<<\1)
|
||
plp ;set flags
|
||
pha ;save to verify
|
||
php
|
||
sta I_port ;interrupt next instruction plus outbound delay
|
||
endm
|
||
I_clr macro ibit ;ibit = interrupt bit
|
||
lda I_port ;turn off interrupt by bit
|
||
and #I_filter-(1<<ibit)
|
||
sta I_port
|
||
endm
|
||
I_clr IRQ_bit ;turn off IRQ
|
||
endif
|
||
endif
|
||
else
|
||
if I_drive = 0 ;totem pole (push/pull, 0 -> I_port to force interrupt)
|
||
I_set macro ibit ;ibit = interrupt bit
|
||
lda I_port ;turn on interrupt by bit
|
||
if ibit > 7 ;set both NMI & IRQ
|
||
and #I_filter-(1<<IRQ_bit|1<<NMI_bit)
|
||
else
|
||
and #I_filter-(1<<\1)
|
||
endif
|
||
plp ;set flags
|
||
pha ;save to verify
|
||
php
|
||
sta I_port ;interrupt next instruction plus outbound delay
|
||
endm
|
||
I_clr macro ibit ;ibit = interrupt bit
|
||
lda I_port ;turn off interrupt by bit
|
||
and #I_filter
|
||
ora #(1<<ibit)
|
||
sta I_port
|
||
endm
|
||
I_clr IRQ_bit ;turn off IRQ & NMI
|
||
I_clr NMI_bit
|
||
if I_ddr != 0 ;with DDR
|
||
lda I_ddr ;set DDR for IRQ & NMI to enabled
|
||
and #I_filter
|
||
ora #(1<<IRQ_bit|1<<NMI_bit)
|
||
sta I_ddr
|
||
endif
|
||
else ;open collector, 0 -> I_DDR or I_port to force interrupt
|
||
if I_ddr != 0 ;with DDR
|
||
I_set macro ibit ;ibit = interrupt bit
|
||
lda I_ddr ;turn on interrupt by bit
|
||
and #I_filter
|
||
if ibit > 7 ;set both NMI & IRQ
|
||
ora #(1<<IRQ_bit|1<<NMI_bit)
|
||
else
|
||
ora #(1<<\1)
|
||
endif
|
||
plp ;set flags
|
||
pha ;save to verify
|
||
php
|
||
sta I_ddr ;interrupt next instruction plus outbound delay
|
||
endm
|
||
I_clr macro ibit ;ibit = interrupt bit
|
||
lda I_ddr ;turn off interrupt by bit
|
||
and #I_filter-(1<<ibit)
|
||
sta I_ddr
|
||
endm
|
||
I_clr IRQ_bit ;turn off IRQ & NMI
|
||
I_clr NMI_bit
|
||
lda I_port ;precharge IRQ & NMI
|
||
and #I_filter-(1<<IRQ_bit|1<<NMI_bit)
|
||
sta I_port
|
||
else ;no DDR
|
||
I_set macro ibit ;ibit = interrupt bit
|
||
lda I_port ;turn on interrupt by bit
|
||
and #I_filter
|
||
if ibit > 7 ;set both NMI & IRQ
|
||
ora #(1<<IRQ_bit|1<<NMI_bit)
|
||
else
|
||
ora #(1<<\1)
|
||
endif
|
||
plp ;set flags
|
||
pha ;save to verify
|
||
php
|
||
sta I_port ;interrupt next instruction plus outbound delay
|
||
endm
|
||
I_clr macro ibit ;ibit = interrupt bit
|
||
lda I_port ;turn off interrupt by bit
|
||
and #I_filter-(1<<ibit)
|
||
sta I_port
|
||
endm
|
||
I_clr IRQ_bit ;turn off IRQ & NMI
|
||
0809 : adfcbf > lda I_port ;turn off interrupt by bit
|
||
080c : 297e > and #I_filter-(1<<IRQ_bit )
|
||
080e : 8dfcbf > sta I_port
|
||
|
||
I_clr NMI_bit
|
||
0811 : adfcbf > lda I_port ;turn off interrupt by bit
|
||
0814 : 297d > and #I_filter-(1<<NMI_bit)
|
||
0816 : 8dfcbf > sta I_port
|
||
|
||
endif
|
||
endif
|
||
endif
|
||
|
||
; IRQ integrity test
|
||
; test for clear flags seen in IRQ vector
|
||
0819 : a902 lda #2 ;set expected interrupt source IRQ
|
||
081b : 8d0302 sta I_src
|
||
push_stat 0
|
||
081e : a900 > lda #0
|
||
0820 : 48 > pha ;use stack to load status
|
||
|
||
I_set IRQ_bit
|
||
0821 : adfcbf > lda I_port ;turn on interrupt by bit
|
||
0824 : 297f > and #I_filter
|
||
> if IRQ_bit > 7 ;set both NMI & IRQ
|
||
> ora #(1<<IRQ_bit|1<<NMI_bit)
|
||
> else
|
||
0826 : 0901 > ora #(1<<IRQ_bit)
|
||
> endif
|
||
0828 : 28 > plp ;set flags
|
||
0829 : 48 > pha ;save to verify
|
||
082a : 08 > php
|
||
082b : 8dfcbf > sta I_port ;interrupt next instruction plus outbound delay
|
||
|
||
082e : ea nop ;allow 6 cycles for interrupt to trip
|
||
082f : ea nop
|
||
0830 : ea nop
|
||
0831 : ad0302 lda I_src
|
||
trap_ne ;IRQ timeout
|
||
0834 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
0836 : ba tsx
|
||
0837 : e0fd cpx #$ff-2 ;original accu & flags remain on stack
|
||
trap_ne ;returned SP
|
||
0839 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
083b : a502 lda irq_f ;flags seen in IRQ vector
|
||
if D_clear = 1
|
||
and #decmode
|
||
trap_ne ;D-flag not cleared
|
||
lda irq_f
|
||
eor lst_f ;turn off unchanged bits
|
||
and #m8-fai-decmode ;mask untested other flags
|
||
trap_ne ;other flags (N,V,Z,C) changed
|
||
else
|
||
083d : 4dfe01 eor lst_f ;turn off unchanged bits
|
||
0840 : 29cb and #m8-fai ;mask untested other flags
|
||
trap_ne ;other flags (N,V,Z,C,D) changed
|
||
0842 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
endif
|
||
0844 : a2ff ldx #$ff ;reset stack pointer
|
||
0846 : 9a txs
|
||
; test all other registers
|
||
0847 : a249 ldx #'I'
|
||
0849 : a052 ldy #'R'
|
||
084b : a902 lda #2 ;set expected interrupt source IRQ
|
||
084d : 8d0302 sta I_src
|
||
push_stat 0
|
||
0850 : a900 > lda #0
|
||
0852 : 48 > pha ;use stack to load status
|
||
|
||
I_set IRQ_bit
|
||
0853 : adfcbf > lda I_port ;turn on interrupt by bit
|
||
0856 : 297f > and #I_filter
|
||
> if IRQ_bit > 7 ;set both NMI & IRQ
|
||
> ora #(1<<IRQ_bit|1<<NMI_bit)
|
||
> else
|
||
0858 : 0901 > ora #(1<<IRQ_bit)
|
||
> endif
|
||
085a : 28 > plp ;set flags
|
||
085b : 48 > pha ;save to verify
|
||
085c : 08 > php
|
||
085d : 8dfcbf > sta I_port ;interrupt next instruction plus outbound delay
|
||
|
||
0860 : 88 dey ;Y count will fail, if instructions are skipped
|
||
0861 : 88 dey
|
||
0862 : 88 dey
|
||
0863 : 88 dey
|
||
0864 : 08 php ;check processor status later
|
||
0865 : e04a cpx #('I'+1) ;returned registers OK?
|
||
trap_ne ;returned X
|
||
0867 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
0869 : c04b cpy #('R'-7)
|
||
trap_ne ;returned Y
|
||
086b : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
086d : c951 cmp #'Q'
|
||
trap_ne ;returned A
|
||
086f : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
0871 : ba tsx
|
||
0872 : e0fc cpx #$ff-3
|
||
trap_ne ;returned SP
|
||
0874 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
0876 : 68 pla ;flags
|
||
0877 : 4dfe01 eor lst_f
|
||
087a : 297d and #$ff-fnz ;ignore flags changed by dey
|
||
trap_ne ;returned flags
|
||
087c : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
087e : a500 lda irq_a ;accu seen in IRQ vector
|
||
0880 : cdff01 cmp lst_a
|
||
trap_ne ;IRQ A received
|
||
0883 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
0885 : a2ff ldx #$ff ;reset stack pointer
|
||
0887 : 9a txs
|
||
; repeat with reversed registers
|
||
0888 : a2b6 ldx #$ff-'I'
|
||
088a : a0ad ldy #$ff-'R'
|
||
088c : a902 lda #2 ;set expected interrupt source IRQ
|
||
088e : 8d0302 sta I_src
|
||
push_stat $ff-intdis
|
||
0891 : a9fb > lda #$ff-intdis
|
||
0893 : 48 > pha ;use stack to load status
|
||
|
||
I_set IRQ_bit
|
||
0894 : adfcbf > lda I_port ;turn on interrupt by bit
|
||
0897 : 297f > and #I_filter
|
||
> if IRQ_bit > 7 ;set both NMI & IRQ
|
||
> ora #(1<<IRQ_bit|1<<NMI_bit)
|
||
> else
|
||
0899 : 0901 > ora #(1<<IRQ_bit)
|
||
> endif
|
||
089b : 28 > plp ;set flags
|
||
089c : 48 > pha ;save to verify
|
||
089d : 08 > php
|
||
089e : 8dfcbf > sta I_port ;interrupt next instruction plus outbound delay
|
||
|
||
08a1 : 88 dey ;Y count will fail, if instructions are skipped
|
||
08a2 : 88 dey
|
||
08a3 : 88 dey
|
||
08a4 : 88 dey
|
||
08a5 : 08 php ;check processor status later
|
||
08a6 : e0b7 cpx #($ff-'I'+1) ;returned registers OK?
|
||
trap_ne ;returned X
|
||
08a8 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
08aa : c0a6 cpy #($ff-'R'-7)
|
||
trap_ne ;returned Y
|
||
08ac : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
08ae : c951 cmp #'Q'
|
||
trap_ne ;returned A
|
||
08b0 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
08b2 : ba tsx
|
||
08b3 : e0fc cpx #$ff-3
|
||
trap_ne ;returned SP
|
||
08b5 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
08b7 : 68 pla ;flags
|
||
08b8 : 4dfe01 eor lst_f
|
||
08bb : 297d and #$ff-fnz ;ignore flags changed by dey
|
||
trap_ne ;returned flags
|
||
08bd : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
08bf : a500 lda irq_a ;accu seen in IRQ vector
|
||
08c1 : cdff01 cmp lst_a
|
||
trap_ne ;IRQ A received
|
||
08c4 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
08c6 : a2ff ldx #$ff ;reset stack pointer
|
||
08c8 : 9a txs
|
||
; retest for set flags seen in IRQ vector
|
||
08c9 : a902 lda #2 ;set expected interrupt source IRQ
|
||
08cb : 8d0302 sta I_src
|
||
push_stat $ff-intdis
|
||
08ce : a9fb > lda #$ff-intdis
|
||
08d0 : 48 > pha ;use stack to load status
|
||
|
||
I_set IRQ_bit
|
||
08d1 : adfcbf > lda I_port ;turn on interrupt by bit
|
||
08d4 : 297f > and #I_filter
|
||
> if IRQ_bit > 7 ;set both NMI & IRQ
|
||
> ora #(1<<IRQ_bit|1<<NMI_bit)
|
||
> else
|
||
08d6 : 0901 > ora #(1<<IRQ_bit)
|
||
> endif
|
||
08d8 : 28 > plp ;set flags
|
||
08d9 : 48 > pha ;save to verify
|
||
08da : 08 > php
|
||
08db : 8dfcbf > sta I_port ;interrupt next instruction plus outbound delay
|
||
|
||
08de : ea nop ;allow 6 cycles for interrupt to trip
|
||
08df : ea nop
|
||
08e0 : ea nop
|
||
08e1 : ad0302 lda I_src
|
||
trap_ne ;IRQ timeout
|
||
08e4 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
08e6 : ba tsx
|
||
08e7 : e0fd cpx #$ff-2 ;original accu & flags remain on stack
|
||
trap_ne ;returned SP
|
||
08e9 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
08eb : a502 lda irq_f ;flags seen in IRQ vector
|
||
if D_clear = 1
|
||
and #decmode
|
||
trap_ne ;D-flag not cleared
|
||
lda irq_f
|
||
eor lst_f ;turn off unchanged bits
|
||
and #m8-fai-decmode ;mask untested other flags
|
||
trap_ne ;other flags (N,V,Z,C) changed
|
||
else
|
||
08ed : 4dfe01 eor lst_f ;turn off unchanged bits
|
||
08f0 : 29cb and #m8-fai ;mask untested other flags
|
||
trap_ne ;other flags (N,V,Z,C,D) changed
|
||
08f2 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
endif
|
||
08f4 : a2ff ldx #$ff ;reset stack pointer
|
||
08f6 : 9a txs
|
||
|
||
; BRK integrity test
|
||
; test for clear flags seen in IRQ vector
|
||
08f7 : a901 lda #1 ;set expected interrupt source BRK
|
||
08f9 : 8d0302 sta I_src
|
||
set_stat 0
|
||
08fc : a900 > lda #0
|
||
08fe : 48 > pha ;use stack to load status
|
||
08ff : 28 > plp
|
||
|
||
0900 : 48 pha ;save entry registers
|
||
0901 : 08 php
|
||
0902 : 00 brk
|
||
0903 : ea nop ;should not be executed
|
||
0904 : ea nop ;allow 6 cycles for interrupt to trip
|
||
0905 : ea nop
|
||
0906 : ea nop
|
||
0907 : ad0302 lda I_src
|
||
trap_ne ;IRQ timeout
|
||
090a : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
090c : ba tsx
|
||
090d : e0fd cpx #$ff-2 ;original accu & flags remain on stack
|
||
trap_ne ;returned SP
|
||
090f : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
0911 : a502 lda irq_f ;flags seen in IRQ vector
|
||
if D_clear = 1
|
||
and #decmode
|
||
trap_ne ;D-flag not cleared
|
||
lda irq_f
|
||
eor lst_f ;turn off unchanged bits
|
||
and #m8-fai-decmode ;mask untested other flags
|
||
trap_ne ;other flags (N,V,Z,C) changed
|
||
else
|
||
0913 : 4dfe01 eor lst_f ;turn off unchanged bits
|
||
0916 : 29cb and #m8-fai ;mask untested other flags
|
||
trap_ne ;other flags (N,V,Z,C,D) changed
|
||
0918 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
endif
|
||
091a : a2ff ldx #$ff ;reset stack pointer
|
||
091c : 9a txs
|
||
; test all other registers
|
||
091d : a242 ldx #'B'
|
||
091f : a052 ldy #'R'
|
||
0921 : a901 lda #1 ;set expected interrupt source BRK
|
||
0923 : 8d0302 sta I_src
|
||
set_stat 0
|
||
0926 : a900 > lda #0
|
||
0928 : 48 > pha ;use stack to load status
|
||
0929 : 28 > plp
|
||
|
||
092a : 48 pha ;save entry
|
||
092b : 08 php
|
||
092c : 00 brk
|
||
092d : 88 dey ;should not be executed
|
||
092e : 88 dey ;Y count will fail, if return address is wrong
|
||
092f : 88 dey
|
||
0930 : 88 dey
|
||
0931 : 88 dey
|
||
0932 : 08 php ;check processor status later
|
||
0933 : e043 cpx #('B'+1) ;returned registers OK?
|
||
trap_ne ;returned X
|
||
0935 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
0937 : c04b cpy #('R'-7)
|
||
trap_ne ;returned Y
|
||
0939 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
093b : c94b cmp #'K'
|
||
trap_ne ;returned A
|
||
093d : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
093f : ba tsx
|
||
0940 : e0fc cpx #$ff-3
|
||
trap_ne ;returned SP
|
||
0942 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
0944 : 68 pla ;flags
|
||
0945 : 4dfe01 eor lst_f
|
||
0948 : 297d and #$ff-fnz ;ignore flags changed by dey
|
||
trap_ne ;returned flags
|
||
094a : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
094c : a500 lda irq_a ;accu seen in IRQ vector
|
||
094e : cdff01 cmp lst_a
|
||
trap_ne ;IRQ A received
|
||
0951 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
0953 : a2ff ldx #$ff ;reset stack pointer
|
||
0955 : 9a txs
|
||
; repeat with reversed registers
|
||
0956 : a2bd ldx #$ff-'B'
|
||
0958 : a0ad ldy #$ff-'R'
|
||
095a : a901 lda #1 ;set expected interrupt source BRK
|
||
095c : 8d0302 sta I_src
|
||
set_stat $ff
|
||
095f : a9ff > lda #$ff
|
||
0961 : 48 > pha ;use stack to load status
|
||
0962 : 28 > plp
|
||
|
||
0963 : 48 pha ;save entry registers
|
||
0964 : 08 php
|
||
0965 : 00 brk
|
||
0966 : 88 dey ;should not be executed
|
||
0967 : 88 dey ;Y count will fail, if return address is wrong
|
||
0968 : 88 dey
|
||
0969 : 88 dey
|
||
096a : 88 dey
|
||
096b : 08 php ;check processor status later
|
||
096c : e0be cpx #($ff-'B'+1) ;returned registers OK?
|
||
trap_ne ;returned X
|
||
096e : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
0970 : c0a6 cpy #($ff-'R'-7)
|
||
trap_ne ;returned Y
|
||
0972 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
0974 : c94b cmp #'K'
|
||
trap_ne ;returned A
|
||
0976 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
0978 : ba tsx
|
||
0979 : e0fc cpx #$ff-3
|
||
trap_ne ;returned SP
|
||
097b : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
097d : 68 pla ;flags
|
||
097e : 4dfe01 eor lst_f
|
||
0981 : 297d and #$ff-fnz ;ignore flags changed by dey
|
||
trap_ne ;returned flags
|
||
0983 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
0985 : a500 lda irq_a ;accu seen in IRQ vector
|
||
0987 : cdff01 cmp lst_a
|
||
trap_ne ;IRQ A received
|
||
098a : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
098c : a2ff ldx #$ff ;reset stack pointer
|
||
098e : 9a txs
|
||
; retest for set flags seen in IRQ vector
|
||
098f : a901 lda #1 ;set expected interrupt source BRK
|
||
0991 : 8d0302 sta I_src
|
||
set_stat $ff
|
||
0994 : a9ff > lda #$ff
|
||
0996 : 48 > pha ;use stack to load status
|
||
0997 : 28 > plp
|
||
|
||
0998 : 48 pha ;save entry registers
|
||
0999 : 08 php
|
||
099a : 00 brk
|
||
099b : ea nop ;should not be executed
|
||
099c : ea nop ;allow 6 cycles for interrupt to trip
|
||
099d : ea nop
|
||
099e : ea nop
|
||
099f : ad0302 lda I_src
|
||
trap_ne ;IRQ timeout
|
||
09a2 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
09a4 : ba tsx
|
||
09a5 : e0fd cpx #$ff-2 ;original accu & flags remain on stack
|
||
trap_ne ;returned SP
|
||
09a7 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
09a9 : a502 lda irq_f ;flags seen in IRQ vector
|
||
if D_clear = 1
|
||
and #decmode
|
||
trap_ne ;D-flag not cleared
|
||
lda irq_f
|
||
eor lst_f ;turn off unchanged bits
|
||
and #m8-fai-decmode ;mask untested other flags
|
||
trap_ne ;other flags (N,V,Z,C) changed
|
||
else
|
||
09ab : 4dfe01 eor lst_f ;turn off unchanged bits
|
||
09ae : 29cb and #m8-fai ;mask untested other flags
|
||
trap_ne ;other flags (N,V,Z,C,D) changed
|
||
09b0 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
endif
|
||
09b2 : a2ff ldx #$ff ;reset stack pointer
|
||
09b4 : 9a txs
|
||
|
||
if NMI_bit < 0
|
||
; test IRQ with interrupts disabled
|
||
ldx #0
|
||
lda #0
|
||
sta I_src
|
||
push_stat intdis
|
||
I_set IRQ_bit ;IRQ pending
|
||
inx
|
||
inx
|
||
inx
|
||
ldx #0
|
||
lda #2 ;now re-enable IRQ
|
||
sta I_src
|
||
cli
|
||
inx
|
||
inx
|
||
inx
|
||
lda I_src ;test IRQ done?
|
||
trap_ne
|
||
ldx #$ff ;purge stack
|
||
txs
|
||
|
||
ldx #0 ;now overlap IRQ & BRK
|
||
lda #3
|
||
sta I_src
|
||
lda #$ff ;measure timing
|
||
sta nmi_count
|
||
sta irq_count
|
||
sta brk_count
|
||
push_stat 0
|
||
I_set IRQ_bit ;trigger IRQ
|
||
else
|
||
; NMI integrity test
|
||
; test for clear flags seen in NMI vector
|
||
09b5 : a904 lda #4 ;set expected interrupt source NMI
|
||
09b7 : 8d0302 sta I_src
|
||
push_stat 0
|
||
09ba : a900 > lda #0
|
||
09bc : 48 > pha ;use stack to load status
|
||
|
||
I_set NMI_bit
|
||
09bd : adfcbf > lda I_port ;turn on interrupt by bit
|
||
09c0 : 297f > and #I_filter
|
||
> if NMI_bit > 7 ;set both NMI & IRQ
|
||
> ora #(1<<IRQ_bit|1<<NMI_bit)
|
||
> else
|
||
09c2 : 0902 > ora #(1<<NMI_bit)
|
||
> endif
|
||
09c4 : 28 > plp ;set flags
|
||
09c5 : 48 > pha ;save to verify
|
||
09c6 : 08 > php
|
||
09c7 : 8dfcbf > sta I_port ;interrupt next instruction plus outbound delay
|
||
|
||
09ca : ea nop ;allow 6 cycles for interrupt to trip
|
||
09cb : ea nop
|
||
09cc : ea nop
|
||
09cd : ad0302 lda I_src
|
||
trap_ne ;NMI timeout
|
||
09d0 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
09d2 : ba tsx
|
||
09d3 : e0fd cpx #$ff-2 ;original accu & flags remain on stack
|
||
trap_ne ;returned SP
|
||
09d5 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
09d7 : a505 lda nmi_f ;flags seen in NMI vector
|
||
if D_clear = 1
|
||
and #decmode
|
||
trap_ne ;D-flag not cleared
|
||
lda nmi_f
|
||
eor lst_f ;turn off unchanged bits
|
||
and #m8-fai-decmode ;mask untested other flags
|
||
trap_ne ;other flags (N,V,Z,C) changed
|
||
else
|
||
09d9 : 4dfe01 eor lst_f ;turn off unchanged bits
|
||
09dc : 29cb and #m8-fai ;mask untested other flags
|
||
trap_ne ;other flags (N,V,Z,C,D) changed
|
||
09de : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
endif
|
||
09e0 : a2ff ldx #$ff ;reset stack pointer
|
||
09e2 : 9a txs
|
||
; test all other registers
|
||
09e3 : a24e ldx #'N'
|
||
09e5 : a04d ldy #'M'
|
||
09e7 : a904 lda #4 ;set expected interrupt source NMI
|
||
09e9 : 8d0302 sta I_src
|
||
push_stat 0
|
||
09ec : a900 > lda #0
|
||
09ee : 48 > pha ;use stack to load status
|
||
|
||
I_set NMI_bit
|
||
09ef : adfcbf > lda I_port ;turn on interrupt by bit
|
||
09f2 : 297f > and #I_filter
|
||
> if NMI_bit > 7 ;set both NMI & IRQ
|
||
> ora #(1<<IRQ_bit|1<<NMI_bit)
|
||
> else
|
||
09f4 : 0902 > ora #(1<<NMI_bit)
|
||
> endif
|
||
09f6 : 28 > plp ;set flags
|
||
09f7 : 48 > pha ;save to verify
|
||
09f8 : 08 > php
|
||
09f9 : 8dfcbf > sta I_port ;interrupt next instruction plus outbound delay
|
||
|
||
09fc : 88 dey ;Y count will fail, if instructions are skipped
|
||
09fd : 88 dey
|
||
09fe : 88 dey
|
||
09ff : 88 dey
|
||
0a00 : 08 php ;check processor status later
|
||
0a01 : e04f cpx #('N'+1) ;returned registers OK?
|
||
trap_ne ;returned X
|
||
0a03 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
0a05 : c046 cpy #('M'-7)
|
||
trap_ne ;returned Y
|
||
0a07 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
0a09 : c949 cmp #'I'
|
||
trap_ne ;returned A
|
||
0a0b : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
0a0d : ba tsx
|
||
0a0e : e0fc cpx #$ff-3
|
||
trap_ne ;returned SP
|
||
0a10 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
0a12 : 68 pla ;flags
|
||
0a13 : 4dfe01 eor lst_f
|
||
0a16 : 297d and #$ff-fnz ;ignore flags changed by dey
|
||
trap_ne ;returned flags
|
||
0a18 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
0a1a : a503 lda nmi_a ;accu seen in NMI vector
|
||
0a1c : cdff01 cmp lst_a
|
||
trap_ne ;NMI A received
|
||
0a1f : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
0a21 : a2ff ldx #$ff ;reset stack pointer
|
||
0a23 : 9a txs
|
||
; repeat with reversed registers
|
||
0a24 : a2b1 ldx #$ff-'N'
|
||
0a26 : a0b2 ldy #$ff-'M'
|
||
0a28 : a904 lda #4 ;set expected interrupt source NMI
|
||
0a2a : 8d0302 sta I_src
|
||
push_stat $ff-intdis
|
||
0a2d : a9fb > lda #$ff-intdis
|
||
0a2f : 48 > pha ;use stack to load status
|
||
|
||
I_set NMI_bit
|
||
0a30 : adfcbf > lda I_port ;turn on interrupt by bit
|
||
0a33 : 297f > and #I_filter
|
||
> if NMI_bit > 7 ;set both NMI & IRQ
|
||
> ora #(1<<IRQ_bit|1<<NMI_bit)
|
||
> else
|
||
0a35 : 0902 > ora #(1<<NMI_bit)
|
||
> endif
|
||
0a37 : 28 > plp ;set flags
|
||
0a38 : 48 > pha ;save to verify
|
||
0a39 : 08 > php
|
||
0a3a : 8dfcbf > sta I_port ;interrupt next instruction plus outbound delay
|
||
|
||
0a3d : 88 dey ;Y count will fail, if instructions are skipped
|
||
0a3e : 88 dey
|
||
0a3f : 88 dey
|
||
0a40 : 88 dey
|
||
0a41 : 08 php ;check processor status later
|
||
0a42 : e0b2 cpx #($ff-'N'+1) ;returned registers OK?
|
||
trap_ne ;returned X
|
||
0a44 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
0a46 : c0ab cpy #($ff-'M'-7)
|
||
trap_ne ;returned Y
|
||
0a48 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
0a4a : c949 cmp #'I'
|
||
trap_ne ;returned A
|
||
0a4c : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
0a4e : ba tsx
|
||
0a4f : e0fc cpx #$ff-3
|
||
trap_ne ;returned SP
|
||
0a51 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
0a53 : 68 pla ;flags
|
||
0a54 : 4dfe01 eor lst_f
|
||
0a57 : 297d and #$ff-fnz ;ignore flags changed by dey
|
||
trap_ne ;returned flags
|
||
0a59 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
0a5b : a503 lda nmi_a ;accu seen in NMI vector
|
||
0a5d : cdff01 cmp lst_a
|
||
trap_ne ;NMI A received
|
||
0a60 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
0a62 : a2ff ldx #$ff ;reset stack pointer
|
||
0a64 : 9a txs
|
||
; retest for set flags seen in NMI vector
|
||
0a65 : a904 lda #4 ;set expected interrupt source NMI
|
||
0a67 : 8d0302 sta I_src
|
||
push_stat $ff-intdis
|
||
0a6a : a9fb > lda #$ff-intdis
|
||
0a6c : 48 > pha ;use stack to load status
|
||
|
||
I_set NMI_bit
|
||
0a6d : adfcbf > lda I_port ;turn on interrupt by bit
|
||
0a70 : 297f > and #I_filter
|
||
> if NMI_bit > 7 ;set both NMI & IRQ
|
||
> ora #(1<<IRQ_bit|1<<NMI_bit)
|
||
> else
|
||
0a72 : 0902 > ora #(1<<NMI_bit)
|
||
> endif
|
||
0a74 : 28 > plp ;set flags
|
||
0a75 : 48 > pha ;save to verify
|
||
0a76 : 08 > php
|
||
0a77 : 8dfcbf > sta I_port ;interrupt next instruction plus outbound delay
|
||
|
||
0a7a : ea nop ;allow 6 cycles for interrupt to trip
|
||
0a7b : ea nop
|
||
0a7c : ea nop
|
||
0a7d : ad0302 lda I_src
|
||
trap_ne ;NMI timeout
|
||
0a80 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
0a82 : ba tsx
|
||
0a83 : e0fd cpx #$ff-2 ;original accu & flags remain on stack
|
||
trap_ne ;returned SP
|
||
0a85 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
0a87 : a505 lda nmi_f ;flags seen in NMI vector
|
||
if D_clear = 1
|
||
and #decmode
|
||
trap_ne ;D-flag not cleared
|
||
lda nmi_f
|
||
eor lst_f ;turn off unchanged bits
|
||
and #m8-fai-decmode ;mask untested other flags
|
||
trap_ne ;other flags (N,V,Z,C) changed
|
||
else
|
||
0a89 : 4dfe01 eor lst_f ;turn off unchanged bits
|
||
0a8c : 29cb and #m8-fai ;mask untested other flags
|
||
trap_ne ;other flags (N,V,Z,C,D) changed
|
||
0a8e : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
endif
|
||
0a90 : a2ff ldx #$ff ;reset stack pointer
|
||
0a92 : 9a txs
|
||
|
||
; test IRQ & NMI with interrupts disabled
|
||
0a93 : a200 ldx #0
|
||
0a95 : a904 lda #4 ;set expected interrupt NMI only
|
||
0a97 : 8d0302 sta I_src
|
||
push_stat intdis
|
||
0a9a : a904 > lda #intdis
|
||
0a9c : 48 > pha ;use stack to load status
|
||
|
||
I_set 8 ;both interrupts pending
|
||
0a9d : adfcbf > lda I_port ;turn on interrupt by bit
|
||
0aa0 : 297f > and #I_filter
|
||
> if 8 > 7 ;set both NMI & IRQ
|
||
0aa2 : 0903 > ora #(1<<IRQ_bit|1<<NMI_bit)
|
||
> else
|
||
> ora #(1<<8 )
|
||
> endif
|
||
0aa4 : 28 > plp ;set flags
|
||
0aa5 : 48 > pha ;save to verify
|
||
0aa6 : 08 > php
|
||
0aa7 : 8dfcbf > sta I_port ;interrupt next instruction plus outbound delay
|
||
|
||
0aaa : e8 inx
|
||
0aab : e8 inx
|
||
0aac : e8 inx
|
||
0aad : ad0302 lda I_src ;test NMI done?
|
||
trap_ne
|
||
0ab0 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
0ab2 : a200 ldx #0
|
||
0ab4 : a902 lda #2 ;now re-enable IRQ
|
||
0ab6 : 8d0302 sta I_src
|
||
0ab9 : 58 cli
|
||
0aba : e8 inx
|
||
0abb : e8 inx
|
||
0abc : e8 inx
|
||
0abd : ad0302 lda I_src ;test IRQ done?
|
||
trap_ne
|
||
0ac0 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
0ac2 : a2ff ldx #$ff ;purge stack
|
||
0ac4 : 9a txs
|
||
|
||
;test overlapping NMI, IRQ & BRK
|
||
0ac5 : a200 ldx #0
|
||
0ac7 : a907 lda #7
|
||
0ac9 : 8d0302 sta I_src
|
||
0acc : a9ff lda #$ff ;measure timing
|
||
0ace : 8d0002 sta nmi_count
|
||
0ad1 : 8d0102 sta irq_count
|
||
0ad4 : 8d0202 sta brk_count
|
||
push_stat 0
|
||
0ad7 : a900 > lda #0
|
||
0ad9 : 48 > pha ;use stack to load status
|
||
|
||
I_set 8 ;trigger NMI + IRQ
|
||
0ada : adfcbf > lda I_port ;turn on interrupt by bit
|
||
0add : 297f > and #I_filter
|
||
> if 8 > 7 ;set both NMI & IRQ
|
||
0adf : 0903 > ora #(1<<IRQ_bit|1<<NMI_bit)
|
||
> else
|
||
> ora #(1<<8 )
|
||
> endif
|
||
0ae1 : 28 > plp ;set flags
|
||
0ae2 : 48 > pha ;save to verify
|
||
0ae3 : 08 > php
|
||
0ae4 : 8dfcbf > sta I_port ;interrupt next instruction plus outbound delay
|
||
|
||
endif
|
||
0ae7 : 00 brk
|
||
0ae8 : e8 inx
|
||
0ae9 : e8 inx
|
||
0aea : e8 inx
|
||
0aeb : e8 inx
|
||
0aec : e8 inx
|
||
0aed : e8 inx
|
||
0aee : e8 inx
|
||
0aef : e8 inx
|
||
0af0 : ad0302 lda I_src ;test all done?
|
||
;may fail due to a bug on a real NMOS 6502 - NMI could mask BRK
|
||
trap_ne ;lost an interrupt
|
||
0af3 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
|
||
; S U C C E S S ************************************************
|
||
; -------------
|
||
success ;if you get here everything went well
|
||
0af5 : 4cf50a > jmp * ;test passed, no errors
|
||
|
||
; -------------
|
||
; S U C C E S S ************************************************
|
||
; check data_segment +0 to +2 for sequence of concurrent interrupts
|
||
; e.g. 0x200 = NMI, 0x201 = IRQ, 0x202 = BRK, lower values = earlier
|
||
0af8 : 4c0008 jmp start ;run again
|
||
|
||
; manual tests for the WAI opcode of the 65c02
|
||
|
||
wai macro
|
||
db $cb ;WAI opcode
|
||
endm
|
||
|
||
; requires single step operation, report = 0
|
||
; set PC to the 1st instruction of the test
|
||
; step to the WAI opcode, then manually tie the IRQ input low
|
||
; continue to step until you see the PC advance, then remove IRQ
|
||
; allow the routine to complete.
|
||
|
||
; WAI with interrupts disabled
|
||
0afb : a2ff ldx #$ff
|
||
0afd : 9a txs
|
||
0afe : a003 ldy #3
|
||
0b00 : a900 lda #0 ;IRQ not expected
|
||
0b02 : 8d0302 sta I_src
|
||
set_stat intdis
|
||
0b05 : a904 > lda #intdis
|
||
0b07 : 48 > pha ;use stack to load status
|
||
0b08 : 28 > plp
|
||
|
||
wai
|
||
0b09 : cb > db $cb ;WAI opcode
|
||
|
||
0b0a : 88 dey
|
||
0b0b : 88 dey
|
||
0b0c : 88 dey
|
||
trap_ne ;skipped opcodes!
|
||
0b0d : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
|
||
success
|
||
0b0f : 4c0f0b > jmp * ;test passed, no errors
|
||
|
||
|
||
; WAI with interrupts enabled
|
||
0b12 : a2ff ldx #$ff
|
||
0b14 : 9a txs
|
||
0b15 : a007 ldy #7
|
||
0b17 : a902 lda #2 ;IRQ expected
|
||
0b19 : 8d0302 sta I_src
|
||
set_stat 0
|
||
0b1c : a900 > lda #0
|
||
0b1e : 48 > pha ;use stack to load status
|
||
0b1f : 28 > plp
|
||
|
||
wai
|
||
0b20 : cb > db $cb ;WAI opcode
|
||
|
||
0b21 : 88 dey
|
||
0b22 : 88 dey
|
||
0b23 : 88 dey
|
||
0b24 : ad0302 lda I_src
|
||
trap_ne ;IRQ vector not called
|
||
0b27 : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
0b29 : 88 dey
|
||
trap_ne ;skipped opcodes!
|
||
0b2a : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
|
||
success
|
||
0b2c : 4c2c0b > jmp * ;test passed, no errors
|
||
|
||
|
||
; manual test for the STP opcode of the 65c02
|
||
|
||
stp macro
|
||
db $db ;STP opcode
|
||
endm
|
||
|
||
; set PC to the 1st instruction of the test, then run
|
||
0b2f : ea nop
|
||
0b30 : ea nop
|
||
stp ;expected end of operation
|
||
0b31 : db > db $db ;STP opcode
|
||
|
||
0b32 : ea nop
|
||
0b33 : ea nop
|
||
trap ;overran STP
|
||
0b34 : 4c340b > jmp * ;failed anyway
|
||
|
||
|
||
;end of manual tests
|
||
|
||
;---------------------------------------------------------------------------
|
||
;trap in case of unexpected IRQ, NMI, BRK, RESET - IRQ, NMI, BRK test target
|
||
0b37 : 88 dey
|
||
0b38 : 88 dey
|
||
0b39 : nmi_trap
|
||
if NMI_bit < 0
|
||
dey
|
||
dey
|
||
dey
|
||
trap ;unexpected NMI
|
||
else
|
||
0b39 : 08 php ;either SP or Y count will fail, if we do not hit
|
||
0b3a : 88 dey
|
||
0b3b : 88 dey
|
||
0b3c : 88 dey
|
||
0b3d : 8503 sta nmi_a ;save regsters during NMI
|
||
0b3f : 8604 stx nmi_x
|
||
0b41 : 68 pla
|
||
0b42 : 48 pha
|
||
0b43 : 8505 sta nmi_f
|
||
0b45 : ad0302 lda I_src ;NMI expected?
|
||
0b48 : 2904 and #4
|
||
trap_eq ;unexpexted NMI - check stack for conditions
|
||
0b4a : f0fe > beq * ;failed equal (zero)
|
||
|
||
0b4c : 68 pla ;test I-flag was set
|
||
0b4d : 48 pha
|
||
0b4e : 2904 and #intdis
|
||
trap_eq ;I-flag not set
|
||
0b50 : f0fe > beq * ;failed equal (zero)
|
||
|
||
0b52 : 68 pla ;return with other flags reversed
|
||
0b53 : 49c3 eor #m8-fai-decmode
|
||
0b55 : 48 pha
|
||
0b56 : ba tsx
|
||
0b57 : bd0201 lda $102,x ;test break on stack
|
||
0b5a : 2910 and #break
|
||
trap_ne ;unexpected B-flag! - this may fail on a real 6502
|
||
0b5c : d0fe > bne * ;failed not equal (non zero)
|
||
|
||
;due to a hardware bug on concurrent BRK & NMI
|
||
0b5e : ad0302 lda I_src ;mark expected NMI has occured
|
||
0b61 : 29fb and #$ff-4
|
||
0b63 : 8d0302 sta I_src
|
||
I_clr NMI_bit
|
||
0b66 : adfcbf > lda I_port ;turn off interrupt by bit
|
||
0b69 : 297d > and #I_filter-(1<<NMI_bit)
|
||
0b6b : 8dfcbf > sta I_port
|
||
|
||
0b6e : a604 ldx nmi_x
|
||
0b70 : e8 inx
|
||
0b71 : 8e0002 stx nmi_count
|
||
0b74 : a949 lda #'I' ;mark (NM)I
|
||
0b76 : 28 plp ;should be reversed by rti
|
||
0b77 : 40 rti
|
||
endif
|
||
|
||
0b78 : res_trap
|
||
trap ;unexpected RESET
|
||
0b78 : 4c780b > jmp * ;failed anyway
|
||
|
||
|
||
0b7b : 88 dey
|
||
0b7c : 88 dey
|
||
0b7d : irq_trap ;BRK & IRQ test
|
||
0b7d : 08 php ;either SP or Y count will fail, if we do not hit
|
||
0b7e : 88 dey
|
||
0b7f : 88 dey
|
||
0b80 : 88 dey
|
||
0b81 : 8500 sta irq_a ;save registers during IRQ/BRK
|
||
0b83 : 8601 stx irq_x
|
||
0b85 : 68 pla
|
||
0b86 : 48 pha
|
||
0b87 : 8502 sta irq_f
|
||
0b89 : ad0302 lda I_src ;IRQ expected?
|
||
0b8c : 2903 and #3
|
||
trap_eq ;unexpexted IRQ/BRK - check stack for conditions
|
||
0b8e : f0fe > beq * ;failed equal (zero)
|
||
|
||
0b90 : 68 pla ;test I-flag was set
|
||
0b91 : 48 pha
|
||
0b92 : 2904 and #intdis
|
||
trap_eq ;I-flag not set
|
||
0b94 : f0fe > beq * ;failed equal (zero)
|
||
|
||
0b96 : 68 pla ;return with other flags reversed
|
||
0b97 : 49c3 eor #m8-fai-decmode
|
||
0b99 : 48 pha
|
||
0b9a : ba tsx
|
||
0b9b : bd0201 lda $102,x ;test break on stack
|
||
0b9e : 2910 and #break
|
||
0ba0 : d021 bne brk_trap
|
||
|
||
0ba2 : ad0302 lda I_src ;IRQ expected?
|
||
0ba5 : 2902 and #2
|
||
trap_eq ;unexpexted IRQ - check stack for conditions
|
||
0ba7 : f0fe > beq * ;failed equal (zero)
|
||
|
||
0ba9 : ad0302 lda I_src ;mark expected IRQ has occured
|
||
0bac : 29fd and #$ff-2
|
||
0bae : 8d0302 sta I_src
|
||
I_clr IRQ_bit
|
||
0bb1 : adfcbf > lda I_port ;turn off interrupt by bit
|
||
0bb4 : 297e > and #I_filter-(1<<IRQ_bit)
|
||
0bb6 : 8dfcbf > sta I_port
|
||
|
||
0bb9 : a601 ldx irq_x
|
||
0bbb : e8 inx
|
||
0bbc : 8e0102 stx irq_count
|
||
0bbf : a951 lda #'Q' ;mark (IR)Q
|
||
0bc1 : 28 plp ;should be reversed by rti
|
||
0bc2 : 40 rti
|
||
|
||
0bc3 : brk_trap
|
||
0bc3 : ad0302 lda I_src ;break expected?
|
||
0bc6 : 2901 and #1
|
||
trap_eq ;unexpected BRK - check stack for conditions
|
||
0bc8 : f0fe > beq * ;failed equal (zero)
|
||
|
||
0bca : ad0302 lda I_src ;mark expected BRK has occured
|
||
0bcd : 29fe and #$ff-1
|
||
0bcf : 8d0302 sta I_src
|
||
0bd2 : a601 ldx irq_x
|
||
0bd4 : e8 inx
|
||
0bd5 : 8e0202 stx brk_count
|
||
0bd8 : a500 lda irq_a
|
||
0bda : a94b lda #'K' ;mark (BR)K
|
||
0bdc : 28 plp ;should be reversed by rti
|
||
0bdd : 40 rti
|
||
|
||
if report = 1
|
||
rep_int = 1
|
||
include "report.i65"
|
||
endif
|
||
|
||
|
||
;system vectors
|
||
if (load_data_direct = 1)
|
||
fffa = org $fffa
|
||
fffa : 390b dw nmi_trap
|
||
fffc : 780b dw res_trap
|
||
fffe : 7d0b dw irq_trap
|
||
else
|
||
vec_init
|
||
vec_bss equ $fffa
|
||
dw nmi_trap
|
||
dw res_trap
|
||
dw irq_trap
|
||
endif
|
||
|
||
fffa = end start
|
||
|
||
|
||
No errors in pass 2.
|
||
Wrote binary from address $0000 through $ffff.
|
||
Total size 65536 bytes.
|
||
Program start address is at $0800 (2048).
|
||
|