From 6ddb57fadb9f08b9d1f6d48dec101587b407f1c2 Mon Sep 17 00:00:00 2001 From: Adrian Conlon Date: Fri, 5 Jan 2018 19:42:34 +0000 Subject: [PATCH] Updated to latest version of the Klaus Dormann 6502 tests. Signed-off-by: Adrian Conlon --- .../test_M6502/roms/6502_functional_test.a65 | 139 +++++++++++--- .../roms/65C02_extended_opcodes_test.a65c | 177 +++++++++++++++--- M6502/test_M6502/roms/report.i65 | 49 ++++- 3 files changed, 313 insertions(+), 52 deletions(-) diff --git a/M6502/test_M6502/roms/6502_functional_test.a65 b/M6502/test_M6502/roms/6502_functional_test.a65 index 333497d..2c4031c 100644 --- a/M6502/test_M6502/roms/6502_functional_test.a65 +++ b/M6502/test_M6502/roms/6502_functional_test.a65 @@ -21,7 +21,7 @@ ; addressing modes with focus on propper setting of the processor status ; register bits. ; -; version 21-oct-2015 +; version 04-dec-2017 ; contact info at http://2m5.de or email K@2m5.de ; ; assembled with AS65 from http://www.kingswood-consulting.co.uk/assemblers/ @@ -73,6 +73,9 @@ ; 24-aug-2015 all self modifying immediate opcodes now execute in data RAM ; added small branch offset pretest ; 21-oct-2015 added option to disable decimal mode ADC & SBC tests +; 04-dec-2017 fixed BRK only tested with interrupts enabled +; added option to skip the remainder of a failing test +; in report.i65 ; C O N F I G U R A T I O N @@ -85,7 +88,7 @@ ROM_vectors = 1 ;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 -load_data_direct = 0 +load_data_direct = 1 ;I_flag behavior (0=force enabled, 1=force disabled, 2=prohibit change, 3=allow ;change) 2 requires extra code and is not recommended. SEI & CLI can only be @@ -1608,35 +1611,40 @@ jsr_ret = *-1 ;last address of jsr = return address ; break & return from interrupt if ROM_vectors = 1 - set_stat 0 + load_flag 0 ;with interrupts enabled if allowed! + pha lda #'B' ldx #'R' - ldy #'K' ;N=0, V=0, Z=0, C=0 + ldy #'K' + plp ;N=0, V=0, Z=0, C=0 brk else - lda #hi brk_ret ;emulated break + lda #hi brk_ret0 ;emulated break pha - lda #lo brk_ret + lda #lo brk_ret0 pha - lda #fao ;set break & unused on stack + load_flag fao ;set break & unused on stack + pha + load_flag intdis ;during interrupt pha - set_stat intdis lda #'B' ldx #'R' - ldy #'K' ;N=0, V=0, Z=0, C=0 + ldy #'K' + plp ;N=0, V=0, Z=0, C=0 jmp irq_trap endif dey ;should not be executed -brk_ret ;address of break return +brk_ret0 ;address of break return php ;either SP or Y count will fail, if we do not hit dey dey dey - cmp #('B'^$aa) ;returned registers OK? + cmp #'B'^$aa ;returned registers OK? + ;the IRQ vector was never executed if A & X stay unmodified trap_ne - cpx #('R'+1) + cpx #'R'+1 trap_ne - cpy #('K'-6) + cpy #'K'-6 trap_ne pla ;returned flags OK (unchanged)? cmp_flag 0 @@ -1644,6 +1652,47 @@ brk_ret ;address of break return tsx ;sp? cpx #$ff trap_ne + if ROM_vectors = 1 + load_flag $ff ;with interrupts disabled if allowed! + pha + lda #$ff-'B' + ldx #$ff-'R' + ldy #$ff-'K' + plp ;N=1, V=1, Z=1, C=1 + brk + else + lda #hi brk_ret1 ;emulated break + pha + lda #lo brk_ret1 + pha + load_flag $ff + pha ;set break & unused on stack + pha ;actual flags + lda #$ff-'B' + ldx #$ff-'R' + ldy #$ff-'K' + plp ;N=1, V=1, Z=1, C=1 + jmp irq_trap + endif + dey ;should not be executed +brk_ret1 ;address of break return + php ;either SP or Y count will fail, if we do not hit + dey + dey + dey + cmp #($ff-'B')^$aa ;returned registers OK? + ;the IRQ vector was never executed if A & X stay unmodified + trap_ne + cpx #$ff-'R'+1 + trap_ne + cpy #$ff-'K'-6 + trap_ne + pla ;returned flags OK (unchanged)? + cmp_flag $ff + trap_ne + tsx ;sp? + cpx #$ff + trap_ne next_test ; test set and clear flags CLC CLI CLD CLV SEC SEI SED @@ -5808,6 +5857,7 @@ test_ind eor #$aa ;N=1, V=1, Z=0, C=1 jmp (ptr_ind_ret) trap ;runover protection + jmp start ;catastrophic error - cannot continue ; target for the jump subroutine test dey @@ -5848,12 +5898,15 @@ test_jsr eor #$aa ;N=1, V=1, Z=0, C=1 rts trap ;runover protection + jmp start ;catastrophic error - cannot continue ;trap in case of unexpected IRQ, NMI, BRK, RESET - BRK test target nmi_trap trap ;check stack for conditions at NMI + jmp start ;catastrophic error - cannot continue res_trap trap ;unexpected RESET + jmp start ;catastrophic error - cannot continue dey dey @@ -5862,41 +5915,81 @@ irq_trap ;BRK test or unextpected BRK or IRQ dey dey dey - ;next 4 traps could be caused by unexpected BRK or IRQ + ;next traps could be caused by unexpected BRK or IRQ ;check stack for BREAK and originating location ;possible jump/branch into weeds (uninitialized space) - cmp #'B' ;registers loaded? + cmp #$ff-'B' ;BRK pass 2 registers loaded? + beq break2 + cmp #'B' ;BRK pass 1 registers loaded? trap_ne cpx #'R' trap_ne - cpy #('K'-3) + cpy #'K'-3 trap_ne sta irq_a ;save registers during break test stx irq_x tsx ;test break on stack lda $102,x - cmp_flag 0 ;break test should have B=1 + cmp_flag 0 ;break test should have B=1 & unused=1 on stack trap_ne ; - no break flag on stack pla - cmp #fai ;should have added interrupt disable + cmp_flag intdis ;should have added interrupt disable trap_ne tsx cpx #$fc ;sp -3? (return addr, flags) trap_ne lda $1ff ;propper return on stack - cmp #hi(brk_ret) + cmp #hi(brk_ret0) trap_ne lda $1fe - cmp #lo(brk_ret) + cmp #lo(brk_ret0) trap_ne - set_stat $ff + load_flag $ff + pha ldx irq_x inx ;return registers with modifications lda irq_a - eor #$aa ;N=1, V=1, Z=0, C=1 but original flags should be restored + eor #$aa + plp ;N=1, V=1, Z=1, C=1 but original flags should be restored rti trap ;runover protection + jmp start ;catastrophic error - cannot continue +break2 ;BRK pass 2 + cpx #$ff-'R' + trap_ne + cpy #$ff-'K'-3 + trap_ne + sta irq_a ;save registers during break test + stx irq_x + tsx ;test break on stack + lda $102,x + cmp_flag $ff ;break test should have B=1 + trap_ne ; - no break flag on stack + pla + ora #decmode ;ignore decmode cleared if 65c02 + cmp_flag $ff ;actual passed flags + trap_ne + tsx + cpx #$fc ;sp -3? (return addr, flags) + trap_ne + lda $1ff ;propper return on stack + cmp #hi(brk_ret1) + trap_ne + lda $1fe + cmp #lo(brk_ret1) + trap_ne + load_flag intdis + pha + ldx irq_x + inx ;return registers with modifications + lda irq_a + eor #$aa + plp ;N=0, V=0, Z=0, C=0 but original flags should be restored + rti + trap ;runover protection + jmp start ;catastrophic error - cannot continue + if report = 1 include "report.i65" endif @@ -6007,4 +6100,4 @@ vec_bss equ $fffa endif end start - + \ No newline at end of file diff --git a/M6502/test_M6502/roms/65C02_extended_opcodes_test.a65c b/M6502/test_M6502/roms/65C02_extended_opcodes_test.a65c index a504861..e18e098 100644 --- a/M6502/test_M6502/roms/65C02_extended_opcodes_test.a65c +++ b/M6502/test_M6502/roms/65C02_extended_opcodes_test.a65c @@ -20,9 +20,9 @@ ; This program is designed to test all additional 65C02 opcodes, addressing ; modes and functionality not available in the NMOS version of the 6502. ; The 6502_functional_test is a prerequisite to this test. -; NMI, IRQ, BRK, STP & WAI are covered in the 6502_interrupt_test. +; NMI, IRQ, STP & WAI are covered in the 6502_interrupt_test. ; -; version 09-feb-2017 +; version 04-dec-2017 ; contact info at http://2m5.de or email K@2m5.de ; ; assembled with AS65 from http://www.kingswood-consulting.co.uk/assemblers/ @@ -65,6 +65,10 @@ ; 24-aug-2015 all self modifying immediate opcodes now execute in data RAM ; 28-aug-2015 fixed decimal adc/sbc immediate only testing carry ; 09-feb-2017 fixed RMB/SMB tested when they shouldn't be tested +; 04-dec-2017 fixed BRK not tested for actually going through the IRQ vector +; added option to skip the remainder of a failing test +; in report.i65 +; added skip override to undefined opcode as NOP test ; C O N F I G U R A T I O N @@ -106,6 +110,10 @@ wdc_op = 1 ;(0=test as NOPs, 1=full test, >1=no test) rkwl_wdc_op = 1 +;skip testing all undefined opcodes override +;0=test as NOP, >0=skip +skip_nop = 0 + ;report errors through I/O channel (0=use standard self trap loops, 1=include ;report.i65 as I/O channel, add 3 kB) report = 0 @@ -552,6 +560,9 @@ test_num = test_num + 1 bss ;uninitialized segment, copy of data at end of code! endif org zero_page +;break test interrupt save +irq_a ds 1 ;a register +irq_x ds 1 ;x register if I_flag = 2 ;masking for I bit in status flag_I_on ds 1 ;or mask to load flags @@ -1131,7 +1142,8 @@ nop_test macro ;\1 = opcode, \2 = # of bytes cpx #0 trap_ne ;x changed endm - + + if skip_nop = 0 nop_test $02,2 nop_test $22,2 nop_test $42,2 @@ -1215,6 +1227,7 @@ nop_test macro ;\1 = opcode, \2 = # of bytes nop_test $db,1 endif next_test + endif ; jump indirect (test page cross bug is fixed) ldx #3 ;prepare table @@ -1310,10 +1323,59 @@ jxp_ok if ROM_vectors = 1 ; test BRK clears decimal mode - sed + load_flag 0 ;with interrupts enabled if allowed! + pha + lda #'B' + ldx #'R' + ldy #'K' + plp ;N=0, V=0, Z=0, C=0 brk - nop -brk_ret + dey ;should not be executed +brk_ret0 ;address of break return + php ;either SP or Y count will fail, if we do not hit + dey + dey + dey + cmp #'B'^$aa ;returned registers OK? + ;the IRQ vector was never executed if A & X stay unmodified + trap_ne + cpx #'R'+1 + trap_ne + cpy #'K'-6 + trap_ne + pla ;returned flags OK (unchanged)? + cmp_flag 0 + trap_ne + tsx ;sp? + cpx #$ff + trap_ne +;pass 2 + load_flag $ff ;with interrupts disabled if allowed! + pha + lda #$ff-'B' + ldx #$ff-'R' + ldy #$ff-'K' + plp ;N=1, V=1, Z=1, C=1 + brk + dey ;should not be executed +brk_ret1 ;address of break return + php ;either SP or Y count will fail, if we do not hit + dey + dey + dey + cmp #($ff-'B')^$aa ;returned registers OK? + ;the IRQ vector was never executed if A & X stay unmodified + trap_ne + cpx #$ff-'R'+1 + trap_ne + cpy #$ff-'K'-6 + trap_ne + pla ;returned flags OK (unchanged)? + cmp_flag $ff + trap_ne + tsx ;sp? + cpx #$ff + trap_ne next_test endif @@ -2534,6 +2596,7 @@ test_ji nop nop trap ;runover protection + jmp start ;catastrophic error - cannot continue ; target for the jump indirect test jxi_adr dw trap_ind @@ -2578,6 +2641,7 @@ test_jxi nop nop trap ;runover protection + jmp start ;catastrophic error - cannot continue ; JMP (abs,x) with bad x nop @@ -2586,33 +2650,96 @@ trap_ind nop nop trap ;near miss indexed indirect jump + jmp start ;catastrophic error - cannot continue ;trap in case of unexpected IRQ, NMI, BRK, RESET nmi_trap trap ;check stack for conditions at NMI + jmp start ;catastrophic error - cannot continue res_trap trap ;unexpected RESET -irq_trap - php ;save decimal flag + jmp start ;catastrophic error - cannot continue + + dey + dey +irq_trap ;BRK test or unextpected BRK or IRQ + php ;either SP or Y count will fail, if we do not hit + dey + dey + dey + ;next traps could be caused by unexpected BRK or IRQ + ;check stack for BREAK and originating location + ;possible jump/branch into weeds (uninitialized space) + cmp #$ff-'B' ;BRK pass 2 registers loaded? + beq break2 + cmp #'B' ;BRK pass 1 registers loaded? + trap_ne + cpx #'R' + trap_ne + cpy #'K'-3 + trap_ne + sta irq_a ;save registers during break test + stx irq_x tsx ;test break on stack lda $102,x - and #break - trap_eq ;check stack for conditions at IRQ - if ROM_vectors = 1 - pla ;test decimal mode cleared - and #decmode - trap_ne ;decimal mode not cleared after BRK - plp ;pop saved flags - pla ;return address low - cmp #lo(brk_ret) - trap_ne ;unexpected BRK - pla ;return address high - cmp #hi(brk_ret) - trap_ne ;unexpected BRK - jmp brk_ret - else - trap_ne ;check stack for conditions at BRK - endif + cmp_flag 0 ;break test should have B=1 & unused=1 on stack + trap_ne ;possible no break flag on stack + pla + cmp_flag intdis ;should have added interrupt disable + trap_ne + tsx + cpx #$fc ;sp -3? (return addr, flags) + trap_ne + lda $1ff ;propper return on stack + cmp #hi(brk_ret0) + trap_ne + lda $1fe + cmp #lo(brk_ret0) + trap_ne + load_flag $ff + pha + ldx irq_x + inx ;return registers with modifications + lda irq_a + eor #$aa + plp ;N=1, V=1, Z=1, C=1 but original flags should be restored + rti + trap ;runover protection + jmp start ;catastrophic error - cannot continue + +break2 ;BRK pass 2 + cpx #$ff-'R' + trap_ne + cpy #$ff-'K'-3 + trap_ne + sta irq_a ;save registers during break test + stx irq_x + tsx ;test break on stack + lda $102,x + cmp_flag $ff ;break test should have B=1 + trap_ne ;possibly no break flag on stack + pla + cmp_flag $ff-decmode ;actual passed flags should have decmode cleared + trap_ne + tsx + cpx #$fc ;sp -3? (return addr, flags) + trap_ne + lda $1ff ;propper return on stack + cmp #hi(brk_ret1) + trap_ne + lda $1fe + cmp #lo(brk_ret1) + trap_ne + load_flag intdis + pha + ldx irq_x + inx ;return registers with modifications + lda irq_a + eor #$aa + plp ;N=0, V=0, Z=0, C=0 but original flags should be restored + rti + trap ;runover protection + jmp start ;catastrophic error - cannot continue if report = 1 include "report.i65" diff --git a/M6502/test_M6502/roms/report.i65 b/M6502/test_M6502/roms/report.i65 index 06f8849..8cf7a09 100644 --- a/M6502/test_M6502/roms/report.i65 +++ b/M6502/test_M6502/roms/report.i65 @@ -78,6 +78,8 @@ rabs jsr rspace ;ask to continue rprt rmsg_cont rerr1 jsr rget + cmp #'S' + beq rskip cmp #'C' bne rerr1 ;restore registers @@ -87,7 +89,44 @@ rerr1 jsr rget tax pla plp - rts + rts +;skip the current test +rskip lda #$f0 ;already end of tests? + cmp test_case + beq rerr1 ;skip is not available + ldx #$ff ;clear stack + txs + inc test_case ;next test + lda #lo(start) ;find begin of test + sta zpt + lda #hi(start) + sta zpt+1 +rskipl1 ldy #4 ;search pattern +rskipl2 lda (zpt),y ;next byte + cmp rmark,y + bne rskipnx ;no match + dey + bmi rskipf ;found pattern + cpy #1 ;skip immediate value + bne rskipl2 + dey + beq rskipl2 + +rskipnx inc zpt ;next RAM location + bne rskipl1 + inc zpt+1 + bne rskipl1 + +rskipf ldy #1 ;pattern found - check test number + lda (zpt),y ;test number + cmp #$f0 ;end of last test? + beq rskipe ;ask to rerun all + cmp test_case ;is next test? + bne rskipnx ;continue searching +rskipe jmp (zpt) ;start next test or rerun at end of tests + +rmark lda #0 ;begin of test search pattern + sta test_case ;show test has ended, ask to repeat report_success @@ -117,7 +156,8 @@ rget ;get character in A ; and #8 ; beq rget1 ;not a real ACIA - so RDRF is not checked - lda $bff0 ;read acia rx reg +; lda $bff0 ;read acia rx reg + lda $f004 ;Kowalski simulator default ;the load can be replaced by a call to a kernal routine ; jsr $ffcf ;example: CHRIN for a C64 cmp #'a' ;lower case @@ -159,7 +199,8 @@ rchar ;report character in A ; beq rchar1 ; pla ;not a real ACIA - so TDRF is not checked - sta $bff0 ;write acia tx reg +; sta $bff0 ;write acia tx reg + sta $f001 ;Kowalski simulator default ;the store can be replaced by a call to a kernal routine ; jsr $ffd2 ;example: CHROUT for a C64 rts @@ -169,7 +210,7 @@ rmsg_start rmsg_stack db 10,13,"regs Y X A PS PCLPCH",10,13,0 rmsg_cont - db 10,13,"press C to continue",10,13,0 + db 10,13,"press C to continue or S to skip current test",10,13,0 rmsg_success db 10,13,"All tests completed, press R to repeat",10,13,0 if rep_int = 1