add support for JMP600 bootloader [fixes Wizard of Words v1.0]

This commit is contained in:
4am 2023-10-02 23:42:07 -04:00
parent 6bf634274f
commit c9af84fa27
12 changed files with 227 additions and 23 deletions

View File

@ -37,6 +37,8 @@
jmp modify2 jmp modify2
;jmodify3 ;jmodify3
jmp modify3 jmp modify3
;jmultimodify
jmp multimodify
;jcompare ;jcompare
jmp compare jmp compare
;jcompare1 ;jcompare1
@ -79,6 +81,10 @@
; set in IDBootFailure() after reading T00 ; set in IDBootFailure() after reading T00
FIRSTFILTER FIRSTFILTER
;gIsJMP600
!byte FALSE ; 0=true, 1=false
; reset before each operation
; set in IDBootloader() after reading T00,S00
;gIsDidatech ;gIsDidatech
!byte FALSE ; 0=true, 1=false !byte FALSE ; 0=true, 1=false
; reset before each operation ; reset before each operation

View File

@ -185,9 +185,10 @@ gIsLowDOS = gPossibleB4BBBasic-$01 ; byte
gIsBaudville = gIsLowDOS-$01 ; byte gIsBaudville = gIsLowDOS-$01 ; byte
gIsEnlightenment = gIsBaudville-$01 ; byte gIsEnlightenment = gIsBaudville-$01 ; byte
gIsDidatech = gIsEnlightenment-$01 ; byte gIsDidatech = gIsEnlightenment-$01 ; byte
gIsJMP600 = gIsDidatech-$01 ; byte
;LASTFILTER ; add new gIs* above this line ;LASTFILTER ; add new gIs* above this line
;gIsInfocom18 is a special case whose ID is not in the regular inspection path ;gIsInfocom18 is a special case whose ID is not in the regular inspection path
gIsInfocom18 = gIsDidatech-$01 ; byte gIsInfocom18 = gIsJMP600-$01 ; byte
;gIs13Sector is a special case whose ID is not in the regular inspection path ;gIs13Sector is a special case whose ID is not in the regular inspection path
gIs13Sector = gIsInfocom18-$01 ; byte gIs13Sector = gIsInfocom18-$01 ; byte
;gMECCFastloadType is a special case integer whose default value cannot be #FALSE ;gMECCFastloadType is a special case integer whose default value cannot be #FALSE
@ -203,7 +204,8 @@ jcompare3 = gDisplayBytes-$03 ; 3-byte
jcompare2 = jcompare3-$03 ; 3-byte jcompare2 = jcompare3-$03 ; 3-byte
jcompare1 = jcompare2-$03 ; 3-byte jcompare1 = jcompare2-$03 ; 3-byte
jcompare = jcompare1-$03 ; 3-byte jcompare = jcompare1-$03 ; 3-byte
jmodify3 = jcompare-$03 ; 3-byte jmultimodify = jcompare-$03 ; 3-byte
jmodify3 = jmultimodify-$03 ; 3-byte
jmodify2 = jmodify3-$03 ; 3-byte jmodify2 = jmodify3-$03 ; 3-byte
jmodify1 = jmodify2-$03 ; 3-byte jmodify1 = jmodify2-$03 ; 3-byte
jmodify = jmodify1-$03 ; 3-byte jmodify = jmodify1-$03 ; 3-byte
@ -226,6 +228,7 @@ modify = jmodify
modify1 = jmodify1 modify1 = jmodify1
modify2 = jmodify2 modify2 = jmodify2
modify3 = jmodify3 modify3 = jmodify3
multimodify = jmultimodify
PrintByID = jPrintByID PrintByID = jPrintByID
SearchTrack = jSearchTrack SearchTrack = jSearchTrack
CallRWTS = jCallRWTS CallRWTS = jCallRWTS
@ -302,6 +305,7 @@ ConstructStandardDelivery = jConstructStandardDelivery
!warn "gDisplayBytes=",gDisplayBytes !warn "gDisplayBytes=",gDisplayBytes
!warn "jcompare=",jcompare !warn "jcompare=",jcompare
!warn "jmodify=",jmodify !warn "jmodify=",jmodify
!warn "jmultimodify=",jmultimodify
!warn "jPrintByID=",jPrintByID !warn "jPrintByID=",jPrintByID
!warn "jSearchTrack=",jSearchTrack !warn "jSearchTrack=",jSearchTrack
!warn "jCallRWTS=",jCallRWTS !warn "jCallRWTS=",jCallRWTS

View File

@ -63,6 +63,7 @@ VerifyInfocom18
beq .setcount ; passport-test-suite/Trinity - Side B.woz [Z=1] here beq .setcount ; passport-test-suite/Trinity - Side B.woz [Z=1] here
lda #$02 ; passport-test-suite/Time Lord.woz [Z=0] here lda #$02 ; passport-test-suite/Time Lord.woz [Z=0] here
; only two sectors when in ID mode ; only two sectors when in ID mode
.setcount .setcount
sta tmp ; sector counter sta tmp ; sector counter
@ -422,3 +423,7 @@ read13
lda $C0E8 lda $C0E8
rts rts
} }
; this prevents branches in this file from crossing a page
; (adjust as necessary)
filler !fill 9

View File

@ -148,6 +148,19 @@ IDBootloader
jmp Trace8B3 jmp Trace8B3
.not8b3 .not8b3
; ;
; Try to identify a bootloader that loads several sectors
; into the text page before loading a non-standard RWTS in
; higher memory.
; Exit via custom trace function if found.
;
+IDJMP600
bcs .notjmp600
jsr PrintByID
!byte s_jmp600
sta gIsJMP600 ; A = 0 (TRUE)
jmp FoundJMP600
.notjmp600
;
; Try to identify all the different MECC fastloader variants. ; Try to identify all the different MECC fastloader variants.
; Exit via custom trace function if found. ; Exit via custom trace function if found.
; (in A=0,X=0; out A=0,X=0) ; (in A=0,X=0; out A=0,X=0)

47
src/id/jmp600.a Normal file
View File

@ -0,0 +1,47 @@
;------------------------------------------------------------------------------
; IDJMP600
; identify JMP600 bootloader
; custom boot sector uses its own RWTS to load one or more sectors into the
; text page and eventually jumps to $600 which uses a second RWTS to load
; and jump to a third RWTS at $B700
;
; in: track buffer contains T00,S00
; A = 0
; X = 0
; out: C clear if JMP600 bootloader was found
; C set if not found
; A = 0
; X = 0
; all other registers & flags clobbered
;-------------------------------
!macro IDJMP600 {
ldy #$11
jsr compare
!byte $01,$A2,$00,$BD,$00,$08,$9D,$00
!byte $02,$E8,$D0,$F7,$4C,$0F,$02,$A9
!byte $A0
; passport-test-suite/Competition Karate.woz [C=0] matches
}
FoundJMP600
;
; We found a JMP600 bootloader. Now we create
; an RWTS that can read the rest of the disk.
; Starting with our built-in RWTS, we modify address
; and data prologues based on the parameters of the
; original disk.
;
lda $84A
lsr
sta UNIV_A1
lda $853
sta UNIV_A2
lda $85D
sta UNIV_A3
lda $888
sta UNIV_D1
lda $891
sta UNIV_D2
lda $89B
sta UNIV_D3
jmp ADStyle

View File

@ -90,3 +90,45 @@ mod_tmp_y=*+1
ldy #$FD ; SMC ldy #$FD ; SMC
clv clv
rts rts
multimodify
; in: (A/Y) -> sector+offsets+values block
; X = number of values (must also be equal number of sectors and offsets)
; out: all registers/flags clobbered
sta @getsector+1
sty @getsector+2
stx @loop+1
clc
adc @loop+1
bcc +
iny
+ sta @getoffset+1
sty @getoffset+2
clc
adc @loop+1
bcc +
iny
+ sta @getvalue+1
sty @getvalue+2
jmp @next
@loop
ldy #$FD ; SMC
@getvalue
lda $FDFD, y ; SMC
sta @cmpvalue
sta @modvalue
@getsector
lda $FDFD, y ; SMC
@getoffset
ldx $FDFD, y ; SMC
jsr compare1
@cmpvalue
!byte $FD ; SMC
bcc @next
jsr modify1
@modvalue
!byte $FD ; SMC
@next
dec @loop+1
bpl @loop
rts

View File

@ -101,6 +101,7 @@ T00_IsNotRWTS
!source "../patchers/t00_baudville.a" !source "../patchers/t00_baudville.a"
!source "../patchers/t00_enlightenment.a" !source "../patchers/t00_enlightenment.a"
!source "../patchers/t00_scottforesman.a" !source "../patchers/t00_scottforesman.a"
!source "../patchers/t00_jmp600.a"
rts rts
!if * > $3200 { !if * > $3200 {

View File

@ -76,6 +76,7 @@ NonRelocatable
!source "id/davidson.a" !source "id/davidson.a"
!source "id/holle.a" !source "id/holle.a"
!source "id/phoenix.a" !source "id/phoenix.a"
!source "id/jmp600.a"
!source "id/555.a" !source "id/555.a"
!source "id/trace32.a" !source "id/trace32.a"
!source "id/trace8b3.a" !source "id/trace8b3.a"

90
src/patchers/t00_jmp600.a Normal file
View File

@ -0,0 +1,90 @@
;------------------------------------------------------------------------------
; #JMP600
; custom boot sector uses its own RWTS to load one or more sectors into the
; text page and eventually jumps to $600 which uses a second RWTS to load
; and jump to a third RWTS at $B700
;
; tested on
; - Dinosaurs v1.0 (Advanced Ideas)
; - Wizard of Words v1.0 (Advanced Ideas)
;------------------------------------------------------------------------------
;
; common to all variants:
;
; T00,S00,$4A: -> D5
; T00,S00,$53: -> AA
; T00,S00,$5D: -> 96
;
; T00,S00,$88: -> D5
; T00,S00,$91: -> AA
; T00,S00,$9B: -> AD
;
; T00,S03,$1A: -> D5
; T00,S03,$23: -> AA
; T00,S03,$2D: -> 96
;
; T00,S03,$57: -> D5
; T00,S03,$60: -> AA
; T00,S03,$6A: -> AD
;
; That's all that's required for the 'basic' version (e.g. Wizard of Words v1.0).
; Its final RWTS loaded into $B700+ is custom but seamlessly supports both
; reading and reading of both protected and unprotected disks. However, there
; are several variants in use. Some variants require additional patches because
; - they don't support reading from unprotected disks (e.g. In Search of the Most Amazing Thing), or
; - they don't support writing to unprotected disks, or
; - they support writing to disks but in some other protected format that is
; different than the program disk (e.g. Competition Karate, Amazon)
; e.g. Competition Karate / Amazon variant:
;
; T00,S06,$53: -> D5
; T00,S06,$58: -> AA
; T00,S06,$5D: -> AD
; T00,S06,$9E: -> DE
; T00,S06,$A3: -> AA
; T00,S06,$A8: -> EB
;
; T00,S07,$46: -> D5
; T00,S07,$4B: -> AA
; T00,S07,$50: -> 96
; T00,S07,$7A: -> DE
; T00,S07,$7F: -> AA
; T00,S07,$84: -> EB
;
; T00,S08,$74: -> D5
; T00,S08,$82: -> AA
; T00,S08,$91: -> 96
;
; TODO need to test on real hardware and see if these are sufficient:
; T00,S06,$49: -> 04 to restore standard timing
; T00,S07,$D2: -> 2C to burn fewer cycles
; T00,S08,$6C: -> 2C to burn fewer cycles
;
;.competitionkaratemulti ; length=21 dec
; !byte $08,$08,$08,$08,$07,$07,$07,$07,$07,$07,$07,$07,$07,$07,$06,$06,$06,$06,$06,$06,$06
; !byte $91,$82,$74,$6C,$F7,$E8,$DA,$D2,$84,$7F,$7A,$50,$4B,$46,$A8,$A3,$9E,$5D,$58,$53,$49
; !byte $96,$AA,$D5,$2C,$AD,$AA,$D5,$2C,$EB,$AA,$DE,$96,$AA,$D5,$EB,$AA,$DE,$AD,$AA,$D5,$04
!zone {
; gTrack = 0
lda gIsJMP600
beq .start
jmp .exit
.jmp600multi ; length=12 dec
;sectors
!byte $03,$03,$03,$03,$03,$03,$00,$00,$00,$00,$00,$00
;offsets
!byte $6A,$60,$57,$2D,$23,$1A,$9B,$91,$88,$5D,$53,$4A
;values
!byte $AD,$AA,$D5,$96,$AA,$D5,$AD,$AA,$D5,$96,$AA,$D5
.start
lda #<.jmp600multi
ldy #>.jmp600multi
ldx #12
jsr multimodify
.exit
}

View File

@ -6,31 +6,21 @@
!zone { !zone {
lda gIsLaureate lda gIsLaureate
bne .exit bne .exit
beq + ; passport-test-suite/Sound Ideas- Word Attack - Disk 1 - Ending Consonants.woz [Z=1] reaches here beq .start ; passport-test-suite/Sound Ideas- Word Attack - Disk 1 - Ending Consonants.woz [Z=1] reaches here
.sectors .laureatemulti
;sectors
!byte $04,$04,$04,$04,$04,$04,$04,$04,$04,$05,$05,$05,$05,$05,$05,$05,$06,$06,$06 !byte $04,$04,$04,$04,$04,$04,$04,$04,$04,$05,$05,$05,$05,$05,$05,$05,$06,$06,$06
.offsets ;offsets
!byte $53,$58,$5D,$9E,$A3,$A8,$E7,$F1,$FC,$35,$3F,$55,$5F,$6A,$91,$9B,$49,$D5,$D6 !byte $53,$58,$5D,$9E,$A3,$A8,$E7,$F1,$FC,$35,$3F,$55,$5F,$6A,$91,$9B,$49,$D5,$D6
.values ;values
!byte $D5,$AA,$AD,$DE,$AA,$EB,$D5,$AA,$AD,$DE,$AA,$D5,$AA,$96,$DE,$AA,$D6,$D5,$20 !byte $D5,$AA,$AD,$DE,$AA,$EB,$D5,$AA,$AD,$DE,$AA,$D5,$AA,$96,$DE,$AA,$D6,$D5,$20
+ lda #$12 .start
sta .loop+1 lda #<.laureatemulti
ldy #>.laureatemulti
.loop ldy #$d1 ldx #19
lda .values,y jsr multimodify
sta .valuea
sta .valueb
lda .sectors,y
ldx .offsets,y
jsr compare1
.valuea !byte $d1
bcc +
jsr modify1
.valueb !byte $d1
+ dec .loop+1
bpl .loop
lda #$0A lda #$0A
ldx #$12 ldx #$12

View File

@ -221,6 +221,7 @@ StringTableLow ; must be kept in sync with constants in enid.a
!byte <.didatech !byte <.didatech
!byte <.anychecksum !byte <.anychecksum
!byte <.unexpected5and3 !byte <.unexpected5and3
!byte <.jmp600
StringTableHigh ; must be kept in sync with constants in enid.a StringTableHigh ; must be kept in sync with constants in enid.a
!byte >.header !byte >.header
@ -381,11 +382,12 @@ StringTableHigh ; must be kept in sync with constants in enid.a
!byte >.didatech !byte >.didatech
!byte >.anychecksum !byte >.anychecksum
!byte >.unexpected5and3 !byte >.unexpected5and3
!byte >.jmp600
.passport .passport
!text "Passport ",$00 !text "Passport ",$00
.header .header
!text "@",s_passport,"by 4am@",s_space7,"@",s_space7," 2023-09-04",$00 !text "@",s_passport,"by 4am@",s_space7,"@",s_space7," 2023-10-02",$00
.bar9 .bar9
!text "_________",$00 !text "_________",$00
.bar18 .bar18
@ -790,3 +792,5 @@ StringTableHigh ; must be kept in sync with constants in enid.a
!text "address field checksum",$8D,$00 !text "address field checksum",$8D,$00
.unexpected5and3 .unexpected5and3
!text "@",s_tfound,"5-and-3 @",s_protectiontrack,$8D,$00 !text "@",s_tfound,"5-and-3 @",s_protectiontrack,$8D,$00
.jmp600
!text "@",s_found,"JMP $0600 @",s_bootloader,$8D,$00

View File

@ -159,3 +159,4 @@ s_probs5 = $9A
s_didatech = $9B s_didatech = $9B
s_anychecksum = $9C s_anychecksum = $9C
s_unexpected5and3 = $9D s_unexpected5and3 = $9D
s_jmp600 = $9E