.title k11rt4 i/o for rt11 version 4 or 5 for Kermit-11 .ident /1.0.01/ ; 08-Mar-84 09:18:25 Brian Nelson ; ; 6-May-85 Added a little more to the TSX message to ; indicate that the TSX version comes up in ; the remote mode. If a set line 0 is performed ; you are changed to a local kermit and send ; receive do not work. Going to server mode ; works fine. Purpose of the message is to alert ; user that the default is remote mode and no ; setting of the line is required. ; ; Copyright (C) 1984 Change Software, Inc. ; ; This is the RT11 version of K11RMS.MAC. It simply tries ; to emulate, as much as is reasonable, what the RMS i/o ; routines do for RSX and RSTS. This strains a few things ; in as much that RT11 does not provide much of anything ; in the sense of file services as compared to that which ; RMS11 v2 provides. Since the whole of Kermit-11 is built ; around RMS11 for i/o we will even take the step to map ; RT11 error codes into RMS11 error codes, thus allowing ; the use of the RMS error routines and removing any need ; to modify Kermit-11 elsewhere. ; We won't really use the RMS error routines since they are ; much to comprehensive for the errors that RT can have. ; ; This routine MUST be in the root segment. ; The RT11 executive must have multiple terminal support. ; ; ; Disk i/o epts ; ; open ( %loc filename, %val channel_number ,%val type ) ; create( %loc filename, %val channel_number ,%val type ) ; getrec( %loc buffer , %val channel_number ) { returns RSZ in R1} ; putrec( %loc buffer , %val record_size ,%val channel_number ) ; close ( %val channel_number ) ; putc ( %val char , %val channel_number ) ; getc ( %val channel_number ) .sbttl non disk i/o entry points ; In all cases, R0 will have the returned error code (zero for success) ; For KBREAD and READ, R1 will have the size of the read ; For BINREAD, R1 will have the character just read ; ; The use of %LOC and %VAL are from VMS Pascal and Fortran. ; %LOC means ADDRESS, whereas %VAL means literal. All call ; formats assume the first argument is at 0(r5), the next ; at 2(r5) and so on, as in: ; ; clr -(sp) ; today's date by default ; mov #datebf ,-(sp) ; where to put the converted string ; mov sp ,r5 ; call ASCDAT ; call ascdat ; simple ; cmp (sp)+ ,(sp)+ ; all done ; ; or by using the CALLS macro (defined in K11MAC.MAC) ; ; calls ascdat ,<#datebf,#0> ; ; ; Any version of Kermit-11 which can not, due to the lack of ; executive support, implement a function should return an ; error of -1 in r0. For instance, RT11 does not have any ; executive primitives to do wildcarding directory lookup. ; ; ; ; ; ASCDAT ( %loc buffer, %val datevalue ) ; ASCTIM ( %loc buffer, %val timevalue ) ; ASSDEV ( %loc device_name ) ; BINREA ( %val lun, %val timeout ) ; BINWRI ( %loc buffer, %val byte_count, %val lun ) ; CANTYP ( %loc device_name, %val lun ) ; CHKABO ( ) ; DODIR ( %loc directory_string, %val lun ) ; DRPPRV ( ) ; DSKUSE ( %loc returned_string ) ; ECHO ( %loc terminal_name ) ; EXIT ( ) ; GETPRV ( ) ; GETUIC ( ) ; GTTNAM ( %loc returned_ttname ) ; KBREAD ( %loc buffer ) ; L$PCRL ( ) ; L$TTYO ( %loc buffer, %val bytecount ) ; LOGOUT ( ) ; NAMCVT ( %loc source_filename, %loc returned_normal_name ) ; NOECHO ( %loc device_name, %val lun ) ; QUOCHK ( ) ; READ ( %loc buffer, %val buffer_length, %val lun, %val block_number ) ; SETCC ( %loc control_c_ast_handler ) ; SETSPD ( %loc device_name, %val speed ) ; SUSPEN ( %val seconds, %val ticks ) ; SYSERR ( %val error_number, %loc error_text_buffer ) ; TTRFIN ( ) ; TTRINI ( ) ; TTSPEE ( %loc terminal_name ) ; TTYDTR ( %loc terminal_name ) ; TTYFIN ( %loc terminal_name, %val lun ) ; TTYHAN ( %loc terminal_name ) ; TTYINI ( %loc terminal_name, %val lun, %val open_flags ) ; TTYPAR ( %loc terminal_name, %val parity_code ) ; TTYRST ( %loc terminal_name ) ; TTYSAV ( %loc terminal_name ) ; TTYSET ( %loc terminal_name ) ; WRITE ( %loc buffer, %val buffer_length, %val lun, %val block_number ) ; XINIT ( ) .sbttl define macros and local i/o database .include /IN:K11MAC.MAC/ .iif ndf,k11inc ,.error ; missing INCLUDE for K11MAC.MAC cr = 15 lf = 12 ff = 14 soh = 1 maxsiz = 1000 errbyt == 52 topmem = 50 .enabl gbl .psect $code ,ro,i,lcl,rel,con .psect rtdir1 ,rw,d,gbl,rel,con .psect rtioda ,rw,d,lcl,rel,con ; Note that for RT11, of course, all files are considered ; to be image files. If there was a RMS11/RT we would have ; had transportability from RSX and RSTS version of disk ; i/o. buflst: .word ttbuf ,buf1 ,buf2 ,buf3 ,buf4 bufdef: .word ttbuf ,buf1 ,buf2 ,buf3 ,buf4 bufsiz: .word ttbsiz ,maxsiz ,maxsiz ,maxsiz ,maxsiz filtyp: .word terminal,text ,text ,text ,text bufp: .word 0 ,0 ,0 ,0 ,0 bufs: .word 0 ,0 ,0 ,0 ,0 mode: .word 1 ,0 ,0 ,0 ,0 blknum: .word 0 ,0 ,0 ,0 ,0 filsiz: .word 0 ,0 ,0 ,0 ,0 fb$stm == 0 fb$var == 0 fb$cr == 0 df$rfm::.word 0 df$rat::.word 0 ttbsiz = 40 ttbuf: .blkb ttbsiz+2 buf1: .blkb maxsiz+2 ; record buffer for GETC channel 1 buf2: .blkb maxsiz+2 ; record buffer for GETC channel 2 buf3: .blkb maxsiz+2 ; record buffer for GETC channel 3 buf4: .blkb maxsiz+2 ; record buffer for GETC channel 4 prtbuf: .blkb 40 ; for doing .print output without cc .word 0 time: .word 0,40 timbuf: .word 0,0 timbf1: .word 0,0 clkflg::.word 0 tenth: .word 0,6 freept::.word 0 ; for the next .fetch lun1 = 1 lun2 = 2 lun3 = 3 lun4 = 4 maxlun = lun4 .sbttl error mapping, error codes defined in overlay K11RTE .psect $pdata cloerr::.word er$sy1 ,er$sy1 ,er$sys ,er$prv csierr::.word er$fnm ,er$dev ,er$sy2 dsterr::.word er$dev enterr::.word er$lby ,er$ful ,er$sy3 ,er$prv ,er$sy3 feterr::.word er$dev ,er$sy4 lokerr::.word er$lby ,er$fnf ,er$sys reaerr::.word er$eof ,er$rer ,er$nop ,er$sys wrierr::.word er$eof ,er$wer ,er$nop ,er$sys twaerr::.word er$que mrkerr::.word er$que renerr::.word er$lby ,er$fnf ,er$iop ,er$prv xcierr::.word er$lby ,er$xco xcspfu::.word er$fun ,er$hrd ,er$nop ,er$sys .word er$sup mterr:: .word er$nin ,er$nat ,er$lun ,er$iop ,er$bsy ,er$buf ,er$sys .psect $code .sbttl one shot init code for Kermit-11 RT11 CONFIG = 300 CONFG2 = 370 SYSGEN = 372 $USRLC = 266 SYSVER = 276 PRO350 = 20000 TSXPLU = 100000 .mcall .qset,.twait,.fetch,.gval,.settop,.serr,.herr,.gtim .enabl lsb xinit:: .gtim #rtwork ,#timbuf ; insure clock rollover .gval #rtwork,#CONFIG ; check for what we are running on bit #1 ,r0 ; can't run on SJ bne 10$ ; ok, FB or XM print #nosj ; no, die 10$: .qset #rtque,#10 ; allocate extra que elements mov @#50 ,freept ; free core list for .fetch add #2 ,freept ; + 2 to pointer above us please .gval #rtwork,#$USRLC ; get the usr address please sub #2 ,r0 ; get everything up to the usr .settop ; simple to do clr proflg ; assume not a pro/350 clr tsxflg ; assume not TSX+ clr tsxcl ; assume not tsx and cl ; ; --- Use tried-and-true method to detect TSX+ ; .serr ; Stop abort from RT-11 mov #12$,r0 ; Set the EMT emt 375 ; Do it bcs 11$ ; Not TSX if an error mov sp,tsxflg ; it's TSX+, folks mov sp,remote ; assume remote if TSX+ print #k$tsx ; say who we are 11$: .herr ; Reenable heavy-duty aborts tst tsxflg ; Where now? beq 20$ ; Not TSX - Go check for pro/350 br 40$ ; TSX - skip to the end 12$: .byte 0,110 ; TSX-Plus get-line-number emt ; ;--- This code is for new versions of RT-11 and TSX-Plus. ; It will not work on TSX-Plus/RT-11 versions < 5.0 so we use the old ; method. Code left in for historical interest. ; ; .gval #rtwork,#SYSGEN ; get the sysgen parameters now ; bit #TSXPLU ,r0 ; TSX+ ; beq 20$ ; no ; mov sp ,tsxflg ; yes, flag as being so ; br 40$ ; and do nothing else at all. 20$: .gval #rtwork,#SYSVER ; check for RT version 5 please cmpb r0 ,#5 ; must be version 5 for PRO/350 blo 30$ ; no, can't be a pro .gval #rtwork,#CONFG2 ; get config word number two bit #PRO350 ,r0 ; is this a pro/350 ? beq 30$ ; no copyz #xc$dev ,#ttname ; yes, set XC as the default copyz #xc$dev ,#ttdial ; simple to do mov sp ,proflg ; it's a pro/350, folks clr remote ; we want to be a local kermit print #xcmsg ; say so br 40$ 30$: calls gttnam ,<#errtxt> ; get the name of the console tt: print #ttn ; and say what it should be print #errtxt ; print it print #crlf ; finish the line 40$: mov sp ,clkflg .gtim #rtwork ,#timbuf ; get current time please .twait #rtwork ,#tenth ; wait a little while (1/10 sec) .gtim #rtwork ,#timbf1 ; get current time please cmp timbuf ,timbf1 ; see if time is same please bne 50$ ; no cmp timbuf+2,timbf1+2 ; continue bne 50$ ; not the same .print #noclock ; yes, issue a warning message clr clkflg ; clear clock flag to NONE 50$: 90$: 100$: clr r0 ; no errors return .save .psect $pdata dkdev: .rad50 /DK / nosj:: .asciz /Kermit-11 may not run correctly on a SJ monitor/ ttn: .asciz /RT-11 default terminal line set to unit / noclock:.ascii /This system does not appear to have a line clock./ .asciz /Kermit-11 may not run correctly./ xc$dev: .asciz /XC0:/ xcmsg: .asciz #PRO/350 comm port set to XC0:# k$tsx: .ascii /TSX-Plus remote mode/ crlf: .byte cr,lf,0 .even .restore .dsabl lsb global .sbttl open a file for rt11 .mcall .csispc,.dstatus,.lookup,.fetch,.enter,.close .psect $code ; OPEN ; ; open ( %loc filename; %val channel_number, %val type ) ; ; input: @r5 filename address ; 2(r5) channel number ; 4(r5) val 'binary' or 'text' or 0 ; ; output: r0 rms (really a mapped RT11 error) error code ; ; ; For RT11, create a buffered image file. It is ; always assumed that channel '0' implies writes ; to the attached console terminal. fopen:: open:: save ; save all the temps please mov @r5 ,r1 mov 2(r5) ,r3 ; channel number here bne 5$ ; not implied terminal open mov sp ,mode+0 ; say we will be writing to it clr bufp+0 ; and there is nothing there yet clr r0 ; no errors are possible br 120$ ; bye 5$: asl r3 ; times 2 for word indexing call setatr ; initialize file control data mov 4(r5) ,filtyp(r3) ; ensure the correct type of file sub #20. ,sp ; allocate buffer for the mov sp ,r2 ; device status call sub #40.*2 ,sp ; allocate a buffer for the mov sp ,r1 ; .csispc data mov @r5 ,r0 ; string address 10$: movb (r0)+ ,(r1)+ ; copy it to the csi buffer bne 10$ ; until a null byte is found. movb #'= ,-1(r1) ; fake an output filespec here clrb @r1 ; and .asciz mov sp ,r1 ; reset pointer (also saving sp) .csispc r1,#defext,r1 ; and try to parse the name mov r1 ,sp ; restore from any switches bcs 80$ ; oops calls fetch ,<@r1> ; load the handler if need be tst r0 ; did it work ? bne 120$ ; no, exit please 20$: tst @r1 ; a device is there ? bne 30$ ; yes mov #^RDK ,@r1 ; no, stuff DK: in please 30$: .lookup #rtwork,2(r5),r1 ; open the file for input bcs 100$ ; can not find it mov r0 ,filsiz(r3) ; save the file size please mov #-1 ,bufp(r3) ; say we need to init the buffer clr mode(r3) ; we are reading today clr r0 ; no errors br 120$ ; and exit 60$: mov #dsterr ,r1 br 110$ 80$: mov #csierr ,r1 ; .csispc error mapping br 110$ ; get the correct error now 90$: mov #feterr ,r1 ; .fetch error codes br 110$ 100$: mov #lokerr ,r1 ; .lookup error mapping br 110$ 110$: movb @#errbyt,r0 ; get the error code now asl r0 ; times 2 for indexing into error map add r0 ,r1 ; now map the rt11 error into a fake mov @r1 ,r0 ; of a rms11 error 120$: add #<40.*2>+20.,sp ; pop all the tiny buffers now. unsave ; pop all the temps return ; and get out defext: .word 0 .word 0 .word 0 .word 0 .sbttl create a file for RT11 .psect $pdata en$siz::.word 0 ; 1/2 largest free or 2nd largest .psect $code .mcall .csispc,.dstatus,.lookup,.fetch,.enter,.close .psect $code ; CREATE ; ; open ( %loc filename; %val channel_number, %val type ) ; ; input: @r5 filename address ; 2(r5) channel number ; 4(r5) val 'binary' or 'text' or 0 ; ; output: r0 rms (really a mapped RT11 error) error code ; ; ; For RT11, create a buffered image file. It is ; always assumed that channel '0' implies writes ; to the attached console terminal. fcreat:: create::save ; save all the temps please mov 2(r5) ,r3 ; channel number here bne 5$ ; not implied terminal open mov sp ,mode+0 ; say we will be writing to it clr bufp+0 ; and there is nothing there yet clr r0 ; no errors are possible br 120$ ; bye 5$: asl r3 ; times 2 for word indexing call setatr ; initialize file control data mov 4(r5) ,filtyp(r3) ; ensure the correct type of file sub #20. ,sp ; allocate buffer for the mov sp ,r2 ; device status call sub #40.*2 ,sp ; allocate a buffer for the mov sp ,r1 ; .csispc data mov @r5 ,r0 ; string address 10$: movb (r0)+ ,(r1)+ ; copy it to the csi buffer bne 10$ ; until a null byte is found. movb #'= ,-1(r1) ; fake an output filespec here clrb @r1 ; and .asciz mov sp ,r1 ; reset pointer (also saving sp) .csispc r1,#defext,r1 ; and try to parse the name mov r1 ,sp ; restore from any switches bcs 80$ ; oops tst @r1 ; any device there ? bne 20$ ; no mov dkdev ,@r1 ; yes, stuff a real name for DK: in 20$: calls fetch ,<@r1> ; try to get the device tst r0 ; well ? bne 120$ ; no 30$: tst 2(r1) ; never allow NFS writes to disk beq 70$ ; force bad filename error mov at$len ,r4 ; any passed file length bne 40$ ; yes mov en$siz ,r4 ; no, use our default 40$: .enter #rtwork,2(r5),r1,r4 ; open the file for input bcs 100$ ; can not find it clr bufp(r3) ; say the buffer is empty mov sp ,mode(r3) ; we are writing today clr r0 ; no errors br 120$ ; and exit 60$: mov #dsterr ,r1 ; error from .dstatus br 110$ 70$: mov #er$fnm ,r0 ; bad filename br 120$ ; and exit 80$: mov #csierr ,r1 ; .csispc error mapping br 110$ ; get the correct error now 90$: mov #feterr ,r1 ; .fetch error codes br 110$ 100$: mov #enterr ,r1 ; .enter error mapping br 110$ 110$: movb @#errbyt,r0 ; get the error code now asl r0 ; times 2 for indexing into error map add r0 ,r1 ; now map the rt11 error into a fake mov @r1 ,r0 ; of a rms11 error 120$: add #<40.*2>+20.,sp ; pop all the tiny buffers now. unsave ; pop all the temps return ; and get out .sbttl init things for open and close ; input: r3 channel number times 2 ; output: nothing setatr: save ; init things for open and create clr filsiz(r3) ; no filesize clr bufp(r3) ; offset into the buffer ? clr bufs(r3) ; nothing is there today clr mode(r3) ; say we are reading today clr blknum(r3) ; no block number yet mov #maxsiz ,bufsiz(r3) ; the size of the buffer (1000) mov #binary ,filtyp(r3) ; always image mode for rt11 mov bufdef(r3),buflst(r3) ; insure buffer address is correct mov buflst(r3),r0 ; and clear it out please mov bufsiz(r3),r1 ; the size please 10$: clrb (r0)+ ; simple sob r1 ,10$ ; next please unsave ; pop and exit return getsiz::mov @r5 ,r1 ; get opened filesize asl r1 ; get the lun times 2 mov filsiz(r1),r1 ; return the size clr r0 ; no errors return ; bye .sbttl close a file .mcall .close ; C L O S E ; ; close (%val lun) ; ; input: @r5 channel number to close ; output: r0 mapped error code ; ; calls: flush(lun) close:: save ; save registers we may have call flush ; dump out any remaining buffer mov @r5 ,r1 ; then disconnect the access stream beq 10$ ; terminal .close r1 ; do the rt close bcc 10$ ; it worked movb @#errbyt,r0 ; it failed, map the rt11 error asl r0 ; to something more descriptive mov cloerr(r0),r0 ; simple br 20$ ; map the error please 10$: clr r0 ; no errors 20$: asl r1 ; channel number times 2 clr bufp(r1) ; buffer_pointer[lun] := 0 clr filsiz(r1) ; no size please unsave ; pop the saved r1 return ; and exit with error in r0 rewind::mov @r5 ,r0 ; get the channel number beq 100$ ; for the terminal, a no-op asl r0 ; times two please mov #-1 ,bufp(r0) ; flag a buffer reload is needed clr bufs(r0) ; nothing is in the buffer clr blknum(r0) ; first block of the disk file 100$: clr r0 ; no errors are possible return ; bye .sbttl read a record from a sequential file ; G E T R E C ; ; getrec( %loc buffer, %val channel_number ) ; ; input: @r5 address of user buffer, at least 80 bytes ; 2(r5) channel number ; ; output: r0 rms sts ; r1 record size ; ; Read the next record from a disk file. Assumes that the ; user has supplied a buffer of 132 characters to return ; the record to. ; ; We really don't need GETREC for RT11 Kermit. It was used ; in the RMS11 version (for RSXM/M+ and RSTS) only in con- ; junction with GETC to fake RMS single character i/o. The ; only two places it is called from is for the TYPE and ; HELP commands (C$TYPE and C$HELP). ; GETREC assumes text (stream ascii) file only. getrec::save ; save registers we may need clr r4 ; recordsize := 0 mov @r5 ,r3 ; the recordbuffer address mov #132. ,r2 ; max size of a record to read clr r1 ; nothing read as of yet 10$: cmpb r1 ,#ff ; exit if ch = form_feed beq 30$ ; mov 2(r5) ,r0 ; the channel number to use call getcr0 ; read the next character now tst r0 ; did it work ? bne 100$ ; no cmpb r1 ,#cr ; exit if ch = carriage_return beq 30$ ; cmpb r1 ,#'z&37 ; exit if ch = control_z beq 30$ ; cmpb r1 ,#lf ; ignore line feeds beq 10$ ; inc r4 ; length := succ(length) movb r1 ,(r3)+ ; yes, stuff the character in sob r2 ,10$ ; up until maxrec size mov #er$rtb ,r0 ; too much data, return maxsize br 100$ ; error 30$: cmpb r1 ,#'z&37 ; we get here on reading a record bne 40$ ; terminator mov #er$eof ,r0 ; control z means end of file clr r1 ; no data is there at all br 100$ 40$: mov r4 ,r1 ; return the record length br 100$ ; all done if ff or lf 100$: unsave ; pop registers we saved return ; bye .sbttl put a record to an rt11 sequential file ; P U T R E C ; ; putrec( %loc buffer, %val record_size, %val channel_number ) ; ; input: @r5 address of user buffer ; 2(r5) record size ; 4(r5) channel number ; ; output: r0 rms sts ; ; Write the next record to a disk file. ; ; Assumption: The record to be written will have a cr/lf ; appended to it unless the filetype is not ; text. In other words, PUTREC provides the ; carriage control unless the file is a ter- ; minal. putrec::save ; save registers we may need mov 2(r5) ,r2 ; the size of the i/o mov @r5 ,r3 ; the buffer address mov 4(r5) ,r1 ; the channel number please bne 10$ ; a real disk file tst r2 ; faking output to a terminal beq 100$ ; nothing at all to do ? print r3 ,r2 ; do the terminal i/o br 100$ ; bye 10$: tst r2 ; the size of the i/o to do beq 30$ ; nothing to do, add carriage control 20$: clr r0 bisb (r3)+ ,r0 ; the character to write out call putcr0 ; channel is passed in r1 tst r0 ; did the write fail ? bne 100$ ; yes, exit asap sob r2 ,20$ ; next ch please 30$: asl r1 ; get the channel number times 2 cmp filtyp(r1),#text ; is this a text file bne 100$ ; no, don't add carriage control in asr r1 ; get the channel number back movb #cr ,r0 ; and add in a cr/lf call putcr0 ; simple movb #lf ,r0 ; and at last the line feed call putcr0 ; do the line feed at the end 100$: unsave ; pop registers we saved return ; bye .sbttl getc get one character from an input file .mcall .readw ; G E T C ; ; getc(%val channel_number) ; ; input: @r5 channel_number ; output: r0 rms error status ; r1 the character just read getc:: mov @r5 ,r0 call getcr0 return getcr0::save ; use for saving the channel# 10$: mov r0 ,r3 ; save the channel number please call .getc ; get the next ch please tst r0 ; did the read work ok ? bne 100$ ; no, exit asl r3 ; get the channel number times 2 cmp filtyp(r3),#text ; if filetype[lun] = text bne 100$ ; then tstb r1 ; if ch = NULL bne 100$ ; then try-again asr r3 ; get origional channel back mov r3 ,r0 ; setup the correct call format br 10$ 100$: unsave return .getc: save ; save temps mov r0 ,r2 ; channel number please mov r0 ,r1 ; for the .readw please asl r2 ; times 2 tst bufs(r2) ; anything in the buffer ? beq 10$ ; no, please load it cmp bufp(r2),#-1 ; need to initialize the buffer? bne 20$ ; no 10$: mov bufsiz(r2),r3 ; we need buffer size in words asr r3 ; convert bytes to words .readw #rtwork,r1,buflst(r2),r3,blknum(r2) bcs 90$ ; it failed, bye inc blknum(r2) ; next time read the next block clr bufp(r2) ; it worked. clear current pointer asl r0 ; convert words read to bytes mov r0 ,bufs(r2) ; and save the record size 20$: mov buflst(r2),r3 ; get the address of the buffer add bufp(r2),r3 ; and point to the next character clr r1 ; to be returned in r1 bisb @r3 ,r1 ; simple inc bufp(r2) ; buffer.pointer := succ(buffer.pointer) dec bufs(r2) ; amountleft := pred( amountleft ) clr r0 ; no errors please br 100$ 90$: movb @#errbyt,r0 ; get the error code asl r0 ; times two mov reaerr(r0),r0 ; map it into a unique global error 100$: unsave return .sbttl putc put a single character to an rms file .mcall .writw ; P U T C ; ; input: @r5 the character to put ; 2(r5) the channel number to use ; ; Buffer single character i/o to internal disk buffer. ; Buffer is dumped if internal buffer is full. ; The local buffers are allocated in CREATE and OPEN. putc:: save ; simply save r1 and call putcr0 mov 2(r5) ,r1 ; to do it. putcr0 will be somewhat clr r0 ; faster to call directly due to the bisb @r5 ,r0 ; overhead involved in setting up an call putcr0 ; argument list. unsave ; pop saved r1 and exit return ; bye putcr0::save ; save registers we use mov r1 ,r2 ; channel number asl r2 ; times 2 of course cmp bufp(r2),bufsiz(r2) ; is the buffer full ? blo 20$ ; no, store some more characters in it movb r0 ,r3 ; yes, save the input character r0 mov bufsiz(r2),r4 ; and setup for a .writw asr r4 ; rt11 needs word count not byte count tst r1 ; channel zero is always terminal beq 3$ ; simple cmp filtyp(r2),#terminal ; check for being a terminal today? bne 4$ ; not a terminal 3$: print buflst(r2),bufsiz(r2) ; a terminal, force it out please br 5$ ; and reinit the buffer now 4$: .writw #rtwork,r1,buflst(r2),r4,blknum(r2); dump this block to disk bcs 90$ ; it failed for some reason 5$: inc blknum(r2) clr bufp(r2) ; pointer := 0 mov buflst(r2),r4 ; it worked. zero the buffer now mov bufsiz(r2),r0 ; get the buffer address and size 10$: clrb (r4)+ ; for i := 1 to bufsiz sob r0 ,10$ ; do buffer[i] := chr(0) movb r3 ,r0 ; ok, restore the old character 20$: mov bufp(r2),r1 ; get the current buffer pointer add buflst(r2),r1 ; and point to a new home for the movb r0 ,@r1 ; the input character in r0 inc bufp(r2) ; pointer := succ( pointer ) clr r0 ; success br 100$ 90$: movb @#errbyt,r0 ; get the rt11 error code asl r0 ; times two mov wrierr(r0),r0 ; map it into a global error code 100$: unsave return .sbttl flush .mcall .writw flush: save mov @r5 ,r1 ; get the internal channel number asl r1 ; times 2 for indexing tst bufp(r1) ; anything in the buffer beq 100$ ; no tst mode(r1) ; writing today ? beq 100$ ; no tst r1 ; terminal today ? beq 20$ ; yes mov bufsiz(r1),r2 ; rt11 likes to have word counts asr r2 ; simple .writw #rtwork,@r5,buflst(r1),r2,blknum(r1) br 100$ 20$: print buflst(r1),bufp(r1) br 100$ 100$: unsave clr r0 return .sbttl fparse parse filename and fill in with defaults ; F P A R S E ; ; input: @r5 input filename, .asciz ; defdir the default directory name string to use ; ; output: 2(r5) expanded filename, .asciz, maximum length 63 bytes ; r0 error codes ; ; For RT11, simply return the passed string. Perhaps later do ; something real. fparse::save mov #defdir ,r0 mov 2(r5) ,r1 10$: movb (r0)+ ,(r1)+ bne 10$ dec r1 copyz @r5 ,r1 ; simple clr r0 ; no errors are possible today unsave return ; bye global .sbttl l$ttyout ; Print a string to the console terminal ; ; Input: @r5 buffer address ; 2(r5) string length ; ; If 2(r5) is zero, then assume .asciz l$ttyo::save ; save registers we may need mov @r5 ,r1 ; get the string address mov 2(r5) ,r2 ; get the string length bne 20$ ; non-zero then mov r1 ,r2 ; count until a null now 10$: tstb (r2)+ ; well ? bne 10$ ; not yet, keep looking sub r1 ,r2 ; get the length now dec r2 ; all done beq 100$ ; nothing to print at all? 20$: mov #prtbuf ,r0 ; now buffer the i/o to avoid mov #36 ,r3 ; the printing of cr/lf at the 30$: tstb (r1)+ ; don't copy nulls please beq 35$ ; ignore if null movb -1(r1) ,(r0)+ ; copy a byte please 35$: dec r2 ; done yet ? beq 40$ ; yes sob r3 ,30$ ; no, next please 40$: movb #200 ,(r0)+ ; insure no carraige control ! clrb @r0 ; must be passed .asciz mov #prtbuf ,r0 ; point back to the start of buffer emt 351 ; do the .print kmon request tst r2 ; any more data to buffer ? bne 20$ ; yes, try again 100$: unsave return 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 ; G E T S Y S ; ; output: r0 operating system ; ; sy$11m (1) for rsx11m ; sy$ias (3) for ias ; sy$rsts (4) for rsts ; sy$mpl (6) for m+ ; sy$rt (7) for rt11 ???? getsys::mov #7 ,r0 ; this is rt11 folks return ; bye .sbttl misc routines ; E X I T ; ; exit to kmon .mcall .exit exit:: .exit ; should always work ok halt ; huh ? .mcall .twait ; mark time request suspen::save ; save temps mov @r5 ,r1 ; sleep time in seconds beq 10$ ; nothing, must be fractional mul #60. ,r1 ; sixty clock ticks in a second clr r0 ; low order part br 20$ ; ignore the fractional part 10$: mov 2(r5) ,r0 ; sleep < 1 second 20$: add r1 ,r0 ; total time to sleep mov r0 ,-(sp) ; setup the timeout block clr -(sp) ; two words please mov sp ,r1 ; point to it .twait #rtwork,r1 ; suspend ourself for a while bcs 30$ ; it worked ok clr r0 ; return success br 100$ ; bye 30$: movb @#errbyt,r0 ; it failed, map the error into asl r0 ; a global error number mov twaerr(r0),r0 ; simple 100$: cmp (sp)+ ,(sp)+ ; pop time buffer and exit unsave ; pop registers return ; bye .sbttl dummy epts for rsx/rsts compatibility ; Dummy routines ; ; DRPPRV ( ) ; GETPRV ( ) ; GETUIC ( ) ; LOGOUT ( ) ; QUOCHK ( ) ; SETCC ( %loc ast_handler ) ; TTRFIN ( ) ; TTRINI ( ) setcc:: getuic:: logout:: quochk:: qspool:: noecho:: echo:: runjob::clr r0 getprv:: drpprv:: throtl::return getatr::mov #er$iop ,r0 return putatr::clr r0 return binfil::clr r0 calls chkext ,<@r5> return chkpar::clr r0 return fixwil::clr r0 return systat:: sercmd::mov #er$iop ,r0 return okuser::mov (sp)+ ,@sp return dskuse::mov @r5 ,r0 clrb @r0 return getmcr::mov @r5 ,r0 clrb @r0 clr r0 return .sbttl detach for the server (a no-op for rt) detach::mov #er$iop ,r0 return ; FETCH( %val rad50(devicename) ) fetch:: .dstat #rtwork,r5 bcs 70$ tst rtwork+4 ; is this handler resident ? bne 50$ ; yes mov @#50 ,-(sp) ; check for space to load it sub freept ,@sp ; simple to do cmp rtwork+2,(sp)+ ; is there sufficient space ? bhi 60$ ; no, error and exit .fetch freept ,r5 ; no, try hard to load the thing bcs 80$ ; no way, map the error code please mov r0 ,freept ; update the free pointer and exit 50$: clr r0 ; no errors br 100$ 60$: mov #er$fet ,r0 br 100$ ; no room, exit 70$: mov #dsterr ,-(sp) ; map a .dstat error br 90$ ; and do it 80$: mov #feterr ,-(sp) ; map a .fetch error 90$: movb @#ERRBYT,r0 ; get the error code asl r0 ; word offsets add (sp)+ ,r0 ; the actual address mov @r0 ,r0 ; get it and exit 100$: return ; bye .sbttl things to do eis instructions .psect $mul:: mov r0 ,-(sp) mov r1 ,-(sp) mov 6(sp) ,r0 mov 10(sp) ,r1 mov r0,-(sp) mov #21,-(sp) clr r0 10$: ror r0 ror r1 bcc 20$ add 2(sp),r0 20$: dec (sp) bgt 10$ cmp (sp)+ ,(sp)+ mov r1 ,10(sp) mov (sp)+ ,r1 mov (sp)+ ,r0 mov (sp) ,2(sp) tst (sp)+ return $div:: mov r0 ,-(sp) mov r1 ,-(sp) mov 6(sp) ,r0 mov 10(sp) ,r1 mov #20,-(sp) mov r1,-(sp) clr r1 e00040: asl r0 rol r1 cmp r1,(sp) bcs e00054 sub (sp),r1 inc r0 e00054: dec 2(sp) bgt e00040 cmp (sp)+ ,(sp)+ mov r1 ,6(sp) mov r0 ,10(sp) mov (sp)+ ,r1 mov (sp)+ ,r0 return $ash:: tst 2(sp) ; left shift or rigth shift? bmi 20$ ; do some divisions by 2 10$: tst 2(sp) ; anything left to do ? ble 100$ ; no, time to go asl 4(sp) ; shift left dec 2(sp) ; all done yet ? br 10$ 20$: tst 2(sp) ; divide by 2 bge 100$ ; all done asr 4(sp) ; keep going inc 2(sp) ; all done yet ? br 20$ 100$: mov (sp)+ ,(sp) ; pop rept count and exit return .end