Apple2-IO-RPi/Apple2/DriveFirmware.asm

210 lines
3.8 KiB
NASM
Raw Normal View History

; Copyright Terence J. Boldt (c)2020-2023
2022-01-11 04:00:58 +00:00
; Use of this source code is governed by an MIT
; license that can be found in the LICENSE file.
; This file contains the source for the firmware
; that allows the Apple II to boot from the card
; and for ProDOS to recognize the card as two
; hard drivers
2020-12-13 04:26:16 +00:00
;ProDOS Zero Page
Command = $42 ;ProDOS Command
Unit = $43 ;ProDOS unit (SDDD0000)
BufferLo = $44
BufferHi = $45
BlockLo = $46
BlockHi = $47
; ProDOS Error Codes
IOError = $27
NoDevice = $28
WriteProtect = $2B
InputByte = $c08e+SLOT*$10
OutputByte = $c08d+SLOT*$10
InputFlags = $c08b+SLOT*$10
OutputFlags = $c087+SLOT*$10
2021-03-27 14:25:42 +00:00
2020-12-13 04:26:16 +00:00
ReadBlockCommand = $01
WriteBlockCommand = $02
2020-12-16 02:05:12 +00:00
GetTimeCommand = $03
2021-03-27 14:25:42 +00:00
ChangeDriveCommand = $04
ExecCommand = $05
LoadFileCommand = $06
SaveFileCommand = $07
2021-07-17 14:17:52 +00:00
MenuCommand = $08
2020-12-13 04:26:16 +00:00
2021-03-27 14:25:42 +00:00
.org SLOT*$100 + $C000
2020-12-13 04:26:16 +00:00
;ID bytes for booting and drive detection
cpx #$20 ;ID bytes for ProDOS and the
cpx #$00 ; Apple Autostart ROM
cpx #$03 ;
2021-03-27 14:25:42 +00:00
ldx #SLOT*$10
stx $2b
stx Unit
;force EPROM to second page on boot
lda #$3f ;set all flags high and page 3 of EPROM for menu
2021-03-27 14:25:42 +00:00
PageJump:
sta OutputFlags
2021-03-27 14:25:42 +00:00
jmp Start ;this jump is only called if coming in from PageJump with A=$0f
;entry points for ProDOS
DriverEntry:
lda #$0f ;set all flags high and page 0 of EPROM
sta OutputFlags
2021-03-27 14:25:42 +00:00
jmp Driver
2020-12-13 04:26:16 +00:00
;load first two blocks and execute to boot
2021-03-27 14:25:42 +00:00
Start:
2020-12-13 04:26:16 +00:00
lda #$01 ;set read command
sta Command
lda #$00 ;block 0
sta BlockLo
sta BlockHi
sta BufferLo ;buffer at $800
lda #$08
sta BufferHi
jsr Driver ;get the block
jmp $801 ;execute the block
; ProDOS Driver code
; First check that this is the right drive
Driver:
ldx Unit
lda Command; Check which command is being requested
beq GetStatus ;0 = Status command
cmp #ReadBlockCommand
2021-05-10 02:41:57 +00:00
beq ReadBlockAndSetTime
2020-12-13 04:26:16 +00:00
cmp #WriteBlockCommand
beq WriteBlock
sec ;set carry as we don't support any other commands
lda #$53 ;Invalid parameter error
rts
; ProDOS Status Command Handler
GetStatus:
ldx #$ff ;low byte number of blocks
ldy #$ff ;high byte number of blocks
lda #$0 ;zero accumulator and clear carry for success
clc
rts
; ProDOS Read Block Command
2021-05-10 02:41:57 +00:00
ReadBlockAndSetTime:
lda BlockHi ; only get the time if block 0002
bne readBlock
lda BlockLo
cmp #$02
bne readBlock
2020-12-16 02:05:12 +00:00
ldy #$00 ;Get the current time on each block read for now
lda #GetTimeCommand
jsr SendByte
getTimeByte:
jsr GetByte
sta $bf90,y
iny
cpy #$04
bne getTimeByte
2021-05-10 02:41:57 +00:00
readBlock:
2020-12-16 02:05:12 +00:00
lda #ReadBlockCommand ;read the block after setting the clock
2020-12-13 04:26:16 +00:00
jsr SendByte
lda BlockLo
jsr SendByte
lda BlockHi
jsr SendByte
2021-10-06 23:30:20 +00:00
txa
jsr SendByte
2020-12-13 04:26:16 +00:00
ldy #$0
jsr read256
inc BufferHi
jsr read256
dec BufferHi
lda #$0 ;zero accumulator and clear carry for success
clc
rts
read256:
jsr GetByte
sta (BufferLo),y
iny
bne read256
rts
; ProDOS Write Block Command
WriteBlock:
lda #WriteBlockCommand
jsr SendByte
lda BlockLo
jsr SendByte
lda BlockHi
jsr SendByte
2021-10-06 23:30:20 +00:00
txa
jsr SendByte
2020-12-13 04:26:16 +00:00
ldy #$0
jsr write256
inc BufferHi
jsr write256
dec BufferHi
lda #$0 ;zero accumulator and clear carry for success
clc
rts
write256:
lda (BufferLo),y
jsr SendByte
iny
bne write256
rts
SendByte:
pha
waitWrite:
lda InputFlags
2021-01-16 17:42:00 +00:00
rol
rol
2021-01-16 03:58:13 +00:00
bcs waitWrite
2020-12-13 04:26:16 +00:00
pla
sta OutputByte
2021-01-16 03:58:13 +00:00
lda #$0e ; set bit 0 low to indicate write started
sta OutputFlags
2020-12-13 04:26:16 +00:00
finishWrite:
lda InputFlags
2021-01-16 17:42:00 +00:00
rol
rol
2021-01-16 03:58:13 +00:00
bcc finishWrite
lda #$0f
sta OutputFlags
2020-12-13 04:26:16 +00:00
rts
GetByte:
2021-01-16 03:58:13 +00:00
lda #$0d ;set read flag low
sta OutputFlags
2020-12-13 04:26:16 +00:00
waitRead:
lda InputFlags
2021-01-16 17:42:00 +00:00
rol
2021-01-16 03:58:13 +00:00
bcs waitRead
lda InputByte
2020-12-13 04:26:16 +00:00
pha
2021-01-16 03:58:13 +00:00
lda #$0f ;set all flags high
sta OutputFlags
2020-12-13 04:26:16 +00:00
finishRead:
lda InputFlags
2021-01-16 17:42:00 +00:00
rol
2021-01-16 03:58:13 +00:00
bcc finishRead
2020-12-13 04:26:16 +00:00
pla
end:
rts
2021-10-06 23:30:20 +00:00
.repeat 251-<end
2020-12-13 04:26:16 +00:00
.byte 0
.endrepeat
2020-12-13 04:26:16 +00:00
.byte 0,0 ;0000 blocks = check status
.byte 23 ;bit set(0=status 1=read 2=write) unset(3=format, 4/5=number of volumes, 6=interruptable, 7=removable)
2021-03-27 14:25:42 +00:00
.byte DriverEntry&$00FF ;low byte of entry