Trailing-Edge
-
PDP-10 Archives
-
ap-c800d-sb
-
metio.mac
There are 5 other files named metio.mac in the archive. Click here to see a list.
; UPD ID= 1991 on 7/30/79 at 5:10 PM by M:<MAYBERRY>
TITLE METIO FOR LIBOL V12A - LSTATS METER I/O CODE
SUBTTL D. WRIGHT JAN, 1979
;THIS SOFTWARE IS FURNISHED UNDER A LICENSE AND MAY ONLY BE USED
; OR COPIED IN ACCORDANCE WITH THE TERMS OF SUCH LICENSE.
;
;COPYRIGHT (C) 1979, BY DIGITAL EQUIPMENT CORPORATION
; HAM 26-JUN-79
; SET VERSION NUMBER TO 2
SUBTTL DEFINITIONS
SEARCH LBLPRM ;DEFINE PARAMETERS.
SEARCH COMUNI
SEARCH FTDEFS
SEARCH UUOSYM
IFN TOPS20,<SEARCH MONSYM, MACSYM>
IFE TOPS20,<SEARCH UUOSYM, MACTEN>
IFN LSTATS,<SEARCH METUNV>
IFN LSTATS,<
EXTERN KILL.,RET.1,RET.2,LMETR. ;CBLIO ROUTINES
IFN TOPS20,<
EXTERN MRTM.E,MRTM.S ;TOPS20 ROUTINES IN CBLIO
>
EXTERN LIBVR.,LIBSW. ;ALSO DEFINED IN CBLIO
;LOWSEG LOCATIONS, DEFINED IN COMUNI, (LOADED IN LILOWS)
EXTERN MROPT.,MROPTT,MRNM6,MRNMA, MRTMB.,MBTIM.,MRHDBP,MRHDFL
EXTERN MRAFFT,MRFKFT,MRTDBP,MRAPN.,MRBLKO,MRBKO.,MRFPGT
EXTERN MRKILL,MRBNUM,MRPSTM,MRRERN,MRLDBL
IFE TOPS20, EXTERN MRCHNN,MRCHCF
IFN TOPS20, EXTERN MRJFN,MRLCJF
IFN DBMS6,< INTERN MRDBST,MRDBEN,MRDBDM
EXTERN MRDDBP
>
IFN FTLSDR,< ;FOR LSTATS.DIR STUFF
IFN TOPS20,<
EXTERN MRLDJF,MRLDNA
>
IFE TOPS20,<
EXTERN MRCHLS,MRLLDR,MRLLDV,MRCFNM,MRCFEX
>
EXTERN MRLBPC,MRLSZL,MRLFPR
>;END IFN FTLSDR
;ENTRY POINTS
ENTRY MRLSET ;SETUP TO WRITE LSTATS FILE
ENTRY MRDMP ;WRITE ONE LSTATS FILE BLOCK IF THERE IS DATA
ENTRY MRDMPT ;DUMP ALL LSTATS DATA AND FINISH WRITING METER FILE
>;END IFN LSTATS
ENTRY MROUT. ;LSTATS ROUTINE TO OUTPUT AN LSTATS BLOCK
HISEG
SALL
T0=0
T1=1
T2=2
T3=3
T4=4
T5=5
AC6=6
AC11=11
SUBTTL METERING STUFF
IFN LSTATS,<
;WORDS IN LSTATS.DIR
.LDBPC==0 ;BYTE PTR TO CURRENT FILENAME
.LDNFL==1 ;# FILES IN DIRECTORY,,# OF CURRENT FILE
.LDSZL==2 ; SIZE LIMIT OF EACH FILE (BLOCKS OR PAGES)
.LDTML==3 ;TIME LIMIT (AS DAYS,,1/3 SECS)
.LDFWR==4 ;TIME OF FIRST WRITE TO CURRENT FILE
.LDAFN==5 ;ASCIZ FILE SPECS
IFN FTLSDR,<
IFE TOPS20, MR.BFB==^D128+5 ;ONE BUFFER + A FEW EXTRA WORDS
IFN TOPS20, MR.BFB==^D128+1 ;ONE BUFFER AND BLANK WORD AT END
>;END IFN FTLSDR
IFE FTLSDR, MR.BFB==^D128 ;ONE BUFFER EXACTLY
IFE TOPS20,<
;TOPS10 LSTATS ROUTINE TO GET A FREE CHANNEL
; RETURNS .+1 IF NONE AVAILABLE, ELSE .+2 WITH NUMBER IN RH(T5)
GMCHAN: SKIPN T5,OPNCH.## ;ANY CHANNELS AVAIL?
POPJ PP, ;NO
MOVE AC6,OPNCBP## ;GET BYTE PTR
HRRI T5,1 ;START WITH 1
MOVEI T2,17 ; UPPER LIMIT
GMCHN2: ILDB AC11,AC6
SOJE AC11,GMCHN1 ; SEE GCHAN. ROUTINE
CAILE T2,(T5)
AOJA T5,GMCHN2
GMCHN0: SETZB T5,AC11 ;USE CHANNEL 0 IF NONE OTHER FREE
GMCHN1: DPB AC11,AC6 ;NOTE CHANNEL UNAVAILABLE
JRST RET.2 ;GIVE SKIP RETURN
>;END IFE TOPS20
>;END IFN LSTATS
IFN LSTATS,<
SUBTTL LSTATS SETUP - OPEN METER FILE
;ROUTINE MRLSET IS CALLED AT RESET TIME TO OPEN THE METER FILE
; PNAME.MTO. IT REMAINS OPEN DURING THE ENTIRE RUN. ON TOPS10
; THIS REQUIRES ONE OF THE PRECIOUS 16 CHANNELS.
MRLSET:
;GET FILENAME TO WRITE THE DATA TO.
;ON TOPS10 USE NNNMTR.TMP
;ON TOPS20 USE <LAST-SIX-DIGITS-OF-TIME-OF-DAY>.MTO;T
IFE TOPS20,<
PJOB T1,
SETZ T3,
MOVEI T4,3 ;WANT 3 DIGITS OF JOB NUMBER
MAKEJN: IDIVI T1,^D10
MOVEI T2,'0'(T2)
LSHC T2,-6
SOJG T4,MAKEJN
HRRI T3,'MTO' ;(T3) = NNNMTO
MOVEM T3,MRNM6 ;STORE SIXBIT FILENAME
>;END IFE TOPS20
IFN TOPS20,<
GTAD ;GET SOME FAIRLY UNIQUE DIGITS
HRRZ T1,T1 ;JUST SAVE RH
SETZ T3,
MOVEI T4,6 ;WANT 6 DIGITS
MAKEJN: IDIVI T1,^D10
MOVEI T2,'0'(T2)
LSHC T2,-6
SOJG T4,MAKEJN
MOVEM T3,MRNM6
>;END IFN TOPS20
; NOW SIXBIT NAME IS IN MRNM6
; MAKE AN ASCII NAME OUT OF IT
MOVE T2,[POINT 6,MRNM6]
MOVE T3,[POINT 7,MRNMA]
MRLST1: ILDB T4,T2
JUMPE T4,MRLST2
ADDI T4,40 ;CONVERT SIXBIT CHAR TO ASCII CHAR
IDPB T4,T3
TLNE T2,760000 ;SKIP IF DONE 6 CHARS
JRST MRLST1
MRLST2:
IFN TOPS20, MOVE 2,[POINT 7,[ASCIZ/.TMP;T/]] ;235123.TMP;T
IFE TOPS20, MOVE 2,[POINT 7,[ASCIZ/.TMP/]] ;WRITE TO NNNMTO.TMP
ILDB T4,T2
IDPB T4,T3
JUMPN T4,.-2
;NOW OUTPUT FILE NAME HAS BEEN OBTAINED. IT IS STORED
;IN SIXBIT IN MRNM6, IN ASCII IN MRNMA.
;
;NOW OPEN THE OUTPUT METER FILE
IFE TOPS20,<
PUSHJ PP,GMCHAN ;GET A FREE CHANNEL TO USE
JRST [OUTSTR [ASCIZ/?NO FREE CHANNELS TO WRITE METER FILE
/]
JRST KILL.] ;THIS SHOULD NEVER HAPPEN!!!
ANDI T5,17 ;T5:= CHANNEL NUMBER
DPB T5,[POINT 4,T5,12] ;SAVE IN AC FIELD OF T5
HLLZ T5,T5 ;FOR MAKING UUOS
MOVEM T5,MRCHNN ;SAVE CHANNEL NUMBER FOR CLOSE LATER
;DO OPEN UUO
MOVEI T1,.IODMP ;BINARY DUMP MODE
MOVSI T2,'DSK' ; TO DEVICE "DSK"
SETZ T3, ;NO BUFFER HEADERS
MOVE T0,[OPEN T1]
OR T0,T5 ;READY TO DO IT
XCT T0
JRST MROPNF ;OPEN FAILED!
SUBTTL LSTATS SETUP - ENTER METER FILE
;DO ENTER UUO
MRLKPF: MOVE T1,MRNM6 ;FILENAME
MOVSI T2,'TMP' ;.TMP
SETZB T3,T4
MOVE T0,[ENTER T1]
OR T0,T5
XCT T0
JRST MRENTF ;ENTER FAILED!
JRST MRENOK ;OK
MRENTF: OUTSTR [ASCIZ/? ENTER FAILED FOR METER FILE: /]
MRTYPF: OUTSTR MRNMA
OUTSTR [ASCIZ/
/]
SETZM MROPT. ;CLEAR BUCKET POINTER, SO WE DON'T
; TRY WRITING ANYMORE
JRST KILL. ;FATAL ERROR IF THIS HAPPENS
MROPNF: OUTSTR [ASCIZ/? OPEN FAILED FOR METER FILE: /]
JRST MRTYPF ;TYPE FILE NAME (EVEN THOUGH THIS
; ERROR IS INDEPENDENT OF THE FILE NAME)
>;END IFE TOPS20
IFN TOPS20,<
MOVX 1,GJ%SHT
HRROI 2,MRNMA
GTJFN
ERJMP MRJSER ;COULDN'T OPEN THE FILE FOR OUTPUT
MOVEM 1,MRJFN ;REMEMBER JFN
MOVX T2,OF%WR
OPENF
ERJMP MRJSER
JRST MRENOK
MRJSER: HRROI 1,[ASCIZ/?JSYS ERROR: /]
PSOUT
MOVEI 1,.PRIOU
HRLOI 2,.FHSLF
SETZ 3,
ERSTR
JFCL
JFCL
HRROI 1,[ASCIZ/ FOR METER FILE /]
PSOUT
MRTYPF: HRROI 1,MRNMA
PSOUT
HRROI 1,[ASCIZ/
/]
PSOUT
JRST KILL. ;FATAL ERROR--DIE OFF
;HERE FOR NON I/O JSYS ERRORS
MRJSSE: HRROI 1,[ASCIZ/?JSYS ERROR: /]
PSOUT
PUSHJ P,TYPFER ;TYPE LAST ERROR IN THIS FORK
JRST KILL. ;FATAL ERROR--DIE OFF
;ROUTINE TO TYPE LAST ERROR IN THIS FORK THEN CRLF
TYPFER: MOVEI T1,.PRIOU
HRLOI T2,.FHSLF
SETZ T3,
ERSTR
JFCL
JFCL
HRROI T1,[ASCIZ/
/]
PSOUT
POPJ PP,
>;END IFN TOPS20
SUBTTL LSTATS SETUP - SETUP CORE
;METER FILE HAS NOW BEEN SUCCESSFULLY OPENED FOR OUTPUT.
;CALL FUNCT. TO GET CORE AT PAGE BOUNDARY
; FOR THE BUCKETS, AND STORE ADDRESS IN MROPT.
MRENOK: MOVEI 16,1+[-5,,0
XWD 0,FUN.A0##
XWD 0,[ASCIZ/LBL/]
XWD 0,FUN.ST##
XWD 0,FUN.A1##
XWD 0,FUN.A2##]
F.PAG==15
MOVEI T1,F.PAG ;FUNCTION WE WANT
MOVEM T1,FUN.A0## ;STORE FUNCTION
SETZM FUN.ST## ;CLEAR STATUS
SETZM FUN.A1## ; AND ADDRESS RETURNED
MOVEI T1,.MBFSZ ;NUMBER OF WORDS TO ALLOCATE
IMULI T1,^D16
ADDI T1,.MBHSZ+.MBTSZ ;ACCOUNT FOR HEADER BLOCK & TRAILER BLOCK
ADDI T1,MBHISL ;CORE FOR FILE/PAGE TABLE
IFN DBMS6, ADDI T1,.MBDSZ ;AND DBMS BLOCK
ADDI T1,MR.BFB ;200 OR MORE WORDS FOR LSTATS.DIR BLOCK
MOVEM T1,FUN.A2## ;STORE AS ARG #2
PUSHJ PP,FUNCT.## ;CALL FUNCT. ROUTINE...
SKIPE FUN.ST## ; STATUS MUST BE 0...
JRST MRNCR ; ? NOPE - NO CORE AVAIL
HRRZ T1,FUN.A1## ;GOT IT -- GET ADDRESS OF START
MOVEM T1,MROPT. ;STORE IN MROPT.
;SETUP THE TABLE OF BUCKET ADDRESSES INDEXED BY CHANNEL NUMBER.
SETZ T2, ;T2=BUCKET BLOCK NUMBER
MRSBA: MOVEM T1,MROPTT(T2) ;STORE ADDRESS OF THIS BLOCK
ADDI T1,.MBFSZ ;NEXT BUCKET BLOCK ADDRESS
CAIGE T2,17 ;DONE ALL?
AOJA T2,MRSBA ;NO, LOOP
MOVEM T1,MRHDBP ; STORE POINTER TO HEADER BLOCK
ADDI T1,.MBHSZ
MOVEM T1,MRTDBP ; STORE POINTER TO TRAILER BLOCK
ADDI T1,.MBTSZ ;TRAILER SIZE
MOVEM T1,MRFPGT ;STORE POINTER TO FILE/PAGE # TABLE
..NXAD==MBHISL
IFN DBMS6,<
..NXAD==.MBDSZ
ADDI T1,MBHISL
MOVEM T1,MRDDBP ;STORE POINTER TO DBMS BLOCK
>
ADDI T1,..NXAD
MOVEM T1,MRLDBL ;STORE POINTER TO LSTATS.DIR BLOCK
SUBTTL LSTATS SETUP - SETUP HEADER BLOCK
;STORE HEADER BLOCK INFORMATION
DEFINE STOR1H(OFFSET),<
HRRZ T2,MRHDBP ;GET OFFSET
MOVEM T1,OFFSET(T2) ;STORE THE WORD
>
MOVEI T1,2 ;VERSION NUMBER OF LSTATS STUFF
STOR1H (MB.VNO) ; STORE IT
;HEADER BLOCK LENGTH IN WORDS
MOVEI T1,.MBHLN
STOR1H (MB.HLN)
;PER-FILE BLOCK LENGTH IN WORDS
MOVEI T1,.MBFLN ;ACTUAL SIZE OF THE BLOCK
STOR1H (MB.FLN)
;MONITOR TYPE (%CNMNT WORD ON TOPS10, BIT 21=1 ON TOPS20)
IFN TOPS20,<
MOVEI T2,4B23 ;MONITOR TYPE 4 = TOPS20
>
IFE TOPS20,<
MOVE T2,[%CNMNT]
GETTAB T2,
MOVEI T2,1B23 ;ASSUME TOPS10
>
;NOW FIND PROCESSOR TYPE. THIS ALGORITHM IS TAKEN
;DIRECTORY FROM THE HARDWARE REFERENCE MANUAL.
MOVNI T1,1 ;MAKE AC ALL 1'S
AOBJN T1,.+1 ;INCREMENT BOTH HALVES
JUMPN T1,MRKA10 ;KA10 IF AC= 1000000 (CARRY BETWEEN HALVES)
BLT T1,0
JUMPN T1,MRKL10 ;KL10 IF AC=1,,1
MRKI10: MOVEI T1,2 ;KI10 - SET PROCESSOR TYPE TO 2
JRST MRSTRP
MRKA10: MOVEI T1,1 ;KA10 - SET PROCESSOR TYPE TO 1
JRST MRSTRP
MRKL10: MOVEI T1,3 ;KL10 - SET PROCESSOR TYPE TO 3
MRSTRP: DPB T1,[POINT 3,T2,17] ;STORE IN 7B17
MOVE T1,T2 ;COPY PROCESSOR/OPERATING SYSTEM INFO TO T1
STOR1H (MB.MON) ;STORE IT IN HEADER WORD
;LIBOL VERSION NUMBER
MOVE T1,LIBVR.
STOR1H (MB.LBV)
;LIBOL SWITCH WORD
MOVE T1,LIBSW.
STOR1H (MB.LSW)
;PROGRAM NAME
IFE TOPS20,<
HRROI T1,-3
GETTAB T1,
MOVE T1,['METER']
>
IFN TOPS20,<
GETNM
SKIPN T1 ;IF NO NAME,
MOVE T1,['METER'] ;USE METER
>
STOR1H (MB.PNM) ;STORE IT
;CPU SERIAL NUMBER
IFN TOPS20,<
MOVEI T1,.APRID ;LH=0 (TABLE OFFSET), RH=.APRID TABLE
GETAB ; GET THE WORD
SETO T1, ;SET TO -1 IF CAN'T GET IT
STOR1H (MB.SER)
>
IFE TOPS20,<
MOVE T1,[%CNSER]
GETTAB T1,
SETO T1,
STOR1H (MB.SER)
>
;MONITOR VERSION NUMBER
IFE TOPS20,<
MOVE T1,[%CNVER]
GETTAB T1,
SETZ T1, ;SET TO ZEROES IF CAN'T GET
STOR1H (MB.MVR)
>
;DATE/TIME PROGRAM STARTED
IFN TOPS20,<
GTAD
STOR1H (MB.DTM)
>
IFE TOPS20,<
MOVE T1,[%CNDTM]
GETTAB T1,
SETO T1, ;COMPATIBLE WITH TOPS20
STOR1H (MB.DTM)
>
;SYSTEM NAME IN ASCIZ
HRRZ T5,MRHDBP ;GET POINTER TO HEADER BLOCK
ADDI T5,MB.SYN ;RH (T5) = PTR TO SYSTEM NAME
HRLI T5,(POINT 7,) ;GET ASCIZ BYTE PTR TO IT
IFN TOPS20,<
HRLZI T4,-32 ;MAX SIZE
MRVRS1: HRRI T1,.SYSVE ;SYSTEM VERSION TEXT
HRL T1,T4 ;WORD NUMBER
GETAB
SETZ T1, ;CLEAR IF CAN'T GET IT
MOVE T2,T1
MOVEI T3,5 ;5 CHARS/WORD
MRVRS2: SETZ T1,
LSHC T1,7
JUMPE T1,MRVRS3 ;END OF STRING
IDPB T1,T5 ;STORE BYTE
SOJG T3,MRVRS2 ;LOOP FOR CHARS IN THIS WORD
AOBJN T4,MRVRS1 ;LOOP FOR MORE WORDS
>;END IFN TOPS20
IFE TOPS20,<
HRLZI T4,-5 ;MAX SIZE
MRVRS1: HRRI T1,.GTCNF ;FROM CONFIG TABLE
HRL T1,T4 ;WORD NUMBER
GETTAB T1,
SETZ T1, ;CLEAR IF CAN'T GET IT
MOVE T2,T1
MOVEI T3,5 ;5 CHARS/WORD
MRVRS2: SETZ T1,
LSHC T1,7 ;GET NEXT CHAR
JUMPE T1,MRVRS3 ;END OF STRING
IDPB T1,T5 ;STORE BYTE
SOJG T3,MRVRS2 ;LOOP FOR CHARS IN THIS WORD
AOBJN T4,MRVRS1 ;LOOP FOR MORE WORDS
>;END IFE TOPS20
MRVRS3:
;BATCH JOB INDICATOR
IFE TOPS20,<
HRROI T1,.GTLIM ;.GTLIM WORD FOR MY JOB
GETTAB T1,
SETZ T1, ;ASSUME NOT A BATCH JOB
TXNN T1,JB.LBT ;BATCH JOB?
TDZA T1,T1 ;NO
SETO T1, ;YES
STOR1H (MB.BAT)
>
IFN TOPS20,<
SETO T1, ;GET INFORMATION FOR CURRENT JOB
HRROI T2,T4 ;ONE WORD TABLE, IN T4
MOVEI T3,.JIBAT ;FIRST (AND ONLY) ENTRY
GETJI ; GET INFO IN T4
ERJMP MRJSSE
MOVE T1,T4 ;GET THE WORD
STOR1H (MB.BAT)
>
;SYSTEM STATUS BITS
IFE TOPS20,<
MOVE T1,[%CNSTS]
GETTAB T1,
SETZ T1, ;ZERO IF CAN'T GET
STOR1H (MB.STS)
>
;SOFTWARE CONFIGURATION INDICATORS
IFE TOPS20,<
MOVE T1,[%CNST2]
GETTAB T1,
SETZ T1,
STOR1H (MB.ST2)
>
;AVERAGE OVERHEAD TIME
; DO IT OHTMS TIMES AND DIVIDE BY OHTMS
OHTMS==^D10 ;NUMBER OF TIMES TO DO IT TO GET THE
; AVERAGE
MOVEI T2,MRFKFT-D.CN ; FAKE FILE TABLE WORD
MOVEM T2,MRAFFT ;SAVE THE FAKE FILE TABLE ADDRESS
SETOM MRFKFT ;(CHANNEL 17 WILL BE RETURNED)
XLIST ;CALL MACROS OHTMS TIMES
REPEAT OHTMS,<
MRTMS. (T1) ;START
MOVEI T2,400+MB.OVH ;BUCKET OFFSET
MOVE T1,MRAFFT ;GET FAKE FILE TABLE ADDRESS
PUSHJ PP,LMETR. ;CALCULATE BUCKET ADDRESS
MRTME. (T1) ;STOP
>
LIST
MOVE T1,@MRTMB. ;GET TOTAL TIME
IDIVI T1,OHTMS ; DIVIDE BY # TIMES DONE
STOR1H (MB.OVH) ;STORE IN HEADER BLOCK
IFN TOPS20,<
SETZ T0, ;CLEAR FOR SHIFT
ASHC T0,^D12 ;SHIFT TIME LEAVING 12 0'S FOR
;JSYS 776 FORMAT
DMOVEM T0,MRBKO. ;SAVE FIXED OVERHEAD TIME
>;END IFN TOPS20
IFE TOPS20,<
MOVEM T1,MRBKO. ;SAVE FIXED OVERHEAD TIME
>
POPJ PP, ;ALL DONE, RETURN
MRNCR:
IFE TOPS20,<
OUTSTR [ASCIZ/? Not enough core available for METERing
/]
>
IFN TOPS20,<
HRROI T1,[ASCIZ/? Not enough core available for METERing
/]
PSOUT
>
JRST KILL. ;SORRY ABOUT THAT, CHIEF.
SUBTTL LSTATS - ROUTINE TO DUMP A FILE BLOCK
;ROUTINE MRDMP IS CALLED BEFORE EVERY OPEN TO WRITE THE
; DATA COLLECTED ON THE FILE PREVIOUSLY OPEN ON THIS CHANNEL.
; IF THERE WASN'T ANY FILE PREVIOUSLY OPEN ON THIS CHANNEL,
; THERE IS NO WORK TO BE DONE.
;
; CALLED WITH T1/ CHANNEL NUMBER (0 THRU 17)
MRDMP: MOVE T4,MROPTT(T1) ; T4 POINTS TO FIRST LOCATION
SKIPN MB.BAS+0(T4) ;SKIP IF ANY OLD INFO THERE
POPJ PP, ;NOTHING TO DO, RETURN
;WRITE OUT THIS CHUNK, AND THEN ZERO IT
PUSHJ PP,MRWONE ;CALL ROUTINE TO WRITE ONE
;NOW ZERO OUT THE BLOCK
SETZM (T4) ;CLEAR FIRST WORD
HRL T4,T4
MOVEI T3,.MBFSZ-1(T4)
ADDI T4,1
BLT T4,(T3)
POPJ PP, ;DONE, RETURN
;ROUTINE MRDMPT IS CALLED AT THE "STOP RUN" TO WRITE
; THE DATA COLLECTED ON ALL FILES WHICH WE HAVEN'T WRITTEN
; OUT YET. MOST OF THE I/O SHOULD BE DONE HERE.
MRDMPT: SKIPN MROPT. ;BUCKETS SETUP?
POPJ PP, ;NO, RETURN AND DO NOTHING
MOVEI T5,17 ;T5=CHANNEL NUMBER TO DO
MRDMT1: MOVE T4,MROPTT(T5) ;GET ADDRESS
SKIPE MB.BAS+0(T4) ;SKIP IF NO INFO TO WRITE
PUSHJ PP,MRWONE ;WRITE ONE
SOJGE T5,MRDMT1 ;LOOP FOR ALL CHANNELS
;IF WE HAVEN'T WRITTEN A HEADER YET, CHECK FOR ACCEPT/DISPLAY
; DATA IN THE TRAILER.. IF ANY WAS RECORDED, WRITE THE HEADER.
SKIPE MRHDFL ;HEADER WRITTEN?
JRST MRWTRL ; YES, WRITE TRAILER NOW
HRRZ T2,MRTDBP ;GET POINTER TO TRAILER
SKIPN MB.ACP(T2) ;ACCEPT DATA?
SKIPE MB.DSP(T2) ; OR DISPLAY DATA?
TRNA ;YES
JRST MRCLOS ;NO, DON'T WRITE HEADER OR TRAILER THEN
;SKIP INTO "WRITE TRAILER" ROUTINE
SUBTTL LSTATS - TRAILER BLOCK SETUP AND WRITE ROUTINE
; WRITE TRAILER BLOCK
DEFINE STOR1T(OFFSET),<
HRRZ T2,MRTDBP ;GET OFFSET
MOVEM T1,OFFSET(T2) ;STORE THE WORD
>
DEFINE STOR2T(OFFSET),<
HRRZ T3,MRTDBP ;GET OFFSET
DMOVEM T1,OFFSET(T3) ;STORE THE WORD
>
MRWTRL: MOVE T1,[MBBT.T,,.MBTSZ] ;TRAILER BLOCK TYPE,,LENGTH
MOVEM T1,@MRTDBP ;STORE IN THE BLOCK
;SIZE OF HISTOGRAM BLOCK
MOVEI T1,MBHISL ;LENGTH OF HISTOGRAM BLOCK
STOR1T (MB.HSL)
;DATE/TIME PROGRAM ENDED
IFN TOPS20,<
GTAD
STOR1T (MB.DTF)
>
IFE TOPS20,<
MOVE T1,[%CNDTM]
GETTAB T1,
SETO T1, ;COMPATIBLE WITH TOPS20
STOR1H (MB.DTF)
>
;KILL FLAG (-1 IF PROGRAM WAS ABORTED, 0 FOR NORMAL TERMINATION)
MOVE T1,MRKILL
STOR1T (MB.KIL)
;TOTAL RUNTIME OF PROGRAM (TICKS)
; ON TOPS20 THIS IS STORED AS A DOUBLE-WORD NUMBER OF TICKS
; ON TOPS10 IT IS JUST ONE WORD.
IFN TOPS20,<
JSYS 776 ;GET TICKS NOW IN T1,T2
ERJMP MRWTRX ;DO NOTHING IF NO CLOCK
DSUB T1,MRPSTM ;SUBTRACT STARTING TICKS
ASHC T1,-^D12 ;SHIFT ZEROES OUT
STOR2T (MB.TRT) ;STORE DOUBLE WORD QUANTITY
MRWTRX:>
IFE TOPS20,<
SETZ T1,
RUNTIME T1,
SUB T1,MRPSTM ;SUBTRACT STARTING TICKS
STOR1T (MB.TRT)
>
; NOW WRITE IT
HRRZ T1,MRTDBP ;START ADDRESS
HRLI T1,-.MBTSZ ;-SIZE
PUSHJ PP,MROUT. ;OUTPUT THE BLOCK
JRST MRCLOS ;OK--GO CLOSE FILE
SUBTTL LSTATS - DBMS SUPPORT ROUTINES
IFN DBMS6,<
;ROUTINES FOR DBMS V6 TO WRITE IN LSTATS FILE
;ALL ROUTINES MUST SAVE ALL ACS EXCEPT T0
;START TIMING, BUCKET NUMBER IN T0
MRDBST: MRTMS. (T0) ;START METER TIMING
POPJ PP,
;STOP TIMING, BUCKET NUMBER IN T0
MRDBEN: ADD T0,MRDDBP ;ADD DBMS BLOCK ADDRESS
MOVEM T0,MRTMB. ;SAVE TIMING BUCKET ADDRESS
MRTME. (T0) ;END METER TIMING
POPJ PP,
;DUMP AND CLEAR DBMS BLOCK
MRDBDM: PUSH PP,T1 ;SAVE ACS WE WILL USE
PUSH PP,T2
PUSH PP,T3
MOVE T1,[MBBT.D,,.MBDSZ]; BLOCK TYPE,, LENGTH
MOVEM T1,@MRDDBP ;STORE HEADER WORD
HRRZ T1,MRDDBP ;GET POINTER TO BLOCK
PUSHJ PP,MROUT. ;WRITE IT
;CLEAR THE BLOCK
HRR T1,MRDDBP ;MAKE T1 = BLT POINTER
MOVEI T2,.MBDSZ(T1) ;LAST LOCATION+1
HRL T1,T1
ADDI T1,1
SETZM @MRDDBP ;CLEAR FIRST LOC
BLT T1,-1(T2) ;FINISH CLEARING BLOCK
POP PP,T3 ;RESTORE SAVED ACS
POP PP,T2
POP PP,T1
POPJ PP,
>;END IFN DBMS6
SUBTTL LSTATS - ROUTINE TO OUTPUT ANY LSTATS BLOCK
;ROUTINE TO OUTPUT AN LSTATS BLOCK
;CALL: MOVE T1,[-LENGTH,,ADDRESS]
; PUSHJ PP,MROUT.
; USES T0,T1,T2,T3
; <RETURNS HERE>
MROUT.: SKIPN MRRERN ;SKIP IF RERUNNING AND WE CAN'T DO I/O
;TO MTO FILE
SKIPN MROPT. ;SKIP IF BUCKETS ARE SETUP
POPJ PP, ; CAN'T DO I/O - RETURN AND DO NOTHING
PUSH PP,T1 ;SAVE THE CALLING ARG
SKIPN MRHDFL ;HAVE WE WRITTEN THE HEADER YET?
PUSHJ PP,MRWRTH ;NO, WRITE IT NOW
HRRZ T1,(PP) ;GET ADDRESS FROM STACK
HLRZ T1,(T1) ;GET BLOCK TYPE
CAIN T1,MBBT.F ;A FILE BLOCK?
JRST MROUF ;YES
CAIN T1,MBBT.D ;A DBMS BLOCK?
JRST MROUD ;YES
CAIN T1,MBBT.S ;A SORT BLOCK?
JRST MROUS ;YES
MROUB:
IFN TOPS20,<
POP PP,T2 ;GET ADDRESS BACK
HLRE T3,T2 ;SIZE IN T3
HRLI T2,(POINT 36,)
MOVE T1,MRJFN ;JFN IN T1
SOUT ;WRITE IT
POPJ PP, ;RETURN
>
IFE TOPS20,<
POP PP,T1 ;GET ARG BACK
SOJA T1,MRWRT1 ;FORM IOWD AND OUTPUT BLOCK
>;END IFE TOPS20
MROUF: HRRZ T1,MRTDBP ;POINT TO TRAILER BLOCK
AOS T2,MB.NFB(T1) ; ANOTHER FILE BLOCK, HOW MANY SO FAR?
CAILE T2,^D100 ;IF .GT. 100, DON'T WRITE ANOTHER ONE.
JRST [POP PP,(PP) ;THROW AWAY SAVED VALUE
POPJ PP, ];AND RETURN
JRST MROUB ;GO OUTPUT IT
MROUS: HRRZ T1,MRTDBP ;POINT TO TRAILER BLOCK
AOS T2,MB.NSB(T1) ; ANOTHER SORT BLOCK, HOW MANY SO FAR?
CAILE T2,^D100 ;IF .GT. 100, DON'T WRITE ANOTHER ONE.
POPJ PP,
JRST MROUB ;GO OUTPUT IT
MROUD: HRRZ T1,MRTDBP ;COUNT TOTAL DBMS BLOCKS
AOS MB.NDB(T1)
JRST MROUB ;AND GO OUTPUT IT
SUBTTL LSTATS - FINISH WRITING METER DATA
;ALL DONE, CLOSE THE METER OUTPUT FILE
MRCLOS: SKIPE MRRERN ;IF RERUN WAS USED, DON'T CLOSE
JRST MRRERC ;BECAUSE FILE IS NOT OPEN!!
IFE TOPS20,<
;DO CLOSE UUO
MOVSI T0,(CLOSE 0,)
OR T0,MRCHNN ;OR IN CHANNEL NUMBER
XCT T0
>;END IFE TOPS20
IFN TOPS20,<
MOVE T1,MRJFN ;GET JFN OF METER FILE
TXO T1,CO%NRJ ;DON'T RELEASE JFN
CLOSF
ERJMP MRJSER
>;END IFN TOPS20
SKIPN MRHDFL ;DID WE WRITE ANYTHING?
JRST MRNOWR ;NO
REPEAT 0,<
; MOST USERS WILL NOT WANT THIS PRINTED OUT.
; IN ANY CASE, IT SHOULDN'T BE PRINTED OUT UNLESS THE DIRECTORY
;FILE LSTATS.DIR IS NOT BEING USED.
OUTSTR [ASCIZ/[LIBOL: METER file (/]
OUTSTR MRNMA
SKIPN MRAPN. ;DID WE APPEND?
JRST [OUTSTR [ASCIZ/) written]
/]
POPJ PP,] ;DONE
OUTSTR [ASCIZ/) appended]
/]
>;END REPEAT 0
IFN FTLSDR,<
IFE TOPS20, JRST GOLSPN ;APPEND TO THE APPROPRIATE FILE
IFN TOPS20, JRST GOLSDR ;APPEND TO THE APPROPRIATE FILE
>;END IFN FTLSDR
IFE FTLSDR,< ;NO LSTATS.DIR
IFE TOPS20, JRST DOAP10 ;GO APPEND TO JOBNAME.MTO
IFN TOPS20, JRST DOAP20 ;. .
>;END IFE FTLSDR
MRNOWR:; OUTSTR [ASCIZ .[LIBOL: No METER data was recorded, because no I/O was done]
;.]
POPJ PP, ;ALL DONE, RETURN
MRRERC: ;OUTSTR [ASCIZ .[LIBOL: No METER data was recorded, because
;program was RERUN]
;.]
POPJ PP, ;RETURN
SUBTTL LSTATS - ROUTINE TO WRITE ONE FILE BLOCK
;MRWONE WRITES OUT ONE CHUNK OF BUCKETS, USING THE
; START ADDRESS IN T4.
MRWONE: MOVE T1,[MBBT.F,,.MBFSZ]; BLOCK TYPE,, LENGTH
MOVEM T1,(T4) ;STORE HEADER WORD
HRRZ T1,T4 ;GET ADDRESS
HRLI T1,-.MBFSZ ;-SIZE IN LH
JRST MROUT. ;WRITE THE BLOCK AND RETURN
SUBTTL LSTATS - ROUTINE TO WRITE THE HEADER
; ROUTINE TO WRITE THE HEADER
;CALL: PUSHJ PP,MRWRTH
; RETURN HERE
;USES T0,T1,T2, T3
MRWRTH: SETOM MRHDFL ;REMEMBER WE WROTE THE HEADER
MOVE T1,[MBBT.H,,.MBHSZ] ;HEADER BLOCK TYPE,,LENGTH
MOVEM T1,@MRHDBP ;STORE IN THE BLOCK
SKIPE MRRERN ;IS THIS A 'RERUN' OF THE PROGRAM?
POPJ PP, ;YES, DON'T DO I/O TO MTO FILE
IFN TOPS20,<
MOVE T1,MRJFN
HRLI T2,(POINT 36,)
HRR T2,MRHDBP ;START ADDRESS
MOVNI T3,.MBHSZ ; 128 WORDS
SOUT ;WRITE IT
POPJ PP, ;RETURN
>
IFE TOPS20,<
HRR T1,MRHDBP ;GET POINTER
SUBI T1,1 ;ADDRESS-1
HRLI T1,-.MBHSZ ;-NO. WORDS
MRWRT1: SETZ T2,
MOVE T0,[OUT T1]
OR T0,MRCHNN ;LOAD UUO WITH CHANNEL NUMBER
XCT T0
POPJ PP, ;WORKED OK
OUTSTR [ASCIZ/? OUT UUO FAILED FOR METER FILE /]
JRST MRTYPF ;TYPE FILE NAME AND DIE OFF
>;END IFE TOPS20
;; ** END OF MRWRTH ROUTINE **
SUBTTL LSTATS - TOPS10 DIRECTORY FILE ROUTINE
;ALWAYS RETURNS .+1. IF SOME FATAL ERROR HAPPENS, TYPES
; A WARNING AND RETURN .+1 ALSO.
IFE TOPS20,<
IFN FTLSDR,<
GOLSPN:
;NOW LOOKUP THE ONE WE WROTE TO FOR READ NOW
;IF ANY ERRORS HAPPEN NOW, THE .TMP FILE WILL GET DELETED
; BY SIMPLY DOING A RENAME TO 0.
MOVE T1,MRNM6
MOVSI T2,'TMP'
SETZB T3,T4
MOVE T5,[LOOKUP T1]
OR T5,MRCHNN
XCT T5 ;LOOKUP
JRST LLKPF3 ;CAN'T!!
;TRY TO FIND LSTATS.DIR
PUSHJ PP,GMCHAN ;GET A FREE CHANNEL TO USE
JRST NFC1 ;CAN'T
ANDI T5,17
DPB T5,[POINT 4,T5,12] ;SAVE IN AC FIELD OF T5
HLLZM T5,MRCHLS ;SAVE CHANNEL
;AND WHILE WE'RE AT IT.. GET CHANNEL FOR "COMMON" FILE TOO
PUSHJ PP,GMCHAN
JRST NFC1 ;CAN'T
ANDI T5,17
DPB T5,[POINT 4,T5,12]
HLLZM T5,MRCHCF ;CHANNEL NUMBER IN AC FIELD
IFE LSPPN,<
MOVSI T1,'COB' ;FIND PPN FOR 'COB'
DEVPPN T1,
MOVE T1,[XWD 5,2] ;DEFAULT
>
IFN LSPPN, MOVE T1,[LSPPN] ;PPN WAS SPECIFIED
MOVEM T1,MRLLDR ;SAVE PPN WHERE LSTATS.DIR IS
;SEARCH THRU SYSTEM SEARCH LIST, AND TRY A LOOKUP ON EACH DEVICE
SETO T4, ;START AT THE BEGINNING
NXTGBS: MOVE T1,[3,,T2] ;SET UP FOR GOBSTR
SETZB T2,T3 ;CLEAR JOB NUMBER, PPN ARGS
GOBSTR T1,
JRST GOBSTE ;?FAILED!
JUMPE T4,NXTGBS ;IGNORE "FENCE"
CAMN T4,[-1] ;NOT FOUND?
JRST NOLSTS ;YES
MOVEM T4,MRLLDV ;SAVE DEVICE NAME
;OPEN THE DEVICE
MOVEI T1,.IODMP ;DUMP MODE
MOVE T2,T4 ;THAT DEVICE
SETZ T3, ;NO BUFFERS
MOVE T4,[OPEN T1] ; PLAIN OPEN UUO
OR T4,MRCHLS ;"OR" IN CHANNEL
XCT T4 ;DO OPEN
JRST LOPNF1 ;OPEN FAILED
;LOOKUP UUO
MOVE T1,[SIXBIT/LSTATS/]
MOVSI T2,'DIR'
SETZ T3,
MOVE T4,MRLLDR ;PPN
MOVE T5,[LOOKUP T1]
OR T5,MRCHLS
XCT T5 ;DO LOOKUP
CAIA ;FAILED - GO CHECK
JRST RD1200 ;FOUND IT
HRRZ T2,T2 ;GET LOOKUP ERROR CODE
CAIE T2,1 ;NO DIRECTORY, OR "NOT FOUND"?
JUMPN T1,LLKPE1 ;NO--OTHER LOOKUP ERROR, COMPLAIN
MOVSI T1,(RELEAS) ;RELEAS THE CHANNEL
OR T1,MRCHLS
XCT T1
MOVE T4,MRLLDV ; GET LAST DEVICE NAME WE TRIED
JRST NXTGBS ;AND DO ANOTHER GOBSTR
;READ IN FIRST 200 WORDS .. AND COPY INFORMATION
RD1200: HRRZ T1,MRLDBL ;POINTER TO BLOCK
SUBI T1,1
HRLI T1,-200 ;FORM IOWD
SETZ T2,
MOVE T3,[IN T1]
OR T3,MRCHLS
XCT T3
CAIA ;OK
JRST LSINF ;PROBLEMS
SETZM MRLFPR ;FILE IS AT BEGINNING NOW
MOVE T4,MRLDBL ;T4 = PTR TO THE BLOCK
MOVE T1,.LDBPC(T4) ;BYTE PTR TO CURRENT FILENAME
MOVEM T1,MRLBPC
MOVE T1,.LDSZL(T4) ;SIZE LIMIT
MOVEM T1,MRLSZL ;SAVE
SKIPN .LDTML(T4) ;ANY TIME LIMIT?
JRST TMLMOK ;NO
;CHECK TIME LIMIT
SKIPN T3,.LDFWR(T4) ;GET TIME OF FIRST WRITE
JRST SETTFW ;NONE THERE--SET TO NOW
ADD T3,.LDTML(T4) ;ADD TIME LIMIT
MOVX T1,%CNDTM ;GET TODAY'S DATE/TIME
GETTAB T1,
SETO T1,
CAML T1,T3 ;IS IT TIME TO INCREMENT?
JRST NXTFLE ;YES!
;TIME LIMIT IS OK -- OPEN FILE FOR APPEND AND CHECK SIZE LIMIT
TMLMOK: MOVE T4,MRLBPC ;BYTE PTR TO CURRENT FILE
PUSHJ PP,LDPOST ;POSITION LD FILE
JRST LRLLC0 ;ERROR - RELEASE CHANNEL AND QUIT
ADD T4,MRLDBL ; NOW GET ACTUAL BYTE PTR TO THE FILE
SUB T4,MRLFPR
;READ FILENAME AND PUT SIXBIT IN MRCFNM,MRCFEX
MOVE T3,T4 ;COPY PTR INCASE ERROR
MOVE T1,[POINT 6,MRCFNM]
TMLMO1: ILDB T2,T3 ;GET ASCII CHARACTER
JUMPE T2,ENDFN1 ;NULL ENDS STRING
CAIN T2,"." ;. STARTS EXTENSION
JRST ENDFN2
SUBI T2,40 ;MAKE SIXBIT CHARACTER
IDPB T2,T1 ;STORE
JRST TMLMO1 ;LOOP
ENDFN2: MOVE T1,[POINT 6,MRCFEX]
TMLMO2: ILDB T2,T3 ;GET ASCII CHARACTER
JUMPE T2,ENDFN1 ;NULL ENDS STRING
SUBI T2,40 ;ELSE STORE CHARACTER
IDPB T2,T1
JRST TMLMO2
ENDFN1: MOVEI T1,.IODMP ;OPEN OLD DEVICE IN DUMP MODE
MOVE T2,MRLLDV
SETZ T3,
MOVE T4,[OPEN T1]
OR T4,MRCHCF ;COMMON FILE CHANNEL
XCT T4
JRST LOPNFC ;OPEN FAILED FOR COMMON FILE
;TO APPEND: DO LOOKUP. IF NO ERROR OR ERROR CODE 0,
; DO ENTER AND USETO.
MOVE T1,MRCFNM ;FILENAME
MOVE T2,MRCFEX ;EXTENSION
SETZ T3,
MOVE T4,MRLLDR ;DIRECTORY
MOVE T5,[LOOKUP T1]
OR T5,MRCHCF
XCT T5 ;DO LOOKUP
JRST LLKPEC ;LOOKUP FAILED FOR COMMON FILE
;CHECK SIZE LIMIT
HLRE T1,T4 ;GET SIZE
JUMPL T1,[MOVM T1,T1 ;SOME # OF WORDS-- ROUND UP TO BLOCKS
IDIVI T1,200
JUMPE T2,.+1
AOJA T2,.+1]
MOVEI AC11,1(T1) ;GET BLOCK # TO USETO TO
SKIPN MRLSZL ;SKIP IF A SIZE LIMIT TO WORRY ABOUT
JRST DOENT ;NONE
CAML T1,MRLSZL ;SIZE TOO BIG?
JRST SIZTBG ;YES: GO ON TO NEXT FILE
DOENT: MOVSI T1,(RELEAS) ;CLOSE & RELEAS LSTATS.DIR
OR T1,MRCHLS
XCT T1
MOVEI T0,^D10 ;# TIMES TO TRY ENTER
DOENTA: MOVE T1,MRCFNM ;NAME
MOVE T2,MRCFEX ;EXTENSION
;NOTE: T3 HAS PROTECTION CODE RETURNED FROM LOOKUP
AND T3,[777000,,0] ;JUST SAVE PROTECTION CODE
MOVE T4,MRLLDR ;DIRECTORY
MOVE T5,[ENTER T1]
OR T5,MRCHCF
XCT T5 ;DO ENTER
JRST LENTEC ;ENTER FAILED FOR COMMON FILE
;DO USETO TO APPEND TO THE FILE
CAIN AC11,1 ;IF BLOCK 1, DON'T BOTHER
JRST NXTRDD
MOVSI T5,(USETO)
OR T5,MRCHCF
HRR T5,AC11 ;BLOCK NUMBER
XCT T5
;READ/WRITE FROM OUR 200 WORD BUFFER UNTIL WE HIT EOF ON THE FIRST FILE
NXTRDD: MOVE T1,MRLDBL ;POINTER TO BLOCK
SUBI T1,1
HRLI T1,-200
SETZ T2,
MOVE T3,[IN T1]
OR T3,MRCHNN
XCT T3
CAIA
JRST LINER1 ;UNTIL ERROR
;OUT UUO
MOVE T1,MRLDBL
SUBI T1,1
HRLI T1,-200
SETZ T2,
MOVE T3,[OUT T1]
OR T3,MRCHCF ;TO COMMON FILE
XCT T3
JRST NXTRDD ;OK--LOOP
JRST LOUTF1 ;OUT FAILED!
;COME HERE WHEN "IN" FAILS - ASSUME EOF
LINER1: PUSHJ PP,DELOF ;DELETE OLD FILE
MOVSI T1,(RELEAS) ;CLOSE THE INPUT FILE
OR T1,MRCHNN
XCT T1
MOVSI T1,(RELEAS) ;CLOSE THE OUTPUT FILE
OR T1,MRCHCF
XCT T1
POPJ PP, ;DONE OK, RETURN
;HERE IF SIZE LIMIT WAS EXCEEDED
; OPEN LSTATS.DIR FOR UPDATING, CHANGE THE THINGS
; AND GO WRITE TO THE NEW FILE
SIZTBG: MOVSI T1,(RELEAS)
OR T1,MRCHCF ;RELEASE OLD COMMON FILE CHANNEL
XCT T1 ;THEN FALL INTO "NXTFLE"
;HERE FROM ABOVE, OR TIME LIMIT WAS EXCEEDED
; OPEN LSTATS.DIR FOR UPDATING, CHANGE THE THINGS
; AND GO WRITE TO THE NEW FILE
NXTFLE: PUSHJ PP,OPNUPD ;OPEN FILE FOR UPDATING
POPJ PP, ;IF ERRORS, JUST RETURN
PUSHJ PP,NEWINF ;PUT IN NEW INFO
POPJ PP, ;ERRORS
JRST TMLMOK ;GO WRITE TO THAT MTO FILE
;HERE IF DATE/TIME OF FIRST WRITE HAS NOT BEEN PUT IN LSTATS.DIR
;DO SO, AND GO WRITE TO THAT MTO FILE
SETTFW: PUSHJ PP,OPNUPD ;OPEN FILE FOR UPDATING
POPJ PP, ;IF ERRORS, JUST RETURN
MOVX T1,%CNDTM ;CURRENT DATE/TIME
GETTAB T1,
SETZ T1, ;IF CAN'T GET IT, MAKE SURE THE NEXT GUY TRIES
HRRZ T4,MRLDBL
MOVEM T1,.LDFWR(T4) ;STORE THE INFO
PUSHJ PP,NEWIOT ;WRITE OUT THE UPDATED DATA BLOCK
POPJ PP, ;PROBLEMS--GIVE IT UP
JRST TMLMOK ;CONTINUE
;ROUTINE TO OPEN LSTATS.DIR FOR UPDATING
;POPJ IF ERROR, SKIP RETURN IF OK.
OPNUPD: MOVEI T0,^D10 ;# TIMES WE WILL TRY
OPNUP0: MOVE T1,['LSTATS']
MOVSI T2,'DIR'
SETZ T3,
MOVE T4,MRLLDR ;PPN
MOVE T5,[ENTER T1]
OR T5,MRCHLS
;ENTER MAY FAIL BECAUSE OF SIMULTANEOUS UPDATE
XCT T5
JRST OPNEU2 ;HOPEFULLY JUST SIMULTANEOUS UPDATE
;HERE WHEN FILE HAS BEEN OPENED
OPNUP1: SETZM MRLFPR ;SET FILE PTR TO ZERO
MOVE T1,[USETI 1] ; AND REALLY DO IT
OR T1,MRCHLS
XCT T1
HRRZ T1,MRLDBL ;NOW READ 1ST 200 WORDS
SUBI T1,1
HRLI T1,-200 ;FORM IOWD
SETZ T2,
MOVE T3,[IN T1]
OR T3,MRCHLS
XCT T3 ;DO IN UUO
JRST RET.2 ;ALL OK
JRST LSINF ;ERROR- GO COMPLAIN AND POPJ
;HERE FOR PROBLEMS
;ENTER FAILED
OPNEU2: HRRZ T2,T2 ;ISOLATE ERROR CODE
CAIN T2,3 ; SIMULATEOUS UPDATE PROBLEM?
JRST OPNEU3 ;YES--WAIT A LITTLE
OUTSTR [ASCIZ/%LIBOL: Can't OPEN LSTATS.DIR for updating: (/]
PUSHJ PP,TYPOT2 ;TYPE ERROR CODE
OUTSTR [ASCIZ/)
/]
JRST LRLLC ;RELEASE LSTATS CHANNEL AND POPJ
OPNEU3: SOJLE T0,OPNEU4 ;WAITED TOO LONG?
MOVEI T1,1 ;SLEEP A SEC
SLEEP T1,
JRST OPNUP0 ;GO TRY AGAIN
OPNEU4: OUTSTR [ASCIZ/%LIBOL: Can't OPEN LSTATS.DIR for updating:
% Someone has it open for UPDATING
/]
JRST LRLLC ;RELEASE LSTATS CHANNEL AND POPJ
;ROUTINE TO PUT NEW INFO IN THE HEADER WHICH SAYS WE
;BUMPED TO THE NEXT FILESPEC.
; IF I/O ERRORS, OR FILE IS EXHAUSTED, POPJ, ELSE POPJ1
;WITH FILE POSITIONED AT 0, NEW INFO IN PLACE, AND 1ST
;DISK BLOCK UPDATED.
; IF POPJ RETURN TAKEN, THE CHANNEL HAS BEEN RELEASED, TOO.
NEWINF: MOVE T4,MRLDBL ;POINTER TO BLOCK
MOVE T1,.LDNFL(T4) ;# FILES,,# OF CURRENT
HLRZ T2,T1 ;# FILES
CAIG T2,(T1) ;IS FILE EXHAUSTED?
JRST FILEXH ; YES, SAY THAT
AOS .LDNFL(T4) ;BUMP # OF CURRENT FILE
MOVX T1,%CNDTM
GETTAB T1,
SETZ T1,
MOVEM T1,.LDFWR(T4) ;SAVE TIME OF FIRST WRITE
;BUMP CURRENT POINTER
MOVE T1,.LDBPC(T4) ;BYTE PTR TO CURRENT
HRRZ T3,T1
CAIL T3,200 ;POINTING TO FIRST BLOCK?
JRST NEWIN1 ;NO, REPOSITION FILE
ADD T1,MRLDBL ;MAKE ABSOLUTE PTR
ILDB T2,T1 ;GET A BYTE
JUMPN T2,.-1 ;LOOK FOR NULL
SUB T1,MRLDBL ;GET RELATIVE BP
MOVEM T1,.LDBPC(T4) ;THAT'S THE STARTING BP ALL RIGHT
MOVEM T1,MRLBPC
;WRITE OUT THE UPDATED BLOCK TO DSK
NEWIOT: MOVE T0,[USETO 1] ;PREPARE TO WRITE 1ST BLOCK
OR T0,MRCHLS
XCT T0
HRRZ T1,MRLDBL
SUBI T1,1
HRLI T1,-200
SETZ T2, ;END OF IO LIST
MOVE T3,[OUT T1]
OR T3,MRCHLS
XCT T3
JRST RET.2 ;EVERYTHING IS COOL .. SKIP RETURN
JRST OUERRL
;CURRENT FILE STARTS OR ENDS IN ANOTHER BLOCK. WE HAVE TO REPOSITION
; THE FILE TO GET THE START OF THE NEXT ONE, REPOSITION AGAIN TO 0,
; AND STORE THE STARTING BP.
NEWIN1: MOVE T4,MRLBPC ;GET BP TO CURRENT NOW
PUSHJ PP,LDPOST ;REPOSITION FILE
JRST LRLLC0 ;ERROR - DELETE OLD FILE, RELEASE, RETURN
MOVE T1,MRLBPC
SUB T1,MRLFPR ;INDEX INTO BLOCK
ADD T1,MRLDBL ; ADD START OF DATA BLOCK
ILDB T2,T1 ;LOOK FOR FIRST NULL
JUMPN T2,.-1
SUB T1,MRLDBL
ADD T1,MRLFPR ;GET RELATIVE BP ADDRESS
MOVEM T1,MRLBPC ;SAVE IT HERE
;NOW WE MUST REPOSITION THE FILE TO WORD 0 AGAIN
SETZ T4,
PUSHJ PP,LDPOST
JRST LRLLC0 ;DELETE OLD FILE, RELEASE STUFF
HRRZ T4,MRLDBL ;T4= START OF BLOCK
MOVE T1,MRLBPC ;STORE AWAY NEW "CURRENT BP"
MOVEM T1,.LDBPC(T4)
JRST NEWIOT ;GO WRITE UPDATED BLOCK TO DSK
;HERE IF WE ARE ALREADY POINTING TO THE LAST FILE IN THE DIRECTORY
FILEXH: OUTSTR [ASCIZ/%LIBOL: LSTATS.DIR is exhausted
/]
JRST LRLLC ;RELEAS LSTATS CHANNEL AND POPJ
;ROUTINE TO POSITION LD FILE
;CALLED WITH BYTE PTR TO THE FILE NAME IN T4 (OR 0)
;RETURNS: PROPER BLOCK IN THE 200 WORD BUFFER
; MRLFPR= NUMBER OF THE 1ST WORD IN THE BLOCK
; POPJ IF ERROR, SKIP RETURN IF OK
LDPOST: HRRZ T3,T4 ;GRAB START OF BYTE PTR
SKIPN T2,MRLFPR ;ZERO IS SPECIAL
JRST LDPST1 ;WE ARE NOW AT START OF FILE
;SET FILE POSITION TO BLOCK WE WANT, READ NEXT 205 WORDS IN
;(THIS ENSURES THAT THE BYTE PTR WILL END WITHIN THE BUFFER).
LDPST0: TRZ T3,177 ;MAKE EVEN START OF BLOCK
MOVEM T3,MRLFPR ;THIS WILL BE FILE POINTER
LSH T3,-7 ;MAKE BLOCK NUMBER
ADDI T3,1
MOVE T0,[USETI (T3)]
OR T0,MRCHLS ;LSTATS CHANNEL
XCT T0 ;POSITION TO THE BLOCK
HRRZ T1,MRLDBL ;POINT TO BLOCK
SUBI T1,1 ;IOWD
HRLI T1,-204 ;READ 204 WORDS IN
SETZ T2,
MOVE T3,[IN T1]
OR T3,MRCHLS
XCT T3
JFCL ;EOF CONDITION IS OK--ASSUME WE GOT THAT
JRST RET.2
;WE ARE POSITIONED AT THE START OF THE FILE
LDPST1: JUMPE T3,RET.2 ;IF WE ARE ALREADY HERE, DON'T BOTHER
JRST LDPST0 ;JUST GO READ IN WHAT WE WANT
;I/O ERRORS FOR LSTATS.DIR
;LOOKUP FAILED FOR THE OLD FILE WE HAD BEEN WRITING
LLKPF3: OUTSTR [ASCIZ/%LIBOL: LOOKUP failed (/]
HRRZ T2,T2
PUSHJ PP,TYPOT2 ;THIS IS WIERD, AND "CAN'T HAPPEN"
OUTSTR [ASCIZ/ for the .TMP file we'd been writing
/]
JRST CNTWLS ;TYPE FINAL MSG AND POPJ
;NO FREE CHANNELS
NFC1: OUTSTR [ASCIZ/%LIBOL: No free channels to write LSTATS data
/]
PUSHJ PP,DELOF ;DELETE OLD FILE
POPJ PP,
;GOBSTR UUO FAILED
GOBSTE: OUTSTR [ASCIZ/%LIBOL: GOBSTR UUO failed
/]
PUSHJ PP,DELOF ;DELETE OLD FILE
JRST CNTWLS
CNTWLS: OUTSTR [ASCIZ/% Can't write LSTATS data
/]
POPJ PP,
;COULDN'T FIND AN LSTATS.DIR
NOLSTS: OUTSTR [ASCIZ/%LIBOL: LSTATS.DIR not found
/]
PUSHJ PP,DELOF ;DELETE OLD FILE
JRST CNTWLS
;OPEN FAILED FOR A SYSTEM DEVICE
LOPNF1: OUTSTR [ASCIZ/%LIBOL: OPEN failed for device /]
PUSHJ PP,TYP6T2 ;TYPE SIXBIT FROM T2
OUTSTR [ASCIZ/
/]
PUSHJ PP,DELOF ;DELETE OLD FILE
JRST CNTWLS
;LOOKUP FAILED FOR LSTATS.DIR - ERROR CODE IN T2
LLKPE1: OUTSTR [ASCIZ/%LIBOL: LOOKUP ERROR (/]
PUSHJ PP,TYPOT2
OUTSTR [ASCIZ/) FOR LSTATS.DIR
/]
LRLLC0: PUSHJ PP,DELOF ;DELETE OLD FILE
LRLLC: MOVSI T1,(RELEAS) ;RELEASE LSTATS CHANNEL
OR T1,MRCHLS
XCT T1
JRST CNTWLS ;THEN GO FINISH MESSAGE
;IN UUO FAILED READING LSTATS.DIR
LSINF: OUTSTR [ASCIZ/%LIBOL: IN UUO failed reading LSTATS.DIR
/]
JRST LRLLC0 ;DELETE OLD FILE, RELEASE CHANNEL, FINISH MSG
;OUT UUO FAILED TRYING TO WRITE THE 1ST BLOCK
OUERRL: OUTSTR [ASCIZ/%LIBOL: OUT UUO failed updating LSTATS.DIR
/]
JRST LRLLC0 ;RELEASE CHANNEL, FINISH MSG
;THIS ROUTINE TYPES OCTAL FROM T2
TYPOT2: IDIVI T2,10 ;YE OLDE
PUSH PP,T3 ; RECURSIVE
SKIPE T2 ;NUMBER PRINT
PUSHJ PP,TYPOT2 ; ROUTINE
POP PP,T2
ADDI T2,"0"
OUTCHR T2
POPJ PP,
;THIS ROUTINE TYPES THE SIXBIT WORD IN T2
TYP6T2: JUMPE T2,RET.1 ;RETURN IF NO MORE CHARACTERS TO TYPE
SETZ T1,
LSHC T1,6 ;SHIFT BYTE INTO RIGHT 6 BITS
ADDI T1,40 ;MAKE ASCII CHARACTER
OUTCHR T1 ;TYPE IT
JRST TYP6T2 ;LOOP AS LONG AS THERE IS SOMETHING TO DO
;I/O ERRORS FOR THE COMMON FILE
;OPEN FAILED FOR COMMON FILE DEVICE
LOPNFC: OUTSTR [ASCIZ/%LIBOL: Can't OPEN device /]
PUSHJ PP,TYP6T2 ;TYPE DEVICE NAME FROM T2
OUTSTR [ASCIZ/
/]
JRST LRLLC0 ;DELETE OLD FILE,
;RELEASE LSTATS.DIR AND POPJ
;LOOKUP FAILED FOR COMMON FILE - THIS IS PROBABLY OK (JUST START A NEW FILE)
; UNLESS THE ERROR IS NOT FILE-NOT-FOUND
LLKPEC: HRRZ T2,T2 ;GET ERROR CODE
JUMPN T2,LLKPEA ;FILE NOT FOUND, NO, BAD ERROR
MOVSI T3,111000 ;LOAD PROTECTION
MOVEI AC11,1 ;USETO TO BLOCK 1
JRST DOENT ;GO DO ENTER
LLKPEA: PUSHJ PP,DELOF ;DELETE OLD FILE
OUTSTR [ASCIZ/%LIBOL: LOOKUP ERROR (/]
PUSHJ PP,TYPOT2
OUTSTR [ASCIZ/ on LSTATS file /]
;TYPE FILE NAME
LLKPE2: MOVE T2,MRCFNM
PUSHJ PP,TYP6T2
OUTCHR ["."]
MOVE T2,MRCFEX
PUSHJ PP,TYP6T2
OUTSTR [ASCIZ/
/]
PUSHJ PP,LRCF ;RELEASE COMMON FILE
JRST LRLLC ;RELEASE LSTATS.DIR AND POPJ
;RELEASE COMMON FILE
LRCF: MOVSI T1,(RELEAS)
OR T1,MRCHCF ;COMMON FILE CHANNEL
XCT T1
POPJ PP, ;SILENTLY, RETURN
;RELEASE OLD FILE WITHOUT DELETING IT
LROLD: MOVSI T1,(RELEAS)
OR T1,MRCHNN
XCT T1
POPJ PP,
;ENTER FAILED FOR COMMON FILE
LENTEC: HRRZ T2,T2 ;GET ERROR CODE RETURNED
CAIN T2,3 ;SIMULATEOUS UPDATE?
JRST LENTE3 ;YES
OUTSTR [ASCIZ/%LIBOL: ENTER failed (/]
PUSHJ PP,TYPOT2 ;TYPE ENTER ERROR CODE
OUTSTR [ASCIZ/) for LSTATS output file /]
PUSHJ PP,DELOF ;DELETE OLD FILE
JRST LLKPE2 ;TYPE FILE NAME AND RELEASE CHANNELS
LENTE3: SOJLE T0,LENTE4 ;TRIED TOO LONG--GIVE UP
MOVEI T1,1 ;NO, SLEEP A SEC
SLEEP T1,
JRST DOENTA ;THEN TRY AGAIN
LENTE4: OUTSTR [ASCIZ/%LIBOL: Can't append to LSTATS output file -
(Someone else has it open for updating)
/]
PUSHJ PP,LRCF ;RELEASE COMMON FILE CHANNEL
JRST LRLLC0 ;DELETE OLD FILE,
;RELEASE LSTATS.DIR AND POPJ
;OUT FAILED TO THE COMMON FILE
LOUTF1: OUTSTR [ASCIZ/%LIBOL: OUT uuo failed for LSTATS common file
/]
PUSHJ PP,LROLD ;RELEASE OLD FILE (NOT DELETED)
PUSHJ PP,LRCF ;RELEASE COMMON FILE
JRST CNTWLS ;TYPE FINAL MSG AND POPJ
>;END IFN FTLSDR
>;END IFE TOPS20
SUBTTL LSTATS - TOPS20 DIRECTORY FILE ROUTINES
;IF ALL OK, RETURNS .+1. IF SOME FATAL ERROR HAPPENS, TYPES A
; WARNING TO THE USER AND RETURNS .+1 ALSO.
IFN TOPS20,<
IFDEF LSDIR,<
GOLSDR:
;TRY TO FIND LSTATS.DIR
MOVE T2,[POINT 7,[LSDIR]] ;LSTATS DIRECTORY
MOVE T1,[POINT 7,MRLDNA] ;POINTER TO ASCII STRING TO MAKE
ILDB T3,T2 ;GET A CHAR
JUMPE T3,.+3 ; IF NULL, STOP
IDPB T3,T1 ;STORE IT
JRST .-3 ;AND LOOP
MOVE T2,[POINT 7,[ASCIZ/LSTATS.DIR/]] ;THE FILENAME
ILDB T3,T2 ;GET A CHAR
IDPB T3,T1 ;STORE
JUMPN T3,.-2 ;LOOP, UNTIL WE'VE STORED LAST NULL
MOVX T1,GJ%SHT!GJ%OLD ;MUST BE AN OLD FILE
MOVE T2,[POINT 7,MRLDNA] ;POINTER TO FILENAME STRING WE JUST MADE
GTJFN
ERJMP LGTJNE ;GTJFN FAILED - GO SEE WHY
MOVEM T1,MRLDJF ;REMEMBER IT'S JFN
;FOUND IT! - OPEN FOR READING
MOVX T2,OF%RD ;JUST OPEN FOR READING
OPENF
ERJMP LOPNFF ;OPENF FAILED
;LSTATS.DIR IS OPEN FOR READING
;READ IN FIRST 200 WORDS .. AND COPY INFORMATION
MOVE T1,MRLDJF
HRRZ T2,MRLDBL ;POINTER TO BLOCK
HRLI T2,(POINT 36,)
MOVNI T3,200 ;-# WORDS
SIN ;COMMIT A SIN
ERJMP LSINF ;UH OH WE'RE IN TROUBLE NOW!
SETZM MRLFPR ;REMEMBER FILE POINTER IS 0
MOVE T4,MRLDBL ;T4 = INDEX TO THE BLOCK
MOVE T1,.LDBPC(T4) ;BYTE PTR TO CURRENT FILENAME
MOVEM T1,MRLBPC ;SAVE FOR LATER
MOVE T1,.LDSZL(T4) ;SIZE LIMIT
MOVEM T1,MRLSZL ;SAVE
SKIPN .LDTML(T4) ; SKIP IF ANY TIME LIMIT TO WORRY ABOUT
JRST TMLMOK ;NO
SKIPN T3,.LDFWR(T4) ;GET TIME OF FIRST WRITE
JRST SETTFW ;NONE THERE--SET TO NOW
ADD T3,.LDTML(T4) ;ADD TIME LIMIT
GTAD ; AND TODAY'S DATE/TIME
CAML T1,T3 ;IS IT TIME TO INCREMENT?
JRST NXTFLE ;YES!
;TIME LIMIT IS OK -- OPEN FILE FOR APPEND AND CHECK SIZE LIMIT
TMLMOK: MOVE T4,MRLBPC ;BYTE PTR TO CURRENT FILE
PUSHJ PP,LDPOST ;POSITION LD FILE
JRST RLDFJ ;ERROR - RELEASE JFN AND QUIT
ADD T4,MRLDBL ; NOW GET ACTUAL BYTE PTR TO THE FILE
SUB T4,MRLFPR
;BUILD FILENAME IN MRLDNA
MOVE T2,[POINT 7,[LSDIR]] ;DIRECTORY, OR LOGICAL DEVICE
MOVE T1,[POINT 7,MRLDNA]
ILDB T3,T2
JUMPE T3,.+3
IDPB T3,T1
JRST .-3
MOVE T2,T4 ;COPY BP
ILDB T3,T2 ;NOW STORE THE FILENAME
IDPB T3,T1
JUMPN T3,.-2
MOVE T2,[POINT 7,MRLDNA] ;POINT TO FILESPEC
MOVX T1,GJ%SHT
GTJFN
ERJMP FJFNF ;GTJFN FAILED -- COMPLAIN!!
MOVEM T1,MRLCJF ;SAVE JFN OF THIS COMMON FILE
SKIPN MRLSZL ;SKIP IF A SIZE LIMIT TO WORRY ABOUT
JRST SIZIOK ;NO - SIZE OK
SIZEF ; SEE IF SIZE LIMIT EXCEEDED
ERJMP SZEFF ;FAILED -- GO SEE WHY
CAML T3,MRLSZL ; SIZE TOO BIG?
JRST SIZTBG ;YES: GO ON TO NEXT FILE
;IT IS OK TO WRITE INTO THIS FILE! OPEN FOR APPEND, AND COPY THE INFO
; INTO IT.
SIZIOK: MOVE T1,MRLDJF ;CLOSE AND RELEASE LSTATS.DIR
CLOSF
ERJMP CLSFE ; * FAILED * ... ???
MOVE T1,MRLCJF ;JFN OF THIS FILE
MOVX T2,OF%APP
OPENF
ERJMP COPNFF ;CAN'T OPEN FOR APPEND!
MOVE T1,MRJFN ;OPEN THE ONE WE WROTE TO FOR READ NOW
MOVX T2,OF%RD
OPENF
ERJMP OOPNFF ;CAN'T!!
;READ/WRITE FROM OUR 200 WORD BUFFER UNTIL WE HIT EOF ON THE FIRST FILE
NXTRDD: MOVE T1,MRJFN ;READ FROM HERE
HRRZ T2,MRLDBL ;POINTER TO BLOCK
HRLI T2,(POINT 36,)
MOVNI T3,200 ;-# WORDS
SIN ;COMMIT A SIN
ERJMP NXTRD1 ;UNTIL ERROR
MOVE T1,MRLCJF ;WRITE TO HERE
HRRZ T2,MRLDBL
HRLI T2,(POINT 36,)
MOVNI T3,200
SOUT
ERJMP NXTRD2
JRST NXTRDD ;LOOP...
;COME HERE WHEN SIN FAILS - ASSUME EOF
NXTRD1: MOVE T1,MRJFN ;CLOSE THE INPUT FILE
TXO T1,CO%NRJ ;DON'T RELEASE JFN
CLOSF
ERJMP .+1 ;IGNORE ERRORS
MOVE T1,MRLCJF ;CLOSE THE OUTPUT FILE
CLOSF
ERJMP .+1 ;IGNORE ERRORS
PUSHJ PP,DELOF ;DELETE OLD FILE
POPJ PP,
;HERE IF SOUT FAILS FOR THE COMMON FILE
NXTRD2: HRROI T1,[ASCIZ/%LIBOL: Can't append to LSTATS file because:
/]
PSOUT
PUSHJ PP,TYPFER ;PRINT LAST ERROR IN THIS FORK
JRST NXTRD1 ;CLOSE FILES AND RETURN
; HERE IF SIZE LIMIT WAS EXCEEDED
; OPEN LSTATS.DIR FOR UPDATING, CHANGE THE THINGS
; AND GO WRITE TO THE NEW FILE
SIZTBG: MOVE T1,MRLCJF ;RELEASE OLD JFN
RLJFN
ERJMP .+1 ;IGNORE ERROR HERE
; HERE IF TIME LIMIT WAS EXCEEDED
;OPEN LSTATS.DIR FOR UPDATING, CHANGE THE THINGS
; AND GO WRITE TO THE NEW FILE
NXTFLE: PUSHJ PP,OPNUPD ;OPEN FILE FOR UPDATING
POPJ PP, ;IF ERRORS, JUST RETURN
PUSHJ PP,NEWINF ;PUT IN NEW INFO
POPJ PP, ;?FILE IS EXHAUSTED!
JRST TMLMOK ;GO WRITE TO THAT MTO FILE
;HERE IF DATE/TIME OF FIRST WRITE HAS NOT BEEN PUT IN LSTATS.DIR
;DO SO, AND GO WRITE TO THAT MTO FILE
SETTFW: PUSHJ PP,OPNUPD ;OPEN FILE FOR UPDATING
POPJ PP, ;IF ERRORS, JUST RETURN
GTAD ;GET DATE/TIME NOW
HRRZ T4,MRLDBL
MOVEM T1,.LDFWR(T4) ;STORE THE INFO
PUSHJ PP,NEWIOT ; WRITE OUT THE UPDATED DATA BLOCK
POPJ PP, ;PROBLEMS-- GIVE IT UP
JRST TMLMOK ;CONTINUE
;ROUTINE TO OPEN LSTATS.DIR FOR UPDATING
;POPJ IF ERROR, SKIP RETURN IF OK.
OPNUPD: MOVE T1,MRLDJF ;CLOSE FILE
TXO T1,CO%NRJ ;DON'T RELEASE JFN!
CLOSF
ERJMP OPNEU1 ;CLOSF SHOULDN'T FAIL!
MOVEI T4,^D10 ;# TIMES WE WILL TRY
OPNUP0: MOVE T1,MRLDJF
MOVX T2,OF%RD!OF%WR ;READ/WRITE ACCESS
OPENF
ERJMP OPNEU2 ;OPENF MAY FAIL BECAUSE OF SIMULATANEOUS ACCESS
;HERE WHEN FILE HAS BEEN OPENED
OPNUP1: SETZM MRLFPR ;SET FILE PTR TO ZERO
MOVE T1,MRLDJF ;GET READY TO READ IT
HRRZ T2,MRLDBL
HRLI T2,(POINT 36,)
MOVNI T3,200 ;-# WORDS
SIN
ERJMP OPNUP2 ;FAILED
JRST RET.2 ;SIGNAL ALL OK
; HERE FOR PROBLEMS
OPNUP2: HRROI T1,[ASCIZ/%LIBOL: Can't update LSTATS.DIR - SIN failed:
/]
PSOUT
OPNERC: PUSHJ PP,TYPFER ;TYPE THE ERROR
OPNERD: MOVE T1,MRLDJF ;NOW RELEASE JFN
RLJFN
ERJMP .+1 ;OH WELL, WE TRIED
POPJ PP, ;RETURN
OPNEU1: HRROI T1,[ASCIZ/%LIBOL: CLOSF failed for LSTATS.DIR: /]
PSOUT
JRST OPNERC
;OPENF FAILED FOR READ/WRITE ACCESS
OPNEU2: CAIN T1,OPNX9 ;INVALID SIMULTANEOUS ACCESS?
JRST OPNEU3 ;YES--WAIT A LITTLE
HRROI T1,[ASCIZ/%LIBOL: Can't OPENF LSTATS.DIR for updating: /]
PSOUT
JRST OPNERC ;TYPE REASON, AND RELEASE JFN
OPNEU3: SOJLE T4,OPNEU4 ;WAITED TOO LONG?
MOVEI T1,^D1000 ;NO, SLEEP A SEC
DISMS
JRST OPNUP0 ;GO TRY AGAIN
OPNEU4: HRROI T1,[ASCIZ/%LIBOL: Can't OPENF LSTATS.DIR for updating: /]
PSOUT
HRROI T1,[ASCIZ/
% Someone has it open for UPDATING
/]
PSOUT
JRST OPNERD ;RELEASE JFN AND POPJ
;ROUTINE TO PUT NEW INFO IN THE HEADER WHICH SAYS WE
;BUMPED TO THE NEXT FILESPEC.
; IF I/O ERRORS, OR FILE IS EXHAUSTED, POPJ, ELSE POPJ1
;WITH FILE POSITIONED AT 0, NEW INFO IN PLACE, AND 1ST DSK BLOCK UPDATED.
; IF POPJ RETURN TAKEN, THE JFN HAS BEEN RELEASED, TOO.
NEWINF: MOVE T4,MRLDBL ;POINTER TO BLOCK
MOVE T1,.LDNFL(T4) ;# FILES,,# OF CURRENT
HLRZ T2,T1 ;# FILES
CAIG T2,(T1) ;IS FILE EXHAUSTED
JRST FILEXH ; YES, SAY THAT
AOS .LDNFL(T4) ;BUMP # OF CURRENT FILE
GTAD ;SAVE LATEST DATE/TIME IN HEADER
MOVEM T1,.LDFWR(T4) ;SAVE TIME OF FIRST WRITE
;BUMP CURRENT POINTER
MOVE T1,.LDBPC(T4) ;BYTE PTR TO CURRENT
HRRZ T3,T1
CAIL T3,200 ;POINTING TO FIRST BLOCK?
JRST NEWIN1 ;NO, REPOSITION FILE
ADD T1,MRLDBL ;MAKE ABSOLUTE PTR
ILDB T2,T1 ;GET A BYTE
JUMPN T2,.-1 ;LOOK FOR NULL
SUB T1,MRLDBL ;GET RELATIVE BP
HRRZ T3,T1
CAIL T3,200 ;SKIP IF STILL IN 1ST BLOCK
JRST NEWIN1 ;NO, HAVE TO REPOSITION FILE
MOVEM T1,.LDBPC(T4) ;THAT'S THE STARTING BP ALL RIGHT
MOVEM T1,MRLBPC ; SAVE AS BP TO CURRENT
;WRITE OUT THE UPDATED BLOCK TO DSK
NEWIOT: MOVE T1,MRLDJF ;TO SFPTR TO 0 TO MAKE SURE
SETZ T2,
SFPTR
ERJMP SFPTE2 ;ERROR...?
MOVE T1,MRLDJF ;NOW GET READY FOR SOUT
HRRZ T2,MRLDBL
HRLI T2,(POINT 36,)
MOVNI T3,200
SOUT ;WRITE THE 1ST BLOCK
ERJMP NEWIES ;ERROR...?
;EVERYTHING IS COOL .. SKIP RETURN
JRST RET.2
;CURRENT FILE STARTS OR ENDS IN ANOTHER BLOCK. WE HAVE TO
;REPOSITION THE FILE TO GET THE START OF THE NEXT ONE,
;REPOSITION AGAIN TO 0, AND STORE THE STARTING BP.
NEWIN1: MOVE T4,MRLBPC ;GET BP TO CURRENT NOW
PUSHJ PP,LDPOST ;REPOSITION FILE
POPJ PP, ;ERROR - POPJ
HLLZ T1,MRLBPC ; GET LH OF BYTE PTR (RH = 0)
ADD T1,MRLDBL ;GET ABSOLUTE BP TO START OF OLD CURRENT
ILDB T2,T1 ;LOOK FOR FIRST NULL
JUMPN T2,.-1
SUB T1,MRLDBL ;GET REL ADDRESS IN THIS BLOCK
ADD T1,MRLFPR ;T1= RELATIVE PTR TO CURRENT FILE
MOVEM T1,MRLBPC ;SAVE HERE FOR A SEC..
;NOW WE MUST REPOSITION THE FILE TO WORD 0 AGAIN
SETZ T4,
PUSHJ PP,LDPOST
POPJ PP, ;ERROR - POPJ
HRRZ T4,MRLDBL
MOVE T1,MRLBPC ;GET SAVED BP
MOVEM T1,.LDBPC(T4) ;STORE IT
JRST NEWIOT ;GO WRITE UPDATED BLOCK TO DSK
;ERROR - SFPTR FAILED TRYING TO GET TO 0
SFPTE2: HRROI T1,[ASCIZ/%LIBOL: SFPTR failed trying to update LSTATS.DIR:
/]
PSOUT
PUSHJ PP,TYPFER
JRST RLDFJ ;RELEASE JFN AND POPJ
;ERROR - SOUT FAILED TRYING TO UPDATE 1ST BLOCK
NEWIES: HRROI T1,[ASCIZ/%LIBOL: SOUT failed trying to update LSTATS.DIR:
/]
PSOUT
PUSHJ PP,TYPFER
JRST RLDFJ ;RELEASE JFN AND POPJ
;HERE IF WE ARE ALREADY POINTING TO THE LAST FILE IN THE DIRECTORY
FILEXH: HRROI T1,[ASCIZ/%LIBOL: LSTATS.DIR is exhausted
/]
PSOUT
MOVE T1,MRLDJF
RLJFN
ERJMP .+1
POPJ PP,
; ROUTINE TO POSITION LSTATS.DIR
;CALL: T4/ BYTE PTR TO IT
; PUSHJ PP,LDPOST
; RETURNS: PROPER BLOCK IN THE 200 WORD BUFFER
; MRLFPR = NUMBER OF THE FIRST WORD IN THE BLOCK
; RETURNS POPJ IF ERROR, SKIP RETURN IF OK
LDPOST: HRRZ T3,T4 ;GRAB START OF BLOCK
SKIPN T2,MRLFPR ;ZERO IS SPECIAL
JRST LDPST1 ;WE ARE NOW AT START OF FILE
;SET FILE POSITION TO WORD WE WANT, READ NEXT 200 WORDS IN.
LDPST0: MOVEM T3,MRLFPR ;NEW FILE POINTER
MOVE T1,MRLDJF ; SET FILE PTR - GET JFN IN T1
MOVE T2,T3 ;T2= BYTE NUMBER
SFPTR
ERJMP SFPTRE ;SFPTR FAILED!
MOVE T1,MRLDJF ;GET READY TO READ IT
HRRZ T2,MRLDBL
HRLI T2,(POINT 36,)
MOVNI T3,200 ;-# WORDS
SIN
ERJMP .+1 ;EOF CONDITION IS OK -- ASSUME WE GOT THAT
JRST RET.2 ;GIVE SKIP RETURN
;WE ARE POSITIONED AT THE START OF THE FILE
LDPST1: JUMPE T3,LDPST0 ;GO READ 1ST 200 WORDS
;WE ARE AT START OF FILE, BUT WANT SOMETHING IN THE MIDDLE SOMEWHERE.
;IF IT IS WITHIN THE FIRST 200 WORDS, DON'T BOTHER REPOSITIONING THE FILE.
;HOWEVER, BE CAREFUL THAT THE WHOLE FILESPEC FITS IN THE FIRST 200 WORDS..
;ELSE WE MUST GO TO LDPST0.
MOVE T2,T4 ;GET POINTER NOW
ADD T2,MRLDBL ;MAKE ACTUAL BYTE PTR
ILDB T1,T2
JUMPN T1,.-1 ;LOOK FOR FIRST NULL
HRRZ T2,T2
SUB T2,MRLDBL
CAIGE T2,200 ;IF LESS THAN 200, WE'RE OK
JRST RET.2 ;IT ALL FITS IN THE BLOCK
JRST LDPST0 ;GO REPOSITION, ETC.
;SFPTR FAILED
SFPTRE: HRROI T1,[ASCIZ/%LIBOL: SFPTR failed for LSTATS.DIR: /]
PSOUT
JRST TYPFER ;TYPE MESSAGE
;I/O ERRORS FOR LSTATS.DIR
;HERE IF GTJFN FAILS FOR LSTATS.DIR
LGTJNE: CAIN T1,GJFX24 ;IF FILE-NOT-FOUND
JRST LGTJN1 ;GO SAY THAT
HRROI T1,[ASCIZ/%LIBOL: GTJFN failed for LSTATS.DIR: /]
PSOUT
JRST TYPFER ;TELL USER
LGTJN1: HRROI T1,[ASCIZ/%LIBOL: LSTATS.DIR not found
/]
PSOUT
POPJ PP, ;RETURN
;HERE IF OPENF FAILED FOR LSTATS.DIR (BUT THE GTJFN HAD WORKED!)
LOPNFF: HRROI T1,[ASCIZ/%LIBOL: Can't OPEN LSTATS.DIR for read: /]
PSOUT
PUSHJ PP,TYPFER ;TYPE REASON
RLDFJ: MOVE T1,MRLDJF ;RELEASE JFN
RLJFN
ERJMP .+1
POPJ PP, ;RETURN
;HERE IF SIN FAILED TRYING TO READ FIRST 200 WORDS OF LSTATS.DIR
LSINF: HRROI T1,[ASCIZ/%LIBOL: Can't read LSTATS.DIR: /]
PSOUT
PUSHJ PP,TYPFER ; EVERYTHING THAT COULD HAPPEN IS WRONG
JRST RLDFJ ;RELEASE JFN AND POPJ
;HERE IF CLOSF FAILED AFTER WE DID ALL WE HAD TO DO IN LSTATS.DIR
CLSFE: HRROI T1,[ASCIZ/%LIBOL: CLOSF failed for LSTATS.DIR: /]
PSOUT
PUSHJ PP,TYPFER ;TYPE WHY
JRST NXTRD1 ;CLOSE INPUT, OUTPUT FILES, AND RETURN
;I/O ERRORS FOR THE COMMON FILE
;HERE IF GTJFN FAILS FOR THE COMMON FILE THE DIRECTORY POINTS US TO
FJFNF: HRROI T1,[ASCIZ/%LIBOL: GTJFN failed for LSTATS file: /]
PSOUT
CMMERR: PUSHJ PP,TYPFER ;TYPE WHAT IS WRONG
HRROI T1,[ASCIZ/% Filespec is: /]
PSOUT
MOVE T1,[POINT 7,MRLDNA] ;STRING WAS HERE
PSOUT
JRST RLDFJ ;RELEASE JFN OF LSTATS.DIR AND POPJ
;THIS ROUTINE IS SIMILAR TO CMMERR, BUT ALSO RELEASES JFN OF COMMON FILE
CMMER1: PUSHJ PP,TYPFER ;TYPE WHAT IS WRONG
HRROI T1,[ASCIZ/% Filespec is: /]
PSOUT
MOVE T1,[POINT 7,MRLDNA]
PSOUT
MOVE T1,MRLCJF ;RELEASE COMMON FILE JFN
RLJFN
ERJMP .+1 ;HOPE WE DID
JRST RLDFJ ;RELEASE JFN OF LSTATS.DIR AND POPJ
;HERE IF SIZEF FAILS FOR COMMON FILE
SZEFF: HRROI T1,[ASCIZ/%LIBOL: SIZEF failed for LSTATS file: /]
PSOUT
JRST CMMERR ;GO TYPE REASON AND FILESPEC
;HERE IF OPENF FAILED - CAN'T OPEN IT FOR APPEND
COPNFF: HRROI T1,[ASCIZ/%LIBOL: Can't append to LSTATS file: /]
PSOUT
JRST CMMER1 ;GO TYPE REASON AND FILESPEC
;HERE IF OPENF FAILED FOR FILE WE HAD WRITTEN
OOPNFF: HRROI T1,[ASCIZ/%LIBOL: Can't append to LSTATS file - /]
PSOUT
HRROI T1,[ASCIZ/OPENF failed for old file:
/]
PSOUT
PUSHJ PP,TYPFER ;TYPE REASON
MOVE T1,MRLCJF ;RELEASE JFN OF COMMON FILE
RLJFN
ERJMP .+1 ;HOPE WE DID
JRST RLDFJ ;RELEASE JFN OF LSTATS.DIR AND POPJ
>;END IFDEF LSDIR
>;END IFN TOPS20
; ROUTINES TO APPEND TO <JOBNAME>.MTO ON DSK:
;(NO LSTATS.DIR IS USED)
SUBTTL TOPS10 JOBNAME.MTO APPEND CODE
IFE FTLSDR,<
IFE TOPS20,<
DOAP10: PUSHJ PP,GMCHAN ;GET A CHANNEL FOR THE .MTO FILE
JRST AP1NFC ;?CAN'T
ANDI T5,17
DPB T5,[POINT 4,T5,12] ;STORE IN AC FIELD
HLLZM T5,MRCHCF ;SAVE CHANNEL NUMBER
;OPEN 'DSK' ON THAT CHANNEL
MOVEI T1,17 ;DUMP MODE
MOVSI T2,'DSK'
SETZ T3,
MOVE T4,[OPEN T1]
OR T4,MRCHCF
XCT T4
JRST AP1ODF ;?OPEN OF DSK FAILED
;OPEN PREVIOUS FILE FOR READING
MOVE T1,MRNM6 ;NAME
MOVSI T2,'TMP' ;EXTENSION
SETZB T3,T4 ;ON DSK:
MOVE T5,[LOOKUP T1]
OR T5,MRCHNN
XCT T5
JRST AP1LKF ;?CAN'T FIND FILE WE'VE BEEN WRITING
;LOOK FOR JOBNAME.MTO
HRROI T1,3
GETTAB T1, ;GET MY PROGRAM NAME
MOVE T1,[SIXBIT/METER/] ;FAILED--GET 'METER'
MOVSI T2,'MTO'
SETZB T3,T4
MOVE T5,[LOOKUP T1] ;SEE IF ALREADY TEHRE
OR T5,MRCHCF
XCT T5
JRST AP1LF1 ;FAILED-- HOPEFULLY JUST NOT THERE
;IT IS THERE, OPEN FOR APPEND
HLRE T1,T4 ;GET SIZE
JUMPL T1,[MOVM T1,T1 ;SOME # OF WORDS-- ROUND UP TO BLOCKS
IDIVI T1,200
JUMPE T2,.+1
AOJA T2,.+1]
MOVEI AC11,1(T1) ;START WRITING TO NEXT BLOCK
MOVEI T0,^D5 ;# OF TIMES TO TRY
DOAPE: MOVSI T2,'MTO'
SETZB T3,T4
MOVE T5,[ENTER T1]
OR T5,MRCHCF
XCT T5
JRST AP1EN0 ;FAILED-- BAD ERROR
MOVSI T1,(USETO) ;PREPARE TO APPEND
OR T1,MRCHCF
HRR T1,AC11 ;GET BLOCK NUMBER TO START AT
XCT T1
JRST CENTOK ;COMMON ENTER OK
;HERE IF LOOKUP FAILED
AP1LF1: HRRZ T2,T2 ;GET ERROR CODE
JUMPN T2,AP1LFF ;OOPS.. SOMETHING BESIDES "FILE NOT FOUND"!
;JUST DO ENTER
MOVSI T2,'MTO'
SETZB T3,T4
MOVE T5,[ENTER T1]
OR T5,MRCHCF
XCT T5
JRST AP1CN1 ;NORMAL ENTER FAILED! (STRANGE!)
;HERE WHEN FILES ARE OPEN - COPY THE DATA
CENTOK: MOVE T1,MRLDBL ;PREPARE FOR "IN" UUO
SUBI T1,1 ; TO OUR 200-WORD BUFFER
HRLI T1,-200
SETZ T2,
MOVE T3,[IN T1]
OR T3,MRCHNN
XCT T3
CAIA ;OK
JRST AP1INE ;FAILED--ASSUME EOF
;OUT
MOVE T1,MRLDBL
SUBI T1,1
HRLI T1,-200
SETZ T2,
MOVE T3,[OUT T1]
OR T3,MRCHCF
XCT T3
JRST CENTOK ;WORKED--GO DO SOME MORE
JRST AP1OUF ;?OUT FAILED!
;HERE WHEN "IN" FAILS, ASSUMING EOF
AP1INE: PUSHJ PP,DELOF ;DELETE OLD FILE
JRST RLCMF ;AND RELEASE COMMON FILE CHANNEL
;ERRORS
AP1NFC: OUTSTR [ASCIZ/%LIBOL: No free channels to append LSTATS data
/]
POPJ PP, ;GIVE UP AND RETURN
;OPEN OF 'DSK' FAILED?
AP1ODF: OUTSTR [ASCIZ/%LIBOL: Can't OPEN DSK to append LSTATS data
/]
POPJ PP, ;GIVE UP AND RETURN
;LOOKUP FAILED FOR FILE WE'VE BEEN WRITING
AP1LKF: OUTSTR [ASCIZ/%LIBOL: LOOKUP faile for .TMP file we wrote!
/]
RLOF: MOVSI T1,(RELEAS)
OR T1,MRCHNN
XCT T1
POPJ PP,
;ENTER FAILED FOR UPDATE OF JOBNAME.MTO
AP1EN0: HRRZ T2,T2 ;GET ERROR CODE
CAIN T2,3 ;SIMULATANEOUS PROBLEM?
JRST AP1EN3 ;YES, WAIT A SEC
OUTSTR [ASCIZ/%LIBOL: ENTER failed for .MTO file on DSK:
/]
PUSHJ PP,DELOF ;DELETE OLD FILE
RLCMF: MOVSI T1,(RELEAS)
OR T1,MRCHCF
XCT T1
POPJ PP,
AP1EN3: SOJLE T0,AP1EN4 ;WAITED TOO LONG
MOVEI T2,1
SLEEP T2,
JRST DOAPE ;TRY ENTER AGAIN
AP1EN4: OUTSTR [ASCIZ/%LIBOL: ENTER failed for .MTO file on DSK:
% Someone else has it OPEN for update
/]
PUSHJ PP,DELOF ;DELETE OLD FILE
JRST RLCMF ;GO RELEASE COMMON CHANNEL
;LOOKUP FAILED OTHER THAN ERROR CODE 0 FOR DSK:JOBNAME.MTO
AP1LFF: OUTSTR [ASCIZ/%LIBOL: LOOKUP failed for .MTO file on DSK:
/]
PUSHJ PP,DELOF ;DELETE OLD FILE
JRST RLCMF ;GO RELEASE COMMON CHANNEL
;NORMAL ENTER FAILED FOR DSK:JOBNAME.MTO
AP1CN1: OUTSTR [ASCIZ/%LIBOL: ENTER failed for .MTO file on DSK:
/]
PUSHJ PP,DELOF ;DELETE OLD FILE
JRST RLCMF ;GO RELEASE COMMON CHANNEL
;"OUT" FAILED FOR .MTO FILE
AP1OUF: OUTSTR [ASCIZ/%LIBOL: OUT failed while appending to .MTO file
/]
PUSHJ PP,DELOF ;TRY TO DELETE OLD FILE
JRST RLCMF ;GO RELEASE COMMON CHANNEL
>;END IFE TOPS20
SUBTTL TOPS20 JOBNAME.MTO APPEND CODE
IFN TOPS20,<
DOAP20: MOVE T1,MRJFN
MOVX T2,OF%RD
OPENF
ERJMP AP20RF
;OPEN (FOR UPDATE) JOBNAME.MTO
GETNM
MOVE T2,[POINT 6,T1]
MOVE T3,[POINT 7,MRNMA] ;BUILD ASCII FILE NAME
DOAP21: ILDB T4,T2
JUMPE T4,DOAP22
ADDI T4,40
IDPB T4,T3
TLNE T2,760000
JRST DOAP21 ;LOOP FOR ALL CHARS
DOAP22: MOVE T2,[POINT 7,[ASCIZ/.MTO/]]
ILDB T4,T2
IDPB T4,T3
JUMPN T4,.-2
;GTJFN
MOVX T1,GJ%SHT
MOVE T2,[POINT 7,MRNMA]
GTJFN
ERJMP AP2GJF ;GTJFN FAILED
MOVEM T1,MRLCJF ;SAVE JFN
;OPENF
MOVEI T5,^D5 ;TIMES WE WILL TRY
DOAP25: MOVE T1,MRLCJF ;GET JFN
MOVX T2,OF%APP ;APPEND ACCESS
OPENF
ERJMP AP25ER ;FAILED (PROBABLY SIMULTANEOUS ACCESS)
;OK--BOTH FILES OPEN -- COPY THE DATA
DOAP26: MOVE T1,MRJFN ;OUR FAMILIAR READ/WRITE LOOP
HRRZ T2,MRLDBL
HRLI T2,(POINT 36,) ;BYTE PTR TO BLOCK
MOVNI T3,200
SIN ;READ DATA FROM .TMP FILE
ERJMP DOAP27 ;UNTIL ERROR
MOVE T1,MRLCJF ; APPEND IT TO COMMON FILE (.MTO)
HRRZ T2,MRLDBL
HRLI T2,(POINT 36,)
MOVNI T3,200
SOUT
ERJMP AP26ER ;SOUT FAILED - COMPLAIN!
JRST DOAP26 ;LOOP
;HERE WHEN SIN FAILS -- ASSUME EOF
DOAP27: MOVE T1,MRJFN ;CLOSE THE INPUT FILE
TXO T1,CO%NRJ ;BUT DON'T RELEAS JFN
CLOSF
ERJMP .+1 ;IGNORE ERRORS CLOSING
MOVE T1,MRLCJF ;CLOSE COMMON FILE
CLOSF ; AND RELEASE JFN
ERJMP .+1 ;IGNORE ERRORS CLOSING
JRST DELOF ;DELETE THE .TMP FILE
;AND RETURN (TO EXIT THE PROGRAM)
;HERE FOR PROBLEMS THAT OCCUR ABOVE
;OPENF FAILED FOR OLD FILE
AP20RF: HRROI T1,[ASCIZ/%LIBOL: OPENF failed for LSTATS .TMP file: /]
PSOUT
PUSHJ PP,TYPFER
MOVE T1,MRJFN
RLJFN ;TRY TO RELEAS JFN
ERJMP .+1 ;IGNORE ERRORS
POPJ PP, ;RETURN
;GTJFN FAILED FOR THE COMMON FILE
AP2GJF: HRROI T1,[ASCIZ/%LIBOL: GTJFN failed for output .MTO file: /]
PSOUT
PUSHJ PP,TYPFER ;WHAT IS IT EXACTLY
JRST DELOF ;DELETE THE OLD FILE AND RETURN
;OPENF FAILED FOR COMMON FILE
AP25ER: CAIE T1,OPNX9 ;SIMULTANEOUS UPDATE?
JRST AP25E1 ;NO -- SOME BAD ERROR
SOJLE T5,AP25E2 ;TRIED SEVERAL TIMES, GIVE UP
MOVEI T1,^D1000 ;SLEEP A SEC
DISMS
JRST DOAP25 ;AND TRY OPENF AGAIN
AP25E1: HRROI T1,[ASCIZ/%LIBOL: OPENF failed for output .MTO file: /]
PSOUT
PUSHJ PP,TYPFER ;WHAT IS IT??
AP25E3: PUSHJ PP,DELOF ;DELETE OLD FILE
MOVE T1,MRLCJF ;AND RELEAS THIS JFN
RLJFN
ERJMP .+1
POPJ PP, ;THEN RETURN
AP25E2: HRROI T1,[ASCIZ/%LIBOL: OPENF failed for output .MTO file
% Someone has it OPEN for updating
/]
PSOUT
JRST AP25E3 ;FINISH ERROR HANDLING
;SOUT FAILED
AP26ER: HRROI T1,[ASCIZ/%LIBOL: SOUT failed trying to append to .MTO file:
/]
PSOUT
PUSHJ PP,TYPFER ;TYPE WHY
JRST DOAP27 ;PRETEND WE ARE AT THE END OF THINGS
>;END IFN TOPS20
>;END IFE FTLSDR
;ROUTINE TO DELETE OLD .TMP FILE
IFE TOPS20,<
DELOF: MOVE T5,[RENAME T1]
OR T5,MRCHNN
SETZB T1,T2
SETZB T3,T4
XCT T5
JRST RENFAI ;RENAME FAILED!
DELOFR: MOVSI T1,(RELEAS)
OR T1,MRCHNN
XCT T1
POPJ PP,
RENFAI: OUTSTR [ASCIZ/%LIBOL: Can't DELETE DSK:/]
OUTSTR MRNMA ;TYPE FILENAME
OUTSTR [ASCIZ/
/]
JRST DELOFR ;GO RELEAS CHANNEL
>;END IFE TOPS20
IFN TOPS20,<
DELOF: MOVE T1,MRJFN ;DELETE THIS FILE
DELF
ERJMP .+2
JRST DELOF1 ;OK, RELEASE JFN AND RETURN
HRROI T1,[ASCIZ/% Can't delete LSTATS .TMP file: /]
PSOUT
PUSHJ PP,TYPFER
DELOF1: MOVE T1,MRJFN
RLJFN
ERJMP .+1 ;IGNORE ERRORS
POPJ PP, ;RETURN
>;END IFN TOPS20
>;END IFN LSTATS
IFE LSTATS,<MROUT.==:RET.1## ;FOR SORT>
END