apple2_dclock/dclock.system.merlin32.s
2018-03-23 07:53:39 -07:00

773 lines
30 KiB
ArmAsm

; Fully disassembled and analyzed source to AE
; DCLOCK.SYSTEM by M.G. - 04/18/2017
; Assembles to a binary match for AE code unless
; FIX_BUGS is set.
; speaking of FIX_BUGS, there are critical bugs in the
; original AE code:
; * When driver loader is initially probing it corrupts the
; Apple //c Memory Expansion Card:
; - it saves, but fails to restore, data at address $080000
; - it fails to reset slinky pointer, and *will* trash $080000-$080007
; * When the clock is read, it corrupts data at address $08xx01
; - John Brooks spotted this, I totally missed this.
; Setting FIX_BUGS to 1 will fix these issues.
; other notes:
; * uses direct block access to read volume directory,
; so won't launch from an AppleShare volume.
; Build instructions:
; ca65 dclock.system.s -l dclock.system.lst
; ld65 -t none -o dclock.system dclock.system.o
; put dclock.system as a SYS file on a ProDOS disk.
FIX_BUGS = 0 ; set to 1 to fix critical bugs
;.setcpu "65C02"
; zero page locations
SLASHOFFS = $00 ; offset of last '/' in our path
MYNAMELEN = $01 ; length of our file name in path
FENTPTR = $02 ; directory file entry pointer
FENTPTRL = FENTPTR
FENTPTRH = FENTPTR+1
CURENT = $04 ; current file entry in block
ENTLEN = $05 ; length of a file entry
ENTPERBLK = $06 ; number of entries per block
DIRBLK = $07 ; directory block to read
DIRBLKL = DIRBLK
DIRBLKH = DIRBLK+1
CHRPTR = $09 ; character pointer for print routine
CHRPTRL = CHRPTR
CHRPTRH = CHRPTR+1
SCRATCH = $0B ; scratch value for BCD range checks
SAVEBYTE = $0C ; slinky overwritten byte save location
BCDTMP = $3A ; location clock driver uses for BCD->Binary
; entry points
PRODOS = $BF00
HOME = $FC58
WAIT = $FCA8
COUT1 = $FDF0
; buffers & other spaces
INBUF = $0200 ; input buffer
PATHBUF = $0280 ; path buffer
CLOCKBUF = $0300 ; clock buffer
RELOCT = $1200 ; reloc target
BLOCKBUF = $1000 ; buffer for BLOCK_READ
READBUF = $1C00 ; I/O buffer for READ
SYSEXEC = $2000 ; location of SYS file executable
CLKCODE = $D742 ; Clock code location
; global Page entries
CLKENTRY = $BF06 ; clock routine entry point
DATELO = $BF90
DATEHI = $BF91
TIMELO = $BF92
TIMEHI = $BF93
MACHID = $BF98 ; machine ID
; I/O and hardware
ROMIn2 = $C082 ; access to read ROM/no write RAM
LCBank1 = $C08B ; Access twice to write bank 1
C8OFF = $CFFF ; C8xx ROM off
SLOT4ROM = $C400 ; Slot 4 ROM space
SLOT4IO = $C0C0 ; Slot 4 I/O space
DPTRL = SLOT4IO+0 ; Slinky data ptr low
DPTRM = SLOT4IO+1 ; Slinky data ptr middle
DPTRH = SLOT4IO+2 ; Slinky data ptr high
DATA = SLOT4IO+3 ; Slinky data byte
; Misc
CLKCODEMAX = $7D
; PRODOS Command
PRODOS_QUIT = $65
PRODOS_READ_BLOCK = $80
PRODOS_OPEN = $C8
PRODOS_READ = $CA
PRODOS_CLOSE = $CC
PRODOS_GET_EOF = $D1
ORG SYSEXEC
; ----------------------------------------------------------------------------
; relocate code from RELOCS to RELOCT
DClockSystem
RSEG1 = rseg1e-rseg1b ; MERLIN32: removed ()
ldy #rseg1e-rseg1b ; MERLIN32: removed ()
:_ lda RELOCS,y
sta RELOCT-1,y
dey
bne :_
; greetings
jsr HOME
jsr iprint
STR 'DClock',0D
jsr iprint
STR 'Copyright (c)1988 Applied Engineering',0D,0D
jsr ClockRead
jsr ValidTime
bcc InstallDriver
; clock not found
jsr iprint
STR 'Can',27,'t find clock',0D
jmp NextSys
; ----------------------------------------------------------------------------
; Install clock driver
InstallDriver
; make language card writable
lda LCBank1
lda LCBank1
; copy clock driver into ProDOS
; should not be hard-coded, Apple says to put it at
; the address pointed to at ($BF07). [PDOS8TRM 6.1.1]
ldx #$00
:_ lda rclkdrv-1,x
sta CLKCODE,x
inx
cpx #CLKCODEMAX
bcc :_
; make LC write-protected
lda ROMIn2
; make sure clock vector is preceded by a JMP
; (it is RTS if no clock installed)
lda #$4C
sta CLKENTRY
; indicate in MACHID that clock is present
lda MACHID
ora #$01
sta MACHID
; ----------------------------------------------------------------------------
; find and launch the next .SYSTEM file
NextSys
; check our path for last / (counted string at $0280)
ldy PATHBUF
:_ lda PATHBUF,y
cmp #'/'
beq :LSlash
dey
bne :_
:LSlash sty SLASHOFFS ; offset of '/' into $00
lda PATHBUF
sec
sbc SLASHOFFS ; calculate length of our name
sta MYNAMELEN ; and put in $01
jsr ReadVolDir1 ; read first block of volume directory
bcs :BadSys ; didn't get anything valid
; Now search for our own file name
:ChkSlf lda (FENTPTR) ; get length & storage type
and #$0F ; mask in length
cmp MYNAMELEN ; same as our name?
bne :INext ; Nope, next entry
tay ; Now let's see if name matches
ldx PATHBUF
:__ lda (FENTPTR),y
cmp PATHBUF,x
bne :INext ; no match, next entry
dex
dey
bne :__
; ok, found ourself, now get next entry and start looking for *.SYSTEM
bra :SNext ; Get next entry and start checking for .SYSTEM file
:INext jsr NextFileEnt ; Get next entry and check again for ourself
bcc :ChkSlf
bra :BadSys ; no more valid entries
; check the current entry for .SYSTEM
:ChkSys lda (FENTPTR) ; get storage & length
and #$F0 ; mask in storage type
beq :SNext ; not valid if 0
cmp #$40
bcs :SNext ; not valid if >= $40 (not type 1-3)
lda (FENTPTR) ; load it again
and #$0F ; but mask in length this time
cmp #$07
bcc :SNext ; not valid if not at least 7 chars long
; now check to see if it ends with .SYSTEM
tay
ldx #$07
:___ lda (FENTPTR),y
cmp dSYSTEM,x
bne :SNext ; no match, next!
dey
dex
bne :___
ldy #$10
lda (FENTPTR),y ; get file Type
cmp #$FF ; is SYS
bne :SNext ; Nope
; okay we have a .SYSTEM file to load
lda (FENTPTR) ; get storage type/name length again
and #$0F ; mask in name length
tay ; and save for copying
; calculate total length of path and put in length byte at PATHBUF
clc
adc PATHBUF
sec
sbc MYNAMELEN
sta PATHBUF
; copy filename into path
tax
:____ lda (FENTPTR),y
sta PATHBUF,x
dex
dey
bne :____
jmp LaunchNext ; proceed to launch code
; get next entry for looking for .SYSTEM
:SNext jsr NextFileEnt
bcc :ChkSys
; if we get here, unable to identify next .SYSTEM
:BadSys cmp #$00 ; error code in accumulator?
beq :NoSys ; no, must have not found it
pha
jsr iprint
STR 'Directory ProDOS error #' ; MERLIN32: removed len $18
pla
jsr PrHex
lda #$8D
jsr COUT1
jmp DoQuit ; quit to ProDOS
:NoSys jsr iprint
STR 'Can',27,'t find next .SYSTEM file',0d ; MERLIN32: REMOVED length $1D, replaced "'" with /27/
jmp DoQuit ; quit to ProDOS
dSYSTEM = * - 1
ASC '.SYSTEM'
; ----------------------------------------------------------------------------
; this looks like some unused debugging code to print the filename
; from the current directory entry pointed to at ($02)
PrintFN
lda (FENTPTR) ; get storage type/name length
pha ; save it
and #$0F ; mask in name length
sta (FENTPTR) ; put in file entry
beq :done ; if zero, we are done
ldy #$01 ; start with byte 1 of entry
:_ lda (FENTPTR),y ; get file name char
ora #$80
jsr COUT1 ; print it
tya
iny
cmp (FENTPTR) ; printed all the chars?
bcc :_ ; nope
lda #$8D ; CR
jsr COUT1
:done pla ; get type/name length
sta (FENTPTR) ; and restore it in file entry
rts
; ----------------------------------------------------------------------------
; get volume directory first block ($0002)
; and initialize values for processing
; returns carry clear if no error, set otherwise
ReadVolDir1
lda #$02
sta DIRBLKL ; initialize block number low byte
stz DIRBLKH ; and high byte
jsr ReadVolDir ; read volume directory block
bcs :done ; done if error
lda BLOCKBUF+$23 ; get entry length
sta ENTLEN ; and save it
lda BLOCKBUF+$24 ; get entries per block
sta ENTPERBLK ; and save it
; intial values $02/03 = $102B (offset of first file entry in first VD block)
lda #<BLOCKBUF+$2b ; MERLIN32: removed ()
sta FENTPTRL
lda #>BLOCKBUF+$2b ; MERLIN32: removed ()
sta FENTPTRH
lda #$02 ; on first block, start with entry #2
sta CURENT
clc
:done rts
; ----------------------------------------------------------------------------
; Update pointer to the next directory file entry
; read the next block if necessary
; returns carry clear if no error, set otherwise
NextFileEnt
; add entry length to current entry pointer at ($02)
clc
lda FENTPTRL
adc ENTLEN
sta FENTPTRL
lda FENTPTRH
adc #$00
sta FENTPTRH
inc CURENT ; increment current entry number
lda CURENT ; and load it
cmp ENTPERBLK ; did the last one in block?
bcc :Okay ; nope, exit
; check if last block of Volume Directory
lda DIRBLKL ; block low byte
bne :NxtBlk ; another one to read
lda DIRBLKH ; block high byte
beq :Fail ; also zero, exit with error
; read next block of Volume Directory
:NxtBlk jsr ReadVolDir ; next volume directory block
bcs :Fail ; read problem, exit with error
lda #<BLOCKBUF+$04 ; now set entry pointer to the offset of the first MERLIN32: removed ()
sta FENTPTRL ; one in the block
lda #>BLOCKBUF+$04 ; ... MERLIN32: removed()
sta FENTPTRH ; ...
lda #$01 ; and set current entry
sta CURENT ; to 1
:Okay clc
rts
:Fail sec
rts
; ----------------------------------------------------------------------------
; read volume directory block at ($07)
; and update to point to next block
; returns carry clear if no error, set otherwise
ReadVolDir
; most recent accessed device into param list for READ_BLOCK
lda $BF30 ; most recent access device
sta :PL_READ_BLOCK+1 ; into READ_BLOCK parameter list
; copy block number to param list
lda DIRBLKL ; copy block number
sta :PL_READ_BLOCK+4 ; to parameter list
lda DIRBLKH ; ...
sta :PL_READ_BLOCK+5 ; ...
jsr PRODOS ; now get the block
DB PRODOS_READ_BLOCK ; READ_BLOCK
DW :PL_READ_BLOCK
bcs :Done ; error, bail out
lda BLOCKBUF+$02 ; otherwise get next block number
sta DIRBLKL ; from offset $02/$03
lda BLOCKBUF+$03 ; and save it
sta DIRBLKH ; ...
:Done rts
; Parameter list for READ_BLOCK
:PL_READ_BLOCK
DB $03 ; param count
DB $00 ; unit number
DW BLOCKBUF ; data buffer
DW $0000 ; block number
; ----------------------------------------------------------------------------
; enable slinky registers, set adddress and save byte we intend to trash
SlinkyEnable
lda C8OFF ; not needed on //c, but release $C8xx firmware
lda SLOT4ROM ; enable slinky registers
lda #$08 ; set addr $080000
sta DPTRH
stz DPTRM
stz DPTRL
lda DATA ; read data byte
sta SAVEBYTE ; save it to restore later
rts
; ----------------------------------------------------------------------------
; Routine to restore trashed byte in slinky RAM
; WARNING: never called by unfixed, so the value is never restored
SlinkyRestore
lda #$08 ; set adddr $080000
sta DPTRH
stz DPTRM
stz DPTRL
lda SAVEBYTE ; get saved byte
sta DATA ; and put it back
lda C8OFF ; not needed on //c, but release $C8xx firmware
rts
; ----------------------------------------------------------------------------
; Write 8 bits to clock
; WARNING: the unfixed code has a bug that trashes 8 bytes in the RAM card at $080000
ClockWrite8b
ldx #$08 ; set adddr $080000
IF FIX_BUGS
stx DPTRH
stz DPTRM
:_1 stz DPTRL ; restore low byte to 0
sta DATA ; write byte
lsr a ; next bit into 0 position
dex
bne :_1 ; MERLIN32: uniq label
ELSE
stz DPTRL
stz DPTRM
stx DPTRH
:_2 sta DATA ; write byte
lsr a ; next bit into 0 position
dex
bne :_2 ; MERLIN32: uniq label
FIN
rts
; ----------------------------------------------------------------------------
; unlock the clock by writing the magic bit sequence
ClockUnlock
ldy #$08
:_ lda unlock,y
jsr ClockWrite8b ; write 8 bits
dey
bne :_
rts
unlock = * - 1
DB $5c, $a3, $3a, $c5, $5c, $a3, $3a, $c5
; ----------------------------------------------------------------------------
; Read 8 bits from the clock
ClockRead8b
ldx #$08 ; set adddr $080000
stz DPTRL
stz DPTRM
stx DPTRH
:_ pha ; save accumulator
lda DATA ; get data byte
lsr a ; bit 0 into carry
pla ; restore accumulator
ror a ; put read bit into position
dex
bne :_
rts
; ----------------------------------------------------------------------------
; read the clock data into memory at CLOCKBUF
; WARNING: unfixed code never restores byte we trashed
ClockRead
jsr SlinkyEnable
jsr ClockUnlock
ldy #$00
:_ jsr ClockRead8b
sta CLOCKBUF,y
iny
cpy #$08 ; have we read 8 bytes?
bcc :_ ; nope
IF FIX_BUGS
jsr SlinkyRestore
FIN
rts
; ----------------------------------------------------------------------------
; validate the DClock data makes sense
; return carry clear if it does, carry set if it does not
ValidTime
; validate ms
ldx #$00
ldy #$99
lda CLOCKBUF
jsr CheckBCD
bcs :bad
; validate seconds
ldx #$00
ldy #$59
lda CLOCKBUF+$01
jsr CheckBCD
bcs :bad
; validate minutes
ldx #$00
ldy #$59
lda CLOCKBUF+$02
jsr CheckBCD
bcs :bad
; validate hours
ldx #$00
ldy #$23
lda CLOCKBUF+$03
jsr CheckBCD
bcs :bad
; validate day of week
ldx #$01
ldy #$07
lda CLOCKBUF+$04
jsr CheckBCD
bcs :bad
; validate day of month
ldx #$01
ldy #$31
lda CLOCKBUF+$05
jsr CheckBCD
bcs :bad
; validate month
ldx #$01
ldy #$12
lda CLOCKBUF+$06
jsr CheckBCD
bcs :bad
; validate year
ldx #$00
ldy #$99
lda CLOCKBUF+$07
jsr CheckBCD
bcs :bad
clc ; all good
rts
:bad sec ; problem
rts
; ----------------------------------------------------------------------------
; Check BCD number in range of [x,y]
; return carry clear if it is, carry set if it is not
CheckBCD
sed ; decimal mode
stx SCRATCH ; lower bound into scratch
cmp SCRATCH ; compare it
bcc :bad ; fail if out of range
sty SCRATCH ; upper bound into scratch
cmp SCRATCH ; compare it
beq :good ; OK if equal
bcs :bad ; fail if out of range
:good cld ; in range
clc
rts
:bad cld ; not in range
sec
rts
; ----------------------------------------------------------------------------
; This code segment is relocated to RELOCT at the start of program
; ----------------------------------------------------------------------------
RELOCS = * - 1
rseg1bm = *
ORG RELOCT
rseg1b = *
; ----------------------------------------------------------------------------
; This routine attempts to execute the next system file that we identified
; in the main code
LaunchNext
jsr PRODOS
DB PRODOS_OPEN ; OPEN
DW :PL_OPEN
bcs :LaunchFail ; if error
lda :PL_OPEN+5 ; copy reference number
sta :PL_GET_EOF+1 ; to parm list for GET_EOF
sta :PL_READ+1 ; and parm list for READ
sta :PL_CLOSE+1 ; and parm list for CLOSE
jsr PRODOS
DB PRODOS_GET_EOF ; GET_EOF
DW :PL_GET_EOF
bcs :LaunchFail ; if error
lda :PL_GET_EOF+4 ; high byte of length
bne :LaunchFail ; bigger than 64K... sheesh
lda :PL_GET_EOF+2 ; copy EOF middle and low
sta :PL_READ+4 ; to read reqest count
lda :PL_GET_EOF+3 ; ...
sta :PL_READ+5 ; ...
jsr PRODOS ; and do read
DB PRODOS_READ ; READ
DW :PL_READ
bcs :LaunchFail ; if error
jsr PRODOS
DB PRODOS_CLOSE ; CLOSE
DW :PL_CLOSE
bcs :LaunchFail ; if error
jmp SYSEXEC ; execute system file
:LaunchFail
pha ; save accumulator
jsr iprint ; because this trashes it
STR 'Can',27,'t start next SYS file: error #' ; MERLIN32: removed length $22
pla ; restore accumulator
jsr PrHex ; print error code
lda #$8D
jsr COUT1
jmp DoQuit ; quit to ProDOS
; parameter list for OPEN
:PL_OPEN
DB $03 ; param count
DW PATHBUF ; pathname address
DW READBUF ; I/O buffer
DB $00 ; reference number
; parameter list for GET_EOF
:PL_GET_EOF
DB $02 ; param count
DB $00 ; ref number
DB $00,$00,$00 ; EOF
; parameter list for READ
:PL_READ
DB $04 ; param count
DB $00 ; ref number
DW SYSEXEC ; data buffer
DW $0000 ; request count
DW $0000 ; transfer count
; parameter list for CLOSE
:PL_CLOSE
DB $01 ; param count
DB $00 ; ref number
; ----------------------------------------------------------------------------
; Routine to print hex number via LUT
; the firmware has a perfectly good routine for this, not sure why
; AE engineers put this in here. Nice wheel, though.
PrHex
pha
lsr a
lsr a
lsr a
lsr a
tax
lda :_,x
jsr COUT1
pla
and #$0F
tax
lda :_,x
jsr COUT1
rts
:_ ASC '0123456789ABCDEF'
; ----------------------------------------------------------------------------
; Execute QUIT call after long delay, exit via RTS if QUIT fails
DoQuit
ldx #$0C
:_ lda #$FF
jsr WAIT
dex
bne :_
jsr PRODOS
DB PRODOS_QUIT
DW :PL_QUIT
rts
; parameter list for QUIT
:PL_QUIT
DB $04 ; param count
DB $00 ; quit type - $00 is only type
DW $0000 ; reserved
DB $00 ; reserved
DW $0000 ; reserved
; unnecessary fill?
brk
brk
; ----------------------------------------------------------------------------
; 'inline print'
; print counted string following JSR
iprint
; get pointer into $09/$0a
pla
clc
adc #$01
sta CHRPTRL
pla
adc #$00
sta CHRPTRH
lda (CHRPTR)
beq :done
; now print the string
tay
ldy #$01
:ploop lda (CHRPTR),y
ora #$80
jsr COUT1
tya
iny
cmp (CHRPTR)
bcc :ploop
; done, fix up stack and exit
:done clc
adc CHRPTRL
sta CHRPTRL
lda #$00
adc CHRPTRH
pha
lda CHRPTRL
pha
rts
; ----------------------------------------------------------------------------
; end of relocated code
rseg1e = *
; fix up org address so we get a clean ref to rclkdrv
; ORG rseg1bm + (rseg1e - rseg1b + 1) ; MERLIN32: remove ()
ORG rseg1bm + rseg1e - rseg1b + 1 ; MERLIN32: remove ()
; ----------------------------------------------------------------------------
; clock driver code inserted into ProDOS at $D742 - this shouldn't be
; hard-coded but it is.
; critical bug: Corrupts byte in $08xx01 of the //c memory expansion card.
rclkdrv = *
ORG CLKCODE
clockdrv
begin = *
lda #$08 ; useless instruction
php
sei
lda SLOT4ROM ; activate slinky registers
; ($08 from above overwritten)
stz DPTRL ; set slinky address to $08xx00
ldy #$08 ; also counter for unlock bytes
sty DPTRH
lda DATA ; get destroyed byte
; (slinky now at $08xx01)
pha ; save value on stack
; unlock dclock registers
:ubytlp lda regulk,y
ldx #$08 ; bit counter
:ubitlp ; MERLIN32: moved outside conditional
IF FIX_BUGS
stz DPTRL ; reset pointer to $08xx00
sta DATA ; write to $08xx00
ELSE
sta DATA ; write to $08xx00
; (!but not on first iteration!)
stz DPTRL ; reset pointer to $08xx00
FIN
lsr a ; next bit into 0 position
dex
bne :ubitlp
dey
bne :ubytlp
; now read 64 bits (8 bytes) from dclock
ldx #$08 ; byte counter
:rbytlp ldy #$08 ; bit counter
:rbitlp pha
lda DATA ; data byte
lsr a ; bit 0 into carry
pla
ror a ; carry into bit 7
dey
bne :rbitlp
; got 8 bits now, convert from BCD to binary
pha
and #$0F
sta BCDTMP
pla
and #$F0
lsr a
pha
adc BCDTMP
sta BCDTMP
pla
lsr a
lsr a
adc BCDTMP
; place in input buffer, which is OK because the ThunderClock driver does this
sta INBUF-1,x
dex
bne :rbytlp
; done copying, now put necessary values into ProDOS time locations
; copy hours to ProDOS hours
lda INBUF+4
sta TIMEHI
; copy minutes to ProDOS minutes
lda INBUF+5
sta TIMELO
; copy month ...
lda INBUF+1
lsr a
ror a
ror a
ror a
; ... and day of month to ProDOS month/day
ora INBUF+2
sta DATELO
; copy year and final bit of month to ProDOS year/month
lda INBUF
rol a
sta DATEHI
stz DPTRL ; set slinky back to $08xx00
pla ; get saved byte
sta DATA ; put it back
plp
rts
; DS1215 unlock sequence (in reverse)
regulk = * - 1
DB $5C, $A3, $3A, $C5, $5C, $A3, $3A, $C5
end = *
;.assert (begin - end + 1) < CLKCODEMAX, error, "DCLOCK driver too big"