.title k11m41 kermit i/o for RSX11M/M+ v4.1 and 2.1 .ident /5.0.01/ ; Bob Denny (see below) ; define macros and things we want for KERMIT-11 .include /IN:K11MAC.MAC/ .iif ndf, k11inc, .error ; INCLUDE for IN:K11MAC.MAC failed .enabl gbl ; Copyright (C) 1983 Change Software, Inc. ; ; ; This software is furnished under a license and may ; be used and copied only in accordance with the ; terms of such license and with the inclusion of ; the above copyright notice. This software or any ; other copies thereof may not be provided or other- ; wise made available to any other person. No title ; to and ownership of the software is hereby trans- ; ferred. ; ; The information in this software is subject to ; change without notice and should not be construed ; as a commitment by the author. ; ; ; binread( %val channel_number, %val timeout ) ; binwrit( %loc buffer, %val buffer_length, %val channel_number ) ; cantyp ( %val channel_number ) ; doconn ( ) ; drpprv ( ) ; echo ( ) ; exit ( ) ; getprv ( ) ; getuic ( ) ; gttnam ( %loc buffer ) ; noecho ( ) ; read ( %loc buffer, %val buffer_length, %val channel_number, ; %val block_number ) ; setdlm ( %val channel_number ) ; suspen ( %val sleep_time ) ; ttyini ( %loc device_name ,%val channel_number , %val ccflag ) ; write ( %loc buffer, %val buffer_length, %val channel_number, ; %val block_number ) ; .sbttl edits ; 20-Jan-84 09:50:18 BDN Test and fix TTSPEED, SETSPD and BINREAD ; ; 03-Mar-84 Bob Denny 4.2.00 [RBD01] ; Rewrote namcvt(). Eliminated FCS parsing ; in favor of home-brew code which can handle ; the infinite variety of filespecs that may ; crop up when doing DECnet remote file access. ; ; 07-Mar-84 Bob Denny 5.0.00 [Edit trails removed] ; Fair rewrite, particularly of terminal handling. ; Changed within the existing KERMIT-11 architecture, ; which is better suited to RSTS/E (which seems to ; have a lot more terminal & communications options). ; Modes for RSX now allow operation at 9600 baud for ; packet communication. CONNECT is still a problem. ; ; 10-Mar-84 Bob Denny 5.0.01 The method used for CONNECT on RSTS/RSX ; will not work reliably on native RSX at baud rates ; over 1200 on a busy system. The "doconn()" routine ; was rewritten. Now there are 2 separate modules. ; Also, the binrea() function is now used only for ; packet reading, and has been greatly simplified. ; ; 16-mar-84 Brian Nelson ; ; Merged origional K11M41 with Bob Denny's mods. .sbttl macros .macro clrfqb ; Clear the "Furk-bee" (I love it!!) call $clrfq .endm clrfqb .macro clrxrb ; Clear the "Exer-bee" call $clrxr .endm clrxrb .macro moverr val,dst movb val ,-(sp) call $mover movb (sp)+ ,dst .endm moverr .iif ndf, r$rsts, r$rsts = 0 ef.tmp = 17 ef.tt = 20 ef.tmo = 21 er.tmo == 176 ; for now, timeout er.nod == 177 ; pseudo error for no data nodata == er.nod .sbttl data areas .psect $idata rw,d,lcl,rel,con fu$def::.word 0 ; if rms needs the DNA filled in .psect $idata rw,d,lcl,rel,con ; ; Terminal settings and parameter lists for line setting ; savass: .byte TC.SLV,0 ; /{no}SLAVE setass: .byte TC.SLV,1 ; /SLAVE=TTnn: assdon: .word 0 savchr: ; Saved line parameters .byte TC.ACR,0 ; /{NO}WRAP .byte TC.FDX,0 ; /{NO}FULLDUPLEX .byte TC.HFF,0 ; /{NO}FORMFEED .byte TC.HHT,0 ; /{NO}TAB .byte TC.NEC,0 ; /{NO}ECHO .byte TC.SLV,0 ; /{NO}SLAVE .byte TC.SMR,0 ; /{NO}LOWERCASE .byte TC.WID,0 ; /WIDTH = n .byte TC.8BC,0 ; /{NO}EIGHTBIT .byte TC.BIN,0 ; /{NO}RPA (BDN 04-Aug-84) savlen = .-savchr savdon: .word 0 ; ; Local line buffer for binary reading ; inilun: .word 0 linbuf: .blkb 140. ; Buffer itself maxlin = .-linbuf ; Maximum read length .even linptr: .word linbuf ; Scan pointer icrem: .rept 15. ; # characters remaining .word 0 .endr ; ; Other r/w data ; .psect $pdata ro,d,lcl,rel,con ; Read-only data datchr: ; Data mode line parameters .byte TC.ACR,0 ; /NOWRAP .byte TC.FDX,1 ; /FULLDUPLEX .byte TC.HFF,1 ; /FORMFEED .byte TC.HHT,1 ; /TAB .byte TC.NEC,1 ; /NOECHO .byte TC.SLV,1 ; /SLAVE .byte TC.SMR,1 ; /LOWERCASE .byte TC.WID,200. ; /WIDTH = 200. .byte TC.8BC,1 ; /EIGHTBIT datlen = . - datchr ibmmod: .byte tc.bin,1 ; /RPA (need to read XON's) .sbttl xinit - assign & attach command terminal .mcall alun$s ,astx$s ,qiow$s .psect $code ; XINIT - Assign and attach the command terminal ; ; This routine assigns and attaches the command terminal (the ; terminal that "ran" this copy of Kermit-11. ; *** N O T E *** Later, this routine should establish a ^C ; AST so that user can abort in-progress file transfers, and ; get Kermit out of server mode without having to send it a ; finish command. I'll wait for Brian to send me his changes ; for graceful transfer abort before I implement this, though. .enabl lsb xinit:: call getsys ; rsx or m+ cmpb r0 ,#sy$mpl ; if m+ require defdev bne 10$ ; not m+ mov sp ,fu$def ; m+, set SY: as def 10$: cmpb r0 ,#sy$pro ; p/os? bne 20$ ; no mov sp ,proflg ; yes, flag it copyz #xk$dev ,#ttname ; also set the device copyz #xk$dev ,#ttdial ; to XK0: clr remote ; and we are local clr con8bit ; clear bit 7 message ,cr ; tell the user calls setspd ,<#ttname,xk$spd,#lun.co> ; try to set the speed tst r0 ; for XK0: bne 20$ ; oops message ; tell them decout xk$spd ; print the speed message ; cr/lf 20$: alun$s #lun.tt,#"TI,#tiunit ; Assign command term. qiow$s #io.att,#lun.tt,#ef.tt,,#kbiost ; Attach it, also clr tcdlu ; don't change tc.dlu call setcc ; enable ^C asts return .save .psect $xkdev ,ro,d,lcl,rel,con xk$dev::.asciz /XK0:/ .even .dsabl lsb xk$spd::.word 9600. .restore global setcc:: qiow$s #io.det,#lun.tt,#ef.tt,,#kbiost qiow$s #io.ata,#lun.tt,#ef.tt,,#kbiost,,<,0,#ttast> return ttast: cmpb (sp) ,#'c&37 ; control C ? bne 100$ ; no call cctrap ; yes, call handler to check it 100$: tst (sp)+ astx$s ; and exit from ast service global .sbttl ttyini - Save & switch line to data mode ; T T Y I N I ; ; ttyini( %loc device_name ,%val channel_number ,%val ccflag ) ; ; ; input: @r5 .asciz string of device name (Ignored on native RSX) ; 2(r5) channel number (LUN) ; 4(r5) mode bits: (Ignored on native RSX) ; ; output: r0 error codes ; ; On RSX, this routine does dynamic switching of terminal from ; interactive mode(s) to data mode(s). The ttysav(), ttyset() ; and noecho() routines are no-ops ... ; ; It is used only for packet communications. The "doconn()" in ; this module handles the setup and restoration of the terminal ; lines for CONNECT modes. ; ; ** Someday, the whole command terminal and communication line handling ; architecture should be smoothed out and simplified, once Brian and ; I get together and compare notes re: native RSX versus emulated RSX, ; and what is required for compatibility without too much pain ... ; ; Added SREX 22-Jun-84 11:15:46 Brian Nelson ; ; Bob Denny ; .mcall srex$s ,exit$s ttyini::save call ttpars ; Get unit number bcs 10$ alun$s 2(r5),r1,r0 ; Assign LUN mov $dsw,r0 ; get the result bcs 10$ ; oops clr r0 ; Make return success clr savdon ; not saved tt settings yet cmp 2(r5),#lun.co ; Command terminal (SAFETY) beq 10$ ; (yes, ignore this) qiow$s #sf.gmc,2(r5),#ef.tt,,#kbiost,,<#savchr,#savlen> mov kbiost,r0 cmpb r0,#IS.SUC ; OK? bne 10$ ; (no) mov sp ,savdon ; we have done the save mov 2(r5) ,inilun ; save this lun (BDN) srex$s #abort ; in case server aborted (BDN) tstb handch ; IBM crap (BDN 04-Aug-84) beq 5$ ; no qiow$s #sf.smc,2(r5),#ef.tt,,#kbiost,,<#ibmmod,#2> ; 5$: qiow$s #sf.smc,2(r5),#ef.tt,,#kbiost,,<#datchr,#datlen> mov kbiost,r0 cmpb r0,#IS.SUC ; OK? bne 10$ ; (no) clr r0 ; Yes - clear r0 = OK 10$: tst proflg ; if a pro/350, ignore errors beq 100$ ; not a 350 clr r0 ; a 350, forget about the errors 100$: unsave return abort: qiow$s #sf.smc,inilun,#ef.tt,,#kbiost,,<#savchr,#savlen> jmp exit ; T T Y F I N ; ; ttyfin( %loc device_name ,%val channel_number ) ; ; ; input: @r5 .asciz string of device name (Ignored on native RSX) ; 2(r5) channel number (LUN) ; ; No need for ttyrst() ; ttyfin:: ; no more abort handling srex$s cmp 2(r5),#lun.co ; Command terminal? beq 10$ ; (yes, skip it) tst savdon ; ever save the crap? beq 10$ ; no, don't reset it qiow$s #sf.smc,2(r5),#ef.tt,,,,<#savchr,#savlen> 10$: clr r0 return ; STUB ROUTINES - Not needed here ; ttrini:: ttrfin:: ttysav:: ttyset:: ttyrst:: noecho:: echo:: clr r0 return .sbttl get terminal name ; G T T N A M ; ; input: @r5 address of 8 character buffer for terminal name ; output: .asciz name of terminal .mcall glun$s gttnam::save ; save temps please mov @r5 ,r3 ; point to output buffer please sub #20 ,sp ; allocate a buffer for GLUN$S mov sp ,r2 ; point to it please glun$s #lun.tt ,r2 ; try it cmpb @#$DSW ,#is.suc ; did it work ? bne 90$ ; no, return the error code please movb g.luna+0(r2),(r3)+ ; get the device name next movb g.luna+1(r2),(r3)+ ; both bytes of it please clr r1 ; get the unit number next please bisb g.lunu(r2),r1 ; simple clr r0 ; now compute the ascii name div #10 ,r0 ; simple (in octal please for RSX) mov r1 ,-(sp) ; save the low order unit number cmp r0 ,#7 ; unit number > 77 octal ? blos 10$ ; no mov r0 ,r1 ; yes, do it again please clr r0 ; simple div #10 ,r0 ; and so on add #'0 ,r0 ; convert to ascii please movb r0 ,(r3)+ ; get the high part copied mov r1 ,r0 ; and now put the next digit back 10$: mov (sp)+ ,r1 ; get the low digit back now add #'0 ,r0 ; convert to ascii add #'0 ,r1 ; likewise movb r0 ,(r3)+ ; move the unit number in now movb r1 ,(r3)+ ; at last .... movb #': ,(r3)+ ; please insert a colon: clrb @r3 ; make it .asciz clr r0 ; no errors br 100$ ; exit 90$: moverr @#$dsw ,r0 ; get the directive error code 100$: add #20 ,sp ; pop glun$s buffer unsave return .sbttl Vanilla read from command terminal ; K B R E A D ; ; Read a line from the command terminal (80 characters max) ; ; Input: @r5 Address of 80 character buffer ; ; Output: r0 = 0 if OK, else error code ; r1 = Number of characters if OK, else 0 ; ; Echoes a on completion to counter Dave Cutler's old ; FORTRAN record processing view of the world. kbread:: qiow$s #io.rvb,#5,#ef.tt,,#kbiost,,<@r5,#80.> clr r0 ; assume no errors mov kbiost+2,r1 ; return bytecount in r1 cmpb kbiost ,#is.suc ; successful read ? beq 100$ ; yes clr r1 ; no data please moverr kbiost ,r0 ; return the error 100$: print #200$ return 200$: .byte lf,0 .sbttl terminal read/write binary mode ; B I N R E A ; ; binread( %val channel_number, %val timeout ) ; ; ; input: @r5 channel number ; 2(r5) timeout (if -1, then no wait) (do this for RSX??) ; ; output: r0 error ; r1 character read ; ; This version uses "normal" reading, as KERMIT sends its packets ; ending in its "EOL" character, which we need to be a . This ; makes reading packets a piece'o cake. We simply buffer lines ; here and scan off characters as needed. Terminal modes have ; been set for reasonably low driver overhead. ; ; No longer used by CONNECT ; binrea::tstb handch ; doing ibm style xon handshaking BDN beq 5$ ; then we must do single char qios BDN call xbinrea ; do that and exit BDN return ; bye BDN 5$: save mov @r5 ,r2 ; lun to use today asl r2 ; fix it for word indexing 10$: tst icrem(r2) ; Anything remaining in current line? bne 20$ ; (yes) call rdlin ; No, maybe issue a read bcs 30$ ; (read error) br 10$ ; Try again 20$: clr r1 ; Move next char unsigned ... bisb @linptr,r1 ; ... into r1 inc linptr ; Advance pointer dec icrem(r2) ; Decrement # characters remaining clr r0 ; Success 30$: unsave return ; Return ; ; RDLIN - Local read routine ; ; Inputs: ; @r5 LUN to read on ; 2(r5) timeout, seconds ; ; Outputs: ; C-bit clear Successful read (something read before timeout) ; icrem = number of characters in this line ; linptr -> 1st character in the line ; ; C-bit set Failed ; R0 = error code ; icrem = 0 rdlin: clr icrem(r2) ; Reset buffer counter mov #linbuf,linptr ; Reset scan pointer 10$: clr r0 ; Assume no timeout mov 2(r5),r1 ; R1 = timeout in seconds ble 20$ ; (no timeout) add #9.,r1 ; Round up to nearest 10 second clicks div #10.,r0 ; Convert to 10 sec. clicks 20$: tst proflg ; pro/350? bne 25$ ; yes tst chario ; force pro/350 style reads today? bne 25$ ; yes qiow$s #,@r5,#ef.tt,,#kbiost,,<#linbuf,#maxlin,r0> br 30$ 25$: clr -(sp) ; get the typehead buffer size mov sp ,r1 ; point to the parameter area movb #tc.tbf ,@r1 ; we want amount in the buffer qiow$s #sf.gmc,@r5,#ef.tt,,,, movb 1(r1) ,r1 ; get the typeahead size bne 26$ ; we have something to get there inc r1 ; nothing, wait for one character 26$: qiow$s #,@r5,#ef.tt,,#kbiost,,<#linbuf,r1,r0> tst (sp)+ ; pop sf.gmc buffer please 30$: cmpb kbiost ,#is.tmo ; timed out on the read ? beq 40$ ; yes movb kbiost,r0 ; Some kind of success? bmi 90$ ; no mov kbiost+2,icrem(r2) ; Yes, set up number of characters mov #linbuf,r1 ; R1 --> line buffer add icrem(r2),r1 ; R1 --> first free byte at end of line movb kbiost+1,(r1) ; Get possible terminator character beq 35$ ; (none) inc icrem(r2) ; Adjust for terminator 35$: clrb (r1) ; Null terminate just for grins clr r0 ; Clear r0 and C-bit return ; Finished 40$: movb #er.tmo ,r0 ; return timeout error code clr icrem(r2) ; just to be safe sec ; say it failed return 90$: clr icrem(r2) ; to be safe sec ; Error return ; bye ; B I N W R I ; ; write( %loc buffer, %val buffer_length, %val channel_number ) ; ; input: @r5 buffer address ; 2(r5) buffer size ; 4(r5) channel number ; output: r0 error code binwri:: qiow$s #io.wal,4(r5),#ef.tt,,#kbiost,,<@r5,2(r5)> clr r0 return .sbttl real binary i/o for doing ^X and ^Z things ; X B I N R E A ; ; binread( %val channel_number, %val timeout ) ; ; ; input: @r5 channel number ; 2(r5) timeout (if -1, then no wait) (do this for RSX??) ; ; output: r0 error ; r1 character read ; xbinre::save ; save a register for a friend clr -(sp) ; allocate a buffer please mov sp ,r2 ; and point to it now clr -(sp) ; allocate a buffer for SF.GMC mov sp ,r3 ; and point to it please cmp 2(r5) ,#-1 ; get without any wait today ? bne 20$ ; no, check for timeouts now movb #tc.tbf ,@r3 ; create a .byte tc.tbf,0 qiow$s #sf.gmc,@r5,#ef.tt,#50,#kbiost,, cmpb kbiost ,#is.suc ; did the read terminal thing work? bne 90$ ; no tstb 1(r3) ; any data in the typeahead buffer? bne 20$ ; yes movb #nodata ,r0 ; fake a psuedo no data error br 100$ ; and exit 20$: clr r0 ; get timeout in seconds into mov 2(r5) ,r1 ; ten second chunks (better than ble 30$ ; -1 timeout or no timeout div #10. ,r0 ; nothing, i suppose) tst r0 ; anything left ? if not, make it bne 30$ ; 10 seconds please inc r0 ; 1 --> 10 seconds 30$: .iif ne, r$rsts,emt 22 ; if testing under RSTS disable echo qiow$s #io.ral!tf.rne!tf.tmo,@r5,#ef.tt,#50,#kbiost,, .iif ne, r$rsts,emt 20 ; if testing under RSTS enable echo clr r1 ; get the character now please bisb @r2 ,r1 ; copy it with sign extension! clr r0 ; assume no errors cmpb #is.suc ,kbiost ; did it work ? beq 100$ ; yes, exit cmpb #is.tmo ,kbiost ; timeout bne 90$ ; no movb #er.tmo ,r0 ; yes br 100$ ; bye 90$: moverr kbiost ,r0 ; no, return the error 100$: cmp (sp)+ ,(sp)+ ; pop the 2 buffers please unsave ; from DIRECTIVE errors return ; bye chkabo::calls xbinrea ,<#lun.tt,#-1> ; simple read on console terminal tst r0 ; did it work ok ? bne 100$ ; no mov r1 ,r0 ; yes, return ch in r0 please return 100$: clr r0 ; it failed return .sbttl normal i/o to the terminal ; S T T Y O U ; ; input: 2(sp) buffer address ; 4(sp) buffer length ; output: 'c' set on error ; 'c' clear on no error ; ; ; L $ T T Y O ; ; l$ttyou( %loc buffer, %val string_length ) ; ; input: @r5 buffer address ; 2(r5) buffer length l$ttyo:: save ; save temps here please movb kbiost ,-(sp) ; save old io status mov 2(r5) ,r0 ; string length bne 20$ ; length was passed mov @r5 ,r0 ; no length, assume .asciz 10$: tstb (r0)+ ; move along looking for a null bne 10$ ; none yet so far sub @r5 ,r0 ; get the length dec r0 ; off by one 20$: qiow$s #io.wvb,#5,#ef.tt,,#kbiost,,<@r5,r0> cmpb kbiost ,#is.suc ; did it work ? bne 90$ ; no, exit with carry set clc ; yes, it worked br 100$ ; exit 90$: sec ; write failed, set error flag and exit 100$: movb (sp)+ ,kbiost unsave ; pop registers that we used return ; and exit sttyou:: mov r5 ,-(sp) mov sp ,r5 add #4 ,r5 call l$ttyo mov (sp)+ ,r5 return l$pcrl:: .print #100$ return 100$: .byte cr,lf,0,0 .sbttl exit kermit and logout .mcall exit$s ,rpoi$s bye: .rad50 /...BYE/ logout:: tst assdon ; ever slave the line? beq 10$ ; no qiow$s #sf.smc,#lun.as,#ef.tt,,#kbiost,,<#savass,#2> ; 10$: rpoi$s #bye exit$s exit:: tst assdon ; ever slave the line? beq 10$ ; no qiow$s #sf.smc,#lun.as,#ef.tt,,#kbiost,,<#savass,#2> ; 10$: exit$s quochk:: clr r0 ; try to see if the logout will work return dskuse:: mov @r5 ,r0 copyz #120$ ,r0 return 120$: .asciz /Can't do space enquiry for RSX/ .even .sbttl cantyp cancel typeahead ; C A N T Y P ; ; cantyp(%val channel_number) ; ; input: @r5 device name ; 2(r5) lun ; ; ; Cantyp tries to dump all pending input on a given terminal ; line. cantyp:: save ; use r0 to point into xrb clr -(sp) ; allocate buffer for SF.SMC mov sp ,r1 ; point to it please movb #tc.tbf ,@r1 ; cancel all typeahead please mov 2(r5) ,r0 ; get the channel number please asl r0 ; purge internally buffer chars clr icrem(r0) ; simple asr r0 ; restore lun bne 10$ ; ok mov #5 ,r0 10$: qiow$s #sf.smc,r0,#ef.tt,,#kbiost,, 100$: tst (sp)+ unsave ; all done return ; bye ; T T X O N ; ; input: @r5 device name ; 2(r5) lun ; output: r0 error code (really, it will be zero) ; ; ; TTXON cancels xoff on a line ttxon:: save ; use r0 to point into xrb clr -(sp) ; allocate buffer for SF.SMC mov sp ,r1 ; point to it please movb #tc.cts ,@r1 ; cancel all typeahead please clrb 1(r1) ; zero means to cancel xoff mov 2(r5) ,r2 ; get the channel number please bne 10$ ; ok mov #5 ,r2 10$: qiow$s #sf.smc,r2,#ef.tmp,,,, qiow$s #io.wal,r2,#ef.tmp,,,,<#200$,#1> 100$: tst (sp)+ unsave ; all done clr r0 ; no errors return ; bye 200$: .byte 'Q&37,0 .sbttl get uic ; G E T U I C ; ; input: nothing ; output: r0 current UIC/PPN of the user .mcall gtsk$s getuic:: sub #40 ,sp ; allocate gtsk buffer mov sp ,r0 ; point to the buffer please gtsk$s r0 ; simple mov g.tspc(r0),r0 ; return the uic add #40 ,sp ; pop the buffer and exit return drpprv:: return getprv:: return .sbttl suspend the job for a while ; S U S P E N ; ; suspend(%val sleep_time) ; ; input: @r5 time to go away for .mcall mrkt$s ,wtse$s suspen:: tst @r5 ; nonzero seconds call ? bne 10$ ; yes mrkt$s #ef.tt,2(r5),#1 ; no, sleep passed # of ticks br 20$ ; and now wait for the timeout 10$: mrkt$s #ef.tt,@r5,#2 ; sleep integral # of seconds 20$: wtse$s #ef.tt return .sbttl ttypar set parity stuff for kermit ; T T Y P A R ; ; ttypar( %loc terminal name, %val paritycode ) ; ; input: @r5 address of terminal name ; 2(r5) parity code ; output: r0 error code .if ne ,0 ; we are doing it in software as of .ift ; 28-Mar-84 09:11:18 (BDN) ttypar:: call ttpars ; get the terminal unit number bcs 100$ ; oops 100$: movb @#$DSW ,r0 ; get any errors return .endc chkpar::clr r0 return .sbttl hangup a terminal, set dtr on a terminal ; T T Y H A N ; ; ttyhan( %loc terminalname ) ; ; input: @r5 address of the terminal name ; output: r0 error code ttyhan::save call ttpars ; the usual, parse the device name bcs 100$ ; oops alun$s #lun.co,r1,r0 ; assign the terminal please qiow$s #io.att,#lun.co,#ef.tt,#50,#kbiost qiow$s #io.hng,#lun.co,#ef.tt,#50,#kbiost moverr kbiost ,r0 qiow$s #io.det,#lun.co,#ef.tt,#50,#kbiost br 100$ ; bye 90$: moverr @#$DSW ,r0 ; return TTPARS error code 100$: unsave return ; raise DTR on a terminal line ; ; T T Y D T R ; ; ttydtr( %loc terminalname ) ; ; input: @r5 address of the terminal name ; output: r0 error code ttydtr:: call ttpars ; the usual, parse the device name bcs 100$ ; oops 100$: movb @#$DSW ,r0 ; return error code and exit return ; bye .sbttl ttspeed get speed for line ; T T S P E E D ; ; input: @r5 name of terminal or address of null for current ; output: r0 current speed ; .psect $pdata splst: .word 0 ,50. ,75. ,110. ,134. ,150. ,200. .word 300. ,600. ,1200. ,1800. ,2000. ,2400. ,3600. .word 4800. ,7200. ,9600. ,19200. ,38400. ,-1 setlst: .word s.0 ,s.50 ,s.75 ,s.110 ,s.134 ,s.150 ,s.200 .word s.300 ,s.600 ,s.1200 ,s.1800 ,s.2000 ,s.2400 ,s.3600 .word s.4800. ,s.7200 ,s.9600 ,s.19.2 ,s.38.4 ,-1 .psect $code ttspee:: save clr -(sp) ; allocate buffer for SF.GMC clr -(sp) call ttpars ; parse the terminal device name bcs 90$ ; error in device name ? alun$s #lun.co,r1,r0 ; assign the terminal please mov sp ,r2 movb #tc.xsp ,@r2 movb #tc.rsp ,2(r2) qiow$s #sf.gmc,#lun.co,#ef.tt,,#kbiost,, movb kbiost ,-(sp) movb (sp)+ ,kbiost clr r0 ; assume zero speed cmpb kbiost ,#is.suc ; did the read speed thing work ? bne 90$ ; not really movb 1(r2) ,r2 ; get the speed setting please clr r1 ; find the index into speed table 10$: cmp setlst(r1),#-1 ; reached the end of table yet ? beq 90$ ; yes, exit cmpb setlst(r1),r2 ; a match yet beq 20$ ; yes tst (r1)+ ; no, index := index + 2 br 10$ ; next please 20$: mov splst(r1),r0 ; return decimal of the speed br 100$ ; bye 90$: 100$: cmp (sp)+ ,(sp)+ unsave return .sbttl set the speed of a terminal line .mcall astx$s ,cmkt$s ,mrkt$s ,qiow$s ; S E T S P D ; ; setspd(%loc devicename, %val speed) ; ; input: @r5 device name ; 2(r5) speed ; 4(r5) lun ; output: r0 error code, 255 if invalid speed setspd::save mov 2(r5) ,r2 ; the speed mov 4(r5) ,r4 ; save the lun call ttpars ; parse the terminal name bcs 90$ ; oops clr r3 ; match the passed speed to the 10$: cmp splst(r3),#-1 ; speed desired to get the index beq 80$ ; end of the table, invalid speed cmp splst(r3),r2 ; a match yet ? beq 20$ ; yes tst (r3)+ ; no, look again please br 10$ ; next 20$: clr -(sp) ; we have the index into speedtable clr -(sp) ; allocate a SF.SMC buffer now mov sp ,r2 ; point to a buffer for SF.SMC movb #tc.xsp ,(r2)+ ; stuff xmitter change code in movb setlst(r3),(r2)+ ; and the desired speed now movb #tc.rsp ,(r2)+ ; stuff receiver change code in movb setlst(r3),(r2)+ ; and the desired speed now mov sp ,r2 ; point back to the buffer now alun$s r4,r1,r0 ; assign the terminal please ; qiow$s #io.att,r4,#ef.tt,#50 mrkt$s #ef.tmo,#2,#2,#spdtmo ; in case we can't get the device qiow$s #sf.smc,r4,#ef.tt,#50,#kbiost,, cmkt$s #ef.tmo,#spdtmo ; we got it ok ; qiow$s #io.det,r4,#ef.tt,#50 cmp (sp)+ ,(sp)+ ; pop the buffer we used clr r0 ; assume success cmpb kbiost ,#is.suc ; did it work ? beq 100$ ; yes, exit without error cmpb kbiost ,#ie.pri ; protection violatian here ? bne 70$ ; no call spwnbd ; yes, try to spawn MCR SET /SPEED= br 100$ ; and exit 70$: moverr kbiost ,r0 ; no, return the error and exit br 100$ ; and exit with the error code 80$: mov #377 ,r0 ; unknown speed br 100$ ; exit 90$: moverr @#$dsw ,r0 ; error from parse br 100$ 100$: unsave ; bye return spdtmo: tst (sp)+ ; remove the event flag number qiow$s #io.kil,r4,#ef.tt,#50,#kbiost movb #ie.abo ,kbiost ; insure that's the error code astx$s ; exit from this timeout ast .sbttl SPWNBD set the line speed via a spawn if SF.SMC fails ; input: ; @r5 address of terminal name, asciz ; 2(r5) speed to do it to ; output: ; r0 error code from spawn spwnbd: sub #70 ,sp ; allocate some buffers please mov sp ,r1 ; pointer to the MCR command line sub #10 ,sp ; pointer to decimal conversion buffer mov sp ,r2 ; simple copyz #200$ ,r1 ; copy over MCR SET/SPEED= strlen r1 ; get over to the end of it now add r0 ,r1 ; simple copyz @r5 ,r1 ; copy the device name over now strlen r1 ; point to the end of it please add r0 ,r1 ; simple deccvt 2(r5) ,r2 ; convert speed to decimal now calls cvt$$ , ; remove any spaces please add r0 ,r2 ; point to the end of the line clrb @r2 ; make it .asciz please mov sp ,r2 ; point back to the start of buffer copyz r2 ,r1 ; stuff the speed data into the mcr add r0 ,r1 ; command buffer movb #': ,(r1)+ ; simple copyz r2 ,r1 ; again add r0 ,r1 ; point to the end of it clrb @r1 ; insure .asciz please add #10 ,sp ; pop the conversion buffer mov sp ,r1 ; point back to the mcre command calls runmcr ,<#1,r1> ; buffer and try the spawn add #70 ,sp ; pop the mcr command buffer return ; and exit 200$: .asciz #SET /SPEED=# ; part of the mcr command line .even .sbttl spawn a set slave command setsla::save ; save temps we may want sub #60 ,sp ; allocate a local buffer mov sp ,r1 ; point to it copyz #200$,r1 ; copy the SET/SLAVE= strlen r1 ; point to the end by getting add r1 ,r0 ; the length and adding it in copyz @r5,r0,#30 ; add the ttname in calls runmcr ,<#1,r1> ; simple to do add #60 ,sp ; restore the stack unsave ; bye clr r0 ; no error return return 200$: .asciz #SET /SLAVE=# .even .sbttl ttpars get unit number from ttname ; T T P A R S ; ; ttpars( %loc ttname ) ; ; output: r0 unit number or 377 for null string ; r1 device name ttpars:: ; NEEDS TO BE GLOBAL(RBD) save ; parse a device name clr r1 ; no device name yet clrb @#$DSW ; set no error as of yet mov #377 ,r0 ; presume no device name mov @r5 ,r3 ; get the string address tstb @r3 ; anything there ? beq 90$ ; no, error cmpb @r3 ,#'X&137 ; i may try this on 350 some day?? beq 10$ ; ok cmpb @r3 ,#'A&137 ; must be of the format ?Tnnn: blo 90$ ; ok so far cmpb @r3 ,#'Z&137 ; must be of the format ?Tnnn: blos 10$ ; no cmpb @r3 ,#'A!40 ; must be of the format ?Tnnn: blo 90$ ; ok so far cmpb @r3 ,#'Z!40 ; must be of the format ?Tnnn: bhi 90$ ; no 10$: bisb (r3)+ ,r1 ; ok, save the first character swab r1 ; and make a place for the next cmpb @r3 ,#'I&137 ; passed 'TI:' ? beq 105$ ; return unit of 377 then please cmpb @r3 ,#'I!40 ; passed 'TI:' ? beq 105$ ; return unit of 377 then please cmpb @r3 ,#'K&137 ; XK: (?) beq 20$ ; yep cmpb @r3 ,#'T&137 ; must be of the format TTnnn: beq 20$ ; ok so far cmpb @r3 ,#'T!40 ; must be of the format TTnnn: bne 90$ ; no 20$: bisb (r3)+ ,r1 swab r1 ; have the device name in r1 now clr r0 ; could use .parse but this is 30$: movb (r3)+ ,r2 ; get the next digit in the string beq 90$ ; hit end of string cmpb r2 ,#': ; end of the device name ? beq 105$ ; yes, exit please cmpb r2 ,#'0 ; in the range '0'..'7' ? blo 90$ ; oops cmpb r2 ,#'7 ; keep checking please bhi 90$ ; bad device name asl r0 ; r0 = r0 * 8 asl r0 ; ditto asl r0 ; and so forth sub #'0 ,r2 ; convert to binary add r2 ,r0 ; and sum the digit in please br 30$ ; next 90$: movb #ie.idu ,@#$dsw ; fake a bad device name and exit sec ; ok br 110$ ; bye 105$: clr @#$dsw ; no errors clc ; success 110$: unsave ; bye return .sbttl assign device ; Fake a device assignment by attaching to a dummy lun. Also ; check for someone else having it via issueing a mark time. ; Thanks to Bob Denny for that one. ; .mcall alun$s ,astx$s ,cmkt$s ,mrkt$s ,qiow$s ,wtse$s assdev::tst proflg beq 1$ clr r0 return 1$: save qiow$s #io.det,#lun.as,#ef.tt ; Detach possible old device call ttpars bcc 5$ jmp 100$ 5$: mov r0 ,r3 ; save the unit number please clr r2 ; flag if we timed out mrkt$s #ef.tmo,#2,#2,#asstmo ; give 2 seconds to do this cmpb r3 ,#377 ; local terminal ? bne 10$ ; no alun$s #lun.as,#"TI,#0 ; assign the terminal please br 20$ 10$: alun$s #lun.as,r1,r3 ; assign the terminal please 20$: qiow$s #io.att,#lun.as,#ef.tt,,#kbiost mov r2 ,r0 ; did we ever time out bne 110$ ; yes, return busy device cmkt$s #ef.tmo,#asstmo ; and cancel the mark time sub #20 ,sp ; allocate a buffer for glun mov sp ,r2 ; and a pointer to it glun$s #lun.tt ,r2 ; get name of the console terminal cmpb r3 ,#377 ; no unit? beq 40$ ; yes, must be TI: cmp g.luna(r2),r1 ; device name of console same as dev? bne 30$ ; no cmpb g.lunu(r2),r3 ; unit number the same ? beq 40$ ; yes 30$: qiow$s #sf.gmc,#lun.as,#ef.tt,,,,<#savass,#2> ; qiow$s #sf.smc,#lun.as,#ef.tt,,,,<#setass,#2> ; mov sp ,assdon ; flag we did the set /slave=ttnn: 40$: add #20 ,sp ; pop glun buffer clr r0 cmpb kbiost ,#is.suc ; did it work beq 110$ ; yes, return error zero cmpb kbiost ,#ie.daa ; ignore already attached errors beq 110$ ; simple to do moverr kbiost ,r0 ; no, get the error code br 110$ ; and exit 100$: moverr @#$DSW ,r0 110$: unsave return asstmo: tst (sp)+ ; remove the event flag number qiow$s #io.kil,#lun.as,#ef.tt,#50,#kbiost moverr #ie.daa ,r2 ; get the error code and exit astx$s ; exit from this timeout ast .sbttl namcvt remove everything but the filename.type .if ne ,0 ; removeed 21-Mar-84 11:02:40 BDN .ift ; new version in K11CVT.MAC ; N A M C V T ; ; input: @r5 source filespec ; 2(r5) address of where to return FILENAME.TYPE .psect rsxdat ,rw,d,lcl,rel,con tmpbuf: .blkb 100. ; Scratch string buffer ;RBD01+ .psect $code namcvt:: save mov #tmpbuf,r4 ; r4 -> scratch buffer copyz (r5),r4,#100. ; Make a scratch copy of raw string strlen r4 ; R0 = length of raw string add r4,r0 ; R0 -> byte after last in string ; ; Locate '.', tack one on if not there. Strip filetype. ; mov r0,r3 ; R3 -> byte after last in string scan #'.,r4 ; R0 = index of '.' or 0 tst r0 ; Is there a '.' bne 10$ ; (yes, strip file type) movb #'.,(r3)+ ; No '.', tack one on clrb (r3) ; Terminate string. br 40$ ; Skip file type stripping 10$: add r4,r0 ; r0 -> char after the '.' mov #3,r1 ; Counter 20$: call isalnum ; Scan forward for up to 3 alphamerics bcs 30$ ; (non-alphameric) inc r0 ; Next sob r1,20$ ; Loop for up to 3 30$: clrb (r0) ; Terminate string here ; ; Now back up from '.' for up to 9 alphamerics or to start of ; string. ; 40$: scan #'.,r4 ; R0 = index of '.' (should be there!) add r4,r0 ; R0 -> char after '.' sub #2,r0 ; R0 -> char before '.' cmp r0,r4 ; Before start? blo 60$ ; (yes) mov #9.,r1 ; Counter 50$: call isalnum ; Scan backward for up to 9 alphamerics bcs 60$ ; (non-alphameric) dec r0 ; Next cmp r0,r4 ; Backed up bast start? blo 60$ ; (yes) sob r1,50$ ; Loop for up to 9 60$: inc r0 ; R0 -> 1st character of file name ; ; Copy the string from R0 to the null ; copyz r0,2(r5) ; Return the stripped specification unsave ; THAT'S IT!! (ain't that easy?) return ; ; Local routines for namcvt() ; ; ISALNUM - Test character for alphamericity ; ; Input: r0 -> character to test ; Output: C-bit: Clear = alphameric else C-set ; .enabl lc ; For safety's sake isalnum: mov r0,-(sp) ; Save r0 movb (r0),r0 ; r0 = character cmp r0,#'a ; Fold character to upper case blo 10$ bic #177440,r0 ; Make it word-wise upper case 10$: cmp r0,#'Z ; Past "Z"? bhi 100$ ; Yes, not alphameric cmp r0,#'0 ; Below "0"? blo 100$ ; Yes, not alphameric cmp r0,#'A ; Above "A"? bhis 90$ ; Yes, alpha cmp r0,#'9 ; Above "9"? bhi 100$ ; Yes, not alphameric 90$: clc ; Indicate alphameric-ness br 110$ 100$: sec ; Non-alphameric 110$: mov (sp)+,r0 ; Restore pointer return .endc ; for new version .sbttl get date and time .enabl lc .mcall gtim$s ascdat::save mov @r5 ,r0 ; r0 := caller result addr sub #16. ,sp ; make room for result mov sp ,r1 ; result addr for gtim$ gtim$s r1 ; get time and date mov g.tida(r1),r2 ; r2 := day jsr pc ,cnvert ; convert and store day movb #'- ,(r0)+ ; insert dash mov g.timo(r1),r2 ; r2 := month asl r2 add g.timo(r1),r2 ; r2 := 3*month add #mnthtab-3,r2 ; r2 := mnthtab[3*month]@ movb (r2)+ ,(r0)+ movb (r2)+ ,(r0)+ ; store month name movb (r2)+ ,(r0)+ movb #'- ,(r0)+ ; insert dash mov @r1 ,r2 ; r2 := year jsr pc ,cnvert ; convert and store year movb #40 ,(r0)+ ; final space clrb @r0 add #16. ,sp unsave return asctim::save mov @r5 ,r0 ; the desitination sub #16. ,sp ; make room for result mov sp ,r1 ; result addr for gtim$ gtim$s r1 ; get time and date mov #3,r3 ; loop count := 3 add #g.tihr,r1 ; start with hours 1$: mov (r1)+,r2 ; begin loop jsr pc,cnvert ; convert to ascii and store dec r3 ; if done beq 2$ ; then exit loop movb #':,(r0)+ ; else insert colon br 1$ ; end loop 2$: clrb @r0 add #16. ,sp unsave return ; cnvert: internal procedure to convert ; integer in r2 to ascii. cnvert: add #366,r2 ;begin loop tstb r2 bpl cnvert ;end loop add #"00-366,r2 ;convert to ascii swab r2 ;reorder bytes movb r2,(r0)+ ;store digit swab r2 movb r2,(r0)+ ;store digit rts pc mnthtab:.ascii /JanFebMarAprMayJunJulAugSepOctNovDec/ .even .sbttl systat get list of users logged in sercmd:: systat:: moverr #-1 ,r0 return .sbttl dodir get a reasonable directory printed ; D O D I R ; ; input: @r5 wildcarded filespec ; 2(r5) channel to do the i/o on, 0 implies TI: ; ; output: r0 error code ; ; note: a value of zero will direct output to the terminal ; ; Unfortunately, RMS does not seem to return all the nice things ; like dates and sizes and protection. While I have written the ; same using FCS (CSI and .PARSE and qio's) for MINITAB and TED ; I would rather not have to do it that way since we are using ; RMS for everything else. dodir:: save ; save these just for kicks sub #140 ,sp ; allocate a buffer for filenames mov sp ,r2 ; and a pointer to the buffer clr -(sp) ; flag to get lookup to do a $parse mov sp ,r4 ; and point to it please mov @r5 ,r1 ; get the filespec address tstb @r1 ; anything passed at all ? bne 10$ ; yes mov #300$ ,r1 ; no, use *.*;* 10$: call 250$ ; print a header out please clr r3 ; filecount := 0 20$: calls lookup ,<#3,r1,r4,r2> ; do the $search via RMS11 now tst r0 ; did it work at all ? bne 90$ ; no, it failed call 200$ ; it worked, decide where to put it inc r3 ; filecount := succ( filecount ) br 20$ ; next please 90$: cmp r0 ,#ER$NMF ; no more files error ? bne 100$ ; no, just exit then tst r3 ; yes, did we ever find anything ? beq 95$ ; no, change error to ER$FNF clr r0 ; found at least one and we got br 100$ ; no more files error from RMS 95$: mov #ER$FNF ,r0 ; no files found 100$: tst (sp)+ add #140 ,sp ; pop resultant filename buffer unsave ; pop registers we used and exit return ; bye 200$: tst 2(r5) ; i/o to TI: or to a disk file ? bne 210$ ; disk print r2 ; terminal print #320$ ; cr/lf br 220$ ; exit 210$: strlen r2 ; to disk, get the string length calls putrec , ; simple 220$: return 250$: copyz #310$ ,r2 ; yes, copy a header over strlen r2 ; get the current length add r2 ,r0 ; point to the null in it calls fparse , ; build the filename string tst r0 ; did that work ok ? bne 260$ ; no, just exit then strlen r2 ; get the current length again add r2 ,r0 ; and point to the end again movb #cr ,(r0)+ ; and stuff a cr/lf/null movb #lf ,(r0)+ ; simple clrb @r0 ; ok call 200$ ; print it and exit 260$: return 300$: .asciz /*.*;*/ ; everything please 310$: .asciz /Directory of / ; a simple header 320$: .byte cr,lf,0 ; the lonely crlf .even .sbttl fix up error codes $mover: tstb 2(sp) bmi 10$ clr 2(sp) return 10$: neg 2(sp) return .sbttl rsxsys sys command for RSX11M/M+ ; 21-Aug-83 16:12:37 Brian Nelson ; 12-Jan-84 09:54:02 Created from MINITAB v82 source ; 07-Mar-84 21:58:10 Bob Denny - Stop instead of wait, nicer. .enabl gbl .mcall spwn$s ,stse$s ,r50$ runjob:: mov #cli... ,r0 call rsxsys return runmcr:: mov #mcr... ,r0 call rsxsys return rsxsys:: save qiow$s #io.det,#lun.tt,#ef.tt,#50,#kbiost mov r0 ,r4 ; save the CLI we want to use sub #12*2 ,sp ; need eight word exit block BDN mov sp ,r2 ; Get address of exit block BDN clr @r2 ; to be safe ? mov 2(r5) ,r1 ; the command buffer address mov r1 ,r3 ; save it strlen r1 ; get the command string length add r0 ,r3 ; point to the end cmpb -(r3) ,#cr ; trailing carriage return ? bne 5$ ; no dec r0 ; yes, fix the length up 5$: mov r0 ,r3 ; save the length clr r0 ; assume no error please spwn$s r4,,,,,#6,,r2,r1,r3 ; do it bcc 10$ ; Ignore error for now moverr @#$DSW ,r0 ; get the error code please qiow$s #io.att,#lun.tt,#ef.tt,#50,#kbiost print #100$ br 20$ 10$: stse$s #6 ; Stop for task to exit 20$: add #12*2 ,sp ; pop exit status block qiow$s #io.att,#lun.tt,#ef.tt,#50,#kbiost unsave ; pop registers and exit return 100$: .asciz <15><12>/Spawn failure for SYS command/<15><12> .even mcr...: r50$ MCR... cli...: r50$ CLI... .sbttl spool to printer .mcall print$ ; can we do this with RMS i/o ????? qspool::movb #1 ,r0 return ; calls open ,<@r5,2(r5)> ; calls rsxspl ,<2(r5)> ;100$: return ; ; ;rsxspl::mov r0 ,-(sp) ; save temps ; mov r1 ,-(sp) ; also this one ; mov @r5 ,r1 ; unit number file is open on ; asl r1 ; get into word offset ; mov fdblst(r1),r1 ; fdb for that file ; clr errsav ; print$ r1,,,#"LP,#1 ; spool file to lp0 now ; bcc 100$ ; moverr f.err(r1) ;100$: mov (sp)+ ,r1 ; pop temps and exit ; mov (sp)+ ,r0 ; ; return ; bye .sbttl detach for the server ; Much simpler for RSX than for RSTS detach::qiow$s #io.det,#5,#ef.tt,,#kbiost clr r0 return .sbttl error message text syserp:: save mov @r5 ,r0 call rmserp .print #200$ unsave return 200$: .byte cr,lf,0,0 syserr:: save ; save a register clr -(sp) ; allocate variable for error # mov sp ,r1 ; and point to it mov @r5 ,@r1 ; if errornumber > 0 bmi 10$ ; then calls direrr ,<#2,r1,2(r5)> ; call fiperr(num,text) br 100$ ; else 10$: calls rmserr ,<#2,r1,2(r5)> ; call rmserr(num,text) 100$: tst (sp)+ unsave return global .sbttl SENBRK send a break, lun assigned, passed in 2(r5) senbrk::save ; save scratch registers please clr -(sp) ; clear a sf.gmc buffer out clr -(sp) ; ditto mov sp ,r2 ; and a pointer to it please movb #tc.xsp ,@r2 ; we want the current speed settings movb #tc.rsp ,2(r2) ; ditto qiow$s #sf.gmc,2(r5),#ef.tmp,#50,,, movb 1(r2) ,r1 ; save the old speed setting please mov sp ,r2 ; reset the buffer address please movb #tc.xsp ,(r2)+ ; stuff xmitter change code in movb #s.50 ,(r2)+ ; and the desired speed now movb #tc.rsp ,(r2)+ ; stuff receiver change code in movb #s.50 ,(r2)+ ; and the desired speed now mov sp ,r2 ; point back to the buffer now qiow$s #sf.smc,2(r5),#ef.tmp,#50,,, qiow$s #io.wal,2(r5),#ef.tmp,#50,,,<#200$,#2> mov sp ,r2 ; reset the buffer address please movb #tc.xsp ,(r2)+ ; stuff xmitter change code in movb r1 ,(r2)+ ; and the old speed now movb #tc.rsp ,(r2)+ ; stuff receiver change code in movb r1 ,(r2)+ ; and the old speed now mov sp ,r2 ; point back to the buffer now qiow$s #sf.smc,2(r5),#ef.tmp,#50,,, cmp (sp)+ ,(sp)+ ; pop local buffer and exit clr r0 ; no errors are ever returned unsave ; pop local registers and exit return ; bye 200$: .byte 0,0 .end