Mercurial > hg > Members > kono > nitros9-code
view level1/cmds/shellplus.asm @ 2437:d26e96c40194
rfm changes
author | aaronwolfe |
---|---|
date | Wed, 24 Feb 2010 00:37:08 +0000 |
parents | bfb8eefc19e1 |
children | 5c6b71612ce4 |
line wrap: on
line source
******************************************************************** * Shellplus - Enhanced shell for NitrOS-9 * * Modified by L. Curtis Boyle from original 2.2 disassembly * * $Id$ * * Edt/Rev YYYY/MM/DD Modified by * Comment * ------------------------------------------------------------------ * 21 ????/??/?? * Original Tandy/Microware version. * * 22/2 ????/??/?? * History and numerous features added. * * 23 2010/01/19 Boisy G. Pitre * Added code to honor S$HUP signal and exit when received to support * networking. nam Shell ttl Enhanced shell for NitrOS-9 * Disassembled 93/04/15 14:58:18 by Disasm v1.6 (C) 1988 by RML * Signals: Signals 2 & 3 are assigned new keys to handle forward/backward * command history. Signal $B (11) is the signal sent out on a key being ready * for normal command processing ifp1 use defsfile endc tylg set Prgrm+Objct atrv set ReEnt+rev rev set $00 edition set 23 mod eom,name,tylg,atrv,start,size u0000 rmb 1 Path # for standard input u0001 rmb 1 Path # for standard output u0002 rmb 1 Path # for standard error u0003 rmb 1 # of 256 byte pages of data mem for frked module u0004 rmb 2 Temp ptr (current parse ptr, mem module ptr,etc) u0006 rmb 2 Size of current group u0008 rmb 2 Pointer to start of current group (past '(') u000A rmb 2 u000C rmb 1 Current char. being processed in command parser u000D rmb 1 # of command groups [ '()' groupings ] u000E rmb 1 unprocessed signal # (0=none waiting) u000F rmb 1 ??? Flag of some sort u0010 rmb 2 ??? (ptr to some module name) u0012 rmb 1 Current working DIR path # u0013 rmb 1 Flag to kill parent process (1=Kill parent) u0014 rmb 1 Flag: If set, a result must not be 0 ??? u0015 rmb 1 u0016 rmb 1 u0017 rmb 1 u0018 rmb 1 Immortal shell (0=NO) * A clearing routine only does u0000 to u0018 u0019 rmb 1 u001A rmb 2 u001C rmb 1 Shell logging on flag (0=OFF) u001D rmb 1 Shell prompting (0=ON) u001E rmb 1 Echo input (0=OFF) u001F rmb 1 Variable expansion (0=ON) u0020 rmb 1 Kill shell on error (0=OFF) u0021 rmb 1 Process # to set priority on u0022 rmb 1 Priority to set (0=don't change) (ours or fork) u0023 rmb 2 u0025 rmb 2 End of data mem ptr (top of stack) u0027 rmb 1 u0028 rmb 1 u0029 rmb 1 u002A rmb 2 u002C rmb 2 u002E rmb 1 u002F rmb 1 u0030 rmb 1 u0031 rmb 1 u0032 rmb 2 u0034 rmb 3 u0037 rmb 1 Flag: 0=Data dir .PWD invalid, 1=valid u0038 rmb 1 Flag: 0=Exec dir .PXD invalid, 1=valid u0039 rmb 1 u003A rmb 1 u003B rmb 2 u003D rmb 1 u003E rmb 2 u0040 rmb 2 Ptr to start of filename (vs. pathname) ('/') * Shell prompt flag u0042 rmb 1 Current working dir path already done flag u0043 rmb 1 u0044 rmb 1 u0045 rmb 1 ??? <>0 means looking for GOTO label? u0046 rmb 1 Flag: 1=GOTO label found? u0047 rmb 1 Error code from ReadLn or signal u0048 rmb 2 Ptr to 1st char after redirection symbols u004A rmb 2 Ptr to text message u004C rmb 2 Size of text message u004E rmb 1 u004F rmb 1 0=no pathname in parm line, else IS pathname u0050 rmb 2 u0052 rmb 2 Current expanded buffer size (max=2048) u0054 rmb 2 Ptr to current char in wildcard filename we are * checking u0056 rmb 2 Ptr to current pos in expanded buffer u0058 rmb 2 Pointer to end of GOTO label name u005A rmb 2 User ID # from F$ID call u005C rmb 1 u005D rmb 1 u005E rmb 1 Device type: 0=SCF (keyboard),1=RBF (Scriptfile) u005F rmb 1 u0060 rmb 1 Data module linked flag: 1= Yes u0061 rmb 2 Ptr to data module name u0063 rmb 2 Ptr to intercept routines data mem u0065 rmb 2 Execution address of linked module u0067 rmb 2 Start address of module u0069 rmb 2 u006B rmb 1 Flag: 0=No module to unlink, <>0 module to unlink u006C rmb 1 u006D rmb 1 Start of device name buffer (start with '/') u006E rmb 71 Actual device name u00B5 rmb 20 Start of another device name buffer ('/') u00C9 rmb 13 u00D6 rmb 13 Standard module header info (M$ID-M$Mem) u00E3 rmb 5 Module name string (reserves 64 chars) u00E8 rmb 3 u00EB rmb 4 u00EF rmb 10 Temp buffer (many uses) u00F9 rmb 6 u00FF rmb 37 Place to point SP when CHAINing u0124 rmb 81 Temporary buffer (used for several things) u0175 rmb 119 Part of temp buffer for ReadLn (200 chars total) u01EC rmb 2 Least sig. 2 digits of process # (ASCII format) u01EE rmb 1 u01EF rmb 2 Holding area for 2 digit ASCII conversions * Shell prompt parsing flags u01F1 rmb 1 Process ID # already done flag u01F2 rmb 1 Standard output device name already done flag u01F3 rmb 1 Quoting on flag in shell prompt string parsing u01F4 rmb 1 Date already done flag u01F5 rmb 1 Time already done flag u01F6 rmb 1 Date OR time already done flag u01F7 rmb 2 Size of expanded shell prompt u01F9 rmb 25 Current shell prompt string u0212 rmb 1 Lead in Line feed for expanded shell prompt u0213 rmb 199 Expanded shell prompt u02DA rmb 6 Date/time packet u02E0 rmb 8 Date string u02E8 rmb 1 Space separating date & time (for shell init) u02E9 rmb 9 Time string (and CR) u02F2 rmb 131 u0375 rmb 131 u03F8 rmb 29 u0415 rmb 2 u0417 rmb 1 u0418 rmb 400 Intercept routines memory area (not used) u05A8 rmb 810 Shell variables (user?) u08D2 rmb 810 Shell variables (shell sub?) u0BFC rmb 80 u0C4C rmb 81 Copy of GOTO label name u0C9D rmb 32 DIR Entry buffer u0CBD rmb 32 Shell logging filename (append mode '+') u0CDD rmb 400 PATH=Buffer (each entry CR terminated) u0E6D rmb 2048 Fully expanded filenames buffer (for wildcards) * Actually,this next block appears to be generic buffers for various functions u166D rmb 80 Process descriptor copies go here (512 bytes) u16BD rmb 1 u16BE rmb 80 u170E rmb 238 u17FC rmb 10 u1806 rmb 2 ??? Ptr to end of shell history buffers u1808 rmb 2 Ptr to where next history entry will go u180A rmb 2 Ptr to start of shell history buffers u180C rmb 1 # of lines in history buffer (1-(u180C)) u180D rmb 1 Current line # in history buffer u180E rmb 1 Original keyboard terminate char u180F rmb 1 Original keyboard interrupt char u1810 rmb 1 u1811 rmb 1 Original end of line NUL count u1812 rmb 1 Flag to indicate if we have to restore PD.OPT u1813 rmb 2 u1815 rmb 808 Shell history copies start here u1B3D rmb 963 Local stack space, etc. size equ . name equ * L000D fcs /Shell/ fcb edition L0013 fcb Prgrm+PCode fcs 'PascalS' fcb Sbrtn+CblCode fcs 'RunC' fcb Sbrtn+ICode L0021 fcs 'RunB' fcb $00,$00,$00,$00,$00,$00,$00,$00,$00 L002E fcb C$LF fcc 'Shell+ v2.2a ' L003C fcb $00 L003D fcc '{@|#}$: ' L0055 fcc '+++START+++' fcb C$CR L0061 fcc '+++END+++' fcb C$CR * Intercept routine L006B stb <u000E Save signal code & return * +++ BGP added for Hang Up * +++ Note we are exiting even if shell is immortal! cmpb #S$HUP lbeq exit * +++ rti start leas -5,s Make 5 byte buffer on stack pshs y,x,d Save Top/bottom of param area, and size of area leax >u1815,u Pointer to start of history buffer area stx >u1808,u Save ptr to where next history entry goes stx >u180A,u Save ptr to start of history buffers leax >$0328,x Setup up another pointer (end of history area?) stx >u1806,u Save it clr >u1812,u Clear flag that we have to restore PD.OPT clr >u180D,u Current line # of history buffer=0 clr >u180C,u # lines in history buffer=0 ldx 2,s Get back top of data area ptr ldb #$FF 255 bytes to clear lbsr L0412 Go clear direct page sts <u0025 Save current stack ptr stu <u0027 Save Data mem ptr (0) leax >u0418,u Point to intercept routines memory area stx <u0063 Save a copy of it leax <L006B,pc Point to intercept routine os9 F$Icpt Setup the intercept routine lbsr L16A6 Get user #, and make 2 digit ASCII ver. @ 1EC lbsr L1674 Make shell logging pathname @ CBD leax <L003D,pc Point to default shell prompt string lbsr L0A14 Go create shell prompt string & prompt itself leax >u05A8,u Point to start of shell variables ldd #C$CR*256+20 Carriage return (blank entries) & all 20 of them L009C sta ,x Mark shell variable as blank leax <81,x Point to next entry (81 bytes/entry) decb Do until all 20 are done (user & shell sub) bne L009C sta >u0CDD,u Init 1st 2 entries of PATH= buffer to CR's sta >u0CDD+1,u puls x,d Get parameter ptr & parameter size std <u0006 Save size of parameter area beq L00BF If no parameters, skip ahead lbsr L041B Pre-Parse parameters passed to this shell lbcs L01FA Error, go handle it tst <u000C Any char being processed in parser? lbne L01F9 Yes, skip ahead L00BF lbsr L0225 Release any keyboard/mouse signals & get PD.OPT inc <u006C ??? Set flag lds ,s++ Since parameters parsed, point SP to top of mem sts <u0025 Save end of data mem ptr (top of stack) stu <u0027 Save start of data mem ptr L00CC leax >L002E,pc Point to Shellplus v2.2 message tst <u001D Shell prompting turned on? bne L00EA No, skip ahead ldy #$000E Length of message lda #$01 Standard out os9 I$Write Write it lbcs L0200 If error writing, terminate Shellplus bsr L00FB Go create date & time strings lbsr L021B Write out date & time lbcs L0200 If error writing, terminate Shellplus L00EA clra sta <u005C leax >L0055,pc Point to '+++START+++' for shell logging lbsr L07E7 Go deal with logging if necessary tst <u001D Shell prompting turn on? bne L0120 No, skip ahead bra L010D Yes, do something else first L00FB clr >u01F6,u Clear date or time done flag lbsr L0B3B Create current date & time strings lda #C$SPAC Put in a space to separate date & time sta >u02E8,u leax >u02E0,u Point to start of date buffer rts * Shell just booted goes here L010D lbsr L09F7 Update expanded shell prompt if needed leax >u0212,u Point to expanded shell prompt ldy >u01F7,u Get size of expanded shell prompt L0119 tst <u001D Shell prompting on? bne L0120 No, continue on lbsr L021F Print shell prompt to standard error * Shell already booted goes here L0120 clra leax >u0124,u Point to an input line of some sort tst <u006B Any module linked? beq L015B No, continue on ldy <u0065 Yes, get Execution address of linked module cmpy <u0069 ??? lbhs L01F0 ldb #200 LDW #200/PSHS X/TFM Y+,X+ pshs x Copy temporary buffer to [<u0065] L0137 lda ,y+ sta ,x+ decb beq L014E Whole buffer copied, skip ahead cmpy <u0069 But don't go past here bhs L0147 cmpa #C$CR Also, stop on carriage return bne L0137 L0147 sty <u0065 Save new start address puls x Restore ptr to u0124 bra L01B9 Skip ahead L014E lda ,y+ cmpy <u0069 bhs L0147 cmpa #C$CR bne L014E bra L0147 L015B tst <u005E We reading from script file? bne L017F Yes, skip ahead ldd #SS.Relea Std input path/release keyboard+mouse signals os9 I$SetStt Release any keyboard signals lbsr L18DB Go modify signals 2&3 to use up/down arrows, * set up to re-enable Kybd signals L016A clr <u000E Clear out last signal received os9 I$SetStt Re-set up SS.SSig * NOTE: This BRA L0177 is required for type-ahead to work bra L0177 Go check for history-related signals * DOESN'T SEEM TO GET BACK TO HERE WHEN ABORT BUG APPEARS L0171 ldx #$0000 Sleep until signal is received os9 F$Sleep L0177 lbra L191C Go check for history signals (PD.QUT & PD.INT) * Signal received is not PD.QUT or PD.INT goes here L017B cmpb #$0B Key pressed signal? bne L0171 No, ignore signal & go to sleep again * Keyboard input received signal goes here L017F leax >u0124,u Point to temp buffer to hold line clra Std Input ldy #200 Maximum 200 char. input lbra L193E Go read the rest of line as ReadLn * Comes here after line or signal received & processed * NOTE: IF LINE RECEIVED, PD.QUT & PD.INT ARE BACK TO NORMAL VALUES L018B bcc L01B9 If no errors, skip ahead cmpb #E$EOF <ESC> key in ReadLn? beq L01F0 Yes, skip ahead L0191 lds <u0025 Get top of stack ptr ldu <u0027 Get back ptr to DP stb <u0047 Save error code tst <u0046 GOTO label active? bne L01A4 Yes, go print error message tst <u0018 Is this shell immortal? bne L01A4 Yes, go print error message tst <u0020 Kill shell on error? bne L01FA Yes, go do that L01A4 os9 F$PErr Print the error message tst <u0046 GOTO label active? lbeq L010D No, go print shell prompt/process next line clr <u0046 Clear GOTO label flag leax >u0C4C,u Point to copy of GOTO label name lbsr L0FD0 Go process GOTO line lbra L010D Go print shell prompt/process next line * No error received on line L01B9 cmpy #$0001 Just 1 char read (just a CR)? bhi L01CE No, go parse parameters lbsr L09F7 Go update date/time & expanded shell prompt leax >u0213,u Point to expanded shell prompt ldy >u01F7,u Get size of expanded shell prompt lbra L0119 Go print shell prompt & get next line from user * No errors-got input line, and there is something in it L01CE lbsr L041B Go parse parameters? pshs cc Save flags tst <u0045 bne L01D9 clr <u0047 Clear error/signal code from ReadLn L01D9 puls cc Restore flags lbcc L010D If no error from parse, do prompts & read again tstb If a 0 error, do prompts & read again lbeq L010D tst <u0045 ??? Do we care about errors? bne L0191 No, go handle error stb <u0047 Save error/signal code bra L0191 Go handle error L01EC fcc 'eof' fcb $0d * <ESC> received L01F0 tst <u001D Shell prompting on? L01F2 bne L01F9 No, skip printing 'eof' to screen leax <L01EC,pc Point to 'eof' string bsr L021B Write it out to std error L01F9 clrb No error flag * Possible shell error - Process L01FA lda <u0018 Is this shell immortal? lbne L0BDA Yes, go close or dupe paths * Shell not immortal, exit routine in here L0200 pshs u,b,cc Preserve regs tst <u006B Shellsub module to unlink? beq L020B No, skip ahead ldu <u0067 Yes, get ptr to module os9 F$UnLink Unlink it L020B puls u,b,cc Restore U, error # & flags * EX with no module name (or non-existant one) go here L020D pshs b,cc Save error # & flags leax >L0061,pc Point to '+++END+++' string lbsr L07E7 Close log file if one is open, restore ID # puls b,cc Restore error # & flags exit os9 F$Exit Terminate shellplus L021B ldy #80 Write up to 80 chars or CR L021F lda #$02 Std error path os9 I$WritLn Write message out & return rts L0225 pshs y,x,d Preserve regs ldd #SS.Relea Std In & Release keyboard & mouse signals os9 I$SetStt bcc L0233 No error, continue lda #$01 Couldn't release keyboard/mouse signals flag bra L0241 L0233 leax >u0124,u Point to buffer for current path options clra clrb CHANGE TO CLRB os9 I$GetStt Get std input path options lda >u0124,u Get Device type (0 (SCF) usually) L0241 sta <u005E Save device type (1=SS.Relea failed) puls pc,y,x,d Restore regs & return * R= command (redirect specific paths) L0245 lda ,x Get char from input line cmpa #'< Is it input path? beq L0251 Yes, skip ahead cmpa #'> Is it an output/error path? lbne L0BCF No, print 'WHAT?' L0251 pshs x Preserve ptr to 1st redirection symbol leay >L03CF,pc Point to modifier symbol table lbsr L092A Go to command line parser lbcs L0BCF Error, print 'WHAT?' ldd ,y Get table offset jsr d,y Call appropriate subroutine stx <u0048 Save ptr to source after redirection symbols puls x Restore ptr to redirection symbols lbcs L0B96 If error in subroutine, close paths & return L026A lda ,x+ Get 1st char again cmpa #'< Input path? beq L026A Yes, skip to next cmpa #'> Output or Error path? beq L026A Yes, skip to next cmpa #'- Overwrite old file? beq L026A Yes, skip to next cmpa #'+ Append to old file? beq L026A Yes, skip to next leax -1,x Point to non-redirect char bsr L02A3 Make name buffer, release signals,parse modifiers clrb Start path # 0 L0281 pshs b Save on stack lda b,u Get special path # beq L028A None, skip ahead os9 I$Close Close the path L028A puls b Get path # back clr b,u Clear out entry incb Next path # cmpb #$03 Done the 3 standard paths yet? blo L0281 No, keep doing until all done ldx <u0048 Get ptr to redirected dev/file name & return rts * Z= command (immortal shell setting, but kill parent process) L0296 inc <u0013 Flag we want to kill parent * I= command (immortal shell setting) L0298 lbsr L0CD2 lbcs L0B96 bsr L02A3 bra L02D4 * Create device name buffer @ u006D, Write NUL to std out, parse special * chars: # @ @+hibit $ ( ), Release any keyboard/mouse signals L02A3 pshs x Prsrve ptr to file/dev name we're redirecting to ldb #SS.DevNm Get device name call code leax <u006D,u Point to buffer for path names lda #'/ Preload '/' for device names sta ,x+ clra Standard input path os9 I$GetStt Get the current device name puls x Get back ptr to file/dev name for redirection lbcs L0B96 Error on GetStt, shut down paths & exit ldy #$0001 One char to write pshs x Prsrve ptr to file/dev name we're redirecting to leax >L003C,pc Point to a NUL lda #1 Std out os9 I$Write Write the NUL out puls x Restore Devname ptr lbcs L0B96 Error on Write, shut down paths & exit lbsr L0A04 Do normal parsing - includes #,@,$e0,$,(,) lbsr L0225 Release signals rts L02D4 inc <u0018 Set 'immortal shell' flag inc <u0019 ??? (flag used by L0B96 routine) lbsr L0B96 clr <u0019 tst <u0013 Do we want to kill the parent process? beq L02FC IFEQ Level-1 ldx <D.Proc ELSE os9 F$ID Get our process ID # into A pshs x Preserve X leax >u166D,u Point to process descriptor buffer os9 F$GPrDsc Get our process descriptor ENDC lda P$PID,x Get our parents process # IFGT Level-1 puls x Restore X ENDC beq L02A3 If parent's process # is 0 (system), skip back clrb S$Kill signal code os9 F$Send Send it to parent lbcs L0191 If error sending signal, go to error routine L02FC clrb No error stb <u0013 Clear 'kill parent' flag rts * Command list L0300 fdb L0B87-L0300 $0887 fcs '*' L0303 fdb L12D1-L0303 $0fce fcs 'W' L0306 fdb L09BD-L0306 $06b7 fcs 'CHD' L030B fdb L09B5-L030B $06aa fcs 'CHX' L0310 fdb L09BD-L0310 $06ad fcs 'CD' L0314 fdb L09B5-L0314 $06a1 fcs 'CX' L0318 fdb L0987-L0318 $066f fcs 'EX' L031C fdb L16DD-L031C $13c1 fcs 'KILL' L0322 fdb L09EF-L0322 $06cd fcs 'X' L0325 fdb L09F3-L0325 $06ce fcs '-X' L0329 fdb L0A14-L0329 $06eb fcs 'P=' L032D fdb L09D7-L032D $06aa fcs 'P' L0330 fdb L09DA-L0330 $06aa fcs '-P' L0334 fdb L09DF-L0334 $06ab fcs 'T' L0337 fdb L09E3-L0337 $06ac fcs '-T' L033B fdb L170D-L033B $13d2 fcs 'SETPR' L0342 fdb L0298-L0342 $ff56 fcs 'I=' L0346 fdb L0245-L0346 $feff fcs 'R=' L034A fdb L0296-L034A $ff4c fcs 'Z=' L034E fdb L0927-L034E $05d9 fcs ';' L0351 fdb L176B-L0351 $141a fcs '.PWD' L0357 fdb L177D-L0357 $1426 fcs '.PXD' L035D fdb L1029-L035D $0ccc fcs 'M=' L0361 fdb L105A-L0361 $0cf9 fcs 'VAR.' L0367 fdb L09E7-L0367 $0680 fcs 'V' L036A fdb L09EA-L036A $0680 fcs '-V' L036E fdb L1209-L036E $0e9b fcs 'PATH=' L0375 fdb L11CF-L0375 $0e5a fcs 'PAUSE' L037C fdb L1126-L037C $0daa fcs 'INC.' L0382 fdb L1131-L0382 $0daf fcs 'DEC.' L0388 fdb L0D62-L0388 $09da fcs 'IF' L038C fdb L0E0B-L038C $0a7f fcs 'THEN' L0392 fdb L0F8C-L0392 $0bfa fcs 'ELSE' L0398 fdb L0E0B-L0398 $0a73 fcs 'FI' L039C fdb L0E0B-L039C $0a6f fcs 'ENDIF' L03A3 fdb L0F3F-L03A3 $0b9c fcs 'CLRIF' L03AA fdb L0FD0-L03AA $0c26 L03AC fcs 'GOTO' L03B0 fdb L0F93-L03B0 $0be3 fcs 'ONERR' L03B7 fdb L09CF-L03B7 $0618 fcs 'L' L03BA fdb L09D3-L03BA $0619 fcs '-L' L03BE fdb L0969-L03BE $05ab fcs 'S.T.A.R.T.U.P' fdb $0000 L03CF fdb L1634-L03CF $1265 fcs '!' L03D2 fdb L1634-L03D2 $1262 fcs '|' L03D5 fdb L12A0-L03D5 $0ecb fcs ';' L03D8 fdb L12C3-L03D8 $0eeb fcs '&' L03DB fdb L1299-L03DB $0ebe fcb $8d CR L03DE fdb L0CCE-L03DE $08f0 fcs '<>>>' L03E4 fdb L0CE5-L03E4 $0901 fcs '<>>' L03E9 fdb L0CD7-L03E9 $08ee fcs '<>' L03ED fdb L0CF3-L03ED $0906 fcs '>>>' L03F2 fdb L0C01-L03F2 $080f fcs '>>' L03F6 fdb L0BFA-L03F6 $0804 fcs '<' L03F9 fdb L0C08-L03F9 $080f fcs '>' L03FC fdb L1277-L03FC $0e7b fcs '#' L03FF fdb L1265-L03FF $0e66 fcs '^' fdb $0000 End of table marker L0404 fcb $0d fcc '()' fcb $ff L0408 fcb $0d fcc '!#&;<>^|' fcb $ff * Subroutine: Clears B bytes of memory to NUL's starting @ U L0412 pshs u Clear memory L0414 clr ,u+ decb L0417 bne L0414 L0419 puls pc,u * Pre-Parse parameters passed to this shell. Will copy from parm area to * buffer area at u0166D, checking for raw mode access allowability if needed * Will also * Entry: X=ptr to parm area L041B ldb #$18 # of bytes to clear in DP L041D bsr L0412 Go clear from <u0000 to <u0018 (immortal off) tst <u006C ??? (If shell is initing, it's 0) lbeq L04CE Shell just initializing, skip ahead leay >u17FC,u Point to end of buffer marker pshs y Save ptr on stack leay >u166D,u Point to buffer for process descriptor L042F bsr L0486 Copy next char (check for '@', possibly eat) cmpa #C$CR CR? beq L04A8 Yes, force CR into ,y & process line from start cmpy ,s At end of buffer? beq L04A8 Yes, force CR, process line from start tst <u001F Variable expansion on? bne L042F No, check next char cmpa #'% Is it a '%' (shell variable)? bne L042F No, do next character leay -1,y Back up destination ptr to where '%' is pshs x Save current position in parms lda ,x Get 1st char of shell variable nam cmpa #'* Shell variable for last error code received? beq L049D Yes,replace shell var with contents of shell var leax >u05A8,u Point to user shell variable contents table cmpa #'% 2nd '%'= shellsub variable bne L0460 No, just user, go process puls x Get back parm ptr leax 1,x Skip over 2nd '%' lda ,x Get shellsub variable # pshs x Save parm ptr leax >u08D2,u Point to shellsub variable contents table * Entry: A=ASCII 0-9 for either shell or shellsub variable # L0460 cmpa #'9 ASCII numeric? bhi L0470 No, skip ahead cmpa #'0 blo L0470 suba #$30 Yes, bump down to binary equivalent ldb #81 Point to proper variable entry within var table mul leax d,x incb ??? Used in TSTB below L0470 bsr L0486 Copy char from var table to pre-parse buffer cmpy 2,s Hit end of pre-parse buffer? beq L04A6 Yes, force CR at end of it, do full parse cmpa #C$CR End of variable? bne L0470 No, keep copying characters leay -1,y Knock ptr back one (to where CR is) puls x Get current parm ptr back tstb ??? flag to skip a byte in parm or not beq L042F Don't skip L0482 leax 1,x Skip to next byte in parm line bra L042F Continue pre-parse * Copy char from parm area to command line buffer - if '@', eat if user is * not super-user (#0) * Entry: X=ptr to current pos. in original parm buffer * Y=ptr to current pos. in new, pre-parsed buffer L0486 lda ,x+ Get char from parms cmpa #'@ Raw mode request? bne L049A Skip ahead if not (SHOULD BE BNE) L048C pshs y,a Preserve regs os9 F$ID Get user ID # cmpy #$0000 Is it 0 (superuser)? (should be leay ,y) puls y,a Restore regs beq L049A Yes, allow char thru rts Otherwise eat it L049A sta ,y+ Save char & return rts * Shell variable '%* (for last error code) requested * Put contents of shell var. into pre-parsed command line buffer L049D puls x Get back parm ptr lbsr L168A Put error code into preparse buffer leay 3,y Skip over error code space we just added bra L0482 Skip over shell varname, continue preparse L04A6 leas 2,s Eat stack L04A8 lda #C$CR Force CR as last char in buffer sta ,y puls y leax >u166D,u Point to start of pre-parse buffer again L04B2 bra L04CE Start real parse L04B4 fcb C$LF fcc 'Expanded line too long' fcb C$CR L04CC fcc '.' L04CD fcb C$CR * Parse command line buffer - already pre-parsed (user 0 RAW mode checks & * shell/shellsub variable expansion is already done) * Entry: X=Ptr to pre-parsed command line buffer L04CE lda ,x Get 1st char from parameter area cmpa #'* Is it a comment? beq L04DE Yes, skip ahead cmpa #': Is it a wildcard on/off? bne L04DA No, skip ahead leax 1,x Bump ptr past it L04DA cmpa #': Is it a wildcard off? * FOLLOWING LINE: BEQ means wildcarding default off, BNE = on beq L04F0 No, go process wildcarding * No wildcard processing L04DE leay >u0E6D,u Point Y to expanded buffer lbsr L1320 Copy param area to buffer area until 1st CR leax >u0E6D,u Point X to expanded (wildcard) buffer ldy #$0800 Max. size of expanded buffer (2048 bytes) lbra L079B Process line without wildcards * Wild carding processor * Entry: X=ptr to current position in pre-parsed parm line * 1st, set up error msg for if buffer gets full L04F0 leay >L04B4,pc Point to 'Expanded line too long' sty <u004A Save it ldy #$0018 Get size of text sty <u004C Save it too leay >L04CD,pc Point to CR sty <u0048 Save ptr sts <u0050 Save stack ptr leay >u0E6D,u Point to fully expanded buffer (2k max) sty <u0056 Save it clra clrb sta <u0012 No current working DIR path sta <u004F Flag no pathname in parm area std <u0052 Expanded buffer size=0 bytes bra L0520 Enter main loop for wildcards L051D lbsr L06FB Special shell chars handling L0520 lbsr L0746 Check for special chars (wildcard or shell) bcc L051D Special shell char process * Wildcard char found pshs x Save current pre-parsed parm ptr bsr L0549 Check from that point for '/' or special char ldx ,s Get previous parm ptr back bcc L0557 No '/' found, open current dir '.' * Found '/' (part of path) - keep looking for last '/' for filename (not * pathname) portion L052D bsr L0549 Check for special char or '/' again bcc L0535 No '/', skip ahead stx <u0040 Save latest ptr to '/' in parm line found so far bra L052D See if any more sub-paths * Found one or more '/' path dividers in current parm line. * Entry: <u0040 - contains the last '/' found L0535 inc <u004F Flag that their IS a pathname from parm line L0538 ldx <u0040 Get last level of path ptr puls y Get original parm ptr sty <u0040 Save it pshs x Save filename ptr on stack lda #C$SPAC Save space over '/' after pathname sta -1,x (before filename) tfr y,x Move original parm ptr to X bra L055E Open dir path * Parse from current position in pre-parsed parm line until shell special * char, or '/' found (path) * Entry: X=ptr to current pos in pre-parsed parm line * Exit: Carry clear if special shell char found * Carry set if '/' found * X=ptr to found char+1 L0549 clrb lda ,x+ Get next char lbsr L05D3 See if shell special char beq L0556 Yes, return cmpa #'/ No, is it a slash (path start)? bne L0549 No, Keep checking for / or special char comb L0556 rts * No '/' found in parm line at all L0557 clr <u004F Flag no pathname from parm line leax >L04CC,pc Point to '.' * Entry: X=ptr to pathname to directory * 0-1,s = Ptr to filename spec we are looking for in this directory L055E lda #DIR.+READ. Open directory in READ mode os9 I$Open lbcs L0776 Error, skip ahead sta <u0012 Save current DIR path puls x Get back ptr to filename (not pathname) lbsr L0746 Check for special shell char or wildcard lbcc L06E3 Special shell char found, ?????? L0572 lda ,x+ Get next char from filename cmpa #C$CR End of user specified filename (CR)? lbeq L0789 Yes, close DIR and proceed cmpa #', Comma? beq L0572 Yes, skip it cmpa #C$SPAC Space? beq L0572 Yes, skip it leax -1,x Other char, point to it. stx <u0054 Save ptr to it lda <u0012 Get DIR path # pshs u Preserve u ldx #$0000 Skip '.' and '..' dir entries ldu #$0040 os9 I$Seek lbcs L0776 Error, skip ahead puls u Restore u L0599 leax >u0C9D,u Point to dir entry buffer pshs x Save ptr lda <u0012 Get Current dir path # ldy #$0020 Read 1 file entry os9 I$Read lbcs L06BD Error, skip ahead puls y Restore pointer to dir filename buffer lda ,y Get 1st char of entry tsta Is it a deleted file? beq L0599 Yes, skip it ldx <u0054 Get ptr to current char in expanded filename bfr bsr L05EF Check wildcard spec against this filename bcs L0599 No match, skip to next file in DIR tst <u004F Was a pathname specified? beq L05CA No, skip ahead ldx <u0040 Yes,get ptr to end of pathname/start of filename lbsr L06FB Check for wildcard stuff there ldx <u0056 Get current pos in expanded buffer lda #'/ Save '/' there (to separate path & filename) sta -1,x L05CA leax >u0C9D,u Point to start of matched DIR entry filename lbsr L06FB Copy filename over, handling quoted chars, etc. bra L0599 On to next DIR entry * Check if shell 'special' char. (except wildcards & shell var) * non-wildcard char in current byte of pre-parsed parm buffer * Entry: X=ptr to next char in parms buffer * A=current char in parms buffer * Exit: BEQ if shell special char found, BNE if just regular char * A=char we were checking L05D3 pshs x Save parms buffer ptr cmpa #'( Group start char? beq L05ED Yes, skip ahead cmpa #') Group end char? beq L05ED Yes, skip ahead cmpa #C$SPAC Space? beq L05ED Yes, skip ahead cmpa #', Comma? beq L05ED Yes, skip ahead leax >L0408,pc Table of other special chars L05E9 cmpa ,x+ Found match or end of table? bhi L05E9 No, keep checking (or fall through if not found) L05ED puls pc,x Exit with BEQ/BNE flag set * IF wildcards were detected in preparsing, filename compares happen here * Entry: X=ptr to current char in user's pre-parse parm line * Y=ptr to start of current DIR entry filename * Exit: Carry set if no match * Carry clear if match L05EF ldd ,x+ Get 2 chars (NOTE: PTR UP BY ONLY 1!) L05F1 cmpa #'* 1st char a * (multi-char wildcard)? beq L064C Yes, go handle it cmpa #'? Question mark (single char wildcard)? beq L0638 Yes, go handle it cmpa #'[ Start of ranged wildcard? lbeq L068B Yes, go handle it bsr L05D3 Not wildcard, check for special shell chars beq L062B It is a special shell char, skip ahead bsr L0631 Just regular char, force uppercase pshs a Save char bsr L062D Force uppercase on DIR filename char eora ,s+ Same char as last parm char? bne L062B No, exit with carry set L060D lda ,y+ Re-get char from DIR filename bpl L05EF Not on last char yet, check next char from parm ldd ,x At end of DIR filename, grab 2 chars from parm bsr L05D3 Check 1st char against special shell chars beq L0621 Found one, skip ahead cmpa #'* Multi-char wildcard char? bne L062B No, no match, exit with carry set tfr b,a 1st char from parm is '*', check 2nd char for bsr L05D3 special shell chars bne L062B None found, no match, exit with carry set L0621 lda -1,y Special char, get last char from DIR filename L0623 anda #$7F Strip 'end of filename' bit flag ldb #C$SPAC Space char std -1,y Save 'fixed' last char & space clrb Flag match rts L062B comb Flag no match rts * Force char to uppercase L062D lda ,y Get char anda #$7F Strip hi-bit L0631 cmpa #'A Need to force uppercase? blo L0637 No, exit anda #$DF Yes, force to uppercase L0637 rts * '?' single char wildcard found L0638 cmpb #'* Is next char a multi-char wildcard? beq L060D Yes, process as if just '*' cmpb #', 2nd char '-' or greater? bhi L060D Yes, process normally lda ,y+ Get next char from DIR filename bpl L062B Not end of filename, Flag no match bra L0623 Save hibit stripped char & space, flag match L0646 lda ,y+ Get next char from DIR filename bpl L0646 Hunt for end of DIR filename bra L0623 Found it, fix hibit and add space to DIR entry * '*' multi-char wildcard found L064C lda ,x+ Get next char after '*' from parm buffer bsr L05D3 Check for shell special char beq L0646 Found one, check if end of DIR filename cmpa #'? Single char wildcard next? beq L067F Yes, Process cmpa #'[ Start of Ranged wildcard next? beq L067F Yes, process bsr L0631 Force char to uppercase pshs a Save it L065E bsr L062D Get next char from DIR filename, force uppercase eora ,s+ Same char? beq L066E Yes, possible resync after '*' L0664 leas -1,s Make room on stack lda ,y+ Re-get un-modified char from DIR filename bpl L065E Not end of filename, try next char leas 1,s Found end of filename, eat temp stack bra L062B Flag no match * Above loop @ L0664/L065E uses sneaky stack stuff * Found possible resync char match after '*' L066E leas -1,s Make room on stack pshs y,x Preserve both DIR & parm ptrs bsr L060D Attempt normal matching using this resync puls y,x Restore ptrs leas 1,s Eat temp stack bcs L0664 No, resync did not work, look for next resync rts worked, exit with carry clear L067B lda ,y+ Get next char in DIR bmi L062B Last char, flag no match * '?' found after '*' in parm buffer L067F pshs y,x Preserve both DIR and parm buffer ptrs ldd -1,x Get previous & current parm chars lbsr L05F1 Do normal sub-parsing from here puls y,x Restore ptrs bcs L067B No match, go to next char in DIR and attemp resync rts Matched, exit with match * Ranged wildcard here * Entry: X=ptr to 1st char in parm buffer AFTER '[' L068B ldd ,x+ Get 1st 2 chars of range sub-string from parm bfr bsr L0631 Force uppercase on 1st char exg b,a Force uppercase on 2nd char bsr L0631 exg b,a cmpa #'] Is 1st char a close range check char? beq L062B Yes, flag no match for '[]' cmpa #'- Range separator char? beq L06A7 Yes, need to get end of range * Special case for [x] - acts as if just normal char (no range or wildcard) sta <u00EF No, save start of range char bsr L062D Get char from DIR filename, force uppercase eora <u00EF Match only range char? beq L06B4 Yes, skip to next char now * Special case for [xyz] - one of these chars must match bra L068B No, see if more single chars to try matching * Actual range - need to get end of range char * Entry: B=high char of range * <u00EF - current char (within range) we are checking against L06A7 inc <u00EF Bump up start char's ascii value cmpb <u00EF Hit end of range yet? beq L068B Yes, try next char lbsr L062D Force char in A to uppercase eora <u00EF equal to current range check char? bne L06A7 No, try next char in range L06B4 ldd ,x+ Get next 2 chars from pathname cmpa #'] End of range? bne L06B4 No, check next char lbra L060D End of range specifier, process normally * Error reading from current DIR L06BD cmpb #E$EOF End of file error? lbne L0776 No, fatal error, leave ldx <u0054 Get ptr to current char in wildcard filename lda -1,x Get last char processed bsr L0714 Add it to new expanded buffer L06C9 lda ,x+ Get next char, inc ptr stx <u0054 Save updated pointer cmpa #C$CR CR? lbeq L0789 Yes, append CR to expanded buffer, we are done lbsr L05D3 See if special shell char bne L06C9 Just regular char, go to next one pshs x Save current ptr to wildcard filename ldx <u0056 Get current ptr in expanded buffer sta -1,x Save that char overtop last char written puls x Get wildcard ptr back bra L06EA Close dir path, go back into loop * If special shell char found right after OPEN of DIR path * Entry: X=ptr to filename (not pathname) - but with special char L06E3 bsr L06FB Process special shell char copying bsr L072F Close DIR path lbra L051D More processing L06EA bsr L072F Close DIR path lbra L0520 Go back to main wildcard processing loop * This chunk (6EF-6F9) is for copying '\x' when x is NOT a wildcard char L06EF lda ,x+ Get quoted char bsr L0739 Check if it is [, * or ? beq L06F9 Yes, add that char to expanded buffer by itself leax -2,x No, bump ptr back to '\' lda ,x+ Get '\' char L06F9 bsr L0714 Append that to output buffer * Special shell chars found goes here * This part copies filename from [,x] to expanded buffer, handling quoted * wildcard chars, and ending on CR or special shell char L06FB lda ,x+ Get char cmpa #'\ Backslash (for quoted char)? beq L06EF Yes, go get quoted char cmpa #C$CR Is it the end of the filename? beq L0789 Yes, append CR to expanded line, we are done bsr L073D Is it '?' or '*' beq L0714 Yes, add that char to expanded buffer lbsr L05D3 Check if shell special char beq L0714 Yes, add to expanded buffer, return from there bsr L0714 No, add to expanded buffer, stay in this loop bra L06FB * Add char to expanded line buffer * Entry: A=char to append to expanded line buffer * <u0056=Current position in expanded line buffer * <u0052=Current size of expanded line buffer * Exit: <u0056 & <u0052 updated L0714 pshs x,a Preserve regs ldx <u0056 Get current pos in expanded buffer sta ,x+ Save char stx <u0056 Save updated expanded buffer ptr ldd <u0052 Get expanded buffer size addd #$0001 Increase by 1 cmpd #2048 Is it full yet? bhi L0773 Yes, exit with expanded line too long error std <u0052 No, save new size puls pc,x,a Restore regs & return * Close DIR path L072F lda <u0012 Get DIR path # beq L0738 If none, exit os9 I$Close Close the path clr <u0012 Clear DIR path # to none L0738 rts * Wildcard checks * Entry: A=char to check * Exit: BEQ if any of the 3 wildcard chars, or BNE if not L0739 cmpa #'[ Range wildcard? beq L0745 L073D cmpa #'? Single char wildcard? beq L0745 cmpa #'* Multi-char wildcard? beq L0745 L0745 rts * Parse for next wildcard or special shell char in pre-parsed parm line * Entry: X=current pos in pre-parse parm line * Exit: X=Same as entry * IF WILDCARD CHAR FOUND: * B=# chars to next wildcard/special char * A=special char found * Carry bit set * IF SPECIAL SHELL CHAR FOUND * B=0 * Carry bit clear * A=special char found: CR ( ) , space ! # & ; < > ^ | L0746 pshs x Save current parm ptr bra L0752 Parse for wildcard or special shell chars L074A lda ,x+ Get char from parms bsr L0739 Do wildcard char check beq L0752 If wildcard char, skip over it & continue leax -1,x not wildcard char, bump back to char after \ L0752 lda ,x+ Get char from parms cmpa #'\ Wildcard quote char (do not expand next)? beq L074A Yes, handle it lbsr L05D3 No, check for other special shell chars beq L0770 Found one, skip ahead bsr L0739 Check for wildcard char bne L0752 None, skip to next char * One of the 3 wildcard chars found, process pshs a Save char tfr x,d Move parm ptr to D subd 1,s Calc distance since last wildcard/special char subd #$0001 -1 puls a B=distance between wild/special chars orcc #Carry Set Carry Flag puls pc,x Exit with original parm ptr in X * Exit with B=0, Carry clear and A=special char found (which ends current * 'chunk' being checked... can included CR) L0770 clrb Clear carry, exit puls pc,x * Expanded buffer full error L0773 comb ldb #E$BufSiz L0776 pshs b,cc Preserve error code bsr L072F Close DIR path (if we had one) ldx <u004A Print text message (usually error msg) ldy <u004C lda #$02 os9 I$WritLn puls b,cc lbra L0191 Exit with error L0789 lda #C$CR Append CR to expanded filenames buffer bsr L0714 ldy <u0052 Get expanded buffer size leax >u0E6D,u Point to start of expanded buffer bsr L072F Close DIR path lds <u0050 Get back original stack ptr * At this point, expanded buffer is complete and ready to go (either through * wildcard routine, or by simply copying the original parm buffer from user) * Main entry point for non-wildcard filename search/match * Entry: X=Ptr to start of expanded buffer L079B tst <u001E Echo required? beq L07A2 No, skip ahead lbsr L021F Print out user entered parm line to std err L07A2 tst <u0043 2=FALSE,0=TRUE, 1=??? (check current IF status, beq L07B0 if any. If TRUE flag set, skip ahead) lbsr L0F69 Do checks for IF type statements tsta Find match? lbpl L0F1D Yes, process IF type statement L07AE clrb No error & return rts * If current IF type statement has result TRUE L07B0 tst <u0045 ??? (Are we looking for a label for GOTO?) beq L07E3 No, skip ahead lda ,x+ Get char from buffer cmpa #'* Comment line? bne L07AE No, exit without error lbsr L091F Yes, get ptr to first non-space char into X leay >u0124,u Point to temp buffer lbsr L100B Copy label from X to Y lda ,x Get 1st char of label cmpa #'\ '*\'? (Which means cancel impending GOTO) beq L07DB Yes, cancel GOTO leax >u0BFC,u No, point to 80 byte buffer (GOTO label holder?) clra default char to check against=NUL L07CF cmpa #C$CR End of label? beq L07DB Yes, cancel GOTO search lda ,x+ Get next char from buffer cmpa ,y+ Is it the label we are looking for? beq L07CF Keep checking till done bra L07AE No match, clear carry & return * Cancelling GOTO search L07DB clr <u0045 Flag we are NOT looking for label for GOTO clrb leas 2,s Eat stack lbra L010D Reprint shell prompt, process from scratch L07E3 bsr L07E7 bra L083A L07E7 pshs x tst <u006C beq L0832 tst <u005C bne L07F5 tst <u001C beq L0832 L07F5 lda ,x cmpa #'* beq L0832 IFGT Level-1 ldy #$0000 Force temporarily to super-user os9 F$SUser ENDC leax >u0CBD,u ldd #$0203 lbsr L0C79 bcs L0832 lbsr L00FB lda #$20 sta <$11,x ldy #$0012 lda <u0012 os9 I$Write bcs L082D ldx ,s ldy #$0800 lda <u0012 os9 I$WritLn L082D lda <u0012 os9 I$Close L0832 IFGT Level-1 ldy <u005A Get original user # back os9 F$SUser Change user # to original ENDC puls pc,x * Parse input line * Entry : X=Ptr to current char in line buffer L083A clra sta <u0022 Flag we don't change priority for forked process sta <u0003 Clear out # pages of data mem for forked process sta <u000E Clear out signal code leay >L0300,pc Point to main command list lbsr L08D1 Parse keywords from list bcs L08A0 Keyword found but generated error, done line cmpa #C$CR Is 1st non-keyword char a carriage return? beq L08A0 Yes, done line sta <u000C Save non-keyword char cmpa #'( Is it a 'start command group' char? bne L087B No, try next leay >L000D,pc Point to 'Shell' sty <u0004 Save pointer to program to fork? leax 1,x Bump ptr past '(' stx <u0008 Save updated ptr L0860 inc <u000D Bump up # of command groups L0862 leay >L0404,pc Point to command group symbols to parse lbsr L08E9 Hunt down EOL, or '(' or ')' (skip quoted text) cmpa #'( Another group start? beq L0860 Yes, bump up # of command groups & continue cmpa #') Group end? bne L0898 No, skip ahead (to print WHAT?) dec <u000D Bump down # of command groups bne L0862 Still more groups, continue parsing for them lda #C$CR Append CR at end of command line sta -1,x bra L087F Check for modifiers L087B bsr L08A3 Check for valid path, do '<>', '#', '^' if needed bcs L08A0 No valid path found, skip ahead * Found valid pathname L087F leay >L0408,pc Point to modifiers table bsr L08E9 Go parse for them tfr x,d Move ')' ptr to D for SUB subd <u0008 Calculate size from 1st '(' to ')' std <u0006 Save size of group (could also be pathname-modifier) leax -1,x leay >L03CF,pc Point to modifier branch table bsr L08D1 Go execute modifier routine bcs L08A0 If error in modifier routine, exit ldy <u0004 Get ptr to first char we started at L0898 lbne L0BCF Not proper, print 'WHAT?' cmpa #C$CR Last char a carriage return? bne L083A No, start parsing again at current position L08A0 lbra L0B96 Done processing line, continue * Entry: X=Ptr to text to check for valid device name (including '.' & '..') * Exit: Carry set if not a valid device name * Carry clear if there was * <u0004 = Ptr to first char where we started at * <u0008 = Ptr to end of redirection/mem size/priority setting chars (if found) L08A3 stx <u0004 Save ptr to current char in input line bsr L08B6 Check for valid device name bcs L08B5 None found, return * Found valid device name L08A9 bsr L08B6 Eat rest of valid pathlist bcc L08A9 Keep going until we are done pathlist leay >L03DE,pc Point to Command list starting at '<>>>' bsr L08D1 Call any redirection, mem size or priority routines stx <u0008 Save ptr to where we are now (end of current set) L08B5 rts L08B6 os9 F$PrsNam Valid OS9 device name? bcc L08C7 Yes, point X to it & return lda ,x+ Not valid, get first char cmpa #'. Is it a period? bne L08CB No, bad path name cmpa ,x+ Is it a double period? beq L08C9 Yes, leave src ptr pointing to name after '..' leay -1,x If single, bump ptr to name after '.' L08C7 leax ,y Point X to pathname in source L08C9 clra No error & return rts L08CB comb Error flag leax -1,x Bump source ptr back ldb #E$BPNam Bad Path Name error & return rts * Entry: Y=ptr to command list (L0300) L08D1 bsr L0907 Go find 1st non-space char pshs y Save command list ptr bsr L092A Parse for keyword or special char bcs L08E2 If no keyword found, skip ahead ldd ,y Keyword found, get offset jsr d,y Go execute routine for command found puls y Restore command list ptr bcc L08D1 No error, continue parsing for keywords rts Subroutine had error, return with it L08E2 clra No error lda ,x Get character (not in command list) puls pc,y Restore command list ptr & return * Start searching at beginning of current command list * For looking for single character modifiers L08E7 puls y L08E9 pshs y Preserve command list ptr lda ,x+ Get next char from command line L08ED tst ,y Check current char in command list bmi L08E7 If done list, start over at beginning cmpa #'" Is it quotes? bne L0901 No, skip ahead L08F5 lda ,x+ Get next char from command line cmpa #C$CR EOL? beq L0905 Yes, exit with A being CR cmpa #'" Is it another set of quotes? bne L08F5 No, keep looking for it (or EOL) lda ,x+ Get char after 2nd quotes L0901 cmpa ,y+ Is it the current command char we are checking? bne L08ED No, try next L0905 puls pc,y Yes, exit with A containing it * Entry: Y=ptr to command list L0907 pshs x Preserve X lda ,x+ Get char from line entered by user cmpa #C$SPAC Is it a space? beq L091D Yes, skip ahead cmpa #', Is it a comma? beq L091D Yes, skip ahead leax >L0408,pc Point to single character modifiers L0917 cmpa ,x+ Found a match? bhi L0917 No, try next until list is done puls pc,x Found it, restore X to point to it & return L091D leas 2,s Eat X off the stack L091F lda #C$SPAC Get space character L0921 cmpa ,x+ Keep searching until non-space char is found beq L0921 leax -1,x Point to 1st non-space char * ; (1st pass) comes here L0927 andcc #^Carry No carry & return rts * Command line parser L092A pshs y,x Preserve command table ptr & input line ptr leay 2,y Skip first offset L092E ldx ,s Get input line ptr L0930 lda ,x+ Get char from input line lbsr L0F0C Convert char to uppercase if lowercases eora ,y+ Check for a match lsla bne L0951 No match, skip to next keyword in table bcc L0930 Keep checking until end of keyword (high bit set) lda -$01,y Get command table char again cmpa #'|+$80 '|' (with high bit set)? beq L094E Yes, exit with carry clear cmpa #'^+$80 '^' (with high bit set)? beq L094E Yes, exit with carry clear cmpa #'A+$80 Any of the other special modifiers with high bit? blo L094E Yes, exit with carry clear bsr L0907 Eat spaces until first non-space char bcs L0951 If special char (!|#,etc.), skip ahead L094E clra puls pc,y,d L0951 leay -1,y Bump search ptr back L0953 lda ,y+ Get char again bpl L0953 Keep getting them until end of keyword sty 2,s Save ptr to next offset on stack ldd ,y++ Get offset bne L092E If not at end of table, keep searching comb End of table, command not found error puls pc,y,x L0961 fcc 'startup' fcb C$CR * Create child shell to run 'startup' file L0969 pshs u,y,x Preserve regs leax L000D,pc Point to 'shell' (module name) leau L0961,pc Point to 'startup' (parameter for 'shell') ldy #$0008 Size of 'startup<CR>' IFGT Level-1 ldd #$111F Program+Objct / 7.5K data area ELSE ldd #$1102 Program+Objct / 512 byte data area ENDC os9 F$Fork Fork a shell to run the startup file bcs L0983 Couldn't fork, exit os9 F$Wait Wait until 'startup' is done L0983 puls u,y,x Restore regs clrb No error & return rts * EX command L0987 lbsr L08A3 Go check for valid device name (module) bcs L09AB If none, exit clra Std in path bsr L09B0 Go close it bsr L09AF Go close Std out bsr L09AF Go close Std Err lbsr L0B87 Go find the end of the input line leax 1,x Bump ptr to 1 past CR tfr x,d Move ptr to D subd <u0008 Calculate size of current command group std <u0006 Save it lbsr L130A Point to module to chain&get its parm size, etc. leas >u00FF,u Point stack to end of DP os9 F$Chain Chain to the new program lbra L01FA Couldn't, go here L09AB clrb No error lbra L020D Close logging file (if any) and exit ShellPlus L09AF inca Inc path # L09B0 pshs a Save path # lbra L0BBC close path if it is open * CHX & CX commands L09B5 clr <u0038 lda #DIR.+EXEC. os9 I$ChgDir rts * CHD & CD commands L09BD lda #DIR.+READ. (bug fix, originally opened in UPDATE) os9 I$ChgDir Change the directory bcs L09CE Error, exit with it clr <u0037 Flag .pwd entry as invalid tst <u0042 WAS ,U beq L09CE bsr L0A04 Go update shell expanded prompt if needed L09CE rts * L command - Logging ON L09CF lda #$01 bra L09D4 * -L command - Logging OFF L09D3 clra L09D4 sta <u001C rts * P command - Prompting ON L09D7 clra bra L09DC * -P command - Prompting OFF L09DA lda #$01 L09DC sta <u001D rts * T command - Echo input ON L09DF lda #$01 bra L09E4 * -T command - Echo input OFF L09E3 clra L09E4 sta <u001E rts * V command - Turn variable expansion ON L09E7 clra bra L09EC * -V command - Turn variable expansion OFF L09EA lda #$01 L09EC sta <u001F rts * X command - Kill Shell when error occurs ON L09EF lda #$01 bra L09F4 * -X command - Kill Shell when error occurs OFF L09F3 clra L09F4 sta <u0020 rts L09F7 tst >u01F4,u Date already done? bne L0A04 Yes, skip ahead tst >u01F5,u Time already done? bne L0A04 Yes, skip ahead rts If neither, we don't need to change prompt? L0A04 pshs y,x Preserve regs bra L0A64 Go update shell expanded prompt if needed * Make shell prompt string the default one L0A08 puls y Restore Y pshs x Preserve X leax >L003D,pc Point to default prompt string bsr L0A14 Put that into working shell prompt string puls pc,x Restore X & return * P= (prompt set) command * Make shell prompt string (default or user-specified) * Entry: X=ptr to source of new shell prompt string L0A14 pshs y Preserve Y leay >u01F9,u Point to working prompt text buffer ldd #C$LF*256+22 Line feed & max count for prompt string+1 sta ,y+ Save LF as first char bsr L0A25 Go copy prompt string (& parse quotes) bra L0A5E Go see if we need to override with default L0A25 clr >u01F3,u Clear quotes in progress flag L0A29 lda ,x+ Get char from default shell prompt cmpa #'" Is it a quotes? bne L0A3F No, skip ahead leax 1,x Yes, bump ptr up 2 past quotes tst >u01F3,u We processing quotes already? bne L0A59 Yes, we are done then inc >u01F3,u Set processing quotes flag leax -1,x Set ptr back to char just after quotes bra L0A29 Check next char L0A3F cmpa #C$CR Carriage return? beq L0A59 Yes, we are done then cmpa #C$SPAC Space? beq L0A4B Yes, skip ahead cmpa #'; Semi-colon? bne L0A51 No, skip ahead * Semi-colon or space found L0A4B tst >u01F3,u We quoting at the moment? beq L0A59 No, we are done L0A51 tstb Char count down to 0? beq L0A29 Yes, continue parsing (but eating them) decb Dec max char counter sta ,y+ Save literal char into current copy bra L0A29 continue parsing L0A59 clr ,y Append NUL to indicate end of string leax -1,x Bump source ptr back to last char & return rts L0A5E cmpb #22 Did the user do a no-length prompt? beq L0A08 Yes, go override with default prompt pshs x Preserve ptr to last char of source prompt * Create expanded shell prompt from shell prompt string L0A64 leay >u01F9,u Point to shell prompt string leax >u0212,u Point to expanded prompt buffer pshs x Preserve it a moment clr >u01F2,u Output device name done already = OFF clr >u01F1,u Process ID # done already = OFF clr >u01F4,u Date done already = OFF clr >u01F5,u Time done already = OFF clr >u01F6,u Date OR time already done once = OFF clr >u0042,u Current working dir name done already = OFF L0A86 lda ,y+ Get char from shell prompt string lbeq L0B29 If end of string, exit cmpa #'# Process ID # wanted? bne L0AA2 No, try next * Process ID # tst >u01F1,u Done it already? bne L0A86 Yes, skip doing it again inc >u01F1,u No, flag it as being done now ldd >u01EC,u Get process # (01-99) std ,x++ Save in expanded prompt buffer bra L0A86 Continue parsing L0AA2 cmpa #'@ Standard output path device name wanted? beq L0AAA Yes, go do it cmpa #$E0 High bit version? bne L0AC7 No, try next * Standard output path device name L0AAA tst >u01F2,u Done it already? bne L0A86 Yes, skip doing it again ldd #$01*256+SS.DevNm Standard out/get device name os9 I$GetStt Get the device name bcs L0A86 Error, skip doing device name inc >u01F2,u Flag it as being done now L0ABD lda ,x+ Get char from device name bpl L0ABD Keep looking until last char (high bit set) anda #$7F Mask out high bit sta -1,x Save the normal char bra L0A86 Continue parsing L0AC7 cmpa #'$ Current working directory wanted? bne L0AF2 No, check next * Current working directory tst <u0042 Done it already? (WAS ,U) bne L0A86 Yes, skip doing it again inc <u0042 Flag it as being done now (WAS ,U) lda #$01 sta <u003D lbsr L176D Go figure out current working directory tstb bne L0AF2 pshs y Save prompt string ptr ldy <u002A Get pointer to current working directory * Copy string: Y=source string ptr, X=prompt buffer ptr L0AE4 lda ,y+ Get char sta ,x+ Save as part of shell prompt text cmpa #C$CR Was it the end? bne L0AE4 No, keep copying leax -1,x Bump ptr back CR puls y Restore source string ptr bra L0A86 Continue parsing L0AF2 cmpa #'( Current Date wanted? bne L0B0A No, check next * Current date tst >u01F4,u Done it already? bne L0A86 Yes, skip doing it again inc >u01F4,u Flag as being done now bsr L0B3B Go get date & time pshs y Save shell prompt string ptr leay >u02E0,u Point to date text bra L0AE4 Copy into expanded prompt L0B0A cmpa #') Current Time wanted? bne L0B24 No, just store the raw ASCII char * Current time tst >u01F5,u Done Time already? lbne L0A86 Yes, skip doing it again inc >u01F5,u Flag as being done now bsr L0B3B Go get date & time pshs y Save shell prompt string ptr leay >u02E9,u Point to time text bra L0AE4 Copy into expanded prompt L0B24 sta ,x+ Save raw character lbra L0A86 Continue parsing L0B29 sta ,x Save NUL to mark end of prompt tfr x,d Move End of prompt ptr to D for subtract subd ,s++ Subtract start of prompt ptr std >u01F7,u Save size of expanded shell prompt puls pc,y,x Restore regs & return * Separator table for date & time strings L0B35 fcc '//' fcb $0d fcc '::' fcb $0d * Get current date (2E0-2E8) & time (2E9-2EF) L0B3B pshs y,x Preserve shell prompt string & shell prompt ptrs tst >u01F6,u Have we already been here before? bne L0B75 Yes, both date & time strings already done inc >u01F6,u Flag we have been here now leax >u02DA,u Point to date/time packet buffer os9 F$Time Get the date/time packet leay >u02E0,u Point to where date string goes clrb Offset into separator table to first '/' L0B54 lda ,x+ Get byte from time packet bsr L0B77 Convert to ASCII pshs b Preserve offset into separator table ldd >u01EF,u Get ASCII version of byte std ,y++ Save into date/string buffers puls b Restore offset into separator table pshs y Preserve date/string buffer ptr leay >L0B35,pc Point to separator's table lda b,y Get current separator puls y Restore date/string buffer ptr sta ,y+ Save separator into date/string buffer incb Point to next separator cmpb #6 On last one? bne L0B54 No, continue converting L0B75 puls pc,y,x Restore prompt & buffer ptrs & return L0B77 pshs y,x,d Preserve regs leay >L0B80,pc Point to routine to copy ASCII digits lbra L16B9 Go convert byte to ASCII equivalent L0B80 ldd $04,s Copy 2 digit ASCII # to 1EF std >u01EF,u rts * Searches for CR in string pointed to by X * '*' Comment lines come here L0B87 lda #C$CR We want to find the CR L0B89 cmpa ,x+ Found it yet? bne L0B89 No, keep looking cmpa ,-x Set up flags & return rts L0B90 pshs d,cc Preserve regs lda #$01 Only do std in & out (not error) bra L0B9A * Any errors from any of the L0300 subroutines go here * If child process had error/status code it goes here (u005D cleared,B=Status * code) L0B96 pshs d,cc Preserve error code, flags & A lda #$02 # of paths to do L0B9A sta <u001A Save it clra Start at path 0 L0B9D bsr L0BA8 Go close (possibly dupe) paths inca Next path # cmpa <u001A Done up to last one yet? bls L0B9D No, do next one * POSSIBLY COULD BE PULS PC,D,CC ror ,s+ Eat CC but shift Carry bit into Carry puls pc,d Restore error code & A & return L0BA8 pshs a Save path # tst <u0019 bmi L0BC4 If high bit set, close path bne L0BBC If 0<u0019<128, get changed path # & close tst a,u Check 'real' path # from DP beq L0BC7 If 0, return os9 I$Close Otherwise, close current path # lda a,u Get 'real' path # os9 I$Dup Dupe it * Close path # on stack, if it is open L0BBC ldb ,s Get path # from stack lda b,u Get real path # from DP beq L0BC7 If none, exit clr b,u Clear out path # L0BC4 os9 I$Close Close the path L0BC7 puls pc,a Exit L0BC9 fcc 'WHAT?' fcb C$CR L0BCF bsr L0B96 Close 3 std paths (possibly dupe) leax <L0BC9,pc Point to 'WHAT?' lbsr L021B Write it out std err clrb coma rts L0BDA inc <u0019 ??? bsr L0B96 Do path closings (possibly dupings) lda #$FF Set flag to just close raw paths sta <u0019 bsr L0B90 Go close std in & std err leax <u006D,u Point to device name buffer lbsr L0CDB lbcs L0200 lda #$02 bsr L0BA8 lbsr L0CFF clr <u0019 lbra L00CC * < processing L0BFA ldd #$0001 orb <u000F bra L0C1A * >> processing L0C01 ldd #$020D stb -$02,x bra L0C0A * > processing L0C08 lda #$01 L0C0A ldb #$02 bra L0C1A * if from z= or i=, A=0, B=3 L0C0E tst a,u Test duped path? bne L0BCF There is one, print 'WHAT' & close paths & return pshs d Save path # & B tst <u0019 bmi L0C34 bra L0C24 L0C1A tst a,u bne L0BCF pshs d ldb #C$CR stb -$01,x L0C24 os9 I$Dup Create duplicate of the standard path lbcs L0CBE Couldn't create dupe, ldb ,s sta b,u lda ,s os9 I$Close L0C34 lda 1,s Get B bmi L0C40 ldb ,s lbsr L0D05 tsta bpl L0C47 L0C40 anda #$0F os9 I$Dup bra L0CBE L0C47 ldb #$0B bita #$02 bne L0C7B pshs a ldd ,x andb #$5F cmpd #$2F57 Is it '/W' puls a bne L0C74 ora #$02 os9 I$Open bcs L0CBE pshs x leax >L003C,pc ldy #$0001 clra os9 I$Write puls x bra L0CBE L0C74 os9 I$Open bra L0CBE L0C79 pshs d L0C7B stb <u004E ldb ,x cmpb #$2B bne L0C96 leax $01,x os9 I$Open bcs L0CB1 pshs u,x ldb #SS.Size os9 I$GetStt os9 I$Seek bra L0CA8 L0C96 cmpb #'- bne L0CB9 leax 1,x os9 I$Open bcs L0CB1 Error opening pshs u,x ldx #$0000 tfr x,u L0CA8 ldb #SS.Size Init size of file to 0 bytes os9 I$SetStt puls u,x bra L0CBE L0CB1 cmpb #E$PNNF Error 216 (path name not found)? beq L0CB9 Yes, create the file orcc #Carry Otherwise, set error flag bra L0CBE L0CB9 ldb <u004E Get file attributes os9 I$Create L0CBE sta <u0012 Save path # (or one we tried to duplicate?) stb 1,s Save possible error code? lda #$00 DO NOT CHANGE-NEED TO PRESERVE CARRY sta <u000F puls pc,d Restore regs & return L0CC8 ldd #$0003 Std in & ??? lbra L0C0E * <>>> processing L0CCE lda #C$CR sta -$04,x * i= & z= both come here right off the bat L0CD2 bsr L0CDB bcc L0CFF L0CD6 rts * <> processing L0CD7 lda #C$CR sta -$02,x L0CDB bsr L0CC8 bcs L0CD6 ldd #$0180 lbra L0C0E * <>> processing L0CE5 lda #C$CR sta -$03,x bsr L0CC8 bcs L0CD6 ldd #$0280 lbra L0C0E * >>> processing L0CF3 lda #C$CR sta -$03,x ldd #$0102 lbsr L0C0E bcs L0CD6 L0CFF ldd #$0281 lbra L0C0E L0D05 pshs x,d ldd ,x++ cmpd #$2F30 bcs L0D2F cmpd #$2F32 bhi L0D2F pshs x,d lbsr L0907 puls x,d bcs L0D2F andb #$03 cmpb 1,s bne L0D31 ldb b,u L0D26 orb #$80 stb ,s puls d leas 2,s rts L0D2F puls pc,x,d L0D31 tst $01,s bne L0D26 pshs x tfr b,a leax >u00B5,u Point to buffer for device name ldb #'/ Put a slash in it stb ,x+ ldb #SS.DevNm Get the device name os9 I$GetStt bcs L0D4F Error, skip ahead leax -1,x Reset ptr to include '/' lda #UPDAT. os9 I$Open L0D4F puls x Restore ptr to beginning (including '/') leas 6,s Eat stack lbra L0CBE L0D56 fcc 'TRUE ' L0D5B fcb $0d L0D5C fcc 'FALSE' fcb $0d L0D62 lda ,x+ cmpa #'[ bne L0D6B lbsr L0E15 L0D6B cmpa #'- lbne L0E3E ldb ,x+ lbsr L0E15 leax -$01,x tfr b,a lbsr L0F0C Convert char to uppercase if lower cmpa #'Y bne L0DBB L0D81 pshs x Preserve X leax >u0124,u Point to buffer ldy #$0001 Read 1 byte from error path??? lda #$02 os9 I$Read lbcs L0F17 lda ,x Get the character read puls x lbsr L0F0C Convert char to uppercase if lower cmpa #'Y Unless char is Y or N, re-read it beq L0DA3 cmpa #'N bne L0D81 L0DA3 pshs a Preserve char on stack leax >L0D5B,pc Point to a Carriage return lda #$02 Print it to std out ldy #$0001 os9 I$WritLn puls a Restore char clrb cmpa #'Y beq L0DF8 Print 'true' if it is a Y bra L0DEE Print 'false' if it is a N L0DBB clrb cmpa #'F beq L0DE0 cmpa #'E bne L0DC8 orb #%00000100 bra L0DE0 L0DC8 cmpa #'R bne L0DD0 orb #%00000001 bra L0DE0 L0DD0 cmpa #'W bne L0DD8 orb #%00000010 bra L0DE0 L0DD8 cmpa #'D lbne L0F17 orb #%10000000 L0DE0 tfr b,a os9 I$Open bcs L0DEE os9 I$Close bra L0DF8 L0DEE lda #$02 sta <u0043 leax >L0D5C,pc Point to 'FALSE' bra L0DFE L0DF8 clr <u0043 leax >L0D56,pc Point to 'TRUE' L0DFE tst <u001E Command echo on? beq L0E0B No, skip ahead ldy #$0006 Print result of IF to std error lda #$02 os9 I$WritLn L0E0B leax >u0124,u lda #C$CR sta ,x clrb rts L0E15 lda ,x+ cmpa #C$SPAC beq L0E15 rts L0E1C cmpa #$3D bne L0E26 lda <u005F ora #$01 bra L0E38 L0E26 cmpa #'< bne L0E30 lda <u005F ora #$02 bra L0E38 L0E30 cmpa #'> bne L0E3C * X command - Kill Shell when error occurs ON lda <u005F ora #$04 L0E38 sta <u005F clra rts L0E3C coma rts L0E3E cmpa #'+ bne L0E46 inc <u0015 bra L0E48 L0E46 leax -1,x L0E48 clr <u005F pshs u leau >u0124,u ldb #180 Clear out 180 bytes @ u0124 lbsr L0412 puls u leay >u0124,u ldb #81 L0E5D lda ,x+ Copy buffer up to CR or 81 chars lbsr L0F0C Convert char to uppercase if lower sta ,y+ cmpa #C$CR lbeq L0F17 bsr L0E1C bcc L0E74 decb bne L0E5D lbra L0F17 L0E74 negb addb #81 stb <u0016 clra sta -$01,y lda ,x bsr L0E1C bcs L0E84 leax $01,x L0E84 leay >u0175,u ldb #81 L0E8A lda ,x+ bsr L0F0C Convert char to uppercase if lower sta ,y+ cmpa #C$CR beq L0E99 decb bne L0E8A bra L0F17 L0E99 negb addb #$51 stb <u0017 clra sta -$01,y tst <u0015 beq L0EE0 leax >u166D,u ldd #$30b4 Store 180 ASCII 0's into buffer L0EAD sta ,x+ decb bne L0EAD leax >u0124,u ldb <u0016 leax b,x leay >u16BD,u bsr L0ED8 leax >u0175,u ldb <u0017 leax b,x leay >u170E,u bsr L0ED8 leax >u166D,u leay >u16BE,u bra L0EE8 L0ED8 lda ,-x sta ,-y decb bne L0ED8 rts L0EE0 leax >u0124,u leay >u0175,u L0EE8 ldb #80 L0EEA lda ,x+ cmpa ,y+ blo L0EFB bhi L0F01 decb bne L0EEA lda <u005F bita #$01 bra L0F05 L0EFB lda <u005F bita #$02 bra L0F05 L0F01 lda <u005F bita #$04 L0F05 lbne L0DF8 lbra L0DEE * Convert char to uppercase if it is a letter L0F0C cmpa #'a Lower case letter? blo L0F16 No, return cmpa #'z Check high range bhi L0F16 No, return suba #$20 Yes, convert to uppercase L0F16 rts L0F17 comb ldb #$01 lbra L0191 L0F1D cmpa #$03 beq L0F3F cmpa #$02 bne L0F2B dec <u0044 blt L0F3F bra L0F43 L0F2B cmpa #$01 bne L0F3B lda <u0043 cmpa #$02 bne L0F43 tst <u0044 beq L0F3F bra L0F43 L0F3B inc <u0044 bra L0F43 L0F3F clr <u0043 clr <u0044 L0F43 clrb rts * Table: 7 bytes/entry: * 1st 5 bytes is name, high bit set & NUL padded * Byte 6 is # bytes actually used L0F45 fcs 'IF' fcb 0,0,0,2,0 fcs 'ELSE' fcb 0,4,1 fcs 'ENDIF' fcb 5,2 fcs 'FI' fcb 0,0,0,2,2 fcs 'CLRIF' fcb 5,3 fcb $ff L0F69 leay <L0F45,pc Point to conditionals table L0F6D ldb 5,y Get actual length of string we are checking os9 F$CmpNam Compare with string pointed to by X bcs L0F80 If they don't match, skip ahead lda 6,y Get conditional token(?) number ldb b,x Get char past end of matching string cmpb #C$CR Is it a CR? beq L0F8B Yes, return cmpb #C$SPAC Is it a space? beq L0F8B Yes, return L0F80 leay 7,y Point to next command in table lda ,y Get 1st char from this entry cmpa #$FF End of table marker? beq L0F8B Yes, return * NOTE: THIS INCA SEEMS TO BE USELESS, AS F$CMPNAM DOESN'T USE A inca No, ??? bra L0F6D Process this one L0F8C lda #$01 sta <u0043 lbra L0E0B L0F93 lbsr L0907 Go find 1st non-space char or single char modifier bne L0F9B clr <u0046 L0F8B rts L0F9B leay >L03AC,pc Point to 'GOTO' ldb #4 4 chars to compare os9 F$CmpNam Does it match? lbcs L0BCF No, print 'WHAT?' leax 4,x Yes, skip X past 'GOTO' lbsr L091F Go find 1st non-space char past 'GOTO' leay >u0C4C,u Point to some sort of buffer lda ,x Get char from GOTO label cmpa #'+ Is label after current pos. in script file? bne L0FBB No, skip ahead sta ,y+ Save '+' in buffer leax 1,x Bump up source ptr past '+' L0FBB bsr L100B Go copy label name into buffer inc <u0046 Set flag that a GOTO was found L0FBF lda ,x+ Get 1st char from user's label again leay >L0408,pc Point to single char modifiers table L0FC5 cmpa ,y+ Illegal modifier char in label name? bhi L0FC5 Not yet, check other modifiers blo L0FBF This char ok, check rest of label name leax -1,x Point to last char (terminator) of label name stx <u0058 Save it & return rts L0FD0 lda ,x cmpa #'+ bne L0FDA leax 1,x bra L0FFB L0FDA tst <u006B beq L0FEA ldy <u0067 ldd $09,y leay d,y sty <u0065 bra L0FFB L0FEA pshs u,x clra ldx #$0000 Seek to beginning tfr x,u os9 I$Seek puls u,x lbcs L0191 L0FFB lbsr L091F leay >u0BFC,u bsr L100B lda #C$CR sta ,x inc <u0045 rts * Copy label from X to buffer @ Y, terminate at 1st illegal char with CR * Exit: X=ptr to start of label name from user's buffer * Y=ptr to start of buffer entry copy of label name L100B pshs y,x Preserve buffer & source ptrs ldb #79 (78 bytes to check) L100F decb Dec # chars left to check beq L1022 If done max, skip ahead lda ,x+ Get char for label sta ,y+ Save in buffer cmpa #'A Is it a letter or higher? bhs L100F Yes, continue copying cmpa #'0 Is it lower than a #? blo L1022 Yes, not allowed, force end of label name cmpa #'9 Is it a #? bls L100F Yes, that is fine L1022 lda #C$CR All others illegal, force CR in buffer copy sta -1,y Save it clrb No error puls pc,y,x Restore regs & return * M= command (???) L1029 ldb #C$CR stb -$01,x tst <u006B bne L1057 tst <u006C bne L1057 lda #Data Data module type pshs u,y,x os9 F$Link bcs L1055 stu <u0067 Save start address of module sty <u0065 Save execution address of module ldd 2,u addd <u0067 subd #$0003 std <u0069 inc <u006B puls u,y,x leax -$01,x lbra L0907 L1055 puls u,y,x L1057 lbra L0BCF * VAR. command L105A leay >u05A8,u lda ,x+ cmpa #'? beq L10C9 cmpa #'= beq L1096 cmpa #C$SPAC beq L1085 cmpa #'; beq L1085 cmpa #'9 bhi L1085 cmpa #'0 bcs L1085 suba #$30 ldb #$51 Multiply by 81 (size of each VAR entry) mul leay d,y lda ,x+ cmpa #'= beq L1096 L1085 leax -$01,x pshs x tfr y,x ldy #$0051 lda #$02 os9 I$ReadLn puls pc,x L1096 ldb #80 lbsr L0A25 lda #C$CR sta ,y rts L10A0 fcb C$LF fcc 'User Variables :' fcb C$CR L10B2 fcb C$LF fcc 'Shell Sub Variables :' fcb C$CR L10C9 pshs x clrb leax >L10A0,pc bsr L10DF leay >u08D2,u clrb leax >L10B2,pc bsr L10DF puls pc,x L10DF pshs y,b lbsr L021B puls y,b L10E6 pshs y,b lda #$51 mul leay d,y leax >u0124,u ldd #'V*256+'A std ,x++ ldd #'R*256+'. std ,x++ lda ,s L10FD adda #$30 ldb #'= std ,x++ L1103 lda ,y+ sta ,x+ cmpa #C$CR bne L1103 leax >u0124,u ldy #$0057 lda #$01 os9 I$WritLn puls y,b bcs L1122 incb cmpb #C$LF bcs L10E6 rts L1122 puls y puls pc,x * INC. command (increment shell variable by 1) L1126 bsr L1144 lbcs L0191 addd #$0001 bra L113A * DEC. command (decrement shell variable by 1) L1131 bsr L1144 lbcs L0191 subd #$0001 L113A bsr L11A7 lda #C$CR sta $05,y ldx <u0048 clrb rts L1144 inc <u0014 leay >u05A8,u lda ,x+ stx <u0048 cmpa #'0 bcs L1161 cmpa #'9 bhi L1161 suba #$30 ldb #81 mul leay d,y tfr y,x bra L1166 L1161 leas 2,s lbra L0BCF L1166 pshs y leas -$05,s tfr s,y clr $03,y clr $04,y L1170 clr $02,y lda ,x+ suba #$30 cmpa #$09 bhi L1195 pshs a lda #10 ldb $03,y mul std ,y lda $04,y ldb #10 mul addd $01,y std $01,y clra puls b addd $01,y std $03,y bra L1170 L1195 ldd 3,y leas 5,s puls pc,y * 2 byte ASCII conversion table L119B fdb 10000 fdb 1000 fdb 100 fdb 10 fdb 1 fdb 0 L11A7 pshs y,x,d pshs b leax >L119B,pc L11AF pshs d ldb #'/ stb 2,s puls d L11B7 inc ,s subd ,x bcc L11B7 addd ,x++ pshs d ldb $02,s stb ,y+ lda $01,x puls d bne L11AF puls b puls pc,y,x,d * PAUSE command - may display text message, and then waits for key press or * mouse button L11CF ldy #394 Write up to 394 chars of pause string lda #$02 To standard error os9 I$WritLn lbcs L0191 tfr y,d Tfr # chars written to D leax d,x Point X to next char after ones written leax -1,x Point to last char written pshs x Save ptr ldd #$02*256+SS.SSig Std Err/Send signal when key pressed ldx #$000A Signal $A is the one to send os9 I$SetStt lbcs L0191 Error, use main shell error handler IFGT Level-1 ldb #SS.MsSig Send signal on mouse button press os9 I$SetStt lbcs L0191 ENDC ldx #$0000 Go to sleep until one of the 2 is received os9 F$Sleep ldb #SS.Relea Signal gotten, release all signals os9 I$SetStt clrb No error & return puls pc,x * Parse PATH=, add paths to PATH buffer list L1209 pshs x Preserve ptr to string after 'PATH=' lda ,x Get 1st char cmpa #'? User requesting current paths? beq L1245 Yes, go do that pshs u Preserve U leau >u0CDD,u Point to PATH= buffer L1217 lda ,x+ Get char from user-requested path cmpa #C$SPAC Space? beq L1217 Yes, eat spaces until 1st real char found sta ,u+ No, save char L121F leay >L0408,pc Point to command modifier list L1223 cmpa ,y+ Match char? bhi L1223 No, our char is higher, check next modifier beq L1237 Found match, skip ahead lda ,x+ No modifier found, get next char sta ,u+ Save in PATH buffer cmpa #C$SPAC Was it a space? bne L121F No, check this char vs. modifier list lda #C$CR Yes, change to CR sta -1,u Save CR instead (terminate 1 path entry) bra L1217 Do whole list * NOTE: ANY modifier (not just CR, but ! # & ; < > ^ |) stops PATH=parsing L1237 leax -1,x Bump ptr back to last char from user stx 2,s Save ptr on stack over original X lda #C$CR Get CR sta -1,u Save CR as current path end sta ,u And 1 extra for parse routines puls u Get U back puls pc,x Restore new X & return L1245 leax >u0CDD,u Point to start of PATH=buffer L1249 ldy #400 Write up to 400 chars to standard out lda #$01 os9 I$WritLn Print text of one path lbcs L0191 Error, go process shell error tfr y,d Tfr # bytes written to D leax d,x Offset X to end of what was printed lda ,x Get char from there cmpa #C$CR CR (end of path list)? bne L1249 No, go write next path out puls x Restore ptr to next set of PATH= leax 1,x Bump ptr up by 1 & return rts * ^ (set priority on the fly) command L1265 ldb #C$CR Plop a CR onto the end stb -$01,x ldb <u0022 Any priority already set? lbne L0BCF Yes, print 'WHAT?' lbsr L16EB Go calculate binary priority into B stb <u0022 Save priority to fork module with lbra L0907 Continue processing for modifiers * # (set data memory size) command L1277 ldb #C$CR stb -1,x ldb <u0003 Already have a data mem size set? lbne L0BCF Yes, print 'WHAT?' lbsr L16EB eora #'K anda #$DF Force uppercase bne L1294 No 'K', just save # of 256 byte pages leax 1,x lda #4 Multiply # of K by 4 to get # pages mul tsta lbne L0BCF Ended up too big, print 'WHAT?' L1294 stb <u0003 Save data mem size to use lbra L0907 Continue processing command line * Carriage return processing L1299 leax -1,x lbsr L145D bra L12A3 * ; (separator) command (also called by others) L12A0 lbsr L1459 L12A3 bcs L12BA lbsr L0B96 Go do the path stuff tst <u005D Is there a module that is unlinking? bne L12AE Yes bsr L12D2 Go wait for child process to die (A=process #) L12AE bcs L12BA If child exited with status/signal code,skip lbsr L0907 Go parse for modifiers cmpa #C$CR Was the next non-space/comma char a CR? bne L12B9 No, skip ahead leas 4,s Yes, eat stack L12B9 clrb No error * Child process had a signal/status code L12BA pshs cc Preserve error status clr <u005D ??? puls cc Restore carry lbra L0B96 ??? Go close some paths & return? * & (background operation) command L12C3 lbsr L1459 bcs L12BA bsr L12BA ldb #$26 lbsr L16B3 bra L12AE * W command - Wait for a child to die L12D1 clra Clear process ID # * Entered here if commands are separated with ';' (or '()' groups) L12D2 pshs a Save ID # of process? L12D4 os9 F$Wait Wait for child to die or until signal received tst <u000E Signal received (which would be in SHELL)? beq L12EC No, child was exited (or got signal), go process * Shell was interrupted by signal while Waiting ldb <u000E Get signal that we received cmpb #S$Abort Was it a PD.QUT (<CTRL>-<E>) (quit 'W'aiting?) bne L1304 No, skip ahead lda ,s Get process # beq L1304 None, exit os9 F$Send Send the signal to the child as well clr ,s Clear process # bra L12D4 Go Wait again (until child dies) * Child F$Exited or was aborted - eat should go here * Entry: A=ID # of deceased child * B=Exit (error) code from child L12EC lbcs L1308 If F$Wait exited with error, return with it cmpa ,s Same process # as one we were waiting for? beq L1304 Yes, exit tst ,s No, was there a specific process # we wanted? beq L12F9 No, skip ahead * Child died, but not the one we were waiting for tstb Was there an error status? beq L12D4 No, ignore dead child and wait for one we wanted * Child died with error on exit L12F9 pshs b Preserve child's exit status code bsr L12BA ??? Go close & re-dupe paths? ldb #'- Get a '-' (for a '-003' process finished msg) lbsr L16B3 Print that out puls b Get back exit status code L1304 tstb is there an Error/signal code? beq L1308 No, exit cmpb #S$Intrpt Yes, is it a keyboard interrupt signal? beq eatchar Yes, eat the key cmpb #S$Abort Keyboard abort signal? bne errexit No, exit with unknown error/signal code * At this point, child died from signal 2 or 3 (CTRL-C or CTRL-E). The corres- * ponding key is also sitting in this devices PD.BUF as the 1st char. We musts * 1) Disable keyboard signal & eat the key from the buffer * 2) Exit from here with Carry set & signal (2 or 3) in B eatchar pshs b,x,y Preserve signal code & regs used ldd #SS.Ready Std in, check for data ready on device os9 I$GetStt Check it bcs NotSCF No chars waiting on device, exit with signal lda <u0018 Is the shell immortal? beq NotSCF No, don't try to eat the char eat clra Standard in path leas -PD.OPT,s Make 32 byte buffer for OPT's leax ,s Point X to it clrb SS.Opt call os9 I$GetStt Get current path options lda ,x Get device type beq Eatkey SCF (not script file) so go eat key NoChar leas PD.OPT,s Eat temp buffer bra NotSCF Exit with signal code (script file got signal) * Have to eat key: Shut echo off 1st Eatkey clr 4,x PD.EKO flag off os9 I$SetStt Shut echo off ldd #SS.Relea Std In, Release keyboard/mouse signals os9 I$SetStt Shut signals off so we don't get stuck leax ,-s Make 1 byte buffer on stack ldy #1 1 byte to read os9 I$Read Eat the char leas 1,s eat buffer ldd #SS.SSig Std In, send signal on key ready ldx #$B Signal to send os9 I$SetStt Turn keyboard signal on again leax ,s Point to temp buffer again inc 4,x PD.EKO flag on clra Std In clrb Set Options os9 I$SetStt Turn echo back on leas PD.OPT,s Deallocate temp buffer ldb u180D,u Get current history line # cmpb #1 First one? bhi Previous No, B=previous one ldb u180C,u Was on first, so get last incb Adjust for dec Previous decb Point to previous one lbsr L19D3 Go get ptr to history lda ,y Get 1st char from previous line in history sta ,x Save char there ldd #SS.Fill Fill keyboard buffer call to Std In ldy #$8001 1 char long, don't append CR os9 I$SetStt Stick that key into the keyboard buffer NotSCF puls b,x,y Restore regs (and exit status byte in B) errexit coma Yes, set carry & exit L1308 puls pc,a * Level 2: If data area <4.25K, force up to 7.5K * Exit: A=Type/language * X=Current source line parsing ptr (module name to chain) * Y=Size of parameter area * U=Ptr to parameter area * B=Size of data area L130A lda #Prgrm+Objct Module type/language ldb <u0003 Get # pages of data mem needed for forked module IFGT Level-1 cmpb #$11 Is it at least 17? bhs L1316 Yes, skip ahead ldb #$1F Otherwise, force to 7.5K minimum stb <u0003 Save it ENDC L1316 andcc #^Carry Clear carry ldx <u0004 Get mem module ptr ldy <u0006 Get size of current command group ldu <u0008 Get ptr to start of current command group rts * Copy string from X to Y until CR is hit L1320 lda ,x+ Get char sta ,y+ Save it cmpa #C$CR Carriage return? bne L1320 No, keep copying rts Done, return * Attempt load in module to execute (it's not in memory) * Entry: X=Ptr to module name L1329 lda #EXEC. 1st, attempt to get it from current Exec DIR os9 I$Open Attempt to open it bcc L1362 Found it, continue * Possible search thru PATH= settings inc <u000F ??? Set flag to indicate using PATH= leax >u0CDD,u Point to start of PATH= list L1336 clrb lda ,x Get 1st char from next PATH= line cmpa #C$CR End of list? lbeq L1564 Yes, ??? leay >u0124,u No, point to temp buffer bsr L1320 Copy path to temp buffer until CR pshs x Preserve ptr to next possible path lda #'/ Add slash since we want file from this path sta -1,y Save at end of path in temp buffer ldx <u0004 Get ptr to module/script name we are looking for bsr L1320 Copy it into temp buffer up to CR leax >u0124,u Point to start of full path list lda #READ. Attempt to open file os9 I$Open puls x Restore ptr to next possible path bcs L1336 Didn't find file there, try next path leax >u0124,u Point to full pathlist again stx <u0004 Replace ptr to module with full pathlist ptr L1362 leax >u00D6,u Point to buffer to hold beginning of file ldy #77 77 bytes to read os9 I$Read Read it in bcc L1373 No error, skip ahead cmpb #E$EOF Just EOF error? bne L13CE No, something more serious, skip ahead L1373 tst <u000F bne L137B ldb #$04 stb <u000F L137B pshs a Save path # a sec ldd M$ID,x Get possible module header bytes cmpd #M$ID12 Legit module header? puls a Restore path # beq L1396 OS9 module, skip ahead * Not module...possible shell script? os9 I$Close Not OS9 module, close file clrb dec <u000F Dec flag lbeq L1564 If 0, skip ahead inc <u000F If not, inc & skip ahead lbra L1564 * Seems to be OS9 module L1396 clr <u000F Clear flag ldy M$Name,x Get offset to module name leax >u00E3,u Point X to offset $E in module cmpy #$000D Does the name start at offset $D? beq L13C0 Yes, skip ahead pshs u Preserve U tfr y,u Move name offset to U ldx #$0000 MSW of file pos=0 os9 I$Seek Go seek to that spot in file puls u Restore U bcs L13CE Error seeking, go handle ldy #64 Go read up to 64 char filename leax >u00E3,u Point to spot to hold filename os9 I$Read Read it in bcs L13CE Error reading, go handle L13C0 pshs a Save path # os9 F$PrsNam Parse module name puls a Restore path # bcs L13CE cmpb #$40 bhi L1422 clrb L13CE pshs b,cc Preserve error status os9 I$Close Close file puls b,cc Restore error status lbcs L162B If error, exit with it (S/B L162C) leax >u00D6,u Point to buffer holding 77 bytes of file lda $06,x ldy $0B,x cmpa #$40 bne L1407 bsr L13EF lbcs L162B lbra L14AF L13EF leax >u00E3,u IFGT Level-1 os9 F$NMLink ELSE pshs u os9 F$Link puls u ENDC bcc L1400 ldx <u0004 IFGT Level-1 os9 F$NMLoad ELSE pshs u os9 F$Load puls u ENDC bcc L1400 rts L1400 leax >u00E3,u stx <u0004 rts L1407 cmpa #$51 bne L1413 bsr L13EF lbcs L162B bra L1427 L1413 cmpa #$11 lbne L14D7 leax >u00E3,u stx <u0010 lbra L15D7 L1422 lbsr L08CB bra L13CE * Call a shellsub module L1427 clra Type/language byte to wildcard:any will match ldx <u0004 Get ptr to module name pshs u Preserve U os9 F$Link Attempt to link it in puls u Restore U lbcs L162B If we couldn't link, Exit with error ldx <u0004 Get ptr to module name again IFGT Level-1 os9 F$UnLoad Unlink it ELSE pshs a,b,x,y,u os9 F$Link os9 F$Unlink os9 F$Unlink puls a,b,x,y,u ENDC lbcs L162B If we couldn't unlink exit with error ldx <u0008 Get ptr to current group (param ptr for shellsub) ldd <u0006 Get size of param area for shellsub leau >u08D2,u Point to shellsub variable area jsr ,y Execute shellsub module pshs b,a,cc Preserve error status & A clra clrb std <u0010 ? (originally pointing to E3 if type $11 module) ldx <u0004 Get shellsub module ptr IFGT Level-1 os9 F$UnLoad Unlink it ELSE pshs a,b,x,y,u os9 F$Link os9 F$Unlink os9 F$Unlink puls a,b,x,y,u ENDC std <u0004 Clear shellsub module ptr inc <u005D Set flag that we should wait for module to exit? puls pc,u,y,x,d,cc restore regs & return L1459 lda #C$CR sta -1,x L145D clr <u0060 pshs u,y,x ldx <u0004 Get ptr to name ldd ,x Get 2 chars of name andb #$5F Force 2nd one to uppercase cmpd #$2F57 Is it a /W? bne L1473 No, check for shellsub comb Yes, exit with bad mode error ldb #E$BMode lbra L162B L1473 clra Wildcard NMLink IFGT Level-1 os9 F$NMLink Link to module ELSE pshs u os9 F$Link Link to module puls u ENDC lbcs L1329 Error, do something cmpa #ShellSub+Objct ShellSub module? beq L1427 Yes, go set up for it ldx <u0004 Get ptr to name back IFGT Level-1 os9 F$UnLoad Drop the link count back down ELSE pshs a,b,x,y,u os9 F$Link os9 F$Unlink os9 F$Unlink puls a,b,x,y,u ENDC pshs y Save data area size ldx <u0004 Get ptr to module name again leay >L000D,pc Point to 'Shell' ldb #$05 Size of 'Shell' os9 F$CmpNam Is the module requested Shell? puls y Restore data area size bcs L14A3 Not shell, skip ahead ldb 5,x Get char right after 'shell' cmpb #C$CR Is 'shell' the only thing on the line? lbeq L158D Yes, skip ahead cmpb #C$SPAC Is it a space? lbeq L158D Yes, skip ahead * Not Shell or Shellsub module L14A3 cmpa #Prgrm+Objct ML program? lbeq L15D7 Yes, skip ahead cmpa #Data Is it a data module? bne L14D7 No, skip ahead inc <u0060 Set flag - data module L14AF inc <u0060 Bump it up ldx <u0004 Get ptr to module name os9 F$PrsNam Parse the name ldy <u0063 Get ptr to Intercept routines data mem ($418) leay $0A,y Bump up to $422 sty <u0008 Save ptr to start of current group sty <u0061 ??? Ptr to data modules name? ldx #60 Max size of group stx <u0006 Save it ldx <u0004 Get ptr to module name L14C8 lda ,x+ Copy it to buffer @ $422 sta ,y+ decb bne L14C8 lda #C$CR Append a CR to it sta ,y+ clrb lbra L1564 * Not 6809 object code or data module either L14D7 sty <u000A Save data area size leax >L0013,pc Point to alternate languages table L14DE tst ,x Is this entry active? lbeq L1629 No, exit with non-existing module error cmpa ,x+ Same module type as we want? beq L14EE Yes, skip ahead L14E8 tst ,x+ No, eat module name bpl L14E8 bra L14DE Try next module * Found run-time language match L14EE ldd <u0008 Get ptr to start of current command group subd <u0004 Calculate size of whole group addd <u0006 Don't include size of current group std <u0006 Save remainder size ldd <u0004 Get ptr to start of sub-module std <u0008 Save it pshs y,x Preserve data area size & primary module ptr leax >L0021,pc Point to 'RUNB' cmpx ,s Is that the run-time module we want? bne L1546 No, skip ahead * RUNB needed - have to () & quote/commas between params ldx <u0008 Yes, get sub-module ptr? leay >u0418,u Point to before sub-module bsr L154B Copy buffer up to next param (or end of line) beq L1535 If it was end of line, add CR & continue ldd ,x Get 2 param chars cmpd #$2822 Is it '("' (RUNB variables ahead?) beq L1546 Yes, skip ahead (we won't have to add them) lda #C$SPAC No, add ' ("' sta ,y+ ldd #$2822 std ,y++ L151F bsr L154B Copy buffer up to next param (or end of line) beq L152E If end of line, add '")' (close params) ldd #$222C Add '","' (Basic09 param separators std ,y++ lda #$22 2nd quote of above sta ,y+ bra L151F Keep doing for all parameters L152E ldd #$2229 Add '")' to end parameter list std ,y++ lda #C$CR Add CR L1535 sta ,y+ tfr y,d Move end of param ptr to D leay >u0418,u Point to start of param sty <u0008 Save as start ptr subd <u0008 Calculate param size std <u0006 Save it L1546 puls y,x Restore data area size & primary module ptr lbra L15D5 * Copy from X to Y until either a CR or a space char is hit * If it finds a space, it will eat them until the next non-space char is found L154B lda ,x+ Get char cmpa #C$SPAC Is it a space? * Was L155B beq L1559 yes, skip ahead cmpa #C$CR Is it the end of the line? beq L155F Yes, bump ptr back to CR & exit sta ,y+ Save the char bra L154B Keep doing it L1559 lda ,x+ Get char L155B cmpa #C$SPAC Is it another space? beq L1559 Yes, keep eating spaces L155F leax -$01,x Bump ptr back to either non-space or CR cmpa #C$CR Is it a CR? & return rts * THIS CMPB / LBEQ SEEMS TO BE USELESS, AS B IS ALWAYS CLEAR COMING INTO THIS * ROUTINE L1564 cmpb #E$BMode lbeq L162B ldx <u0006 Get size of current group leax 5,x Bump it up by 5??? stx <u0006 Save new size tst <u0060 Data module linked? bne L1592 Yes, skip ahead ldx <u0004 Get module name ptr ldu 4,s lbsr L0BFA Set up paths lbcs L162B If error, exit with it bra L1592 Start up shell with '-P X PATH=(current)' * L1581 is for sub-shells (?), L1586 for normal shells L1581 fcc '-P X ' Prompting off/exit on error L1586 fcc 'PATH= ' For inheriting parent shell's paths fcb C$CR L158D leax <L1586,pc Point to 'path=' bra L1595 Skip ahead L1592 leax <L1581,pc Point to '-p x ' L1595 leay >u166D,u Point to about-to-be merged buffer lbsr L1320 Copy up until CR leay -1,y Point to CR leax >u0CDD,u Point to copy of current path= * Copy all paths to buffer, changing <CR> separated ones with Spaces L15A2 lda ,x Get char cmpa #C$CR CR? beq L15B1 Yes, don't copy this buffer lbsr L1320 Copy up until CR lda #C$SPAC Replace CR with Space sta -1,y bra L15A2 Continue copying CR marked blocks until done L15B1 lda #'; Replace final CR with ; (command separator) sta -1,y tst <u0060 beq L15BE ldd #'M*256+'= If flag set, append 'M=' std ,y++ L15BE ldx <u0008 Get ptr to start of current group lbsr L1320 Copy up until CR leax >u166D,u Point to merged buffer again stx <u0008 Make it the new current group start tfr y,d Move end buffer ptr to D pshs x Push merged buffer ptr for SUBD subd ,s++ Calculate size of merged buffer std <u0006 Save merged buffer size leax >L000D,pc Point to 'shell' L15D5 stx <u0004 Save ptr to module name to fork L15D7 ldx <u0004 Get ptr to module name to fork lda #Prgrm+Objct IFGT Level-1 os9 F$NMLink Get memory requirement stuff from it ELSE pshs u os9 F$Link Get memory requirement stuff from it tfr u,y puls u ENDC bcc L15E5 Got it, continue IFGT Level-1 os9 F$NMLoad Couldn't get, try loading it ELSE pshs u os9 F$Load Couldn't get, try loading it tfr u,y puls u ENDC bcs L162B Still couldn't get, can't fork L15E5 IFEQ Level-1 ldy M$Mem,y ENDC tst <u0003 Memory size specified? bne L15F2 Yes, skip ahead tfr y,d No, tfr modules mem size to D addd <u000A ??? Add to something addd #$00FF Round up to nearest page sta <u0003 Save # of pages need for data mem L15F2 clr ,-s Clear byte on stack to store original priority ldb <u0022 Get priority we want to set new program at beq DnePrior 0=Use inherited priority, skip ahead IFEQ Level-1 ldx <D.Proc ELSE leax >u166D,u Point to place to hold Process descriptor os9 F$ID Get our process # os9 F$GPrDsc Get our process descriptor ENDC ldb P$Prior,x Get our priority stb ,s Save it ldb <u0022 Get priority for new process os9 F$SPrior Set our priority so child will inherit it DnePrior lbsr L130A Go setup Fork entry registers os9 F$Fork Create the new process pshs d,cc Preserve error (if any) & new process # ldb 3,s Get original priority back beq L1609 Priority didn't change, ignore it os9 F$ID Get our process # into A os9 F$SPrior Reset our priority back to normal L1609 lda #Prgrm+Objct Std 6809 module ldx <u0010 Get ptr to some other module name (?) bne L1611 There is one, unlink it instead ldx <u0004 Get ptr to command name L1611 IFGT Level-1 os9 F$UnLoad Bump link count down back to normal? ELSE pshs a,b,x,y,u os9 F$Link os9 F$Unlink os9 F$Unlink puls a,b,x,y,u ENDC clra clrb std <u0010 Zero out other module name ptr std <u0004 Clear out ptr to main command name lda <u0060 Check if data module needs to be unlinked too cmpa #$01 Just 1 link to it? bne L1627 No, skip ahead lda #Data Data module ldx <u0061 Get ptr to name of data module IFGT Level-1 os9 F$UnLoad Bump link count down back to normal ELSE pshs a,b,x,y,u os9 F$Link os9 F$Unlink os9 F$Unlink puls a,b,x,y,u ENDC L1627 puls cc,d Get back F$FORK error/process # leas 1,s Eat priority byte puls pc,u,y,x Restore regs & return L1629 ldb #E$NEMod Non-existing module error L162B coma puls pc,u,y,x L162E fcc '/pipe' fcb C$CR L1634 pshs x leax <L162E,pc Point to '/pipe' ldd #$0103 lbsr L0C0E puls x bcs L169E lbsr L1459 bcs L169E lda ,u bne L1653 os9 I$Dup bcs L169E sta ,u L1653 clra os9 I$Close lda #$01 os9 I$Dup lda #$01 lbsr L0BA8 lda #$02 lbra L0BA8 * Filename for shell log-append mode because of leading '+' L1666 fcc '+/dd/log/uxxx' fcb $0d * Make shell logging filename @ u0CBD,u L1674 leax <L1666,pc Point to log name string (append mode) leay >u0CBD,u Point to buffer to hold shell log name lbsr L1320 Copy name to buffer leay -4,y Point to where 1st digit will go lda <u005A+1 Get LSB of user #? pshs y,x,d Preserve regs leay <L1693,pc Point to routine bra L16B9 Go convert digits & append to logname L168A lda <u0047 pshs y,x,d leay <L1693,pc bra L16B9 L1693 ldy $0B,s Get ptr to where shell log # goes ldd $03,s Get 1st 2 digits of # std ,y++ Save in shell log pathname lda $05,s Get last digit sta ,y Save it too L169E rts L169F ldd 4,s Get last 2 digits of process # (ASCII) std >u01EC,u Save it & return rts L16A6 pshs y,x,d Preserve End of parm ctr & others os9 F$ID Get user's ID # sty <u005A Save it leay <L169F,pc Point to routine bra L16B9 * Set up to write out process # when forked? L16B3 pshs y,x,d leay >L021B,pc * Entry: A=Process ID # * Exit: L01EC=2 lower digits of process # in ASCII format L16B9 pshs y,x,b Preserve regs leax 1,s Point X to X value on stack ldb #$2F Init B to start @ '0' in loop L16BF incb Bump ASCII # up suba #100 Start with 100's digit bhs L16BF More left, keep going stb ,x+ Save digit ldb #$3A Init so loop starts @ ASCII '9' L16C8 decb Bump ASCII # down adda #10 Do 10's digit bhs L16C8 Still more, keep going stb ,x+ Save 10's digit adda #$30 Bump 1's digit up to ASCII equivalent ldb #C$CR Add carriage return std ,x Save overtop Y on the stack leax ,s Point X to B on stack jsr ,y leas 5,s Eat stack puls pc,y,x,d Restore other regs & return * KILL command L16DD bsr L16EB Go get process # to kill cmpb #2 Trying to kill the system process or 1st shell? bls L170A Yes, print 'WHAT?' & ignore it tfr b,a Move process # to proper reg L16E5 clrb S$Kill signal os9 F$Send Send it to the process & return rts * Set priority - subroutine to calculate binary version of # * (used for both process # & priority values) L16EB clrb Initialize # for loop L16EC lda ,x+ This loop will calculate the binary version suba #$30 Of the ASCII # pointed to by X cmpa #9 bhi L16FD pshs a lda #10 mul addb ,s+ bcc L16EC Keep going until overflows past 255 L16FD lda ,-x Get last char done bcs L1708 If #>255, eat RTS & exit with error tst <u0014 If flag is set, return bne L169E tstb Otherwise, check if # is 0 bne L169E No, return L1708 leas 2,s Yes, eat RTS address & exit with error L170A lbra L0BCF Print 'WHAT?' * SETPR routine L170D bsr L16EB Go calculate process # stb <u0021 Save it lbsr L0907 Find next field (after commas/spaces) bsr L16EB Go calculate priority (into B) lda <u0021 Get process # os9 F$SPrior Set it's priority & return rts L171C fcc 'pwd: bad name in path' fcb C$CR L1732 fcc '.......................................' L1759 fcc '.' fcb C$CR L175B fcc 'pwd: read error' fcb C$CR L176B clr <u003D L176D pshs y,x leay >u02F2,u lda #$81 tst <u0037 beq L178F ldx <u0039 bra L17F7 * .PXD command L177D clr <u003D pshs y,x leay >u0375,u lda #$85 tst <u0038 beq L178F ldx <u003B bra L17F7 L178F sta <u0029 sty <u003E L1794 leax >$0080,y lda #$0D sta ,x stx <u002A leax <L1759,pc stx <u0040 bsr L1801 lbsr L183C L17A9 ldd <u002C std <u0032 lda <u002E sta <u0034 bsr L1828 beq L17D3 lda <u0012 os9 I$Close lbcs L188C ldx <u0040 leax -$01,x stx <u0040 bsr L1801 bsr L183C bsr L1817 leax >u03F8,u lbsr L1859 bra L17A9 L17D3 lda <u0012 ldb #SS.DevNm leax >u00B5,u os9 I$GetStt Get device name bsr L1859 L17E0 lda <u0012 os9 I$Close ldx <u002A lda <u0029 bita #$04 bne L17F3 inc <u0037 stx <u0039 bra L17F7 L17F3 inc <u0038 stx <u003B L17F7 ldy #$0083 lda #$01 clrb lbra L18A0 L1801 lda <u0029 os9 I$Open sta <u0012 rts L1809 lda <u0012 leax >u03F8,u ldy #$0020 os9 I$Read rts L1817 bsr L1809 bcs L1896 leax >u0415,u leay >u0032,u bsr L1830 bne L1817 rts L1828 leax >u002C,u leay >u002F,u L1830 ldd ,x++ cmpd ,y++ bne L183B lda ,x cmpa ,y L183B rts L183C bsr L1809 ldd >u0415,u std <u002F lda >u0417,u sta <u0031 bsr L1809 ldd >u0415,u std <u002C lda >u0417,u sta <u002E rts L1859 os9 F$PrsNam bcs L1890 ldx <u002A pshs b incb clra std <u0032 tfr x,d subd <u0032 cmpd <u003E bls L1881 puls b L1871 lda ,-y anda #$7F sta ,-x decb bne L1871 lda #$2F sta ,-x stx <u002A rts L1881 lda #'* sta ,-x stx <u002A leas 3,s lbra L17E0 L188C pshs b,cc bra L18AB L1890 leax >L171C,pc bra L189A L1896 leax >L175B,pc L189A leas $02,s ldd #$02FF L18A0 stx <u002A pshs b,cc tst <u003D bne L18AB os9 I$WritLn L18AB puls b,cc puls y,x,pc L18DB leax >u0124,u Point to PD.OPT work copy area clrb Change to CLRB os9 I$GetStt Get current PD.OPT settings tst ,x Check device type bne L1914 If not SCF, don't bother changing stuff inc >u1812,u Set flag that we are changing key defs ldd <$10,x Get PD.INT & PD.QUT chars (<CTRL>-<C> & <E>) sta >u180F,u Save copy of PD.INT stb >u180E,u Save copy of PD.QUT ldd #$0A0C Reload with up & down arrows std <$10,x Save 'em lda $06,x Get PD.NUL count (normally 0) sta >u1811,u Save copy lda #$05 Replace with 5 sta $06,x clra clrb Reset path options to new settings os9 I$SetStt Do SS.OPT Setstat * Non-SCF devices go here (ex. a script file would be RBF) L1914 ldb #SS.SSig Send signal on data ready ldx #$000B Signal code to send is $B rts Do SetStt, clear <E signal copy, and go to L191C * X still is $B from L1914 * Called when F$Sleep @ L0171 is interrupted by keyboard/mouse signal L191C ldb <u000E Get Signal code clr <u000E Clear memory copy of signal code cmpb #S$Abort Keyboard abort signal (<CTRL>-<E>)? bne L1928 No, check next bsr L1932 Write CR out if no history yet bra L1967 Go backwards in history L1928 cmpb #S$Intrpt Keyboard interrupt signal (<CTRL>-<C>)? lbne L017B No check for $B signal (ignore rest) bsr L1932 Write CR out if no history yet bra L1991 Go forwards in history * Keyboard abort or Keyboard interrupt signal L1932 tst >u180D,u Any history entries yet? bne L193D Yes, exit bsr L1959 Otherwise, put CR into the buffer os9 I$WritLn Write it out to standard out L193D rts L193E os9 I$ReadLn Read in line bcc L194E No error, skip ahead * NOTE: WHEN THE ABORT A SUB-FORKED PROGRAM BUG OCCURS, THE ABOVE READLN * EXITS WITH CARRY SET & B=0 (ERROR 0) cmpb #$02 Go backward in history buffer signal? beq L1967 Yes, go process cmpb #$03 Go forward in history buffer signal? beq L1991 Yes, go process lbra L1AAA Go to parse routine with unknown error in B L194E cmpy #$0001 Just 1 char. read? lbeq L1AA3 Yes, go do normal parsing (no error) lbra L1A37 Otherwise, change chars <$0D to spaces * Change A to be standard output??? L1959 lda #C$CR Carriage return leax >u0124,u ??? Point to a buffer sta ,x Save CR there clra Prepare to write the CR out to standard input ldy #$0001 rts * PD.QUT (redefined to be go back 1 in history buffer) L1967 tst >u180C,u Any lines in history buffer? bne L1972 Yes, go move between them bsr L1959 Otherwise, put CR into buffer lbra L1AA3 Go to normal command parse routine (no error) L1972 ldb >u180D,u Get 'current' history buffer entry # cmpb #$01 On the first one? bls L1987 Yes, go wrap to end cmpb >u180C,u Somehow point past last one? bhi L1987 Yes, bump back to last one decb Bump to previous one stb >u180D,u Save it as 'current' history entry # bra L19B3 L1987 ldb >u180C,u Get highest history entry # stb >u180D,u Make it the current bra L19B3 * PD.INT (redefined to be go forward 1 in history buffer) L1991 tst >u180C,u Any lines in history buffer? bne L199C Yes, go move between them bsr L1959 Otherwise, put CR into buffer lbra L1AA3 Go to normal command parse routine (no error) L199C ldb >u180D,u Get current history entry # cmpb >u180C,u Higher or same as last one? bhs L19AD Yes, wrap to beginning incb Bump to next one stb >u180D,u Save it as 'current' history entry # bra L19B3 L19AD ldb #$01 Set history entry # to 1st one stb >u180D,u Save as 'current' entry # L19B3 bsr L19D3 Go get ptr to history entry buffer we want sty >u1813,u Save 'current' history buffer ptr leax >u0213,u Point to expanded shell prompt ldy >u01F7,u Get size of expanded shell prompt lda #$02 Std Err os9 I$WritLn Write out shell prompt bsr L19EA Go Pre-load Read buffer with history entry bcc L1A0E No error, continue lbra L1AAA Normal parse with error * Find history entry # we want * Entry: B=history entry # (1-xx) * Exit: Y=Ptr to history buffer entry we wanted L19D3 ldy >u180A,u Get ptr to start of history buffers L19D8 decb Bump down counter to find entry we want beq L19E9 Found it, skip ahead L19DB tst ,y+ Wrong one, search for end of one we are checking beq L19E4 Found it, skip ahead lbsr L1A97 Wrap buffer ptr if we hit end of all history bra L19DB Keep searching for end of current entry L19E4 lbsr L1A97 Wrap buffer ptr if we hit end of all history bra L19D8 Go into next entry in history buffers L19E9 rts Found it, leave L19EA leax >u0124,u Point to temp buffer ldy >u1813,u Get ptr to current history buffer clrb Set counter to 0 (size of buffer) L19F4 lda ,y+ Get char from history buffer beq L1A00 Found end, continue sta ,x+ Put it in temp buffer incb Bump up size counter lbsr L1A97 Wrap buffer ptr if we hit end of all history bra L19F4 Continue copying into temp buffer L1A00 clra D=Size of history buffer decb tfr d,y Move to proper register leax >u0124,u Point to buffer history copy ldb #SS.Fill Fill it with current selected history command os9 I$SetStt rts * Successfull SS.Fill of history buffer goes here L1A0E lda #$01 Write copy of history buffer to std out os9 I$Write leax >u0124,u Point to current history buffer again tfr y,d Transfer # bytes written to D lda #C$BSP Backspace char L1A1B sta ,x+ Fill buffer with backspaces decb bne L1A1B leax >u0124,u Write them to reset cursor to start of line lda #$01 os9 I$Write ldd #SS.Relea Eat keyboard signal (SS.Fill cleared it out of os9 I$SetStt the read buffer already) ldb #SS.SSig Setup for re-enabling it ldx #$000B Signal Code to send on keypress=$B lbra L016A Go re-enable signal send on keyboard/mouse input * Put new entry into history buffers, adjusting # used, etc.) * Entry: Y=# bytes read in ReadLn <>1 L1A37 pshs y Preserve # bytes read tfr y,d Move to D reg ldy >u1808,u Get ptr to where next history entry will go leax >u0124,u Point to line entered by user L1A44 lda ,x+ Get char cmpa #C$CR Control char <$0d? bhs L1A4C No, save the character lda #C$SPAC Replace with space char if it is L1A4C sta ,y+ Save char into history buffer bsr L1A97 Wrap to beginning if we hit end cmpy >u180A,u Point to entry #1? bne L1A59 No, continue bsr L1A7B Yes, make new #1 entry & drop # of entries L1A59 decb Drop # bytes left in new entry bne L1A44 Not done, continue clr ,y+ mark end with NUL bsr L1A97 Wrap to beginning if at end cmpy >u180A,u Pointing to entry #1? bne L1A69 No, skip ahead bsr L1A7B Yes, make new #1 entry & drop # of entries L1A69 inc >u180C,u Increase # of entries sty >u1808,u Save ptr to where next history entry will go puls y Restore # bytes read clra Point to temp buffer again leax >u0124,u bra L1AA3 Normal parse, no error * Reset ptr to new 1st entry in history buffer L1A7B pshs y Preserve current location in history buffer ldy >u180A,u Point to start of 1st entry L1A82 tst ,y+ End of current entry? beq L1A8A Yes, skip ahead bsr L1A97 No, wrap if we need to bra L1A82 Keep going until we find end L1A8A bsr L1A97 Wrap if we need to dec >u180C,u Dec # lines in history buffer sty >u180A,u Save new 'start of history' ptr puls pc,y Restore current location & return * If we are at end of history buffers, wrap to beginning (raw, has nothing to * do with entry #'s) L1A97 cmpy >u1806,u Are we at end of buffer for history entries? bne L1AA2 No, continue on leay >u1815,u Yes, reset to beginning of history buffers L1AA2 rts L1AA3 bsr L1AB1 Reset std paths to normal <CTRL>-<E>/<C> settings andcc #^Carry No error lbra L018B Normal command processing L1AAA bsr L1AB1 Reset std paths to normal <CTRL>-<E>/<C> settings orcc #Carry Error lbra L018B Normal command processing * Reset all 3 standard paths' NUL counts/Keyboard Interrupt/Terminate settings L1AB1 pshs x,d Preserve regs tst >u1812,u Check flag beq L1AD6 If 0 skip ahead leas <-PD.OPT,s Make 32 byte buffer on stack leax ,s Point X to buffer clrb CHANGE TO CLRB clra Standard input path bsr L1AD8 Restore NUL counts, Keyboard Intrpt/Terminate lda #$01 Do same for Standard output path bsr L1AD8 lda #$02 Do same for Standard Error path bsr L1AD8 leas <PD.OPT,s Eat stack buffer clr >u1812,u Reset PD.OPT flags to 0 clr >u180D,u L1AD6 puls pc,x,d Restore regs & return * Restore path options to preserved Keyboard terminate/interrupt & end of * line NUL counts L1AD8 pshs a Preserve path # os9 I$GetStt Get current PD.OPT settings lda >u180E,u Get copy of Keyboard terminate char sta <PD.QUT-PD.OPT,x Save into buffered copy lda >u180F,u Get copy of Keyboard interrupt char sta <PD.INT-PD.OPT,x Save into buffered copy lda >u1811,u Get copy of end of line NUL count sta $6,x Save into buffered copy puls a Get path # back os9 I$SetStt Reset path options with restored values & return rts emod eom equ * end