Applecorn/mainmem.svc.s
2022-12-14 23:09:17 -05:00

1260 lines
45 KiB
ArmAsm

* MAINMEM.SVC.S
* (c) Bobbi 2021 GPLv3
*
* Main memory entry points called by Applecorn MOS running in
* aux memory. Each entry point performs some ProDOS service,
* then returns to aux memory.
* 12-Oct-2021 OSFIND exits with bad filename, allows OPENIN(dir),
* exits with MFI error, returns error if no more buffers,
* OPENOUT doesn't try to delete if nothing to delete.
* 13-Oct-2021 OSFIND implementes CLOSE#0.
* 13-Oct-2021 FIND, BGET, BPUT optimised passing registers to main.
* 13-Oct-2021 ARGS, EOF returns errors, optimised.
* 15-Oct-2021 LOADFILE updated.
* 16-Oct-2021 LOADFILE only reads object info once.
* 17-Oct-2021 SAVEFILE updated.
* 18-Oct-2021 Optimised CREATE, removed dead code, RDDATA and WRDATA.
* 23-Oct-2021 Moved all the OSFILE routines together.
* Optimised entry and return from OSFILE routines.
* DELETE returns 'Dir not empty' when appropriate.
* 29-Oct-2021 DRVINFO reads current drive if "".
* 01-Nov-2021 DRVINFO checks reading info on a root directory.
* 02-Nov-2021 SETPERMS passed parsed access byte.
* 03-Nov-2021 Optimised CAT/EX/INFO, DESTROY.
* *BUG* RENAME won't rename between directories, eg RENAME CHARS VDU/CHARS.
* ProDOS file handling to rename a file
RENFILE >>> ENTMAIN
JSR PREPATH ; Preprocess arg1
JSR MFtoTMP ; Stash arg1
JSR COPYMF21 ; Copy arg2
JSR PREPATH ; Preprocess arg2
JSR COPYMF12 ; Put it back in MOSFILE2
JSR TMPtoMF ; Recover arg1->MOSFILE
RENFILE2 LDA #<MOSFILE
STA RENPL+1
LDA #>MOSFILE
STA RENPL+2
LDA #<MOSFILE2
STA RENPL+3
LDA #>MOSFILE2
STA RENPL+4
JSR MLI
DB RENCMD
DW RENPL
>>> XF2AUX,RENRET
* Set name of volume dir, for *TITLE
* Special case of renaming
* Case 1: Disk ID in MOSFILE, title in MOSFILE2
* Case 2: Title in MOSFILE, MOSFILE2=""
*
DISKTITLE >>> ENTMAIN
LDA MOSFILE2
BNE :CASE1
JSR COPYMF12 ; Copy title->MOSFILE2
LDA #$01 ; MOSFILE2 = "@"
STA MOSFILE1
LDA #'@'
STA MOSFILE1 + 1
:CASE1 JSR PREPATH ; Preprocess pathname
BCS :EXIT
JSR TRUNCPATH ; Truncate to top level
JMP RENFILE2
:EXIT >>> XF2AUX,RENRET
* ProDOS file handling for file copying
* Returns with ProDOS error code in A
COPYFILE >>> ENTMAIN
JSR MFtoTMP ; Swap MOSFILE & MOSFILE2
JSR COPYMF21
JSR TMPtoMF2
JSR PREPATH ; Preprocess arg2 (in MOSFILE)
JSR WILDONE ; Handle any wildcards
JSR EXISTS ; See if destination exists
STA :DESTTYPE ; Stash for later
JSR MFtoTMP ; Swap MOSFILE & MOSFILE2 again
JSR COPYMF21
JSR TMPtoMF2
JSR PREPATH ; Preprocess arg1
SEC ; Force wildcard lookup
JSR WILDCARD ; Handle any wildcards in arg1
BCS :NONE
JSR HASWILD
BCC :MAINLOOP ; No wildcards in final segment
LDA :DESTTYPE ; Wildcards, check dest type
CMP #$02 ; Existing directory?
BNE :BADDEST ; If not, error
BRA :MAINLOOP ; Source: wildcard, dest: dir
:NOWILD
:MAINLOOP
LDA MOSFILE2 ; Length
STA :OLDLEN
JSR EXISTS ; See if source is file or dir
CMP #$02 ; Directory
BEQ :SKIP ; Skip directories
LDA :DESTTYPE ; Check dest type
CMP #$02 ; Existing directory?
BNE :NOTDIR
LDY MOSFILE2 ; Dest idx = length
LDA MOSFILE2,Y ; Get last char
CMP #'/' ; Is it slash?
BEQ :HASSLSH
LDA #'/' ; Add a '/' separator
STA MOSFILE2+1,Y
INY
:HASSLSH LDX #$00 ; Source id
:APPLOOP CPX MATCHBUF ; At end?
BEQ :DONEAPP
LDA MATCHBUF+1,X ; Appending MATCHBUF to MOSFILE2
STA MOSFILE2+1,Y
INX
INY
BRA :APPLOOP
:DONEAPP STY MOSFILE2 ; Update length
:NOTDIR LDA :DESTTYPE ; Recover destination type
JSR COPY1FILE ; Copy an individual file
BCS :COPYERR
:SKIP JSR WILDNEXT
BCS :NOMORE
LDA :OLDLEN ; Restore MOSFILE2
STA MOSFILE2
BRA :MAINLOOP
JSR CLSDIR
:EXIT >>> XF2AUX,COPYRET
:NONE JSR CLSDIR
LDA #$46 ; 'File not found'
BRA :EXIT
:BADDEST JSR CLSDIR
LDA #$5E ; 'Wildcards' error
BRA :EXIT
:NOMORE JSR CLSDIR
LDA #$00
BRA :EXIT
:COPYERR PHA
JSR CLSDIR
PLA
BRA :EXIT
:DESTTYPE DB $00
:OLDLEN DB $00
* Copy a single file
* Source is in MOSFILE, DEST in MOSFILE2
* Returns with carry set if error, carry clear otherwise
* Returns with ProDOS error code in A
* Buffer COPYBUF is used for the file copy, to avoid trashing
* directory block in RDBUF (when doing wilcard search)
COPY1FILE LDA #<MOSFILE
STA GINFOPL+1
STA OPENPL2+1
LDA #>MOSFILE
STA GINFOPL+2
STA OPENPL2+2
JSR GETINFO ; GET_FILE_INFO
BCS :ERR
LDA #<MOSFILE2
STA GINFOPL+1
STA DESTPL+1
LDA #>MOSFILE2
STA GINFOPL+2
STA DESTPL+2
JSR MLI ; DESTROY
DB DESTCMD
DW DESTPL
LDA #$07 ; Fix num parms in PL
STA GINFOPL
LDA #$C3 ; Default permissions
STA GINFOPL+3
JSR MLI ; Call CREATE with ..
DB CREATCMD ; .. PL from GET_FILE_INFO
DW GINFOPL
LDX #$0A ; Num parms back as we found it
STX GINFOPL
BCS :ERR ; Error creating dest file
LDA #$00 ; Look for empty slot
JSR FINDBUF
STX :BUFIDX1
JSR BUFADDR
BCS :ERR ; No I/O bufs available
STA OPENPL2+3
STY OPENPL2+4
JSR MLI
DB OPENCMD
DW OPENPL2
BCS :ERR ; Open error
BRA :S1
:ERR SEC ; Report error
RTS
:S1 LDA OPENPL2+5 ; File ref num
STA RDPLCP+1
LDX :BUFIDX1
STA FILEREFS,X ; Record the ref number
LDA #<MOSFILE2
STA OPENPL2+1
LDA #>MOSFILE2
STA OPENPL2+2
LDA #$00 ; Look for empty slot
JSR FINDBUF
STX :BUFIDX2
JSR BUFADDR
BCS :ERRCLS1 ; No I/O bufs available
STA OPENPL2+3
STY OPENPL2+4
JSR MLI
DB OPENCMD
DW OPENPL2
BCS :ERRCLS1
LDA OPENPL2+5 ; File ref num
STA WRTPLCP+1
LDX :BUFIDX2
STA FILEREFS,X ; Record the ref number
:MAINLOOP JSR MLI ; Read a block
DB READCMD
DW RDPLCP
BCC :RDOKAY
CMP #$4C ; Is it EOF?
BEQ :EOFEXIT
BRA :ERRCLS2 ; Any other error
:RDOKAY LDA RDPLCP+6 ; Trans count MSB
STA WRTPLCP+4 ; Request count MSB
LDA RDPLCP+7 ; Trans count MSB
STA WRTPLCP+5 ; Request count MSB
JSR MLI ; Write a block
DB WRITECMD
DW WRTPLCP
BCS :ERRCLS2 ; Write error
BRA :MAINLOOP
:EOFEXIT CLC ; No error
PHP
LDA #$00
PHA
:CLOSE2 LDA WRTPLCP+1 ; Close output file
STA CLSPL+1
JSR CLSFILE
LDX :BUFIDX2
STZ FILEREFS,X
:CLOSE1 LDA RDPLCP+1 ; Close input file
STA CLSPL+1
JSR CLSFILE
LDX :BUFIDX1
STZ FILEREFS,X
PLA
PLP
RTS
:ERRCLS1 SEC
PHP
PHA
BRA :CLOSE1
:ERRCLS2 SEC
PHP
PHA
BRA :CLOSE2
:BUFIDX1 DB $00
:BUFIDX2 DB $00
* ProDOS file handling for MOS OSFIND OPEN call
* Options in A: $40 'r', $80 'w', $C0 'rw'
OFILE >>> ENTMAIN
AND #$C0 ; Keep just action bits
PHA ; Preserve arg for later
JSR PREPATH ; Preprocess pathname
BCS :JMPEXIT1 ; Bad filename
PLA
PHA
CMP #$80 ; Is it "w"?
BEQ :NOWILD ; If so, no wildcards
JSR WILDONE ; Handle any wildcards
:NOWILD JSR EXISTS ; See if file exists ...
TAX
CMP #$02 ; ... and is a directory
BNE :NOTDIR
PLA ; Get action back
BPL :NOTDIR2 ; OPENIN(dir) allowed
LDA #$41 ; $41=Directory exists
PHA ; Balance PLA
:JMPEXIT1 PLA
:JMPEXIT JMP FINDEXIT
:NOTDIR PLA
:NOTDIR2 CMP #$80 ; Write mode
BNE :S1
TXA
BEQ :S0 ; No file, don't try to delete
JSR DODELETE
BCS FINDEXIT ; Abort if error
:S0 LDX #$00 ; LOAD=$0000
LDY #$00
JSR CREATEFILE
BCS FINDEXIT ; Abort if error
* Looking for a buffer should be done before creating a file
:S1 LDA #$00 ; Look for empty slot
JSR FINDBUF
STX BUFIDX
JSR BUFADDR
BCS NOBUFFS ; No empty slot (BUFIDX=FF)
STA OPENPL2+3
STY OPENPL2+4
LDA #<MOSFILE
STA OPENPL2+1
LDA #>MOSFILE
STA OPENPL2+2
JSR MLI
DB OPENCMD
DW OPENPL2
BCS FINDEXIT
LDA OPENPL2+5 ; File ref number
LDX BUFIDX
STA FILEREFS,X ; Record the ref number
FINDEXIT JSR CHKNOTFND ; Convert NotFound to $00
>>> XF2AUX,OSFINDRET
NOBUFFS LDA #$42 ; $42=File buffers full
BNE FINDEXIT
BUFIDX DB $00
* ProDOS file handling for MOS OSFIND CLOSE call
* ProDOS can do CLOSE#0 but we need to manually update FILEREFS
CFILE >>> ENTMAIN
LDX #$00 ; Prepare for one file
TYA ; File ref number
BNE :CFILE1 ; Close one file
LDX #$03 ; Loop through all files
:CFILE0 LDA FILEREFS,X
BEQ :CFILE3 ; Not open, try next
:CFILE1 PHX
PHA
STA CLSPL+1
JSR CLSFILE
BCS :CFILEERR ; Error occured during closing
PLA
JSR FINDBUF
BNE :CFILE2
LDA #$00
STA FILEREFS,X ; Release buffer
:CFILE2 PLX
:CFILE3 DEX
BPL :CFILE0 ; Loop to close all files
LDA #$00
BEQ FINDEXIT
:CFILEERR PLX ; Balance stack
PLX
BCS FINDEXIT
* ProDOS file handling for MOS OSGBPB call
* A=1 : Write bytes to disk, using new seq pointer
* A=2 : Write bytes to disk, ignoring seq pointer
* A=3 : Read bytes from disk, using new seq pointer
* A=4 : Read bytes from disk, ignoring seq pointer
* All others unsupported
GBPB >>> ENTMAIN
PHA
LSR A ; Test LSB of A
BCC :L1 ; A=2 or 4, no seek
LDA GBPBHDL ; File ref number
STA GMARKPL+1
LDY #$02 ; Copy file pointer
:L0 LDA GBPBPTR+0,Y
STA GMARKPL+2,Y
DEY
BPL :L0
JSR MLI ; Perform seek
DB SMARKCMD
DW GMARKPL
BCC :L1
JMP :ERR
:L1 PLA
PHA
CMP #$02 ; If A<=2 WRITE
BCC :WRITE
:READ LDA GBPBHDL ; File ref number
STA READPL2+1
JSR MLI ; Read one byte from file
DB READCMD
DW READPL2
BCS :ERR
LDA GBPBDAT+0
STA ZPMOS+0
LDA GBPBDAT+1
STA ZPMOS+1
LDA BLKBUF
STA WRCARDRAM ; Write to aux
STA (ZPMOS) ; Store byte in aux mem
STA WRMAINRAM ; Write to main again
BRA :UPDCB
:WRITE LDA #<BLKBUF ; Start of destination
STA A4L
LDA #>BLKBUF
STA A4H
LDA GBPBDAT ; Start & end of source buf
STA A1L ; (one byte buffer)
STA A2L
LDA GBPBDAT+1
STA A1H
STA A2H
:S0 CLC ; AUX -> Main
JSR AUXMOVE
LDA GBPBHDL ; File ref number
STA WRITEPL+1
LDA #$01 ; Write one byte
STA WRITEPL+4
LDA #$00
STA WRITEPL+5
JSR WRTFILE ; Write one byte to file
:UPDCB INC GBPBDAT+0 ; Increment data pointer
BNE :S1
INC GBPBDAT+1
:S1 INC GBPBPTR+0 ; Increment file pointer
BNE :S2
INC GBPBPTR+1
BNE :S2
INC GBPBPTR+2
:S2 LDA GBPBNUM+0 ; Decrement number of bytes
BNE :S3
LDA GBPBNUM+1
BEQ :ZERO ; Zero remaining, done!
DEC GBPBNUM+1
:S3 DEC GBPBNUM+0
JMP :L1
:ERR
:ZERO PLA ; Throw away A
>>> ALTZP ; Control block can be in ZP!
STA WRCARDRAM ; Write to aux
LDA GBPBAUXCB+0 ; Copy control block back to aux
STA $B0+0 ; $B0 in AltZP is temp FS workspace
LDA GBPBAUXCB+1
STA $B0+1
LDY #$0C
:L2 LDA GBPBBLK,Y
STA ($B0),Y
DEY
BPL :L2
STA WRMAINRAM ; Write to main again
>>> MAINZP
>>> XF2AUX,OSGBPBRET
* ProDOS file handling for MOS OSBGET call
* Returns with char read in A and error num in Y (or 0)
FILEGET >>> ENTMAIN
STY READPL2+1 ; File ref number
JSR MLI
DB READCMD
DW READPL2
TAY ; Error number in Y
BCS :EXIT
LDY #$00 ; 0=Ok
LDA BLKBUF
:EXIT >>> XF2AUX,OSBGETRET
* ProDOS file handling for MOS OSBPUT call
* Enters with char to write in A
FILEPUT >>> ENTMAIN
STA BLKBUF ; Byte to write
STY WRITEPL+1 ; File ref number
LDA #$01 ; Bytes to write
STA WRITEPL+4
LDA #$00
STA WRITEPL+5
JSR WRTFILE
BCS :FILEPUT2
LDA #$00 ; 0=Ok
:FILEPUT2 >>> XF2AUX,OSBPUTRET
* ProDOS file handling for FSC $01 called by OSBYTE $7F EOF
* Returns EOF status in A ($FF for EOF, $00 otherwise)
* A=channel to test
FILEEOF >>> ENTMAIN
STA GEOFPL+1
STA GMARKPL+1
JSR MLI
DB GEOFCMD
DW GEOFPL
TAY
BCS :EXIT ; Abort with any error
JSR MLI
DB GMARKCMD
DW GMARKPL
TAY
BCS :EXIT ; Abort with any error
SEC
LDA GEOFPL+2 ; Subtract Mark from EOF
SBC GMARKPL+2
STA GEOFPL+2
LDA GEOFPL+3
SBC GMARKPL+3
STA GEOFPL+3
LDA GEOFPL+4
SBC GMARKPL+4
STA GEOFPL+4
LDA GEOFPL+2 ; Check bytes remaining
ORA GEOFPL+3
ORA GEOFPL+4
BEQ :ISEOF ; EOF -> $00
LDA #$FF ; Not EOF -> $FF
:ISEOF EOR #$FF ; EOF -> $FF, Not EOF ->$00
LDY #$00 ; 0=No error
:EXIT >>> XF2AUX,CHKEOFRET
* ProDOS file handling for OSARGS flush commands
FLUSH >>> ENTMAIN
STY FLSHPL+1 ; File ref number
JSR MLI
DB FLSHCMD
DW FLSHPL
JMP TELLEXIT
* ProDOS file handling for OSARGS set ptr command
* GMARKPL+1=channel, GMARKPL+2,+3,+4=offset already set
SEEK >>> ENTMAIN
JSR MLI
DB SMARKCMD
DW GMARKPL
JMP TELLEXIT
* ProDOS file handling for OSARGS get ptr command
* and for OSARGs get length command
* A=ZP, Y=channel
SIZE LDX #$02 ; $02=SIZE, Read EXT
BNE TELL2
TELL LDX #$00 ; $00=TELL, Read PTR
TELL2 STY GMARKPL+1 ; File ref number
PHA ; Pointer to zero page
CPX #$00 ; OSARGS parameter
BEQ :POS
JSR MLI
DB GEOFCMD
DW GMARKPL ; MARK parms same as EOF parms
BRA :S1
:POS JSR MLI
DB GMARKCMD
DW GMARKPL
:S1 PLX ; Pointer to ZP control block
BCS TELLEXIT ; Exit with error
>>> ALTZP ; Alt ZP & Alt LC on
LDA GMARKPL+2
STA $00,X
LDA GMARKPL+3
STA $01,X
LDA GMARKPL+4
STA $02,X
STZ $03,X ; Sizes are $00xxxxxx
>>> MAINZP ; Alt ZP off, ROM back in
LDA #$00 ; 0=Ok
TELLEXIT >>> XF2AUX,OSARGSRET
ZPMOS EQU $30
* ProDOS file MOS OSFILE calls
CALLFILE >>> ENTMAIN
JSR FILEDISPATCH
>>> XF2AUX,OSFILERET
FILEDISPATCH CMP #$00
BEQ SVCSAVE ; A=00 -> SAVE
CMP #$FF
BEQ SVCLOAD ; A=FF -> LOAD
CMP #$06
BEQ DELFILE ; A=06 -> DELETE
BCC INFOFILE ; A=01-05 -> INFO
CMP #$08
BEQ MAKEDIR ; A=08 -> MKDIR
RTS
SVCSAVE JMP SAVEFILE
SVCLOAD JMP LOADFILE
INFOFILE JSR UPDPATH ; Process path and get info
JMP COPYFB ; Copy back to aux mem
* ProDOS file handling to delete a file
* Called by AppleMOS OSFILE
* Return A=0 no object, A=1 file deleted, A=2 dir deleted
* A>$1F ProDOS error
DELFILE JSR UPDPATH ; Process path and get info
JSR COPYFB ; Copy back to aux mem
PHA ; Save object type
JSR DODELETE
BCC :DELETED ; Success
TAX ; X=error code
PLA ; Get object back
CPX #$4E
BNE :DELERROR ; Not 'Insuff. access', return it
LDX #$4F ; Change to 'Locked'
CMP #$02
BNE :DELERROR ; Wasn't a directory, return 'Locked'
LDA FBATTR+0
AND #$08
BNE :DELERROR ; Dir locked, return 'Locked'
LDX #$54 ; Change to 'Dir not empty'
:DELERROR TXA
JSR CHKNOTFND
PHA
:DELETED PLA ; Get object back
:EXIT RTS
DODELETE LDA #<MOSFILE ; Attempt to destroy file
STA DESTPL+1
LDA #>MOSFILE
STA DESTPL+2
JSR MLI
DB DESTCMD
DW DESTPL
RTS
* ProDOS file handling to create a directory
* Invoked by AppleMOS OSFILE
* Return A=$02 on success (ie: 'directory')
* A>$1F ProDOS error, translated by OSFILE handler
MAKEDIR JSR UPDPATH ; Process path and get info
CMP #$02
BEQ :EXIT1 ; Dir already exists
LDA #$0D ; OBJT='Directory'
STA CREATEPL+7 ; ->Storage type
LDA #$0F ; TYPE='Directory'
LDX #$00 ; LOAD=$0000
LDY #$00
JSR CREATEOBJ
BCS :EXIT ; Failed, exit with ProDOS result
JSR UPDFB ; Update FILEBLK, returns A=$02
:EXIT1 JSR COPYFB ; Copy FILEBLK to aux mem
:EXIT RTS
* ProDOS file handling for MOS OSFILE LOAD call
* Invoked by AppleMOS OSFILE
* Return A=01 if successful (meaning 'file')
* A>$1F ProDOS error, translated by FILERET
LOADFILE LDX #4
:LP LDA FBLOAD,X ; Get address to load to
STA ZPMOS,X
DEX
BPL :LP
JSR PREPATH ; Preprocess pathname
JSR WILDONE ; Handle any wildcards
JSR UPDFB ; Get object info
CMP #$20
BCS :JMPEXIT ; Error occured
CMP #$01 ; Is it a file
BEQ :ISFILE
ROL A ; 0->0, 2->5
EOR #$05 ; 0->5, 2->0
ADC #$41 ; 0->$46, 2->$41
:JMPEXIT JMP :EXIT2 ; Return error
:ISFILE LDA ZPMOS+4 ; If FBEXEC is zero, use addr
BEQ :CBADDR ; in the control block
LDA FBLOAD+0 ; Otherwise, use file's address
STA ZPMOS+0
LDA FBLOAD+1
STA ZPMOS+1
:CBADDR JSR OPENMOSFILE
BCS :EXIT2 ; File not opened
:L1 LDA OPENPL+5 ; File ref number
JSR READDATA ; Read data from open file
PHA ; Save result
LDA OPENPL+5 ; File ref num
STA CLSPL+1
JSR CLSFILE
PLA
BNE :EXIT2
JSR COPYFB ; Copy FILEBLK to auxmem
LDA #$01 ; $01=File
:EXIT2 RTS
* A=channel, MOSZP+0/1=address to load to, TO DO: MOS+4/5=length to read
READDATA STA READPL+1
:RDLP JSR RDFILE
BCS :READERR ; Close file and return any error
LDA #<BLKBUF ; LSB of start of data buffer
STA A1L ; A1=>start of data buffer
ADC READPL+6 ; LSB of trans count
TAX ; X=>LSB end of data buffer
LDA #>BLKBUF ; MSB of start of data buffer
STA A1H ; A1=>start of data buffer
ADC READPL+7 ; MSB of trans count
TAY ; Y=>MSB end of data buffer
TXA
BNE :L2
DEY
:L2 DEX ; XY=XY-1, end address is start+len-1
STX A2L ; A2=>end of data buffer
STY A2H
LDA ZPMOS+0 ; A4=>address to load to
STA A4L
LDA ZPMOS+1
STA A4H
INC ZPMOS+1 ; Step to next block
INC ZPMOS+1
SEC ; Main -> AUX
JSR AUXMOVE ; A4 updated to next address
JMP :RDLP
:READERR CMP #$4C
BNE :EXITERR
:EXITOK LDA #$00 ; $00=Success
:EXITERR RTS
* ProDOS file handling for MOS OSFILE SAVE call
* Invoked by AppleMOS OSFILE
* Return A=01 if successful (ie: 'file')
* A>$1F ProDOS error translated by FILERET
SAVEFILE SEC ; Compute file length
LDA FBEND+0
SBC FBSTRT+0
STA LENREM+0
LDA FBEND+1
SBC FBSTRT+1
STA LENREM+1
LDA FBEND+2
SBC FBSTRT+2
BNE :TOOBIG ; >64K
LDA FBEND+3
SBC FBSTRT+3
BEQ :L0 ; >16M
:TOOBIG LDA #$2C ; Bad byte count - file too long
RTS
:L0 JSR PREPATH ; Preprocess pathname
JSR EXISTS ; See if file exists ...
CMP #$01
BEQ :NOTDIR ; Overwrite file
BCC :NOFILE ; Create new file
CMP #$02
BNE :JMPEXIT2
LDA #$41 ; Dir exists, return $41
:JMPEXIT2 JMP :EXIT2
:NOTDIR LDA #<MOSFILE ; Attempt to destroy file
STA DESTPL+1
LDA #>MOSFILE
STA DESTPL+2
JSR MLI
DB DESTCMD
DW DESTPL
BCS :EXIT2 ; Error trying to delete
:NOFILE LDX FBLOAD+0 ; Auxtype = load address
LDY FBLOAD+1
JSR CREATEFILE
BCS :JMPEXIT2 ; Error trying to create
JSR OPENMOSFILE
BCS :JMPEXIT2 ; Error trying to open
LDA OPENPL+5 ; File ref number
JSR WRITEDATA
:EXIT1 PHA ; Save result
LDA OPENPL+5 ; File ref num
STA CLSPL+1
JSR CLSFILE
PLA
BNE :EXIT2 ; Error returned
JSR UPDFB ; Update FILEBLK
JSR COPYFB ; Copy FILEBLK to aux mem
LDA #$01 ; Return A='File'
:EXIT2 CMP #$4E
BNE :EXIT3 ; Change 'Insuff. access'
LDA #$4F ; to 'Locked'
:EXIT3 RTS
* A=channel, FBSTRT+0/1=address to save from
* LENREM+0/1=length to write
WRITEDATA STA WRITEPL+1
:L1 LDA #$00 ; 512 bytes request count
STA WRITEPL+4
LDA #$02
STA WRITEPL+5
LDA LENREM+1
CMP #$02
BCS :L15 ; More than 511 bytes remaining
STA WRITEPL+5
LDA LENREM+0
STA WRITEPL+4
ORA WRITEPL+5
BEQ :SAVEOK ; Zero bytes remaining
:L15 SEC
LDA LENREM+0 ; LENREM=LENREM-count
SBC WRITEPL+4
STA LENREM+0
LDA LENREM+1
SBC WRITEPL+5
STA LENREM+1
CLC
LDA FBSTRT+0
STA A1L ; A1=>start of this block
ADC WRITEPL+4
STA FBSTRT+0 ; Update FBSTRT=>start of next block
TAX ; X=>end of this block
LDA FBSTRT+1
STA A1H
ADC WRITEPL+5
STA FBSTRT+1
TAY
TXA
BNE :L2
DEY
:L2 DEX ; XY=XY-1, end address is start+len-1
STX A2L ; A2=>end of data buffer
STY A2H
:S2 LDA #<BLKBUF
STA A4L
LDA #>BLKBUF
STA A4H
CLC ; Aux -> Main
JSR AUXMOVE ; Copy data from aux to local buffer
JSR WRTFILE ; Write the data
BCS :WRITEERR
JMP :L1 ; Loop back for next block
:SAVEOK ; Enter here with A=$00
:WRITEERR RTS
LENREM DW $0000 ; Remaining length
CREATEFILE LDA #$01 ; Storage type - file
STA CREATEPL+7
LDA #$06 ; Filetype BIN
CREATEOBJ STA CREATEPL+4 ; Type = BIN or DIR
STX CREATEPL+5 ; Auxtype = load address
STY CREATEPL+6
JMP CRTFILE
* Process pathname and read object info
UPDPATH JSR PREPATH ; Process pathname
BCC UPDFB ; If no error, update control block
RTS
* Update FILEBLK before returning to aux memory
* Returns A=object type or ProDOS error
UPDFB LDA #<MOSFILE
STA GINFOPL+1
LDA #>MOSFILE
STA GINFOPL+2
JSR GETINFO ; Call GET_FILE_INFO
BCC :UPDFB1
JMP CHKNOTFND
:UPDFB1 LDA GINFOPL+5 ; Aux type LSB
STA FBLOAD
STA FBEXEC
LDA GINFOPL+6 ; Aux type MSB
STA FBLOAD+1
STA FBEXEC+1
STZ FBLOAD+2
STZ FBEXEC+2
STZ FBLOAD+3
STZ FBEXEC+3
*
LDA GINFOPL+3 ; Access byte
CMP #$40 ; Locked?
AND #$03 ; ------wr
PHP
STA FBATTR+0
ASL A ; -----wr-
ASL A ; ----wr--
ASL A ; ---wr---
ASL A ; --wr----
PLP
BCS :UPDFB2
ORA #$08 ; --wrl---
:UPDFB2 ORA FBATTR+0 ; --wrl-wr
STA FBATTR+0
*
LDA GINFOPL+11 ; yyyyyyym
PHA
ROR A ; ?yyyyyyy m
LDA GINFOPL+10 ; mmmddddd m
PHA
ROR A ; mmmmdddd
LSR A ; -mmmmddd
LSR A ; --mmmmdd
LSR A ; ---mmmmd
LSR A ; ----mmmm
STA FBATTR+2
PLA ; mmmddddd
AND #31 ; ---ddddd
STA FBATTR+1
PLA ; yyyyyyym
SEC
SBC #81*2 ; Offset from 1981
BCS :UPDFB3 ; 1981-1999 -> 00-18
ADC #100*2 ; 2000-2080 -> 19-99
:UPDFB3 PHA ; yyyyyyym
AND #$E0 ; yyy-----
ORA FBATTR+1 ; yyyddddd
STA FBATTR+1
PLA ; yyyyyyym
AND #$FE ; yyyyyyy0
ASL A ; yyyyyy00
ASL A ; yyyyy000
ASL A ; yyyy0000
ORA FBATTR+2 ; yyyymmmm
STA FBATTR+2
STZ FBATTR+3
JSR OPENMOSFILE ; Open file
BCS :ERR
LDA OPENPL+5 ; File ref number
STA GMARKPL+1
JSR MLI ; Call GET_EOF MLI
DB GEOFCMD
DW GMARKPL ; MARK parms same as EOF
LDA GMARKPL+2
STA FBSIZE+0
LDA GMARKPL+3
STA FBSIZE+1
LDA GMARKPL+4
STA FBSIZE+2
STZ FBSIZE+3
LDA OPENPL+5 ; File ref number
STA CLSPL+1
JSR CLSFILE
LDA #$01 ; Prepare A=file
LDX GINFOPL+7
CPX #$0D ; Is it a directory?
ADC #$00 ; Becomes A=2 for directory
:UPDFB5 RTS
:ERR
CHKNOTFND CMP #$44 ; Convert ProDOS 'not found'
BEQ :NOTFND ; into result=$00
CMP #$45
BEQ :NOTFND
CMP #$46
BNE :CHKNOTFND2
:NOTFND LDA #$00
:CHKNOTFND2 RTS
* Quit to ProDOS
QUIT JSR AUDIOSTOP ; Shut down audio, dereg ISR
INC PWRDUP ; Invalidate powerup byte
STA PAGE1 ; PAGE2 off
STA CLRALTCHAR ; Alt font off
JSR MLI
DB QUITCMD
DW QUITPL
RTS
* Used for *CAT, *EX and *INFO
* On entry: b7=0 - short info (*CAT)
* b7=1 - long info (*INFO, *EX)
* b6=0 - single entry, parameter is object (*INFO)
* b6=1 - multiple items, parameter is dir (*CAT, *EX)
*
CATALOG >>> ENTMAIN
STA CATARG ; Stash argument
CMP #$80 ; Is it *INFO?
BNE :NOTINFO
JMP INFO ; Handle entry for *INFO
:NOTINFO LDA MOSFILE ; Length of pathname
BEQ :NOPATH ; If zero use prefix
JSR PREPATH ; Preprocess pathname
JSR WILDONE ; Handle any wildcards
JSR EXISTS ; See if path exists ...
BEQ :NOTFND ; Not found
CMP #$02
BEQ :DIRFOUND
LDA #$0D ; Becomes Not a directory
:NOTFND EOR #$46 ; $00->$46, $xx->$4B
BNE CATEXIT
:NOPATH JSR GETPREF ; Fetch prefix into PREFIX
LDX #<PREFIX ; XY=>prefix
LDY #>PREFIX
BRA :OPEN
:DIRFOUND LDX #<MOSFILE ; XY=>specified directory
LDY #>MOSFILE
:OPEN STX OPENPL+1 ; Open the specified directory
STY OPENPL+2
JSR OPENFILE
BCS CATEXIT ; Can't open dir
CATREENTRY LDA OPENPL+5 ; File ref num
STA READPL+1
JSR RDFILE
BCS :CATERR
JSR COPYAUXBLK
>>> XF2AUX,PRONEBLK
:CATERR CMP #$4C ; EOF
BNE :NOTEOF
LDA #$00
:NOTEOF PHA
LDA OPENPL+5 ; File ref num
STA CLSPL+1
JSR CLSFILE
PLA
CATEXIT >>> XF2AUX,STARCATRET
* Handle *INFO
INFO JSR PREPATH ; Preprocess pathname
SEC
JSR WILDCARD ; Handle any wildcards
JSR EXISTS ; Check matches something
BNE INFOFIRST ; Match found, start listing
LDA #$46 ; No match, error Not found
INFOEXIT CMP #$4C ; EOF
BNE INFOCLS
LDA #$00 ; EOF is not an error
INFOCLS PHA
JSR CLSDIR ; Be sure to close it!
PLA
BRA CATEXIT
* PRONEBLK call returns here ...
CATALOGRET >>> ENTMAIN
LDA CATARG
CMP #$80 ; Is this an *INFO call?
BNE CATREENTRY ; No, go back and do another CAT/EX
INFOREENTRY JSR WILDNEXT2 ; Start of new block
BCS INFOEXIT ; No more matches
INFOFIRST LDA WILDIDX
CMP #$FF ; Is WILDNEXT about to read new blk?
BEQ :DONEBLK ; If so, print this blk first
JSR WILDNEXT2
BCC INFOFIRST ; Find more entries
:DONEBLK JSR COPYAUXBLK
>>> XF2AUX,PRONEBLK
CATARG DB $00
* Helper routine to truncate path in MOSFILE at the top level
* So '/FOO/BAR/BAZ' -> '/FOO'
TRUNCPATH LDX #$01 ; Skip over initial '/'
:L1 CPX MOSFILE ; See if end of string
BCS :EXIT ; If so ... we are done
BEQ :EXIT
LDA MOSFILE+1,X ; Get char from path
CMP #'/' ; See if it is slash
BNE :NOTSLASH
STX MOSFILE ; If so, truncate here
BRA :EXIT
:NOTSLASH INX
BRA :L1
:EXIT RTS
* Set prefix. Used by *CHDIR/*DRIVE to change directory
* Y= $00 - CHDIR, select any directory
* Y<>$00 - DRIVE, must select root
*
SETPFX >>> ENTMAIN
PHY ; Save CHDIR/DRIVE flag
JSR PREPATH ; Preprocess pathname
BCS :EXIT
JSR WILDONE ; Handle any wildcards
LDA #$2E
BCS :EXIT ; Exit with wildcard path
PLY ; CHDIR/DRIVE flag
CPY #$00 ; If <> $00 (ie: *DRIVE)
BEQ :CHDIR
JSR TRUNCPATH ; Truncate to top level
:CHDIR LDA #<MOSFILE
STA SPFXPL+1
LDA #>MOSFILE
STA SPFXPL+2
JSR MLI ; SET_PREFIX
DB SPFXCMD
DW SPFXPL
:EXIT >>> XF2AUX,CHDIRRET
* Obtain info on total/used blocks
DRVINFO >>> ENTMAIN
LDA MOSFILE
BNE :DRVINF2
INC MOSFILE
LDA #'@'
STA MOSFILE+1 ; Convert "" to "@"
:DRVINF2 JSR PREPATH
BCS :EXIT
JSR TRUNCPATH ; Truncate to top level
LDA #<MOSFILE
STA GINFOPL+1
LDA #>MOSFILE
STA GINFOPL+2
JSR GETINFO ; GET_FILE_INFO
BCS :EXIT
LDA GINFOPL+7
CMP #$0F
BNE :EXIT1 ; Not a drive, exit with 'Bad drive'
>>> ALTZP ; Alt ZP & Alt LC on
LDA GINFOPL+8 ; Blocks used LSB
STA AUXBLK+0
LDA GINFOPL+9 ; Blocks used MSB
STA AUXBLK+1
LDA GINFOPL+5 ; Total blocks LSB
STA AUXBLK+2
LDA GINFOPL+6 ; Total blocks MSB
STA AUXBLK+3
>>> MAINZP ; ALt ZP off, ROM back in
LDA #$00 ; $00=Ok
:EXIT CMP #$46
BNE :EXIT2
:EXIT1 LDA #$2A ; Change 'Not found' to 'Bad drive'
:EXIT2 >>> XF2AUX,FREERET
* Change file permissions, for *ACCESS
* Filename in MOSFILE, access mask in A
*
SETPERM >>> ENTMAIN
PHA ; Save access mask
JSR PREPATH ; Preprocess pathname
BCS :SYNERR
CLC
JSR WILDCARD ; Handle any wildcards
BCS :NONE
BCC :MAINLOOP
* STZ :LFLAG
* STZ :WFLAG
* STZ :RFLAG
* LDX MOSFILE2 ; Length of arg2
* INX
*:L1 DEX
* CPX #$00
* BEQ :MAINLOOP
* LDA MOSFILE2,X ; Read arg2 char
* CMP #'L' ; L=Locked
* BNE :S1
* STA :LFLAG
* BRA :L1
*:S1 CMP #'R' ; R=Readable
* BNE :S2
* STA :RFLAG
* BRA :L1
*:S2 CMP #'W' ; W=Writable
* BNE :ERR2 ; Bad attribute
* STA :WFLAG
* BRA :L1
:SYNERR LDA #$40 ; Invalid pathname syn
BRA :EXIT
:NONE JSR CLSDIR
LDA #$46 ; 'File not found'
BRA :EXIT
:MAINLOOP LDA #<MOSFILE
STA GINFOPL+1
LDA #>MOSFILE
STA GINFOPL+2
JSR GETINFO ; GET_FILE_INFO
BCS :EXIT
PLA ; Access byte
PHA
* LDA GINFOPL+3 ; Access byte
* AND #$03 ; Start with R, W off
* ORA #$C0 ; Start with dest/ren on
* LDX :RFLAG
* BEQ :S3
* ORA #$01 ; Turn on read enable
*:S3 LDX :WFLAG
* BEQ :S4
* ORA #$02 ; Turn on write enable
*:S4 LDX :LFLAG
* BEQ :S5
* AND #$3D ; Turn off destroy/ren/write
:S5 STA GINFOPL+3 ; Access byte
JSR SETINFO ; SET_FILE_INFO
JSR WILDNEXT
BCC :MAINLOOP
* BCS :NOMORE
:NOMORE JSR CLSDIR
LDA #$00
* BRA :EXIT
:EXIT PLX ; Drop access byte
>>> XF2AUX,ACCRET
:ERR2 LDA #$53 ; Invalid parameter
BRA :EXIT
*:LFLAG DB $00 ; 'L' attribute
*:WFLAG DB $00 ; 'W' attribute
*:RFLAG DB $00 ; 'R' attribute
* Multi file delete, for *DESTROY
* Filename in MOSFILE
*
MULTIDEL >>> ENTMAIN
JSR PREPATH ; Preprocess pathname
BCS :EXIT
* CLC ; CC already set
JSR WILDCARD ; Handle any wildcards
BCC :MAINLOOP
LDA #$46 ; 'File not found'
BRA :DELERR
:MAINLOOP JSR DODELETE
BCS :DELERR
JSR WILDNEXT
BCC :MAINLOOP ; More to do
LDA #$00 ; $00=Done
:DELERR PHA
JSR CLSDIR
PLA
:EXIT >>> XF2AUX,DESTRET
* Read machid from auxmem
MACHRD LDA ROMIN
LDA ROMIN
LDA $FBC0
SEC
JSR $FE1F
BRA MAINRDEXIT
* Read mainmem from auxmem
MAINRDMEM STA A1L
STY A1H
LDA ROMIN
LDA ROMIN
LDA (A1L)
MAINRDEXIT >>> XF2AUX,NULLRTS ; Back to an RTS