Google
 

Trailing-Edge - PDP-10 Archives - tops10_tools_bb-fp64b-sb - 10,7/klepto/klepto.mac
There are 3 other files named klepto.mac in the archive. Click here to see a list.
	TITLE	KLEPTO - THE KLEPTOMANIAC (A FAST RIPOFF)
	SUBTTL	DEFINITONS
	SEARCH	MAC10,JOBDAT,UUOSYM,COMMOD
	TWOSEG	760000			;MAKE ROOM FOR LARGE LOWSEG
	SALL				;SAVE THE FORESTS

;STEPHEN M. WOLFE
;MR1-2/S43
;DIGITAL EQUIPMENT CO.
;TECHNICAL SUPPORT GROUP (HOSS)
;200 FOREST ST.
;MARLBORO, MA., 01752
;(617) 467-5583

;AC'S
	F=0				;FLAGS
	T1=1				;TEMPS
	T2=T1+1
	T3=T2+1
	T4=T3+1
	P1=5				;PRESERVED
	P2=P1+1
	P3=P2+1
	P4=P3+1
	U=11				;ADDR OF UDB
	TC=12				;ADDR OF TCB
	FL=13				;ADDR OF FIL
	DDB=14				;ADDR OF DDB
	B=15				;ADDR OF BUFFER
	P=17				;PDL

;FLAGS
	F.DIR==1B35			;RIB OF DIRECTORY
	F.PRM==1B34			;PRIME RIB
	F.RIB==1B33			;ALREADY STEPPED OVER RIB
	F.SUM==1B32			;ALREADY DID CHECKSUM
	F.STR==1B31			;STR DATA FROZEN
	F.HW==1B30			;HIGH WATER
	F.VHW==1B29			;VERY HIGH WATER (PANIC)
	F.P2==1B28			;PASS 2
	F.SAT==1B27			;SAT.SYS
	F.TTY==1B26			;OUTPUT TO PHYSICAL TTY
	F.702==1B25			;7.02 MONITOR (OR LATER)
	F.SHC==1B24			;SAT HAS CHANGED
	F.TTC==1B23			;USE TTCALL
	F.LOK==1B22			;I OWN THE INTERLOCK
	F.NOC==1B21			;NO CTRL-C ALLOWED
	F.CC==1B20			;CTRL-C WAS TYPED
	F.SIL==1B19			;NO OUTPUT TO LOG FILE (TTCALL ONLY)
	F.ALL==1B18			;WE ARE DOING ALL

;ASSEMBLY PARAMETERS
	PAGLSH==11			;BITS PER PAGE
	PAGSIZ==1_PAGLSH		;WORDS PER PAGE
	IFNDEF	BADDR,<BADDR==3>	;NUMBER OF BAD CFPS IT TAKES TO
					;DECLARE THAT DIR IS CLOBBERED
	IFNDEF	UPN,<UPN==2>		;PAGES TO ALLOCATE AT A TIME
	NSTR==^D36			;NUMBER OF STRS
	IFNDEF	NSEC,<NSEC==2>		;SECONDS TO SLEEP
	IFNDEF	MINVRT,<MINVRT=^D512>	;MINIMUM AMOUNT OF SWAP SPACE
	IFNDEF	DOWNN,<DOWNN==5>	;NUMBER OF PAGES TO RELEASE
	PDLSIZ==200			;SIZE OF PDL
	BLKSIZ==200			;SIZE OF A DISK BLOCK
	FOOSIZ==33			;SIZE OF SCRATCH SPACE
	NUNT==^D15			;MAX UNITS IN A STR
	IFG	<1_ACSUN1>-1-NUNT,<PRINTX NUNT TOO SMALL>
	IFNDEF	NSLT,<NSLT==10>		;NUMBER OF COPIES OF KLEPTO
	IFNDEF	NHCOR,<NHCOR==640>	;NUMBER OF WORDS OF HISEG FREE CORE
	IFNDEF	FTCIH,<FTCIH==0>	;CODE IN HISEG
					;THERE IS A BUG IN SETUWP
					;WHICH LEAVES THE HISEG UNCACHED
	IFNDEF	LPTWID,<LPTWID==^D132>	;WIDTH OF LPT
	IFNDEF	LNM,<LNM=='STR'>	;LOGICAL NAME
	IFNDEF	MAXSAF,<MAXSAF==^D500>	;MAX SAFETY BLOCKS PER UNIT
	IFNDEF	NDDBS,<NDDBS==5>	;NUMBER OF DDBS
	IFNDEF	NBLK,<NBLK==^D25>	;NUMBER OF BLOCKS TO READ AT ONCE
	IFNDEF	NSKP,<NSKP==^D9>	;NUMBER OF BLOCKS WE ARE WILLING
					; TO SKIP (CALIBRATED FOR AN RP06)
	IFNDEF	DSKP,<DSKP==^D20>	;BPT ON AN RP06
	IFNDEF	FTDBUG,<FTDBUG==-1>	;DEBUGGING CODE
	IFNDEF	FTCHK,<FTCHK==0>	;CODE TO CHECK TREE CONSISTENCY
	IFNDEF	FTGCHK,<FTGCHK==0>	;CODE TO CHECK CONSISTENCY OF GARBAGE
	IFNDEF	TREG,<TREG==^D20>	;THRESHOLD FOR TYPING REGIONS
	IFNDEF	RMAR,<RMAR==^D15>	;RIGHT MARGIN
	IFNDEF	HWN,<HWN==6>		;HIGH WATER NUMERATOR
	IFNDEF	HWD,<HWD==7>		;HIGH WATER DEMONINATOR
	IFNDEF	VHWN,<VHWN==7>		;VERY HIGH WATER NUMERATOR
	IFNDEF	VHWD,<VHWD==^D8>	;VERY HIGH WATER DEMONINATOR

;OPDEFS
	OPDEF	PJRST[JRST]		;POPPING JRST
	DEFINE	FALL(AA),<IF2 <IFN AA-.,<PRINTX CANNOT FALL TO AA>>>

;CH'S
	TL==16				;TEMP LOG
	TO==17				;LOG FILE

;MONITOR SYMBOLS
	KONCPU==32			;THE 7.01 VERSION OF THIS SYMBOL
	UNI2ND==72			;THE 7.01 VERSION OF THIS SYMBOL

;ABBREVIATIONS:
;CC=CLUSTER COUNT
;CS=CHECK SUM
;CA=CLUSTER ADDRESS
;BN=BLOCK NUMBER (USUALLY RELATIVE TO STR)
;RTP=RETRIEVAL POINTER
;FORMAT OF TCB (TASK CONTROL BLOCK)
	TCBLNK==0			;LH=BACKWARD, RH=FORWARD
	TCBSON==1			;LH=LEFT SON, RH=RIGHT SON
	TCBRNT==2			;LH=ADDR OF PARENT TCB
	TCBBAL==2			;BALANCE FACTOR (HEIGHT OF LEFT SUBTREE
					; MINUS HEIGHT OF RIGHT SUBTREE)
		TCSBAL==3
		TCNBAL==^D20
		BALMO==1		;TOO RIGHT HEAVY
		BALM==2			;RIGHT HEAVY (ACCEPTABLY SO)
		BALZ==3			;PERFECTLY BALANCED
		BALP==4			;LEFT HEAVY (ACCEPTABLY SO)
		BALPO==5		;TOO LEFT HEAVY
	TCBGAR==2			;SIZE (GC ONLY)
		TCSGAR==^D15
		TCNGAR==^D35
		TCMGAR==<1_TCSGAR>-1
	TCBCOD==3			;FUNCTION CODE
		TCSCOD==3
		TCNCOD==2
	TCBRBC==3			;(RIB ONLY) COUNT OF RIBS
		TCSRBC==^D8		;I.E. PRIME RIB IS NUMBER 0
		TCNRBC==^D10
		IFG DESRBC-TCSRBC,<PRINTX TCSRBC TOO SMALL>
	TCBCX==3			;CHANNEL INDEX (OF DDB)
		TCSCX==4		; THIS FIELD ONLY FILLED IN FOR
		TCNCX==^D14		; ONE TCB PER BUFFER
		CXACT==<1_TCSCX>-1	;TRUST ME, I'M ACTIVE
		IFG NDDBS-<CXACT-1>,<PRINTX NDDBS TOO LARGE>
	TCBCOR==3			;(EXTENDED RIB) FILE IS
		TCPCOR==1B15		;CORRUPT, IT HAS FREE
					;AND/OR MULTIPLY USED CLUSTERS
	TCBMBB==3			;MIGHT BE BAD
		TCPMBB==1B16		;THIS BLOCK IS SUSPECTED OF HAVING
					;AN I/O ERROR
	TCBFIL==3			;RH=ADDR OF FIL
	TCBBLK==4			;BN RELATIVE TO STR
	TCBFLR==5			;(EXTENDED RIB) CONTENTS OF RIBFLR
	TCBREL==5			;(SUM+DIR) BN RELATIVE TO FILE
	TCBSUM==6			;(SUM ONLY) CHECKSUM
	TCBSIZ==6			;(EXTENDED RIB) RIBSIZ FROM PRIME RIB
	TCBLFT==7			;(EXTENDED RIB) BLOCKS LEFT IN RIBALC
	TCBCST==10			;(CORRUPT EXTENDED RIB)
					; COPY OF F2CST AND M2CST

;FORMAT OF FIL (FILE BLOCK)
	FILNAM==0			;FILE NAME
	FILEXT==1			;LH=EXTENSION
	FILDAD==FILEXT			;RH=PARENT FIL (0 IF MFD)
	FILCFP==2			;LH=CFP FOR THIS FILE
					;USED IN CHECKING RIBUFD
	FILCNT==2			;USE COUNT
		FISCNT==^D18
		FINCNT==^D35
;FORMAT OF A REG (REGION)
	REGNXT==0			;RH=NEXT REGION
	REGLOW==1			;LOWEST CLUSTER IN REGION
	REGHI==2			;HIGHEST CLUSTER IN REGION

;FORMAT OF A CST (CLUSTER LIST HEADER)
	CSTREG==0			;RH=ADDR OF 1ST REG
	CSTNUM==1			;TOTAL CLUSTERS IN ALL REGIONS
	CSTCNT==2			;TOTAL NUMBER OF REGIONS

;FORMAT OF A DDB
	DDBHDR==0			;RING HEADER (3 WORDS)
	DDBLBN==3			;LOW BN
	DDBHBN==4			;HIGH BN
	DDBPUN==5			;PHYS UNIT NAME CURRENTLY OPEN
	DDBBUF==6			;ADDR OF BUFFER
	DDBCH==7			;CHANNEL NUMBER
	DDBCX==10			;CHANNEL INDEX (INTO USRJDA)
	DDBTCB==11			;1ST TCB
	DDBDON==12			;NON-0 IF I/O DONE
	DDBERR==13			;NON-0 IF I/O ERROR

;FORMAT OF SNF
	SNFNXT==0			;NEXT SNF ON SYSTEM
	SNFLNK==1			;NEXT SNF THIS LIST
	SNFNAM==2			;NAME OF STR
	SNFCHN==3			;BIT MASK OF CHANNELS
	SNFVRT==4			;PAGES OF SWAPPING SPACE (0 IF NOT ASL)

;FORMAT OF A CNF
	CNFNXT==0			;ADDR OF NEXT CNF
	CNFNAM==1			;PHYS NAME OF UNIT
	CNFALT==2			;NAME OF ALTERNATE PORT
	CNFSTR==3			;NAME OF STR
	CNFCHN==4			;BIT MASK OF CHANNELS
	CNFVRT==5			;PAGES OF SWAPPING SPACE (0 IF NOT ASL)

;LOG FILE
	LOGNXT==0			;ADDR OF NEXT LOG
	LOGDEV==1			;STR NAME FILE IS ON
	LOGNAM==2			;FILENAME

;SIZES
	SIZFIL==3			;SIZE OF FIL
	SIZDIR==6			;SIZE OF DIR TCB
	SIZSUM==7			;SIZE OF SUM TCB
	SIZRIB==5			;SIZE OF PRIME RIB TCB
	SIZXRB==10			;SIZE OF EXTENDED RIB TCB
	SIZCXR==SIZXRB+SCSTL		;SIZE OF CORRUPT EXTENDED RIB
	SIZREG==3			;SIZE OF REG
	SIZCST==3			;SIZE OF CST
	SIZDDB==14			;SIZE OF DDB
	SIZGAR==3			;SIZE OF SMALLEST BLOCK
	SIZCNF==6			;SIZE OF CNF
	SIZLOG==3			;SIZE OF LOG FILE
	SIZSNF==5			;SIZE OF SNF
	SUBTTL	EDIT HISTORY

	VWHO==0		;DEC
	VMAJOR==1
	VMINOR==0
	VEDIT==1	;6-8-84/SMW DEVELOPEMENT COMPLETED,
			;COMMENCE EDIT HISTORY

	LOC	.JBVER
	BYTE	(3)VWHO(9)VMAJOR(6)VMINOR(18)VEDIT
	SUBTTL	DATA AREAS
	RELOC	0
FAST:	BLOCK	1			;FIRST WORD IN CORE (UNUSED)
CRSHAC:	BLOCK	20			;ACS SAVED BY DIE
TDEV:	BLOCK	3			;OPEN BLOCK FOR TEMP LOG
LDEV:	BLOCK	3			;OPEN BLOCK FOR LOG FILE
LFIL:	BLOCK	.RBEXT+1		;ENTER BLOCK FOR LOG FILE
TFIL:	BLOCK	.RBEXT+1		;LOOKUP/ENTER BLOCK FOR TEMP LOG
PTH:	BLOCK	.PTMAX			;PATH OF LOG FILE
	PTHL==.-PTH
SVWCNT:	BLOCK	1			;SAVED VALUE OF WRDCNT
WRDCNT:	BLOCK	1			;WORDS ALLOCATED
FREMEM:	BLOCK	1			;1ST FREE CORE BLOCK
SSL:	BLOCK	NSTR*3+.FSDSO		;ARG BLOCK FOR CHANGING SSL
FOO:	BLOCK	FOOSIZ			;SCRATCH SPACE
SYSPPN:	BLOCK	1			;PPN OF SYS
MFDPPN:	BLOCK	1			;PPN OF MFD
FFAPPN:	BLOCK	1			;OPR PPN
ROOT:	BLOCK	1			;ROOT NODE
QTCB:	BLOCK	1			;TCB QUEUE
DEV:	BLOCK	3			;OPEN BLOCK
IBUF:	BLOCK	3			;RING HEADER FOR TEMP LOG
OBUF:	BLOCK	3			;RING HEADER FOR LOG FILE
SAVBUF:	BLOCK	1			;ADDR OF 1ST BUF FOR LOG FILE
HW:	BLOCK	1			;HIGH WATER LEVEL
VHW:	BLOCK	1			;VERY HIGH WATER LEVEL
ZER:!					;ZERO AT BEGINNING OF STR
LSTCST:	BLOCK	SIZCST			;LOST CLUSTERS
FRECST:	BLOCK	SIZCST			;FREE CLUSTERS
MULCST:	BLOCK	SIZCST			;MULTIPLY USED CLUSTERS
BADCST:	BLOCK	SIZCST			;BAD CLUSTERS
					;THIS LIST CONTAINS ONLY THOSE
					;BAD BLOCKS WHICH WERE NOT
					;INCLUDED IN BADBLK.SYS
ZERL==.-ZER				;LENGTH
SCST:!					;ZERO AT BEGINING OF PASS 2 RIB
F2CST:	BLOCK	SIZCST			;FREE CLUSTERS THIS RIB
M2CST:	BLOCK	SIZCST			;MULTIPLY USED CLUSTERS THIS RIB
SCSTL==.-SCST				;LENGTH
HPOS:	BLOCK	1			;HORIZONTAL POSITION OF CURSOR
MARGIN:	BLOCK	1			;VALUE OF HPOS AT RIGHT MARGIN
ACTCNT:	BLOCK	1			;NUMBER OF ACTIVE DDBS
DDBS:	BLOCK	1			;NUMBER OF DDBS TOTAL
DONCNT:	BLOCK	1			;USED BY FNDDON
USRJDA:	BLOCK	NDDBS+1			;ADDR OF DDB
TBUDB:	BLOCK	NUNT			;ADDR OF UDB (NOT IN ORDER)
DEFIN:	BLOCK	.FSNUN			;DEFINE A STR
TABUDB:	BLOCK	NUNT			;ADDR OF UNIT DATA BLOCK
	DEFINL==.-DEFIN			;SIZE OF ARG BLOCK
LOGSTR:	BLOCK	1			;STR LOG FILE IS ON
OURCPU:	BLOCK	1			;CURRENT CPU (BIT MASK)
OVERN:	BLOCK	1			;NUMBER OF BAD FREE CLUSTERS
MYSGN:	BLOCK	1			;MY HISEG NUMBER
MYSLT:	BLOCK	1			;MY SLOT NUMBER
MYJOB:	BLOCK	1			;MY JOB NUMBER
RUNSTR:	BLOCK	1			;STR PROGRAM WAS RUN FROM
LOGS:	BLOCK	1			;ADDR OF 1ST LOG
MONVER:	BLOCK	1			;MONITOR VERSION
MYSNF:	BLOCK	1			;ADDR OF SNF WE ARE DOING
PDL:	BLOCK	PDLSIZ			;PUSH DOWN LIST
CMD:	BLOCK	2			;COMMAND LIST
SAVPC:	BLOCK	1			;COPY OF .EROPC

;THESE LOCATIONS ARE USED AS LOCAL VARIABLES INSIDE DORIB
DRPHY:	BLOCK	1	;BN RELATIVE TO STR
DRBLK:	BLOCK	1	;BN RELATIVE TO STR (WORKING COPY)
DRCA:	BLOCK	1	;CLUSTER ADDRESS
DRCS:	BLOCK	1	;CHECKSUM
DRCC:	BLOCK	1	;CLUSTER COUNT
DRCNT:	BLOCK	1	;BLOCKS LEFT IN THIS CLUSTER
DRLFT:	BLOCK	1	;BLOCKS LEFT TILL RIBALC RUNS OUT
DRREL:	BLOCK	1	;BN RELATIVE TO FILE
DRSIZ:	BLOCK	1	;RIBSIZ IN BLOCKS
	SUBTTL	STRUCTURE DATA BLOCK
	DEFINE	SE(AA),<
	IFN	.-SDB-AA,<PRINTX BB IS WRONG>
	BLOCK	1>

SDB:!
ALIAS:	SE	.FSSNM	;NAME STR MOUNTED AS
NUN:	SE	.FSSNU	;NUMBER OF UNITS
HLBN:	SE	.FSSHL	;HIGHEST LOGICAL BLOCK NUMBER
SSIZ:	SE	.FSSSZ	;SUM OF UDBBPU
	SE	.FSSRQ	;RESERVED QUOTA
	SE	.FSSRF	;RESERVED QUOTA LEFT
STAL:	SE	.FSSTL	;SUM OF UDBTAL
SOVR:	SE	.FSSOD	;OVERDRAW
SPT1:	SE	.FSSMP	;1ST RTP FOR MFD
S1PT:	SE	.FSSML	;NON-0 IF MFD ONLY HAS 1 RTP
SUN1:	SE	.FSSUN	;1ST UNIT FOR MFD
STRY:	SE	.FSSTR	;RETRIES ON ERROR
BIGBPU:	SE	.FSSBU	;BIGGEST UDBBPU
BPSC:	SE	.FSSBC	;BLOCKS PER SUPER CLUSTER
SCPU:	SE	.FSSSU	;SUPER CLUSTERS PER UNIT
	SE	.FSSIG	;IGNORED
CCBP:	SE	.FSSCC	;BP TO CLUSTER COUNT
CSBP:	SE	.FSSCK	;BP TO CHECKSUM
CABP:	SE	.FSSCA	;BP TO CLUSTER ADDRESS
SPVS:	SE	.FSPVT	;NON-0 IF PRIVATE STR
SPPN:	SE	.FSPPN	;OWNER PPN
CRSBN:	SE	.FSSCR	;BN (REL TO STR) OF CRASH.EXE
SK4C:	SE	.FSK4C	;K FOR CRASH
	SZSDB==.-SDB	;PORTION OF SDB PASSED TO STRUUO
BPC:	BLOCK	1	;BLOCKS PER CLUSTER
BPCS:	BLOCK	1	;BITS PER CHECKSUM
MBPCS:	BLOCK	1	;MINUS BITS PER CHECKSUM
BPLU:	BLOCK	1	;BLOCKS PER LARGEST UNIT
HUN:	BLOCK	1	;HIGHEST UNIT NUMBER
STRNAM:	BLOCK	1	;NAME OF STR (IN HOME BLOCKS)
MFDBN:	BLOCK	1	;BN OF RIB OF MFD
WL:	BLOCK	1	;SOME UNIT IN STR IS WRITE LOCKED
SDL:	BLOCK	1	;POSITION IN SDL OR -1
PSSL:	BLOCK	1	;POSITION IN SSL OR 0
SSLSTS:	BLOCK	1	;STATUS OF THIS STR IN SSL
	SUBTTL	UNIT DATA BLOCK
	ZZ==0
	DEFINE	ITM(AA),<
	AA==ZZ
	ZZ==ZZ+1>

	DEFINE	ITEM(AA,BB),<
	IFN	BB-ZZ,<PRINTX BB IS WRONG>
	ITM	AA>

;UNIT DATA BLOCK (UDB)
	ITEM	UDBNAM,.FSUNM		;UNIT NAME
	ITEM	UDBHID,.FSUID		;PACK SERIAL NUMBER (SIXBIT)
	ITEM	UDBLOG,.FSULN		;LOGICAL UNIT NAME (E.G. DSKB0)
	ITEM	UDBLUN,.FSULU		;LOGICAL UNIT NUMBER
	ITEM	UDBAWL,.FSUDS		;STATUS (E.G. SOFWARE WRITE-LOCK)
	ITEM	UDBGRP,.FSUGP		;BLOCKS TO TRY ALLOCATING ON OUTPUT
	ITEM	UDBTAL,.FSUTL		;BLOCKS LEFT (MINUS SAFETY FACTOR)
	ITEM	UDBBPC,.FSUBC		;BLOCKS PER CLUSTER
	ITEM	UDBCPS,.FSUCS		;CLUSTERS PER SAT
	ITEM	UDBWPS,.FSUWS		;WORDS PER SAT
	ITEM	UDBSIC,.FSUSC		;SATS IN CORE
	ITEM	UDBSPU,.FSUSU		;SATS PER UNIT
	ITEM	UDBSPT,.FSUSP		;POINTER TO SPT
	ITEM	UDBSLB,.FSUSB		;1ST BLOCK FOR SWAPPING
	ITEM	UDBK4S,.FSUKS		;K FOR SWAPPING
	SZUDB==ZZ			;PORTION OF UDB PASSED TO STRUUO
	ITM	UDBBLK			;BN WHERE UNIT START (RELATIVE TO STR)
	ITM	UDBSSF			;ENTRIES IN SPT SO FAR
	ITM	UDBBPU			;BLOCKS PER UNIT
	ITM	UDBHLB			;HIGHEST LEGAL BLOCK
	ITM	UDBCYL			;BLOCKS PER CYLINDER
	ITM	UDBBPT			;BLOCKS PER TRACK
	ITM	UDBPST			;POINTER TO TABLE OF POINTERS
					;INDEX BY SAT NUMBER, TABLE GIVES
					;POINTER TO SAT BLOCK
	ITM	UDBHWP			;NON-0 IF HARDWARE WRITE PROTECT
	ITM	UDBASL			;POSITION IN ASL OR -1
	ITM	UDBALT			;NAME OF ALTERNATE PORT
	ITM	UDBCPU			;CPU SPECIFICATION (7.01 ONLY)
	ITM	UDBSKP			;BLOCKS TO SKIP
	SIZUDB==ZZ			;SIZE OF UDB
	SUBTTL	SHARED WRITEABLE HISEG
	RELOC				;TO HISEG

JBTJOB:	BLOCK	NSLT			;TABLE OF JOB NUMBERS
JBTCHN:	BLOCK	NSLT			;BIT MASK OF CHANNELS
LOKJOB:	0				;JOB THAT OWNS INTERLOCK
ILOCK:	-1				;THE INTERLOCK
BLST:	BLOCK	1			;LIST OF UNPROCESSED STRS
ALST:	BLOCK	1			;LIST OF STRS ALREADY DONE
SLST:	BLOCK	1			;LIST OF ALL STRS ON SYSTEM
JBTLOG:	BLOCK	NSLT			;STR NAME LOG IS OPEN ON
JBTSTR:	BLOCK	NSLT			;STR NAME IN PROGRESS
CNTSLT:	0				;NUMBER OF SLOTS IN USE
CONFIG:	BLOCK	1			;POINTER TO LIST OF CNF'S
CHNNAM:	BLOCK	^D36+1			;TABLE OF NAMES OF CHANNELS
HFF:	BLOCK	1			;FIRST FREE ADDR IN HISEG
HCOR:	BLOCK	NHCOR			;HISEG FREE CORE
	SUBTTL	IMPURE DATA

	RELOC				;BACK TO LOWSEG
IFN FTCIH,<
LOW:!	RELOC				;BACK TO HISEG
HI:	PHASE	LOW
>
INTR:	XWD	4,TRAP			;INTERRUPT BLOCK
	ER.ICC				;CTRL-C ONLY
	BLOCK	2

;END OF IMPURE CODE
IFN FTCIH,<
	DEPHASE
	LOWL==.-HI			;SIZE OF IMPURE DATA
	RELOC				;BACK TO LOWSEG
	BLOCK	LOWL			;ALLOCATE SPACE
	RELOC				;BACK TO HISEG
>
	SUBTTL	BYTE POINTERS

FIYCNT:	POINT	FISCNT,FILCNT(FL),FINCNT ;USE COUNT
FIZCNT:	POINT	FISCNT,FILCNT(P3),FINCNT ;USE COUNT
RIYRBA:	POINT	DESRBA,RIBXRA(B),DENRBA	;RIB ADDRESS
RIYRBU:	POINT	DESRBU,RIBXRA(B),DENRBU	;RIB UNIT
TCWCOD:	POINT	TCSCOD,TCBCOD(T1),TCNCOD ;FUNCTION CODE
TCXCOD:	POINT	TCSCOD,TCBCOD(T2),TCNCOD ;FUNCTION CODE
TCZCOD:	POINT	TCSCOD,TCBCOD(P2),TCNCOD ;FUNCTION CODE
TCYCOD:	POINT	TCSCOD,TCBCOD(TC),TCNCOD ;FUNCTION CODE
TCXRBC:	POINT	TCSRBC,TCBRBC(T2),TCNRBC ;RIB COUNT
TCYRBC:	POINT	TCSRBC,TCBRBC(TC),TCNRBC ;RIB COUNT
TCZRBC:	POINT	TCSRBC,TCBRBC(P2),TCNRBC ;RIB COUNT
BAYNBB:	POINT	BASNBB,BAFNBB(P2),BANNBB ;NUMBER BAD BLOCKS IN REGION-1
BAYNBR:	POINT	BASNBR,BAFNBR(B),BANNBR	;NUMBER OF BAD REGIONS
TCXBAL:	POINT	TCSBAL,TCBBAL(T1),TCNBAL ;BALANCE FACTOR
TCZBAL:	POINT	TCSBAL,TCBBAL(T2),TCNBAL ;BALANCE FACTOR
TCWBAL:	POINT	TCSBAL,TCBBAL(T3),TCNBAL ;BALANCE FACTOR
TCVBAL:	POINT	TCSBAL,TCBBAL(T4),TCNBAL ;BALANCE FACTOR
TCYCX:	POINT	TCSCX,TCBCX(TC),TCNCX	;CHANNEL INDEX
TCXCX:	POINT	TCSCX,TCBCX(T1),TCNCX	;CHANNEL INDEX
TCZCX:	POINT	TCSCX,TCBCX(T2),TCNCX	;CHANNEL INDEX
DEYSIZ:	PNTR	@DDBBUF(DDB),BF.SIZ	;SIZE OF BUFFER
TCXGAR:	POINT	TCSGAR,TCBGAR(T2),TCNGAR ;SIZE OF GARBAGE
TCZGAR:	POINT	TCSGAR,TCBGAR(T4),TCNGAR ;SIZE OF GARBAGE
TCWGAR:	POINT	TCSGAR,TCBGAR(T3),TCNGAR ;SIZE OF GARBAGE
SPYCLA:	POINT	CLASIZ,(T2),CLAPOS	;CA PART OF SPT
SPYTAL:	POINT	TALSIZ,(T2),TALPOS	;CC PART OF SPT
	SUBTTL	INITIALIZATION

KLEPTO:	JFCL				;NO CCL
	RESET				;STOP I/O
	MOVE	P,[IOWD PDLSIZ,PDL]	;SETUP PDL
	SETZ	F,			;SETUP FLAGS
IFN FTCIH,<
	MOVE	T1,[XWD HI,LOW]		;COPY IMPURE DATA
	BLT	T1,LOW+LOWL-1
>
	MOVEI	T1,INTR			;CTRL-C TRAP
	HRRM	T1,.JBINT
	SETZM	WRDCNT			;NO CORE ALLOCATED YET
	SETZM	FREMEM
	PUSHJ	P,MAKDDB		;BUILD DDBS
	SETOM	OURCPU			;ASSUME ALL CPUS
	MOVE	T1,[%LDMFD]		;PPN OF MFD
	GETTAB	T1,
	 PUSHJ	P,DIE
	MOVEM	T1,MFDPPN
	MOVE	T1,[%LDFFA]		;PPN OF OPR
	GETTAB	T1,
	 PUSHJ	P,DIE
	MOVEM	T1,FFAPPN
	HRROI	T1,.GTPPN		;OUR PPN
	GETTAB	T1,
	 PUSHJ	P,DIE
	CAME	T1,FFAPPN		;ARE WE THE OPR?
	JRST	NOTOPR			;NO
	MOVE	T1,[%LDSYS]		;PPN OF SYS
	GETTAB	T1,
	 PUSHJ	P,DIE
	MOVEM	T1,SYSPPN
	PJOB	T1,			;MY JOB NUMBER
	MOVEM	T1,MYJOB
	HRROI	T1,.GTSGN		;MY HISEG NUMBER
	GETTAB	T1,
	 PUSHJ	P,DIE
	HRRZM	T1,MYSGN
	SETZ	T1,			;WRITE ENABLE THE HISEG
	SETUWP	T1,
	 PUSHJ	P,DIE
	PUSHJ	P,GETLOK		;GET INTERLOCK
	PUSHJ	P,ALIVE			;MAKE SURE EVERYBODY ALIVE
	PUSHJ	P,GETSLT		;GET A SLOT NUMBER
	PUSHJ	P,LUNIT			;BUILD LIST OF UNITS
	PUSHJ	P,BDALL			;BUILD LIST OF STRS
	PUSHJ	P,GIVLOK		;GIVE UP INTERLOCK
	PUSHJ	P,OPENLG		;OPEN DEVICE FOR LOG
	PUSHJ	P,SWATER		;SET WATER LEVEL
	SETZM	LOGS			;NO TMP LOGS YET
	MOVSI	T1,LNM			;ALL?
	DEVNAM	T1,
	 JRST	NOSTR
	CAMN	T1,[SIXBIT /ALL/]
	JRST	DOALL			;YES
	PUSHJ	P,GETLOK		;GET INTERLOC
	PUSHJ	P,PIKONE		;TELL THE WORLD WHICH STR
	 JRST	XIT
	PUSHJ	P,PIKLOG		;PICK STR FOR LOG
	PUSHJ	P,GIVLOK		;GIVE UP INTERLOCK
	PUSHJ	P,DOSTR			;PROCESS THE STR
	PUSHJ	P,GETLOK		;GET INTERLOCK BACK
	PUSHJ	P,ALIVE			;CHECK IF EVERYBODY ALIVE
	PUSHJ	P,DONSTR		;TELL THE WORLD WE ARE DONE
	JRST	XIT			;EXIT
;HERE TO DO ALL THE STR'S ON THE SYSTEM
DOALL:	HRROI	T1,.GTRDV		;GET STR RUN FROM
	GETTAB	T1,
	 PUSHJ	P,DIE
	MOVEM	T1,RUNSTR
	PUSHJ	P,GETLOK		;GET INTERLOCK
	TRO	F,F.ALL			;WE ARE DOINT ALL
DOALL1:	PUSHJ	P,PIKSTR		;PICK A STR
	 JRST	XIT			;NONE
	PUSHJ	P,PIKLOG		;PICK A STR FOR LOG FILE
	PUSHJ	P,GIVLOK		;GIVE UP INTERLOCK
	PUSHJ	P,GC			;COLLECT GARBAGE
	PUSHJ	P,CDOWN			;GIVE CORE AWAY
	PUSHJ	P,DOSTR			;PROCESS THE STR
	PUSHJ	P,GETLOK		;GET INTERLOCK
	PUSHJ	P,ALIVE			;SEE IF ANYBODY DIED
	PUSHJ	P,DONSTR		;FLAG THAT STR IS DONE
	PUSHJ	P,CPYLG			;COPY TMP LOG
	 JFCL
	SETZM	RUNSTR			;OK TO DO RUN STR NOW
	JRST	DOALL1			;LOOP

;HERE IF NOT OPR
NOTOPR:	OUTSTR	[ASCIZ /Must be OPR
/]
	EXIT

;HERE IF LOGICAL NAME ISN'T ASSIGNED
NOSTR:	OUTSTR	[ASCIZ /STR isn't assigned
/]
	PUSHJ	P,GETLOK		;GET INTERLOCK
XIT:	PUSHJ	P,CPYLG			;COPY TMP LOG
	 JRST	XIT2			;DO IT LATER
	PUSHJ	P,GIVSLT		;GIVE UP SLOT NUMBER
	PUSHJ	P,GIVLOK		;GIVE UP INTERLOCK
	EXIT
XIT2:	PUSHJ	P,SLEEPY		;SLEEP AWHILE
	JRST	XIT
	SUBTTL	MAINSTREAM

;ROUTINE TO PROCESS A STR
DOSTR:	MOVE	T1,WRDCNT		;SAVE WORD COUNT
	MOVEM	T1,SVWCNT
	PUSHJ	P,BEGIN			;TELL WHICH STR WE ARE DOING
	MOVE	T1,[XWD QTCB,QTCB]	;QUEUE IS INITIALY EMPTY
	MOVEM	T1,QTCB
	SETZM	ROOT
	SETZM	ZER			;ZERO AT BEGINNING OF STR
	MOVE	T1,[XWD ZER,ZER+1]
	BLT	T1,ZER+ZERL-1
	TRZ	F,F.P2			;PASS 1
	PUSHJ	P,PUNT			;COMPILE TABLE OF UNITS
	PUSHJ	P,RDHOM			;READ HOME BLOCKS
	 JRST	BADHOM
	PUSHJ	P,RSTR			;REMOVE STR
	 JRST	DOSTR1
	PUSHJ	P,BDMFD			;BUILD TCB FOR MFD
DOPASS:	HRRZ	T1,QTCB			;CRANK UP I/O
	PUSHJ	P,CRANK0
	 PUSHJ	P,DIE
	HRRZ	TC,QTCB			;1ST TCB
LOOP:	LDB	T1,TCYCOD		;DISPATCH
	PUSHJ	P,@FUNX(T1)
	PUSHJ	P,SCHED			;PICK NEXT TCB
	 SETZ	T1,			;DONE WITH PASS 1
	PUSH	P,T1			;SAVE ADDR OF NEXT TCB
	PUSHJ	P,DECFIL		;DECREMENT USE COUNT ON FIL
	PUSHJ	P,UNLINK		;UNLINK TCB FROM TREE
	PUSHJ	P,UNLNK			;UNLINK TCB FROM QUEUE
IFN FTCHK,<
	PUSHJ	P,CHECK			;CHECK CONSISTENCY
>
	PUSHJ	P,GIVTCB		;RETURN TCB TO FREE LIST
	POP	P,TC			;RESTORE ADDR OF NEXT TCB
	JUMPN	TC,LOOP

;HERE WHEN DONE WITH A COMPLETE PASS OF THE STR
	TRNE	F,F.P2			;DONE WITH PASS 1 OR 2?
	JRST	DONE2			;PASS 2
;HERE WHEN DONE WITH PASS ONE
IFN FTDBUG,<
	HRRZ	T1,QTCB			;FORGOT ANY TCBS?
	CAIE	T1,QTCB
	PUSHJ	P,DIE
>
	PUSHJ	P,DOSAT			;PROCESS SAT BLOCKS
	 JRST	DOSTR4			;CAN'T FIND SATS
	PUSHJ	P,CSTSO			;PRINT CST'S
	PUSHJ	P,BADOVR		;OVERLAP OF FREE AND BAD
	MOVEM	T1,OVERN
	PUSHJ	P,DEBAD			;DEALLOCATE BAD REGIONS
	PUSHJ	P,DELST			;DEALLOCATE LOST REGIONS
	PUSHJ	P,DEFRE			;DEALLOCATE FREE REGIONS
	SKIPN	T1,OVERN		;ANY OVERLAP?
	JRST	NOVR			;NO
	MOVEI	T2,[ASCIZ /All/]	;TOTAL OVERLAP?
	CAME	T1,FRECST+CSTNUM
	MOVEI	T2,[ASCIZ /Some/]	;NO, PARTIAL OVERLAP
	PUSHJ	P,STRO
	MOVEI	T2,[ASCIZ / free clusters are bad blocks/]
	PUSHJ	P,STRDSP
NOVR:	MOVE	T1,FRECST+CSTNUM	;FREE CLUSTERS
	SUB	T1,OVERN		;MINUS BAD FREE
	ADD	T1,MULCST+CSTNUM	;MULTIPLY USED CLUSTERS
	JUMPN	T1,DOP2			;GO IF NEED PASS TWO
	MOVEI	T2,[ASCIZ /No need for pass two/]
	PUSHJ	P,STRDSP
	JRST	DONE2			;DONE
;HERE TO START PASS TWO
DOP2:	MOVEI	T2,MULCST		;MARK MULTIPLY USED CLUSTERS IN
	PUSHJ	P,MARKC			; SAT SO THEY ARE EASY TO SPOT
	TRO	F,F.P2			;FLAG PASS TWO
	MOVEI	T2,[ASCIZ /Begining pass two/]
	PUSHJ	P,STRDSP
	PUSHJ	P,BDMFD			;BUILD TCB FOR MFD
	JRST	DOPASS

DOSTR1:	MOVEI	T2,[ASCIZ /Cannot remove STR, aborting/]
	JRST	DOSTR2

BADHOM:	MOVEI	T2,[ASCIZ /Error while reading home blocks, aborting/]
DOSTR2:	PUSHJ	P,STRDSP
	JRST	DOSTR4

;HERE WHEN DONE WITH PASS TWO
DONE2:	PUSHJ	P,DEMUL			;DEALLOCATE MULTIPLY USED REGIONS
	PUSHJ	P,ASTR			;PUT STR BACK
	 PUSHJ	P,DOSTR3
DOSTR4:	PUSHJ	P,DESTR			;DEALLOCATE CORE
	MOVE	T1,WRDCNT		;MISS SOMETHING?
	CAME	T1,SVWCNT
	PUSHJ	P,DIE			;BUG
	POPJ	P,

DOSTR3:	MOVEI	T2,[ASCIZ /Error while mounting STR/]
STRDSP:	PUSHJ	P,STRO
	PJRST	DSPACE
	SUBTTL	SCHEDULER

;TC PASSES ADDR OF PREVIOUS TCB
;T1 RETURNS ADDR OF NEXT TCB
;NOSKIP IF QUEUE IS EMPTY
SCHED:	LDB	T2,TCYCX		;FIND DDB
	MOVE	DDB,USRJDA(T2)
	MOVE	T1,TC			;START AT CURRENT TCB
SCHED3:	HRRZ	T1,TCBLNK(T1)		;GET NEXT TCB
	CAIN	T1,QTCB
	JRST	SCHED0
	MOVE	T2,TCBBLK(T1)		;IS IT IN CACHE?
	CAMLE	T2,DDBHBN(DDB)
	JRST	SCHED0			;NO
	PUSHJ	P,FITP			;FIT IN CORE?
	 JRST	SCHED3			;NO
SCHED2:	LDB	T2,TCYCX		;COPY CHANNEL INDEX
	DPB	T2,TCXCX
	JRST	CPOPJ1
SCHED0:	MOVE	T1,TC			;BACK TO BEGINING
SCHED4:	HLRZ	T1,TCBLNK(T1)		;GET PREVIOUS TCB
	CAIN	T1,QTCB
	JRST	SCHED1
	MOVE	T2,TCBBLK(T1)		;IS IT IN CACHE?
	CAMGE	T2,DDBLBN(DDB)
	JRST	SCHED1			;NO
	PUSHJ	P,FITP			;FIT IN CORE?
	 JRST	SCHED4			;NO
	JRST	SCHED2			;DO IT

;HERE IF CURRENT DDB IS EXHAUSTED
SCHED1:	SOS	ACTCNT			;ONE LESS USEFUL DDB
	MOVEI	T1,CXACT		;MARK ACTIVE BEFORE INVALIDATE
	DPB	T1,TCYCX
	SETOM	DDBHBN(DDB)		;INVALIDATE CACHE
	PUSHJ	P,CRANK			;CRANK UP SOME I/O
	 POPJ	P,			;NOTHING TO CRANK UP
	MOVE	T1,DDBTCB(DDB)		;1ST TCB IN THAT DDB
	JRST	CPOPJ1
;ROUTINE TO CRANK UP I/O:
;START UP A TRANSFER ON THIS DDB,
;AND WAIT FOR I/O TO FINISH ON SOME (OTHER) DDB.
;DDB PASSES CURRENT DDB
;TC PASSES CURRENT TCB
;SKIP IF THERE IS INDEED STUFF TO DO
CRANK:	PUSHJ	P,NEXT			;FIND A TCB TO START
	 JRST	CRANK1			;NONE
CRANK0:	PUSH	P,TC			;SAVE CURRENT TCB
	MOVE	TC,T1			;TCB TO START
	PUSHJ	P,LODTCB		;START IT
	POP	P,TC			;RESTORE CURRENT TCB
	MOVE	T1,ACTCNT		;ALL DDBS ACTIVE?
	CAMN	T1,DDBS
	JRST	CRANK2			;YES
	PUSHJ	P,FNDIDL		;NO, FIND ONE THAT'S IDLE
	JRST	CRANK			;START SOMETHING ON IT
CRANK1:	SETOM	DDBHBN(DDB)		;INVALIDATE CACHE
	SKIPN	ACTCNT			;ANY DDBS ACTIVE?
	POPJ	P,			;NO
CRANK2:	PUSHJ	P,FNDDON		;WAIT FOR SOME DDB TO FINISH
	JRST	CPOPJ1
;FIND AN IDLE DDB
;DDB RETURNS DDB
FNDIDL:	MOVEI	T1,1
FNDID1:	MOVE	DDB,USRJDA(T1)
	SKIPL	DDBHBN(DDB)
	AOJA	T1,FNDID1
	POPJ	P,

;WAIT TILL SOME DDB FINISHES I/O
;DDB RETURNS DDB
FNDDON:	PUSHJ	P,SAVE1			;SAVE AC
FNDDN1:	MOVE	P1,DDBS			;THIS MANY DDBS TO TEST
FNDDN2:	AOS	T1,DONCNT		;NEXT DDB TO TEST
	IDIV	T1,DDBS
	MOVE	DDB,USRJDA+1(T2)
	SKIPGE	DDBHBN(DDB)		;I/O ALREADY STARTED?
	JRST	FNDDN3			;NO
	SKIPN	DDBDON(DDB)		;I/O ALREADY DONE?
	PUSHJ	P,DOIN			;OR JUST FINISHED NOW?
	 POPJ	P,			;YES, A WINNER
FNDDN3:	SOJG	P1,FNDDN2		;TEST EACH DDB
	MOVE	T1,[HB.RIO+HB.RWJ+^D1000] ;NONE DONE, SLEEP AWHILE
	HIBER	T1,
	 HALT
	JRST	FNDDN1			;TRY AGAIN

;ROUTINE TO FIND THE CX OF A TCB (IF ANY)
;T1 PASSES TCB
;T2 RETURNS CX
;CPOPJ IF I/O NOT IN PROGRESS
;CPOPJ1 IF FOUND
FNDCX:	MOVE	T2,DDBS			;THIS MANY DDBS TO TEST
	MOVE	T3,TCBBLK(T1)		;BN OF THIS TCB
FNDCX1:	MOVE	T4,USRJDA(T2)		;ADDR OF DDB
	CAML	T3,DDBLBN(T4)		;BN IN CACHE?
	CAMLE	T3,DDBHBN(T4)
	SOJG	T2,FNDCX1		;NO, TEST NEXT DDB
	SKIPE	T2			;FOUND IT?
	AOS	(P)			;YES, SKIP
	POPJ	P,
;FIND NEXT TCB TO START I/O ON
;TC PASSES CURRENT TCB
;T1 RETURNS NEXT TCB
;CPOPJ1 IF OK
;CPOPJ IF NO TCB'S STARTABLE
NEXT:	PUSHJ	P,SAVE2			;SAVE AC
	PUSHJ	P,TWATER		;TEST WATER LEVEL
	MOVE	T2,TCBBLK(TC)		;FIND UDB
	IDIV	T2,BPLU
	HRRZ	U,TABUDB(T2)
	IDIV	T3,UDBCYL(U)		;CYL NUMBER
	IMUL	T3,UDBCYL(U)		;1ST BN THIS CYL
	ADD	T3,UDBBLK(U)		;BN RELATIVE TO STR
	MOVE	P1,T3			;SAVE IT IN A SAFE PLACE
	MOVE	P2,P1			;1ST BN ON NEXT CYL
	ADD	P2,UDBCYL(U)
NEXT14:	MOVE	T1,TC			;FIND LAST TCB IN PREVIOUS CYL
NEXT1:	HLRZ	T1,TCBLNK(T1)
	CAIN	T1,QTCB
	JRST	NEXT3
	CAMG	P1,TCBBLK(T1)
	JRST	NEXT1
;HERE TO FIND NEXT TCB THAT DOES NOT HAVE I/O ACTIVE
NEXT3:	HRRZ	T1,TCBLNK(T1)		;STEP TO NEXT TCB
	CAIN	T1,QTCB
	JRST	NEXT10			;NONE
NEXT13:	LDB	T2,TCXCX		;GET CHANNEL INDEX
	CAIN	T2,CXACT		;RESTARTING CURRENT DDB?
	JRST	NEXT3			;YES, PITCH IT
	JUMPN	T2,NEXT4		;I/O ACTIVE IF NON-0
	PUSHJ	P,FNDCX			;DON'T KNOW, SEARCH AND FIND OUT
	 JRST	NEXT6			;NOT ACTIVE
NEXT4:	MOVE	T2,USRJDA(T2)		;GET DDB ADDR
	MOVE	T2,DDBHBN(T2)		;HIGHEST BN IN TRANSIT
NEXT5:	HRRZ	T1,TCBLNK(T1)		;FIND 1ST TCB BEYOND THAT BN
	CAIN	T1,QTCB
	JRST	NEXT10
	CAML	T2,TCBBLK(T1)
	JRST	NEXT5
	JRST	NEXT13			;SEE IF IT'S ACTIVE
;HERE WITH T1=1ST TCB THAT DOES NOT HAVE I/O ACTIVE
NEXT6:	CAMG	P2,TCBBLK(T1)		;SAME CYL?
	JRST	NEXT7			;NO
	PUSHJ	P,FITP			;WILL TCB FIT IN CORE?
	 JRST	NEXT3			;NO
	JRST	CPOPJ1			;YES, A WINNER
;HERE IF MUST MOVE HEADS IN
;HERE WITH T1=1ST TCB ON NEXT CYL
NEXT7:	SKIPN	ACTCNT			;I/O ACTIVE ON THIS CYL?
	JRST	NEXT9			;NO, MOVE HEADS NOW
	POPJ	P,			;YES, DON'T MOVE TILL I/O DONE
NEXT8:	HRRZ	T1,TCBLNK(T1)
	CAIN	T1,QTCB
	JRST	NEXT11
;HERE IF MUST MOVE HEADS, FIND ANY TCB THAT WILL FIT IN CORE
NEXT9:	PUSHJ	P,FITP			;THIS ONE FIT?
	 JRST	NEXT8			;NO
	JRST	CPOPJ1			;YES, A WINNER
;HERE IF MUST MOVE HEADS OUT
;HERE WITH T1=QTCB
NEXT10:	SKIPE	ACTCNT			;I/O ACTIVE ON THIS CYL?
	POPJ	P,			;YES, DON'T MOVE TILL I/O DONE
;HERE IF MUST MOVE HEADS, FIND ANY TCB THAT WILL FIT IN CORE
NEXT11:	HRRZ	T1,TCBLNK(T1)		;STEP TO NEXT TCB
	CAMG	P1,TCBBLK(T1)		;BACK TO ORIGINAL CYL?
	JRST	NEXT15			;YES, NOTHING WE CAN DO
	PUSHJ	P,FITP			;THIS ONE FIT?
	 JRST	NEXT11			;NO
	JRST	CPOPJ1			;YES, A WINNER

;HERE IF ABSOLUTELY NOTHING TO DO
NEXT15:	TRZN	F,F.VHW			;IGNORE VHW (MAY GO VIRTUAL)
	TRZE	F,F.HW			;DIDN'T WORK, TRY IGNORING HW
	JRST	NEXT14
	POPJ	P,
	SUBTTL	WATER LEVEL

;ROUTINE TO SET THE WATER LEVEL(S)
;BY "HIGH WATER" WE MEAN THAT WE'RE LOW ON CORE AND SHOULD
;TRY TO BE MORE CONSERVATIVE.
SWATER:	MOVE	T2,[%NSCMX]		;GET CORMAX
	GETTAB	T2,
	 PUSHJ	P,DIE
	HRROI	T3,.GTCVL		;CURRENT PHYS LIMIT
	GETTAB	T3,
	 PUSHJ	P,DIE
	ANDI	T3,77777
	LSH	T3,PAGLSH
	CAMLE	T2,T3			;TAKE THE LOWER OF THE TWO
	MOVE	T2,T3
	HRRZ	T3,.JBHRL		;HIGHEST ADDR IN HISEG
	CAMLE	T2,T3			;TAKE THE LOWER OF THE TWO
	MOVE	T2,T3
	HLRZ	T3,.JBHRL		;MINUS SIZE OF HISEG
	SUB	T2,T3
	SUB	T2,.JBFF		;MINUS STUFF ALREADY USED
	ADD	T2,WRDCNT
	MOVE	T1,T2			;SAVE IT
	IMULI	T2,HWN			;COMPUTE FRACTION
	IDIVI	T2,HWD
	MOVEM	T2,HW			;HIGH WATER
	IMULI	T1,VHWN			;COMPUTE FRACTION
	IDIVI	T1,VHWD
	MOVEM	T1,VHW			;VERY HIGH WATER
	POPJ	P,

;ROUTINE TO TEST THE WATER LEVEL AND SET THE BITS IN F
TWATER:	MOVE	T1,WRDCNT		;HIGH WATER?
	CAMGE	T1,HW
	TRZA	F,F.HW+F.VHW		;NO
	TROA	F,F.HW			;YES
	POPJ	P,
	CAMGE	T1,VHW			;VERY HIGH WATER?
	TRZA	F,F.VHW			;NO
	TRO	F,F.VHW			;YES
	POPJ	P,
;ROUTINE TO TEST IF THERE'S ENOUGH CORE TO PROCESS A GIVEN TCB
;T1 PASSES ADDR OF TCB
;SKIP IF THERE'S ENOUGH CORE
;FNCSUM ALWAYS MAKES CORE SIZE SMALLER SO THEY'RE ALWAYS SAFE.
;FNCDIR ALWAYS MAKES YOU BIGGER SO NEVER DO THOSE IF YOU'RE LOW ON CORE.
;FNCRIB SOMETIMES MAKES YOU BIGGER AND SOMETIMES SMALLER, IT TAKES
;EXTENSIVE CHECKING.
;DIRECTORY RIBS ALWAYS MAKE YOU BIGGER.
;OTHER RIBS SOMETIMES MAKE YOU BIGGER AND SOMETIMES SMALLER, DEPENDING
;ON HOW MANY RTP'S HAVE TO BE CHECKSUMED. THERE'S NO WAY TO TELL
;WITHOUT ACTUALLY READING THE RIB. BUT MOST RIBS ONLY HAVE ONE RTP,
;AND WILL PROBABLY MAKE YOU SMALLER. WE TEND TO GIVE RIBS THE BENEFIT
;OF A DOUBT AND ASSUME THEY MAKE US SMALLER. IF, HOWEVER, WE'RE AT
;THE VERY HIGH WATER MARK, THEN BE VERY CONSERVATIVE AND DON'T PROCESS
;ANY RIBS. WE ASSUME THAT THE ONLY WAY TO GET TO THE VERY HIGH WATER MARK
;IS BY HAVING LOTS OF RIBS WITH LOTS OF RTP'S. THUS THERE MUST BE LOTS
;OF CHECKSUMS IN THE QUEUE. PROCESS ONLY THE CHECKSUMS.
FITP:	TRNN	F,F.HW			;HIGH WATER MARK?
	JRST	CPOPJ1			;NO, IT'S SAFE
	LDB	T2,TCWCOD		;YES, BETTER CHECK IT
	PJRST	@FUNHW(T2)

;ROUTINE TO TEST IF THERE'S ENOUGH CORE TO PROCESS A RIB
HWRIB:	HRRZ	T2,TCBFIL(T1)		;DIR?
	HLRZ	T2,FILEXT(T2)
	CAIE	T2,'SFD'
	CAIN	T2,'UFD'
	POPJ	P,			;YES, DON'T PROCESS
	TRNN	F,F.VHW			;VERY HIGH WATER?
	AOS	(P)			;NO, GIVE HIM BENEFIT OF DOUBT
	POPJ	P,			;YES, DON'T PROCESS
	SUBTTL	DISPATCH TABLES

	DEFINE	FUNC,<
	XX	FNCRIB,DORIB,SIZRIB,GIVRIB,HWRIB
	XX	FNCDIR,DODIR,SIZDIR,GIVBLK,CPOPJ
	XX	FNCSUM,DOSUM,SIZSUM,GIVBLK,CPOPJ1
>

	DEFINE	XX(AA,BB,CC,DD,EE),<
	AA==.-FUNX
	BB
>
FUNX:	FUNC

	DEFINE	XX(AA,BB,CC,DD,EE),<CC>
FUNSZ:	FUNC

	DEFINE	XX(AA,BB,CC,DD,EE),<DD>
FUNGV:	FUNC

	DEFINE	XX(AA,BB,CC,DD,EE),<EE>
FUNHW:	FUNC
	SUBTTL	PROCESS A RIB

DORIB:	PUSHJ	P,SAVE1			;PRESERVE ACS
	TRZ	F,F.DIR+F.RIB+F.PRM+F.SAT ;CLEAR FLAGS
	LDB	T1,TCYRBC		;PRIME RIB?
	SKIPN	T1
	TRO	F,F.PRM			;YES, LIGHT FLAG
	TRNN	F,F.P2			;WHICH PASS?
	JRST	DORIB5			;ONE
	SETZM	SCST			;TWO, ZERO CST
	MOVE	T1,[XWD SCST,SCST+1]
	BLT	T1,SCST+SCSTL-1
	MOVE	T1,TCBCOR(TC)		;SAVED CST?
	TRNN	F,F.PRM
	TLNN	T1,(TCPCOR)
	JRST	DORIB5			;NO
	MOVEI	T1,SCST			;YES, RESTORE IT
	HRLI	T1,TCBCST(TC)
	BLT	T1,SCST+SCSTL-1
DORIB5:	PUSHJ	P,REDTCB		;READ THE BLOCK
	 JRST	DORB10			;I/O ERROR
	PUSHJ	P,RBCK			;CHECK REASONABLENESS
	 JRST	DORB11
	MOVE	P1,RIBFIR(B)		;AOBJN TO RTP'S
	ADD	P1,B
	HRRZ	T1,FILDAD(FL)		;GO IN NOT MFD
	JUMPN	T1,DORIB6
	TRNN	F,F.PRM			;GO UNLESS PRIME RIB
	JRST	DORIB6
	SETZM	S1PT			;MFD ONLY HAS ONE RTP?
	SKIPN	2(P1)
	SETOM	S1PT			;YES
DORIB6:	HLRZ	T1,FILEXT(FL)		;DIRECTORY?
	CAIE	T1,'UFD'
	CAIN	T1,'SFD'
	TRO	F,F.DIR			;YES, SET FLAG
	CAIN	T1,'SYS'		;SET FLAG IF SAT.SYS
	TRO	F,F.SAT
	MOVS	T1,RIBNAM(B)
	MOVE	T2,RIBPPN(B)
	CAIN	T1,'SAT'
	CAME	T2,SYSPPN
	TRZ	F,F.SAT
	TRNN	F,F.PRM			;PRIME RIB?
	SKIPA	T1,TCBFLR(TC)		;NO, GET FLR FROM TCB
	MOVEI	T1,1			;YES, 1ST BLOCK
	MOVEM	T1,DRREL		;STORE 1ST LOGICAL RECORD
	MOVE	T1,RIBSIZ(B)		;SIZE OF FILE IN BLOCKS
	IDIVI	T1,BLKSIZ
	SKIPE	T2
	ADDI	T1,1
	TRNN	F,F.PRM			;PRIME RIB?
	MOVE	T1,TCBSIZ(TC)		;NO, GET SIZE FROM TCB
	MOVEM	T1,DRSIZ		;STORE SIZE
	TRNN	F,F.PRM			;PRIME RIB?
	SKIPA	T1,TCBLFT(TC)		;NO, GET LEFTOVER FROM TCB
	MOVE	T1,RIBALC(B)		;YES, FROM RIB
	MOVEM	T1,DRLFT		;STORE BLOCKS LEFT

;HERE TO PROCESS AN RTP
DORIB1:	SKIPN	T1,(P1)			;PICK UP RTP
	JRST	DORIB2			;EOF
	LDB	T2,CCBP			;GET CC
	JUMPN	T2,DORIB3		;UNIT OR GROUP?

;HERE FOR UNIT POINTER
	TRZ	T1,RIPNUB		;GET RID OF NOISE BIT
	CAMLE	T1,HUN			;IN RANGE?
	JRST	DORB12			;NO
	HRRZ	U,TABUDB(T1)		;YES, POINT TO NEW UDB
	JRST	DORIB4

;HERE FOR GROUP POINTER
DORIB3:	MOVEM	T2,DRCC			;STORE CC
	LDB	T3,CSBP			;STORE CS
	MOVEM	T3,DRCS
	LDB	T4,CABP			;STORE CA
	MOVEM	T4,DRCA
	ADD	T2,T4			;IN RANGE?
	IMUL	T2,BPC
	SUBI	T2,1
	CAMLE	T2,UDBHLB(U)
	JRST	DORB12			;NO
	PUSHJ	P,DOGRP			;PROCESS THE GROUP
;HERE WHEN ALL DONE WITH RTP
DORIB4:	AOBJN	P1,DORIB1		;LOOP
	SKIPN	DRLFT			;RIB EXACTLY FULL?
	JRST	DORIB2			;YES, FORCE EOF
	SKIPN	RIBXRA(B)		;BETTER BE EXTENDED
	JRST	DORB12
	LDB	P1,RIYRBU		;BN OF EXTENDED RIB
	IMUL	P1,BPLU
	LDB	T1,RIYRBA
	IMUL	T1,BPC
	ADDB	P1,T1
	PUSHJ	P,LEGALP		;IN RANGE?
	 JRST	DORB12			;NO
	MOVE	T2,F2CST+CSTCNT		;SET T2 NON-ZERO TO SAVE CST
	ADD	T2,M2CST+CSTCNT
	TRNN	F,F.P2
	SETZ	T2,
	MOVEI	T1,SIZXRB		;ALLOCATE CORE
	SKIPE	T2
	MOVEI	T1,SIZCXR
	PUSH	P,T2
	PUSHJ	P,GETBLK
	MOVE	T1,P1			;LINK THE TCB
	PUSHJ	P,LNKTCB
	MOVEI	T1,FNCRIB		;FUNCTION CODE
	DPB	T1,TCXCOD
	LDB	T1,TCYRBC		;RIB COUNT
	ADDI	T1,1
	DPB	T1,TCXRBC
	HRRM	FL,TCBFIL(T2)		;ADDR OF FIL
	LDB	T1,FIYCNT		;BUMP USE COUNT OF FIL
	ADDI	T1,1
	DPB	T1,FIYCNT
	MOVE	T1,DRREL		;BN RELATIVE TO FILE
	SUBI	T1,1			;ACCOUNT FOR SPARE RIB
	MOVEM	T1,TCBFLR(T2)		;SHOULD BE 1ST LOGICAL RECORD
	MOVE	T1,DRSIZ		;SIZE
	MOVEM	T1,TCBSIZ(T2)
	MOVE	T1,DRLFT		;BLOCKS LEFT
	MOVEM	T1,TCBLFT(T2)
	MOVSI	T1,(TCPCOR)
	ANDCAM	T1,TCBCOR(T2)
	POP	P,T3
	JUMPE	T3,CPOPJ		;GO IF NOT SAVING CST
	IORM	T1,TCBCOR(T2)		;LIGHT FLAG
	MOVEI	T1,TCBCST(T2)
	HRLI	T1,SCST			;SAVE CST
	BLT	T1,TCBCST+SCSTL-1(T2)
	POPJ	P,
;HERE ON EOF
DORIB2:	SKIPE	RIBXRA(B)		;SHOULDN'T BE EXTENDED
	PUSHJ	P,DORB16
	SKIPE	DRLFT			;TEST RIBALC
	PUSHJ	P,DORB13
	MOVE	T1,DRREL		;TEST RIBSIZ
	SUBI	T1,2
	CAMGE	T1,DRSIZ
	PUSHJ	P,DORB14
	FALL	DNRIB

;COMMON EXIT FOR DORIB
DNRIB:	TRNN	F,F.P2			;WHICH PASS?
	POPJ	P,			;PASS 1
	PUSHJ	P,CSTTO			;PASS 2, TYPE CST'S
	MOVEI	P1,F2CST		;DEALLOCATE FREE
	PUSHJ	P,DECST
	MOVEI	P1,M2CST		;DEALLOCATE MULTIPLE
	PJRST	DECST
DORB10:	PUSHJ	P,MRKIT			;NOT A LOST CLUSTER
	MOVEI	T2,[ASCIZ /Error while reading RIB/]
	PUSHJ	P,DORB15
	PJRST	DNRIB

DORB12:	PUSHJ	P,DORB16
	PJRST	DNRIB

DORB16:	MOVEI	T2,[ASCIZ /Bad pointer in RIB/]
DORB15:	PUSHJ	P,STRO
	TRNE	F,F.PRM			;GO IF PRIME RIB
	JRST	DORB99
	MOVEI	T1," "
	PUSHJ	P,CO
	LDB	T1,TCYRBC
	PUSHJ	P,OCTO
DORB99:	MOVEI	T2,[ASCIZ / of /]
	PUSHJ	P,STRO
	PUSHJ	P,FILO
	PJRST	DSPACE

DORB11:	PUSHJ	P,MRKIT			;NOT A LOST CLUSTER
	MOVEI	T2,[ASCIZ /RIB error on /]
	PUSHJ	P,STRO
	PUSHJ	P,FILO
	TRNE	F,F.PRM			;GO IF PRIME RIB
	JRST	DORB17
	MOVEI	T2,[ASCIZ / (RIB /]
	PUSHJ	P,STRO
	LDB	T1,TCYRBC
	PUSHJ	P,OCTO
	MOVEI	T1,")"
	PUSHJ	P,CO
DORB17:	PUSHJ	P,DSPACE
	JRST	DNRIB

DORB13:	SKIPA	T2,[SIXBIT /RIBALC/]
DORB14:	MOVE	T2,[SIXBIT /RIBSIZ/]
	PUSHJ	P,SIXO
	MOVEI	T2,[ASCIZ / is wrong in /]
	PUSHJ	P,STRO
	PUSHJ	P,FILO
	PJRST	DSPACE

DIE:	MOVEM	17,CRSHAC+17		;SAVE THE ACS
	MOVEI	17,CRSHAC
	BLT	17,CRSHAC+16
	MOVE	17,CRSHAC+17
	PUSHJ	P,GIVIF			;GIVE UP INTERLOCK
	HALT				;FATAL ERROR
;SUBROUTINE TO PROCESS A GROUP
DOGRP:	TRNE	F,F.SAT			;SAT.SYS?
	PUSHJ	P,SAVSAT		;YES, SAVE CA
	TRZ	F,F.SUM			;HAVEN'T DONE CHECKSUM YET
	MOVE	T2,DRCA			;BN RELATIVE TO STR
	IMUL	T2,BPC
	ADD	T2,UDBBLK(U)
	MOVEM	T2,DRPHY
	MOVN	T1,DRCC			;BUMP COUNT
	IMUL	T1,BPC
	ADDM	T1,DRLFT
DOGRP1:	MOVE	T1,DRPHY		;LIGHT SAT BIT
	PUSHJ	P,MARKIT
	PUSHJ	P,DOCLT			;PROCESS THIS CLUSTER
	MOVE	T1,BPC			;BUMP BN
	ADDM	T1,DRPHY
	SOSE	DRCC			;LOOP
	JRST	DOGRP1
	POPJ	P,
;ROUTINE TO PROCESS ALL THE BLOCKS IN A CLUSTER
DOCLT:	MOVE	T1,DRPHY		;WORKING COPY OF BN
	MOVEM	T1,DRBLK
	MOVE	T1,BPC			;SETUP LOOP
	MOVEM	T1,DRCNT
DOCLT1:	TRON	F,F.RIB			;STEPPED OVER THE RIB YET?
	JRST	DOCLT2			;NO, THIS IS IT
	MOVE	T1,DRREL		;PAST EOF?
	CAMLE	T1,DRSIZ
	JRST	DOCLT4			;YES, IGNORE IT
	TRNN	F,F.DIR			;IS THIS A DIR?
	JRST	DOCLT3			;NO, WE'RE ONLY INTERESTED IN 1ST BLOCK
	PUSHJ	P,BDDIR			;YES, BUILD A TCB TO READ THIS BLOCK
	AOS	DRREL			;BUMP BN RELATIVE TO FILE
DOCLT2:	AOS	DRBLK			;LOOP
	SOSE	DRCNT
	JRST	DOCLT1
	POPJ	P,
DOCLT3:	TRON	F,F.SUM			;DONE CHECKSUM YET?
	PUSHJ	P,BDSUM			;NO, DO IT NOW
DOCLT4:	MOVE	T1,DRCNT		;DUMP BN RELATIVE TO FILE
	ADDM	T1,DRREL
	POPJ	P,
;ROUTINE TO BUILD A TCB FOR CHECKSUM
BDSUM:	MOVE	T1,RIBSTS(B)		;ALWAYS BAD CHECKSUM?
	TRNN	T1,RIPABC
	TRNE	F,F.P2			;OR PASS TWO?
	POPJ	P,			;YES, DON'T BOTHER
	MOVEI	T1,SIZSUM		;ALLOCATE CORE
	PUSHJ	P,GETBLK
	MOVE	T1,DRBLK		;LINK THE TCB
	PUSHJ	P,LNKTCB
	MOVEI	T1,FNCSUM		;FUNCTION CODE
	DPB	T1,TCXCOD
	HRRM	FL,TCBFIL(T2)		;ADDR OF FIL
	LDB	T1,FIYCNT		;BUMP USE COUNT OF FIL
	ADDI	T1,1
	DPB	T1,FIYCNT
	MOVE	T1,DRREL		;BN RELATIVE TO FILE
	MOVEM	T1,TCBREL(T2)
	MOVE	T1,DRCS			;EXPECTED CHECKSUM
	MOVEM	T1,TCBSUM(T2)
	POPJ	P,
;ROUTINE TO BUILD A TCB FOR THE DATA BLOCK OF A DIRECTORY
BDDIR:	MOVEI	T1,SIZDIR		;ALLOCATE CORE
	PUSHJ	P,GETBLK
	MOVE	T1,DRBLK		;LINK THE TCB
	PUSHJ	P,LNKTCB
	MOVEI	T1,FNCDIR		;FUNCTION CODE
	DPB	T1,TCXCOD
	HRRM	FL,TCBFIL(T2)		;ADDR OF FIL
	LDB	T1,FIYCNT		;BUMP USE COUNT OF FIL
	ADDI	T1,1
	DPB	T1,FIYCNT
	MOVE	T1,DRREL		;BN RELATIVE TO FILE
	MOVEM	T1,TCBREL(T2)
	POPJ	P,
	SUBTTL	READ A DIRECTORY BLOCK

DODIR:	PUSHJ	P,SAVE3			;SAVE ACS
	PUSHJ	P,REDTCB		;READ THE BLOCK
	 JRST	IOERR			;I/O ERROR
	PUSHJ	P,DRCK			;CHECK
	 JRST	DIRERR			;BAD DIRECTORY
	MOVE	P1,B			;START WITH ENTRY ZERO
DODIR1:	SKIPN	T2,(P1)			;EMPTY SLOT?
	JRST	DODIR2			;YES, IGNORE IT
	HLLZ	T1,FILEXT(FL)		;MFD POINTS TO SELF
	HLR	T1,FILCFP(FL)
	CAMN	T1,1(P1)
	CAME	T2,FILNAM(FL)
	JRST	DODIR3
	JRST	DODIR2
DODIR3:	MOVEI	T1,SIZFIL		;BUILD A FIL
	PUSHJ	P,GETBLK
	MOVE	P3,T2
	MOVE	T1,(P1)			;STORE FILENAME IN FIL
	MOVEM	T1,FILNAM(P3)
	HLLZ	T1,1(P1)		;STORE EXTENSION IN FIL
	HLLM	T1,FILEXT(P3)
	HRRM	FL,FILDAD(P3)		;STORE ADDR OF PARENT FIL
	LDB	T1,FIYCNT		;BUMP PARENT COUNT
	ADDI	T1,1
	DPB	T1,FIYCNT
	MOVEI	T1,1			;SET FIL USE COUNT
	DPB	T1,FIZCNT		;(TCB WILL POINT TO FIL)
	HRRZ	T1,1(P1)		;STORE CFP IN FIL
	HRLM	T1,FILCFP(P3)
	JUMPE	T1,DODIR9		;ZERO IS ILLEGAL
	IMUL	T1,BPSC			;BN RELATIVE TO STR
	PUSHJ	P,LEGALP		;LEGAL BN?
	 JRST	DODIR9			;NO
	MOVEI	T1,SIZRIB		;BUILD A TCB
	PUSHJ	P,GETBLK
	MOVE	P2,T2
	HRRZ	T1,1(P1)		;LINK IT
	IMUL	T1,BPSC
	PUSHJ	P,LNKTCB
	MOVEI	T1,FNCRIB		;STORE FUNCTION IN TCB
	DPB	T1,TCZCOD
	SETZ	T1,			;PRIME RIB
	DPB	T1,TCZRBC
	HRRM	P3,TCBFIL(P2)		;STORE ADDR OF FIL IN TCB
DODIR2:	ADDI	P1,2			;LOOP
	CAIE	P1,BLKSIZ(B)
	JRST	DODIR1
	PJRST	TWATER			;WATER LEVEL HAS CHANGED SIGNIFICANTLY
;HERE IF BAD CFP
DODIR9:	MOVEI	T2,[ASCIZ /Bad CFP for /]
	PUSHJ	P,STRO
	PUSH	P,FL
	MOVE	FL,P3
	PUSHJ	P,FILO
	PUSHJ	P,DECFIL
	POP	P,FL
	PUSHJ	P,DSPACE
	JRST	DODIR2

;ROUTINE TO CHECK IF DIRECTORY DATA BLOCK IS OK
DRCK:	MOVE	P1,B			;START WITH ENTRY ZERO
	SETZ	P2,			;NO BAD ONES YET
DRCK1:	SKIPN	(P1)			;ZERO FILENAME?
	JRST	DRCK3			;YES, EMPTY SLOT
	HRRZ	T1,1(P1)		;GET CFP
	JUMPE	T1,DRCK2		;ZERO IS ILLEGAL
	IMUL	T1,BPSC			;CONVERT TO BN
	PUSHJ	P,LEGALP		;LEGAL BN?
	 JRST	DRCK2			;NO
	JRST	DRCK4			;YES
;HERE IF FILENAME IS ZERO
DRCK3:	SKIPE	1(P1)			;EXTENSION MUST BE ZERO TOO
	POPJ	P,			;FAIL IMMEDIATELY
	JRST	DRCK4
;HERE IF CFP IS BAD
DRCK2:	ADDI	P2,1			;COUNT IT
	CAIN	P2,BADDR		;REACH THRESHOLD?
	POPJ	P,			;YES
DRCK4:	ADDI	P1,2			;LOOP
	CAIE	P1,BLKSIZ(B)
	JRST	DRCK1
	JRST	CPOPJ1			;OK
	SUBTTL	CONFIGURATION

;ROUTINE TO BUILD A LIST OF ALL THE STRUCTURES ON THE SYSTEM
BDALL:	PUSHJ	P,SAVE2
	MOVEI	P1,CONFIG-CNFNXT	;PRESET PRED
BDALL1:	HRRZ	P1,CNFNXT(P1)		;STEP TO NEXT CNF
	JUMPE	P1,CPOPJ
	MOVE	P2,CNFSTR(P1)		;NAME OF STR
	MOVEI	T2,SLST-SNFNXT		;PRESET PRED
BDALL2:	HRRZ	T2,SNFNXT(T2)		;STEP TO NEXT SNF
	JUMPE	T2,BDALL3
	CAMN	P2,SNFNAM(T2)		;ALREADY KNOW ABOUT THAT ONE?
	JRST	BDALL4			;YES
	JRST	BDALL2			;NO
BDALL3:	MOVEI	T1,SIZSNF		;ALLOCATE AN SNF
	PUSHJ	P,GETHI
	MOVE	T1,SLST			;LINK TO SLST
	MOVEM	T1,SNFNXT(T2)
	MOVEM	T2,SLST
	MOVE	T1,BLST			;LINK TO BLST
	MOVEM	T1,SNFLNK(T2)
	MOVEM	T2,BLST
	MOVEM	P2,SNFNAM(T2)		;SAVE STR NAME
BDALL4:	MOVEI	T4,CONFIG-CNFNXT	;PRESET PRED
	SETZB	T1,T3
BDALL5:	HRRZ	T4,CNFNXT(T4)		;STEP TO NEXT UNIT
	JUMPE	T4,BDALL6
	CAME	P2,CNFSTR(T4)		;THIS STR?
	JRST	BDALL5			;NO
	IOR	T1,CNFCHN(T4)		;CHAN MASK
	ADD	T3,CNFVRT(T4)		;SWAPPING SPACE
	JRST	BDALL5
BDALL6:	MOVEM	T1,SNFCHN(T2)		;CHANNEL MASK
	MOVEM	T3,SNFVRT(T2)		;SWAPPING SPACE
	JRST	BDALL1			;NEXT UNIT
;ROUTINE TO BUILD THE UDBS FOR ALL THE UNITS IN A STR
;ALIAS PASSES THE NAME OF THE STR
;TBUDB RETURNS A LIST OF POINTERS TO UDBS
;HUN RETURNS HIGHEST UNIT NUMBER
PUNT:	PUSHJ	P,SAVE1			;SAVE P1
	PUSHJ	P,GETLOK
	MOVEI	P1,CONFIG-CNFNXT	;START WITH 1ST UNIT ON SYSTEM
	SETZM	NUN			;NO UNITS SO FAR
	SETZM	WL			;NO UNITS WRITE LOCKED (YET)
	SETZM	SSIZ			;SUM OF UDBBPU
	SETZM	BIGBPU			;BIGGEST UDBBPU
PUNT1:	HRRZ	P1,CNFNXT(P1)		;STEP TO NEXT CNF
	JUMPE	P1,GIVLOK
	MOVE	T1,CNFSTR(P1)		;OUR STR?
	CAME	T1,ALIAS
	JRST	PUNT1			;NO
	MOVEI	T1,SIZUDB		;BUILD A UDB
	PUSHJ	P,GETZER
	MOVE	U,T2
	MOVE	T1,CNFNAM(P1)		;UNIT NAME
	MOVEM	T1,UDBNAM(U)
	MOVE	T1,CNFALT(P1)		;ALTERNATE PORT
	MOVEM	T1,UDBALT(U)
	AOS	T3,NUN			;BUMP COUNT
	CAILE	T3,NUNT			;TOO BIG?
	PUSHJ	P,DIE			;YES
	SUBI	T3,1			;HIGHEST UNIT
	MOVEM	T3,HUN
	MOVEM	U,TBUDB(T3)
	PUSHJ	P,MOVCPY		;SAVE CAPACITY
	JRST	PUNT1			;AND TRY NEXT UNIT
;ROUTINE TO COMPILE A LIST OF ALL THE UNITS ON THE SYSTEM
LUNIT:	PUSHJ	P,SAVE2			;SAVE AC
	SETZ	P1,			;NONE SO FAR
LUNIT1:	SYSPHY	P1,			;GET NAME OF NEXT UNIT ON SYSTEM
	 PUSHJ	P,DIE
	JUMPE	P1,CPOPJ
	MOVEM	P1,FOO+.DCNAM		;GET STATS
	MOVE	T1,[XWD FOOSIZ,FOO]
	DSKCHR	T1,
	 PUSHJ	P,DIE
	SKIPE	FOO+.DCSNM		;IN A STR?
	TLNE	T1,(DC.STS)		;AND UNIT OK?
	JRST	LUNIT1			;NO, SKIP IT
	MOVEI	P2,CONFIG-CNFNXT	;SEE IF IT ALREADY EXISTS
LUNIT2:	HRRZ	P2,CNFNXT(P2)
	JUMPE	P2,LUNIT3		;NO
	CAME	P1,CNFNAM(P2)
	CAMN	P1,CNFALT(P2)
	JRST	LUNIT4			;YES
	JRST	LUNIT2
;HERE IF UNIT DOES NOT YET EXIST
LUNIT3:	MOVEI	T1,SIZCNF		;ALLOCATE A CNF
	PUSHJ	P,GETHI
	MOVE	P2,T2			;PUT IT IN A SAFE PLACE
	MOVE	T1,CONFIG		;LINK IT
	MOVEM	T1,CNFNXT(P2)
	MOVEM	P2,CONFIG
	SETZM	CNFVRT(P2)		;NO SWAP SPACE YET
;HERE IF UNIT DOES EXIST, STORE NEW INFO
LUNIT4:	MOVEM	P1,CNFNAM(P2)		;SAVE PHYS UNIT NAME
	MOVE	T1,FOO+.DCSNM		;STR NAME
	MOVEM	T1,CNFSTR(P2)
	MOVE	T1,FOO+.DCK4S		;K FOR SWAPPING
	LSH	T1,1			;PAGES
	SKIPL	FOO+.DCPAS		;IN ASL?
	MOVEM	T1,CNFVRT(P2)		;YES, THESE PAGES COUNT
	MOVE	T2,P1			;CHANNEL MASK
	PUSHJ	P,GETCHN
	MOVEM	T1,CNFCHN(P2)
	MOVE	T2,FOO+.DCALT		;ALTERNATE UNIT NAME
	MOVEM	T2,CNFALT(P2)
	JUMPE	T2,LUNIT1		;LOOP
	PUSHJ	P,GETCHN		;CHANNEL MASK OF ALTERNATE
	IORM	T1,CNFCHN(P2)		;ADD IT IN
	JRST	LUNIT1			;LOOP

;ROUTINE TO ALLOCATE SOME CORE IN THE HISEG
;T1 PASSES SIZE (DESTROYED)
;T2 RETURNS ADDR
;NOTE THAT HISEG CORE IS NEVER RETURNED
GETHI:	MOVE	T2,HFF			;FIRST FREE
	ADDB	T1,HFF			;NEW FIRST FREE
	CAILE	T1,HCOR+NHCOR
	PUSHJ	P,DIE
	POPJ	P,
;ROUTINE TO BUILD A BIT MASK FOR THE CHANNEL NUMBER
;T2 PASSES THE UNIT NAME
;T1 RETURNS BIT MASK
;THIS ROUTINE IS A HORRIBLE CROCK. WE REALLY OUGHT TO USE THE
;CHANNEL NUMBER RETURNED BY THE DSKCHR UUO, BUT THE INFORMATION
;RETURNED BY DSKCHR SIMPLY CAN'T BE TRUSTED. THERE ARE SIMPLY
;TOO MANY MONITOR BUGS INVOLVED. THE VALUE RETURNED BY DSKCHR IS
;A CHANNEL NUMBER PER CPU RATHER THAN SYSTEM WIDE. MOREOVER,
;PRE-703 MONITORS RETURN A BAD CHANNEL NUMBER FOR THE SECOND
;KDB OF A 16 DRIVE RP20 STRING.
;THE ALGORITHM USED HERE DOESN'T ALWAYS GENERATE PERFECT RESULTS,
;BUT IT'S USUALLY PRETTY GOOD.
GETCHN:	HLRZS	T1,T2			;BLOW THE UNIT NUMBER
	TRZ	T1,77			;IS IT AN RP20?
	CAIE	T1,'RN '
	JRST	GTCHN1			;NO
	MOVE	T3,MONVER		;PRE-703?
	CAIL	T3,70300
	JRST	GTCHN1			;NO
	TRNN	T2,1			;ALTERNATE KDB?
	SUBI	T2,1			;YES, USE NAME OF PRIME KDB
GTCHN1:	SKIPN	T3,CHNNAM		;SEARCH FOR KON NAME
	JRST	GTCHN2
GTCHN3:	CAMN	T2,CHNNAM(T3)
	JRST	GTCHN4			;FOUND IT
	SOJN	T3,GTCHN3
GTCHN2:	AOS	T3,CHNNAM		;NOT FOUND, ROOM FOR ANOTHER?
	CAILE	T3,^D36
	SOS	T3,CHNNAM		;NOPE, BACK UP
	MOVEM	T2,CHNNAM(T3)		;STORE CHANNEL NAME
GTCHN4:	MOVEI	T1,1			;BUILD A BIT MASK
	LSH	T1,-1(T3)
	POPJ	P,
	SUBTTL	READ HOME BLOCKS

;ROUTINE TO READ THE HOME BLOCKS OF ALL THE UNITS ON A STR
;SKIP IF OK
RDHOM:	PUSHJ	P,SAVE1			;SAVE P1
	SETZM	TABUDB
	MOVE	T1,[XWD TABUDB,TABUDB+1]
	BLT	T1,TABUDB+NUNT-1
	TRZ	F,F.STR			;STR INFO ISN'T FROZEN YET
	MOVE	P1,HUN			;SET UP LOOP
RDHOM1:	MOVE	U,TBUDB(P1)		;GET ADDR OF UDB
	PUSHJ	P,RDHM			;READ THE HOME BLOCK
	 POPJ	P,
	MOVE	T1,HOMLUN(B)		;GET LOGICAL UNIT NUMBER
	SKIPE	TABUDB(T1)		;ALREADY HAVE ONE OF THOSE?
	POPJ	P,			;YES
	HRLI	U,SZUDB			;SAVE ADDR OF UDB
	MOVEM	U,TABUDB(T1)
	PUSHJ	P,MOVSTR		;COPY STUFF TO STRUCTURE DATA BLOCK
	 POPJ	P,
	PUSHJ	P,MOVUDB		;COPY STUFF TO UDB
	PUSHJ	P,BLDSAT		;BUILD SAT TABLES
	SOJGE	P1,RDHOM1		;LOOP FOR EACH UNIT
	MOVE	T1,MFDBN		;TEST IF BN IS LEGAL
	PUSHJ	P,LEGALP
	 POPJ	P,
	MOVE	T1,MFDBN
	IDIV	T1,BPSC
	SKIPN	T2
	AOS	(P)
	POPJ	P,
;ROUTINE TO READ A HOME BLOCK
;U PASSES UDB
;SKIP IF OK
RDHM:	MOVEI	T3,1			;READ 1ST HOME BLOCK
	PUSHJ	P,REDBLK
	 JRST	RDHM1			;ERROR
	PUSHJ	P,HMCK			;VALID HOME BLOCK?
	 JRST	RDHM1			;NO
	JRST	CPOPJ1			;YES, WIN
RDHM1:	MOVEI	T3,12			;READ 2ND HOME BLOCK
	PUSHJ	P,REDBLK
	 POPJ	P,
	PJRST	HMCK			;AND CHECK IF VALID

;ROUTINE TO READ A BAT BLOCK
;U PASSES UDB
;SKIP IF OK
RDBT:	MOVEI	T3,2			;READ 1ST BAT BLOCK
	PUSHJ	P,REDBLK
	 JRST	RDBT1			;ERROR
	PUSHJ	P,BTCK			;VALID BAT BLOCK?
	 JRST	RDBT1			;NO
	JRST	CPOPJ1			;YES, WIN
RDBT1:	MOVEI	T3,13			;READ 2ND BAT BLOCK
	PUSHJ	P,REDBLK
	 POPJ	P,
	FALL	BTCK			;AND CHECK IF VALID

;ROUTINE TO TEST VALIDITY OF A BAT BLOCK
;SKIP IF OK
BTCK:	MOVS	T1,BAFNAM(B)		;NAME
	MOVE	T2,BAFCOD(B)		;UNLIKELY CODE
	CAIN	T1,'BAT'
	CAIE	T2,CODBAT
	POPJ	P,
	HLRE	T1,BAFFIR(B)		;USEABLE WORDS
	MOVNS	T1
	HRRZ	T2,BAFFIR(B)		;1ST WORD
	ADD	T2,T1			;LAST WORD +1
	LDB	T3,BAYNBR		;REGIONS
	ADD	T3,BAFCNT(B)
	LSH	T3,1			;TIMES WORDS PER ENTRY
	CAMG	T3,T1			;TOO MANY ENTRIES?
	CAILE	T2,BAFCOD		;NOT ENOUGH SLOTS?
	POPJ	P,
	JRST	CPOPJ1
	SUBTTL	TCB I/O

;ROUTINE TO READ THE BLOCK POINTED TO BY A TCB
;TC PASSES ADDR OF TCB
;U RETURNS ADDR OF UDB
;FL RETURNS ADDR OF FIL
;SKIP IF OK
REDTCB:	HRRZ	FL,TCBFIL(TC)		;LOAD ADDR OF FIL
	LDB	T1,TCYCX		;FIND DDB
IFN FTDBUG,<
	SKIPN	T1
	PUSHJ	P,DIE
>
	MOVE	DDB,USRJDA(T1)
	MOVE	T1,TCBBLK(TC)		;FIND UNIT
IFN FTDBUG,<
	CAML	T1,DDBLBN(DDB)
	CAMLE	T1,DDBHBN(DDB)
	PUSHJ	P,DIE
>
	IDIV	T1,BPLU
	HRRZ	U,TABUDB(T1)
	SKIPE	DDBERR(DDB)		;ERROR?
	POPJ	P,			;YES
	MOVE	B,TCBBLK(TC)		;ADDR OF BLOCK
	SUB	B,DDBLBN(DDB)
	IMULI	B,BLKSIZ
	ADD	B,DDBBUF(DDB)
	ADDI	B,2
	JRST	CPOPJ1
;HERE TO LOAD A TCB INTO CACHE
;TC PASSES TCB
;DDB PASSES DDB
;WE WILL READ SEVERAL BLOCKS ALL AT ONCE
;FIND THE HIGHEST DO-ABLE BLOCK
LODTCB:	PUSHJ	P,SAVE4			;SAVE ACS
	MOVE	P2,TCBBLK(TC)		;FIND UNIT
	IDIV	P2,BPLU
	HRRZ	U,TABUDB(P2)
	MOVE	P2,TCBBLK(TC)		;GET BN BACK
	MOVE	T1,TCBMBB(TC)		;SUSPECTED BAD?
	TLNE	T1,(TCPMBB)
	JRST	LDTCB9			;YES, DO SINGLE BLOCK XFER
	MOVE	P1,P3			;BN RELATIVE TO UNIT
	IDIV	P1,UDBCYL(U)		;CYL NUMBER
	ADDI	P1,1			;NEXT CYL
	IMUL	P1,UDBCYL(U)		;1ST BN OF NEXT CYL
	ADD	P1,UDBBLK(U)		;RELATIVE TO STR
	MOVE	P4,TCBBLK(TC)		;1ST BN THAT WON'T FIT IN BUFFER
	ADDI	P4,NBLK
	HRRZ	T1,TC			;START WITH CURRENT TCB
	PUSHJ	P,XTRAP			;WANT EXTRA BLOCK?
	 TLO	T1,(1B0)		;NO
LDTCB2:	MOVE	P2,T1			;SAVE LAST GOOD TCB
LDTCB4:	HRRZ	T1,TCBLNK(T1)		;STEP TO NEXT TCB
	CAIN	T1,QTCB			;IF END
	JRST	LDTCB3			;THEN DON'T DO IT
	CAMLE	P1,TCBBLK(T1)		;SAME CYL?
	CAMG	P4,TCBBLK(T1)		;AND FIT IN BUFFER?
	JRST	LDTCB3			;NO, DON'T DO IT
	LDB	T2,TCXCX		;SEE IF I/O ALREADY ACTIVE
	JUMPN	T2,LDTCB3		;YES
	PUSHJ	P,FNDCX			;DON'T KNOW, SEARCH AND FIND OUT
	 SKIPA	T3,TCBMBB(T1)		;NOT ACTIVE
	JRST	LDTCB3			;ACTIVE
	TLNE	T3,(TCPMBB)		;SUSPECTED BAD?
	JRST	LDTCB3			;YES, DON'T DO IT
	PUSHJ	P,FITP			;FIT IN CORE?
	 JRST	LDTCB4			;NO
	MOVE	T2,TCBBLK(P2)		;BN OF LAST GOOD TCB
	ADD	T2,UDBSKP(U)		;1ST BN THAT WOULD SKIP TOO MUCH
	ADDI	T2,2
	SKIPL	P2			;WANT EXTRA BLOCK?
	ADDI	T2,1			;YES, ONE MORE
	CAMG	T2,TCBBLK(T1)		;SKIP TOO MUCH?
	JRST	LDTCB3			;YES, DON'T DO IT
	PUSHJ	P,XTRAP			;DOES CURRENT TCB WANT EXTRA?
	 TLO	T1,(1B0)		;NO
	MOVE	T2,TCBBLK(T1)		;NEXT BN
	ADDI	T2,1
	TRNE	F,F.702			;7.02?
	CAME	T2,P4			;AND EXTRA WON'T FIT IN BUFFER?
	JRST	LDTCB2
	CAME	T2,P1			;AND EXTRA IS SAME CYL?
	SKIPGE	T1			;AND WE WANT EXTRA?
	 JRST	LDTCB2
	FALL	LDTCB3			;YES, PUT THIS TCB IN NEXT XFER
					; SO WE WON'T HAVE TO DO A ONE
					; BLOCK XFER JUST TO PICK UP
					; THE EXTRA

;HERE WITH P2=LAST GOOD TCB
LDTCB3:	MOVE	T1,P2			;LAST GOOD TCB
	MOVE	P2,TCBBLK(T1)		;THIS BN
	SKIPGE	T1			;WANT EXTRA BLOCK?
	JRST	LDTCB9			;NO
	ADDI	P2,1			;NEXT BN
	CAMGE	P2,P1			;SAME CYL?
	CAML	P2,P4			;AND FIT IN BUFFER?
	SUBI	P2,1			;NO EXTRA

;HERE WITH P2=HIGHEST BN TO READ
LDTCB9:	MOVEM	P2,DDBHBN(DDB)		;HIGH
	MOVE	T1,TCBBLK(TC)		;LOW
	MOVEM	T1,DDBLBN(DDB)
	SETZM	DDBERR(DDB)		;NO ERRORS SO FAR
	SETZM	DDBDON(DDB)		;NOT DONE YET
	SUB	P2,TCBBLK(TC)		;NUMBER OF BLOCKS
	ADDI	P2,1
	IMULI	P2,BLKSIZ		;NUMBER OF WORDS
	PUSHJ	P,OPNUDB		;MAKE SURE UNIT IS OPEN
	 PUSHJ	P,DIE			;IT WENT AWAY
	MOVE	T1,DDBCX(DDB)		;STORE CHANNEL INDEX
	DPB	T1,TCYCX
	MOVEM	TC,DDBTCB(DDB)		;STORE 1ST TCB
	AOS	ACTCNT			;ANOTHER DDB ACTIVE
	MOVE	T3,P3			;POSITION
	PUSHJ	P,SUSETI
	TRNE	F,F.702			;BIG BUFFERS?
	JRST	LDTCB6			;YES

;HERE IF 7.01
	MOVE	T4,DDBBUF(DDB)		;BUILD IOWD
	ADDI	T4,1
	MOVNS	P2
	HRL	T4,P2
	MOVEM	T4,CMD
LDTCB7:	PUSHJ	P,IOXCT			;READ IT
	IN	CMD
	 JRST	DOIN1
	PUSHJ	P,SETDMP		;ERROR, CLEAR BIT
	MOVE	T4,DDBBUF(DDB)		;BUILD SINGLE IOWD
	ADDI	T4,1
	HRLI	T4,-BLKSIZ
	CAMN	T4,CMD			;ALREADY SINGLE?
	JRST	DOIN2			;YES, HARD ERROR
	MOVEM	T4,CMD
	PUSHJ	P,SETMBB		;SET "MIGHT BE BAD"
	MOVE	T4,DDBLBN(DDB)		;ONLY ONE BLOCK
	MOVEM	T4,DDBHBN(DDB)
	MOVE	T3,P3			;RE-POSITION
	PUSHJ	P,SUSETI
	JRST	LDTCB7			;TRY AGAIN

;HERE IF 7.02
LDTCB6:	ADDI	P2,1			;STORE SIZE OF BUFFER
	DPB	P2,DEYSIZ
	PUSHJ	P,DOIN			;START I/O
	 POPJ	P,			;FINISHED
	POPJ	P,			;NOT FINISHED
;ROUTINE DO NON-BLOCKING INPUT
;SKIP IF NOT DONE
DOIN:	PUSHJ	P,IOXCT			;DO IT
	IN
	 JRST	DOIN1
	PUSHJ	P,IOXCT			;ERROR?
	STATO	IO.ERR
	 JRST	CPOPJ1			;NO, MERELY UNFINISHED
	PUSHJ	P,SETBUF		;YES, CLEAR ERROR BIT
	LDB	T1,DEYSIZ		;GET SIZE
	CAIN	T1,BLKSIZ+1		;ALREADY SINGLE?
	JRST	DOIN2			;YES, HARD ERROR
	MOVEI	T1,BLKSIZ+1		;NO, MAKE SINGLE
	DPB	T1,DEYSIZ
	PUSHJ	P,SETMBB		;SET "MIGHT BE BAD"
	MOVE	T2,DDBLBN(DDB)		;RE-POSITION
	MOVEM	T2,DDBHBN(DDB)
	IDIV	T2,BPLU
	PUSHJ	P,SUSETI
	JRST	DOIN			;TRY AGAIN
DOIN2:	SETOM	DDBERR(DDB)		;HARD ERROR
DOIN1:	SETOM	DDBDON(DDB)		;IO DONE
	POPJ	P,
;THIS ROUTINE IS CALLED WHENEVER THERE IS AN I/O ERROR.
;WE DON'T KNOW WHICH BLOCK IT IS THAT'S BAD,
;SO WE WILL LIGHT TCPMBB IN ALL THE TCB'S.
;THIS BIT WILL CAUSE EACH BLOCK TO BE READ SEPERATELY (WITHOUT
;TRYING TO XFER SEVERAL BLOCKS IN THE SAME BUFFER).
;THIS IS THE FASTEST ALGORITHM FOR FINDING THE BAD BLOCK.
;YES, IT'S EVEN FASTER THAN A BINARY SEARCH!
;IT TURNS OUT THAT THE LIMITING FACTOR ON HOW FAST YOU CAN SEARCH IS
;THE NUMBER OF TIMES THAT THE BAD BLOCK IS READ. EACH TIME THAT
;WE ATTEMPT TO READ THE BAD BLOCK, THE MONITOR WILL END UP DOING
;NUMEROUS RETRIES. THIS CAN TAKE UPWARDS OF 1/2 SEC EACH TIME!
;THUS THE GOAL IN DESIGNING THE SEARCH ALGORITHM IS THAT THE BAD
;BLOCK (WHEREVER IT MAY BE) SHOULD ONLY BE READ ONCE.
SETMBB:	MOVE	T1,DDBTCB(DDB)		;1ST TCB IN XFER
STMBB1:	MOVSI	T2,(TCPMBB)		;LIGHT THE BIT
	IORM	T2,TCBMBB(T1)
	HRRZ	T1,TCBLNK(T1)		;STEP TO NEXT TCB
	CAIN	T1,QTCB
	POPJ	P,
	MOVE	T2,TCBBLK(T1)		;END OF XFER?
	CAMG	T2,DDBHBN(DDB)
	JRST	STMBB1			;NO, LOOP
	POPJ	P,			;YES
;ROUTINE TO DECIDE WHETHER TO READ AN EXTRA BLOCK
;T1 PASSES ADDR OF TCB
;SKIP IF SHOULD READ AN EXTRA BLOCK
;THIS DOES NOT IMPLY THAT THE EXTRA BLOCK WILL FIT IN THE BUFFER,
;IT MERELY IMPLIES THAT WE WOULD LIKE TO READ THE EXTRA BLOCK IF IT
;WILL, IN FACT, FIT.
XTRAP:	HRRZ	T2,TCBLNK(T1)		;NEXT TCB
	CAIN	T2,QTCB
	JRST	XTRAP1
	MOVE	T3,TCBBLK(T1)		;NEXT BN
	ADDI	T3,1
	CAMN	T3,TCBBLK(T2)		;CONSECUTIVE TCB'S?
	POPJ	P,			;YES
XTRAP1:	LDB	T2,TCWCOD		;RIB?
	CAIE	T2,FNCRIB
	POPJ	P,			;NO
	HRRZ	T2,TCBFIL(T1)		;YES, RIB OF DIR?
	HLRZ	T2,FILEXT(T2)
	CAIE	T2,'SFD'
	CAIN	T2,'UFD'
	JRST	CPOPJ1			;YES, READ XTRA
	TRNN	F,F.P2			;NOT DIR, IS IT PASS 2?
	AOS	(P)			;PASS 1, READ XTRA
	POPJ	P,			;PASS 2, NO XTRA
	SUBTTL	I/O

;ROUTINE TO READ A DISK BLOCK (WITHOUT HAVING A TCB)
;U PASSES A POINTER TO THE UDB
;DDB PASSES A POINTER TO THE DDB
;T3 PASSES BN RELATIVE TO UNIT
;NOSKIP IF ERROR
;SKIP IF OK
REDBLK:	SETOM	DDBHBN(DDB)		;INVALIDATE CACHE
	MOVE	B,DDBBUF(DDB)		;ADDR OF BUFFER
	ADDI	B,2
	JUMPE	T3,CPOPJ
	PUSHJ	P,OPNUDB		;MAKE SURE UNIT IS OPEN
	 POPJ	P,
	PUSHJ	P,CODMP			;SWITCH TO DUMP
	PUSHJ	P,SUSETI		;POSITION
	MOVEI	T1,-1(B)		;BUILD IOWD
	HRLI	T1,-BLKSIZ
	MOVEM	T1,CMD
	PUSHJ	P,IOXCT			;READ IT
	IN	CMD
	 JRST	CPOPJ1			;WIN
	PJRST	SETDMP			;CLEAR ERROR BIT

;ROUTINE TO WRITE A DISK BLOCK (WITHOUT HAVING A TCB)
;U PASSES A POINTER TO THE UDB
;DDB PASSES A POINTER TO THE DDB
;B PASSES A POINTER TO THE BUFFER
;T3 PASSES BN RELATIVE TO UNIT
;NOSKIP IF ERROR
;SKIP IF OK
WRTBLK:	JUMPE	T3,CPOPJ
	PUSHJ	P,OPNUDB		;MAKE SURE UNIT IS OPEN
	 POPJ	P,
	PUSHJ	P,CODMP			;SWITCH TO DUMP
	PUSHJ	P,SUSETO		;POSITION
	MOVEI	T1,-1(B)		;BUILD IOWD
	HRLI	T1,-BLKSIZ
	MOVEM	T1,CMD
	PUSHJ	P,IOXCT			;WRITE IT
	OUT	CMD
	 JRST	CPOPJ1			;WIN
	PJRST	SETDMP			;CLEAR ERROR BIT
;ROUTINE TO EXECUTE AN IO INSTRUCTION ON THE RIGHT CHANNEL
;DDB PASSES DDB
;CLOBBERS T1 (ALL OTHER ACS PRESERVED)
IOXCT:	MOVE	T1,DDBCH(DDB)		;GET CHANNEL NUMBER
	LSH	T1,^D18+4+1		;POSTION IT
	ADD	T1,@(P)			;ADD REST OF INSTRUCTION
	AOS	(P)			;SKIP OVER INSTRUCTION
	XCT	T1			;DO THE INSTRUCTION
	 POPJ	P,			;NOSKIP
	JRST	CPOPJ1			;SKIP

;ROUTINE TO OPEN THE UNIT
;U PASSES UDB
;DDB PASSES DDB
OPNUDB:	MOVE	T1,UDBNAM(U)		;ALREADY OPEN?
	CAMN	T1,DDBPUN(DDB)
	JRST	CPOPJ1			;YES
	MOVEM	T1,DEV+.OPDEV		;NO, OPEN IT NOW
	MOVEI	T1,.IODMP
	TRNE	F,F.702
	MOVE	T1,[UU.AIO+UU.LBF+.IOBIN]
	MOVEM	T1,DEV+.OPMOD
	MOVEI	T1,DDBHDR(DDB)
	MOVEM	T1,DEV+.OPBUF
	SETZM	DDBPUN(DDB)
	PUSHJ	P,IOXCT
	OPEN	DEV
	 POPJ	P,
	MOVE	T1,UDBNAM(U)
	MOVEM	T1,DDBPUN(DDB)
	MOVE	T1,DDBBUF(DDB)
	TLO	T1,(BF.VBR)
	MOVEM	T1,DDBHDR+.BFADR(DDB)
;UNDER 7.02 THERE DOES NOT APPEAR TO BE ANY ADVANTAGE TO SETTING CPU.
;USING QUEUED PROTOCOL IS ONLY GOING TO TAKE ONE TICK, AND GETTING THE
;REQUEST QUEUED A TICK FASTER WOULDN'T REALLY HELP. THE DRIVE WILL STILL
;BE BUSY DOING THE CURRENT TRANSFER FOR AT LEAST A TICK ANYWAY.
;THUS THERE IS NO ADVANTAGE TO SETTING CPU. IN FACT, IT'S ACTUALLY AN
;ADVANTAGE TO BE RUNNABLE ON LOTS OF CPU'S. THAT WAY, YOU'RE MORE LIKELY
;TO GET CPU TIME WHEN YOU NEED IT IN A HURRY.
	TRNE	F,F.702			;7.01?
	JRST	CPOPJ1			;NO ADVANTAGE
	MOVE	T1,UDBCPU(U)		;CPU(S) THIS UNIT IS ON
	CAMN	T1,OURCPU
	JRST	CPOPJ1			;YES
	MOVEM	T1,OURCPU		;NO
	HRLI	T1,.STCPU		;SET US THERE
	SETUUO	T1,
	 JFCL
	JRST	CPOPJ1

;PUT DDB INTO BUFFERED MODE
SETBUF:	PUSHJ	P,IOXCT
	SETSTS	.IOBIN
	POPJ	P,

;PUT DDB INTO DUMP MODE
SETDMP:	PUSHJ	P,IOXCT
	SETSTS	.IODMP
	POPJ	P,

;COROUTINE TO PUT DDB IN DUMP MODE
CODMP:	TRNN	F,F.702			;ALREADY IN DUMP MODE?
	POPJ	P,			;YES
	PUSHJ	P,SETDMP		;NO, SWITCH
	POP	P,T1			;PRUNE STACK
	PUSHJ	P,@T1			;CALL CALLER
	 CAIA
	AOS	(P)
	JRST	SETBUF			;SWITCH BACK

;ROUTINE TO PERFORM A SUPER USETI
;THE SUSET UUO REALLY SUCKS.
;I PREFER A GOOD OLD USETI.
;BUT SOME PEOPLE PATCH SUPER USETI OUT OF THEIR MONITOR.
;T3 PASSES BN (RELATIVE TO UNIT)
;CLOBBERS T1
SUSETI:	MOVE	T1,DDBCH(DDB)		;GET CHANNEL NUMBER
	LSH	T1,^D23			;POSITION IT
SUSET2:	ADD	T1,T3			;BN
	SUSET.	T1,
	 PUSHJ	P,DIE			;SERVES YOU RIGHT!
	POPJ	P,

;ROUTINE TO PERFORM A SUPER USETO
;T3 PASSES BN (RELATIVE TO UNIT)
;CLOBBERS T1
SUSETO:	MOVE	T1,DDBCH(DDB)		;GET CHANNEL NUMBER
	LSH	T1,^D23			;POSITION IT
	TLO	T1,(SU.SOT)		;OUTPUT
	JRST	SUSET2
	SUBTTL	CHECK HOME BLOCKS

;ROUTINE TO TEST THE VALIDITY OF A HOME BLOCK
;SKIP IF HOME BLOCK IS OK
HMCK:	MOVS	T1,HOMNAM(B)		;NAME MUST BE HOME
	MOVE	T2,HOMCOD(B)		;MUST HAVE MAGIC NUMBER
	CAIN	T1,'HOM'
	CAIE	T2,CODHOM
	POPJ	P,
	LDB	T1,[POINT 6,HOMCLP(B),11];BITS PER CA
	LDB	T2,[POINT 6,HOMCLP(B),5];CA BIT POSITION
	JUMPN	T2,CPOPJ		;MUST BE ZERO
	LDB	T2,[POINT 6,HOMCKP(B),11];BITS PER CS
	LDB	T3,[POINT 6,HOMCKP(B),5];CS BIT POSITION
	CAIG	T1,CLASIZ		;CA MUST FIT IN SPT
	CAME	T1,T3			;SIZE OF CA IS POSITION OF CS
	POPJ	P,
	ADD	T1,T2			;COMBINED SIZE OF CA AND CS
	LDB	T2,[POINT 6,HOMCNP(B),11];BITS PER CC
	LDB	T3,[POINT 6,HOMCNP(B),5];CC BIT POSITION
	CAME	T1,T3			;SIZE OF CA+CS IS POSITION OF CC
	POPJ	P,
	ADD	T1,T2			;COMBINED SIZE OF CA, CS, AND CC
	SKIPN	HOMREF(B)		;CAN'T NEED REFRESHING
	CAIE	T1,^D36			;MUST FILL THE WORD EXACTLY
	POPJ	P,
	SKIPL	T1,HOMLUN(B)		;LOGICAL UNIT NUMBER
	CAMLE	T1,HUN
	POPJ	P,
	SUB	T1,HUN			;NUMBER OF UNITS REMAINING
	SKIPN	T2,HOMNXT(B)		;IS THIS SUPPOSED TO BE LAST UNIT?
	JUMPN	T1,CPOPJ		;YES, BETTER BE NONE LEFT
	SKIPN	T1			;NONE LEFT?
	JUMPN	T2,CPOPJ		;YES, IT'S SUPPOSED TO BE THE LAST
	SKIPLE	T1,HOMBSC(B)		;BPSC MUST BE POSITIVE
	SKIPG	T2,HOMBPC(B)		;BPC MUST BE POSITIVE
	POPJ	P,
	IDIV	T1,T2			;BPC MUST DIVIDE BPSC
	JUMPN	T2,CPOPJ
	SKIPLE	T1,HOMSPU(B)		;SATS PER UNIT
	CAILE	T1,^D100		;BE REASONABLE
	POPJ	P,
	SKIPLE	T2,HOMSIC(B)		;SATS IN CORE
	CAMLE	T2,T1
	POPJ	P,
	MOVE	T1,BIGBPU		;BIGGEST UNIT
	IDIV	T1,HOMBSC(B)		;SUPER CLUSTERS ON BIGGEST UNIT
	SKIPE	T2
	ADDI	T1,1
	CAMN	T1,HOMSCU(B)
	SKIPN	HOMSNM(B)		;MUST BE PART OF A STR
	POPJ	P,
	SKIPE	T1,HOMK4S(B)		;SWAP SPACE MUST FIT
	MOVE	T1,HOMSLB(B)
	MOVE	T2,HOMK4S(B)
	LSH	T2,3
	ADD	T1,T2
	CAMLE	T1,UDBBPU(U)
	POPJ	P,
	JRST	CPOPJ1
	SUBTTL	MFD TCB

;ROUTINE TO BUILD A TCB FOR MFD
;SKIP IF OK
BDMFD:	MOVEI	T1,SIZFIL		;ALLOCATE CORE FOR FIL
	PUSHJ	P,GETBLK
	MOVE	FL,T2
	MOVE	T1,MFDPPN		;FILENAME
	MOVEM	T1,FILNAM(FL)
	MOVSI	T1,'UFD'		;EXT
	MOVEM	T1,FILEXT(FL)
	MOVE	T1,MFDBN		;CFP
	IDIV	T1,BPSC
	HRLZM	T1,FILCFP(FL)
	MOVEI	T1,1			;USE COUNT
	DPB	T1,FIYCNT
	MOVEI	T1,SIZRIB		;ALLOCATE CORE FOR TCB
	PUSHJ	P,GETBLK
	MOVE	T1,MFDBN		;PUT TCB IN QUEUE
	PUSHJ	P,LNKTCB
	MOVE	TC,T2
	MOVEI	T1,FNCRIB		;FUNCTION CODE
	DPB	T1,TCYCOD
	SETZ	T1,			;PRIME RIB
	DPB	T1,TCYRBC
	HRRM	FL,TCBFIL(TC)		;ADDR OF FIL
	POPJ	P,
	SUBTTL	STORE HOME BLOCK INFO

;ROUTINE TO COPY INFO FROM HOME BLOCKS TO STR DATA BLOCK
;SKIP IF OK
MOVSTR:	MOVE	T1,HOMSNM(B)		;STRUCTURE NAME
	TRNN	F,F.STR
	MOVEM	T1,STRNAM
	CAME	T1,STRNAM
	POPJ	P,
	MOVE	T1,NUN			;HIGHEST LEGAL BN
	IMUL	T1,BIGBPU
	SUBI	T1,1
	MOVEM	T1,HLBN
	MOVEI	T1,T1			;BP TO CC
	HLL	T1,HOMCNP(B)
	TLZ	T1,77
	TRNN	F,F.STR
	MOVEM	T1,CCBP
	CAME	T1,CCBP
	POPJ	P,
	HLL	T1,HOMCKP(B)		;BP TO CS
	TLZ	T1,77
	TRNN	F,F.STR
	MOVEM	T1,CSBP
	CAME	T1,CSBP
	POPJ	P,
	LDB	T2,[POINT 6,CSBP,11]	;BITS PER CHECKSUM
	MOVEM	T2,BPCS
	MOVNS	T2
	HRRZM	T2,MBPCS
	HLL	T1,HOMCLP(B)		;BP TO CA
	TLZ	T1,77
	TRNN	F,F.STR
	MOVEM	T1,CABP
	CAME	T1,CABP
	POPJ	P,
	MOVE	T1,HOMBPC(B)		;BLOCKS PER CLUSTER
	TRNN	F,F.STR
	MOVEM	T1,BPC
	CAME	T1,BPC
	POPJ	P,
	MOVE	T1,HOMBSC(B)		;BLOCKS PER SUPER CLUSTER
	TRNN	F,F.STR
	MOVEM	T1,BPSC
	CAME	T1,BPSC
	POPJ	P,
	IMUL	T1,HOMSCU(B)		;BLOCKS PER (LARGEST) UNIT
	TRNN	F,F.STR
	MOVEM	T1,BPLU
	CAME	T1,BPLU
	POPJ	P,
	MOVE	T1,HOMPT1(B)		;BN FOR MFD
	TRNN	F,F.STR
	MOVEM	T1,SPT1
	CAME	T1,SPT1
	POPJ	P,
	LDB	T2,CABP
	IMUL	T2,BPC
	MOVE	T1,HOMUN1(B)
	TRNN	F,F.STR
	MOVEM	T1,SUN1
	CAME	T1,SUN1
	POPJ	P,
	IMUL	T1,BPLU
	ADD	T1,T2
	MOVEM	T1,MFDBN
	MOVE	T1,HOMOVR(B)		;OVERDRAW
	TRNN	F,F.STR
	MOVEM	T1,SOVR
	CAME	T1,SOVR
	POPJ	P,
	MOVEI	T1,DSKTRY		;RETRIES ON ERROR
	MOVEM	T1,STRY
	MOVE	T1,HOMPVS(B)		;NON-0 IF PRIVATE STR
	ANDI	T1,HOPPVS
	SKIPE	T1
	SETO	T1,
	TRNN	F,F.STR
	MOVEM	T1,SPVS
	CAME	T1,SPVS
	POPJ	P,
	MOVE	T1,HOMOPP(B)		;OWNER PPN
	TRNN	F,F.STR
	MOVEM	T1,SPPN
	CAME	T1,SPPN
	POPJ	P,
	MOVE	T1,HOMCRS(B)		;BN (REL TO STR) OF CRASH.EXE
	TRNN	F,F.STR
	MOVEM	T1,CRSBN
	CAME	T1,CRSBN
	POPJ	P,
	MOVE	T1,HOMSCU(B)		;SUPER CLUSTERS PER UNIT
	TRNN	F,F.STR
	MOVEM	T1,SCPU
	CAME	T1,SCPU
	POPJ	P,
	MOVE	T1,HOMK4C(B)		;K FOR CRASH
	TRNN	F,F.STR
	MOVEM	T1,SK4C
	CAME	T1,SK4C
	POPJ	P,
	TRO	F,F.STR			;STR DATA ACCEPTED
	JRST	CPOPJ1
;ROUTINE TO COPY INFO FROM HOME BLOCK TO UDB
MOVUDB:	MOVE	T1,HOMLUN(B)		;LOGICAL UNIT NUMBER
	MOVEM	T1,UDBLUN(U)
	IMUL	T1,BPLU			;LOWEST BLOCK
	MOVEM	T1,UDBBLK(U)
	MOVE	T1,UDBBPU(U)		;HIGHEST LEGAL BLOCK
	IDIV	T1,HOMBPC(B)
	IMUL	T1,HOMBPC(B)
	SUBI	T1,1
	MOVEM	T1,UDBHLB(U)
	MOVE	T1,HOMSPU(B)		;SATS PER UNIT
	MOVEM	T1,UDBSPU(U)
	MOVE	T1,UDBBPU(U)		;CLUSTERS PER SAT
	IDIV	T1,BPC
	SUBI	T1,1
	IDIV	T1,UDBSPU(U)
	ADDI	T1,1
	MOVEM	T1,UDBCPS(U)
	SUBI	T1,1			;WORDS PER SAT
	IDIVI	T1,^D36
	ADDI	T1,1
	MOVEM	T1,UDBWPS(U)
	MOVE	T1,HOMHID(B)		;PACK SERIAL NUMBER
	MOVEM	T1,UDBHID(U)
	MOVE	T1,HOMGRP(B)		;BLOCKS TO TRY ALLOCATE
	MOVEM	T1,UDBGRP(U)
	MOVE	T1,HOMBPC(B)		;BLOCKS PER CLUSTER
	MOVEM	T1,UDBBPC(U)
	MOVE	T1,HOMSIC(B)		;SATS IN CORE
	MOVEM	T1,UDBSIC(U)
	MOVE	T1,HOMSLB(B)		;1ST BLOCK FOR SWAPPING
	MOVEM	T1,UDBSLB(U)
	MOVE	T1,HOMK4S(B)		;K FOR SWAPPING
	MOVEM	T1,UDBK4S(U)
	MOVSI	T1,(FS.UWL)		;SOFTWARE WRITE LOCK
	AND	T1,WL
	MOVEM	T1,UDBAWL(U)
	MOVE	T1,HOMLUN(B)		;LOGICAL UNIT NAME
	IDIVI	T1,10
	SKIPE	T1
	ADDI	T1,'0'
	ADDI	T2,'0'
	ROT	T2,-6
	SKIPE	T1
	LSHC	T1,-6
	MOVE	T1,ALIAS
MVUDB1:	LSH	T1,-6
	TRNN	T1,77
	JRST	MVUDB1
MVUDB2:	LSHC	T1,6
	TLNN	T1,770000
	JRST	MVUDB2
	MOVEM	T1,UDBLOG(U)
	POPJ	P,
;ROUTINE TO SAVE CAPACITY
MOVCPY:	MOVE	T1,UDBNAM(U)		;DO A DSKCHR
	MOVEM	T1,FOO+.DCNAM
	MOVE	T1,[XWD FOOSIZ,FOO]
	DSKCHR	T1,
	 PUSHJ	P,DIE
	TLNE	T1,(DC.AWL+DC.HWP)	;WRITE LOCK?
	SETOM	WL			;YES
	SETZM	UDBHWP(U)
	TLNE	T1,(DC.HWP)		;HARDWARE WRITE PROTECT?
	SETOM	UDBHWP(U)		;YES
	MOVE	T1,FOO+.DCUSZ		;SAVE UNIT SIZE
	MOVEM	T1,UDBBPU(U)
	ADDM	T1,SSIZ			;SUM OF UDBBPU
	CAMLE	T1,BIGBPU		;BIGGEST UDBBPU?
	MOVEM	T1,BIGBPU		;YES
	LOAD	T1,FOO+.DCUCH,DC.UCY	;BLOCKS PER CYLINDER
	MOVEM	T1,UDBCYL(U)
	LOAD	T1,FOO+.DCUCH,DC.UCT	;BLOCKS PER TRACK
	MOVEM	T1,UDBBPT(U)
	IMULI	T1,NSKP			;BLOCKS TO SKIP
	ADDI	T1,DSKP/2		;ROUND
	IDIVI	T1,DSKP
	MOVEM	T1,UDBSKP(U)
	MOVE	T1,FOO+.DCPAS		;POSITION IN ASL
	MOVEM	T1,UDBASL(U)
	MOVE	T1,FOO+.DCPSD		;POSITION IN SDL
	MOVEM	T1,SDL
	SKIPE	T1,FOO+.DCALT		;DON'T OVERWRITE WITH ZERO
	MOVEM	T1,UDBALT(U)		;SAVE ALTERNATE UNIT
	TRNN	F,F.702			;7.01?
	PUSHJ	P,FNDCPU		;YES, FIND WHICH CPU
	POPJ	P,
;ROUTINE TO FIND WHICH CPU(S) THE DISK IS ON.
;WE ONLY NEED THIS INFORMATION IN 7.01, BUT
;7.01 IS THE ONLY MONITOR WHERE DSKCHR DOESN'T
;RETURN THIS INFO.
FNDCPU:	SETOM	UDBCPU(U)		;ASSUME ALL CPUS
	MOVE	T1,[%LDUNI]		;ADDR OF 1ST UDB
	GETTAB	T1,
	 HALT
FNCPU1:	HLRZS	T1			;IN RH
	JUMPE	T1,CPOPJ		;GIVE UP
	MOVEI	T2,UNINAM(T1)		;GET NAME OF UNIT
	PEEK	T2,
	CAME	T2,UDBNAM(U)		;OURS?
	CAMN	T2,UDBALT(U)
	JRST	FNCPU2			;YES
	MOVEI	T1,UNISYS(T1)		;NO, GET ADDR OF NEXT UDB
	PEEK	T1,
	JRST	FNCPU1			;LOOP
FNCPU2:	MOVEI	T2,UNIKON(T1)		;ADDR OF KDB
	PEEK	T2,
	MOVEI	T2,KONCPU(T2)		;CPU WORD
	PEEK	T2,
	LSH	T2,-41			;BIT MASK
	MOVEI	T3,SP.CR0
	LSH	T3,(T2)
	MOVEM	T3,UDBCPU(U)
	SKIPN	UDBALT(U)		;2ND PORT?
	POPJ	P,			;NO
	MOVEI	T1,UNI2ND(T1)		;YES, GET ADDR OF 2ND UDB
	PEEK	T1,
	MOVEI	T2,UNINAM(T1)		;DOUBLE CHECK
	PEEK	T2,
	CAME	T2,UDBNAM(U)
	CAMN	T2,UDBALT(U)
	CAIA
	POPJ	P,
	MOVEI	T1,UNIKON(T1)		;2ND KDB
	PEEK	T1,
	MOVEI	T1,KONCPU(T1)		;CPU WORD
	PEEK	T1,
	LSH	T1,-41			;BIT MASK
	MOVEI	T2,SP.CR0
	LSH	T2,(T1)
	IORM	T2,UDBCPU(U)
	POPJ	P,
	SUBTTL	CHECK A RIB

;ROUTINE TO TEST FOR A VALID RIB
;NOSKIP IF RIB ERROR
;SKIP IF RIB IS OK
RBCK:	MOVE	T1,TCBBLK(TC)		;BN MUST POINT TO SELF
	IDIV	T1,BPLU
	MOVE	T1,RIBCOD(B)		;MAGIC NUMBER MUST BE RIGHT
	CAIN	T1,CODRIB
	CAME	T2,RIBSLF(B)
	POPJ	P,
	MOVE	T1,FILNAM(FL)		;CHECK FILENAME AND EXT
	MOVE	T2,FILEXT(FL)
	XOR	T2,RIBEXT(B)
	CAMN	T1,RIBNAM(B)
	TLNE	T2,-1
	POPJ	P,
	HRRZ	T2,RIBFIR(B)		;1ST RTP MUST BE IN RANGE
	CAIL	T2,RIBTIM+1
	CAILE	T2,RIBCOD-1
	POPJ	P,
	HLRE	T3,RIBFIR(B)		;LAST RTP MUST BE IN RANGE
	SUBM	T2,T3
	CAIL	T3,1(T2)
	CAILE	T3,RIBCOD
	POPJ	P,
	LDB	T3,TCYRBC		;GET RIB COUNT
	SKIPE	T3			;EXTENDED RIB?
	SKIPA	T4,TCBFLR(TC)		;YES, GET 1ST LOGICAL RECORD
	MOVEI	T4,1			;NO, BLOCK ONE
	SUBI	T4,1			;LAST BLOCK IN PREVIOUS RIB
	CAME	T4,RIBFLR(B)		;CHECK IT
	POPJ	P,
	ADD	T2,B			;GET 1ST RTP
	MOVE	T1,(T2)
	MOVE	T4,UDBLUN(U)		;IS IT UNIT POINTER?
	CAIN	T1,RIPNUB(T4)
	AOSA	T2			;YES, POINT TO 1ST GROUP
	JUMPE	T3,CPOPJ		;NO, ERROR UNLESS EXTENDED RIB
	MOVE	T1,(T2)			;GET 1ST GROUP POINTER
	LDB	T1,CABP			;GET CLUSTER ADDRESS
	IMUL	T1,BPC			;MUST POINT TO SELF
	CAME	T1,RIBSLF(B)
	POPJ	P,
	MOVE	T1,RIBUFD(B)		;CHECK UPWARD POINTER
	PUSHJ	P,CTINT			;CONVERT TO INTERNAL FORMAT
	IDIV	T1,BPSC
	JUMPN	T2,CPOPJ
	HLRZ	T3,FILCFP(FL)
	HRRZ	T2,FILDAD(FL)
	SKIPE	T2
	HLRZ	T3,FILCFP(T2)
	CAME	T1,T3
	POPJ	P,
	MOVE	T1,FL			;FIND UFD
RBCK1:	HRRZ	T2,FILDAD(T1)
	SKIPE	T2
	MOVE	T1,T2
	HLRZ	T2,FILEXT(T1)
	CAIE	T2,'UFD'
	JRST	RBCK1
	MOVE	T2,FILNAM(T1)
	CAMN	T2,RIBPPN(B)
	AOS	(P)
	POPJ	P,
	SUBTTL	CHECKSUMING

;READ A DATA BLOCK AND COMPUTE THE CHECKSUM
DOSUM:	PUSHJ	P,REDTCB		;READ THE BLOCK
	 JRST	IOERR			;I/O ERROR
	MOVE	T1,0(B)			;COMPUTE THE CHECKSUM
	PUSHJ	P,FOLD
	CAMN	T2,TCBSUM(TC)		;IS IT RIGHT?
	POPJ	P,			;YES
	MOVEI	T2,[ASCIZ /Checksum error on file /]
	PUSHJ	P,STRO
	PUSHJ	P,FILO			;TYPE FILESPEC
	PUSHJ	P,CRLFO
	MOVEI	T2,[ASCIZ /Block /]
	PUSHJ	P,STRO
	MOVE	T1,TCBREL(TC)
	PUSHJ	P,OCTO
	MOVEI	T2,[ASCIZ / relative to file, cluster /]
	PUSHJ	P,STRO
	MOVE	T1,TCBBLK(TC)
	IDIV	T1,BPC
	PUSHJ	P,OCTO
	PJRST	DSPACE

DIRERR:	SKIPA	T2,[[ASCIZ /Directory is overwritten with garbage /]]
IOERR:	MOVEI	T2,[ASCIZ /Error while reading file /]
	PUSHJ	P,STRO
	PUSHJ	P,FILO			;TYPE FILESPEC
	PUSHJ	P,CRLFO
	MOVEI	T2,[ASCIZ /Block /]
	PUSHJ	P,STRO
	MOVE	T1,TCBREL(TC)
	PUSHJ	P,OCTO
	MOVEI	T2,[ASCIZ / relative to file, block /]
	PUSHJ	P,STRO
	MOVE	T1,TCBBLK(TC)
	PUSHJ	P,BNO
	MOVEI	T2,[ASCIZ / relative to STR/]
	PUSHJ	P,STRO
DSPACE:	PUSHJ	P,CRLFO
	TRNE	F,F.TTY			;BREAK OUTPUT
	PUSHJ	P,BUFO
	PJRST	CRLFO

;ROUTINE TO COMPUTE FOLDED CHECKSUM
;T1 PASSES WORD TO BE FOLDED
;T2 RETURNS CHECKSUM
FOLD:	SETZ	T2,
	LSHC	T1,@MBPCS
	ROT	T2,@BPCS
	JUMPE	T1,CPOPJ
	ADD	T1,T2
	JRST	FOLD
	SUBTTL	CORE MANAGEMENT

;ROUTINE TO DECREMENT USE COUNT OF FIL
DECFIL:	LDB	T1,FIYCNT		;DECREMENT COUNT
	SUBI	T1,1
	DPB	T1,FIYCNT
	JUMPN	T1,CPOPJ		;OTHER USERS, LEAVE ALONE
	PUSH	P,FL			;LAST USER, SAVE ADDR
	HRRZ	FL,FILDAD(FL)		;GET ADDR OF PARENT
	SKIPE	FL			;MFD?
	PUSHJ	P,DECFIL		;NO, DECREMENT PARENT'S COUNT
	POP	P,FL			;RESTORE ADDR OF SON
	FALL	GIVFIL			;RETURN SON TO FREE LIST

;ROUTINE TO GIVE A FIL BACK TO FREE CORE
GIVFIL:	MOVEI	T1,SIZFIL
	MOVE	T2,FL
	PJRST	GIVBLK

;ROUTINE TO GIVE A TCB AWAY
GIVTCB:	LDB	T3,TCYCOD		;GET FUNCTION CODE
	MOVE	T1,FUNSZ(T3)		;SIZE
	MOVE	T2,TC			;ADDR
	PJRST	@FUNGV(T3)		;DISPATCH

;ROUTINE TO GIVE A RIB TCB AWAY
GIVRIB:	LDB	T3,TCYRBC		;PRIME OR EXTENDED?
	JUMPE	T3,GIVBLK
	MOVEI	T1,SIZXRB		;EXTENDED
	MOVE	T3,TCBCOR(TC)		;CORRUPT?
	TLNE	T3,(TCPCOR)
	MOVEI	T1,SIZCXR		;YES, CORRUPT EXTENDED RIB
	PJRST	GIVBLK
;ROUTINE TO GET A BLOCK OF CORE AND ZERO IT
;T1 PASSES SIZE OF BLK (NOT PRESERVED)
;T2 RETURNS ADDR OF BLK
GETZER:	PUSHJ	P,GETBLK		;ALLOCATE THE CORE
	SETZM	(T2)			;ZERO 1ST WORD
	CAIN	T1,1			;ONLY ONE WORD?
	POPJ	P,			;YES, ALL DONE
	HRLZI	T3,0(T2)		;BLT THE REST
	HRRI	T3,1(T2)
	ADDI	T1,-1(T2)
	BLT	T3,(T1)
	POPJ	P,

;ROUTINE TO GET A CORE BLOCK
;T1 PASSES SIZE OF BLK (PRESERVED)
;T2 RETURNS ADDR OF BLK
GETBLK:	ADDM	T1,WRDCNT		;BUMP COUNT
	PUSHJ	P,TRYBLK		;ANY FREE BLOCKS?
	 POPJ	P,			;YES
	MOVE	T2,.JBFF		;NO, ENOUGH CORE TO MAKE 1?
	ADD	T2,T1
	CAMG	T2,.JBREL
	JRST	GTBLK1			;YES
	PUSHJ	P,GC			;NO, GARBAGE COLLECT
	PUSHJ	P,CDOWN			;CORE DOWN
	PUSHJ	P,TRYBLK		;TRY AGAIN
	 POPJ	P,			;WIN
	MOVE	T2,.JBFF		;STILL LOSE, GET MORE CORE
	ADD	T2,T1
	MOVE	T3,T2
	TRNN	F,F.HW
	ADDI	T3,<UPN-1>*PAGSIZ
	CORE	T3,
	 PUSHJ	P,DIE
GTBLK1:	EXCH	T2,.JBFF		;T2=ADDR OF BLK
	POPJ	P,

;ROUTINE TO CORE DOWN
;THIS ROUTINE MAY ONLY BE CALLED IMMEDIATELY AFTER GC
;PRESERVES T1
CDOWN:	HRRZ	T2,FREMEM		;LAST FREE NODE
	JUMPE	T2,CPOPJ
	HLRZ	T3,(T2)			;SIZE OF LAST NODE
	CAIGE	T3,DOWNN*PAGSIZ		;BIG ENOUGH?
	POPJ	P,			;NO
	ADD	T3,T2			;1ST ADDR BEYOND LAST NODE
	CAME	T3,.JBFF		;LAST NODE IS AT VERY END?
	POPJ	P,			;NO
	HRRZ	T3,(T2)			;UNLINK LAST NODE
	HRRM	T3,FREMEM
	MOVEM	T2,.JBFF		;NEW VALUE OF 1ST FREE
	CORE	T2,			;PITCH SOME CORE
	 PUSHJ	P,DIE			;MONITOR BUG?
	POPJ	P,
;ROUTINE TO TRY TO FIND A FREE CORE BLK
;T1 PASSES SIZE OF BLK (PRESERVED)
;T2 RETURNS ADDR OF BLK
;SKIP IF FAIL
TRYBLK:	PUSHJ	P,SAVE3			;SAVE P1-P3
	MOVEI	P1,-1			;FLAG NONE SO FAR
	MOVEI	P2,FREMEM		;POINT TO 0TH FREE BLK
TRYLOP:	MOVE	P3,P2			;ADVANCE TO NEXT BLK
	HRRZ	P2,(P3)
	JUMPE	P2,TRY1			;QUIT IF NO MORE BLKS
	HLRZ	T4,(P2)			;GET SIZE OF BLK
	CAML	T4,T1			;BIG ENOUGH?
	CAML	T4,P1			;AND SMALLEST SO FAR?
	JRST	TRYLOP			;NO
	SUB	T4,T1			;YES, COMPUTE SIZE OF LEFTOVER
	CAIGE	T4,SIZGAR		;IS IT JUST A CRUMB?
	JUMPN	T4,TRYLOP		;YES, DON'T LEAVE CRUMBS AROUND
	HLRZ	P1,(P2)			;REMEMBER WHERE IT IS
	MOVE	T3,P3
	JUMPN	T4,TRYLOP		;GO IF NOT PERFECT MATCH
TRY1:	CAIN	P1,-1			;QUIT IF NO WINNERS AT ALL
	JRST	CPOPJ1
	HRRZ	T2,(T3)			;ADDR OF BEST
	CAMG	P1,T1			;TOO BIG?
	JRST	TRYESY			;NO, JUST RIGHT
	MOVE	P2,T2			;COMPUTE ADDR OF LEFTOVER
	ADD	P2,T1
	SUB	P1,T1			;COMPUTE SIZE OF LEFTOVER
	HRL	P1,(T2)			;SPLIT INTO TWO BLKS
	MOVSM	P1,(P2)
	HRL	P2,T1
	MOVEM	P2,(T2)
TRYESY:	HRRZ	T4,(T2)			;UNLINK THE BLK
	HRRM	T4,(T3)
	POPJ	P,

;ROUTINE TO RETURN A BLOCK OF CORE TO FREE LIST
;T1 PASSES SIZE OF BLK
;T2 PASSES ADDR OF BLK
GIVBLK:
IFN FTGCHK,<
	CAIGE	T1,SIZGAR
	PUSHJ	P,DIE
>
	HRRZ	T3,FREMEM		;ADD TO HEAD OF LIST
	HRL	T3,T1
	MOVEM	T3,(T2)
	MOVEM	T2,FREMEM
	MOVNS	T1			;MAINTAIN COUNT
	ADDM	T1,WRDCNT
	POPJ	P,
;GARBAGE COLLECT ROUTINE
;COMBINES CONSECUTIVE FRAGMENTS
;T1 IS PRESERVED
GC:	PUSH	P,T1			;SAVE T1
	PUSH	P,QTCB			;SAVE THE TCB TREE
	PUSH	P,ROOT
	MOVE	T1,[XWD QTCB,QTCB]	;BUILD A NULL TREE
	MOVEM	T1,QTCB
	SETZM	ROOT
GC1:	HRRZ	T2,FREMEM		;ADDR OF NEXT FREE NODE
	JUMPE	T2,GC2
	HRRZ	T1,(T2)			;UNLINK IT FROM FREE LIST
	HRRM	T1,FREMEM
	HLRZ	T1,(T2)			;SIZE OF NODE
	PUSHJ	P,LNKGAR		;LINK IT INTO THE TREE
IFN FTGCHK,<
	PUSHJ	P,CHECK			;CHECK CONSISTENCY
>
	JRST	GC1			;LOOP
GC2:	HRRZ	T2,QTCB			;ADDR OF NEXT NODE
	CAIN	T2,QTCB			;QUIT IF LAST NODE
	JRST	GC3
	HRRZ	T1,TCBLNK(T2)		;UNLINK IT FROM TREE
	HRRM	T1,QTCB
	LDB	T1,TCXGAR		;LINK IT TO FREE LIST
	HRL	T1,FREMEM
	MOVSM	T1,(T2)
	HRRM	T2,FREMEM
	JRST	GC2			;LOOP
GC3:	POP	P,ROOT			;RESTORE TCB TREE
	POP	P,QTCB
	JRST	TPOPJ			;RESTORE T1
;ROUTINE TO LINK A FREE NODE INTO THE TREE
;USED ONLY FOR GC
;T1 PASSES SIZE OF BLK
;T2 PASSES ADDR OF BLK
LNKGAR:
IFN FTDBUG,<
	CAIGE	T1,SIZGAR
	PUSHJ	P,DIE
>
	DPB	T1,TCXGAR		;STORE SIZE
	SETZM	TCBSON(T2)		;NO SONS
	MOVEI	T4,BALZ			;WE ARE BALANCED
	DPB	T4,TCZBAL
	SKIPN	T4,ROOT			;1ST TIME?
	JRST	LNKGR4			;YES
LNKGR1:	MOVE	T3,T4			;ADVANCE TO NEXT NODE
	CAML	T2,T3			;TOO HIGH OR TOO LOW?
	JRST	LNKGR2			;TOO HIGH, GO RIGHT
	HLRZ	T4,TCBSON(T3)		;TOO LOW, GET LEFT SON
	JUMPN	T4,LNKGR1		;AND LOOP
	LDB	T4,TCXGAR		;JUST BEFORE PARENT?
	ADD	T4,T2
	CAME	T4,T3
	JRST	LNKG10			;NO
	LDB	T4,TCXGAR		;YES, TOO BIG TO COMBINE?
	LDB	T1,TCWGAR
	ADD	T4,T1
	CAIG	T4,TCMGAR
	JRST	LNKGR6			;NO, GO COMBINE
LNKG10:	HLRZ	T4,TCBLNK(T3)		;PRED
	CAIN	T4,QTCB
	JRST	LNKGR7			;NO PRED
	LDB	T1,TCZGAR		;JUST AFTER PRED?
	ADD	T1,T4
	CAME	T1,T2
	JRST	LNKGR7			;NO
	LDB	T1,TCZGAR		;YES, TOO BIG TO COMBINE?
	LDB	T4,TCXGAR
	ADD	T1,T4
	HLRZ	T4,TCBLNK(T3)
	CAIG	T1,TCMGAR
	JRST	LNKGR8			;NO, GO COMBINE
LNKGR7:	HRLM	T2,TCBSON(T3)		;MAKE US THE LEFT SON
	HRLM	T3,TCBRNT(T2)
;HERE WITH T4=PRED, T3=SUCC
LNKGR3:	HRRM	T2,TCBLNK(T4)		;WE ARE SUCC OF PRED
	HRLM	T2,TCBLNK(T3)		;WE ARE PRED OF SUCC
	HRL	T3,T4			;PRED,,SUCC
	MOVEM	T3,TCBLNK(T2)
	PUSH	P,T2			;PRESERVE ADDR OF NODE
	PUSHJ	P,ADDND1		;ADD NODE TO BALANCE FACTOR
T2POPJ:	POP	P,T2
	POPJ	P,

LNKGR2:	HRRZ	T4,TCBSON(T3)		;GET RIGHT SON
	JUMPN	T4,LNKGR1		;AND LOOP
	LDB	T4,TCWGAR		;JUST AFTER PARENT?
	ADD	T4,T3
	CAME	T4,T2
	JRST	LNKG11			;NO
	LDB	T4,TCWGAR		;YES, TOO BIG TO COMBINE?
	LDB	T1,TCXGAR
	ADD	T4,T1
	CAIG	T4,TCMGAR
	JRST	LNKGR5			;NO, GO COMBINE
LNKG11:	HRRZ	T4,TCBLNK(T3)		;GET SUCC
	LDB	T1,TCXGAR		;JUST BEFORE SUCC?
	ADD	T1,T2
	CAME	T1,T4
	JRST	LNKG12			;NO
	LDB	T1,TCZGAR		;YES, TOO BIG TO COMBINE?
	LDB	T4,TCXGAR
	ADD	T1,T4
	HRRZ	T4,TCBLNK(T3)
	CAIG	T1,TCMGAR
	JRST	LNKGR9			;NO, GO COMBINE
LNKG12:	HRRM	T2,TCBSON(T3)		;MAKE US THE RIGHT SON
	HRLM	T3,TCBRNT(T2)
	EXCH	T3,T4			;OTHER WAY AROUND
	JRST	LNKGR3

;HERE TO INSERT THE VERY FIRST NODE
LNKGR4:	MOVEM	T2,ROOT			;WE BECOME THE ROOT
	HRRZS	TCBRNT(T2)		;NO PARENT
	MOVE	T3,[XWD QTCB,QTCB]	;LINK US
	MOVEM	T3,TCBLNK(T2)
	HRRZM	T2,QTCB
	HRLM	T2,QTCB
	POPJ	P,

;HERE TO INSERT JUST AFTER T3
LNKGR5:	MOVE	T4,T3
;HERE TO INSERT JUST AFTER T4
LNKGR8:	LDB	T1,TCXGAR		;TOTAL SIZE
	LDB	T3,TCZGAR
	ADD	T1,T3
	DPB	T1,TCZGAR
	HRRZ	T2,TCBLNK(T4)		;NEXT
	ADD	T1,T4			;JUST BEFORE NEXT?
	CAME	T1,T2
	POPJ	P,
	LDB	T1,TCZGAR		;YES, TOO BIG TO COMBINE?
	LDB	T3,TCXGAR
	ADD	T1,T3
	CAILE	T1,TCMGAR
	POPJ	P,			;YES, DON'T COMBINE
;HERE WHEN WE JUST INSERTED A NODE WHICH IS AN EXACT FIT
;FOR WHAT WAS MISSING BETWEEN TWO OTHER NODES. COMBINE INTO A SINGLE NODE.
;HERE WITH T4=LOW NODE, T2=HIGH NODE
LNKGR0:	PUSH	P,T4			;LOW NODE
	PUSH	P,T2			;HIGH NODE
	PUSH	P,TC			;SAVE TC
	MOVE	TC,T2			;UNLINK HIGH NODE
	PUSHJ	P,UNLINK
	PUSHJ	P,UNLNK
	POP	P,TC			;RESTORE TC
	POP	P,T2			;HIGH NODE
	POP	P,T4			;LOW NODE
	JRST	LNKGR8			;GO COMBINE

;HERE TO INSERT JUST BEFORE T3
LNKGR6:	MOVE	T4,T3
;HERE TO INSERT JUST BEFORE T4
LNKGR9:	HLRZ	T1,TCBRNT(T4)		;COPY PARENT
	HRLM	T1,TCBRNT(T2)
	SKIPN	T1
	MOVEI	T1,ROOT-TCBSON
	HLRZ	T3,TCBSON(T1)		;LEFT SON OF PARENT OR RIGHT?
	CAMN	T3,T4
	HRLM	T2,TCBSON(T1)		;LEFT
	CAME	T3,T4
	HRRM	T2,TCBSON(T1)		;RIGHT
	HLRZ	T1,TCBLNK(T4)		;PRED
	HRRM	T2,TCBLNK(T1)		;PRED HAS NEW SUCC
	HRRZ	T1,TCBLNK(T4)		;SUCC
	HRLM	T2,TCBLNK(T1)		;SUCC HAS NEW PRED
	HLRZ	T1,TCBSON(T4)		;LEFT SON HAS NEW PARENT
	SKIPE	T1
	HRLM	T2,TCBRNT(T1)
	HRRZ	T1,TCBSON(T4)		;RIGHT SON HAS NEW PARENT
	SKIPE	T1
	HRLM	T2,TCBRNT(T1)
	MOVE	T1,TCBSON(T4)		;COPY SONS
	MOVEM	T1,TCBSON(T2)
	MOVE	T1,TCBLNK(T4)		;COPY LINKS
	MOVEM	T1,TCBLNK(T2)
	LDB	T1,TCVBAL		;COPY BALANCE FACTOR
	DPB	T1,TCZBAL
	LDB	T1,TCXGAR		;TOTAL SIZE
	LDB	T3,TCZGAR
	ADD	T1,T3
	DPB	T1,TCXGAR
	HLRZ	T4,TCBLNK(T2)		;PRED
	CAIN	T4,QTCB
	POPJ	P,			;NO PRED
	LDB	T1,TCZGAR		;JUST AFTER PRED?
	ADD	T1,T4
	CAME	T1,T2
	POPJ	P,			;NO
	LDB	T1,TCXGAR		;YES, TOO BIG TO COMBINE?
	LDB	T3,TCZGAR
	ADD	T1,T3
	CAIG	T1,TCMGAR
	JRST	LNKGR0			;NO, GO COMBINE
	POPJ	P,			;YES, DON'T COMBINE
;ROUTINE TO INSERT A TCB INTO THE QUEUE
;T1 PASSES BN
;T2 PASSES ADDR OF TCB (PRESERVED)
LNKTCB:	MOVEM	T1,TCBBLK(T2)		;STORE BN
	SETZB	T4,TCBSON(T2)		;WE HAVE NO SONS
	DPB	T4,TCZCX		;NOT YET LOADED INTO CACHE
	MOVSI	T4,(TCPMBB)		;NOT BAD
	ANDCAM	T4,TCBMBB(T2)
	MOVEI	T4,BALZ			;WE ARE BALANCED
	DPB	T4,TCZBAL
	SKIPN	T4,ROOT			;1ST TIME?
	JRST	LNKTC4			;YES
LNKTC1:	MOVE	T3,T4			;ADVANCE TO NEXT NODE
	CAML	T1,TCBBLK(T3)		;TOO HIGH OR TOO LOW?
	JRST	LNKTC2			;TOO HIGH, GO RIGHT
	HLRZ	T4,TCBSON(T3)		;TOO LOW, GET LEFT SON
	JUMPN	T4,LNKTC1		;AND LOOP
	HRLM	T2,TCBSON(T3)		;NONE, MAKE US THE LEFT SON
	HRLM	T3,TCBRNT(T2)
	HLRZ	T4,TCBLNK(T3)		;GET PRED
;HERE WITH T4=PRED, T3=SUCC
LNKTC3:	HRRM	T2,TCBLNK(T4)		;WE ARE SUCC OF PRED
	HRLM	T2,TCBLNK(T3)		;WE ARE PRED OF SUCC
	HRL	T3,T4			;PRED,,SUCC
	MOVEM	T3,TCBLNK(T2)
	PUSH	P,T2			;PRESERVE ADDR OF NODE
	PUSHJ	P,ADDND1		;ADD NODE TO BALANCE FACTOR
IFN FTCHK,<
	PUSHJ	P,CHECK			;CHECK CONSISTENCY
>
	JRST	T2POPJ

LNKTC2:	HRRZ	T4,TCBSON(T3)		;GET RIGHT SON
	JUMPN	T4,LNKTC1		;AND LOOP
	HRRM	T2,TCBSON(T3)		;NONE, MAKE US THE RIGHT SON
	HRLM	T3,TCBRNT(T2)
LNKTC5:	HRRZ	T4,TCBLNK(T3)		;GET SUCC
	EXCH	T3,T4			;OTHER WAY AROUND
	JRST	LNKTC3

LNKTC4:	MOVEM	T2,ROOT			;WE BECOME THE ROOT
	HRRZS	TCBRNT(T2)		;NO PARENT
	HRRZ	T3,QTCB			;LINK US
	JRST	LNKTC5
;ROUTINE TO UNLINK THE TCB FROM THE QUEUE
UNLNK:	HLRZ	T1,TCBLNK(TC)		;UNLINK TCB FROM QUEUE
	HRRZ	T2,TCBLNK(TC)
	HRRM	T2,TCBLNK(T1)
	HRLM	T1,TCBLNK(T2)
	POPJ	P,

;ROUTINE TO UNLINK THE TCB FROM THE TREE
UNLINK:	HLRZ	T1,TCBSON(TC)		;DO WE HAVE A LEFT SON?
	JUMPE	T1,UNLNK6		;NO, EASY
	HRRZ	T1,TCBSON(TC)		;DO WE HAVE A RIGHT SON?
	JUMPE	T1,UNLNK6		;NO, EASY

;HERE WHEN WE HAVE BOTH LEFT AND RIGHT SONS
	MOVE	T1,TC			;NODE TO MOVE OUT
	LDB	T4,TCXBAL		;ITS BALANCE FACTOR
	HRRZ	T2,TCBLNK(T1)		;NODE TO MOVE IN
	CAIL	T4,BALZ
	HLRZ	T2,TCBLNK(T1)
IFN FTDBUG,<
	HLRZ	T3,TCBSON(T2)		;LEFT CORNER CAN'T HAVE LEFT SON
	CAIL	T4,BALZ
	HRRZ	T3,TCBSON(T2)		;RIGHT CORNER CAN'T HAVE RIGHT SON
	SKIPE	T3
	PUSHJ	P,DIE
>
	PUSH	P,T1			;NODE TO MOVE OUT
	PUSH	P,T2			;NODE TO MOVE IN
	PUSHJ	P,ORPHAN		;UNLINK CORNER NODE FROM ITS OLD POSITION
	POP	P,T2			;NODE TO MOVE IN
	POP	P,T1			;NODE TO MOVE OUT
	PJRST	SWAP			;T2 NODE TAKES PLACE OF T1 NODE

;HERE WHEN WE DON'T HAVE BOTH SONS
UNLNK6:	MOVE	T2,TC			;NODE TO UNLINK
	PJRST	ORPHAN			;BLAST IT
;ROUTINE TO ADD A NODE TO THE BALANCE FACTOR
;T1 PASSES ADDR OF NODE JUST ADDED
ADDNOD:	MOVE	T2,T1			;WANT IT IN OTHER AC
;HERE WITH ADDR OF NODE IN T2
ADDND1:	HLRZ	T1,TCBRNT(T2)		;PARENT
	JUMPE	T1,CPOPJ		;NOBODY TO ADD TO
	MOVEI	T3,1			;ASSUME LEFT SON
	HLRZ	T4,TCBSON(T1)		;LEFT SON?
	CAME	T4,T2
	SETO	T3,			;NO, RIGHT SON
	LDB	T4,TCXBAL		;PARENT'S BALANCE FACTOR
	ADD	T4,T3			;UPDATE BALANCE FACTOR
	DPB	T4,TCXBAL
	CAIL	T4,BALM			;WITHIN TOLERANCE?
	CAILE	T4,BALP
	PUSHJ	P,REBAL			;NO, REBALANCE THE TREE
	CAIE	T4,BALZ			;PERFECTLY BALANCED?
	JRST	ADDNOD			;NO, ADJUST GRANDPA TOO
	POPJ	P,			;YES
;ROUTINE TO SUBTRACT A NODE FROM THE BALANCE FACTOR
;T1 PASSES ADDR OF PARENT (OF NODE JUST DELETED)
;T3 PASSES DELTA FACTOR
SUBNOD:	LDB	T4,TCXBAL		;GET PARENT'S BALANCE FACTOR
	ADD	T4,T3			;ADJUST BALANCE FACTOR
	DPB	T4,TCXBAL
	CAIL	T4,BALM			;WITHIN TOLERANCE?
	CAILE	T4,BALP
	PUSHJ	P,REBAL			;NO, REBALANCE THE TREE
	CAIE	T4,BALZ			;PERFECTLY BALANCED?
	POPJ	P,			;NO, ALL DONE
	MOVE	T2,T1			;YES, SAVE ADDR OF PARENT
	HLRZ	T1,TCBRNT(T2)		;GET ADDR OF GRANDPA
	JUMPE	T1,CPOPJ		;NONE
	MOVEI	T3,1			;ASSUME FATHER IS RIGHT SON OF GRANDPA
	HRRZ	T4,TCBSON(T1)		;TRUE?
	CAME	T4,T2
	SETO	T3,			;NO, LEFT SON
	JRST	SUBNOD			;UPDATE GRANDPA'S BALANCE FACTOR
;ROUTINE TO REBALANCE THE TREE
;I.E. MAKE BALANCE FACTOR BETTER BY ROTATING THE ROOT NODE
;T1 PASSES ROOT OF UNBALANCED SUBTREE
;T4 PASSES BALANCE FACTOR OF ROOT
;T1 RETURNS NEW ROOT
;T4 RETURNS NEW BALANCE FACTOR OF NEW ROOT
REBAL:	PUSHJ	P,SAVE2			;SAVE ACS
	HLRZ	P1,TCBRNT(T1)		;ORIGINAL PARENT OF ORIGINAL ROOT
	CAIL	T4,BALZ			;RIGHT HEAVY OR LEFT HEAVY?
	JRST	REBL			;LEFT HEAVY
	FALL	REBR			;RIGHT HEAVY

;HERE IF RIGHT HEAVY
REBR:	HRRZ	T2,TCBSON(T1)		;B
IFN FTDBUG,<
	SKIPN	T2
	PUSHJ	P,DIE
>
	HLRZ	T3,TCBSON(T2)		;Y OR C
	LDB	T4,TCZBAL		;B BALANCE
IFN FTDBUG,<
	CAIL	T4,BALM
	CAILE	T4,BALP
	PUSHJ	P,DIE
>
	CAIN	T4,BALP
	JRST	REBR3
	MOVE	P2,[EXP BALZ,BALM]-BALM(T4)
	DPB	P2,TCXBAL
	MOVE	P2,[EXP BALZ,BALP]-BALM(T4)
	DPB	P2,TCZBAL
	HRRM	T3,TCBSON(T1)
	SKIPE	T3
	HRLM	T1,TCBRNT(T3)
	HRLM	T1,TCBSON(T2)
	HRLM	T2,TCBRNT(T1)
	JRST	REBCM

;CASE R3
REBR3:
IFN FTDBUG,<
	SKIPN	T3
	PUSHJ	P,DIE
>
	LDB	T4,TCWBAL
IFN FTDBUG,<
	CAIL	T4,BALM
	CAILE	T4,BALP
	PUSHJ	P,DIE
>
	MOVEI	P2,BALZ
	DPB	P2,TCWBAL
	MOVE	P2,[EXP BALP,BALZ,BALZ]-BALM(T4)
	DPB	P2,TCXBAL
	MOVE	P2,[EXP BALZ,BALZ,BALM]-BALM(T4)
	DPB	P2,TCZBAL
	HLRZ	T4,TCBSON(T3)		;Y1
	HRRM	T4,TCBSON(T1)
	SKIPE	T4
	HRLM	T1,TCBRNT(T4)
	HRRZ	T4,TCBSON(T3)		;Y2
	HRLM	T4,TCBSON(T2)
	SKIPE	T4
	HRLM	T2,TCBRNT(T4)
	HRLZM	T1,TCBSON(T3)
	HRLM	T3,TCBRNT(T1)
	HRRM	T2,TCBSON(T3)
	HRLM	T3,TCBRNT(T2)
	JRST	REBCOM
;HERE IF LEFT HEAVY
REBL:	HLRZ	T2,TCBSON(T1)		;B
IFN FTDBUG,<
	SKIPN	T2
	PUSHJ	P,DIE
>
	HRRZ	T3,TCBSON(T2)		;Y OR C
	LDB	T4,TCZBAL		;B BALANCE
IFN FTDBUG,<
	CAIL	T4,BALM
	CAILE	T4,BALP
	PUSHJ	P,DIE
>
	CAIN	T4,BALM
	JRST	REBL3
	MOVE	P2,[EXP BALP,BALZ]-BALZ(T4)
	DPB	P2,TCXBAL
	MOVE	P2,[EXP BALM,BALZ]-BALZ(T4)
	DPB	P2,TCZBAL
	HRLM	T3,TCBSON(T1)
	SKIPE	T3
	HRLM	T1,TCBRNT(T3)
	HRRM	T1,TCBSON(T2)
	HRLM	T2,TCBRNT(T1)
	JRST	REBCM

;CASE L3
REBL3:
IFN FTDBUG,<
	SKIPN	T3
	PUSHJ	P,DIE
>
	LDB	T4,TCWBAL
IFN FTDBUG,<
	CAIL	T4,BALM
	CAILE	T4,BALP
	PUSHJ	P,DIE
>
	MOVEI	P2,BALZ
	DPB	P2,TCWBAL
	MOVE	P2,[EXP BALZ,BALZ,BALM]-BALM(T4)
	DPB	P2,TCXBAL
	MOVE	P2,[EXP BALP,BALZ,BALZ]-BALM(T4)
	DPB	P2,TCZBAL
	HLRZ	T4,TCBSON(T3)		;Y2
	HRRM	T4,TCBSON(T2)
	SKIPE	T4
	HRLM	T2,TCBRNT(T4)
	HRRZ	T4,TCBSON(T3)		;Y1
	HRLM	T4,TCBSON(T1)
	SKIPE	T4
	HRLM	T1,TCBRNT(T4)
	HRLZM	T2,TCBSON(T3)
	HRLM	T3,TCBRNT(T2)
	HRRM	T1,TCBSON(T3)
	HRLM	T3,TCBRNT(T1)
	FALL	REBCOM
;COMMON EXIT FOR REBAL
REBCOM:	MOVE	T2,T3

;HERE WITH:
;T1=OLD ROOT
;T2=NEW ROOT
;P1=OLD PARENT OF OLD ROOT
REBCM:	HRLM	P1,TCBRNT(T2)
	SKIPN	P1
	MOVEI	P1,ROOT-TCBSON
	HRRZ	T3,TCBSON(P1)
	CAMN	T3,T1
	HRRM	T2,TCBSON(P1)
	CAME	T3,T1
	HRLM	T2,TCBSON(P1)
	MOVE	T1,T2			;NEW ROOT
	LDB	T4,TCXBAL		;NEW BALANCE OF NEW ROOT
	POPJ	P,
;ROUTINE TO UNLINK A NODE FROM THE TREE
;THIS ROUTINE ONLY WORKS IF THE NODE DOESN'T HAVE BOTH SONS
;T2 PASSES NODE
ORPHAN:	HLRZ	T1,TCBRNT(T2)		;GET PARENT
	HRRZ	T3,TCBSON(T2)		;GET SON (IF ANY)
	SKIPN	T3
	HLRZ	T3,TCBSON(T2)
	SKIPE	T3			;HAVE SON?
	HRLM	T1,TCBRNT(T3)		;YES, SON GETS NEW PARENT
	SKIPN	T1			;DELETING THE ROOT?
	MOVEI	T1,ROOT-TCBSON		;YES, FAKE PARENT
	HRRZ	T4,TCBSON(T1)		;RIGHT SON OR LEFT SON?
	CAME	T4,T2
	JRST	ORPHN1			;LEFT
	HRRM	T3,TCBSON(T1)		;RIGHT, PARENT GETS NEW SON
	MOVEI	T3,1			;DELTA BALANCE FACTOR
	HLRZ	T1,TCBRNT(T2)		;GET PARENT BACK
	JUMPN	T1,SUBNOD
	POPJ	P,

;HERE IF THE NODE WE'RE DELETING IS A LEFT SON
ORPHN1:
IFN FTDBUG,<
	HLRZ	T4,TCBSON(T1)		;DOUBLE CHECK
	CAME	T4,T2
	PUSHJ	P,DIE
>
	HRLM	T3,TCBSON(T1)		;PARENT GETS NEW SON
	SETO	T3,			;DELTA BALANCE FACTOR
	HLRZ	T1,TCBRNT(T2)		;GET PARENT BACK
	JUMPN	T1,SUBNOD
	POPJ	P,
;T2 NODE TAKES T1 NODE'S PLACE
;PRESERVES T1 AND T2
SWAP:	HRRZ	T3,TCBSON(T1)		;COPY RIGHT SON (IF ANY)
	HRRM	T3,TCBSON(T2)
	SKIPE	T3			;RIGHT SON EXIST?
	HRLM	T2,TCBRNT(T3)		;YES, RIGHT SON GETS NEW PARENT
	HLRZ	T3,TCBSON(T1)		;COPY LEFT SON (IF ANY)
	HRLM	T3,TCBSON(T2)
	SKIPE	T3			;LEFT SON EXIST?
	HRLM	T2,TCBRNT(T3)		;YES, LEFT SON GETS NEW PARENT
	LDB	T3,TCXBAL		;COPY BALANCE FACTOR
	DPB	T3,TCZBAL
	HLRZ	T3,TCBRNT(T1)		;COPY PARENT
	HRLM	T3,TCBRNT(T2)
	SKIPN	T3			;ROOT?
	MOVEI	T3,ROOT-TCBSON		;YES, FAKE PARENT
	HRRZ	T4,TCBSON(T3)		;ARE WE RIGHT SON?
	CAMN	T4,T1
	HRRM	T2,TCBSON(T3)		;YES, UPDATE PARENT
	CAME	T4,T1			;ARE WE LEFT SON?
	HRLM	T2,TCBSON(T3)		;YES, UPDATE PARENT
	POPJ	P,
IFN FTCHK,<IFN FTGCHK,<PRINTX FTCHK AND FTGCHK CANNOT BOTH BE ON>>
IFN FTCHK!FTGCHK,<
;ROUTINE TO CHECK THE ENTIRE TREE FOR CONSISTENCY
CHECK:	SETZ	T1,			;START AT ROOT
	SKIPE	T2,ROOT
	PUSHJ	P,CTREE
	HRRZ	T1,QTCB			;DOES 1ST NODE POINT TO 2ND?
	HLRZ	T1,TCBLNK(T1)
	CAIE	T1,QTCB
	PUSHJ	P,DIE			;NO
	POPJ	P,

;ROUTINE TO CHECK A NODE FOR CONSISTENCY
;T2 PASSES NODE TO BE CHECKED
;T1 PASSES WHAT PARENT SHOULD BE
;T2 RETURNS HEIGHT OF TREE
CTREE:	HLRZ	T3,TCBRNT(T2)		;CHECK PARENT
	CAME	T3,T1
	PUSHJ	P,DIE
	MOVE	T1,T2			;SAVE ADDR OF ROOT
	HRRZ	T2,TCBLNK(T1)		;NEXT NODE POINT BACK TO US?
	HLRZ	T3,TCBLNK(T2)
	CAME	T3,T1
	PUSHJ	P,DIE
IFN FTCHK,<
	HRLOI	T3,377777		;ASSUME LAST NODE
	CAIE	T2,QTCB			;TRUE?
	MOVE	T3,TCBBLK(T2)		;NO, GET BN OF NEXT NODE
	CAMG	T3,TCBBLK(T1)		;NEXT BN SHOULD BE GREATER
>
IFN FTGCHK,<
	CAIN	T2,QTCB			;LAST NODE?
	MOVSI	T2,1			;YES, FAKE ADDR
	LDB	T3,TCVGAR		;LAST ADDR IN THIS NODE
	ADD	T3,T1
	CAMGE	T2,T3			;NEXT NODE SHOULD BE GREATER ADDR
>
	PUSHJ	P,DIE
	HLRZ	T2,TCBSON(T1)		;LEFT SUBTREE
	JUMPE	T2,CTREE1		;NONE
IFN FTCHK,<
	MOVE	T3,TCBBLK(T2)		;LEFT BN SHOULD BE LESS
	CAML	T3,TCBBLK(T1)
>
IFN FTGCHK,<
	CAML	T2,T1			;LEFT SON SHOULD BE LOWER ADDR
>
	PUSHJ	P,DIE
	PUSH	P,T1			;SAVE ROOT
	PUSHJ	P,CTREE			;CHECK LEFT SUBTREE
	POP	P,T1			;RESTORE ROOT
CTREE1:	PUSH	P,T2			;SAVE HEIGHT OF LEFT SUBTREE
	HRRZ	T2,TCBSON(T1)		;RIGHT SUBTREE
	JUMPE	T2,CTREE2		;NONE
IFN FTCHK,<
	MOVE	T3,TCBBLK(T2)		;RIGHT BN SHOULD BE GREATER
	CAMG	T3,TCBBLK(T1)
>
IFN FTGCHK,<
	CAMG	T2,T1			;RIGHT SON SHOULD BE HIGHER ADDR
>
	PUSHJ	P,DIE
	PUSH	P,T1			;SAVE ROOT
	PUSHJ	P,CTREE			;CHECK RIGHT SUBTREE
	POP	P,T1			;RESTORE ROOT
CTREE2:	POP	P,T3			;RESTORE HEIGHT OF LEFT SUBTREE
	MOVEI	T4,BALZ(T3)		;COMPUTE BALANCE FACTOR
	SUB	T4,T2
	CAIL	T4,BALM			;IN RANGE?
	CAILE	T4,BALP
	PUSHJ	P,DIE
	LDB	T1,TCXBAL		;MATCH EXPECTED?
	CAME	T1,T4
	PUSHJ	P,DIE
	CAMGE	T2,T3			;HEIGHT OF THIS TREE IS GREATER OF
	MOVE	T2,T3			; HEIGHTS OF SUBTREES
	AOJA	T2,CPOPJ		; PLUS ONE
>
	SUBTTL	BLOCK NUMBER

;ROUTINE TO TEST IF BN IS LEGAL
;T1 PASSES BN RELATIVE TO STR (DESTROYED)
;SKIP IF LEGAL
LEGALP:	IDIV	T1,BPLU
	CAMLE	T1,HUN
	POPJ	P,
	HRRZ	T1,TABUDB(T1)
	CAMG	T2,UDBHLB(T1)
	AOS	(P)
	POPJ	P,

;ROUTINE TO CONVERT BN TO INTERNAL FORMAT
CTINT:	IDIV	T1,BIGBPU
	IMUL	T1,BPLU
	ADD	T1,T2
	POPJ	P,

;ROUTINE TO CONVERT BN TO MONITOR FORMAT
CTMON:	IDIV	T1,BPLU
	IMUL	T1,BIGBPU
	ADD	T1,T2
	POPJ	P,
	SUBTTL	PRINTOUT ROUTINES

;ROUTINE TO PRINT A FILESPEC
FILO:	HLRZ	T1,FILEXT(FL)		;A UFD?
	CAIN	T1,'UFD'
	JRST	FILO1			;YES
	MOVE	T2,FILNAM(FL)		;FILENAME
	PUSHJ	P,SIXO
	JRST	FILO2
FILO1:	MOVEI	T1,"["
	PUSHJ	P,CO
	PUSHJ	P,PPNO			;PRINT PPN
	MOVEI	T1,"]"
	PUSHJ	P,CO
FILO2:	MOVEI	T1,"."			;DOT
	PUSHJ	P,CO
	HLLZ	T2,FILEXT(FL)		;EXTENSION
	PUSHJ	P,SIXO
	HRRZ	T1,FILDAD(FL)		;QUIT IF MFD
	JUMPE	T1,CPOPJ
	HRRZ	T1,FILDAD(T1)		;OR UFD
	JUMPE	T1,CPOPJ
	MOVEI	T1,"["			;BEGIN PATH
	PUSHJ	P,CO
	PUSH	P,FL
	HRRZ	FL,FILDAD(FL)
	PUSHJ	P,PTHO
	MOVEI	T1,"]"			;END PATH
	PUSHJ	P,CO
FLPOPJ:	POP	P,FL
	POPJ	P,

;ROUTINE TO PRINT THE PATH
PTHO:	PUSH	P,FL			;SAVE CURRENT FILE
	HRRZ	FL,FILDAD(FL)		;ADDR OF PARENT
	JUMPE	FL,FLPOPJ		;QUIT IF NONE
	PUSHJ	P,PTHO			;PRINT PARENT FIRST
	POP	P,FL			;RESTORE SELF
	HLRZ	T1,FILEXT(FL)		;SFD OR UFD?
	CAIN	T1,'UFD'
	PJRST	PPNO			;UFD
	MOVEI	T1,","			;SFD
	PUSHJ	P,CO
	MOVE	T2,FILNAM(FL)		;SFD NAME
	PJRST	SIXO

;ROUTINE TO OUTPUT A PPN
PPNO:	HLRZ	T1,FILNAM(FL)		;UFD, PRINT PROJECT
	PUSHJ	P,OCTO
	MOVEI	T1,","
	PUSHJ	P,CO
	HRRZ	T1,FILNAM(FL)		;PRINT PROGRAMMER
	PJRST	OCTO
;ROUTINE TO OUTPUT A BLOCK NUMBER (RELATIVE TO STR)
BNO:	PUSHJ	P,CTMON			;CONVERT TO MONITOR FORMAT
	FALL	OCTO

;ROUTINE TO OUTPUT AN OCTAL/DECIMAL NUMBER
;T1 PASSES THE NUMBER
OCTO:	SKIPA	T3,[10]
DECO:	MOVEI	T3,^D10
DECO1:	IDIV	T1,T3
	HRLM	T2,(P)
	SKIPE	T1
	PUSHJ	P,DECO1
	HLRZ	T1,(P)
	ADDI	T1,"0"
	JRST	CO

;ROUTINE TO OUTPUT A CHAR
;T1 PASSES THE CHAR
CO:	CAIN	T1,12			;LINE FEED?
	SETZM	HPOS			;YES, RESET POSITION
	AOS	HPOS			;NO, COUNT HORIZONTAL POSITION
	TRNE	F,F.TTC			;USE TTCALL?
	OUTCHR	T1			;YES
	TRNE	F,F.SIL			;TTCALL ONLY?
	POPJ	P,			;YES
CO3:	SOSGE	OBUF+.BFCTR
	JRST	CO2
	IDPB	T1,OBUF+.BFPTR
	POPJ	P,
CO2:	PUSHJ	P,BUFO
	JRST	CO3

;ROUTINE TO OUTPUT A BUFFER
BUFO:	OUT	TO,
	 POPJ	P,
	PUSHJ	P,DIE

;ROUTINE TO OUTPUT A SIXBIT NAME
;T2 PASSES THE NAME
SIXO:	LSHC	T1,6
	ANDI	T1,77
	ADDI	T1,"A"-'A'
	PUSHJ	P,CO
	JUMPN	T2,SIXO
	POPJ	P,
;ROUTINE TO OUTPUT A CRLF
CRLFO:	MOVEI	T2,[BYTE (7)15,12]
	FALL	STRO

;ROUTINE TO OUTPUT AN ASCIZ STRING
;T2 PASSES ADDR OF STRING
STRO:	HRLI	T2,(POINT 7)
STRO1:	ILDB	T1,T2
	JUMPE	T1,CPOPJ
	PUSHJ	P,CO
	JRST	STRO1

;ROUTINE TO PRINT UNIT NAME
UNITO:	MOVE	T2,UDBNAM(U)
	PUSHJ	P,SIXO
	MOVEI	T1,"("
	PUSHJ	P,CO
	MOVE	T2,UDBLOG(U)
	PUSHJ	P,SIXO
	MOVEI	T1,")"
	PJRST	CO
;ROUTINE TO TELL THE OPR WHICH STR WE ARE DOING
BEGIN:	PUSHJ	P,ONTTC			;USE TTCALLS
	MOVEI	T2,[ASCIZ /Beginning /]
	PUSHJ	P,STRO
	MOVE	T2,ALIAS
	PUSHJ	P,SIXO
	PJRST	DSPACE

;COROUTINE TO TURN ON TTCALLS
ONTTC:	TRNE	F,F.TTY			;LOG DIRECTLY TO TTY?
	POPJ	P,			;YES
	TRO	F,F.TTC			;NO, DO THIS PART DIRECTLY
	POP	P,T1			;CALL THE CALLER
	PUSHJ	P,@T1
	TRZ	F,F.TTC			;STOP TTCALLS
	POPJ	P,

;COROUTINE TO ENTER TTCALL ONLY MODE
ONSIL:	SKIPN	LOGSTR			;LOG FILE TO DISK?
	POPJ	P,			;NO
	TRO	F,F.TTC+F.SIL		;TTCALL ONLY
	POP	P,T1			;CALL THE CALLER
	PUSHJ	P,@T1
	TRZ	F,F.TTC+F.SIL		;NORMAL OUTPUT
	POPJ	P,
	SUBTTL	ADD/REMOVE STR

;ROUTINE TO REMOVE A STR
RSTR:	PUSHJ	P,RASL			;REMOVE FROM ASL
	 JRST	RSTR3			;PUT OTHER UNITS BACK
	PUSHJ	P,RSDL			;REMOVE FROM SDL
	 JRST	RSTR3
	PUSHJ	P,RSSL			;REMOVE FROM SSL
	 JRST	RSTR2
	MOVEI	T1,.FSREM		;FUNCTION CODE
	MOVEM	T1,FOO+.FSFCN
	MOVE	T1,ALIAS		;STR NAME
	MOVEM	T1,FOO+.FSMNM
	MOVE	T1,[XWD 2,FOO]		;REMOVE IT
	STRUUO	T1,
	 JRST	RSTR0
	JRST	CPOPJ1

RSTR0:	PUSHJ	P,ASSL			;PUT BACK IN SSL
	 JFCL
RSTR2:	PUSHJ	P,ASDL			;PUT BACK IN SDL
	 JFCL
RSTR3:	PUSHJ	P,AASL			;PUT BACK IN ASL
	POPJ	P,

;ROUTINE TO REMOVE STR FROM ASL (IF NECESSARY)
RASL:	MOVE	T1,HUN			;HIGHEST UNIT NUMBER
RASL2:	HRRZ	U,TABUDB(T1)		;ADDR OF UDB
	SKIPGE	UDBASL(U)		;IN ASL?
	JRST	RASL1			;NO
	MOVEI	T2,UDBNAM(U)		;REMOVE FROM ASL
	HRLI	T2,.DUSWP
	DISK.	T2,
	 CAIA
	JRST	RASL1
	CAIE	T2,DUOIP%		;MIGRATE ALREADY IN PROGRESS?
	POPJ	P,
	PUSHJ	P,SLPY			;SLEEP AWHILE
	JRST	RASL2			;TRY AGAIN
RASL1:	SOJGE	T1,RASL2		;LOOP FOR EACH UNIT
	JRST	CPOPJ1

;ROUTINE TO REMOVE STR FROM SDL (IF NECESSARY)
RSDL:	SKIPGE	SDL			;IN SDL?
	JRST	CPOPJ1			;NO
	MOVE	T1,[XWD .DURSD,ALIAS]	;YES, REMOVE IT
	DISK.	T1,
	 POPJ	P,
	JRST	CPOPJ1

;ROUTINE TO REMOVE STR FROM SSL (IF NECESSARY)
RSSL:	SETZM	FOO+.DFGJN		;SSL=JOB 0
	SETOM	FOO+.DFGNM		;1ST STR PLEASE
	SETZB	T2,PSSL			;POSITION IN SSL
	MOVEI	T3,SSL+.FSDSO		;ADDR TO STORE 1ST STR
RSSL1:	ADDI	T2,1			;BUMP POSITION
	MOVE	T1,[XWD 5,FOO]		;GET NEXT STR
	GOBSTR	T1,
	 POPJ	P,
	MOVE	T1,FOO+.DFGNM
	CAME	T1,ALIAS		;OUR STR?
	JRST	RSSL2			;NO
	MOVEM	T2,PSSL			;YES, SAVE POSITION
	MOVE	T1,FOO+.DFGST		;SAVE STATUS
	MOVEM	T1,SSLSTS
	JRST	RSSL1			;LOOP
RSSL2:	JUMPE	T1,RSSL3		;GO IF END OF LIST
	MOVEM	T1,.DFJNM(T3)		;COPY STR NAME
	MOVE	T1,FOO+.DFGST		;AND STATUS
	MOVEM	T1,.DFJST(T3)
	SETZM	.DFJDR(T3)
	ADDI	T3,3			;BUMP ADDR
	JRST	RSSL1			;LOOP
RSSL3:	SKIPN	PSSL			;IN SSL?
	JRST	CPOPJ1			;NO
	MOVEI	T1,.FSDSL		;FUNCTION
	MOVEM	T1,SSL+.FSFCN
	SETZM	SSL+.FSDJN		;JOB 0
	MOVEI	T1,DF.SRM		;REMOVE
	MOVEM	T1,SSL+.FSDFL
	MOVSI	T1,-SSL(T3)		;DO IT
	HRRI	T1,SSL
	STRUUO	T1,
	 POPJ	P,
	JRST	CPOPJ1
;ROUTINE TO ADD A STR
ASTR:	MOVEI	T1,.FSDEF		;FUNCTION
	MOVEM	T1,DEFIN+.FSFCN
	MOVE	T1,[XWD SZSDB,SDB]	;ADDR OF STR DATA BLOCK
	MOVEM	T1,DEFIN+.FSNST
	MOVE	T1,[XWD DEFINL,DEFIN]
	STRUUO	T1,
	 POPJ	P,
	PUSHJ	P,ASSL			;PUT BACK IN SSL
	 JFCL
	PUSHJ	P,ASDL			;PUT BACK IN SDL
	 JFCL
	PUSHJ	P,AASL			;PUT BACK IN ASL
	JRST	CPOPJ1

;ROUTINE TO ADD STR TO ASL
AASL:	MOVE	T1,HUN			;HIGHEST UNIT NUMBER
AASL1:	HRRZ	U,TABUDB(T1)		;ADDR OF UDB
	SKIPGE	UDBASL(U)		;BELONGS IN ASL?
	JRST	AASL2			;NO
	MOVEI	T2,UDBNAM(U)		;PUT UNIT IN ASL
	HRLI	T2,.DUASW
	DISK.	T2,
	 JFCL
AASL2:	SOJGE	T1,AASL1		;LOOP FOR EACH UNIT
	POPJ	P,

;ROUTINE TO ADD STR TO SDL
ASDL:	SKIPGE	SDL			;BELONGS IN SDL?
	JRST	CPOPJ1			;NO
	MOVE	T1,[XWD .DUASD,ALIAS]	;YES, PUT IT BACK
	DISK.	T1,
	 POPJ	P,
	JRST	CPOPJ1

;ROUTINE TO ADD STR TO SSL
ASSL:	SKIPN	PSSL			;BELONGS IN SSL?
	JRST	CPOPJ1			;NO
	SETZM	FOO+.DFGJN		;SSL=JOB 0
	SETOM	FOO+.DFGNM		;1ST STR PLEASE
	SETZ	T2,			;POSITION IN SSL
	MOVEI	T3,SSL+.FSDSO		;ADDR TO STORE 1ST STR
ASSL1:	ADDI	T2,1			;BUMP POSITION
	CAME	T2,PSSL			;NOW?
	JRST	ASSL4			;NO, NOT YET
	MOVE	T1,ALIAS		;YES, COPY STR NAME
	MOVEM	T1,.DFJNM(T3)
	MOVE	T1,SSLSTS		;COPY STATUS
	MOVEM	T1,.DFJST(T3)
	SETZM	.DFJDR(T3)
	ADDI	T3,3			;BUMP ADDR
ASSL4:	MOVE	T1,[XWD 5,FOO]		;GET NEXT STR
	GOBSTR	T1,
	 POPJ	P,
	SKIPN	T1,FOO+.DFGNM
	JRST	ASSL3			;GO IF END OF LIST
	MOVEM	T1,.DFJNM(T3)		;COPY STR NAME
	MOVE	T1,FOO+.DFGST		;AND STATUS
	MOVEM	T1,.DFJST(T3)
	SETZM	T1,.DFJDR(T3)
	ADDI	T3,3			;BUMP ADDR
	JRST	ASSL1			;LOOP
ASSL3:	MOVEI	T1,.FSDSL		;FUNCTION
	MOVEM	T1,SSL+.FSFCN
	SETZM	SSL+.FSDJN		;JOB 0
	MOVEI	T1,DF.SRM		;REMOVE
	MOVEM	T1,SSL+.FSDFL
	MOVSI	T1,-SSL(T3)		;DO IT
	HRRI	T1,SSL
	STRUUO	T1,
	 POPJ	P,
	JRST	CPOPJ1
	SUBTTL	LOG FILE

;ROUTINE TO OPEN THE DEVICE FOR THE LOG FILE
OPENLG:	MOVSI	T1,'LPT'		;DEVICE NAME
	MOVE	T2,T1
	DEVTYP	T1,			;IS IT SPOOLED?
	 SETZ	T1,
	TLNE	T1,(TY.SPL)
	JRST	OPNLG5			;YES
	MOVE	T1,T2			;IS IT ASSIGNED?
	DEVCHR	T1,
	TRNE	T1,DV.ASC
	TLNN	T1,(DV.AVL)
OPNLG5:	MOVSI	T2,'TTY'		;NO, USE TTY
	MOVEM	T2,LDEV+.OPDEV		;OPEN IT
	MOVEI	T2,.IOASC
	MOVEM	T2,LDEV+.OPMOD
	MOVSI	T2,OBUF
	MOVEM	T2,LDEV+.OPBUF
	OPEN	TO,LDEV
	 PUSHJ	P,DIE
	OUTBUF	TO,			;BUILD BUFFERS NOW
	MOVE	T2,OBUF+.BFADR		;SAVE ADDR OF 1ST BUF
	MOVEM	T2,SAVBUF
	MOVEI	T2,LPTWID-RMAR		;SET MARGIN
	MOVEM	T2,MARGIN
	MOVEI	T1,TO			;GET DEVICE TYPE (TTY MIGHT BE LOGICAL)
	DEVCHR	T1,
	TLNE	T1,(DV.DSK)		;DISK?
	JRST	OPNLG1			;YES
	SETZM	LOGSTR			;NO
	TLNN	T1,(DV.TTY)		;TTY?
	POPJ	P,			;NO
	TRO	F,F.TTY			;YES
	MOVEI	T1,TO			;GET UDX
	IONDX.	T1,
	 PUSHJ	P,DIE
	MOVEM	T1,FOO+.TOUDX
	MOVEI	T1,.TOWID		;FUNCTION
	MOVEM	T1,FOO+.TOFNC
	MOVE	T1,[XWD 2,FOO]		;READ WIDTH OF TTY
	TRMOP.	T1,
	 MOVEI	T1,LPTWID
	SUBI	T1,RMAR			;BACK OFF
	MOVEM	T1,MARGIN		;SET MARGIN
	POPJ	P,

;HERE IF DISK
OPNLG1:	PUSHJ	P,GETLOK		;GET INTERLOCK
	MOVE	T1,LDEV+.OPDEV		;WHAT STR IS LOG?
	MOVEM	T1,FOO+.DCNAM
	MOVE	T1,[XWD FOOSIZ,FOO]
	DSKCHR	T1,
	 JRST	OPNLG2
	MOVE	T1,FOO+.DCSNM
	JRST	OPNLG4

;HERE IF DSKCHR FAILED (STR WAS PROBABLY YANKED)
OPNLG2:	MOVE	T1,LDEV+.OPDEV		;UN-DO LOGICAL NAME
	DEVNAM	T1,
	 SETO	T1,
	PUSHJ	P,INPROG		;IS IT IN PROGRESS?
	 JRST	OPNLG4			;YES
	MOVE	T1,SLST			;NO, I GUESS IT DOESN'T MATTER
	MOVE	T1,SNFNAM(T1)
OPNLG4:	MOVEM	T1,LOGSTR		;YES, THAT'S IT THEN
	PUSHJ	P,GIVLOK		;GIVE UP INTERLOCK
	MOVE	T1,LDEV+.OPDEV		;GET PATH
	MOVEM	T1,PTH+.PTFCN
	MOVE	T1,[XWD PTHL,PTH]
	PATH.	T1,
	 CAIA
	POPJ	P,
	HRRE	T1,.PTFRD		;GET DEFAULT PATH
	MOVEM	T1,PTH+.PTFCN
	MOVE	T1,[XWD PTHL,PTH]
	PATH.	T1,
	 PUSHJ	P,DIE
	POPJ	P,
;ROUTINE TO COPY TEMP LOGS BACK TO THE STR WHERE THEY BELONG
;CALL WITH INTERLOCK
;EXIT CPOPJ IF STILL MORE TO GO
;EXIT CPOPJ1 IF ALL DONE
CPYLG:	PUSHJ	P,SAVE1
CPYLG0:	SKIPN	P1,LOGS			;ANY TO COPY?
	JRST	CPOPJ1			;NO
	MOVE	T1,LOGSTR		;NAME OF LOG STR
	MOVEM	T1,LDEV+.OPDEV
	TRNE	F,F.ALL			;DOING ALL?
	JRST	CPYLG6			;YES, MUST WAIT FOR STR TO FINISH
	MOVE	T2,CNTSLT		;WE AREN'T, BUT ARE WE THE ONLY ONE?
	CAIN	T2,1
	JRST	CPYLG5			;ONLY ONE, IT'LL NEVER FINISH
CPYLG6:	MOVEI	T2,ALST-SNFLNK		;IS LOG STR DONE?
CPYLG3:	HRRZ	T2,SNFLNK(T2)
	JUMPE	T2,CPOPJ
	CAME	T1,SNFNAM(T2)
	JRST	CPYLG3
CPYLG5:	MOVE	T1,LOGDEV(P1)		;IS TMP STR AVAILABLE?
	MOVEM	T1,TDEV+.OPDEV
	PUSHJ	P,INPROG
	 POPJ	P,			;NO
	MOVE	T2,MYSLT		;YES, LOCK IT DOWN
	MOVEM	T1,JBTLOG(T2)
	PUSHJ	P,GIVLOK		;GIVE AWAY INTERLOCK
	MOVEI	T1,.IOBIN
	MOVEM	T1,LDEV+.OPMOD
	MOVEM	T1,TDEV+.OPMOD
	MOVSI	T1,OBUF
	MOVEM	T1,LDEV+.OPBUF
	MOVEI	T1,.RBEXT
	MOVEM	T1,LFIL+.RBCNT
	MOVEM	T1,TFIL+.RBCNT
	MOVEI	T1,PTH
	MOVEM	T1,LFIL+.RBPPN
	MOVE	T1,LOGNAM(P1)
	MOVEM	T1,LFIL+.RBNAM
	MOVEM	T1,TFIL+.RBNAM
	MOVSI	T1,'LST'
	MOVEM	T1,LFIL+.RBEXT
	MOVEI	T1,IBUF
	MOVEM	T1,TDEV+.OPBUF
	MOVE	T1,FFAPPN
	MOVEM	T1,TFIL+.RBPPN
	MOVSI	T1,'TMP'
	MOVEM	T1,TFIL+.RBEXT
	OPEN	TL,TDEV			;OPEN TEMP ON ANOTHER CH
	 JRST	CPYLG4
	LOOKUP	TL,TFIL
	 JRST	CPYLG4
	OPEN	TO,LDEV			;RE-OPEN LOG FILE
	 JRST	CPYLG4
	ENTER	TO,LFIL
	 JRST	CPYLG4
	MOVE	T1,SAVBUF		;USE SAME BUFFERS
	MOVEM	T1,OBUF+.BFADR
	MOVEI	T2,TDEV			;COMPUTE SIZE OF BUFFERS
	DEVSIZ	T2,
	 PUSHJ	P,DIE
	HLRZ	T1,T2
	IMULI	T1,(T2)
	PUSHJ	P,GETBLK		;ALLOCATE SPACE
	PUSH	P,T1			;SAVE SIZE
	PUSH	P,T2			;SAVE ADDR
	EXCH	T2,.JBFF		;ALLOCATE BUFFERS
	INBUF	TL,
	MOVEM	T2,.JBFF
CPYLG1:	PUSHJ	P,CI			;INPUT A CHAR FROM TEMP FILE
	 JRST	CPYLG2			;EOF
	PUSHJ	P,CO			;COPY TO LOG FILE
	JRST	CPYLG1
CPYLG2:	CLOSE	TO,			;CLOSE THE LOG FILE
	STATZ	TO,IO.ERR
	 PUSHJ	P,DIE
	SETZM	FOO			;DELETE TEMP LOG
	RENAME	TL,FOO
	 JFCL
	RELEAS	TL,
	POP	P,T2			;RESTORE ADDR
	POP	P,T1			;RESTORE SIZE
	PUSHJ	P,GIVBLK		;GIVE BACK SPACE
CPYLG4:	PUSHJ	P,GETLOK		;GET INTERLOCK
	MOVE	T1,MYSLT		;FREE TMP STR
	SETZM	JBTLOG(T1)
	MOVE	T1,LOGNXT(P1)		;UNLINK LOG
	MOVEM	T1,LOGS
	MOVEI	T1,SIZLOG		;RETURN LOG
	MOVE	T2,P1
	PUSHJ	P,GIVBLK
	JRST	CPYLG0			;DO ANOTHER LOG
;ROUTINE TO INPUT A CHAR FROM THE TEMP LOG
;T1 RETURNS THE CHAR
CI:	SOSGE	IBUF+.BFCTR
	JRST	CI2
	ILDB	T1,IBUF+.BFPTR
	JRST	CPOPJ1
CI2:	IN	TL,
	 JRST	CI
	STATZ	TL,IO.ERR
	 PUSHJ	P,DIE
	POPJ	P,

;ROUTINE TO TEST IF STR IS IN PROGRESS
;T1 PASSES STR NAME
;NOSKIP IF IN PROGRESS
;SKIP IF NOT IN PROGRESS
INPROG:	PUSHJ	P,SAVE1
	MOVEI	P1,NSLT-1
INPRG1:	CAMN	T1,JBTSTR(P1)
	POPJ	P,
	SOJGE	P1,INPRG1
	JRST	CPOPJ1
;ROUTINE TO PICK A STR FOR THE LOG FILE (AND OPEN IT)
;YOU MUST HAVE THE INTERLOCK TO CALL THIS ROUTINE
PIKLOG:	PUSHJ	P,SAVE1
PKLOGA:	SKIPN	T1,LOGSTR		;LOG ON DISK?
	POPJ	P,			;NO
	MOVEI	T2,.IOASC
	MOVEM	T2,LDEV+.OPMOD
	MOVSI	T2,OBUF
	MOVEM	T2,LDEV+.OPBUF
	MOVEI	T2,.RBEXT
	MOVEM	T2,LFIL+.RBCNT
	MOVE	T2,ALIAS
	MOVEM	T2,LFIL+.RBNAM
	CAMN	T1,ALIAS		;DOING LOG STR?
	JRST	PKLOG2			;YES, MUST USE TMP STR
	MOVE	T3,ALIAS		;PROCESS STR AND LOG STR THE ONLY ONES?
	MOVEI	T2,SLST-SNFNXT
PKLOGC:	HRRZ	T2,SNFNXT(T2)
	JUMPE	T2,PKLOGD		;YES, MUST GO ON ACTUAL STR
	CAME	T3,SNFNAM(T2)
	CAMN	T1,SNFNAM(T2)
	JRST	PKLOGC
	TRNN	F,F.ALL			;NO, DOING ALL?
	JRST	PKLOGD			;NOT ALL, DON'T CARE IF DONE
	MOVEI	T2,ALST-SNFLNK		;LOG STR PROCESSED YET?
PKLOG1:	HRRZ	T2,SNFLNK(T2)
	JUMPE	T2,PKLOG2		;NO
	CAME	T1,SNFNAM(T2)
	JRST	PKLOG1
	JRST	PKLOG3
;HERE TO PUT ON ACTUAL STR IF NOT IN PROGRESS
PKLOGD:	PUSHJ	P,INPROG		;LOG STR IN PROGRESS?
	 JRST	PKLOG2			;YES, PUT ON TMP STR (IF ONE EXISTS)
;HERE TO PUT LOG ON ACTUAL STR
PKLOG3:	MOVEM	T1,LDEV+.OPDEV
	OPEN	TO,LDEV
	 JRST	PKLOG2
	MOVE	T1,SAVBUF		;USE SAME BUFFERS
	MOVEM	T1,OBUF+.BFADR
	MOVEI	T1,PTH
	MOVEM	T1,LFIL+.RBPPN
	MOVSI	T1,'LST'
	MOVEM	T1,LFIL+.RBEXT
	ENTER	TO,LFIL
	 JRST	PKLOG2
	JRST	PKLOGB
;HERE TO PUT LOG FILE ON TMP STR
;TRY ALL STRS COMPLETED
PKLOG2:	MOVEI	P1,ALST-SNFLNK
PKLOG5:	HRRZ	P1,SNFLNK(P1)
	JUMPE	P1,PKLOG4
	MOVE	T1,SNFNAM(P1)
	PUSHJ	P,TRYLG
	 JRST	PKLOG9			;WIN
	JRST	PKLOG5
;TRY WHAT WORKED FOR SOMEBODY ELSE
PKLOG4:	MOVEI	P1,NSLT-1
PKLOG6:	SKIPN	T1,JBTLOG(P1)
	JRST	PKLOG7
	PUSHJ	P,TRYLG
	 JRST	PKLOG9
PKLOG7:	SOJGE	P1,PKLOG6
;TRY ANYTHING AT ALL
	MOVEI	P1,BLST-SNFLNK
PKLOG8:	HRRZ	P1,SNFLNK(P1)
	JUMPE	P1,PKLOG0
	MOVE	T1,SNFNAM(P1)
	PUSHJ	P,TRYLG
	 JRST	PKLOG9
	JRST	PKLOG8
;HERE IF NO PLACE AT ALL TO PUT LOG
;WAIT FOR SOMETHING TO FINISH
PKLOG0:	PUSHJ	P,SLEEPY		;SLEEP AWHILE
	JRST	PKLOGA			;TRY AGAIN
;HERE WHEN WE SUCCESSFULLY CREATED A TMP LOG
PKLOG9:	MOVEI	T1,SIZLOG		;ALLOCATE A LOG BLOCK
	PUSHJ	P,GETBLK
	MOVE	T1,ALIAS		;FILE NAME
	MOVEM	T1,LOGNAM(T2)
	MOVE	T1,LDEV+.OPDEV		;DEVICE
	MOVEM	T1,LOGDEV(T2)
	MOVE	T1,LOGS			;LINK IT
	MOVEM	T1,LOGNXT(T2)
	MOVEM	T2,LOGS
PKLOGB:	MOVE	T1,MYSLT		;TELL THE WORLD
	MOVE	T2,LDEV+.OPDEV
	MOVEM	T2,JBTLOG(T1)
	POPJ	P,

;ROUTINE TO ATTEMPT THE CREATION OF A TMP LOG
;T1 PASSES STR TO TRY FOR
;CPOPJ IF SUCCEED
;CPOPJ1 IF FAIL
TRYLG:	CAMN	T1,LOGSTR		;PUTING TMP FILE ON ACTUAL STR?
	JRST	CPOPJ1			;YES, THAT'S SILLY
	MOVEM	T1,LDEV+.OPDEV		;SAVE DEVICE NAME
	OPEN	TO,LDEV
	 JRST	CPOPJ1
	MOVE	T3,SAVBUF		;USE SAME BUFFERS
	MOVEM	T3,OBUF+.BFADR
	MOVE	T3,FFAPPN		;ENTER TEMP FILE
	MOVEM	T3,LFIL+.RBPPN
	MOVSI	T3,'TMP'
	MOVEM	T3,LFIL+.RBEXT
	ENTER	TO,LFIL
	 JRST	CPOPJ1
	POPJ	P,

;ROUTINE TO SLEEP
SLEEPY:	PUSHJ	P,GIVLOK		;GIVE UP INTERLOCK
	PUSHJ	P,SLPY			;SLEEP AWHILE
	PJRST	GETLOK			;GET INTERLOCK BACK

;SLEEP
SLPY:	MOVEI	T1,NSEC			;SLEEP AWHILE
	SLEEP	T1,
	POPJ	P,
	SUBTTL	BAT BLOCK

;ROUTINE TO PROCESS BAT BLOCK FOR A GIVEN UNIT
DOBT:	PUSHJ	P,SAVE4			;SAVE ACS
	PUSHJ	P,RDBT			;READ BAT BLOCK
	 JRST	DOBT9			;ERROR
	LDB	P1,BAYNBR		;NUMBER OF BAD REGIONS
	ADD	P1,BAFCNT(B)
	JUMPE	P1,CPOPJ
	HRRZ	P2,BAFFIR(B)		;OFFSET OF 1ST ENTRY
	ADD	P2,B			;ADDR OF 1ST ENTRY
DOBT2:	LDB	P3,BAYNBB		;NUMBER OF BAD BLOCKS
	ADDI	P3,1
	MOVE	P4,BAFELB(P2)		;WORD CONTAINING BN
	MOVE	T1,BAFNBB(P2)		;WORD CONTAINING BAPNTP
	TRNE	T1,BAPNTP		;NEW ENTRY OR OLD?
	TLZA	P4,BATMSK		;NEW
	TLZ	P4,-1			;OLD
DOBT3:	CAMLE	P4,UDBHLB(U)		;LEGAL BN?
	JRST	DOBT4			;NO
	MOVE	T1,UDBBLK(U)		;CONVERT TO BN RELATIVE TO STR
	ADD	T1,P4
	PUSH	P,T1
	PUSHJ	P,SATBIT		;FIND SAT BIT
	POP	P,T1
	TDNE	T2,(T3)			;ALREADY LIT? (I.E. IN BADBLK?)
	JRST	DOBT5			;YES, IGNORE
	IORM	T2,(T3)			;NO, LIGHT IT NOW
	IDIV	T1,BPC			;CONVERT BN TO CLUSTER
	MOVEI	T2,BADCST		;ADD CLUSTER TO BAD LIST
	PUSHJ	P,ADDLST
DOBT5:	ADDI	P4,1			;NEXT BN
	SOJG	P3,DOBT3		;LOOP FOR EACH BLOCK IN REGION
DOBT4:	ADDI	P2,2			;ADDR OF NEXT ENTRY
	SOJG	P1,DOBT2		;LOOP FOR EACH ENTRY
	POPJ	P,

;HERE IF BAT BLOCK IS BAD
DOBT9:	MOVEI	T2,[ASCIZ /Error while reading BAT blocks on /]
	PUSHJ	P,STRO
	PUSHJ	P,UNITO
	PJRST	DSPACE
;ROUTINE TO COMPUTE THE AMOUNT OF OVERLAP BETWEEN THE BAD LIST
;AND THE FREE LIST
;T1 RETURNS THE NUMBER OF CLUSTERS OF OVERLAP
BADOVR:	SETZ	T1,			;NONE SO FAR
	SKIPE	FRECST+CSTCNT		;QUIT EARLY IF NONE AT ALL
	SKIPN	BADCST+CSTCNT
	POPJ	P,
	PUSHJ	P,SAVE2
	MOVEI	P1,FRECST+CSTREG-REGNXT	;PRESET PRED
	MOVEI	P2,BADCST+CSTREG-REGNXT
	HRRZ	P1,REGNXT(P1)		;NEXT FREE REGION
	JUMPE	P1,CPOPJ
BDOVR1:	HRRZ	P2,REGNXT(P2)		;NEXT BAD REGION
	JUMPE	P2,CPOPJ
BDOVR2:	MOVE	T2,REGLOW(P1)		;DO THEY OVERLAP?
	CAMLE	T2,REGHI(P2)
	JRST	BDOVR1			;NO, GET NEXT BAD REGION
	MOVE	T3,REGHI(P1)
	CAML	T3,REGLOW(P2)
	JRST	BDOVR3			;YES
BDOVR4:	HRRZ	P1,REGNXT(P1)		;NO, GET NEXT FREE REGION
	JUMPN	P1,BDOVR2
	POPJ	P,
;HERE WHEN THERE IS DEFINITELY SOME OVERLAP
BDOVR3:	CAMGE	T2,REGLOW(P2)		;GET HIGHEST OF THE LOWS
	MOVE	T2,REGLOW(P2)
	CAMLE	T3,REGHI(P2)		;GET LOWEST OF THE HIGHS
	MOVE	T3,REGHI(P2)
	SUBM	T3,T2			;BUMP COUNT
	ADDI	T1,1(T2)
	CAMN	T3,REGHI(P1)		;WHICH REGION ENDS FIRST?
	JRST	BDOVR4			;FREE
	JRST	BDOVR1			;BAD
	SUBTTL	SAT BLOCKS

;ROUTINE TO ALLOCATE SPACE FOR SAT BLOCKS
BLDSAT:	PUSHJ	P,SAVE3			;SAVE ACS
	MOVE	T1,UDBSPU(U)		;BUILD SPT
	PUSHJ	P,GETZER
	HRL	T2,UDBSPU(U)
	MOVEM	T2,UDBSPT(U)
	SETOM	UDBSSF(U)		;NO ENTRIES IN SPT YET
	MOVE	T1,UDBSPU(U)		;BUILD PST
	PUSHJ	P,GETBLK
	MOVEM	T2,UDBPST(U)
	MOVE	P1,UDBSPU(U)		;HIGHEST SAT NUMBER
	SUBI	P1,1
	MOVE	T1,UDBBPU(U)		;NUMBER OF FULL CLUSTERS
	IDIV	T1,BPC
	MOVE	T2,P1			;TOTAL BITS IN ALL BUT LAST SAT
	IMUL	T2,UDBCPS(U)
	SUB	T1,T2			;NUMBER OF CLUSTERS IN LAST SAT
	SKIPA	P2,T1
BLDST1:	MOVE	P2,UDBCPS(U)		;CLUSTERS PER SAT
	MOVE	T1,UDBWPS(U)		;ALLOCATE SPACE FOR SAT
	PUSHJ	P,GETZER
	MOVE	P3,T2			;PUT PNTR IN SAFE PLACE
	MOVE	T1,P1			;STORE IT
	ADD	T1,UDBPST(U)
	MOVEM	P3,(T1)
	MOVE	T1,P2			;CLUSTERS THIS SAT
	IDIVI	T1,^D36			;WORDS
	MOVEI	T3,^D36			;UNUSED BITS
	SUBM	T3,T2
	SETZ	T3,			;BUILD MASK
	SETO	T4,
	LSHC	T3,(T2)
	HRLI	P3,T1			;INDEX REGISTER
	CAMGE	T1,UDBWPS(U)		;IS LAST WORD COMPLETELY FULL?
	IORM	T3,@P3			;NO, MARK UNUSED BITS
BLDST2:	ADDI	T1,1			;POINT AT NEXT WORD
	CAML	T1,UDBWPS(U)		;DONE?
	JRST	BLDST3			;YES
	SETOM	@P3			;NO, MARK ALL BITS
	JRST	BLDST2			;LOOP
BLDST3:	SOJGE	P1,BLDST1		;DO NEXT SAT
	POPJ	P,
;ROUTINE TO STORE CA IN SPT
SAVSAT:	TRON	F,F.RIB			;STEPPED OVER RIB YET?
	POPJ	P,			;NO, STEP NOW
	AOS	T2,UDBSSF(U)		;GOT THEM ALL?
	CAML	T2,UDBSPU(U)
	POPJ	P,			;YES
	ADD	T2,UDBSPT(U)		;NEXT ADDR IN SPT
	MOVE	T1,DRCA			;STORE CA
	DPB	T1,SPYCLA
	POPJ	P,
;ROUTINE TO READ THE SAT BLOCKS AND SEE IF THEY ARE RIGHT
;IN THIS ROUTINE:
;P1=ADDR WITHIN SAT BLOCK (AS READ FROM DISK)
;P2=SAT NUMBER
;P3=WORD NUMBER
;P4=ADDR WITHIN SAT BLOCK (AS COMPUTED)
;NOSKIP IF CAN'T FIND SATS
DOSAT:	PUSHJ	P,SAVE4			;SAVE ACS
	MOVE	T1,HUN			;FOUND ALL SATS?
DOSAT7:	HRRZ	U,TABUDB(T1)
	MOVE	T2,UDBSSF(U)
	ADDI	T2,1
	CAMGE	T2,UDBSPU(U)
	JRST	DOSAT8			;NO
	SOJGE	T1,DOSAT7
	SETZM	STAL			;NUMBER FREE BLOCKS (STR WIDE)
	MOVE	T1,HUN			;HIGHEST UNIT NUMBER
DOSAT1:	HRRZ	U,TABUDB(T1)		;UNIT DATA BLOCK
	SKIPN	UDBAWL(U)		;WRITE LOCKED?
	SKIPE	UDBHWP(U)
	PUSHJ	P,DOSTWL		;YES
	MOVE	T1,UDBBPU(U)		;COMPUTE SAFETY FACTOR
	MOVE	T2,UDBK4S(U)
	LSH	T2,3
	SUB	T1,T2
	IDIVI	T1,UNVRSF
	CAILE	T1,MAXSAF
	MOVEI	T1,MAXSAF
	MOVNS	T1
	MOVEM	T1,UDBTAL(U)
	ADDM	T1,STAL			;BUMP STR WIDE COUNTER
	PUSHJ	P,DOBT			;DO BAT BLOCKS
	MOVE	P2,UDBSPU(U)		;HIGHEST SAT NUMBER
	SUBI	P2,1
DOSAT2:	MOVE	T3,P2			;ADDR OF SAT
	ADD	T3,UDBPST(U)
	MOVE	P4,(T3)
	TRZ	F,F.SHC			;SAT IS OK SO FAR
	MOVE	T2,P2			;READ THE SAT BLOCK
	ADD	T2,UDBSPT(U)
	SETZ	T3,
	DPB	T3,SPYTAL
	LDB	T3,SPYCLA
	IMUL	T3,BPC
	PUSHJ	P,REDBLK
	 PUSHJ	P,DOSTRE		;ERROR
	SETZ	P3,			;START WITH WORD 0
	MOVE	P1,B
DOSAT4:	MOVE	T1,(P1)			;PICK UP A SAT WORD
	CAMN	T1,(P4)			;RIGHT?
	JRST	DOSAT5			;YES
	TRO	F,F.SHC			;MUST REWRITE SAT
	TDZ	T1,(P4)			;DO LOST CLUSTERS
	MOVEI	T2,LSTCST
	PUSHJ	P,DOCST
	MOVE	T1,(P4)			;DO FREE CLUSTERS
	TDZ	T1,(P1)
	MOVEI	T2,FRECST
	PUSHJ	P,DOCST
	MOVE	T2,(P4)			;GET FREE CLUSTERS BACK
	TDZ	T2,(P1)
	SKIPA	T1,(P4)			;FIX THE SAT WORD
DOSAT5:	TDZA	T2,T2
	MOVEM	T1,(P1)
	MOVEM	T2,(P4)			;SET SAT TO FREE CLUSTERS
	MOVE	T2,(P1)			;COUNT FREE CLUSTERS
	PUSHJ	P,CNTSAT
	MOVE	T2,P2			;BUMP COUNT IN SPT
	ADD	T2,UDBSPT(U)
	LDB	T3,SPYTAL
	ADD	T3,T4
	DPB	T3,SPYTAL
	IMUL	T4,BPC			;CONVERT CLUSTERS TO BLOCKS
	ADDM	T4,UDBTAL(U)		;BUMP UNIT COUNT
	ADDM	T4,STAL			;BUMP STR COUNT
	ADDI	P3,1			;LOOP FOR EACH WORD
	ADDI	P1,1
	CAME	P3,UDBWPS(U)
	AOJA	P4,DOSAT4
;JFCL THIS LOCATION TO PREVENT SAT BLOCKS FROM BEING WRITTEN
SATPAT:	TRNN	F,F.SHC			;REWRITE SAT?
	JRST	DOSAT6			;NO
	SKIPN	UDBAWL(U)		;WRITE PROTECT?
	SKIPE	UDBHWP(U)
	JRST	DOSAT6			;YES, DON'T TRY TO WRITE
	MOVE	T2,P2			;COMPUTE BN OF SAT
	ADD	T2,UDBSPT(U)
	LDB	T3,SPYCLA
	IMUL	T3,BPC
	PUSHJ	P,WRTBLK		;WRITE THE SAT
	 PUSHJ	P,DOSTWE		;ERROR
DOSAT6:	SOJGE	P2,DOSAT2		;LOOP FOR EACH SAT
	MOVE	T1,UDBLUN(U)		;LOOP FOR EACH UNIT
	SOJGE	T1,DOSAT1
	JRST	CPOPJ1

;HERE IF CAN'T FIND ALL THE SATS
DOSAT8:	MOVEI	T2,[ASCIZ /Can't find SAT.SYS
SATs will not be processed
STR will not be mounted/]
	PJRST	STRDSP
;HERE IF ERROR WHILE READING SAT BLOCK
;B PASSES ADDR SAT WOULD HAVE BEEN READ INTO
;P4 PASSES ADDR OF SAT (AS COMPUTED)
;P2 PASSES SAT NUMBER
DOSTRE:	HRLZ	T1,P4			;COPY SAT BLOCK
	HRR	T1,B
	MOVE	T2,B
	ADD	T2,UDBWPS(U)
	BLT	T1,-1(T2)
	TRO	F,F.SHC			;WRITE SAT
	MOVEI	T2,[ASCIZ /read/]
	JRST	DOSTER

;HERE IF ERROR WHILE WRITTING SAT BLOCK
;P2 PASSES SAT NUMBER
DOSTWE:	MOVEI	T2,[ASCIZ /writt/]
DOSTER:	PUSH	P,T2			;SAVE READ/WRITE
	MOVEI	T2,[ASCIZ /Error while /]
	PUSHJ	P,STRO
	POP	P,T2
	PUSHJ	P,STRO
	MOVEI	T2,[ASCIZ /ing SAT block /]
	PUSHJ	P,STRO
	MOVE	T1,P2
	PUSHJ	P,OCTO
	MOVEI	T2,[ASCIZ / on unit /]
	PUSHJ	P,STRO
	PUSHJ	P,UNITO
	PJRST	DSPACE

;HERE IF UNIT IS WRITE LOCKED
DOSTWL:	PUSHJ	P,UNITO
	MOVEI	T2,[ASCIZ / is write locked and SATs will not be corrected/]
	PUSHJ	P,STRO
	PJRST	DSPACE
;ROUTINE TO COUNT THE NUMBER OF ZERO BITS IN A SAT WORD
;T2 PASSES THE CORRECTED VERSION OF THE SAT WORD
;T4 RETURNS THE COUNT
CNTSAT:	SETZ	T4,			;COUNTER
CNTST2:	SETCMM	T2			;COMPLIMENT
	JFFO	T2,CNTST3		;COUNT LEADING "ONES"
	POPJ	P,			;NO MORE
CNTST3:	SETZ	T1,			;PURGE LEADING "ONES"
	ROTC	T1,(T3)
	SETCMM	T2			;RETURN TO NORMAL SPACE TIME
	JFFO	T2,CNTST4		;COUNT ZEROES
	MOVEI	T4,^D36			;ALL ZEROES
	POPJ	P,
CNTST4:	ADD	T4,T3			;ADD TO TOTAL
	SETO	T1,			;PURGE ZEROES
	ROTC	T1,(T3)
	JRST	CNTST2			;LOOP
;LIGHT A SAT BIT
MRKIT:	MOVE	T1,TCBBLK(TC)		;BN OF RIB
	FALL	MARKIT

;ROUTINE TO LIGHT A SAT BIT
;T1 PASSES BN
MARKIT:	PUSH	P,T1			;SAVE BN
	PUSHJ	P,SATBIT		;FIND THE RIGHT BIT
	POP	P,T1			;RESTORE BN
	TRNE	F,F.P2			;PASS TWO?
	JRST	MARKI2			;YES
	TDNE	T2,(T3)			;ALREADY ON?
	JRST	ADDMUL			;YES
	IORM	T2,(T3)			;NO, LIGHT IT NOW
	POPJ	P,

;HERE IF PASS TWO
MARKI2:	TDNN	T2,(T3)			;TARGET BN?
	POPJ	P,			;NO
	IDIV	T1,BPC			;CONVERT BN TO CLUSTER
	MOVEI	T2,MULCST		;IS IT FREE OR MULTIPLE?
	PUSHJ	P,FNDREG
	 SKIPA	T2,[F2CST]		;FREE
	MOVEI	T2,M2CST		;MULTIPLY USED
	PJRST	ADDLST			;ADD IT TO CLUSTER LIST
;ROUTINE TO FIND THE SAT BIT CORRESPONDING TO A GIVEN BN
;T1 PASSES BN (DESTROYED)
;T2 RETURNS BIT MASK
;T3 RETURNS WORD ADDR
SATBIT:	IDIV	T1,BPLU			;UNIT AND OFFSET
	HRRZ	T1,TABUDB(T1)		;UNIT DATA BLOCK
	IDIV	T2,BPC			;CLUSTER
	IDIV	T2,UDBCPS(T1)		;SAT AND OFFSET
	ADD	T2,UDBPST(T1)		;ADDR OF PST ENTRY
	IDIVI	T3,^D36			;WORD AND BIT
	ADD	T3,(T2)			;WORD ADDR
	MOVSI	T2,(1B0)		;BUILD MASK
	MOVNS	T4
	LSH	T2,(T4)
	POPJ	P,

;SUBROUTINE TO FIND A REGION
;T1 PASSES CLUSTER NUMBER
;T2 PASSES CST
;T3 RETURNS REG
;SKIP IF FOUND
FNDREG:	MOVEI	T3,CSTREG-REGNXT(T2)	;PRESET PRED
FNDRG1:	HRRZ	T3,REGNXT(T3)		;STEP TO NEXT REGION
	JUMPE	T3,CPOPJ		;QUIT IF NO MORE REGIONS
	CAMGE	T1,REGLOW(T3)		;WITHIN REGION?
	POPJ	P,
	CAMLE	T1,REGHI(T3)
	JRST	FNDRG1			;NO, KEEP SEARCHING
	JRST	CPOPJ1			;YES, WIN

;SUBROUTINE TO MARK A CLUSTER LIST
;T2 PASSES CST
MARKC:	PUSHJ	P,SAVE2			;SAVE AC
	MOVEI	P1,CSTREG-REGNXT(T2)	;PRESET PRED
MARKC1:	HRRZ	P1,REGNXT(P1)		;STEP TO NEXT REGION
	JUMPE	P1,CPOPJ		;QUIT IF NO MORE REGIONS
	MOVE	P2,REGLOW(P1)		;1ST CLUSTER
MARKC2:	MOVE	T1,P2			;CONVERT CLUSTER TO BN
	IMUL	T1,BPC
	PUSHJ	P,SATBIT		;FIND CORRESPONDING BIT
	IORM	T2,(T3)		 	;LIGHT BIT
	CAME	P2,REGHI(P1)		;END OF REGION?
	AOJA	P2,MARKC2		;NO, NEXT CLUSTER
	JRST	MARKC1			;YES, NEXT REGION
	SUBTTL	CLUSTER LIST

;ROUTINE TO ADD A BIT MASK TO THE CLUSTER LIST
;T1 PASSES THE BIT MASK
;T2 PASSES ADDR OF CST
;U PASSES UDB
;P2 PASSES SAT NUMBER
;P3 PASSES WORD NUMBER
DOCST:	JUMPE	T1,CPOPJ		;NONE IS EASY
	PUSHJ	P,SAVE4			;SAVE ACS
	MOVE	T3,UDBBLK(U)		;1ST CLUSTER ON UNIT
	IDIV	T3,BPC
	MOVE	P1,T3
	IMUL	P2,UDBCPS(U)		;1ST CLUSTER IN SAT
	ADD	P1,P2
	IMULI	P3,^D36			;1ST CLUSTER IN WORD
	ADD	P1,P3
	MOVE	P2,T1			;SAVE THE BITS
	MOVE	P4,T2			;ADDR OF CST
DOCST1:	JFFO	P2,DOCST2		;FIND FIRST ONE
	POPJ	P,			;NO MORE ONES, ALL DONE
DOCST2:	ADD	P1,P3			;CLUSTER IN QUESTION
	MOVE	T1,P1			;ADD IT TO THE LIST
	MOVE	T2,P4
	PUSHJ	P,ADDLST
	LSH	P2,1(P3)		;SHIFT OUT THE ONE
	AOJA	P1,DOCST1		;AND LOOP

;ROUTINE TO ADD A MULTIPLY USED CLUSTER TO THE LIST
;T1 PASSES BN
ADDMUL:	IDIV	T1,BPC			;CONVERT TO CLUSTER
	MOVEI	T2,MULCST		;ADD IT
	FALL	ADDLST

;ROUTINE TO ADD A CLUSTER TO THE LIST
;T1 PASSES CLUSTER NUMBER
;T2 PASSES ADDR OF CST (DESTROYED)
ADDLST:	PUSHJ	P,SAVE4			;SAVE ACS
	DMOVE	P1,T1			;COPY ARGS
	MOVEI	P3,CSTREG-REGNXT(P2)	;PRESET PRED
ADDLS1:	MOVE	P4,P3			;SAVE AS PRED
	HRRZ	P3,REGNXT(P4)		;STEP TO NEXT REGION
	JUMPE	P3,ADDLS3		;END OF LIST
	MOVE	T1,REGLOW(P3)		;CLUSTER JUST BEFORE REGION
	SUBI	T1,1
	CAMGE	P1,T1			;FAR ENOUGH?
	JRST	ADDLS3			;NO, BUILD NEW REGION
	CAME	P1,T1			;JUST BEFORE?
	JRST	ADDLS2			;NO, CHECK IF INTERIOR
	MOVEM	P1,REGLOW(P3)		;NEW LOWEST
	JRST	ADDLS4
ADDLS2:	MOVE	T1,REGHI(P3)		;CLUSTER JUST PAST REGION
	ADDI	T1,1
	CAMLE	P1,T1			;BEYOND END?
	JRST	ADDLS1			;YES, TEST NEXT REGION
	CAME	P1,T1			;JUST PAST?
	POPJ	P,			;NO, DUPLICATE CLUSTER
	MOVEM	P1,REGHI(P3)		;NEW HIGHEST
	AOS	CSTNUM(P2)		;ANOTHER CLUSTER
	HRRZ	T2,REGNXT(P3)		;NEXT REGION BEYOND
	JUMPE	T2,CPOPJ
	ADDI	T1,1			;NEXT CLUSTER BEYOND
	CAME	T1,REGLOW(T2)		;ADJACENT?
	POPJ	P,			;NO
	MOVE	T3,REGHI(T2)		;YES, MERGE THE TWO REGIONS
	MOVEM	T3,REGHI(P3)
	SOS	CSTCNT(P2)		;ONE LESS REGION
	HRRZ	T3,REGNXT(T2)		;UNLINK 2ND REGION
	HRRM	T3,REGNXT(P3)
	MOVEI	T1,SIZREG		;PITCH IT
	PJRST	GIVBLK

;HERE TO BUILD A NEW REGION
ADDLS3:	MOVEI	T1,SIZREG		;ALLOCATE CORE
	PUSHJ	P,GETBLK
	MOVEM	P1,REGLOW(T2)		;STORE BOUNDS
	MOVEM	P1,REGHI(T2)
	MOVEM	P3,REGNXT(T2)		;LINK IT
	HRRM	T2,REGNXT(P4)
	AOS	CSTCNT(P2)		;ANOTHER REGION
ADDLS4:	AOS	CSTNUM(P2)		;ANOTHER CLUSTER
	POPJ	P,
;ROUTINE TO PRINT CST'S (PASS ONE)
CSTSO:	MOVEI	T1,[ASCIZ /lost/]
	MOVEI	T2,[ASCIZ /marked in SAT but not in any file/]
	MOVEI	T3,LSTCST
	PUSHJ	P,CSTA
	MOVEI	T1,[ASCIZ /free/]
	MOVEI	T2,[ASCIZ /not marked in SAT but in some file/]
	MOVEI	T3,FRECST
	PUSHJ	P,CSTA
	MOVEI	T1,[ASCIZ /multiply used/]
	MOVEI	T2,[ASCIZ /belonging to more than one file/]
	MOVEI	T3,MULCST
	PJRST	CSTA

;ROUTINE TO PRINT CST'S (PASS TWO)
CSTTO:	MOVEI	T1,[ASCIZ /free/]
	MOVEI	T3,F2CST
	PUSHJ	P,CSTO
	MOVEI	T1,[ASCIZ /multiply used/]
	MOVEI	T3,M2CST
	FALL	CSTO

;ROUTINE TO PRINT A CST
;T1 PASSES CST NAME (SHORT FORM)
;T2 PASSES CST NAME (LONG FORM), USED ONLY IN PASS ONE
;T3 PASSES CST
;FL PASSES FIL, USED ONLY IN PASS TWO
CSTO:	SKIPN	CSTCNT(T3)		;ANY?
	POPJ	P,			;NO
;ENTER HERE TO ALWAYS PRINT
CSTA:	PUSHJ	P,SAVE3			;SAVE AC'S
	DMOVE	P1,T1
	MOVE	P3,T3
	MOVE	T1,CSTNUM(P3)		;NUMBER OF CLUSTERS
	PUSHJ	P,DECO
	MOVEI	T1,40
	PUSHJ	P,CO
	MOVE	T2,P1			;SHORT NAME
	PUSHJ	P,STRO
	MOVEI	T2,[ASCIZ / cluster/]
	PUSHJ	P,STRO
	MOVEI	T1,"s"
	MOVE	T2,CSTNUM(P3)
	CAIE	T2,1
	PUSHJ	P,CO
	MOVEI	T2,[ASCIZ / in /]
	PUSHJ	P,STRO
	MOVE	T1,CSTCNT(P3)		;NUMBER OF REGIONS
	PUSHJ	P,DECO
	MOVEI	T2,[ASCIZ / region/]
	PUSHJ	P,STRO
	MOVEI	T1,"s"
	MOVE	T2,CSTCNT(P3)
	CAIE	T2,1
	PUSHJ	P,CO
	MOVEI	T1,40
	PUSHJ	P,CO
	TRNE	F,F.P2			;WHICH PASS?
	JRST	CSTO9			;PASS TWO
	MOVEI	T1,"("			;PASS ONE
	PUSHJ	P,CO
	MOVE	T2,P2
	PUSHJ	P,STRO
	MOVEI	T1,")"
	PUSHJ	P,CO
	JRST	CSTO8
CSTO9:	MOVEI	T2,[ASCIZ /in file /]
	PUSHJ	P,STRO
	PUSHJ	P,FILO
CSTO8:	PUSHJ	P,CRLFO
	SKIPN	T1,CSTCNT(P3)		;OVER THRESHOLD?
	PJRST	CRLFO
	CAILE	T1,TREG
	TRNN	F,F.TTY			;AND PHYS TTY?
	SKIPA	P1,CSTREG(P3)		;NO, GET 1ST REG
	PJRST	CRLFO			;YES
CSTO1:	MOVE	T2,HPOS			;BEGINING OF LINE?
	MOVEI	T1,","
	CAIE	T2,1
	PUSHJ	P,CO			;NO, PRINT A COMMA
	MOVE	T1,REGLOW(P1)		;PRINT 1ST CLUSTER
	PUSHJ	P,OCTO
	MOVE	T1,REGLOW(P1)		;ONLY ONE CLUSTER?
	CAMN	T1,REGHI(P1)
	JRST	CSTO2			;YES
	MOVEI	T1,"-"			;NO, PRINT LAST CLUSTER
	PUSHJ	P,CO
	MOVE	T1,REGHI(P1)
	PUSHJ	P,OCTO
CSTO2:	MOVE	T1,HPOS			;PAST MARGIN?
	CAML	T1,MARGIN
	PUSHJ	P,CRLFO			;YES, NEWLINE
	HRRZ	P1,REGNXT(P1)		;NEXT REGION
	JUMPN	P1,CSTO1		;LOOP
	PJRST	DSPACE			;EXTRA CRLF
	SUBTTL	ALLOCATE/DEALLOCATE

;ROUTINE TO DEALLOCATE BAD REGIONS
DEBAD:	PUSHJ	P,SAVE1
	MOVEI	P1,BADCST
	PJRST	DECST

;ROUTINE TO DEALLOCATE LOST REGIONS
DELST:	PUSHJ	P,SAVE1
	MOVEI	P1,LSTCST
	PJRST	DECST

;ROUTINE TO DEALLOCATE FREE REGIONS
DEFRE:	PUSHJ	P,SAVE1
	MOVEI	P1,FRECST
	PJRST	DECST

;ROUTINE TO DEALLOCATE MULTIPLY USED REGIONS
DEMUL:	PUSHJ	P,SAVE1
	MOVEI	P1,MULCST
	FALL	DECST

;ROUTINE TO DEALLOCATE ALL THE REGIONS IN A CLUSTER LIST
;P1 PASSES ADDR OF CLUSTER LIST
DECST:	HRRZ	T2,CSTREG(P1)		;ADDR OF 1ST REGION
	JUMPE	T2,CPOPJ
	HRRZ	T3,REGNXT(T2)		;UNLINK IT
	HRRM	T3,CSTREG(P1)
	MOVEI	T1,SIZREG		;PITCH IT
	PUSHJ	P,GIVBLK
	JRST	DECST			;LOOP
;ROUTINE TO DEALLOCATE ALL THE UNITS IN A STR
DESTR:	PUSHJ	P,SAVE3			;SAVE ACS
	SKIPGE	P2,HUN			;HIGHEST UNIT NUMBER
	POPJ	P,
DESTR1:	SKIPN	U,TBUDB(P2)		;UNIT DATA BLOCK
	JRST	DESTR6
	SKIPN	P1,UDBPST(U)		;ADDR OF PST
	JRST	DESTR5
	MOVE	P3,UDBSPU(U)		;SATS PER UNIT
DESTR2:	MOVE	T2,(P1)			;ADDR OF SAT
	MOVE	T1,UDBWPS(U)		;WORDS PER SAT
	PUSHJ	P,GIVBLK		;DEALLOCATE CORE
	SOSE	P3			;LOOP FOR EACH SAT
	AOJA	P1,DESTR2
	MOVE	T2,UDBPST(U)		;DEALLOCATE PST ITSELF
	MOVE	T1,UDBSPU(U)
	PUSHJ	P,GIVBLK
DESTR5:	MOVE	T1,UDBSPU(U)		;DEALLOCATE SPT
	HRRZ	T2,UDBSPT(U)
	SKIPE	T2
	PUSHJ	P,GIVBLK
	MOVE	T2,U			;DEALLOCATE UDB ITSELF
	MOVEI	T1,SIZUDB
	PUSHJ	P,GIVBLK
DESTR6:	SOJGE	P2,DESTR1		;LOOP FOR EACH UNIT
	POPJ	P,
;ROUTINE TO BUILD THE DDBS
MAKDDB:	PUSHJ	P,SAVE1			;SAVE AC
	MOVE	T1,[%CNDVN]		;MONITOR VERSION
	GETTAB	T1,
	 HALT
	HLRZM	T1,MONVER
	MOVE	T1,MONVER
	CAIL	T1,70200		;7.02 (OR LATER)?
	TRO	F,F.702			;YES
	MOVEI	P1,1			;1 DDB FOR 7.01
	TRNE	F,F.702
	MOVEI	P1,NDDBS		;MANY FOR 7.02
	MOVEM	P1,DDBS
MKDDB1:	MOVEI	T1,SIZDDB		;ALLOCATE CORE FOR DDB
	PUSHJ	P,GETBLK
	MOVE	DDB,T2
	MOVEM	DDB,USRJDA(P1)
	MOVEI	T1,NBLK*BLKSIZ+3	;ALLOCATE CORE FOR BUFFER
	PUSHJ	P,GETBLK
	ADDI	T2,1			;STEP OVER DEVIOS WORD
	MOVEM	T2,DDBBUF(DDB)		;STORE IN DDB
	HRLI	T2,NBLK*BLKSIZ+1	;POINT BUFFER AT SELF
	MOVEM	T2,(T2)
	SETOM	DDBHBN(DDB)		;INVALIDATE CACHE
	SETZM	DDBPUN(DDB)		;NO UNIT OPEN YET
	MOVEM	P1,DDBCH(DDB)		;CHANNEL NUMBER
	MOVEM	P1,DDBCX(DDB)		;CHANNEL INDEX (SAME FOR NOW)
	SOJG	P1,MKDDB1
	SETZM	ACTCNT			;NO DDBS ACTIVE YET
	POPJ	P,
	SUBTTL	INTERLOCK

;ROUTINE TO GET THE INTERLOCK
GTLOK1:	MOVE	T1,MYJOB		;RESTART?
	CAMN	T1,LOKJOB
	JRST	GTLOK2			;YES
	TRZ	F,F.NOC			;CTRL-C OK
	MOVEI	T1,22			;SLEEP AWHILE
	HIBER	T1,
	 PUSHJ	P,DIE
;ENTER HERE
GETLOK:
IFN FTDBUG,<
	TRNE	F,F.LOK
	PUSHJ	P,DIE
>
	TRO	F,F.NOC			;NO CTRL-C
	AOSE	ILOCK			;GOT IT?
	JRST	GTLOK1			;NO
GTLOK2:	TRO	F,F.LOK			;YES
	MOVE	T1,MYJOB		;WE GOT IT
	MOVEM	T1,LOKJOB
	POPJ	P,

;ROUTINE TO GIVE UP THE INTERLOCK
GIVIF:	TRNN	F,F.LOK			;OWN IT?
	POPJ	P,			;NO
GIVLOK:
IFN FTDBUG,<
	TRNN	F,F.LOK
	PUSHJ	P,DIE
>
	SETZM	LOKJOB			;NOBODY OWNS IT
	SETOM	ILOCK			;GIVE IT AWAY
	TRZ	F,F.LOK+F.NOC		;CTRL-C OK
	TRZE	F,F.CC			;WAS CTRL-C TYPED?
	MONRT.				;YES, EXIT
	POPJ	P,			;CONTINUE

;CTRL-C TRAP
TRAP:	MOVEM	T1,SAVPC		;SAVE AN AC
	MOVE	T1,INTR+.EROPC		;PICK UP PC
	SETZM	INTR+.EROPC		;RE-ENABLE TRAP
	EXCH	T1,SAVPC		;STORE PC
	TRNE	F,F.NOC			;CTRL-C ALLOWED?
	TROA	F,F.CC			;NO, SET FLAG
	MONRT.				;YES, EXIT
	JRSTF	@SAVPC			;CONTINUE
;ROUTINE TO CHECK IF ALL THE OTHER JOBS ARE STILL ALIVE
ALIVE:	MOVEI	T1,NSLT-1		;START WITH HIGHEST SLOT
ALIVE1:	SKIPN	JBTJOB(T1)		;SLOT IN USE?
	JRST	ALIVE2			;NO
	HRLZ	T2,JBTJOB(T1)		;YES, GET HIS HISEG
	HRRI	T2,.GTSGN
	GETTAB	T2,
	 PUSHJ	P,DIE
	TLZ	T2,-1
	CAMN	T2,MYSGN		;SAME AS OUR HISEG?
	JRST	ALIVE2			;YES, STILL ALIVE
	SETZM	JBTSTR(T1)		;NO, DECLARE HIM DEAD
	SETZM	JBTJOB(T1)
	SETZM	JBTCHN(T1)
	SETZM	JBTLOG(T1)
	SOS	CNTSLT
ALIVE2:	SOJGE	T1,ALIVE1		;LOOP FOR EACH SLOT
	POPJ	P,
;ROUTINE TO DEALLOCATE A SLOT
GIVSLT:	MOVE	T1,MYSLT
	SETZM	JBTJOB(T1)
	SOS	CNTSLT
	POPJ	P,

;ROUTINE TO ALLOCATE A SLOT NUMBER
GETSLT:	MOVEI	T1,NSLT-1		;START WITH HIGHEST SLOT
GTSLT1:	SKIPE	T2,JBTJOB(T1)		;SLOT IN USE?
	CAME	T2,MYJOB		;BY ME?
	SOJGE	T1,GTSLT1		;NO, TRY NEXT SLOT
	JUMPGE	T1,GTSLT5		;YES
;HERE IF WE DON'T ALREADY HAVE A SLOT,
;FIND AN EMPTY ONE
	MOVEI	T1,NSLT-1		;START WITH HIGHEST SLOT
GTSLT3:	SKIPN	JBTJOB(T1)		;EMPTY?
	JRST	GTSLT4			;YES
	SOJGE	T1,GTSLT3		;NO, TRY NEXT
	PUSHJ	P,DIE
GTSLT4:	AOSA	T2,CNTSLT		;BUMP COUNT
GTSLT5:	MOVE	T2,CNTSLT		;GET COUNT
	MOVEM	T1,MYSLT		;SAVE SLOT NUMBER
	MOVE	T3,MYJOB		;TELL OUR JOB NUMBER
	MOVEM	T3,JBTJOB(T1)
	CAIE	T2,1			;ARE WE THE ONLY SLOT?
	POPJ	P,			;NO
	FALL	FIRSLT			;YES

;ROUTINE TO SET UP THE SHARED HISEG
;CALLED ONLY BY THE FIRST JOB
FIRSLT:	MOVEI	T1,HCOR			;INITIALIZE FIRST FREE
	MOVEM	T1,HFF
	SETZM	SLST			;NO STRS YET
	SETZM	BLST
	SETZM	ALST
	SETZM	CONFIG
	POPJ	P,
	SUBTTL	PICK A STR

;ROUTINE TO PICK A STR TO PROCESS
;RULES:
;TRY TO DO THE LOG STR EARLY (BUT DON'T DO IT FIRST).
;IF YOU DO THE LOG STR FIRST, YOU MIGHT BLOW AWAY
;THE JSL OF SOMEBODY WHO DID AN "ASSIGN DSK LPT".
;LET HIM RUN LONG ENOUGH TO FIGURE OUT WHAT HIS JSL IS.
;TRY NOT TO DO THE RUN STR FIRST, SOMEBODY MIGHT STILL BE
;LOADING THE EXE FILE AND WOULD GET A TRANSMISSION ERROR.
;DO NOT, UNDER ANY CIRCUMSTANCE, DO A STR THAT SOME OTHER JOB
;HAS A LOG FILE OPEN ON.
;TRY NOT TO DO A STR IF SOMEBODY ELSE IS ALREADY USING THAT CHANNEL.
PIKSTR:	PUSHJ	P,SAVE4
	SETZ	P4,			;NOTHING GOOD YET
	SKIPN	RUNSTR			;1ST TIME?
	SKIPN	P1,LOGSTR		;LOG FILE TO DISK?
	JRST	PKSTR2			;NO
	MOVEI	P2,BLST-SNFLNK		;LOG STR ALREADY STARTED?
PKSTR1:	HRRZ	P2,SNFLNK(P2)
	JUMPE	P2,PKSTR2		;YES
	CAME	P1,SNFNAM(P2)
	JRST	PKSTR1
	MOVEI	P3,NSLT-1		;NO, SOMEBODY HAS LOG FILE HERE?
PKSTR0:	CAMN	P1,JBTLOG(P3)
	JRST	PKSTR2			;YES, DON'T DO IT
	SOJGE	P3,PKSTR0
	PUSHJ	P,SWPOK			;OK TO REMOVE FROM ASL?
	 JRST	PKSTR2			;NO
	MOVE	P4,P2			;WE CAN DO THIS STR
	MOVE	P2,SNFCHN(P4)		;ANYBODY USING THIS CHAN?
	MOVEI	P3,NSLT-1
PKSTRA:	TDNE	P2,JBTCHN(P3)
	JRST	PKSTR2			;YES, TRY TO FIND SOMETHING BETTER
	SOJGE	P3,PKSTRA
	JRST	PKSTR9			;DO IT
;HERE IF CAN'T DO LOG STR 1ST
PKSTR2:	MOVEI	P2,BLST-SNFLNK		;PRESET PRED
PKSTR5:	HRRZ	P2,SNFLNK(P2)		;STEP TO NEXT STR
	JUMPE	P2,PKSTR7
	MOVE	P1,SNFNAM(P2)
	MOVEI	P3,NSLT-1		;SOMEBODY HAS LOG FILE HERE?
PKSTR6:	CAMN	P1,JBTLOG(P3)
	JRST	PKSTR5			;YES, DO NOT DO THIS STR
	SOJGE	P3,PKSTR6
	PUSHJ	P,SWPOK			;OK TO REMOVE FROM ASL?
	 JRST	PKSTR5			;NO
	MOVE	P4,P2			;NO, WE CAN DO THIS STR
	SKIPE	RUNSTR			;1ST TIME?
	CAME	P1,LOGSTR		;YES, DON'T DO LOG STR
	CAMN	P1,RUNSTR		;IS IT THE RUN STR?
	JRST	PKSTR5			;YES, TRY TO FIND SOMETHING BETTER
	MOVE	P1,SNFCHN(P2)		;SOMEBODY USING THIS CHANNEL?
	MOVEI	P3,NSLT-1
PKSTR8:	TDNE	P1,JBTCHN(P3)
	JRST	PKSTR5			;YES, TRY TO FIND SOMETHING BETTER
	SOJGE	P3,PKSTR8
	JRST	PKSTR9			;NO, GOOD, DO IT NOW
;HERE WHEN THERE'S NOTHING REALLY GOOD TO DO
;DO ONE OF THE UNPLEASANT STRS
PKSTR7:	JUMPE	P4,PKSTR3		;GO IF NONE
;HERE WHEN WE FOUND A PERFECT STR
PKSTR9:	MOVE	P1,SNFNAM(P4)		;GET STR NAME
	MOVEI	P2,BLST-SNFLNK		;FIND PRED
PKSTRB:	CAMN	P4,SNFLNK(P2)
	JRST	PKSTR4
	HRRZ	P2,SNFLNK(P2)
	JRST	PKSTRB
PKSTR4:	MOVE	P3,SNFLNK(P4)		;UNLINK FROM BLST
	MOVEM	P3,SNFLNK(P2)
	MOVEM	P4,MYSNF		;SAVE ADDR OF SNF
	MOVEM	P1,ALIAS		;STR NAME WE WILL DO
	MOVE	P2,MYSLT		;TELL EVERYBODY IT'S OURS
	MOVEM	P1,JBTSTR(P2)
	MOVE	P1,SNFCHN(P4)		;TELL THEM WHAT CHANNEL
	MOVEM	P1,JBTCHN(P2)
	JRST	CPOPJ1

;HERE IF THERE'S ABSOLUTELY NOTHING TO DO
PKSTR3:	MOVE	T1,CNTSLT		;ARE WE THE ONLY ONE?
	CAIE	T1,1
	POPJ	P,			;NO
	MOVEI	P2,BLST-SNFLNK		;PRESET PRED
PKSTRC:	HRRZ	P2,SNFLNK(P2)		;STEP TO NEXT STR
	JUMPE	P2,CPOPJ
	PUSHJ	P,SWPOK			;PARANOIA
	 PUSHJ	P,SWPBAD		;CAN'T REMOVE FROM ASL
	JRST	PKSTRC
;ROUTINE TO TELL THE WORLD ABOUT THE ONE STR WE ARE DOING
;CALL WITH INTERLOCK
PIKONE:	PUSHJ	P,SAVE4
	MOVSI	P1,LNM			;NAME STR MOUNTED AS
	MOVEM	P1,FOO+.DCNAM
	MOVE	P1,[XWD FOOSIZ,FOO]
	DSKCHR	P1,
	 PUSHJ	P,DIE
	SKIPA	P1,FOO+.DCSNM
PKONE3:	PUSHJ	P,SLEEPY		;SLEEP AWHILE
	MOVEI	P2,BLST-SNFLNK		;FIND OUR STR'S SLOT
PKONE1:	MOVE	P4,P2
	HRRZ	P2,SNFLNK(P4)
	JUMPE	P2,CPOPJ		;NOT FOUND
	CAME	P1,SNFNAM(P2)
	JRST	PKONE1
	PUSHJ	P,SWPOK			;OK TO REMOVE FROM ASL?
	 JRST	PKONE2			;NO
	EXCH	P2,P4			;YES
	JRST	PKSTR4
PKONE2:	MOVE	T1,CNTSLT		;ARE WE THE ONLY ONE?
	CAIE	T1,1
	JRST	PKONE3			;NO
SWPBAD:	PUSHJ	P,ONSIL			;TTCALL ONLY
	MOVEI	T2,[ASCIZ /Cannot process STR /]
	PUSHJ	P,STRO
	MOVE	T2,SNFNAM(P2)
	PUSHJ	P,SIXO
	MOVEI	T2,[ASCIZ / (not enough swapping space)/]
	PJRST	STRDSP
;ROUTINE TO TEST IF IT'S OK TO REMOVE A STR FROM THE ASL
;P2 PASSES ADDR OF SNF
;SKIP IF OK
SWPOK:	SKIPN	SNFVRT(P2)		;IN ASL?
	JRST	CPOPJ1			;NO, IT'S OK
	MOVE	T1,[%SWVRT]		;NUMBER OF FREE PAGES OF SWAP SPACE
	GETTAB	T1,
	 PUSHJ	P,DIE
	CAMG	T1,SNFVRT(P2)		;REMOVING ALL FREE PAGES?
	POPJ	P,			;YES, DON'T DO THAT
;HERE TO ADD UP ALL THE SWAPPING SPACE THAT'S LEFT
	SETZ	T1,
	MOVEI	T2,BLST-SNFLNK
SWPOK1:	HRRZ	T2,SNFLNK(T2)
	JUMPE	T2,SWPOK2
	CAME	T2,P2
	ADD	T1,SNFVRT(T2)
	JRST	SWPOK1
SWPOK2:	MOVEI	T2,ALST-SNFLNK
SWPOK3:	HRRZ	T2,SNFLNK(T2)
	JUMPE	T2,SWPOK4
	ADD	T1,SNFVRT(T2)
	JRST	SWPOK3
SWPOK4:	CAIL	T1,MINVRT		;BELOW TRESHHOLD?
	AOS	(P)			;OK
	POPJ	P,
;ROUTINE TO TELL THE WORLD WE ARE DONE WITH THE STR
DONSTR:	MOVE	T2,MYSNF		;LINK IT TO ALST
	MOVE	T3,ALST
	MOVEM	T3,SNFLNK(T2)
	MOVEM	T2,ALST
	MOVE	T1,MYSLT		;CLEAR TABLES
	SETZM	JBTSTR(T1)
	SETZM	JBTLOG(T1)
	SETZM	JBTCHN(T1)
	SKIPN	LOGSTR
	POPJ	P,
	CLOSE	TO,
	STATZ	TO,IO.ERR
	 PUSHJ	P,DIE
	POPJ	P,
	SUBTTL	SAVE AC

SAVE1:	EXCH	P1,(P)
	HRLI	P1,(P)
	PUSHJ	P,CJRA
	 SOS	-1(P)
	JRST	RET1

SAVE2:	EXCH	P1,(P)
	HRLI	P1,(P)
	PUSH	P,P2
	PUSHJ	P,CJRA
	 SOS	-2(P)
	JRST	RET2

SAVE3:	EXCH	P1,(P)
	HRLI	P1,(P)
	PUSH	P,P2
	PUSH	P,P3
	PUSHJ	P,CJRA
	 SOS	-3(P)
	JRST	RET3

SAVE4:	EXCH	P1,(P)
	HRLI	P1,(P)
	PUSH	P,P2
	PUSH	P,P3
	PUSH	P,P4
	PUSHJ	P,CJRA
	 SOS	-4(P)
RET4:	POP	P,P4
RET3:	POP	P,P3
RET2:	POP	P,P2
RET1:	POP	P,P1
CPOPJ1:	AOSA	(P)			;SKIP RETURN
TPOPJ:	POP	P,T1			;RESTORE T1
CPOPJ:	POPJ	P,

CJRA:	JRA	P1,(P1)

	END	KLEPTO