Mercurial > hg > Members > kono > nitros9-code
view 3rdparty/utils/tlindner/sdrive.asm @ 3129:ce9f4ff0a380
l1 rel.asm: Do the module padding in a more portable way
Add comments to describe what's happening and why.
author | Neal Crook <foofoobedoo@gmail.com> |
---|---|
date | Wed, 30 Nov 2016 22:10:56 +0000 |
parents | dfc911234945 |
children |
line wrap: on
line source
******************************************************************** * sdrive - Assign disk image to SDC slot * * $Id$ * * Edt/Rev YYYY/MM/DD Modified by * Comment * ------------------------------------------------------------------ * 1 2014/11/30 tim lindner * Started writing code. nam sdrive ttl Assign disk image to SDC slot ifp1 use defsfile endc * Here are some tweakable options DOHELP set 0 1 = include help info STACKSZ set 32 estimated stack size in bytes PARMSZ set 256 estimated parameter size in bytes * Module header definitions tylg set Prgrm+Objct atrv set ReEnt+rev rev set $00 edition set 1 ********************************************************************* *** Hardware Addressing ********************************************************************* CTRLATCH equ $FF40 ; controller latch (write) CMDREG equ $FF48 ; command register (write) STATREG equ $FF48 ; status register (read) PREG1 equ $FF49 ; param register 1 PREG2 equ $FF4A ; param register 2 PREG3 equ $FF4B ; param register 3 DATREGA equ PREG2 ; first data register DATREGB equ PREG3 ; second data register ********************************************************************* *** STATUS BIT MASKS ********************************************************************* BUSY equ %00000001 READY equ %00000010 FAILED equ %10000000 mod eom,name,tylg,atrv,start,size org 0 slot rmb 1 cleartop equ . everything up to here gets cleared at start * Finally the stack for any PSHS/PULS/BSR/LBSRs that we might do rmb STACKSZ+PARMSZ size equ . * The utility name and edition goes here name fcs /sdrive/ fcb edition * Place constant strings here timeout fcc /Timeout./ fcb C$LF fcb C$CR timeoutL equ *-timout targetInUse fcc /Target in use./ fcb C$LF fcb C$CR targetInUseL equ *-targetInUse dirNotFound fcc /Directory not found./ fcb C$LF fcb C$CR dirNotFoundL equ *-dirNotFound pathNameInvalid fcc /Pathname is invalid./ fcb C$LF fcb C$CR pathNameInvalidL equ *-pathNameInvalid miscHardware fcc /Miscellaneous hardware error./ fcb C$LF fcb C$CR miscHardwareL equ *-miscHardware unknown fcc /Unknown error./ fcb C$LF fcb C$CR unknownL equ *-unknown targetNotFound fcc /Target not found./ fcb C$LF fcb C$CR targetNotFoundL equ *-targetNotFound IFNE DOHELP HlpMsg fcb C$LF fcc /Use: sdrive <slot> <image name>/ fcb C$LF fcb C$CR fcc / <slot> is a number, either 0 or 1./ fcb C$LF fcb C$CR fcc / <image name> is valid path and image name./ fcb C$LF fcb C$CR fcc / <image name> can be blank to eject an image./ fcb C$LF fcb C$CR HlpMsgL equ *-HlpMsg ENDC * * Here's how registers are set when this process is forked: * * +-----------------+ <-- Y (highest address) * ! Parameter ! * ! Area ! * +-----------------+ <-- X, SP * ! Data Area ! * +-----------------+ * ! Direct Page ! * +-----------------+ <-- U, DP (lowest address) * * D = parameter area size * PC = module entry point abs. address * CC = F=0, I=0, others undefined * The start of the program is here. * main program start cmpd #1 lbeq ShowHelp subb #1 sbca #0 clr d,x put null at end of parameter area (not CR) leas -3,s make some more room in parameter area lbsr SkipSpcs clr slot,u lda ,x+ cmpa #'0 beq slot0 cmpa #'1 beq slot1 lbra ShowHelp slot1 inc slot,u slot0 lbsr SkipSpcs skip forward to parameter leax -2,x back off two bytes * Add M: to start of parameter ldd #"M: std ,x * setup SDC for command mode orcc #IntMasks mask interrupts lbsr CmdSetup bcc sendCommand andcc #^IntMasks unmask interrupts ldb #$f6 Not ready error code lbra Exit sendCommand ldb #$e0 orb slot,u stb CMDREG send to SDC command register exg a,a wait lbsr txData transmit buffer to SDC lda #$0 sta CTRLATCH andcc #^IntMasks unmask interrupts bcc ExitOK tstb beq timeOutError bitb #$20 bne targetInUseError bitb #$10 bne targetNotFoundError bitb #$08 bne miscHardwareError bitb #$04 bne pathNameInvalidError bra unknownError targetInUseError ldy #targetInUseL leax targetInUse,pc bra wrtErr targetNotFoundError ldy #targetNotFoundL leax targetNotFound,pc bra wrtErr miscHardwareError ldy #miscHardwareL leax miscHardware,pc bra wrtErr pathNameInvalidError ldy #pathNameInvalidL leax pathNameInvalid,pc bra wrtErr unknownError ldy #unknownL leax unknown,pc bra wrtErr timeOutError ldy #timeoutL leax timeout,pc wrtErr lda #$2 os9 I$Write ExitOK clrb Exit os9 F$Exit ShowHelp equ * IFNE DOHELP leax >HlpMsg,pcr point to help message ldy #HlpMsgL get length lda #$02 std error os9 I$Write write it ENDC bra ExitOk * Stolen from BASIC09 * Convert # in D to ASCII version (decimal) L09BA pshs y,x,d Preserve End of data mem ptr,?,Data mem size pshs d Preserve data mem size again leay <L09ED,pc Point to decimal table (for integers) L09C1 ldx #$2F00 L09C4 puls d Get data mem size L09C6 leax >$0100,x Bump X up to $3000 subd ,y Subtract value from table bhs L09C6 No underflow, keep subtracting current power of 10 addd ,y++ Restore to before underflow state pshs d Preserve remainder of this power ldd ,y Get next lower power of 10 tfr x,d Promptly overwrite it with X (doesn't chg flags) beq L09E6 If finished table, skip ahead cmpd #$3000 Just went through once? beq L09C1 Yes, reset X & do again * lbsr L1373 Go save A @ [<u0082] ldb 11,u Write A to output buffer sta b,u incb stb 11,u ldx #$2F01 Reset X differently bra L09C4 Go do again L09E6 * lbsr L1373 Go save A @ [<u0082] ldb 11,u Write A to output buffer sta b,u incb stb 11,u leas 2,s Eat stack puls pc,y,x,d Restore regs & return * Table of decimal values L09ED fdb $2710 10000 fdb $03E8 1000 fdb $0064 100 fdb $000A 10 fdb $0001 1 fdb $0000 0 ********************************************************************* * Setup Controller for Command Mode ********************************************************************* * EXIT: * Carry cleared on success, set on timeout * All other registers preserved * CmdSetup pshs x,a ; preserve registers lda #$43 ; put controller into.. sta CTRLATCH ; Command Mode ldx #0 ; long timeout counter = 65536 busyLp lda STATREG ; read status lsra ; move BUSY bit to Carry bcc setupExit ; branch if not busy leax -1,x ; decrement timeout counter bne busyLp ; loop if not timeout lda #0 ; clear A without clearing Carry sta CTRLATCH ; put controller back in emulation setupExit puls a,x,pc ; restore registers and return ********************************************************************* * Send 256 bytes of Command Data to SDC Controller ********************************************************************* * ENTRY: * X = Data Address * * EXIT: * B = Status * Carry set on failure or timeout * All other registers preserved * txData pshs u,y,x ; preserve registers ldy #DATREGA ; point Y at the data registers * Poll for Controller Ready or Failed. comb ; set carry in anticipation of failure ldx #0 ; max timeout counter = 65536 txPoll ldb -2,y ; read status register bmi txExit ; branch if FAILED bit is set bitb #READY ; test the READY bit bne txRdy ; branch if ready leax -1,x ; decrement timeout counter beq txExit ; exit if timeout bra txPoll ; poll again * Controller Ready. Send the Data. txRdy ldx ,s ; re-load data address into X ldb #128 ; 128 words to send (256 bytes) txWord ldu ,x++ ; get data word from source stu ,y ; send to controller decb ; decrement word loop counter bne txWord ; loop until done * Done sending data, wait for result ldx #0 ; wait for result comb ; assume error txWait ldb -2,y ; load status bmi txExit ; branch if failed lsrb ; clear carry if not busy bcc txExit ; test ready bit leax -1,x ; decrememnt timeout counter bne txWait ; loop back until timeout txExit puls x,y,u,pc ; restore registers and return ********************************************************************* * Retrieve 256 bytes of Response Data from SDC Controller ********************************************************************* * ENTRY: * X = Data Storage Address * * EXIT: * B = Status * Carry set on failure or timeout * All other registers preserved * rxData pshs u,y,x ; preserve registers ldy #DATREGA ; point Y at the data registers * Poll for Controller Ready or Failed. comb ; set carry in anticipation of failure ldx #0 ; max timeout counter = 65536 rxPoll ldb -2,y ; read status register bmi rxExit ; branch if FAILED bit is set bitb #READY ; test the READY bit bne rxRdy ; branch if ready leax -1,x ; decrement timeout counter beq rxExit ; exit if timeout bra rxPoll ; poll again * Controller Ready. Grab the Data. rxRdy ldx ,s ; re-load data address into X ldb #128 ; 128 words to read (256 bytes) rxWord ldu ,y ; read data word from controller stu ,x++ ; put into storage decb ; decrement word loop counter bne rxWord ; loop until done clrb ; success! clear the carry flag rxExit puls x,y,u,pc ; restore registers and return ********************************************************************* * This routine skip over spaces and commas ********************************************************************* * Entry: * X = ptr to data to parse * Exit: * X = ptr to first non-whitespace char * A = non-whitespace char SkipSpcs lda ,x+ cmpa #C$SPAC beq SkipSpcs leax -1,x rts emod eom equ * end