Mercurial > hg > Members > kono > nitros9-code
changeset 3018:e7ffcda58117
Initial comming for my snakes demonstration and the sdir command.
Snakes draws an animated snake in text and graphics windows.
Sdir is used the display the directory on the CoCoSDC SD card.
author | tlindner |
---|---|
date | Thu, 27 Nov 2014 21:01:17 -0800 (2014-11-28) |
parents | 545a59f51fee |
children | 9abe4d732538 |
files | 3rdparty/utils/makefile 3rdparty/utils/tlindner/defsfile 3rdparty/utils/tlindner/makefile 3rdparty/utils/tlindner/sdir.asm 3rdparty/utils/tlindner/snake.asm |
diffstat | 5 files changed, 791 insertions(+), 1 deletions(-) [+] |
line wrap: on
line diff
--- a/3rdparty/utils/makefile Thu Nov 27 20:03:48 2014 -0800 +++ b/3rdparty/utils/makefile Thu Nov 27 21:01:17 2014 -0800 @@ -1,6 +1,6 @@ include ../../rules.mak -dirs = boisy dasm smartwatch supercomm dladd winfo gene +dirs = boisy dasm smartwatch supercomm dladd winfo gene tlindner # Make all components all:
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/3rdparty/utils/tlindner/defsfile Thu Nov 27 21:01:17 2014 -0800 @@ -0,0 +1,2 @@ + use os9.d + use scf.d
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/3rdparty/utils/tlindner/makefile Thu Nov 27 21:01:17 2014 -0800 @@ -0,0 +1,44 @@ +include ../../../rules.mak + +DEPENDS = ./makefile + +CMDS = snake sdir + +ALLOBJS = $(CMDS) + +DSKTLU = tlindner_utils.dsk + +DSKS = $(DSKTLU) + +all: banner $(ALLOBJS) $(DEPENDS) + +banner: + @$(ECHO) "**************************************************" + @$(ECHO) "* *" + @$(ECHO) "* tlindner utilities *" + @$(ECHO) "* *" + @$(ECHO) "**************************************************" + +$(DSKTLU): $(CMDS) + $(RM) $@ + $(OS9FORMAT) -q $@ -n"tindner utilities" + $(MAKDIR) $@,CMDS + $(OS9COPY) $(CMDS) $@,CMDS + $(OS9ATTR_EXEC) $(foreach file,$(CMDS),$@,CMDS/$(file)) + +dsk: all $(DSKS) + +dskcopy: dsk + $(CP) $(DSKS) $(DSKDIR) + +dskclean: + $(RM) $(DSKS) + +clean: dskclean + $(RM) $(ALLOBJS) + +info: + @echo "*** tlindner utilities ***" + @$(foreach dsk, $(DSKS), $(ECHO) $(dsk);) + +
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/3rdparty/utils/tlindner/sdir.asm Thu Nov 27 21:01:17 2014 -0800 @@ -0,0 +1,370 @@ +******************************************************************** +* sdir - Print directory of SDC card +* +* $Id$ +* +* Edt/Rev YYYY/MM/DD Modified by +* Comment +* ------------------------------------------------------------------ +* 1 2014/11/20 tim lindner +* Started writing code. + + nam sdir + ttl Print directory of SDC card + + 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 +buffer rmb 256*5 + +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 /sdir/ + fcb edition +* Place constant strings here +header fcc /SDC Directory: / +headerL equ *-header +basepath fcc /L:*.*/ + fcb 0 +timoutError fcc /Timeout./ +carrigeReturn fcb C$LF + fcb C$CR +timoutErrorL equ *-timoutError + +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 + +miscHardwareError fcc /Miscellaneous hardware error./ + fcb C$LF + fcb C$CR +miscHardwareErrorL equ *-miscHardwareError + +notInitiated fcc /Listing not initiated./ + fcb C$LF + fcb C$CR +notInitiatedL equ *-notInitiated + +* +* 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 equ * +* create path string in buffer area + decb + pshs u,x,d + leax basepath,pc + ldd ,x++ copy 'L:' + std ,u++ + puls d + cmpd #$0 + bne copyParameterArea +* use '*.*' if user suplied no parameter + ldd ,x++ copy '*.*' and null + std ,u++ + ldd ,x++ + std ,u++ + puls x,u + ldd #$5 Length of buffer + bra printHeader +copyParameterArea equ * + puls x + pshs d +cpaLoop lda ,x+ + sta ,u+ + decb + beq cpaDone + bra cpaLoop +cpaDone clra put null at end of parameter string + sta ,u+ + puls d + addd #3 + puls u + leas 0,y clobber parameter area, put stack at top, giving us as much RAM as possible +printHeader equ * + pshs d + lda #1 Output path (stdout) + ldy #headerL + leax header,pc header in x + os9 I$Write Send the value to the device driver +* print path + puls y + leax buffer,u buffer in x + os9 I$Write Send the value to the device driver + ldy #2 length of buffer + leax >carrigeReturn,pcr buffer in x + os9 I$Write Send the value to the device driver +* wait until our next tick to communicate with the SDC + ldx #$1 + os9 F$Sleep +* setup SDC for directory processing + orcc #IntMasks mask interrupts + lbsr CmdSetup + bcc sendCommand + ldb #$f6 Not ready error code + lbra Exit +sendCommand equ * + ldb #$e0 load initial directory listing command + stb CMDREG send to SDC command register + exg a,a wait + leax buffer,u point to transfer bufer + lbsr txData transmit buffer to SDC + bcc getBuffer + tstb + beq timeOut + bitb #$10 + bne targetDirectoryNotFoundError + bitb #$08 + bne miscellaneousHardwareError + bitb #$04 + bne pathNameInvalidError + lbra Exit +getNextDirectoryPage equ * + leax 256*2,x + tfr s,d + pshs d + cmpx ,s++ + bhi printBuffer + leax -256,x +getBuffer equ * + ldb #$3e set parameter #1 + stb PREG1 + ldb #$c0 set command code + stb CMDREG send to SDC command register + lbsr rxData + bcc checkBuffer + tstb + beq timeOut + bitb #$8 + bne notInitiatedError + bra Exit +timeOut equ * + leax >timoutError,pcr point to help message + ldy #timoutErrorL get length +genErr clr CTRLATCH + andcc #^IntMasks unmask interrupts + lda #$02 std error + os9 I$Write write it + clrb clear error + bra ExitNow +targetDirectoryNotFoundError equ * + leax >dirNotFound,pcr + ldy #dirNotFoundL + bra genErr +miscellaneousHardwareError equ * + leax >miscHardwareError,pcr + ldy #miscHardwareErrorL + bra genErr +pathNameInvalidError equ * + leax >pathNameInvalid,pcr + ldy #pathNameInvalidL + bra genErr +notInitiatedError equ * + leax >notInitiated,pcr + ldy #notInitiatedL + bra genErr + +* Check buffer for nulled entry. This signifies the end +checkBuffer equ * + lda #16 + leau ,x go back to start of buffer +cbLoop ldb ,u + beq printBuffer + leau 16,u + deca + beq getNextDirectoryPage + bra cbLoop + +printBuffer equ * + clr CTRLATCH + andcc #^IntMasks unmask interrupts + clrb + tfr dp,a + tfr d,u + lda #1 Output path (stdout) +pbLoop ldy #8 length of buffer + leax ,u + os9 I$Write Send the value to the device driver + ldy #2 length of buffer + leax >carrigeReturn,pcr buffer in x + os9 I$Write Send the value to the device driver + leau 16,u + ldb ,u + beq ExitOK + bra pbLoop + + +ExitOk clrb +Exit clr CTRLATCH + andcc #^IntMasks unmask interrupts +ExitNow os9 F$Exit + +********************************************************************* +* 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 + ldx #0 ; wait for result +* Done sending data, wait for result + comb ; assume error +txWait ldb -2,y ; load status + bmi txExit ; branch if failed + bitb #READY ; test ready bit + bne txExitOK ; branch if ready + leax -1,x ; decrememnt timeout counter + bne txWait ; loop back until timeout +txExitOK andcc #^1 +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
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/3rdparty/utils/tlindner/snake.asm Thu Nov 27 21:01:17 2014 -0800 @@ -0,0 +1,374 @@ +******************************************************************** +* Snake - Animate slithering snake in window +* +* $Id$ +* +* Edt/Rev YYYY/MM/DD Modified by +* Comment +* ------------------------------------------------------------------ +* 1 2014/10/08 tim lindner +* Started writing code. + + nam Snake + ttl Animate slithering snake in window + + ifp1 + use defsfile + endc + +* Here are some tweakable options +DOHELP set 1 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 + + mod eom,name,tylg,atrv,start,size + + org 0 +bail_flag rmb 1 +windowx rmb 1 +windowy rmb 1 +time rmb 3 +RND rmb 3 +snakesize rmb 1 +head rmb 1 +IOBUF rmb 4 +IOBUF2 rmb 4 +cleartop equ . everything up to here gets cleared at start +* Finally the stack for any PSHS/PULS/BSR/LBSRs that we might do +snakebuffer rmb 80*2 + rmb STACKSZ+PARMSZ +size equ . + +* The utility name and edition goes here +name fcs /Snake/ + fcb edition + +* Place constant strings here + IFNE DOHELP +HlpMsg fcb C$LF + fcc /Use: Snake <size>/ + fcb C$LF + fcb C$CR + fcc / Size is less than window width./ + 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 + +* 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 + +* The start of the program is here. +* Before any command line processing is done, we clear out +* our static memory from U to cleartop, then determine the +* size of our data area (minus the stack). +start pshs u,x save registers for later + leax <cleartop,u point to end of area to zero out + IFNE H6309 + subr u,x subtract U from X + tfr x,w and put X in W + clr ,-s put a zero on the stack + tfm s,u+ and use TFM to clear starting at U + leas 1,s clean up the stack + ELSE + pshs x save end pointer on stack +clrnxt clr ,u+ clear out + cmpu ,s done? + bne clrnxt branch if not + leas 2,s else clear stack + ENDC + puls x,u and restore our earlier saved registers + + lda ,x+ get first char + cmpa #C$CR CR? + lbeq ShowHelp +* Process command line (a single number) +nextChar suba #'0' Convert ASCII number to value + adda snakesize,u add stored value with new value + sta snakesize,u store new snake size + lda ,x get next char + cmpa #'0' + blo clDone + cmpa #'9' + bhi clDone + lda #10 + ldb snakesize,u multiply current snake size by 10 + mul + stb snakesize,u + lda ,x+ get next char + bra nextChar +clDone cmpb #80 + ble timeSeed + ldb #80 + stb snakesize,u +* Get Time packet to seed PRNG +timeSeed leax time,u + os9 F$Time +* Turn off cursor + leax IOBUF,u + lda #$05 05 20 is turn off cursor code + sta ,x + lda #$20 + sta 1,x + ldy #2 Length of buffer + lda #1 Output path (stdout) + os9 I$Write Send the value to the device driver +* Add intercept + leax IR,pcr + os9 F$Icpt +* Clear screen + leax IOBUF,u + lda #$0c load 'clear screen' code for VDGINT/WINDINT + sta ,x + ldy #1 Length of buffer + lda #1 Output path (stdout) + os9 I$Write Send the value to the device driver +* get window size + ldx #$0 + ldy #$0 + lda #1 Output Path (stdout) + ldb #SS.ScSiz Request screen size + os9 I$Getstt Make requst + bcs sizeErr + tfr x,d + cmpb #$0 + beq sizeErr RBF returns screen size of $0 + decb + stb windowx,u + tfr y,d + decb + stb windowy,u + bra compare +* If SS.ScSiz returns an error set the window up for 80 x 24 +sizeErr ldd #$4f17 + std windowx,u +* compare parameter with window width +compare ldb snakesize,u + cmpb windowx,u + lbpl ShowHelp bail if parameter is larger than window +* fill snake buffer with initial X coordinate for snake positions +* figure out starting horizontal position of tail + clra + ldb snakesize,u + lsrb divide snake size by 2 + stb ,-s + ldb windowx,u + lsrb divide window width by 2 + subb ,s+ subtract half snake length from half window width + addb #$20 Offset Horizontal position for VDGINT and WINDINT + leax snakebuffer,u + lda snakesize,u +filloopx stb ,x++ + incb + deca + bne filloopx fill snake buffer with initial horizontal coordinates +* Figure out starting vertical of snake body + ldb windowy,u + lsrb divide window height by 2 + addb #$20 Offset vertical position for VDGINT or WINDINT + leax snakebuffer+1,u + lda snakesize,u +filloopy stb ,x++ + deca + bne filloopy fill snake buffer with initial vertical coordinates +* Draw initial snake + leax IOBUF,u + lda #$02 + sta 0,x + lda #'*' + sta 3,x + ldy #4 Length of buffer + lda snakesize,u + pshs u + leau snakebuffer,u +idrlp ldb ,u+ + stb 1,x + ldb ,u+ + stb 2,x + exg a,b + lda #1 Output path (stdout) + os9 I$Write Send the value to the device driver + exg b,a + deca + bne idrlp Initial draw loop + puls u + lda snakesize,u set head pointer to snake size minus one + deca + sta head,u +* Offset Window size for VDGINT and WINDINT + ldd windowx,u + addd #$2020 + std windowx,u +* Set up IOBUF2 for Easing + lda #$02 + sta IOBUF2,u + lda #' ' + sta IOBUF2+3,U +MainLoop equ * +* Erase tail + leay snakebuffer,u + leax IOBUF2,u + ldb head,u + incb + cmpb snakesize,u + bne mlcont1 + clrb +mlcont1 stb head,u Store new head offset + lslb + ldd b,y offset into table pointer + sta 1,x + stb 2,x + ldy #4 + lda #1 Output path (stdout) + os9 I$Write Send the value to the device driver +* Find position for new head +* Set Reg X to point to current head position +copyhead ldb head,u get head index + lslb + leax snakebuffer,u + abx advance X to correct position + lsrb + bne mlcont2 if not zero + ldb snakesize,u get snake buffer size +mlcont2 decb Back up one index + lslb multiply by two + leay snakebuffer,u point y to start of buffer + leay b,y advance Y to correct position + ldd ,y load coordinates + std ,x save coordinates +* pick random number +pick ldb #$4 + lbsr RAND + deca + beq goLeft + deca + beq goDown + deca + beq goRight + +* Remember: coordinates are pre-offset by a value of $20 + +goUp ldb 1,x + cmpb #$20 + beq pick + decb + stb 1,x + bra drawHead + +goDown ldb 1,x + incb + cmpb windowy,u + beq pick + stb 1,x + bra drawHead + +goLeft ldb ,x + cmpb #$20 + beq pick + decb + stb ,x + bra drawHead + +goRight ldb ,x + cmpb windowx,u + beq pick + incb + stb ,x + bra drawHead + +drawHead ldd ,x + std IOBUF+1,u + lda #$1 + ldy #$4 + leax IOBUF,u + os9 I$Write Send the value to the device driver + lda bail_flag + bne ExitOK + ldx #$1 + os9 F$Sleep + lbra MainLoop + +ExitOk bsr bail + clrb +Exit os9 F$Exit + +IR lda #$ff + sta bail_flag,u + rti +* Turn on cursor +bail leax IOBUF,u + lda #$05 05 21 is turn on cursor code + sta ,x + lda #$21 + sta 1,x + ldy #2 Length of buffer + lda #1 Output path (stdout) + os9 I$Write Send the value to the device driver + rts + +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 + +* THIS IS A FAST MEDIUM GRADE RANDOM NUMBER GENERATOR +* From SYMMETRY by Robert Gault, 1993 +* LENGTH OF NON-REPEATING SEQUENCE = 16,777,215 +* INTERMEDIATE OUTPUT 0 - 255 OR 0 - .996078431 STEPS OF .003921568 +* ENTER: REG.B = N+1 +* EXIT: REG.A = 0 TO N +* REG.B = FRACTIONAL PART OF N +RAND LDA RND+2,u GET 19TH BIT + ANDA #%00100000 + LSLA ALIGN IT WITH 24TH BIT + LSLA FASTER THAN SHIFTING TO RIGHT + LSLA + ROLA + EORA RND+2,u XOR BITS 19&24 + LSRA RESULT TO CARRY + ROR RND,u FEED RESULT INTO RANDOM NUMBER + ROR RND+1,u AND SHIFT TO THE RIGHT + ROR RND+2,u + LDA RND,u + MUL + RTS + + emod +eom equ * + end + \ No newline at end of file