Trailing-Edge
-
PDP-10 Archives
-
de-10-omona-v-mc9
-
filio.mac
There are 17 other files named filio.mac in the archive. Click here to see a list.
TITLE FILIO LEVEL-D DISK SERVICE ROUTINE V17617
SUBTTL DESIGNED BY T.HASTINGS,T.WACHS,C.WHITE CODED BY T.WACHS/TW 24 OCT 78
SEARCH F,S
$RELOC
$HIGH
;THIS SOFTWARE IS FURNISHED UNDER A LICENSE AND MAY ONLY BE USED
; OR COPIED IN ACCORDANCE WITH THE TERMS OF SUCH LICENSE.
;
;COPYRIGHT (C) 1973,1974,1975,1976,1977,1978 BY DIGITAL EQUIPMENT CORP., MAYNARD, MASS.
XP VFILIO,17617
;ASSEMBLY INSTRUCTIONS: FILIO,FILIO/C_S,FT50S,FILIO
ENTRY FILIO
FILIO::
;ASSEMBLY PARAMETER FOR FNDFIL INTERLOCK
;BITS IN THE ACCESS TABLE STATUS WORD
ACPCRE==:40
ACPSUP==:20
ACPUPD==:10
ACPREN==:200
ACRSUP==:2
ACPNIU==:400000
ACMCNT==:377400
ACPSMU==:4
IOSMON==400000 ;THIS FILE IS CURRENTLY DOING MONITOR IO
IOSAU==200000 ;THIS FILE HAS THE ALTER-UFD RESOURCE
IOSUPR==100000 ;SUPER USETI/USETO DONE ON THIS CHAN
IOSDA==40000 ;THIS FIL HAS DISK ALLOCATION QUEUE
IOSRIB==20000 ;RIB IS IN MONITOR BUFFER
IOSRDC==10000 ;THIS USER CHANNEL HAS READ COUNT UP FOR FILE
IOSWLK==4000 ;FILE (WHOLE STR) IS SOFTWARE WRITE-=LOCKED
; EITHER FOR ALL JOBS OR FOR THIS JOB ONLY
IOSALC==2000 ;DONT CHANGE ACCALC WHEN GIVING UP BLOCKS OF A FILE
IOSFIR==1000 ;COMPUTE AND STORE OR CHECK THE CHECKSUM
IOSHMS==IOBEG ;HUNG-DEVICE MESSAGE ALREADY TYPED
IOSRST==IOFST ;RESET (RELEASE) WAS DONE ON A SPOOLED DEVICE
;THE FOLLOWING S BITS ARE DEFINED IN COMMON.MOD
;BECAUSE THEY WANT TO BE IN THE SAME POSITION IN S AS IN RIB STATUS WORD
;IOSHRE=100 ;HARD READ ERROR ENCOUNTERED
;IOSHWE=200 ;HARD WRITE ERROR ENCOUNTERED
;IOSSCE=400 ;SOFTWARE CHECKSUM ERROR ENCOUNTERED OR HARD POSITIONING ERROR
;IOSERR=IOSHRE+IOSHWE+IOSSCE
;IOSMER=-IOSERR
DEFINE NOSCHEDULE <>
DEFINE SCHEDULE <>
IFE FTCBDB,<
DEFINE CBDBUG<>
>
IFN FTCBDB,<
DEFINE CBDBUG(A,B)<
AOSA .+1
0
IFIDN <A>,<Y><
PUSH P,T1
MOVE T1,JOB##
CAME T1,CBUSER##
HALT .
POP P,T1
>
IFIDN <B>,<Y><
PUSHJ P,CKBAS##
>
>
>
REPEAT 0,<
NOTE ABOUT STATES OF CHANNELS, KONTROLLERS, UNITS, FILES:
C K U F
IDLE I I I I I
SEEK WAIT SW SW
SEEK S S
POSITION WAIT PW PW PW
POSITION P P P
TRANSFER WAIT TW TW TW
TRANSFER(BUSY) T OR B B B T T
NOTE ABOUT QUEUES:
THERE ARE 2 QUEUES OF FILES
SW/PW QUEUE FOR EACH UNIT
TW QUEUE FOR CHANNEL
A FILE IS ONE AND ONLY ONE OF THE FOLLOWING CONDITIONS:WITH RESPECT TO QUEUES
A.THE ONLY FILE IN SW/PW QUEUE FOR A UNIT (UNIT IN SW STATE)
B.ONE OF PERHAPS MANY FILES IN PW QUEUE FOR A UNIT(UNIT IN PW,P,TW,OR T STATE)
C.ONE OF PERHAPS MANY IN TW QUEUE FOR CHANNEL(CHAN AND KONTROL IN B STATE)
D.NONE OF THE ABOVE (FILE IN I,P, OR T STATE)
NOTE: #MEANS INSTRUCTION IS EXECUTED WITH ALL DISK PI CHANNELS OFF
%MEANS INSTRUCTION IS EXECUTED WHILE JOB## HAS CB RESOURCE
*MEANS INSTRUCTION MAY BE EXECUTED AT INTERRUPT LEVEL
(TO SAVE TYPING USED ONLY FOR INSTRUCTIONS NOT IN INTERRUPT MODULE ITSELF)
THE FOLLOWING TECO MACRO WILL PRODUCE A LISTING OF ONLY THE SUBROUTINE
NAMES AND COMMENTS PRIOR TO THEM:
ERDEV1:FILSER.MAC$EWDEV2:FILSER.SUB$
<_;SUBROUT$;0L.U1 !NTST! :S:$"GA ONTST$'.U20L1A-59"E LONTST$' Q2JI
$Q1,.PW 0,.K>EF
>
SUBTTL ALLOCATION/DEALLOCATION
CLASIZ=:^D23
TALSIZ=:^D36-CLASIZ
CLAPOS=:^D35
TALPOS=:^D35-CLASIZ
CLAMAX=1B<TALPOS>-1 ;MAX CLUSTER ADDR. (BYTE OF ALL 1'S)
DSKSCN=100000 ;SCANNING SATS FROM DISK
STRTAD=200000 ;ALLOCATE STARTING AT A SPECIFIED LOCATION
SATCHG=400000 ;SAT TABLE IN CORE DIFFERS FROM SAT TABLE ON DISK (SIGN BIT)
;SUBROUTINE TO GET A CHUNK OF BLOCKS (ANYWHERE IN STR)
;ENTER WITH T2=HOW MANY TO GET, U=PREFERRED UNIT
;EXIT CPOPJ1 IF GOT ALL (OR SOME) ON DESIRED UNIT
;EXIT CPOPJ IF GOT ALL (OR SOME) ON A DIFFERENT UNIT, WITH T3=UNIT-CHANGE POINTER
;EXIT CPOPJ WITH T3=0 IF GOT NONE (STR FULL).
;THIS ROUTINE GETS EITHER ALL, OR THE LARGEST AVAILABLE CHUNK IN STR
;RETURNS WITH T2=RETRIEVAL POINTER, T1=NUMBER OF BLOCKS OBTAINED
TAKCHK::PUSHJ P,SAVE2## ;SAVE P1,P2
MOVE P1,U ;FOR END-TEST
SETZ P2, ;INDICATE NO BEST UNIT YET
HRRZS T2 ;WANT EXACTLY C(T2) BLOCKS
;THE ABOVE INSTRUCTION WAS CHANGED FROM HRROS SINCE, WHEN THE DISKS GET FULL,
; THIS LOOP CONSUMES LOTS OF TIME. EVENTUALLY, WE WILL CHANGE IT SO THAT
; THE LARGEST HOLE IN A SAT GETS STORED IN CORE, AT WHICH POINT THIS
; WILL GO BACK THE WAY IT WAS
TAKCH1: SETZ T1, ;WANT THEM ANYWHERE ON UNIT
PUSHJ P,TAKBLK ;TRY TO GET THE BLOCKS
JRST TAKCH2 ;CANT GET THAT MANY ON THIS UNIT
CAIN U,(P1) ;GOT ALL WE ASKED FOR. SAME UNIT?
PJRST CPOPJ1## ;YES - SKIP RETURN
LDB T3,UNYLUN## ;NO. GET LOG. UNIT NO.
TRO T3,RIPNUB## ;MAKE A UNIT-CHANGE POINTER
POPJ P, ;AND NON-SKIP RETURN
;HERE ON NOT-AVAILABLE RETURN FROM TAKBLK
TAKCH2: CAIG T2,(P2) ;THIS UNIT BEST SO FAR?
JRST TAKCH3 ;NO
MOVE P2,T2 ;YES. SAVE SIZE OF LARGEST HOLE
HRL P2,U ;SAVE UNIT OF LARGEST HOLE
TAKCH3: HLRZ U,UNISTR##(U) ;STEP TO NEXT UNIT IN STR
JUMPN U,TAKCH4 ;END OF STR CHAIN?
HRRZ U,UNISTR##(P1) ;YES, STR DATA BLOCK LOC
HLRZ U,STRUNI##(U) ;1ST UNIT IN STR
TAKCH4: HRRZ T2,T1 ;RESTORE NUMBER OF BLOCKS TO GET
CAIE U,(P1) ;BACK WHERE WE STARTED?
JRST TAKCH1 ;NO, TRY THIS UNIT
JUMPE P2,CPOPJ## ;RETURN IF STR IS FULL
HLRZ U,P2 ;NOT FULL - SET BEST UNIT
HRRZ T2,P2 ;LARGEST CONTIGUOUS CHUNK AVAILABLE
JRST TAKCH1 ;GO SETTLE FOR LARGEST HOLE
;ROUTINE TO ALLOCATE BLOCKS FROM DISK
;ENTER WITH T1= WHERE TO START (OR 0 IF DONT CARE)
;T2= HOW MANY TO ALLOCATE
;LH(T2)=0 IF TAKE N OR LESS
;LH(T2)=-1 IF TAKE EXACTLY N
;RETURNS CPOPJ IF UNSUCCESSFUL WITH T2=LARGEST HOLE FOUND, T1= ORIGINAL T2, T3=0
;RETURNS CPOPJ1 IF OK, WITH T1= NUMBER OF BLOCKS TAKEN T2 = CLUSTER POINTER FOR GROUP
;T3 POINTS TO FILE STRUCTURE DATA BLOCK
TAKBLK::
PUSHJ P,SAVE4## ;SAVE P1-P4
PUSH P,W ;SAVE W
HLL W,T2 ;SAVE LH(T2) (=-1 IF EXACTLY N BLOCKS)
HRRZS T2 ;SET T2=POSITIVE NUMBER
SKIPN T2 ;MORE THAN 18 BITS WORTH?
MOVEI T2,-1 ;YES, ASK FOR MAX
SKIPN DINITF## ;IN ONCE-ONLY (REFRESHER)?
CAMG T2,UNITAL##(U) ;NO, REQUESTING MORE THAN ARE AVAILABLE?
JRST TAKBL ;NO. GET SOME BLOCKS
MOVE T3,T2 ;YES. AMONT TO GET INTO T3
SKIPLE T2,UNITAL##(U) ;ANY BLOCKS AT ALL?
JUMPGE W,TAKBL ;YES. REQUEST MAXIMUM OF UNITAL BLOCKS
MOVEI T2,0
TLNE F,OCLOSB ;NO. IS CLOSE HAPENING?
AOJA T2,TAKBL ;YES. TRY TO GET 1 BLOCK ANYWAY
;(THERE ARE BLOCKS IN SAT TABLES WHICH ARE NOT IN UNITAL, AND ARE ONLY
;GIVEN UP DURING A CLOSE UUO)
MOVE T1,T3 ;NOT CLOSE. INDICATE 0 SPACE FOUND
JRST TAKBLT ;AND TAKE ERROR RETURN
TAKBL:
LDB T4,UNYBPC## ;NUMBER OF BLOCKS PER CLUSTER
HRRZ P3,T2 ;DESIRED NUMBER OF BLOCKS
ADDI P3,-1(T4) ;CONVERT TO NUMBER OF CLUSTERS
IDIV P3,T4
MOVE T3,UNISTR##(U)
SETO T2, ;COMPUTE LARGEST ALLOWED GROUP SIZE
LDB T2,STYCNP##(T3) ;LARGEST FIELD
CAIGE T2,(P3) ;ASKING FOR TOO MUCH?
HRR P3,T2 ;YES, REDUCE REQUEST
PUSHJ P,SUPDA ;QUEUE FOR DISK ALLOCATION IF DONT ALREADY HAVE (ENTER)
SKIPN DEVUNI##(F)
JRST TAKBLU
HLRZ P1,UNISAB##(U) ;LOC OF FIRST SAT BUFFER
JUMPE T1,TAKBLA ;GO IF NO START ADDRESS SPECIFIED
;HERE WHEN A START ADDRESS SPECIFIED
SETZ T2,
JUMPL T1,TAKBLM ;NEGATIVE BLOCK NOS ARE ILLEGAL
CAML T1,UNIBPU##(U) ;REQUESTED BLOCK ABOVE TOP OF UNIT?
JRST TAKBLM ;YES, ERROR RETURN
IDIV T1,T4 ;NO, CONVERT TO CLUSTER ADDRESS
PUSH P,T1 ;SAVE CLUSTER ADDRESS
PUSHJ P,CHKNEW ;TEST FOR NEWLY-MOUNTED STR
MOVE T1,(P) ;RESTORE ADR
PUSHJ P,FNSAT ;GET SAT BUFFER FOR THIS BLOCK
SKIPN DEVUNI##(F) ;IF UNIT WAS REMOVED,
PJRST TAKBL3 ; TAKE ERROR RETERN
HRLM P1,UNISAB##(U) ;SAVE BUFFER LOC IN UNISAT
HLRE T1,SABSCN##(P1) ;-LENGTH OF WHOLE SAT DATA AREA
ADDI P1,SABBIT##(T2) ;SET RH(P1)=FIRST SAT WORD FOR ADDRESS
ADD T1,T2 ;T1=-NUMBER OF DATA WORDS AFTER THIS ONE
HRLM T1,P1 ;P1=AOBJN WORD TO SCAN TABLE
TAKBL1: MOVEI P2,0 ;P2 WILL CONTAIN LARGEST HOLE FOUND
MOVE T1,(P1) ;FIRST WORD TO LOOK AT
LSH T1,(T3) ;POSITION TO RIGHT BIT
JUMPL T1,TAKBL3 ;HOLE=0 IF 1ST BIT SET
HRLM T2,P2 ;SAVE POSITION OF HOLE
DPB T3,[POINT 6,P3,17]
MOVNS T3
MOVEI T4,^D36(T3) ;SET NUMBER OF BITS LEFT IN WORD
JFFO T1,.+2 ;COMPUTE NUMBER OF LEADING 0'S
MOVEI T2,^D36(T3) ;REST OF WORD EMPTY
TLO P3,STRTAD ;INDICATE START ADDR. SPECIFIED
PUSH P,P1 ;SAVE LOC OF 1ST DATA WORD
PUSHJ P,GETZR## ;TRY TO GET N 0'S
JRST TAKBL2 ;CANT GET ENOUGH
POP P,T1 ;FOUND THEM REMOVE GARBAGE FROM PD LIST
HLRZ P2,UNISAB##(U) ;LOC OF SAT BUFFER
JRST TAKBLQ ;MARK BLOCKS, HOUSEKEEP AND EXIT
;HERE WHEN WE COULDNT GET N CONTIGUOUS BLOCKS
TAKBL2: POP P,P1 ;PICK UP DATA LOC AGAIN
JUMPL W,TAKBL3 ;GO IF EXACTLY N NEEDED
HRR P3,P2 ;WE CAN DO WITH LESS GET LARGEST AVAILABLE
LDB T3,[POINT 6,P3,17]
HLRZ T2,P2 ;RESTORE POSITION
JRST TAKBL1 ;GO GET THEM
;HERE WHEN N NOT AVAILABLE, WE NEEDED EXACTLY N BLOCKS
TAKBL3: POP P,T1 ;TAKE STUFF OFF PD LIST
HRRZ T2,P2 ;LARGEST HOLE AVAILABLE (CLUSTER COUNT)
JRST TAKBLM ;CONVERT TO BLOCK COUNT, ERROR RETURN
;HERE WHEN A STARTING ADDRESS WAS NOT SPECIFIED
TAKBLA: PUSHJ P,CHKNEW ;MAKE SURE SATS HAVE BEEN READ
TLZ P3,677777 ;LH(P3) WILL HAVE INDEX OF LARGEST HOLE FOUND
MOVEI P2,0 ;LH(P2) WILL HAVE SIZE OF LARGEST HOLE
;THE LEFT HALF OF ACS ARE BEING USED
;BECAUSE THE PD LIST GETS VERY LONG
;IF SAT BLOCKS MUST BE READ
MOVE T1,UNIDES##(U)
TLNN T1,UNPMSB## ;DOES THE UNIT HAVE ONLY 1 SAT TABLE?
JRST TAKBLC ;YES, SKIP THE FANCY STUFF
;TRY TO FIND A SAT BLOCK IN CORE CONTAINING ENOUGH CONSECUTIVE 0'S
TAKBLB: HRRZ T1,SABTAL##(P1) ;FREE BLOCKS LEFT IN THIS SAT
CAIGE T1,(P3) ;ENOUGH TO SATISFY USER?
JRST TAKBLG ;NO. LOOK AT NEXT SAT BUFFER
;HERE WHEN THE CURRENT SAT TABLE MAY HAVE ENOUGH CONTIGUOUS CLUSTERS
;SCAN FIRST FROM WHERE THE SCAN LEFT OFF THE LAST TIME
TAKBLC: MOVE T1,SABHOL##(P1) ;BIGGEST HOLE IN SAT
CAIGE T1,(P3) ;TRYING FOR MORE THAN BIGGEST?
JUMPGE T1,TAKBLF ;YES, SKIP SCAN IF WE KNOW SIZE OF HOLE
; (SABHOL=-1 IF WE DONT KNOW THE SIZE)
MOVSI T1,SABBIT##(P1) ;SET UP AN AOBJN WORD FOR SCAN
HRLZ T2,SABSCN##(P1) ;COMPUTE DISTANCE FROM START TO C(SABSCN)
SUB T2,T1 ;=+N
ADD T2,SABSCN##(P1) ;LH=DISTANCE FROM C(SABSCN) TO TOP
;RH=WHERE TO START LOOKING
PUSH P,P1 ;SAVE THE BUFFER LOC
MOVE P1,T2 ;AOBJN WORD FOR SCAN
HRRI P2,0 ;SET BEST SO FAR TO 0
PUSHJ P,GETZ## ;AND TRY TO GET N 0'S
TAKBLD: SKIPA T2,(P) ;COULDN'T GET THEM
JRST TAKBLP ;FOUND THEM - UPDATE AND EXIT
;HERE WHEN N WERENT AVAILABLE FROM WHERE SCAN LAST LEFT OFF
;RESCAN TABLE FROM THE START
MOVEI P1,SABBIT##(T2) ;FIRST DATA LOC IN BUFFER
HLL P1,SABSCN##(T2) ;-LENGTH OF ENTIRE DATA AREA
PUSHJ P,GETZ## ;SCAN WHOLE SAT TABLE
SKIPA T1,P2 ;STILL CANT FIND ENOUGH
JRST TAKBLP ;FOUND THEM - WRAP UP
;HERE WHEN THE CURRENT SAT BUFFER DOESN'T HAVE ENOUGH
POP P,P1 ;RESTORE BUFFER LOC
TAKBLF: HRR P2,T1
HRRZM P2,SABHOL##(P1) ;SAVE SIZE OF LARGEST HOLE
HLRZ T1,P2 ;PREVIOUS MAXIMUM
CAIL T1,(P2) ;WAS THIS SAT TABLE BETTER?
JRST TAKBLG ;NO
HRLS P2 ;YES. SAVE SIZE IN LH(P2)
HRR P2,P1 ;SET RH(P2)=BUFFER LOC
LDB T1,SAYNDX## ;GET INDEX OF SAT
DPB T1,[POINT 11,P3,17] ;SAVE INDEX IN LH(P3)
TAKBLG: HRR P2,P1 ;RH(P2)=CURRENT BUFFER
MOVE T1,UNIDES##(U) ;DOES THIS UNIT HAVE ONLY 1 SAT?
TLNN T1,UNPMSB##
JRST TAKBLL ;YES. CANT GET ENOUGH
TLNE P3,DSKSCN ;NO. SCANNING SATS FROM DISK?
JRST TAKBLI ;YES. READ NEXT ONE
HLRZ P1,SABRNG##(P1) ;NO. STEP TO NEXT IN-CORE SAT TABLE
HLRZ T1,UNISAB##(U) ;BACK WHERE WE STARTED?
CAME P1,T1
JRST TAKBLB ;NO. TRY RHIS SAT TABLE
;HERE WHEN ALL SAT TABLES IN CORE ARE THROUGH
;NOTICE THAT WHILE WE WERE LOOKING AT ONLY IN-CORE SAT TABLES WE
;SCANNED ONLY THOSE WHICH HAD A CHANCE OF SUCCESS.
;NOW ALL SAT'S WILL BE LOOKED AT SINCE WE WANT TO FIND
;THE MAXIMUM NUMBER OF CONTIGUOUS BITS IF WE CANT GET ENOUGH
TLO P3,DSKSCN ;INDICATE READING SATS FROM DISK
HLR P2,UNISAB##(U) ;POINT P2 TO CURRENT SAT
TROA P4,-1 ;START AT SAT TABLE 0
TAKBLI: LDB P4,SAYNDX## ;INDEX OF LAST SAT LOOKED AT
ADD P4,UNISPT##(U) ;COMPUTE ABSOLUTE ADDR. OF ITS POINTER
TAKBLJ: SKIPN 1(P4) ;IS TABLE EXHAUSTED?
JRST TAKBLL ;YES. CANT GET ENOUGH
LDB T1,[POINT TALSIZ,1(P4),TALPOS] ;FREE COUNT OF THIS SAT
HLRZ T3,P2 ;LARGEST HOLE FOUND SO FAR
CAIG T1,(T3) ;THIS SAT HAVE AT LEAST THAT MANY FREE BITS?
AOJA P4,TAKBLJ ;NO. DONT BOTHER SCANNING IT
SUB P4,UNISPT##(U) ;YES. COMPUTE INDEX OF SATSPT TABLE
MOVEI P4,1(P4)
PUSHJ P,SDWNDA ;UNQUEUE,REQUEUE FOR DISK ALLOCATION
PUSHJ P,SUPDA ;SO ANY WAITING REQUEST WILL BE
; SATISFIED BEFORE WE DO IO
SKIPN DEVUNI##(F)
JRST TAKBLU
PUSHJ P,SATST ;GET THE CORRESPONDING SAT TABLE
MOVE T1,SABHOL##(P2) ;SIZE OF BIGGEST HOLE IN SAT
HLRZ T2,P2 ;BIGGEST HOLE FOUND SO FAR
CAMGE T1,T2 ;WORTH WHILE TO SCAN THE SAT?
JUMPGE T1,TAKBLI ;NOT IF WE REALLY KNOW (SABHOL POSITIVE)
PUSH P,P2 ;SAVE BUFFER LOC
HRRI P2,0 ;PRESET LARGEST HOLE IN SAT
JRST TAKBLD ;GO SCAN THE SAT (FROM BEGINNING)
;HERE WHEN ALL SATS SCANNED, NONE HAS ENOUGH
TAKBLL: TLNE P2,-1 ;FIND ANY AT ALL?
JUMPGE W,TAKBLN ;YES, NEED EXACTLY N?
HLRZ T2,P2 ;YES. T2=LARGEST HOLE FOUND
TAKBLM: HRRZ T1,P3 ;T1=SIZE REQUESTED
LDB T3,UNYBPC## ;CONVERT BOTH CLUSTER COUNTS
IMUL T1,T3 ;TO BLOCK NUMBERS
IMUL T2,T3
SETZ T3, ;T3=0 ON ERROR
POP P,W ;RESTORE W
PJRST SDWNDA ;GIVE UP DA QUEUE AND ERROR RETURN
;HERE WHEN NOT ENOUGH WERE FOUND, BUT A LESSER AMOUNT WILL DO
TAKBLN: LDB P4,[POINT 11,P3,17] ;INDEX OF BEST SAT TABLE
PUSHJ P,SATST ;GET CORRESPONDING SAT BLOCK IN
PUSH P,P2 ;SAVE BUFFER LOC
HLRZ P3,P2 ;SIZE OF LARGEST HOLE
TAKBLO: MOVE T1,(P) ;SET UP AN AOBJN WORD
HLL P1,SABSCN##(T1) ; FOR THE BUFFER
HRRI P1,SABBIT##(T1)
MOVEI P2,0 ;SET LARGEST HOLE TO 0
PUSHJ P,GETZ## ;GET THE BLOCKS
JRST TAKBLR ;SOMEBODY SNUCK IN!
;HERE WHEN A BUNCH OF BLOCKS HAVE BEEN OBTAINED
;THE BUFFER LOC IS ON THE PD LIST
TAKBLP: POP P,P2 ;GET BUFFER LOC
HRRZ T1,P4 ;POSITION OF HOLE
SUBI T1,SABBIT##(P2) ;CONVERT TO A CLUSTER NUMBER
IMULI T1,^D36
HLRZ T2,P4
MOVNS T2 ;-BIT POSITION IN WORD
ADDI T1,^D36(T2) ;CLUSTER NO RELATIVE TO START OF SAT
LDB T2,[POINT CLASIZ,SABFIR##(P2),CLAPOS] ;FIRST ADDRESS IN SAT
ADD T1,T2 ;COMPUTE ACTUAL CLUSTER NUMBER (RELATIVE TO UNIT)
PUSH P,T1 ;SAVE IT ON THE LIST
HRRM P4,SABSCN##(P2) ;SET WHERE TO START NEXT TIME
;HERE WITH CLUSTER ADDRESS ON PD LIST
TAKBLQ: PUSHJ P,SETOS## ;MARK THE BLOCKS IN THE SAT TABLE
STOPCD .+1,DEBUG,BAO, ;++BIT ALREADY ONE
HRRZS P3 ;P3=PLUS NUMBER OF CLUSTERS GOTTEN
PUSHJ P,FIXCNT ;UPDATE COUNTS
POP P,T2 ;RESTORE CLUSTER ADDRESS
MOVE T3,UNISTR##(U) ;LOC OF STRUCTURE DB
DPB P3,STYCNP##(T3) ;SAVE CLUSTER COUNT IN T2
PUSHJ P,SDWNDA ;GIVE UP DA QUEUE (EXCEPT IF SIM UPDATE)
JRST WPOPJ1## ;AND TAKE GOOD RETURN
;HERE WHEN THE BEST-SO-FAR WHICH WE CAREFULLY COMPUTED IS NO LONGER
;THERE - SOMEONE HAS SNUCK IN WHEN WE UNQUEUED AND GRABBED A CHUNK
;OUT OF THE HOLE WE REMEMBERED
TAKBLR: JUMPE P2,TAKBLS ;START ALL OVER IF NOTHING LEFT IN SAT
MOVE P3,P2 ;SOMETHING LEFT - SETTLE FOR IT
JRST TAKBLO ;GO TAKE THE BLOCKS
TAKBLS: POP P,T1 ;REMOVE JUNK FROM PD LIST
SKIPLE T2,UNITAL##(U) ;ANY BLOCKS AT ALL IN UNIT?
JRST TAKBLA ;YES. TRY OVER FROM BEGINNING
HRRZ T1,P3 ;NO. RESTORE AMOUNT REQUESTED
TAKBLT: POP P,W ;RESTORE W
PUSH P,T1 ;SAVE ACS
PUSH P,T2
TLNE S,IOSDA
PUSHJ P,SDWNDA ;GIVE UP THE DA RESOURCE
MOVEI T4,.ERFUL ;STR - FULL ERROR
MOVE T1,UNISTR##(U) ;STR DB LOC
SKIPG STRTAL##(T1) ;STR FULL?
PUSHJ P,SETINJ ;YES, SET UP FOR INTERCEPT
JFCL ;NOT ENABLED OR STR NOT FULL
SETZ T3, ;T3=0 ON ERROR
POP P,T2 ;RESTORE ACS
PJRST TPOPJ## ;AND EXIT
;HERE IF THE UNIT WAS YANKED (DEVUNI=0)
TAKBLU: SETZB T2,T3 ;INDICATE NO ROOM
PUSHJ P,DWNDA ;GIVE UP DA
PJRST TPOPJ## ;AND REURN
IFN FTCCIN!FTDAEM!FTOPRERR,<
;ROUTINE TO CALL SETINT
SETINJ::PUSH P,J ;SAVE J
PUSH P,M ;SETINT CLOBBERS M
LDB J,PJOBN## ;JOB NUMBER
MOVE R,JBTADR##(J) ;SET R
SKIPE J ;SKIP IF SWAP I/O
PUSHJ P,SETINT## ;TEST INTERCEPT
SOS -2(P) ;SET FOR NON-SKIP RETURN
POP P,M ;RESTORE M
PJRST JPOPJ1## ;INTERCEPT SET
> ;END FTCCIN!FTDAEM!FTOPRERR
;SUBROUTINE TO RETURN BLOCKS (DEALLOCATE)
;ENTER WITH T1= DISK ADDRESS T2= HOW MANY TO DEALLOCATE
GIVBLK::PUSHJ P,SAVE4## ;SAVE P1-P4
PUSHJ P,UPDA ;GET DA RESOURCE
HLRZ P1,UNISAB##(U) ;LOC OF FIRST SAT TABLE
LDB T4,UNYBPC## ;NUMBER OF BLOCKS PER CLUSTER
ADDI T2,-1(T4) ;CONVERT BLOCK COUNT TO CLUSTERS
IDIV T2,T4
MOVNM T2,P3 ;P3=-N FOR UPDATING COUNTS
IDIV T1,T4 ;CONVERT TO CLUSTER ADDRESS
PUSHJ P,FNSAT ;FIND THE SAT FOR THIS ADDRESS
MOVEI T4,SABBIT##(P1)
ADDI T4,(T2) ;POSITION IN TABLE
MOVEI T1,^D36
SUBI T1,(T3) ;POSITION
MOVN T3,P3 ;COUNT
PUSHJ P,CLRBTS## ;CLEAR THE BITS
STOPCD DWNDA,DEBUG,BAZ, ;++BIT ALREADY ZERO
SETOM SABHOL##(P1) ;INDICATE SIZE OF LARGEST HOLE IN SAT UNKNOWN
HRRZ P2,P1 ;SET BUFFER LOC
PUSHJ P,FIXCNT ;UPDATE SOME COUNTS
PJRST DWNDA ;GIVE UP DA RESOURCE AND RETURN
;SUBROUTINE TO UPDATE SOME COUNTS
;ENTER WITH P3 = HOW MANY CLUSTERS (PLUS-ALLOCATION, NEG - DEALLOCATION)
; P2=LOC OF SAT BUF.
;RETURNS WITH T1=NUMBER OF BLOCKS
FIXCNT: MOVN T1,P3 ;-NUMBER OF CLUSTERS
LDB T4,UNYBPC##
IMUL T1,T4 ;-NUMBER OF BLOCKS
SKIPE DINITF## ;IF IN ONCE-ONLY
JRST FIXCN2 ;JUST SET SATCHG BIT
MOVN T2,P3 ;-NUMBER OF CLUSTERS
ADDM T1,UNITAL##(U) ;UPDATE UNIT FREE-TALLY
MOVE T3,UNISTR##(U) ;UPDATE STR FREE-TALLY
JUMPE T3,FIXCN3 ;IF SUPER I/O
ADDM T1,STRTAL##(T3)
MOVE T3,DEVUFB##(F) ;UPDATE USERS QUOTA
JUMPE T3,FIXCN1 ;CANT UPDATE QUOTA IF NO UFD
MOVE T4,UFBTAL##(T3)
ADDM T1,UFBTAL##(T3)
JUMPLE T1,FIXCN1 ;IF INCREASING UFBTAL,
JUMPL T4,FIXCN1 ; UFBTAL WAS POSITIVE
SKIPGE UFBTAL##(T3) ; AND UFBTAL HAS OVERFLOWED
HRLOS UFBTAL##(T3) ; MAKE IT HUGELY POSITIVE AGAIN
FIXCN1: HRRZ T3,DEVACC##(F) ;UPDATE HIGHEST BLOCK ALLOCATED
JUMPE T3,FIXCN3
MOVNS T1
TLNN S,IOSALC ;LEAVE ACCALC ALONE IF BIT IS ON
ADDM T1,ACCALC##(T3)
FIXCN3: ADD T2,SABTAL##(P2)
TRNE T2,400000 ;COUNT GO NEGATIVE?
MOVEI T2,0 ;YES, SET TO 0
HRRM T2,SABTAL##(P2)
LDB T3,SAYNDX## ;AND IN SATSPT TABLE
ADD T3,UNISPT##(U)
DPB T2,[POINT TALSIZ,(T3),TALPOS]
FIXCN2: MOVSI T4,SATCHG ;INDICATE THAT THE SAT
ORM T4,SABFIR##(P2) ; BLOCK HAS CHANGED
IFE FTCCIN!FTDAEM!FTOPRERR,<
SETINJ::
>
MOVMS T1
POPJ P, ;AND RETURN
;SUBROUTINE TO FIND THE SAT BUFFER ASSOCIATED WITH A GIVEN DISK ADDRESS
;IT MAY WRITE OUT A CURRENT SAT AND READ IN A NEW ONE
;ENTER WITH P1=LOC OF 1ST SAT BUFFER IN RING, T1 = DESIRED CLUSTER ADDRESS
;EXIT WITH T2=RELATIVE LOC IN SAT TABLE WITHIN SAT BLOCK
; T3=BIT POSITION P1=BUFFER LOC
;P3,T1 UNCHANGED, P1,P2, P4 CHANGED
FNSAT: HRRZ T2,UNICPS##(U) ;NUMBER OF CLUSTERS/SAT TABLE
MOVE P2,P1 ;USE P2 FOR END TEST
;TRY TO FIND A SAT IN CORE FOR THIS CLUSTER ADDRESS
FNSA1: LDB T3,[POINT CLASIZ,SABFIR##(P1),CLAPOS] ;FIRST DISK ADDRESS IN SAT BUFFER
CAMGE T1,T3 ;IN THIS SAT?
JRST FNSA2 ;NO
ADD T3,T2 ;MAYBE, CHECK IF OVER TOP
CAMGE T1,T3 ;THIS THE ONE?
JRST FNSA4 ;YES, COMPUTE POSITION
FNSA2: HLRZ P1,SABRNG##(P1) ;STEP TO NEXT SAT BUFFER
CAME P1,P2 ;THROUGH?
JRST FNSA1 ;NO. TEST IT
;HERE WHEN THE DESIRED SAT IS NOT IN CORE. READ IT FROM DISK
PUSH P,T1 ;SAVE CLUSTER ADDRESS
IDIV T1,T2 ;COMPUTE INDEX TO SATPFI TABLE
MOVE P4,T1 ;STUFF IT IN P4
PUSHJ P,NEWSAT ;WRITE THE CURRENT SAT, READ IN NEW
POP P,T1 ;RESTORE CLUSTER ADDRESS
;HERE WHEN DESIRED SAT IS IN CORE
;T1=CLUSTER ADDRESS, P1 = LOC OF BUFFER
FNSA4: LDB T2,[POINT CLASIZ,SABFIR##(P1),CLAPOS] ;1ST ADDRESS OF SAT
SUBM T1,T2 ;-DESIRED ADDRESS
IDIVI T2,^D36 ;COMPUTE WORD COUNT, SHIFT NUMBER
POPJ P,
;SUBROUTINE TO ENTER REQUEST IN DISK-ALLOCATION QUEUE
;ALL ACS RESPECTED
;CALL SUPDA IF MIGHT ALREADY HAVE DA
SUPDA:: TLNN S,IOSDA ;DONT GET DA IF ALREADY HAVE IT
UPDA:: PUSHJ P,DAWAIT## ;HAVE TO WAIT
TLO S,IOSDA ;HAVE DA QUEUE,LIGHT BIT
PJRST STRIOS ;SAVE S AND RETURN
;SUBROUTINE TO UNQUEUE DA REQUEST
;ALL ACS RESPECTED
SDWNDA::
IFN FTDSIM,<
;SUBROUTINE TO GIVE UP DA EXCEPT FOR A SIM UPDATE FILE
;ALL ACS RESPECTED
PUSH P,T1 ;SAVE AN AC
HRRZ T1,DEVACC##(F) ;A.T. LOC
JUMPE T1,SDWND1 ;JUMP IF NO A.T. (SUPER I/O OR THE REFRESHER)
MOVE T1,ACCSMU##(T1)
TRNE T1,ACPSMU ;SIM UPDATE?
JRST TPOPJ## ;YES, DONT GIVE UP DA
SDWND1: POP P,T1 ;NO, GIVE UP DA RESOURCE
>
DWNDA:: TLZN S,IOSDA ;CLEAR THE BIT
STOPCD CPOPJ,DEBUG,DHD,;++ DON'T HAVE DA
PUSHJ P,DAFREE## ;DECREMENT REQUEST COUNT
PJRST STRIOS ;STORE S IN DDB AND RETURN
;SUBROUTINE TO GET THE AU RESOURCE
;ALL ACS RESPECTED
UPAU::
PUSHJ P,AUWAIT## ;WAIT A WHILE
TLO S,IOSAU ;HAVE IT NOW
PJRST STOIOS## ;RETURN
;SUBROUTINE TO RELEASE AU RESOURCE
;ALL ACS RESPECTED
DWNAU:: TLZN S,IOSAU ;CLEAR THE BIT
STOPCD CPOPJ,DEBUG,DHA,;++DON'T HAVE AU
PUSHJ P,AUFREE## ;DECREMENT COUNT
PJRST STOIOS## ;SAVE S AND RETURN
;SUBROUTINE TO FIND A PARTICULAR SAT IN THE
;SAT-BUFFER RING. IF NOT FOUND, READ IT IN
;ENTER WITH P2 = LOC OF SAT BUFFER, P4 = INDEX OF SAT ADDRESS TABLE TO READ
;P2 ON EXIT CONTAINS THE BUFFER LOC. LH(P2) UNCHANGED.
SATST: HRRZ T2,P2 ;LOC OF BUFFER (FOR END TEST)
SATS2: LDB T3,SAYNDX## ;INDEX OF THIS SAT
CAIN T3,(P4) ;RIGHT ONE?
POPJ P, ;YES. RETURN
HLR P2,SABRNG##(P2) ;NO. STEP TO NEXT
CAIE T2,(P2) ;THROUGH?
JRST SATS2 ;NO. TEST THIS BUFFER
;YES. READ THE SAT
;SUBROUTINE TO (POSSIBLY) WRITE THIS SAT, READ IN NEW ONE
;ENTER WITH P2 = LOC OF SAT BUFFER, P4 = INDEX OF SAT ADDRESS TABLE TO READ
NEWSAT: SKIPG SABFIR##(P2) ;CURRENT SAT BEEN CHANGED?
PUSHJ P,SATWRT ;YES. WRITE IT OUT
;READ NEW SAT
;SUBROUTINE TO READ A NEW SAT BLOCK INTO CORE
;THE SAT DISK ADDRESS IS SPECIFIED BY P4 (INDEX INTO ADDRESS TABLE)
;P2 POINTS TO THE IN-CORE BUFFER
SATRED::DPB P4,SAYNDX## ;SAVE INDEX TO SATPFI IN MON BUFFER
MOVE T4,P4 ;INDEX TO SATSPT INTO T4
PUSHJ P,SATADR ;SET UP PARAMETERS TO READ
HRRM T3,SABTAL##(P2) ;SAVE FREE SPACE FOR THIS SAT
PUSH P,DEVUNI##(F) ;SAVE DEVUNI
MOVEM U,DEVUNI##(F) ;UNIT WE'RE TALKING TO
PUSHJ P,MONRED ;READ THE SAT
POP P,DEVUNI##(F) ;RESTORE DEVUNI
LDB T4,SAYNDX## ;RESET INDEX
HRRZ T1,UNICPS##(U) ;NO. OF CLUSTERS PER SAT
IMUL T1,T4 ;TIMES INDEX = FIRST ADDR OF SAT
DPB T1,[POINT CLASIZ,SABFIR##(P2),CLAPOS] ;SAVE IN BUFFER
MOVEI T1,SABBIT##(P2) ;START SCAN AT 1ST LOC OF TABLE
HRRM T1,SABSCN##(P2) ;SAVE IN SCAN LOC
SETOM SABHOL##(P2) ;DONT KNOW SIZE OF LARGEST HOLE
SKIPE DINITF## ;IF IN ONCE-ONLY
PJRST SATW2 ;DONT CHECK FOR CONSISTENCY
HLL T1,SABRNG##(P2) ;STEP TO NEXT SAB IN RING
HLLM T1,UNISAB##(U) ;THIS WILL BE NEXT TO BE READ INTO
MOVE T1,SABSCN##(P2) ;AOBJN WORD FOR SAT BITS
SKIPN T2,T3 ;CHECK FOR ERRORS DETECTED IN MONRED
PUSHJ P,SATCN ;COUNT 0'S IN SAT
HRRZ T1,SABTAL##(P2) ;DONE. # OF 0'S WE EXPECT
CAIN T1,(T2) ;RIGHT?
PJRST SATW2 ;YES. ZERO SATCHG AND RETURN
SATBAD: MOVSI T2,UNPSER## ;NO. COUNT NO OF SOFTWARE ERRS
ADDM T2,UNIMCT##(U) ; UP BY ONE IN UNIT DB
MOVNS T1 ;DECREASE UNIT AND STR TALLIES
LDB T2,UNYBPC##
IMULI T1,(T2) ;CONVERT TO # OF BLOCKS
ADDM T1,UNITAL##(U) ;BY THE AMOUNT WE EXPECTED
HRRZ T3,UNISTR##(U)
ADDM T1,STRTAL##(T3)
SETZ T1, ;T1=0
ADD T4,UNISPT##(U) ;POINT TO RIGHT SPT WORD
DPB T1,[POINT TALSIZ,(T4),TALPOS] ;SET COUNT IN THIS SAT TO 0
HRRM T1,SABTAL##(P2) ;SET SABTAL FOR THIS SAT TO 0
MOVE T1,SABSCN##(P2) ;SET ALL BITS IN THE SAT TO 1
SETOM (T1)
AOBJN T1,.-1
PJRST SATW2 ;ZERO SATCHG (SO SAT WONT BE WRITTEN) AND EXIT
;ROUTINE TO COUNT 0 BITS IN A TABLE
;ARGS: T1=AOBJN POINTER TO TABLE
;VALUES: T2=NO. OF 0-BITS IN TABLE
;RESPECTS T4
SATCN:: SETZ T2, ;T2 WILL COUNT 0'S FOUND
PUSH P,T4
SATR1: MOVE T3,(T1) ;COUNT 0-BITS IN 0(T1)
SETCMB T4,T3 ;ITS EASIER TO COUNT 1'S
LSH T4,-1
AND T4,[333333,,333333]
SUB T3,T4
LSH T4,-1
AND T4,[333333,,333333]
SUBB T3,T4 ;EACH OCTAL DIGIT REPLACED BY NUMBER OF 1S IN IT
LSH T4,-3
ADD T3,T4
AND T3,[070707,,070707]
IDIVI T3,77 ;CASTING OUT 63'S
ADDI T2,(T4) ;ACCUMULATE ANSWER IN T2
AOBJN T1,SATR1 ;COUNT BITS IN NEXT WORD
PJRST T4POPJ## ;DONE, ANSWER IN T2
;SUBROUTINE TO WRITE OUT A SAT BLOCK
;THE BUFFER IS SPECIFIED BY P2
SATWRT::TLNE S,IOSDA ;JOB HAVE DA RESOURCE?
JRST SATW1 ;YES
PUSHJ P,UPDA ;NO, GET IT (PROBABLY COMING FROM WTUSAT)
PUSHJ P,SATW1 ;WRITE THE SAT
PJRST DWNDA ;GIVE UP DA AND RETURN
SATW1: LDB T4,SAYNDX## ;INDEX TO SATPFI
PUSHJ P,SATADR ;SET PARAMETERS FOR WRITE
PUSH P,DEVUNI##(F)
MOVEM U,DEVUNI##(F) ;UNIT WE'RE TALKING TO
PUSHJ P,MONWRT ;WRITE THE SAT
POP P,DEVUNI##(F)
SATW2: MOVSI T1,SATCHG ;ZERO SAT-CHANGED BIT
ANDCAM T1,SABFIR##(P2)
POPJ P, ;AND RETURN
;SUBROUTINE TO SET UP PARAMETERS FOR A SAT-BLOCK READ OR WRITE
;ENTER WITH T4=INDEX TO SATSPT TABLE P2=BUFFER ADDRESS
;EXIT WITH T1=IOWD FOR THE SAT T2= DISK ADDRESS T3=NO. OF FREE CLUSTERS
SATADR: ADD T4,UNISPT##(U) ;COMPUTE ADDRESS OF SATSPT ENTRY
LDB T2,[POINT CLASIZ,(T4),CLAPOS] ;GET DISK ADDRESS
LDB T3,UNYBPC## ;NO OF BLOCKS PER CLUSTER
IMUL T2,T3 ;CONVERT CLUSTER ADR TO BLOCK ADR
LDB T3,[POINT TALSIZ,(T4),TALPOS] ;GET FREE COUNT
MOVEI T1,SABBIT##-1(P2) ;ADDRESS FOR IOWD
HLL T1,SABSCN##(P2) ;LENGTH OF DATA
POPJ P, ;RETURN
;SUBROUTINE TO CHECK FOR VIRGIN SAB RING (NEWLY MOUNTED F/S)
; IF NEW RING (CLUST. ADR.=-1), READ SATS FROM DISK INTO ALL SABS
;CALL WITH P1=1ST SAB ADR
CHKNEW: LDB T1,[POINT CLASIZ,SABFIR##(P1),CLAPOS] ;CLUSTER ADDR.
CAME T1,[EXP CLAMAX] ;-1?
POPJ P, ;NO. OK
PUSHJ P,SAVE4## ;YES-THIS IS NEW F/S, NEED TO GET SATS
SETZM P4 ;P4=1ST INDEX (SABNDX)
MOVE P2,P1 ;P2=1ST SAB ADDR
CHKNE2: PUSHJ P,SATRED ;READ THE SAT
HLRZ P2,SABRNG##(P2) ;STEP TO NEXT SAB
CAIE P2,(P1) ;BACK WHERE WE STARTED?
AOJA P4,CHKNE2 ;NO - BUMP INDEX AND READ NEXT SAT
POPJ P, ;YES - FINISHED
;SUBROUTINE TO CHECK THE AMOUNT OF SPACE A USER WANTS TO ALLOCATE AGAINST HIS QUOTA
;ALSO CHECKS TO SEE THAT THERE IS SPACE ON THE STR
;ENTER WITH T2= AMOUNT TO ALLOCATE, LH(T2)=-1 IF FROM OUTPUT
;EXIT WITH T2 = AMOUNT ALLOWED (MAY BE 0)
;IF 0 IOBKTL HAS BEEN SET IN S, DEVIOS
CHKQTA::PUSH P,T2 ;SAVE AMOUNT TO ALLOCATE
TLZ T2,400000 ;MAKE SURE ITS POSITIVE
MOVE T4,DEVUFB##(F) ;CHECK AGAINST QUOTA
MOVE T3,UNISTR##(U)
SKIPG STRTAL##(T3) ;ROOM ON STR?
JRST CHKQT1 ;NO, COMPLAIN TO USER
IFE FTDQTA,< ;IF NO QUOTAS,
PJRST TPOPJ## ;OK RETURN IF STR NOT FULL
>
IFN FTDQTA,<
MOVE T1,UFBTAL##(T4)
TLNN F,OCLOSB ;DOING A CLOSE?
CAMG T2,T1 ; OR BELOW QUOTA?
PJRST TPOPJ## ;YES. OK
SUB T1,STROVR##(T3) ;CHECK MAX OVERDRAW ALLOWED IN STR
CAML T2,T1 ;TAKE LESSER OF KOYGRP, AMOUNT LEFT
SKIPLE T2,T1 ;IF 0 OVERDRAW USED UP
JRST CHKQT2 ;CHECK IF QUOTA JUST GOING NEGATIVE
;HERE IF QUOTA IS NOW EXHAUSTED OR STR IS FULL
CHKQT1: MOVEI T4,.ERFUL
PUSHJ P,SETINJ ;IS USER INTERCEPTING DISK FULL?
SKIPA ;NO, STOP JOB
JRST CHKQ1B ;YES, LIGHT AN ERROR BIT AND RETURN
>;END FTDQTA
IFE FTDQTA,<
CHKQT1:
>
MOVE T1,JOB##
MOVE T1,JBTSTS##(T1) ;DOES JOB WANT TO STOP ON FULL?
TRNE T1,JS.SFL
PUSHJ P,SAVSTS ;YES, SAVE A RECORD OF RESOURCES JOB OWNS
JRST CHKQ1B ;OWNS MON BUF - LOSE
SKIPGE -1(P) ;IF CAME FROM OUTPUT,
SOS DEVRET##(F)
PUSH P,S
TRO S,DR
PUSHJ P,DSKFUL ;COMPLAIN TO USER, STOP JOB
POP P,S
MOVEM S,DEVIOS(F)
POP P,T3 ;CONTINUED - RECORD OF RESOURCES
PUSHJ P,RESSTS ;GET BACK RESOURCES
POP P,T2 ;RESTORE AMOUNT TO GET
JUMPGE T2,CHKQTA
AOS DEVRET##(F)
JRST CHKQTA ;AND TEST AGAIN
;HERE FOR ERROR RETURN
CHKQ1B: POP P,(P) ;REMOVE AMOUNT TO GET FROM LIST
SETZ T2, ;INDICATE CANT GET ANY BLOCKS
PJRST ERRFUL ;LIGHT BIT AND RETURN
IFN FTDQTA,<
;HERE WHEN QUOTA IS NEGATIVE OR ABOUT TO GO NEGATIVE.
;IF QUOTA IS GT 0 TYPE A WARNING MESSAGE TO USER
CHKQT2: SKIPGE UFBTAL##(T4) ;QUOTA ALREADY NEG?
PJRST TPOPJ## ;YES. MESSAGE ALREADY TYPED
HRRM T2,(P) ;SAVE REDUCED AMOUNT
MOVEI T4,.ERQEX ;QUOT1 - EXHAUSTED ERROR
PUSHJ P,SETINJ ;SET INTERCEPT IF HE WANTS IT
SKIPA ;NO ENABLED - TYPE MESSAGE
JRST CHKQT4 ;INTERCEPTING - SKIP MESSAGE
PUSH P,S ;SAVE ACS WHICH SCNSER WILL USE
PUSH P,U
PUSH P,J
PUSH P,F
MOVEI U,0
PUSHJ P,TTYFNU## ;FIND TTY FOR CURRENT JOB; SET U,J,F
JUMPE U,CHKQT3
PUSHJ P,INLMES##
ASCIZ /
[EXCEEDING QUOTA ON /
IFN FTSTR,<
MOVE T2,(P) ;DDB
MOVE T2,DEVACC##(T2) ;AT
LDB T1,ACYFSN## ;FSN
MOVE T1,TABSTR##(T1) ;STR DATA BLOCK LOC
> ;END CONDITIONAL ON FTSTR
IFE FTSTR,<
MOVE T1,TABSTR## ;ADDT OF STR DATE BLOCK
>
MOVE T2,STRNAM##(T1) ;STR NAME
PUSHJ P,PRNAME## ;TYPE IT
PUSHJ P,INLMES##
ASCIZ /]
/
PUSHJ P,TTYSTR## ;START TTY TYPING (LEAVE JOB RUNNING, IN USER MODE)
CHKQT3: POP P,F ;RESTORE ACS
POP P,J
POP P,U
POP P,S
CHKQT4: HRRZ T2,(P)
PJRST TPOPJ## ;RETURN TO CALLER
> ;END CONDITIONAL ON FTDQTA
;SUBROUTINE TO STOP JOB ON DISK FULL OR QUOTA EXHAUSTED
;TYPES "?DISK FULL OR QUOTA EXHAUSTED FOR XXX "
;CONTINUE RETRIES
DSKFUL::MOVEM S,DEVIOS(F) ;SAVE S FROM SCNSER
PUSH P,F ;SAVE F
PUSHJ P,TTYFUW## ;FIND USERS TTY
PUSHJ P,TSETBI## ;CLEAR TYPE-AHEAD
PUSHJ P,PRQM## ;"?"
PUSHJ P,INLMES## ;AND THE MESSAGE....
IFN FTDQTA,<
ASCIZ /QUOTA OR STORAGE EXHAUSTED ON /
>
IFE FTDQTA,<
ASCIZ /STORAGE EXHAUSTED ON /
>
MOVE T1,(P) ;GET F
MOVE T1,DEVUNI##(T1) ;UNIT
MOVE T1,UNISTR##(T1) ;STR NAME
MOVE T2,STRNAM##(T1) ;TELL USER THE NAME
PUSHJ P,PRNAME## ;START TYPING
PUSHJ P,HOLD0##
POP P,F ;RESTORE F
MOVE S,DEVIOS(F) ;RESTORE S
PJRST WSCHED## ;AND RESCHEDULE
;SUBROUTINE TO SEE IF THE CURRENT POINTER CAN BE UPDATED
;ENTER WITH T2=NUMBER OF BLOCKS DESIRED
;DEVRET(F) POINTING TO CURRENT RETRIEVAL POINTER
;EXIT WITH T2= LESSER OF ORIGINAL VALUE, AMOUNT LEFT IN CURRENT POINTER
;T1= ORIGINAL T2 IF T2=0 THE CURRENT POINTER IS FULL
CHKADD::
IFN FTDSIM,<
MOVE T1,DEVACC##(F) ;IF A SIM UPDATE FILE WITH MORE THAN 1 WRITER,
LDB T3,ACYWCT## ; IF A PNTR IS ADDED TO AND THE ADDING DDB CLOSES FIRST,
SOJG T3,CHKAD0 ; THE NEW PNTR WILL BE OVERWRITTEN, SO
; DONT ALLOW ADDING TO PNTRS IF GTR THAN 1 WRITER
>
MOVEM T2,T1 ;DESIRED AMOUNT
MOVE T2,@DEVRET##(F) ;CURRENT POINTER
MOVE T4,UNISTR##(U) ;STR ADR
LDB T3,STYCNP##(T4) ;NO OF CLUSTERS IN CURRENT POINTER
LDB T2,STYCLP##(T4) ;ADR OF 1ST CLUSTER
ADD T2,T3 ;HIGHEST ADR(+1) IN PNTR
HRRZ T3,UNICPS##(U) ;NO OF CLUSTERS IN A SAT
IDIV T2,T3 ;LAST CLUSTER IN PNTR=LAST IN SAT?
JUMPE T3,CHKAD1 ;YES, CANT ADD TO CURRENT PNTR
;(ELSE PROBLEMS IN DELETING BLOCKS)
SETO T2, ;NO
MOVE T3,UNISTR##(U) ;STRUCTURE DB LOC
LDB T4,STYCNP##(T3) ;LARGEST LLOWED FIELD
MOVE T2,@DEVRET##(F) ;CURRENT POINTER
LDB T2,STYCNP##(T3) ;CURRENT GROUP SIZE
SUBM T4,T2 ;MAX ADDITIONAL CLUSTERS
CAMLE T2,T1 ;SKIP IF REQUESTED TOO MUCH
MOVE T2,T1 ;CAN HAVE ALL WE REQUESTED
POPJ P, ;RETURN
;HERE IF BLOCK IS FIRST IN NEW SAT
IFN FTDSIM,<
CHKAD0: MOVE T1,T2 ;ORIGINAL T2 INTO T1
>
CHKAD1: SETZ T2, ;RETURN CANT ADD
POPJ P,
IFN FTDHIA&FTATTACH,<
ATTCOD==0
DETCOD==1
XCHCOD==2
;HERE TO SWITCH TWO UNITS BUT LEAVE THE DATA BASE ALONE -
;EG TO PUT DSKB2 ON DPA5 WHEN DPA2 GOES DOWN, LEAVE SATS, ETC THE SAME
XCHDSK::PUSHJ P,SAVE3## ;SAVE SOME ACS
PUSHJ P,COMUNI ;SET UP U FOR FIRST UNIT
POPJ P, ;NO UNIT OR LOGICAL-UNIT MATCH
MOVE P1,U ;SAVE FIRST UNIT
LDB P3,UNYPUN## ;GET ITS NUMBER
POP P,U ;RESTORE U FOR COMCON
PUSHJ P,COMUNI ;GET SECOND UNIT
POPJ P, ;NONE OR LOGICAL MATCH
LDB P2,UNYPUN## ;NUMBER OF SECOND UNIT
HRRZ T1,UNIKON##(U) ;KONTROLLER
HRRZ T2,UNIKON##(P1)
CAIE T1,(T2) ;UNITS ON SAME KONTROLLER?
JRST UPOPJ## ;NO, CANT EXCHANGE THEM
MOVE T1,UNIBPU##(U) ;IF UNIBPU DOESNT MATCH,
CAME T1,UNIBPU##(P1)
JRST UPOPJ## ; THEN DIFFERENT TYPE UNITS, CANT EXCHANGE THEM
DSKOFF ;CANT ALLOW DISK INTERRUPTS WHILE FIDDLING
SKIPE T1,UNISTS##(U) ;UNIT IDLE
CAIL T1,OWCOD## ; OR IN SOME OPR WAIT STATE?
SKIPA T1,UNISTS##(P1) ;YES, 1ST UNIT IS OK
JRST XCHUN2 ;NO, CANT EXCHANGE
CAIGE T1,OWCOD## ;2ND UNIT IDLE OR IN OPR WAIT?
JUMPN T1,XCHUN2 ;CANT EXCHANGE IF NOT
MOVEI T1,O2COD## ;IF UNITS ARENT IN OPR WAK
SKIPN UNISTS##(U) ; PUT THEM THERE, ELSE A
MOVEM T1,UNISTS##(U) ; WRONG PACK COULD BE WRITTEN
SKIPN UNISTS##(P1)
MOVEM T1,UNISTS##(P1)
DPB P3,UNYPUN## ;OK - EXCHANGE THE UNITS
MOVEM U,@KONPTR##(T2) ;MAKE INTERRUPTS FOR 1 UNIT POINT
EXCH P1,U ; AT THE OTHER UDB,
DPB P2,UNYPUN## ;MAKE THE UDB POINT AT DIFFERENT
EXCH P2,P3 ; PHYSICAL UNITS,
MOVEM U,@KONPTR##(T2)
MOVE T1,UNINAM##(U) ;CHANGE PHYSICAL NAMES IN THE UDBS
EXCH T1,UNINAM##(P1)
MOVEM T1,UNINAM##(U)
IFN FTDUAL,<
MOVE T1,UNI2ND##(U) ;GET ALTERNATE UNITS
MOVE T2,UNI2ND##(P1)
MOVEM T2,UNI2ND##(U) ;EXCHANGE THEM
MOVEM T1,UNI2ND##(P1)
HRRM U,UNI2ND##(T2) ;EXCHANGE THE BACKWARDS POINTERS
HRRM P1,UNI2ND##(T1)
>
;STILL IN FTDHIA&FTATTACH CONDITIONAL
MOVEI T1,UNIHCT##(U) ;MAKE THE ERROR STATE
MOVEI T2,UNIHCT##(P1) ; STAY WITH THE DRIVE
HRLI T1,-3 ; EXCHANGE UNIHCT,SCT,MCT
XCHUN1: MOVE T3,(T1)
EXCH T3,(T2)
MOVEM T3,(T1)
ADDI T2,1
AOBJN T1,XCHUN1
AOS -1(P) ;INDICATE WE WON
XCHUN2: DSKON ;ALLOW DISK INTERRUPTS AGAIN
MOVE F,P1 ;GET 1ST UDB INTO F
HRLZI T1,XCHCOD ;CODE FOR XCH
PUSHJ P,DSKCSC ;CALL DAEMON
JRST UPOPJ## ;RESTORE U AND RETURN TO COMCON
;HERE TO TAKE A UNIT OFF-LINE
;RETURNS CPOPJ IF NOT A DSK
;RETURNS CPOPJ1 IF CANT DETACH
;RETURNS CPOPJ2 IF OK
DETDSK::PUSHJ P,COMUNT ;SET UP U (COMCON ALREADY HAS U PUSHED)
POPJ P, ;NO UNIT OR LOGICAL MATCH
IFN FTDUAL,<
JUMPL U,DETDS2 ;EASY IF AN ALTERNATE UNIT
>
HRRZ T1,UNISTR##(U) ;UNIT IN A STR?
JUMPN T1,UPOPJ1## ;CANT DETACH IF IT IS
IFN FTDUAL,<
SKIPG T1,UNI2ND##(U) ;DOES THIS HAVE AN ALTERNATE PATH?
JRST DETDS2 ;NO
HRROS UNI2ND##(U)
HRRZS UNI2ND##(T1) ;YES, SWITCH MAIN AND ALTERNATE UNITS
MOVE T2,UNISYS##(U)
HLLM T2,UNISYS##(T1) ;UNLINK MAIN UNIT, INSERT ALTERNATE
HRLM T1,UNISYS##(T4)
MOVEI T2,O2COD##
MOVEM T2,UNISTS##(U) ;SET SO THIS UNIT WONT BE USED
DETDS2:>
MOVEI T1,UNVDWN## ;OK, INDICATE UNIT IS DOWN
DPB T1,UNYUST##
HRLZI T1,DETCOD ;CODE TO SAY DETACH
PUSHJ P,DSKCSC ;CALL DAEMON
POP P,U
PJRST CPOPJ2## ;AND RETURN
;STILL IN FTDHIA&FTATTACH CONDITIONAL
;HERE TO ATTACH A UNIT
;RETURNS NON-SKIP IF UNIT IS DOWN
;CPOPJ1 IF WE CANT CALL CPY ROUTINE NOW (TRY LATER)
;CPOPJ2 IF ALL IS OK
ATTDSK::PUSHJ P,COMUNT ;SET UP U
JRST [TLO U,400000 ;NO MATCH
POPJ P,]
LDB T1,UNYUST## ;GET UNIT STATUS
IFN FTDUAL,<
SKIPLE UNI2ND##(U) ;IF IT HAS A 2ND PORT
JUMPE T1,ATTUN0 ; 0 IS OK (JUST DETACHED 1ST PORT)
>
CAIE T1,UNVDWN## ;DOWN?
JRST UPOPJ## ;NO, CANT ATTACH IT
ATTUN0: PUSH P,J ;YES, SAVE J FOR COMCON
MOVE J,UNIKON##(U) ;KONTROLLER DATA BLOCK
MOVSI T1,KOPDWN## ;CLEAR KONTROL-IS-DOWN BIT
ANDCAM T1,KONDWN##(J)
SKIPGE KONCPY##(J) ;CAN WE CALL CPY ROUTINE IF KONTROL BUSY?
SKIPL KONTAB##(J) ;NO, IS KONTROLLER BUSY?
TLZA J,-1 ;WE CAN TELL UNIT TYPE NOW
JRST ATTUN4 ;CANT GET KONTROL - TRY LATER
PUSHJ P,@KONCPY##(J) ;DETERMINE UNIT TYPE, CAPACITY
JRST ATTUN5 ;UNIT DOWN
MOVEM T1,UNIBPU##(U) ;SAVE BLOCKS PER UNIT
MOVEM T2,UNIBPM##(U) ;SAVE BLKS PER UNIT INCL. MAINT CYLS
IFN FTRP04,<
MOVEM T3,UNIBUC##(U) ;SAVE BLOCKS PER UNIT IN 10/11 COMPAT. MODE
>
DPB W,UNYBPY## ;SAVE # BLOCKS PER CYLINDER
HLRZ T3,W ;NO OF BLOCKS PER TRACK
DPB T3,UNYBPT## ;BLOCKS/TRACK
DPB T4,UNYUTP## ;UNIT TYPE
IFN FT22BIT,<
MOVSI T1,CP.22B##
HRRZ T3,UNICHN##(U)
TLNE T4,KOP22B##
JRST ATTUN1 ;22-BIT CHAN?
MOVSI T2,1 ;NO, IF MORE THAN 256K,
CAMGE T2,MEMSIZ##
JRST ATTUN5 ; CAN'T USE THE UNIT
ANDCAM T1,CHB22B##(T3) ;OK - CLEAR THE BIT IN CDB
JRST ATTUN2
ATTUN1: IORM T1,CHB22B##(T3) ;22-BIT CHAN - SET THE BIT
>
;STILL IN FTDHIA&FTATTACH CONDITIONAL
ATTUN2: AOS -2(P) ;CPOPJ2 IS GOODNESS
MOVEI T1,UNVNPM## ;INDICATE NO PACK MOUNTED
DPB T1,UNYUST## ; (EG, UNIT IS UP)
MOVSI T1,UNPOFL##
ANDCAM T1,UNIUST##(U)
SETZM UNISTS##(U)
IFN FTDUAL,<
JUMPGE U,ATTUN3 ;GO IF NOT AN ALTERNATE PATH
SETZM UNISTS##(U) ;ALTERNATE - MARK IT USABLE
JRST ATTUN4 ;CALL DAEMON AND EXIT
ATTUN3: HLRZ T1,J ;GET SERIAL NUMBER
HLL T1,UNISER##(U) ;DRIVE TYPE
PUSHJ P,MATUN ;SEARCH FOR A MATCH
JRST ATTUN4 ;NO MATCH
HRROM T2,UNI2ND##(U) ;MATCH, MAKE THIS AN ALTERNATE
HRRZM U,UNI2ND##(T2) ;POINT MAIN UNIT AT THIS ONE
MOVE T1,UNISTR##(T2)
HRRM T1,UNISTR##(U)
PUSHJ P,UNLUN ;UNLINK UNIT FROM SYSUNI CHAIN
>
ATTUN4: POP P,J ;RESTORE J
HRLZI T1,ATTCOD ;CODE TO SAY ATTACH
PUSHJ P,DSKCSC ;CALL DAEMON
JRST UPOPJ1## ;RESTORE U AND RETURN
ATTUN5: MOVEI T1,UNVDWN## ;UNIT IS STILL DOWN - SET THE BYTE
DPB T1,UNYUST## ;IN UNISTS AGAIN
SOS -2(P) ;NON-SKIP RETURN
IFN FTDUAL,<
JRST ATTUN3
>
IFE FTDUAL,<
JRST ATTUN4
>
;ROUTINE TO SET UP U FOR THE COMMAND
;RETURNS WITH ORIGINAL U ON PD LIST
COMUNI: PUSHJ P,SETLGL## ;PRIV'D JOB?
POPJ P, ;NO, ERROR RETURN
PUSHJ P,CTXDEV## ;YES, GET DEVICE TYPED BY USER
MOVE T1,T2 ;DEVICE INTO T1 FOR DEVSRC
COMUNT: EXCH U,(P) ;SAVE U ON LIST
PUSH P,U
SETO T2, ;NEED A COMPLETE MATCH ON UNIT NAME
PUSHJ P,SRUNI## ;FIND MATCHING UNIT
JFCL ;NO SUCH UNIT
CAIA ;LOGICAL MATCH
JRST CPOPJ1## ;PHYSICAL MATCH - GOOD RETURN
IFN FTDUAL,<
HLRZ U,SYSUNI## ;START AT 1ST UNIT
COMUN2: SKIPLE T2,UNI2ND##(U) ;IS THERE AN ALTERNATE TO UNIT?
CAME T1,UNINAM##(T2);YES, IS IT OUR UNIT?
JRST COMUN3 ;NO
HRROI U,(T2) ;YES, SET LH(U)=-1 AS A FLAG
JRST CPOPJ1## ;AND TAKE FOUND-RETURN
COMUN3: HLRZ U,UNISYS##(U) ;NO MATCH, TRY NEXT
JUMPN U,COMUN2
>
MOVE U,(P) ;RESTORE U
EXCH U,-1(P)
JRST T2POPJ## ;TAKE GARBAGE OFF STACK AND BADNESS-RETURN
;ROUTINE TO PICK UP OPERATOR COMMENTS AND CALL
;DAEMON FOR DISK CONFIGURATION STATUS CHANGE
;CALL WITH CODE FOR ATT, DET, XCH IN T1, F + U SETUP
;CALLED WITH U FOR COMCON ON -1 OF STACK
DSKCSC: EXCH U,-1(P) ;GET LDB FOR COMCON
PUSH P,T1
PUSHJ P,SKIPS1## ;SKIP OVER TO OPR COMMENTS
TDZA T2,T2 ;NONE THERE
PUSHJ P,CTEXT## ;PICK THEM UP
POP P,T1
EXCH U,-1(P) ;GET BACK UDB
ROT T2,^D12 ;GET FIRST TWO CHARS OR RIGHT
DPB T2,[POINT 12,T1,11] ;AND PUT THEM INTO T1
HRRI T1,.ERCSC ;GET CODE FOR DAEMON
PUSH P,J ;SAVE J
PUSHJ P,DAEEIM## ;WAKE DAEMON
JRST JPOPJ## ;RETURN
> ;END FTDHIA&FTATTACH
IFN FTDUAL,<
;ROUTINE TO TEST IF MATCHING SERIAL-NUMBERS (EG DUAL-PORTED DRIVES) EXIST
;CALL WITH T1= DRIVE SERIAL NUNBER, U=UNIT
;RETURNS CPOPJ IF NO MATCH
;RETURNS CPOPJ1 IF A MATCH, T2= MATCHING UNIT
;PRESERVES T4
MATUN:: JUMPE T1,CPOPJ## ;NO MATCH IF NO SERIAL NUMBER
HLRZ T2,SYSUNI## ;START AT FIRST UNIT IN SYSTEM
MATUN1: CAMN T1,UNISER##(T2) ;MATCH?
CAIN T2,(U) ;YES, FOR A DIFFERENT UNIT?
JRST MATUN2 ;NO MATCH
MOVE T3,UNIPUN##(U) ;MAKE SURE DRIVE NUMBERS MATCH
XOR T3,UNIPUN##(T2)
TRNN T3,7
JRST CPOPJ1## ;MATCH-SKIP RETURN
MATUN2: HLRZ T2,UNISYS##(T2) ;STEP TO NEXT UNIT
JUMPN T2,MATUN1 ;AND TEST IT
POPJ P, ;NO MATCH, NON-SKIP
;SUBROUTINE TO UNLINK A UNIT FROM UNISYS CHAIN
;ENTER WITH U=UNIT
;PRESERVES T4
UNLUN:: MOVEI T1,DIFUSY## ;PRESET PREDECESSOR
UNLUN1: HLRZ T2,UNISYS##(T1) ;START AT 1ST UNIT
JUMPE T2,CPOPJ## ;SYSTEM ERROR?
CAIN T2,(U) ;MATCH?
JRST UNLUN2 ;YES, UNLINK IT
MOVE T1,T2 ;NO, RESET PREDECESSOR
JRST UNLUN1 ;AND TRY NEXT UNIT
UNLUN2: MOVE T3,UNISYS##(U) ;GET THIS UNIT'S LINK
HLLM T3,UNISYS##(T1) ;AND SAVE IT IN PREDECESSOR'S
POPJ P, ;AND RETURN
> ;END FTDUAL
SUBTTL USETI/USETO
USETI0::
HRRE W,M ;BLOCK NUMBER
CAME W,[-1] ;BLOCK -1 MEANS END OF FILE, IS IT?
CAML W,MUSTMX## ;TRYING TO READ AN EXTENDED RIB?
CAIA ;YES
TLZ W,-1 ;NO, CLEAR HIGH BITS FOR COMPATIBILITY.
FUSETI::
IFN FTSPL,<
SKIPGE DEVSPL(F) ;IF THIS IS A SPOOLED DDB,
POPJ P, ;USETI IS A NO-OP
>
PUSHJ P,WAIT1## ;MAKE SURE ALL I/O IS DONE
TLNN F,LOOKB ;LOOKUP DONE?
JRST SETSUP ;NO. SUPER USETI IF PRIVILEGED
HRRZ U,DEVUNI(F)
JUMPE U,CPOPJ## ;GO AWAY IF UNIT WAS REMOVED
PUSHJ P,CLSNAM## ;SET RIGHT NAME IN DDB FOR RIBCHK
; (FILE MIGHT BE RENAMED)
HRRZ T1,DEVACC##(F) ;YES. LOC OF ACCESS TABLE
PUSHJ P,SAVE1## ;SAVE P1
IFN FTDMRB,< ;IF MULTIPLE RIBS
MOVE P1,W ;GET USETI ARGUMENT TO P1
CAMGE P1,MUSTMX## ;SKIP IF RH(M) POSSIBLE EXTEDNED RIB
JRST USETI2 ;NOT LOOKING FOR EXTENDED RIBS
AOJGE P1,USETI2 ;IF -1 OR POSITIVE, NOT EXTENDED
HRRZ U,DEVUNI##(F) ;GET CURRENT UNIT
PUSHJ P,PTRTST ;READ POINTERS, RE-WRITE IF CHANGED
PJRST GVMNB0 ;ERROR READING RIB
SKIPL DEVRIB##(F) ;PRIME RIB?
JRST USETI1 ;YES, GET EXTENDED
PUSHJ P,REDRIB ;NO, READ PRIME RIB
PJRST GVMNB0 ;ERROR READING RIB
USETI1: PUSHJ P,PTRNXT ;GET EXTENDED RIB
JRST USET1B ;EITHER RIB ERROR OR NONE
AOJN P1,USETI1 ;JUMP BACK IF NOT THIS RIB
HRRM T2,DEVUNI(F) ;(NEW) UNIT
PUSHJ P,PTRBLT ;GET POINTERS TO DDB
MOVE T1,DEVMBF##(F) ;IOWD TO MONITOR BUFFER
MOVE T2,RIBFLR##+1(T1) ;GET RELATIVE BLOCK NUMBER OF RIB
MOVE T3,T2 ;ALSO GET TOT3 FOR SCNPTR
MOVEI T1,DEVRB1##(F) ;ADDRESS OF IN CORE POINTERS
HRLI T1,MPTRLN## ;MAKE AOBJN WORD
PUSHJ P,SCNPTR ;SET UP DEVBLK, DEVREL, DEVLFT
PJRST GVMNB0 ;THIS IS SO "OUTTA SIGHT" IT SHOULD HALT
MOVNS DEVREL##(F) ;FLAG SO OUTPUT NEXT IS ILLEGAL
PUSHJ P,GVMNB0 ;GIVE UP MONITOR BUFFER
TLZ S,IOSFIR ;RIBS AREN'T CHECKSUMMED
JRST USETI4 ;AND EXIT.
USET1B: SKIPN T3 ;ERROR READING RIB?
TRO S,IOBKTL ;NO, NON-EXISTANT, SET IOBKTL
PJRST GVMNB0 ;GIVE UP MONITOR BUFFER AND EXIT
> ;END CONDITIONAL ON FTDMRB
USETI2: MOVE P1,ACCWRT##(T1) ;HIGHEST RELATIVE BLOCK WITH DATA IN THE FILE
JUMPL W,USETI5 ;OK UNLESS USETI -1
CAML P1,W ;ASKING FOR A BLOCK PAST EOF?
TLOA P1,400000 ;NO, SET P1 NEGATIVE AS A SWITCH
USETI5: MOVE W,P1 ;YES, INITIALLY SCAN FOR LAST BLOCK
PUSHJ P,USET00 ;FIND THE PNTR TO THE BLOCK
POPJ P, ;RIB ERROR
SKIPL DEVBLK##(F)
JRST USETI3
MOVEM W,DEVREL##(F)
PUSHJ P,FNDPTR
POPJ P,
USETI3: JUMPL P1,USETI4 ;GO IF NOT PAST EOF
AOS DEVBLK##(F) ;PAST EOF - UPDATE PNTRS IN DDB
AOS DEVREL##(F) ;SO NEXT INPUT/OUTPUT WILL GET LAST BLOCK
SOS DEVLFT##(F) ; OF FILE PLUS 1
TLZ S,IOSFIR
TDOA S,[XWD IOEND,IODEND] ;INDICATE USETI PAST EOF
USETI4: TDZ S,[XWD IOEND,IODEND] ;OK - ZERO EOF BITS
IFN FTDMRB,<
PUSHJ P,EXTCKS ;LIGHT IOSFIR IF 1ST BLOCK IN EXT. RIB
>
PJRST STOIOS## ;STORE S AND TAKE GOOD RETURN
USETO0::
HRRE W,M ;BLOCK NUMBER
CAME W,[-1] ;IF NOT USETO TO LAST BLOCK OF THE FILE
TLZ W,-1 ; CLEAR HIGH ORDER BITS
FUSETO::
IFN FTSPL,<
SKIPGE DEVSPL(F) ;IF THIS IS A SPOOLED DDB,
POPJ P, ; USETO IS A NOOP
>
PUSHJ P,WAIT1## ;WAIT FOR I/O TO FINISH
TLNN F,ENTRB ;ENTER BEEN DONE?
JRST SETSUP ;NO. (FIRST) SUPER USETO IF LEGAL
MOVE T1,W ;YES, ARGUMENT
AOJN T1,USETO7 ;USETO -1 MEANS LAST BLOCK XFERRED
MOVE T1,DEVREL##(F) ;IS THERE A LAST BLOCK?
SOJG T1,USET0A ;YES, DO A USETO TO IT
MOVE T1,DEVACC##(F) ;NO, UPDATE FILE?
HRLZ T1,ACCSTS##(T1)
TLNE T1,ACPUPD
HLRZ T1,DEVLRL##(F) ;YES. GET DEVREL BEFORE ENTER
USET0A: MOVE W,T1 ;DO USETO TO THAT BLOCK
USETO7:
IFN FTDMRB,<
MOVE T1,DEVACC##(F) ;LOC OF A.T
MOVE T1,ACCWRT##(T1) ;HIGHEST WRITTEN BLOCK
CAML T1,W ;TRY TO SETO PAST HIGHEST.
JRST USETO6 ;NO, OK
PUSH P,W ;YES. FIRST FIND HIGHEST
MOVE W,T1 ;SO THAT LAST RIB WILL BE
PUSHJ P,USET00 ;READ AND DEYRBC SET RIGHT
PJRST TPOPJ## ;RIB ERROR
POP P,W ;RESTORE W
>
PUSHJ P,GETALC ;GET ADJUSTED ACCALC
CAMG T1,W ;WANT ONE BELOW HIGHEST?
JRST USETO1 ;NO. HAVE TO ALLOCATE
USETO6: PUSHJ P,USET00 ;YES. SET UP CORRECT POINTERS
POPJ P, ;RIB ERROR
IFN FTDMRB,< ;IF MULTIPLE RIBS
AOSE DEVBLK(F) ;IF DEVBLK=-1, CAN'T FIND BLOCK
JRST USETO8 ;USETO TO ALLOCATED BLOCKS
IFN FTDSIM,<
PUSHJ P,GETALC ;SINCE ANOTHER JOB MAY HAVE ALLOCATED,
CAMG T1,W ; AND ACCWRT ISN'T YET TO ITS FINAL VALUE,
JRST USETO1 ; WE MUST REPEAT THE TEST (DEVRIB NOW POINTS TO LAST RIB)
>
;HERE IF DOING A USETO TO LAST BLOCK IN RIB
PUSHJ P,USETO9 ;ZERO ALLOCATED, UNWRITTEN BLOCKS
SUBI W,1 ;POINT W TO LAST "REAL" BLOCK
PUSHJ P,USET00 ;GET POINTERS INTO CORE (SHOULD ALSO GET HIGHEST)
POPJ P, ;RIB ERROR
MOVE T3,DEVLPC##(F) ;GET LAST POINTER IN CORE FLAG
TLZN T3,DEPLPC## ;CLEAR IT TO FOOL SCNPTR
STOPCD CPOPJ##,DEBUG,DBZ, ;++DEPLPC BIT ZERO
MOVEM T3,DEVLPC##(F) ;RETURN TO DDB
MOVE T3,W ;GET LAST BLOCK ALLOCATED
ADDI T3,1
MOVE T2,DEVFLR##(F) ;INITIAL BLOCK IN DDB
MOVEI T1,DEVRB1##(F) ;SCAN POINTERS IN DDB
HRLI T1,MPTRLN## ;STARTING AT DEVRB1
PUSHJ P,SCNPTR ;FIND THE POINTER
STOPCD .+1,DEBUG,HIF, ;++HOLE IN FILE
HRRZM T1,DEVRET##(F) ;SET DEVRET TO LAST POINTER
HRROS DEVRSU##(F) ;SET DEVRSU TO -1
HLLZS DEVLFT##(F) ;CLEAR DEVLFT SO NXTBLK WILL NOT FIND
HRLZI T1,DEPLPC## ;LIGHT DEPLPC AGAIN
IORM T1,DEVLPC##(F) ;IN DDB
POPJ P, ;AND EXIT
USETO8: SOS DEVBLK##(F) ;RETURN DEVBLK TO PROPER VALUE
PUSHJ P,EXTCKS ;SET IOSFIR IF FIRST BLOCK IN EXTENDED RIB
> ;END FTDMRB CONDITIONAL
MOVEM S,DEVIOS(F) ;SAVE S
;CHECK TO SEE IF A USETO IS SETTING OUTOUT PAST THE LAST BLOCK WRITTEN
; IF SO, WRITE 0'S IN THE INTERVENING BLOCKS
USETO9: SKIPN T1,W ;SETTING BLOCK 0?
POPJ P, ;YES. THIS IS NON-ALLOCATING
SUBI T1,1 ;LAST BLOCK TO ZERO
HRRZ T2,DEVACC##(F) ;LOC OF A.T.
CAMG T1,ACCWRT##(T2) ;PAST HIGHEST BLOCK WRITTEN?
POPJ P, ;NO, OK
PUSHJ P,SAVE2## ;YES, SAVE SOME ACS
HRRZ P1,DEVACC##(F) ;LOC OF A.T.
MOVE P2,T1 ;HIGHEST BLOCK TO ZERO
USETZ1: MOVE T1,ACCWRT##(P1) ;BLOCK-1 TO ZERO
AOS W,T1 ;BLOCK WE WANT TO ZERO
PUSHJ P,USET00 ;SET DEVBLK FOR THIS BLOCK
POPJ P, ;RIB ERROR
SKIPGE DEVBLK##(F) ;SEMI-GOOD RETURN!
POPJ P, ;SHOUD NEVER HAPPEN, BUT....
PUSHJ P,GTMNBF ;GET MON BUF
MOVSI T2,1(T1) ;1ST WORD IN MON BUF
HRRI T2,2(T1) ;SET TO ZERO MON BUF
SETZM 1(T1)
BLT T2,BLKSIZ##(T1) ;ZERO THE BUFFER
MOVE T2,DEVBLK##(F) ;BLOCK TO WRITE
IFN FTDSIM,<
CAMG W,ACCWRT##(P1) ;SOMEBODY JUST WRITE THE BLOCK?
JRST [PUSHJ P,GVMNB0 ;YES, WE'RE DONE
JRST USETZ2]
MOVSI T3,DEPUWZ## ;INDICATE USETO WRITING ZEROES
IORM T3,DEVUWZ##(F) ; IN CASE ANOTHER DDB IS CURRENTLY
; WRITING THIS BLOCK, ACCWRT NOT YET UPDATED
>
PUSHJ P,MONWRT ;WRITE A BLOCK OF 0'S
IFN FTDSIM,<
MOVSI T3,DEPUWZ## ;USETO NOT NOW WRITING ZEROES
ANDCAM T3,DEVUWZ##(F)
MOVE T1,DEVREL##(F) ;IF THIS WRITE DIDN'T HAPPEN (REAL WRITER SNUCK IN)
CAMLE T1,ACCWRT##(P1) ; THEN DON'T CHANGE ACCWRT
>
AOS ACCWRT##(P1) ;BUMP NUMBER OF BLOCKS WRITTEN
PUSHJ P,GVMNB0 ;RETURN MON BUF (DONT WANT TO MONOPOLIZE IT)
LDB J,PJOBN## ;JOB NUMBER
MOVE T1,JBTSTS##(J) ;JBTSTS
TLNN T1,CNTRLC ;JOB TYPED ^C?
JRST USETZ2 ;NO
PUSH P,F
PUSHJ P,STOP1## ;YES, RETURN TO MONITOR MODE
POP P,F ;RESTORE ACS WIPED BY STOP,
MOVE S,DEVIOS(F)
PUSHJ P,WSCHED## ;STOP JOB
;CONTINUE TYPED
;FALL INTO USETZ2
USETZ2: CAMLE P2,ACCWRT##(P1) ;HAVE WE FINISHED YET?
JRST USETZ1 ;NO, WRITE NEXT BLOCK
AOS W,P2 ;YES, SET M FOR BLOCK WE ORIGINALLY WANTED
PUSHJ P,USET00 ;SET DDB POINTERS
POPJ P, ;RIB ERROR
MOVEM S,DEVIOS(F) ;SAVE S (NEW IOSFIR)
MOVE T2,DEVACC(F) ;GET LOC OF A.T.
MOVEI T1,BLKSIZ## ;GET SIZE OF BLOCK
DPB T1,ACYLBS## ;FORCE AS FINAL BLOCK'S WORD COUNT
POPJ P, ;THROUGH - RETURN
;SUBROUTINE TO OBTAIN THE HIGHEST BLOCK ALLOCATED
;RETURNS WITH NUMBER IN T1
GETALC::MOVE T1,DEVACC##(F) ;LOC OF A.T
MOVE T1,ACCALC##(T1) ;ACCALC
IFN FTDMRB,<
LDB T2,DEYRBC## ;CURRENT RIB NUMBER
LSH T2,1 ;2 NON-DATA BLOCKS PER RIB
SUB T1,T2 ;ADJUST ACCALC
>
POPJ P, ;AND RETURN
UDSD==100 ;USER WANTS TO WRITE FORMAT
UDSX==200 ;FILIO WANTS TO WRITE FORMATS
MNTCYL==100000 ;ON IF USER WANTS MAINTENANCE CYLS
;HERE ON SUSET. UUO
USUSET::
IFN FTDSUP,<
MOVE W,M ;SAVE AC BYTE
MOVE M,T1 ;SAVE ARGUMENT
PUSHJ P,VALUUO ;DSK INITED ON THIS CHAN?
PJRST IOIERR## ;"IO TO UNASSIGNED CHAN"
TLO M,400000 ;YES, INDICATE SUSET.
AOS (P) ;SET FOR SKIP (GOOD) RETURN
JRST SETSU0 ;AND DO SUPER USETI/O
>
SETSUP:
IFN FTDSUP,< ;IF SUPER USETI/USETO
JFCL ILLINS## ;CHANGE TO JRST TO MAKE SUPER USETI/O ILLEGAL
SETSU0: HRRZ U,TABST0## ;LOC OF 1ST FS
HLRZ U,STRUNI##(U) ;U=LOC OF 1ST UNIT IN 1ST STR
MOVE T1,DEVNAM(F) ;NAME USER INITED
PUSHJ P,ALIASD## ;IS NAME AN ALIAS FOR "DSK"?
JRST SETIMP## ;YES, GIVE THE USER IO.IMP
PUSHJ P,SRSTR## ;NO. AN STR NAME?
SKIPA ;NO
JRST SETSU1 ;YES.
PUSHJ P,SRUNI## ;A UNIT NAME?
POPJ P, ;NO - RETURN WITHOUT DOING ANYTHING
JFCL
PUSHJ P,PRVJB## ;YES. PRIVILEGED?
JRST SETSU7 ;NO. ILLEGAL
SETSUX: SKIPL T1,M ;BLOCK NOT IN M IF SUSET.
PUSHJ P,GETWDU## ;UNIT NAME - GET BLOCK NUMBER
TLZ T1,777740
HRRM U,DEVUNI##(F) ;SAVE UNIT IN DDB
SETOM DEVREL##(F) ;INDICATE UNIT WAS INITED(NOT STR)
TRNN S,UDSD ;WRITE FORMAT?
TRZA S,UDSX ;NO, CLEAR THE BIT
TRO S,UDSX ;YES, INDICATE WRITING FORMATS
JRST SETSU2 ;AND CONTINUE
SETSU1: TRZ S,UDSX ;INDICATE NOT WRITING FORMATS
PUSHJ P,PRVJB## ;PRIV'D
JRST SETS3A ;NO. ILLEGAL
JUMPG M,SETS1A ;GO IF SUPER USET
MOVE T1,M ;GET BLOCK NO
TLZ T1,777740 ;CLEAR UNWANTED BITS
CAMN T1,[37,,-1] ;SUSET. TO LAST BLOCK XFERRED?
SETO T1, ;YES, MAKE IT -1
JRST SETS1B
SETS1A: PUSHJ P,GETWDU## ;GET BLOCK NUMBER
SETS1B: CAME T1,[-1] ;USETO TO LAST BLOCK XFERRED?
JRST SETS1C ;NO
HRRZ U,DEVUNI##(F) ;YES, GET RIGHT UNIT
SOS T1,DEVBLK##(F) ;GET BLOCK NUMBER
JRST SETSU4 ;AND CONTINUE
SETS1C: PUSHJ P,ADR2UN## ;SET U TO RIGHT UNIT IN STR FOR THIS BLOCK
JRST SETSU3 ;ILLEGAL BLOCK NUMBER - LIGHT IOBKTL
SETSU2: CAML T1,UNIBPU##(U) ;HIGHER THAN HIGHEST BLOCK ON UNIT?
JRST SETSU6 ;YES. LIGHT IOBKTL
JUMPL T1,SETSU3
TLNE M,MNTCYL ;WANT MAINT CYL?
JRST SETSU3 ;YES, ERROR (THIS BLOCK NOT IN MAINT CYLS)
MOVEM T1,DEVBLK##(F) ;NO, SAVE BLOCK NO IN DDB
SETSU4: SUB T1,UNIBPU##(U) ;-DISTANCE TO END OF UNIT
SETSU5: MOVNS T1 ;NO OF BLOCKS LEFT TO END OF UNIT
TLNE T1,-1 ;MORE THAN 256 K BLOCKS?
MOVEI T1,-1 ;YES, MAX TRANSFER IS 256K
HRRM T1,DEVLFT##(F) ;SAVE IN DEVLFT
TLZ S,IOSFIR ;MAKE SURE IOSFIR=0
TLOA S,IOSUPR ;INDICATE SUPER USETI/USETO
> ;END CONDITIONAL ON FTDSUP
ERRFUL:
SETSU3: TRO S,IOBKTL ;INDICATE TOO HIGH A BLOCK NUMBER
PJRST STOIOS## ;SAVE S AND RETURN
SETS3A: JUMPGE M,SETSU3 ;GO IF SUPER USETI/O
SETS3B: SOS (P) ;SUSET, - NON-SKIP RETURN
MOVE M,W ;RESTORE AC (PUUOAC)
PJRST RTM1##
;HERE IF BLOCK ABOVE HIGHEST BLOCK ON UNIT
IFN FTDSUP,<
SETSU6:
IFN FTRP04,<
MOVSI T2,DEPCPT## ;COMPATABILITY MODE?
TDNN T2,DEVCPT##(F)
JRST SETSU8 ;NO
CAMLE T1,UNIBUC##(U) ;YES, IS IT A LEGAL BLOCK?
JRST SETSU3 ;NO, LIGHT AN ERROR BIT
MOVEM T1,DEVBLK##(F) ;YES, SAVE BLOCK NUMBER
SUB T1,UNIBUC##(U) ;DISTANCE TO END OF UNIT
JRST SETSU5 ; AND FINISH UP
SETSU8:>
CAMG T1,UNIBPM##(U) ;MAINT CYL?
SKIPL DEVREL##(F) ;YES, UNIT (NOT STR) INITED?
JRST SETSU3 ;NO - IOBKTL
TLNN M,MNTCYL ;WANT MAINT CYL (OR SUPER USET)?
JRST SETSU3 ;NO, ERROR
MOVEM T1,DEVBLK##(F) ;YES, SAVE BLOCK
SUB T1,UNIBPM##(U) ;DISTANCE TO END OF MAINT CYL
JRST SETSU5 ;FINISH UP
;HERE IF UNPRIU'S SUSET/USET TO A UNIT
SETSU7: JUMPGE M,SETSU3 ;ERROR IF SUPER USET
MOVE T1,.C0JOB## ;SUSET.
MOVE T1,JBTPPN##(T1) ;PPN OF REGISTER
CAMN T1,UMDPPN## ;USER-MODE DIAGNOSTICS? [6,6]
TLNN M,MNTCYL ;TO MAINT CYL?
JRST SETS3B ;NO, ERROR
JRST SETSUX ;YES, OK
> ;END FTDSUP
;HERE IF THE REQUESTED BLOCK IS HIGHER THAN THE HIGHEST ALLOCATED
USETO1: PUSHJ P,SAVE2##
MOVE P1,W ;SAVE REQUESTED BLOCK
MOVE P2,T1 ;SAVE 1ST UNALLOCATED BLOCK NUMBER
SOS W,T1 ;SET RH(M) TO HIGHEST ALLOCATED
PUSHJ P,USET00 ;GET POINTERS INTO CORE FOR HIGHEST BLOCK
POPJ P, ;RIB ERROR
IFN FTDMRB,<
SKIPL DEVBLK(F) ;FIND THE BLOCK?
JRST USET1A ;YES
MOVSI T1,DEPLPC ;NO, IS IT REALLY THERE?
TDNN T1,DEVLPC(F)
STOPCD CPOPJ,DEBUG,PLP,;++ PAST LAST POINTER
HRROS DEVRSU(F) ;YES, SET DEVRSU TO EXTEND THE RIB
USET1A:
>
MOVE T2,P1 ;TOP BLOCK TO ALLOCATE
MOVE T1,P2 ;FIRST BLOCK TO ALLOCATE
SUB T2,T1 ;TOTAL NUMBER TO ALLOCATE
ADDI T2,1
IFN FTDQTA,<
PUSH P,T2 ;SAVE NUMBER REQUESTED
PUSHJ P,CHKQTA ;CAN WE GET THAT MANY?
JUMPLE T2,TPOPJ## ;NO. ERROR RETURN (IOBKTL SET)
CAMGE T2,(P) ;DID WE GET ALL WE ASKED FOR?
TRO S,IOBKTL ;NO, TELL THE USER HE ONLY GOT SOME
> ;END CONDITIONAL ON FTDQTA
IFE FTDQTA,<
PUSHJ P,CHKQTA ;CHECK IF STR FULL
>
MOVEM P1,DEVREL##(F) ;SAVE REQUESTED BLOCK IN DDB
IFE FTDSIM,<
MOVEM T2,P1 ;SAVE NUMBER TO GET
>
IFN FTDSIM,<
MOVEM T2,P2 ;NUMBER TO GET
>
IFN FTDQTA,<
POP P,T1 ;NUMBER REQUESTED
SUBM T2,T1 ;MINUS NUMBER ALLOWED
ADDM T1,DEVREL##(F) ;ADJUST REQUESTED BLOCK BY NO. OBTAINED
> ;END CONDITIONAL ON FTDQTA
IFN FTDSIM,<
MOVE T1,DEVACC##(F)
MOVE T1,ACCSMU##(T1) ;SIM UPDATE FILE?
TRNN T1,ACPSMU
JRST USET1C ;NO, CONTINUE
PUSHJ P,GTMB2 ;YES, GET MON BUF NOW TO AVOID DEADLY EMBRACE
PUSHJ P,UPDA ; (MUST GET MON BUF BEFORE DA, SIM UPDATE NEEDS BOTH)
PUSHJ P,GETALC ;GET CURRENT NO OF BLOCKS ALLOCATED
AOS W ;HAS IT CHANGED (ANOTHER USETO ALLOCATING)
CAMN T1,W
SOJA W,USET1C ;NO, WE'RE OK
MOVE W,P1 ;YES, EXTRICATE OURSELVES
POP P,(P)
POP P,P2
POP P,P1 ;MAKE STACK RIGHT
PUSHJ P,DWNDA ;GIVE UP RESOURCES
PUSHJ P,GVMNB0
JRST USETO7 ;AND TRY AGAIN
USET1C: MOVE P1,P2 ;RESTORE NUMBER TO GET
> ;END FTDSIM
PUSHJ P,CHKADD ;CAN WE ADD TO CURRENT POINTER?
JUMPLE T2,USETO3 ;NO. GET SPACE ANYWHERE
AOSE T1,DEVBLK##(F) ;YES SET T1= 1ST BLOCK
PUSHJ P,TAKBLK ;GET BLOCKS AT PREVIOUS END
JRST USETO3 ;CANT GET ANY THERE
PUSHJ P,ADDPTR ;GOT SOME - ADD TO CURRENT POINTER
USETO2: SUB P1,T1 ;DECREMENT AMOUNT TO GET
JUMPLE P1,USETO5 ;FINISH UP IF GOT ENOUGH
;HERE TO GET BLOCKS ANYWHERE
USETO3:
MOVSI T3,1 ;DECREMENT TOTAL NO. OF POINTERS
ADDB T3,DEVRSU##(F) ;TOO MANY?
JUMPGE T3,USET3B ;YES, TRY TO GET AN EXTENDED RIB
USET3C: SETZ T3,
AOS T1,DEVRET##(F) ;POINT DEVRET TO 1ST EMPTY POINTER LOC
CAILE T1,DEVRBN##(F) ;FILLED THE DDB?
PUSHJ P,WRTPTR ;YES. WRITE THE POINTERS
JUMPN T3,USETO4 ;RETURN WITH IOBKTL IF RIB ERR
MOVE T2,P1 ;NUMBER TO GET
MOVEI T1,0 ;ANYWHERE
PUSHJ P,TAKBLK ;GET SOME BLOCKS
SKIPA ;NOT AVAILABLE ON THIS UNIT
JRST USET3A ;GOT THEM
HLRE T1,DEVRSU##(F) ;IF 1 SLOT LEFT,
AOJGE T1,USET5A ;CANT EXTEND RIB,
PUSHJ P,NEXTUN ;STEP TO ANOTHER UNIT IN STR
JRST USET5A ;ALL UNITS FULL - SETTLE FOR WHAT WE GOT SO FAR
USET3A: MOVEM T2,@DEVRET##(F) ;SAVE POINTER (OR UNIT-CHANGE) IN DDB
HRRZ T3,DEVACC##(F) ;LOC OF A.T.
MOVEI T4,ACP1PT## ;ENSURE THAT 1PT IS OFF
ANDCAM T4,ACCUN1##(T3) ;SINCE WE JUST GENERATED A NEW POINTER
TLO S,IOSFIR ;INDICATE CHECKSUM MUST BE COMPUTED
JRST USETO2 ;GET MORE BLOCKS IF NEEDED
USET3B:
IFN FTDMRB,< ;IF MULTIPLE RIBS
MOVN T1,P1 ;GET -NUMBER OF BLOCKS LEFT TO GET
ADDM T1,DEVREL##(F) ;SET DEVREL TO END OF RIB FOR EXTRIB
PUSHJ P,EXTRIB ;CREATE AN EXTENDED RIB
IFE FTDSIM,<
JRST USETO4 ;COULDN'T GET THE RIB, ERROR
>
IFN FTDSIM,<
JRST [PUSHJ P,USETO4
PJRST DOWNIF]
>
ADDM P1,DEVREL##(F) ;RESET DEVREL TO BLOCK TO GET
ADDI P1,2 ;ACCOUNT FOR REDUNDANT AND EXTENDED RIB
SUB P1,T1 ;DECREMENT AMOUNT TO GET
ADDI P1,2 ;ACCOUNT FOR 2 RIBS
PUSHJ P,CPYEXT## ;SET UP THE DDB
IFE FTDSIM,<
POPJ P, ;RIB ERROR
>
IFN FTDSIM,<
JRST DOWNIF
>
JUMPLE P1,USETO5 ;FINISH UP IF GOT ENOUGH
JRST USET3C ;NOT ENOUGH, GO GET MORE
> ;END OF CONDITIONAL FTDMRB
USETO4: PUSHJ P,ERRFUL ;TOO MANY POINTERS,LIGHT IOBKTL
MOVNS P1 ;AMOUNT WE WERE UNABLE TO GET
ADDM P1,DEVREL##(F) ;ADJUST DEVREL
POPJ P, ;AND RETURN TO USER
;HERE IF UNIT OR RIB FULL
USET5A: PUSHJ P,USETO4 ;LIGHT AN ERROR BIT, ADJUST DEV REL
SOS DEVRET##(F) ;ADJUST DEVRET
MOVSI T1,-1
ADDM T1,DEVRSU##(F) ; AND DEVRSU (INCR'D AT USETO3)
;HERE WHEN ALL BLOCKS HAVE BEEN ALLOCATED
USETO5: MOVE W,DEVREL##(F) ;RESET W TO REQUESTED BLOCK
SKIPLE P1 ;IF COULDN'T GET ALL WE REQUESTED,
SUB W,P1 ;ADJUST BLOCK NUMBER
PUSHJ P,WRTPTR ;WRITE OUT RET POINTERS LEFT IN DDB
JUMPN T3,USETO4 ;RETURN WITH IOBKTL IF RIB ERR
IFN FTDSIM,<
PUSHJ P,DOWNIF ;RETURN DA IF WE OWN IT
SKIPE T1,DEVMBF##(F) ;HAVE MON BUF (SIM UPD)?
PUSHJ P,GVMNBF ;YES, RETURN IT
>
PUSH P,DEVRSU##(F) ;SAVE DEVRSU (USETZ1 MAY CHANGE IT)
PUSHJ P,USETO6 ;ZERO ALLOCATED, UNWRITTEN BLOCKS
POP P,DEVRSU##(F) ;RESTORE DEVRSU
PJRST STRIOS ;SAVE S AND EXIT THE UUO
IFN FTDSIM,<
;SUBROUTINE TO GIVE UP THE DA IF WE OWN IT
;ALWAYS RETURNS CPOPJ
DOWNIF::
TLNE S,IOSDA ;HAVE DA (SIM UPDATE)?
PJRST DWNDA ;YES, RETURN IT
POPJ P,
>
;SUBROUTINE TO ADD TO CURRENT POINTER
;ENTER WITH ACS SET AS IN GOOD RETURN FROM TAKBLK-
;T2=CLUSTER POINTER FOR NEW GROUP, T3=ADDRESS OF STRUCTURE DB
;EXIT WITH T1= NUMBER OF NEW BLOCKS GOTTEN
;AND UPDATED POINTER IN @DEVRET AND T2
ADDPTR::PUSH P,T1 ;SAV NO. OF BLOCKS GOTTEN
LDB T1,STYCNP##(T3) ;NO. OF CLUSTERS GOTTEN (AT END)
MOVE T2,@DEVRET##(F) ;CURRENT POINTER
LDB T4,STYCNP##(T3) ;CLUSTER COUNT
ADD T4,T1 ;PLUS NEW AMOUNT
HRRZ T1,DEVACC##(F) ;LOC OF A.T.
CAME T2,ACCPT1##(T1) ;IS THIS PNTR THE 1ST?
SETZ T1, ;NO. INDICATE BY T1=0
DPB T4,STYCNP##(T3) ;SAVE NEW CLUSTER COUNT
MOVEM T2,@DEVRET##(F) ;SAVE POINTER
JUMPE T1,TPOPJ## ;IS THIS 1ST PNTR?
MOVEM T2,ACCPT1##(T1) ;YES. SAVE IT IN A.T.
JRST TPOPJ## ;RESTORE T1 AND RETURN
;SUBROUTINE TO STEP TO NEXT UNIT IN FILE STRUCTURE WHICH HAS SPACE LEFT
;IF ALL SPACE IS GONE, RETURN CPOPJ WITH IOBKTL SET
;GOOD RETURN WITH U=DEVUNI= LOC OF NEW UNIT, AND T1=0
;AND A CHANGE-UNIT POINTER STORED IN @DEVRET AND LEFT IN T2
NEXTUN::HRRZ T1,UNISTR##(U) ;LOC OF STR DB
HLRZ T1,STRUNI##(T1) ;LOC OF 1ST UNIT IN STR
SKIPA ;TEST IF IT HAS ANY SPACE
NEXTU1: HLRZ T1,UNISTR##(T1) ;STEP TO NEXT UNIT IN STR
JUMPE T1,ERRFUL ;STR IS FULL IF AT END OF UNITS
SKIPG UNITAL##(T1) ;NO. UNIT HAVE ANY SPACE?
JRST NEXTU1 ;NO. TRY NEXT UNIT
MOVE U,T1 ;YES. SET U
HRRM U,DEVUNI##(F) ;AND DEVUNI
LDB T2,UNYLUN## ;GET LOGICAL UNIT NUMBER
TRO T2,RIPNUB## ;MAKE SURE NON-0
MOVEM T2,@DEVRET##(F) ;SAVE IN DDB
SETZ T1, ;MAKE SURE T1=0
JRST CPOPJ1## ;AND TAKE GOOD RETURN
;SUBROUTINE TO DO THE WORK FOR USETO/USETI
;HALTS IF NO POINTERS TO THE BLOCK
;RETURNS CPOPJ IF THERE IS A RIB ERROR
;SKIP - RETURN IF EVERYTHING IS OK
;ENTER WITH RH(M)=DESIRED BLOCK
;EXIT WITH DEVRET, DEVBLK, DEVLFT, DEVREL SET UP
USET00: HRRZ U,DEVUNI##(F)
JUMPE U,CPOPJ## ;ERROR IF UNIT WAS YANKED
HRRZ U,DEVFUN##(F) ;UNIT FOR 1ST POINTER IN DDB
HRRM U,DEVUNI##(F) ;SAVE IN DEVUNI (WILL CHANGE IF UNIT-CHANGE IS READ)
MOVE T2,DEVFLR##(F) ;LOWEST REL BLOCK OF POINTERS IN DDB
MOVE T3,W ;BLOCK NUMBER TO GET
CAML T2,T3 ;IS DESIRED BLOCK BELOW THIS FLOOR?
JRST USTRIB ;YES. READ IN WHOLE RIB
MOVEI T1,DEVRB1##(F) ;NO. SCAN THE POINTERS IN CORE
HRLI T1,MPTRLN## ; STARTING AT DEVRB1
PUSHJ P,SCNPTR ;TRY TO FIND POINTER TO BLOCK
JRST USTRIB ;NOT THERE - READ WHOLE RIB
;FOUND IT. DEVBLK,DEVREL,DEVLFT ARE SET UP
HRRZ T2,DEVRET##(F) ;CURRENT POINTER LOC
CAIN T2,DEVRBN##(F) ;POINTING TO LAST PNTR SLOT?
SKIPE DEVRB2##(F) ;YES, IS 2ND PTR 0? (YES IF SET DDB FROM
;A.T., MORE PNTRS LEFT IN RIB)
HRRM T1,DEVRET##(F) ;NO, SET DEVRET TO THIS POINTER
SUB T1,T2 ;DISTANCE BY WHICH WE CHANGED DEVRET
HRLZS T1 ;IN LH
ADDM T1,DEVRSU##(F) ;UPDATE DEVRSU BY THAT AMOUNT
JRST CPOPJ1## ;AND TAKE GOOD RETURN
IFN FTDMRB,<
;SUBROUTINE TO TURN ON IOSFIR FOR FIRST BLOCK IN EXTENDED RIB
EXTCKS: MOVE T1,DEVUNI##(F) ;NEWUX WIPES RH (DEVUNI)
LDB T2,DEYRBU## ;UNIT OF RIB
PUSHJ P,NEWUX
JFCL
EXCH T1,DEVUNI##(F) ;RESET DEVUNI GET RIB UNIT
MOVE T2,DEVRIB##(F) ;POINTER TO (EXTENDED) RIB
PUSHJ P,GRPAD ;GET BLOCK NUMBER OF RIB
ADDI T2,1 ;FIRST BLOCK PAST RIB?
CAME T2,DEVBLK##(F)
POPJ P, ;NO
CAMN T1,DEVUNI##(F) ;YES, RIGHT UNIT?
TLO S,IOSFIR ;YES, CHECKSUM TIME
PJRST STOIOS##
>
;HERE IF THE POINTERS IN THE DDB DON'T ENCOMPASS THE DESIRED BLOCK
;READ IN THE RIB, AND SCAN IT FROM THE BEGINNING
USTRIB: PUSHJ P,PTRTST ;READ POINTERS, REWRITE RIB IF POINTERS HAVE CHANGED
JRST GVMNB0 ;ERROR READING RIB
PUSHJ P,SAVE1
SETO P1,
USTRB5:
IFN FTDMRB,< ;IF MULTIPLE RIBS
LDB T2,DEYRBU## ;GET UNIT OF CURRENT RIB
PUSHJ P,NEWUX ;SET U
STOPCD GVMNB0,DEBUG,NSU, ;++NO SUCH UNIT
MOVE T2,DEVMBF##(F) ;IOWD FOR MONITOR BUFFER
MOVE T2,RIBFLR##+1(T2) ;FIRST WORD OF CURRENT RIB
SKIPL DEVRIB##(F) ;IF POSITIVE COULD BE OLD TYPE RIB
>
MOVEI T2,0 ;WHICH HAS NO RIBFLR WORD
MOVE T3,W ;BLOCK NUMBER TO GET
IFN FTDMRB,< ;IF MULTIPLE RIBS
CAML T2,T3 ;BLOCK BELOW FLOOR OF CURRENT RIB?
JUMPN T2,USTRB6 ;JUMP IF PRIME RIB
>
PUSHJ P,SCNPT0 ;SCAN THE CURRENT RIB
IFN FTDMRB,< ;IF MULTIPLE RIBS
JRST USTRB7 ;NOT HERE, LOOK IN NEXT RIB
>
IFE FTDMRB,< ;IF NO MULTIPLE RIBS
PJRST GVMNB0
>
MOVEM T2,DEVFLR##(F) ;SET LOWEST RELATIVE BLOCK IN DDB
HRRM U,DEVFUN##(F) ;SET CORRESPONDING UNIT
PUSHJ P,PTRBLT ;BLT POINTERS TO DDB
AOS (P) ;SET FOR SKIP RETURN
PJRST GVMNB0 ;RETURN MONITOR BUFFER AND EXIT
IFN FTDMRB,< ;IF MULTIPLE RIBS
;HERE WHEN WE MUST START LOOKING AT THE PRIME RIB
USTRB6:
AOJN P1,GVMNB0
PUSHJ P,REDRIB ;READ THE PRIME RIB
PJRST GVMNB0 ;ERROR READING THE RIB
PUSHJ P,SPTRW ;SET UP AOBJN WORD FOR THE RIB
JRST USTRB8 ;SET UP TO SCAN THE PRIME RIB
> ;END CONDITIONAL ON FTDMRB
;HERE TO GET THE NEXT RIB IN THE CHAIN
IFN FTDMRB,<
USTRB7: PUSHJ P,PTRNXT ;GET THE NEXT RIB IN THE CHAIN
;IF MULTIPLE RIBS
JRST USTRB9 ;EITHER ERROR OR COULDN'T FIND THE BLOCK
USTRB8: MOVE T3,W ;BLOCK NUMBER TO GET
JRST USTRB5 ;SCAN THE RIB
;HERE ON NON-SKIP RETURN FROM PTRNXT, EITHER RIB ERROR OR NO NEXT RIB
USTRB9: PJUMPN T3,GVMNB0 ;RETURN CPOPJ IF RIB ERROR
PUSHJ P,GVMNB0 ;NO NEXT RIB, GIVE UP THE BUFFER
SETOM DEVBLK##(F) ;SET DEVBLK TO -1 AS A FLAG
PUSHJ P,DDBZRO## ;ZERO DDB PNTR SPACE SINCE DEYRLC IS WRONG
JRST CPOPJ1## ;TAKE A SEMI-GOOD RETURN
> ;END CONDITIONAL ON FTDMRB
;SUBROUTINE TO READ THE POINTERS INTO CORE, COMPARE THE OLD POINTERS IN THE
;RIB WITH THE NEW POINTERS IN THE DDB, AND REWRITE THE RIB IF THEY DIFFER
;SUBROUTINE GETS A MONITOR BUFFER AND RETURNS WITH THE RIB IN IT
;RETURNS WITH T1=AOBJN WORD FOR WHOLE GROUP OF PNTRS IN RIB
;RETURNS CPOPJ IF ERROR READING RIB (STILL WITH MON BUF
;RETURNS CPOPJ1 NORMALLY
PTRTST:
IFN FTDSIM,<
HRRZ T1,DEVACC##(F)
JUMPE T1,PTRTS0
MOVE T1,ACCSTS##(T1) ;SIM UPDATE FILE?
TRNN T1,ACPSMU
JRST PTRTS0
PUSHJ P,GTMNBF ;YES, GET MON BUF
PUSHJ P,UPDA ;AND DA TO PREVENT RACE IF WE WRITE RIB
PTRTS0:>
PUSHJ P,PTRCUR ;READ THE POINTERS INTO CORE
IFE FTDSIM,<
JUMPN T3,CPOPJ## ;RETURN IF ERROR READING RIB
>
IFN FTDSIM,<
JUMPN T3,DOWNIF
>
HLRZ T3,DEVEXT(F) ;EXTENSION
LDB T4,PUUOAC## ;CHAN NUMBER
MOVE T4,USRJDA##(T4) ;WAS AN ENTER DONE ON THIS CHAN?
TLNE T4,ENTRB+OUTPB ; (IF NOT THIS DDB DIDN'T CHANGE THE PNTRS)
CAIN T3,(SIXBIT /UFD/) ;"UFD"?
JRST USTRB4 ;YES, PNTRS IN THE RIB ARE RIGHT
;HERE WHEN THERE ARE PNTRS IN THE DDB WHICH MAY NOT BE IN THE RIB - CHECK THEM
MOVE T3,UNISTR##(U) ;GET ADDRESS OF STRUCTURE DATA BLOCK
SETO T2, ;PUT ONE'S IN T2
LDB T4,STYCLP##(T3) ;CREATE MASK FOR CLUSTER POINTER
;PART OF RETRIEVAL POINTER
LDB T2,DEYRLC## ;POINTER LOC IN THE RIB
ADD T1,T2 ;POINT TO 1ST RIB PNTR - CORRESPONDING TO DEVRB1
MOVEI T2,DEVRB1##(F) ;POINT T2 TO DDB POINTERS
HRLI T2,MPTRLN## ;MAKE T2 AN AOBJN WORD
USTRB1: SKIPN T3,(T2) ;GET A PNTR FROM DDB
JRST USTRB3 ;ALL DONE
CAMN T3,(T1) ;SAME AS PNTR IN RIB?
JRST USTRB2 ;YES
EXCH T3,(T1) ;NO. SAVE PNTR IN MON BUF
JUMPE T3,USTR1A ;IF OLD PNTR=0, OK
XOR T3,(T1) ;XOR RIB WITH MON BUF
TDNE T3,T4 ;IF PNTR PARTS EQUAL, SKIP
STOPCD .+1,DEBUG,PNE, ;++POINTERS NOT EQUAL
;FALL INTO USTR1A
USTR1A: TLZ T1,-1 ;ZERO LH(T1) - WAS MRIBLN
USTRB2: AOBJP T2,USTRB3 ;SKIP IF ALL DDB PNTRS LOOKED AT
AOJA T1,USTRB1 ;LOOK AT NEXT POINTER
;HERE WHEN ALL POINTERS HAVE BEEN COMPARED, CHANGED PNTRS STORED IN MON BUF
USTRB3: SKIPL T1 ;T1 NEG IF ALL PNTRS COMPARED
PUSHJ P,WRTRIB ;WRITE THE MON BUF AS 1ST RIB
USTRB4:
IFN FTDSIM,<
PUSHJ P,DOWNIF ;RETURN DA IF WE OWN IT
>
PUSHJ P,SPTRW ;SET T1 AS AN AOBJN WD FOR PNTRS AGAIN
JRST CPOPJ1## ;AND TAKE GOOD-RETURN
;SUBROUTINE TO SCAN A BLOCK OF RETRIEVAL POINTERS TO FIND THE GROUP POINTER
;FOR A PARTICULAR BLOCK
;ENTER WITH:
;T1=AOBJN WORD FOR THE SET OF POINTERS
;T2=INITIAL RELATIVE BLOCK OF THE SET OF POINTERS
;T3=DESIRED RELATIVE BLOCK
;ENTER AT SCNPT0 TO SCAN WHOLE RIB (IN MON BUF)
;EXIT WITH:
;T1=ADDRESS OF THE POINTER, LH=-NUMBER OF POINTERS LEFT
;T2=RELATIVE BLOCK NUMBER OF POINTER
;DEVLFT,DEVBLK,DEVREL SET IN THE DDB
;EXIT CPOPJ IF THE POINTER WAS NOT FOUND
;SKIP-RETURN IF THE POINTER WAS FOUND
SCNPT0::PUSHJ P,SPTRW ;SET T1=AOBJN WORD FOR WHOLE RIB
SCNPTR::PUSHJ P,SAVE2## ;SAVE P1,P2
LDB T4,UNYBPC## ;NUMBER OF BLOCKS PER CLUSTER
PUSH P,T3 ;SAVE DESIRED BLOCK
SUB T3,T2 ;T3=RELATIVE BLOCK NUMBER IN SET
IDIV T3,T4 ;T3=DESIRED CLUSTER
MOVE T2,UNISTR##(U) ;LOC OF FILE STRUCTURE DB
HLLZ P1,STYCNP##(T2) ;SET UP POS, SIZE OF POINTER COUNT FIELD
TLO P1,T1 ;POINTER TO CLUSTER COUNT
SETZ P2, ;CLEAR REGISTER TO ACCUMULATE BLOCK COUNT
SCNPT1: LDB T2,P1 ;GET NUMBER OF CLUSTERS IN THIS POINTER
JUMPN T2,SCNPT2 ;REAL POINTER IF NON-0
SKIPN T2,(T1) ;UNIT CHANGE OR END OF POINTERS
PJRST TPOPJ## ;END OF POINTERS. ERROR RETURN
TRZ T2,RIPNUB## ;REMOVE BIT 18 (REST IS A LOGICAL UNIT NUMBER)
PUSHJ P,NEWUNI ;SET UP U, DEVUNI(F)
JRST TPOPJ## ;INVALID UNIT -NOT FOUND RETURN
SCNPT3: AOBJN T1,SCNPT1 ;GO BACK TO TEST NEXT POINTER
JRST TPOPJ## ;RAN OUT OF POINTERS, ERROR RETURN
;HERE WHEN A REAL POINTER HAS BEEN FOUND
SCNPT2: ADD P2,T2 ;PLUS LENGTH OF GROUP
CAML T3,P2 ;IS DESIRED CLUSTER IN THIS POINTER?
JRST SCNPT3 ;NO, STEP TO NEXT
LDB P1,UNYBPC## ;YES. NUMBER OF BLOCKS PER CLUSTER
SUB P2,T2 ;SET P2 BACK TO BEGINNING OF GROUP
SUB T3,P2 ;T3=CLUSTER IN GROUP
IMUL T3,P1 ;T3=BLOCK NUMBER IN GROUP
ADD T3,T4 ;T3= DISTANCE OF BLOCK FROM START OF PNTR
POP P,T4 ;BLOCK NUMBER TO GET
SKIPE T3 ;AT 1ST BLOCK OF A GROUP?
CAIN T4,1 ;IS IT BLOCK 1?
TLOA S,IOSFIR ;YES, SET CHECKSUM BIT
TLZ S,IOSFIR ;NO, CLEAR CHECHSUM BIT
IMUL T2,P1 ;T2=RELATIVE BLOCK NUMBER OF START OF PNTR
SUB T2,T3 ;COMPUTE NUMBER OF BLOCKS LEFT IN GROUP
HRRM T2,DEVLFT##(F) ;SAVE IN DDB
IFN FTDMRB,<
HRRZ T2,T1 ;GET ADDRESS PORTION OF POINTER
CAIG T2,DEVRBN##(F) ;SKIP IF NOT POINTING TO DDB
CAIGE T2,DEVRB1##(F) ;SKIP IF POINTING TO DDB
JRST SCNP2A ;NOT IN DDB, MUST BE IN MONITOR BUFFER
MOVE T2,DEVLPC##(F) ;GET WORD CONTAINING LAST POINTER FLAG
TLNN T2,DEPLPC## ;IS POINTER IN DDB?
JRST SCNP2D ;NO, PROCEED
SKIPE 1(T1) ;YES, IS NEXT SLOT EMPTY?
JRST SCNP2A ;NO, CHECK TO SEE IF THIS IS LAST SLOT
HRRZ T2,DEVLFT##(F) ;IS LAST, MAKE LAST BLOCK UNAVAILABLE
SOJE T2,CPOPJ## ;JUMP IF NO BLOCKS AVAILABLE
JRST SCNP2C ;STORE THE NEW VALUE OF DEVLFT
SCNP2A: HRRZ T2,DEVLFT##(F) ;RETURN DEVLFT TO T4
AOBJN T1,SCNP2B ;ABOUT TO RUN OUT OF POINTERS?
SOJE T2,SCNP2E ;YES, MAKE LAST BLOCK UNAVAILABLE
SCNP2B: SUB T1,[XWD 1,1] ;RESTORE AOBJN WORD
SCNP2C: HRRM T2,DEVLFT##(F) ;STORE IN DDB
> ;END FTDMRB
SCNP2D: MOVEM T4,DEVREL##(F) ;=CURRENT RELATIVE BLOCK
MOVE T2,T4 ;GET DEVREL INTO T2
SUB T2,T3 ;SET TO RELATIVE BLOCK OF START OF GROUP
SKIPN T4 ;USETI/O TO BLOCK 0?
TLZ S,IOSFIR ;YES. DONT COMPUTE CHECKSUM (ITS FOR BLOCK 1)
MOVE T4,UNISTR##(U)
MOVE T4,STYCLP##(T4) ;SET T4=POINTER TO CLUSTER ADDRESS
HRRI T4,(T1)
LDB T4,T4 ;T4=CLUSTER ADDRESS
IMUL T4,P1 ;1ST LOGICAL BLOCK ADR. IN POINTER
ADD T3,T4 ;+DISTANCE TO DESIRED BLOCK
MOVEM T3,DEVBLK##(F) ;=LOGICAL ADR. OF DESIRED BLOCK
JRST CPOPJ1## ;TAKE GOOD RETURN
IFN FTDMRB,<
SCNP2E: MOVSI T1,DEPLPC## ;TELL CALLER WHY HE LOST
IORM T1,DEVLPC##(F)
POPJ P,
>
;SUBROUTINE TO READ A RIB BLOCK, AND STORE THE POINTERS IN THE DDB
;ENTER AT RDPTRA IF THE POINTERS ARE ALREADY IN THE MON BUF
RDPTRS::PUSHJ P,PTRGET ;READ THE RIB BLOCK INTO A MON BUF
RDPTRA::PUSHJ P,PTRCPY ;COPY CURRENT POINTERS FROM MON BUF TO DDB
PJRST GVMNB0 ;GIVE UP MON BUF AND RETURN
;SUBROUTINE TO WRITE POINTERS
WRTPTR: PUSHJ P,PTRCUR ;READ THE RIB
SKIPN T3 ;DONT TRUST ANYTHING IF RIB ERR
PUSHJ P,PTRWRT ;BLT POINTERS INTO MON BUF, WRITE THEM
PJRST GVMNBX ;GIVE UP MON BUF AND RETURN
;SUBROUTINE TO GET THE CURRENT POINTERS INTO CORE
;RETURNS T3=0 IF OK, NON-0 IF RIB ERROR
PTRCUR::SKIPN DEVMBF##(F) ;SKIP IF WE ALREADY HAVE BUFFER
PUSHJ P,GTMNBF ;GET THE MONITOR BUFFER
PUSHJ P,RIBCUR ;READ THE CURRENT RIB
PJRST SPTRW ;SET UP A POINTER AND RETURN
;SUBROUTINE TO COPY POINTERS INTO MON BUF AND WRITE IT
;ENTER WITH T1=AOBJN WORD FOR ENTIRE MONITOR BUFFER
;RETURNS WITHOUT GIVING UP THE MON BUFFER
PTRWRT::PUSHJ P,DD2MN ;COPY DDB POINTERS INTO MONITOR BUF
STOPCD .+1,DEBUG,TMP, ;++TOO MANY POINTERS
;SHOULDN'T HAPPEN SINCE DEVRSU DIDNT GO POSITIVE
HRRZ T2,T1 ;SAVE CURRENT POINTER LOC
PUSHJ P,SPTRW ;MINUS ORIGINAL POINTER LOC
SUBI T2,-1(T1)
DPB T2,DEYRLC## ;=CURRENT POSITION IN MON BUF
PJRST WRTRIB ;WRITE THE RIB AND RETURN
;SUBROUTINE TO FIND WRITING DDBS ASSOCIATED WITH A FILE
;CALL FNDDDB THE FIRST TIME, TO FIND OTHER DDBS CALL FNDDDN WITH
; T2 AS RETURNED FROM THE FIRST CALL
;CALL WITH T1= L(AT)
;RETURNS CPOPJ IF NO MORE (OR NONE) WRITING DDBS
;RETURNS CPOPJ1 NORMALLY, WITH T2=ADR OF NEXT DDB
FNDDDB: MOVEI T2,DSKDDB## ;START AT FIRST
FNDDDN: HLRZ T2,DEVSER(T2) ;STEP TO NEXT
HRRZ T1,DEVACC##(F) ;LOC OF A.T.
MOVSI T3,DVDSK
FNDDD1: TDNN T3,DEVMOD(T2) ;DSK DDB?
IFN FTSPL,<
SKIPGE DEVSPL(T2)
>
IFE FTSPL,<
CAIA
>
SKIPA T4,DEVACC##(T2) ;YES, POINTING AT RIGHT A.T.?
POPJ P, ;NO
IFN FTDSIM,<
CAIE T2,(F) ;IGNORE OUR OWN DDB
>
CAIE T1,(T4)
JRST NXTDDB ;NO, TRY NEXT
MOVE T4,DEVWRT##(T2) ;YES, IS THIS DDB WRITING?
TRNE T4,DEPWRT##
JRST CPOPJ1## ;YES, FOUND THE GUY
NXTDDB: HLRZ T2,DEVSER(T2) ;NO, TRY NEXT
JUMPN T2,FNDDD1
POPJ P, ;COULDNT FIND THE DDB - RETURN NON SKIP
;SUBROUTINE TO FIND CURRENT POINTERS FOR A FILE IN SOME DDB
;CALLED WHEN THE ACCESS TABLE INDICATES POINTERS SHOULD BE THERE,
; BUT THE POINTERS ARE NOT IN THE DDB
;SOME DDB HAS ALLOCATED NEW BLOCKS, THE NEW POINTERS AREN'T YET IN THE RIB
;NON-SKIP RETURN IF COULDN'T FIND THE BLOCK
;NORMAL RETURN IS CPOPJ1
FNDPTR: HRRZ T1,DEVACC##(F) ;LOC OF A.T.
MOVE T2,ACCCNT##(T1) ;STATUS OF FILE
TRNE T2,ACPUPD ;UPDATE?
TRNN T2,ACMCNM## ;YES, IS THIS THE ONLY READER?
JRST FIXDDB ;YES, CANT FIND A WRITING DDB
PUSHJ P,FNDDDB ;FIND THE WRITER
JRST FIXDDB ;NONE THERE - REREAD THE RIB
;HERE WHEN THE RIGHT DDB HAS BEEN FOUND
MOVSI T1,DEVRB1##(T2) ;COPY THE CURRENT PNTRS INTO DDB
HRRI T1,DEVRB1##(F) ; (MAY INCLUDE POINTERS WHICH ARE ALREADY
BLT T1,DEVRBN##(F) ; IN THE RIB)
MOVE T1,DEVLPC##(F) ;SET UP ALL THE DDB PARAMETERS
MOVE T3,DEVLPC##(T2)
TLNE T3,DEPLPC##
TLOA T1,DEPLPC##
TLZ T1,DEPLPC##
MOVEM T1,DEVLPC##(F)
LDB T1,DEXRLC##
DPB T1,DEYRLC##
HLRE T1,DEVRSU##(T2)
HRLM T1,DEVRSU##(F)
MOVE T3,DEVRET##(T2)
SUBI T3,DEVRB1##(T2)
ADDI T3,DEVRB1##(F)
HRRM T3,DEVRET##(F)
MOVE T1,DEVFLR##(T2)
MOVEM T1,DEVFLR##(F)
MOVE T1,DEVRIB##(T2)
MOVEM T1,DEVRIB##(F)
MOVE T1,DEVFUN##(T2)
HRRM T1,DEVFUN##(F)
;DDB IS ALL SET (IF WE FOUND THE WRITER). CALL USETI TO SET FOR THE RIGHT BLOCK
; WILL FIND IT IN THE DDB POINTERS IF THERE, IF THE UPDATER CLOSED THEY SHOULD
; HAVE BEEN WRITTEN BACK INTO THE RIB (COULDN'T FIND THE RIGHT DDB)
FIXDDB: PUSH P,W ;SAVE W
MOVE W,DEVREL##(F) ;BLOCK WE'RE LOOKING FOR
PUSHJ P,USET00 ;GO SET UP FOR IT
CAIA
SKIPG DEVBLK##(F) ;SEMI-GOOD RETURN?
SOS -1(P) ;STILL COULDN'T FIND THEM (SYSTEM ERROR?)
POP P,W ;RESTORE W
PJRST CPOPJ1## ;EVERYTHING WORKED!
;SUBROUTINE TO READ THE CURRENT RIB
;RETURNS CPOPJ, IF T3 NON-ZERO, ERROR READING RIB
;RETURNS UNIT OF RIB IN T2
RIBCUR::PUSH P,U ;SAVE CURRENT UNIT
LDB T2,DEYRBU## ;GET CURRENT RIB LOGICAL UNIT NUMBER
PUSHJ P,NEWUNI ;SET UP U,DEVUNI
STOPCD UDEERR,DEBUG,UDE, ;++UNIT DOESN'T EXIST
LDB T2,DEYRBA## ;GET CURRENT RIB CLUSTER ADDRESS
LDB T3,UNYBPC## ;BLOCKS PER CLUSTER FOR THIS UNIT
IMUL T2,T3 ;BLOCK NUMBER IN T2
MOVE T1,DEVMBF##(F) ;GET IOWD FOR MONITOR BUFFER
PUSHJ P,MONRED ;READ THE BLOCK
PUSHJ P,RIBCHK ;MAKE SURE ITS A VALID RIB
UDEERR: SKIPA T3,[-1] ;RIB ERROR, SET T3=-1
SETZ T3, ;T3=0 INDICATES RIB OK
MOVE T2,U ;RIB-UNIT IN T2
POP P,U ;RESTORE CURRENT UNIT
HRRM U,DEVUNI##(F) ;AND SAVE IN DDB
POPJ P, ;AND RETURN
IFN FTDMRB,< ;IF MULTIPLE RIBS
;SUBROUTINE TO GET THE NEXT RIB IN A CHAIN INTO CORE
;RETURNS CPOPJ1 WITH NEXT RIB IN CORE, CPOPJ IF NONE OR ERROR
;IF CPOPJ RETURN AND T3 NON-0, ERROR,T3=0,NO NEXT RIB
PTRNXT::SETZ T3, ;T3=0 INDICATES NO RIB ERROR
MOVE T2,DEVMBF##(F) ;IOWD FOR MONITOR BUFFER
SKIPL DEVRIB##(F) ;IS CURRENT RIB EXTENDED
SKIPN RIBFLR##+1(T2) ;NO, IS THIS AN EXTENDABLE FILE
SKIPN T2,RIBXRA##+1(T2) ;GET THE NEXT RIB ADDRESS
POPJ P, ;NONE, END OF CHAIN
MOVEM T2,DEVRIB##(F) ;MAKE NEXT RIB CURRENT RIB
PUSHJ P,PTRCUR ;READ THE RIB
JUMPN T3,CPOPJ## ;NON-SKIP RETURN IF ERROR
JRST CPOPJ1## ;GOOD RETURN
> ;END CONDITIONAL ON FTDMRB
SUBTTL MISCELLANEOUS FUNCTIONS
IFN FTRP04,<
;UNLOAD A DRIVE
UNLOAD::PUSHJ P,GETWDU ;GET USERS ARGUMENT
MOVNI T2,1 ;WHOLE WORD MUST MATCH
PUSHJ P,SRUNI## ;IS IT A UNIT NAME?
PJRST ECOD1## ;NO - ERROR 1
JFCL
SKIPN UNILOG##(U) ;YES, IS IT IN A FILE STRUCTURE?
SKIPE UNISTS##(U) ;NO, IS IT IDLE?
PJRST ECOD2## ;NOT IDLE OR IN AN STR - ERROR 2
MOVE J,UNIKON##(U) ;KONTROLLER DATA BLOCK
SKIPGE KONUNL##(J) ;DOES DEVICE UNLOAD?
PJRST ECOD3## ;NO, ERROR 3
PUSHJ P,@KONUNL##(J) ;YES, UNLOAD IT
JFCL ;IGNORE IF UNIT NOT READY
MOVEI T2,O2COD## ;MARK UNIT AS DOWN,
MOVEM T2,UNISTS##(U) ; NO ONCE-A-MINUTE TYPOUT
IFN FTDUAL,<
SKIPE T1,UNI2ND##(U)
MOVEM T2,UNISTS##(T1)
>
PJRST CPOPJ1## ;AND TAKE GOOD RETURN
IFN FTDSUP,<
SETCPT::SKIPA T1,[IORM T1,DEVCPT##(F)]
CLRCPT::MOVE T1,[ANDCAM T1,DEVCPT##(F)]
PUSHJ P,SAVE1## ;SAVE P1
MOVE P1,T1 ;SET TO CLEAR/SET THE BIT
PUSHJ P,GETWDU## ;GET USERS ARG
PUSHJ P,DVCNSG## ;FIND THE DDB
PJRST ECOD1## ;NONE SUCH - ERROR 1
PUSHJ P,VALUUX ;LEGAL?
PJRST ECOD1## ;NO, ERROR 1
MOVE U,DEVUNI##(F) ;YES, GET UNIT (IF SET UP)
JUMPE U,SETCP1
MOVE J,UNIKON##(U) ;KONTROLLER DATA BLOCK LOC
SKIPG KONRDC##(J) ;DOES DEVICE HAVE COMPAT. MODE?
PJRST ECOD2## ;NO, ERROR 2
SETCP1: MOVSI T1,DEPCPT## ;YES, GET THE BIT
XCT P1 ;SET/CLEAR BIT IN DDB
PJRST CPOPJ1## ;AND TAKE GOOD RETURN
>
IFE FTDSUP,<
SETCPT::
CLRCPT::POPJ P,0
>>
IFE FTRP04,<
SETCPT::
CLRCPT::
UNLOAD::POPJ P,0
>
SUBTTL DIAG DIAGNOSTIC UUO
IFN FTDHIA,<
DIAUUO::PUSH P,T1
HRR M,T1 ;ADR OF BLOCK
PUSHJ P,GETWDU## ;GET ARG
CAIN T1,7 ;WANT KON/UNIT NUMBER?
JRST DIAKUN ;YES, TELL HIM
IFN FTVM,<
SKIPE .UPMP+.UPVRT
JRST [POP P,(P)
JRST DIAVRT]
>
SETZ T1, ;SAY WE WANT USER-IOT
PUSHJ P,STOTAC## ;TRPSET CALLS GETTAC
PUSHJ P,TRPSET## ;CHECK PRIVS, SET USRIOT
;IF GOING TO DO IO HE'D BETTER BE ALL IN CORE
JRST [POP P,(P) ;NO PRIVS, LOSE
JRST DIANPV]
POP P,T1 ;OK, RESTORE ARGUMENT
PUSHJ P,STOTAC## ;TRPSET STORES 0 IN THE AC
HLRE T2,T1 ;NUMBER OF ARGUMENTS
JUMPGE T2,WRONGN ;ILLEGAL N
PUSHJ P,SAVE3##
MOVN P1,T2 ;P1=NUMBER OF ARGS
CAIGE P1,2 ;LEGAL?
JRST WRONGN ;ILLEGAL N
HRR M,T1 ;LOC OF ARG BLOCK
PUSHJ P,GETWDU## ;GET FUNCTION
CAILE T1,MXDIAG ;LEGAL?
JRST BADFNC ;ILLEGAL FNCN
MOVE P2,T1 ;P2=FUNCTION CODE
PUSHJ P,GETWD1## ;GET KON/UNIT WORD
LDB T2,[POINT 7,T1,6] ;CONTROLLER NUMBER
HLRZ U,SYSUNI## ;FIND IT
DIADRS: HRRZ P3,UNIKON##(U)
MOVE T3,KONREG##(P3)
ADDI T3,KONEBK##(P3) ;POINT AT FIRST IO WORD IN KDB
LDB T3,[POINT 7,(T3),9] ;KON NUM FROM KDB
CAIN T3,(T2) ;MATCH?
JRST DIAUNI ;YES, WE HAVE KON
HLRZ U,UNISYS##(U) ;NO, STEP TO NEXT
JUMPN U,DIADRS ;AND TEST IT
JRST ILLKON ;ILLEGAL CONTROLLER NUMBER
DIAUNI: LDB T1,[POINT 3,T1,29] ;UNIT
DIAUN1: LDB T2,UNYPUN## ;UNIT FROM UDB
CAMN T1,T2 ;RIGHT ONE?
JRST DIAUN2 ;YES, SEE IF SAME KON
HLRZ U,UNISYS##(U) ;NO, TRY NEXT
JUMPN U,DIAUN1
JRST ILLUNI ;NO MATCH, YOU LOSE
DIAUN2: HRRZ T1,UNIKON##(U) ;ON SAME KONTROLLER?
CAIE T1,(P3)
JRST ILLUNI ;ILLEGAL UNIT
;STILL IN FTDHIA CONDITIONAL
DIADSP: MOVE F,.PDDIA##(W) ;SET UP F
PUSHJ P,@DIAFNC-1(P2) ;AND DISPATCH
NOUIO: SKIPA T1,[XC.UIO] ;LOST - CLEAR USRIOT
JRST CPOPJ1## ;;WON - SKIP RETURN
ANDCAM T1,JOBPD1##(R)
MOVSI T1,NSWP!NSHF ;JOB IS NOW MOVABLE
LDB T2,PJOBN##
ANDCAM T1,JBTSTS##(T2)
POPJ P,
DIAFNC: DIAASU ;(1)ASSIGN SINGLE UNIT
DIAAAU ;(2)ASSIGN ALL UNITS
DIARCU ;(3)RELEASE CHAN AND ALL UNITS
DIASCP ;(4)SPECIFY CHANNEL PROGRAM
DIARCP ;(5)RELEASE CHAN PROGRAM
DIAGCS ;(6)GET CHAN STATUS
DIAKUN ;(7)GET KONTROLLER AND UNIT
MXDIAG==.-DIAFNC
;STILL IN FTDHIA CONDITIONAL
;HERE TO ASSIGN SOME UNIT
DIAASU: JUMPN F,DIUNAA ;ALREADY HAVE SOME UNITS ASS'D
HRRZ T1,UNISTR##(U) ;TEST DIAG-MODE WHEN INPL'D
JUMPN T1,UNNDMD ;NOT IN DIAGNOSTIC MODE
HLRZ T1,UNIDIA##(U) ;JOB WHICH OWNS THIS UNIT
CAME T1,.C0JOB## ;SOME OTHER JOB HAVE IT?
JUMPN T1,UNAAJB ;UNIT ASS'D TO ANOTHER JOB
JUMPN T1,DIAHVF ;HAVE A DDB SET IF F NON-0
PUSHJ P,FAKDDB## ;GET A DDB
JRST DINEFC ;NOT ENOUGH "FREE" CORE
HRL F,.C0JOB## ;SET UNIDIA=JOB,,DDB
MOVEM F,UNIDIA##(U)
JRST DIAHVF ;AND CONTINUE
;HERE TO ASSIGN ALL UNITS ON A CHANNEL
DIAAAU: JUMPN F,DIUNAA ;ALREADY HAVE SOME UNITS ASS'D
HRRZ T1,U ;SAVE STARTING-POINT
DIAAA1: HRRZ T2,UNISTR##(T1) ;TEST DIAG MODE WHEN IMPL'D
JUMPN T2,UNNDMD ;NOT IN DIAG MODE
HLRZ T2,UNIDIA##(T1) ;UNIT IN DIAG FOR SOME OTHER JOB?
CAME T2,.C0JOB##
JUMPN T2,UNAAJB ;UNIT ASS'D TO ANOTHER JOB
SKIPE T2 ;HAVE A DDB SET UP ALREADY?
HRRZ F,UNIDIA##(T1) ;YES, SAVE IT
HLRZ T1,UNICHN##(T1) ;STEP TO NEXT UNIT ON CHAN
CAIE T1,(U) ;BACK WHERE WE STARTED?
JRST DIAAA1 ;NO, TEST IT
JUMPN F,DIAAA2 ;GO IF WE HAVE A DDB
PUSHJ P,FAKDDB## ;NONE, GET ONE
JRST DINEFC ;NOT ENOUGH CORE
DIAAA2: HRL F,.C0JOB## ;SET JOB,,DDB IN EVERY UNIT
HRRZ T1,U ; ON THE CHANNEL
DIAAA3: SKIPN UNIDIA##(T1)
MOVEM F,UNIDIA##(T1) ;(IF NONE THERE ALREADY)
HLRZ T1,UNICHN##(T1)
CAIE T1,(U)
JRST DIAAA3
DIAHVF: HRRZ F,UNIDIA##(U) ;MAKE SURE F IS RIGHT
MOVEM U,DEVUNI##(F) ;SAVE U IN DDB
MOVEM F,.PDDIA##(W) ;SAVE DDB IN PDB
DIAHV1: SKIPN T1,DIADSK## ;SOME OTHER JOB IN DIAG ALREADY?
JRST DIASCH ;NO
MOVEI T1,2 ;YES, SLEEP FOR A WHILE
PUSH P,F ;SLEEP ZEROS F
PUSHJ P,SLEEP## ;AND THEN TRY AGAIN
POP P,F
JRST DIAHV1
;STILL IN FTDHIA CONDITIONAL
DIASCH: HRRZM F,DIADSK## ;INDICATE WE WANT TO STOP IO
DPB T1,PDVTIM## ;NO HUNG TIMER TILL WE START
MOVSI T1,NSWP!NSHF ;SO SCHEDULER WONT TRY TO
IORM T1,JBTSTS##(J) ; SWAP US (STORE JOB IN FORCE)
PUSHJ P,SETACT## ;SET DDB IOACT
HRRZ T2,UNICHN##(U) ;SAY WHAT CHANNEL WE WANT TO STOP IO ON
MOVEM T2,DIACHN##
PUSHJ P,WINDWN ;ANNY IO GOING?
PUSHJ P,WAIT1## ;YES, WAIT FOR IT TO STOP
;HERE WHEN ALL IO IS STOPPED ON THE DESIRED CHANNEL.
HRROS DIADSK## ;INDICATE WE'RE IN MIDDLE OF TEST
SETZM @DIACHN## ;CHAN IS BUSY
CAIGE P1,3 ;TIME LIMIT GIVEN?
TDZA T1,T1 ;NO, SET FOR 1 MINUTE
PUSHJ P,GETWD1## ;YES, GET IT
IDIVI T1,^D1000 ;CONVERT TO SECS
SKIPN T1 ;IF LESS THAN 1 SEC (OR NONE),
MOVEI T1,^D60 ; SET TIMER FOR 1 MINUTE
IDIVI T1,^D120 ;CONVERT TO 2-MINUTE CHUNKS
MOVEM T1,DEVDIA##(F) ;REMAINDER IN 2-MINUTE UNITS
JFFO T2,.+2 ;COMPUTE N SUCH THAT 2^N .GT. TIME
MOVEI T3,44-1
MOVNS T3
ADDI T3,44
DPB T3,PDVTIM## ;STORE N IN DDB
PUSHJ P,SETACT## ;MAKE SURE IOACT IS ON
PUSHJ P,GETWDU
HLLZ T2,T1 ;GET KONTROLLER DEVICE CODE
LSH T2,-3
TLO T2,(CONO) ;MAKE A CONO DEV,0
XCT T2 ; AND EXECUTE IT
PJRST CPOPJ1## ;AND TAKE GOOD RETURN
;STILL IN FTDHIA CONDITIONAL
;HERE FROM STOP1A ON ^C OR EXIT, HALT,... ;OR FROM HNGDSK
DIACLR::PUSHJ P,FNDPDB## ;FIND FDB
POPJ P, ;THERE ISN'T ONE
SKIPN F,.PDDIA##(W) ;DIAG GOING?
POPJ P, ;NO
MOVE U,DEVUNI##(F) ;YES, SET UP U
PUSHJ P,SAVE3##
HRRZ P3,UNIKON##(U) ;AND P3
TLOA F,-1 ;INDICATE FROM DIACLR
;HERE TO RELEASE ALL UNITS
DIARCU: JUMPE F,CPOPJ1## ;EXIT IF NO DIAG SET UP
PUSHJ P,DIARCX ;GIVE UP IOWD BLOCK IF ONE EXISTS
PUSH P,U ;SAVE U
DIARC1: HRRZ T1,UNIDIA##(U) ;DDB WHICH HAS UNIT
CAME T1,.PDDIA##(W) ;OURS?
JRST DIARC2 ;NO
SETOM UNICYL##(U) ;YES, SET SO WE'LL SEEK ON IT
SETZM UNIDIA##(U) ;NOW NO DIAG FOR IT
DIARC2: HLR U,UNICHN##(U) ;STEP TO NEXT UNIT
CAME U,(P) ;BACK WHERE WE STARTED?
JRST DIARC1 ;NO, TEST IT
SETZM .PDDIA##(W) ;DONT HAVE A DIAG GOING
SETZM DIADSK##
SETOM @DIACHN## ;CHAN IS FREE
SKIPL F
PUSHJ P,NOUIO ;CLEAR SOME BITS
PUSHJ P,CLRDDB## ;RETURN DDB SPACE
MOVE T1,[CRNKUP,,1] ;CAUSE CANT START IO ON UUO LEVEL
IDPB T1,CLOCK## ; SO COME BACK WITH PD LIST NOT MAPPED
IDPB U,CLOCK## ;SAY WHICH UNIT (CHAN)
JUMPL F,TPOPJ## ;NON-SKIP IF CALLED FROM DIACLR
SETZB T1,F ;MAKE SURE CRNKUP IS CALLED BEFORE JOB CAN
PUSHJ P,SLEEP## ; DO ANOTHER DIAG.
JRST TPOPJ1## ;AND TAKE SKIP-RETURN
;STILL IN FTDHIA CONDITIONAL
;HERE TO SET UP A SCHANNEL PROGRAM
DIASCP: JUMPE F,NOASUN ;NO ASS'D UNITS
PUSHJ P,DIARCX ;RETURN ANY IOWD
PUSHJ P,GETWD1## ;GET IOWD
MOVEM T1,DEVDMP##(F) ;UNRELOCATED IOWD
HLRE T2,T1 ;LENGTH OF IOWD
JUMPE T2,IOWCPB ;TOO BIG IF 0
MOVEI T1,1(T1) ;START ADDRESS
MOVNS T2 ;+LENGTH
ADDI T2,-1(T1) ;TOP ADDRESS
TRC T2,(T1) ;LEGAL?
TRNE T2,777000
JRST [SETZM DEVDMP##(F) ;IOWD CROSSES PAGE BOUNDARY
JRST IOWCPB]
HRR M,T1 ;DO ADDRESS CHECKING WITH GETWD
PUSHJ P,GETWDU## ; TO INSURE THAT THE PAGE IS IN CORE
IFN FTKA10,<
HRRZ P2,KONCOM##(P3) ;WHERE IOWD IS TO GO
MOVE T2,DEVDMP##(F) ;IOWD
ADDI T2,(R) ;RELOCATE IT
MOVEM T2,(P2) ;AND SAVE IN LOW CORE
SETZM 1(P2) ;TERMINATE LIST
>
IFN FTKI10!FTKL10,<
MOVE P3,UNICHN##(U) ;CHAN (FOR MAPIO)
SETZ P1, ;SAY FIRST CALL
MOVE T2,DEVDMP##(F) ;GET IOWD
ADDI T2,(R) ;RELOCATE IT
HLL T2,DEVDMP##(F) ;LH MIGHT HAVE INCR'D
PUSHJ P,MAPIO## ;RELOCATE THE IOWD
JRST [SETZM DEVDMP##(F)
JRST DINEFC] ;NO LOW-CORE BLOCKS
SETZM (P1) ;TERMINATE LIST
>
MOVE T1,UNIKON##(U) ;LOC OF KDB
MOVE T1,KONIOC##(T1) ;INITIAL CONTROL WD ADDR
MOVE T2,CHB22B##(P3) ;RH20?
TLNE T2,CP.RH2##
TLO P2,RH2JMP## ;YES, MAKE IT A JUMP
MOVEM P2,(T1) ;POINT ICWA AT CORE-BLOCK
PUSHJ P,STOTAC## ;TELL USER ICWA
IFN FTKL10,<
PUSHJ P,CSDMP## ;SWEEP CACHE
>
JRST CPOPJ1## ;AND TAKE GOOD RETURN
;STILL IN FTDHIA CONDITIONAL
;HERE TO RETURN A CHAN PROGRAM
DIARCP: JUMPE F,CPOPJ1## ;NOTHING TO DO IF NO DDB
AOS (P)
DIARCX: SKIPN DEVDMP##(F) ;NOTHING TO DO IF NO IOWD
POPJ P,
SETZM DEVDMP##(F) ;NOW NO IOWD
IFN FTKI10!FTKL10,<
MOVE T1,@KONIOC##(P3) ;GET LOC OF CORE-BLOCK
PJRST RTNIOW## ;RETURN THE SPACE
>
IFN FTKA10,<
POPJ P,
>
;HERE TO TELL USER FINAL CHAN STATS
DIAGCS: JUMPE F,CPOPJ1## ;NOTHING TO DO IF NO DDB
MOVE P2,KONIOC##(P3) ;GET ICWA
CAILE P1,4 ;ASKING FOR TOO MUCH?
MOVEI P1,4 ;YES, MAX HE CAN GET
DIAGC1: MOVE T1,(P2) ;GET A WORD FROM LOW CORE
PUSHJ P,PUTWD1## ;TELL USER
SOJLE P1,CPOPJ1## ;DONE IF 0
AOJA P2,DIAGC1 ;GET NEXT WORD
;STILL IN FTDHIA CONDITIONAL
;HERE TO TEST IF IO IS GOING ON A CHANNEL
;RETURNS NON-SKIP IF YES, SKIPS IF CHAN IS NOW IDLE
WINDWN: DSKOFF
SKIPL @DIACHN## ;#IO XFER HAPPENING?
JRST WINDW2 ;#YES, NON-SKIP
MOVE T2,U ;#NO, TEST UNITS FOR POSITIONING
WINDW1: MOVE T1,UNISTS##(T2) ;#UNIT SEEKING?
CAIE T1,PCOD##
CAIN T1,SCOD##
JRST WINDW2 ;#YES, NON-SKIP
HLR T2,UNICHN##(T2) ;#NO, TEST NEXT UNIT
CAME T2,U ;#BACK WHERE WE STARTED?
JRST WINDW1 ;#NO
AOS (P) ;#YES, ALL IS NOW QUIET
WINDW2: DSKON
POPJ P,
;ROUTINE TO START UP IO ON A CHANNEL AGAIN
;CALLED ON PI7 SINCE CANT START IO FOR DIFFERENT JOB ON UUO LEVEL
CRNKUP: PUSHJ P,SAVE1##
MOVE U,T1
MOVE P1,DIACHN## ;POINT P1 AT CHANNEL
SETZM @DIACHN## ;SO ON-CYLINDER UNITS WONT START IO
PUSH P,U
CRNKU1: HLR U,UNICHN##(U) ;NEXT UNIT
DSKOFF
MOVE T1,UNISTS##(U) ;#DOES IT WANT POSITIONING?
CAIN T1,PWCOD##
PUSHJ P,CRNPOS ;#YES, START THE SEEK GOING
CAME U,(P) ;#BACK WHERE WE STARTED?
JRST CRNKU1 ;#NO, TEST NEXT UNIT
POP P,(P) ;#YES, REMOVE JUNK FROM PDL
MOVE J,UNIKON##(U) ;#GO START AN XFER IF CHNQUE NOT EMPTY
DSKOFF ;#FIGHT RACE
SKIPL KONTAB##(J) ;#DID A SEEK FINISH?
PJRST PIKTRX ;#NO, START IO IF CHNQUE NOT EMPTY
DSKON ;#YES, FORGET IT
POPJ P, ;# SINCE IO IS ALREADY GOING
;STILL IN FTDHIA CONDITIONAL
;HERE TO GET KONTROLLER/UNIT
DIAKUN: POP P,(P) ;REMOVE SAVED T1
PUSHJ P,GETWD1## ;GET DEVICE NAME
PUSHJ P,DEVSRG## ;FIND IT
IFE FTDUAL,<
JRST ILLUNI ;NO SUCH UNIT
>
IFN FTDUAL,<
JRST TRYALT
>
MOVE T2,DEVMOD(F)
TLNE T2,DVMTA ;IS IT A MAG TAPE?
JRST DIAMTA## ;YES, LET TAPUUO DO IT
CAIE F,DSKDDB## ;NO, SOME FLAVOR OF DISK
MOVE T1,DEVNAM(F)
PUSHJ P,MSKUNI## ;SET UP A MASK
PUSHJ P,SRUNI## ;FIND UDB
JRST ILLUNI ;NOT A DISK UNIT
JFCL
DIAKU1: MOVE T1,UNIKON##(U) ;KONTROLLER
MOVE T2,KONREG##(T1)
ADDI T2,KONEBK##(T1) ;POINT TO FIRST I/O INST IN KDB
LDB T2,[POINT 7,(T2),9] ;GET KONTROLLER CODE
LDB T1,UNYPUN## ;UNIT NUMBER
LSH T1,3 ;POSITION IT
LSH T2,2 ;POSITION KONTROLLER DEVICE CODE
HRL T1,T2
AOS (P) ;SET FOR SKIP-RETURN
PJRST STOTAC## ;TELL USER KON,,UNIT AND RETURN
IFN FTDUAL,<
;HERE ON NO UNIT, TEST SECOND UNITS
TRYALT: HLRZ T2,SYSUNI## ;START AT BEGINNING
TRYAL1: SKIPLE T3,UNI2ND##(T2) ;IS THERE A SECOND UNIT?
CAME T1,UNINAM##(T3) ;YES, RIGHT ONE?
CAIA
JRST TRYAL2 ;WE FOUND IT
HLRZ T2,UNISYS##(T2) ;NO MATCH, TRY NEXT
JUMPN T2,TRYAL1
PJRST ILLUNI ;NO SUCH UNIT, ERROR
TRYAL2: MOVE U,T3 ;POINT U AT SECOND UNIT
JRST DIAKU1 ;AND TELL HIM WHAT IT IS
>
ERCODE DIANPV,1 ;NOT ENOUGH PRIVS
ERCODE WRONGN,2 ;ILLEGAL NUMBER OF ARGS
ERCODE ILLKON,3 ;ILLEGAL CONTROLLER NUMBER
ERCODE ILLUNI,4 ;ILLEGAL UNIT NUMBER
ERCODE DIUNAA,5 ;SOME UNITS ALREADY ASSIGNED
ERCODE UNNDMD,6 ;UNIT NOT IN DIAG MODE
ERCODE UNAAJB,7 ;UNIT ASS'D TO ANOTHER JOB
ERCODE DINEFC,10 ;NOT ENOUGH FREE CORE
ERCODE NOASUN,11 ;NO ASSIGNED UNITS
ERCODE IOWCPB,12 ;IOWD CROSSES PAGE BOUNDARY
ERCODE BADFNC,13
ERCODE DIAVRT,14
> ;END FTDHIA
SUBTTL MONITOR MODE IO ROUTINES
;SUBROUTINE TO COPY POINTERS FROM DDB TO MONITOR BUFFER
;ENTER WITH T1=AOBJN WORD FOR ENTIRE MONITOR BUFFER
;EXIT CPOPJ IF MON BUF IS FULL AND THERE ARE MORE POINTERS IN DDB
;EXIT CPOPJ1 NORMALLY, WITH T1=LOC OF LAST PNTR STORED
;PRESERVES T4
DD2MN:: MOVE T3,UNISTR##(U) ;GET ADDR OF STR DATA BLOCK
SETO T2, ;CREATE MASK FOR
LDB T2,STYCLP##(T3) ;CLUSTER POINTER
MOVEM T2,MSKLOC ;STORE IT FOR LATER
MOVSI T2,DEPLPC## ;LAST POINTER IN CORE BIT
ANDCAM T2,DEVLPC##(F) ;CLEAR THE BIT IF IT WAS ON
LDB T2,DEYRLC## ;CURRENT POINTER LOC
HRLS T2
ADD T1,T2 ;UPDATE AOBJN WORD
MOVSI T2,MPTRLN## ;LENGTH OF A BUNCH OF POINTERS
HRRI T2,DEVRB1##(F) ;LOC OF 1ST POINTER
HRRM T2,DEVRET(F) ;SET DEVRET=DEVRB1
SKIPE (T2) ;FIRST POINTER EMPTY
JUMPGE T1,CPOPJ## ;NO, POPJ IF NO SLOTS
DD2MN2: SKIPN T3,(T2) ;GET A POINTER FROM DDB
SOJA T1,CPOPJ1## ;ALL POINTERS COPIED - RETURN
EXCH T3,(T1) ;STUFF IT IN MON BUF
JUMPE T3,DD2MN3 ;IF OLD PNTR=0, OK
XOR T3,(T1) ;XOR RIB & MON BUF
TDNE T3,MSKLOC ;IF PNTRS EQUAL, SKIP
STOPCD .+1,DEBUG,PDA, ;++POINTERS WITH DIFFERENT ADDRESSES
DD2MN3: SETZM (T2) ;ZERO THE WORD IN DDB
AOBJP T2,CPOPJ1## ;THROUGH WHEN DDB RUNS OUT
AOBJN T1,DD2MN2 ;DO ANOTHER
SKIPN (T2) ;MON BUF FULL. MORE POINTERS?
SOJA T1,CPOPJ1## ;NO, TAKE SKIP-RETURN
TWOLOC: POPJ P,2 ;MON BUF FULL AND MORE TO GO
$LOW
MSKLOC: Z ;WORD FOR MASK FOR CLUSTER POINTER
$HIGH
;SUBROUTINE TO READ RIB POINTERS INTO CORE
;GETS A MONITOR BUFFER AND READS THE RIB INTO IT
;RETURNS T3=0 IF OK, T3=-1 IF ERROR; AND T1=AOBJN WORD FOR THE POINTERS
PTRGET::PUSHJ P,BUFRIB ;GET MON BUF, READ RIB INTO IT
SKIPA T3,[-1] ;RIB ERROR - RETURN T3=-1
SETZ T3, ;OK - T3=0
PJRST SPTRW ;SET T1=AOBJN WORD AND RETURN
;SUBROUTINE TO COPY CURRENT POINTERS FROM MON BUF TO DDB
;ENTER WITH RIB IN MON BUF, T1=AOBJN WORD FOR POINTERS
;EXIT WITH POINTERS COPIED INTO DDB, DEYRLC UPDATED
PTRCPY::LDB T2,DEYRLC## ;PREVIOUS POINTER RELATIVE LOC
SKIPN DEVRB2(F) ;DEVRB2=0?(PNTR CAME FROM A.T. IF YES)
SKIPA T2,TWOLOC ;YES. START AT 3RD ENTRY IN PNTRS
ADDI T2,PTRLEN## ;+LENGTH OF A BUNCH OF POINTERS
HRLS T2
ADD T1,T2 ;UPDATE AOBJN WORD (WAS FOR WHOLE POINTER AREA)
;AND FALL INTO PTRBLT
;SUBROUTINE TO BLT POINTERS FROM MONITOR BUFFER TO DDB
;ENTER WITH T1=AOBJN WORD FOR CURRENT POINTERS IN MONITOR BUFFER
PTRBLT::MOVE T3,T1 ;SAVE CURRENT AOBJN WRD
PUSHJ P,SPTRW ;GET AOBJN WRD FOR WHOLE MON BUF
HRRZ T2,T3 ;CURRENT PNTR LOC
SUBI T2,(T1) ;-ORIGINAL PNTR LOC=NEW DEYRLC
MOVE T1,T3 ;RESTORE CURRENT AOBJN WORD
;SUBROUTINE TO COPY POINTERS FROM MON BUF TO DDB
; STORE DEVRLC WITHOUT COMPUTING
;ENTER T1=AOBJN WORD FOR POINTERS
PTRBL1::DPB T2,DEYRLC## ;SAVE IN DDB
HLLM T1,DEVRSU##(F) ;-NO OF PNTRS LEFT
MOVSI T2,MPTRLN##
HRRI T2,DEVRB1##(F) ;AOBJN WORD FOR DDB
HRRM T2,DEVRET##(F)
PTRBL2: SKIPA T3,(T1) ;NEXT POINTER
PTRBL3: SETZ T3, ;POINTERS DONE-ZERO
MOVEM T3,(T2) ;SAVE IN DDB
AOBJP T2,PTRBL4 ;COUNT DDB WORD
AOBJN T1,PTRBL2 ;GET NEXT POINTER
JRST PTRBL3 ;THOUGH WITH MON BUF
PTRBL4: MOVE T3,DEVLPC##(F) ;GET LAST POINTER IN CORE WORD
AOBJN T1,PTRBL5 ;JUMP IF MORE POINTER SLOTS IN RIB
PUSHJ P,GTLPT## ;GET LAST RIB POINTER
SKIPE T2 ; DONT LIGHT DEPLPC IF 0
TLOA T3,DEPLPC## ;NO MORE LEFT, LAST POINTER IS IN CORE
PTRBL5: TLZ T3,DEPLPC## ;LAST IS NOT IN CORE
MOVEM T3,DEVLPC##(F) ;RESTORE THE FLAG
IFE FTDSIM,<
POPJ P, ;AND RETURN
>
IFN FTDSIM,<
;IF ACYWCT .GTR. 0 ,ACPSBC=1, THEN IF A WRITER CHANGES A CHECKSUM
; AND THE NEW PNTR ISNT IN SOME OTHER WRITER'S DDB, IF THE 2ND WRITER
; THEN CHANGES THE CHECKSUM OF THAT PNTR AND CLOSES BEFORE THE 1ST WRITER
; (ORIGINAL ALLOCATER), THEN WHEN THE 1ST WRITER CLOSE WE HAVE A
; CHECKSUM ERROR IN THE FILE.
;HENCE, WE HAVE TO SCAN DDBS FOR A WRITER WITH THIS PNTR, USE THE PNTER
; FROM THE FOUND DDB.
;STILL IN FTDSIM CONDITIONAL
HRRZ T1,DEVACC##(F) ;LOC OF A.T.
JUMPE T1,CPOPJ##
LDB T2,ACYWCT## ;ARE THERE MULTIPLE WRITERS?
SOJLE T2,CPOPJ##
MOVE T2,ACCSBC##(T1) ;YES, HAS CHECKSUM CHANGED?
TRNN T2,ACPSBC##
JRST CPOPJ## ;NO
MOVE T3,UNISTR##(U) ;YES, SET UP A MASK FOR
SETO T2, ; ADDR PORTION OF RETRIEVAL POINTERS
LDB T2,STYCLP##(T3)
MOVEM T2,MSKLOC ;AND SAVE IT AWAY
PUSHJ P,FNDDDB ;FIND A WRITING DDB FOR THE FILE
POPJ P, ;NONE (DDB COULD BE IN EXTRIB)
PTRBL6: HRRZ T3,DEVUNI##(T2) ;SAME UNIT?
CAIE T3,(U)
JRST PTRBLB ;NO, TRY NEXT DDB
MOVEI T3,DEVRB1##(F) ;YES, SET AN AOBJN WORD FOR OUR DDB
HRLI T3,MPTRLN##
PTRBL7: MOVEI T1,DEVRB1##(T2) ;SET AOBJN WORD FOR POINTERS
HRLI T1,MPTRLN## ; FOR THE FOUND-DDB
PTRBL8: MOVE T4,(T1) ;GET A DDB POINTER
JUMPE T4,PTRBL9 ;KEEP GOING IF 0 (PNTR MIGHT BE IN DEVRBN)
XOR T4,(T3) ;IS IT IN OUR DDB?
TDNN T4,MSKLOC
JRST PTRBLA ;YES, COPY PNTRS TO OUR DDB
PTRBL9: AOBJN T1,PTRBL8 ;NO, TRY NEXT PNTR
SKIPE 1(T3) ;ANY MORE PNTRS IN OUR DDB?
AOBJN T3,PTRBL7 ;YES, TRY TO MATCH NEXT POINTER
JRST PTRBLB ;NO MATCH - TRY NEXT DDB
;HERE WHEN WE FOUND A MATCH BETWEEN THE WRITING DDB AND OUR DDB
PTRBLA: MOVE T4,(T1) ;GET PNTR FROM THE FOUND DDB
MOVEM T4,(T3) ;AND STUFF IT IN OURS
AOBJP T3,CPOPJ## ;THROUGH IF OUR DDB FULL
SKIPN (T3) ; OR IF NO MORE PNTRS IN OUR DDB
JRST CPOPJ##
SKIPE 1(T1) ;IS THERE ANOTHER PNTR IN FOUND DDB?
AOBJN T1,PTRBLA ;YES, COPY INTO OUR DDB
PTRBLB: PUSHJ P,FNDDDN ;NO, FIND ANOTHER WRITING DDB
POPJ P, ;NO - DONE
JRST PTRBL6 ;FOUND ANOTHER - TEST ITS PNTRS
>
;SUBROUTINE TO SET AN AOBJN WORD FOR POINTERS IN THE MONITOR BUFFER
;ENTER WITH LOC OF MON BUF IN DEVMBF, EXIT WITH T1=AOBJN WORD
;T2-T4 RESPECTED
SPTRW:: HRRZ T1,DEVMBF##(F) ;LOC OF MON BUF (-1)
ADD T1,RIBFIR##+1(T1) ;AOBJN WORD (-1)
AOJA T1,CPOPJ## ;MAKE REAL AOBJN WORD AND RETURN
;SUBROUTINE TO GET A MONITOR BUFFER
;RETURNS WITH DEVMBF=T1=AN IOWD FOR THE BUFFER
IFN FTDSIM,<
;PRESERVES T2
GTMB2: PUSH P,T2 ;SAVE T2
PUSHJ P,GTMNBF ;GET THE MON BUF
PJRST T2POPJ## ;RESTORE T2 AND RETURN
>
GTMNB:
;(SAME AS GTMNBF, PRESERVES T3)
GTMNBF::SKIPE T1,DEVMBF##(F) ;JOB ALREADY HAVE MONITOR BUFFER?
STOPCD CPOPJ##,DEBUG,AHB, ;++ALREADY HAVE BUFFER
GTMNB0::SETZ T4, ;DONT ADJUST MQREQ THE FIRST TIME
GTMNB4: PUSHJ P,MQWAIT## ;GET A MON BUF, WAIT IF NONE FREE
ADDM T4,MQREQ## ;ADJUST MQREQ IF THIS IS SECOND TRY TO GET THE BUFFER
GTMNB1::HRRZ T1,BUFLST## ;GOT ONE - SCAN LIST FOR A FREE BUF
NOSCHEDULE
GTMNB2: SKIPL T2,(T1) ;THIS BUFFER FREE?
JRST GTMNB3 ;YES. MARK IT
HRRZ T1,T2 ;NO. STEP TO NEXT IN LIST
JUMPN T1,GTMNB2 ;TRY THIS ONE
MOVNI T4,1 ;ADJUST MQREQ AFTER NEXT CALL TO MQWAIT
;(CANT SOS MQREQ NOW, AS THAT COULD MAKE MQREQ=-1)
JRST GTMNB4 ;COUNTS ARE OFF - REQUEUE THIS REQUEST
;(CAN HAPPEN IF SCHED GIVES MON BUF TO A USER WHO IS
;SWAPPED, AND NEXT MON BUF IS RELEASED- MQFRE1 IS OFF)
GTMNB3: HRROS (T1) ;MARK BUFFER AS TAKEN
SCHEDULE
HRLI T1,MBLKSZ## ;SET T1 AS AN IOWD FOR THE BUFFER
MOVEM T1,DEVMBF##(F) ;SAVE IOWD IN DDB
LDB T2,PJOBN## ;JOB NUMBER
MOVEM T2,MBFJOB##(T1) ;SAVE IN BUFFER
PJRST MBFCNT ;COUNT CURRENT NO OF MON BUFS AND RETURN
GVMNBX::
IFN FTDSIM,<
TLNE S,IOSDA ;HAVE DA?
POPJ P, ;YES, KEEP MON BUF
>
;SUBROUTINE TO RTURN MONITOR BUFFER
;ENTER WITH DEVMBF(F)= LOC OF MONITOR BUFFER
GVMNB0::SKIPN T1,DEVMBF##(F) ;LOC OF MON BUFFER
STOPCD CPOPJ##,DEBUG,DHB, ;++DON'T HAVE BUFFER
;SUBROUTINE TO RETURN THE MONITOR BUFFER
;ENTER WITH T1 = ADDRESS OF THE BUFFER
;(PRESERVES T3 ON THE SLY)
GVMNBF::HRRZS (T1) ;MARK BUFFER FREE
HRLZS MBFJOB##(T1) ;SAVE LAST USER
PUSHJ P,MQFREE## ;BUMP COUNT OF FREE BUFFERS
TLZ S,IOSRIB ;NO LONGER HAVE RIB IN MONITOR BUFFER
SETZM DEVMBF##(F) ;NO LONGER HAVE MONITOR BUFFER
;COUNT THE NUMBER OF FREE MONITOR BUFFERS
;NEEDED SINCE IF 2 JOBS GIVE UP 2 DIFFERENT MON BUFS BEFORE THE SCHEDULER HAS LOOKED AT MQREQ
;ONLY 1 JOB WILL BE GIVEN A MON BUF, AND THE OTHER BUF WILL NOT BE USED AGAIN
MBFCNT: SKIPE DINITF## ;ONCE-ONLY?
PJRST STRIOS ;YES--RETURN
SETOM MQFRE1## ;INITIALIZE COUNT
HRRZ T2,BUFLST## ;LOC OF 1ST MON BUF
SKIPL T2,(T2) ;FREE?
AOS MQFRE1## ;YES. INCREMENT NO OF FREE BUFS-1
HRRZS T2 ;ADR FIELD ALONE
JUMPN T2,.-3 ;TEST NEXT MON BUF
PJRST STRIOS ;RETURN WITH MQFRE1=# OF MON BUFS -1
;SUBROUTINE TO GET THE MONITOR BUFFER, READ RIB INTO IT
;RETURNS CPOPJ IF ERROR, CPOPJ1 NORMALLY, T1=IOWD
BUFRIB::PUSHJ P,GTMNBF ;GET MONITOR BUFFER
;AND FALL INTO REDRIB
;SUBROUTINE TO READ THE PRIME RIB, T1=IOWD
;RETURNS CPOPJ IF ERROR, CPOPJ1 NORMALLY
REDRIB::JSP T4,SAVUN ;PUSH U, SET UP U FOR RIB
PUSHJ P,PRMRIB ;SET UP TO READ THE PRIME RIB
SKIPN T1,DEVMBF##(F) ;GET IOWD TO MONITOR BUFFER
STOPCD RESUNI,DEBUG,MHB, ;++MUST HAVE BUFFER
PUSHJ P,MONRED ;READ THE RIB INTO MON BUF
PUSHJ P,RIBCHK ;CHECK RIB
JRST RESUNI ;RIB ERR - NON SKIP RETURN
AOS -1(P) ;GOOD RIB - SET FOR SKIP RETURN
;FALL INTO RESUNI AND RETURN
;SUBROUTINE TO RESTORE DEVUNI
;ENTER WITH CURRENT U SAVE ON PD LIST
RESUNI: POP P,U ;CURRENT U
SKIPE DEVUNI##(F) ;UNLESS UNIT(STR) WAS REMOVED
HRRM U,DEVUNI##(F) ;SAVE IN DDB
POPJ P, ;AND RETURN
;SUBROUTINE TO WRITE A RIB
WRTRIB::PUSHJ P,SAVE1##
PUSH P,U ;SAVE CURRENT UNIT
HRRZ T1,DEVACC##(F) ;LOC OF A.T.
JUMPE T1,WRTRB1
MOVE P1,ACCSTS##(T1) ;STATUS OF FILE
TRNN P1,ACPUPD ;UPDATE?
JRST WRTRB1 ;NO
MOVE T4,DEVMBF##(F) ;MON BUF
SKIPE T3,ACCWRT##(T1) ;NO OF BLOCKS IN FILE
SUBI T3,1
LSH T3,BLKLSH## ;CONVERT TO WORDS
HRRZ T2,T1
LDB T1,ACYLBS## ;SIZE OF LAST BLOCK
ADD T3,T1 ;NO OF WORDS IN FILE
MOVEM T3,RIBSIZ##+1(T4) ;SAVE IN RIB
MOVE T1,ACCALC##(T2) ;NO OF BLOCKS ALLOCATED
EXCH T1,RIBALC##+1(T4) ;SAVE IN RIB
CAMN T1,RIBALC##+1(T4) ;DID THE SIZE CHANGE?
JRST WRTRB1 ;NO
SETZ P1, ;YES, SET P1=0 AS A FLAG
PUSHJ P,RIBSAT## ;YES, WRITE CHANGED SATS (IN CASE OF A CRASH)
;HERE WITH SATS WRITTEN IF NECCESSARY
WRTRB1: LDB T2,DEYRBU## ;GET CURRENT RIB LOGICAL UNIT NUMBER
PUSHJ P,NEWUNI ;SET UP U,DEVUNI
STOPCD RESUNI,DEBUG,NXU, ;++NON X UNIT
LDB T2,DEYRBA## ;GET CURRENT RIB CLUSTER ADDRESS
LDB T3,UNYBPC## ;BLOCKS PER CLUSTER, THIS UNIT
IMUL T2,T3 ;BLOCK NUMBER IN T2
SKIPN T1,DEVMBF##(F) ;GET IOWD TO MONITOR BUFFER
STOPCD RESUNI,DEBUG,NMB, ;++NEED MONITOR BUFFER
MOVEM T2,RIBSLF##+1(T1) ;SAVE BLOCK NUMBER
MOVEI T3,CODRIB##
MOVEM T3,RIBCOD##+1(T1) ;INDICATE BLOCK IS A RIB
PUSHJ P,MONWRT ;WRITE HE MON BUF
IFN FTDMRB,<
JUMPN P1,WRTRB3 ;IF AN UPDATE FILE
SKIPL P1,DEVRIB##(F) ; AND IN EXTENDED RIB
JRST WRTRB3
PUSH P,RIBALC##+1(T1) ;SAVE RIBSIZ, RIBALC
PUSH P,RIBSIZ##+1(T1)
PUSHJ P,REDRIB ;READ THE PRIME RIB
JRST [POP P,(P) ;RIB ERROR
POP P,(P)
JRST WRTRB2]
POP P,RIBSIZ##+1(T1)
POP P,RIBALC##+1(T1) ;RESTORE RIBSIZ, RIBALC
PUSHJ P,RIBAD ; FOR USE IN CASE OF CRASH
JSP T4,RIBUN
PUSHJ P,MONWRT ;REWRITE PRIME RIB
WRTRB2: MOVEM P1,DEVRIB##(F)
PUSHJ P,RIBCUR ;REREAD EXTENDED RIB
JFCL
WRTRB3:> ;END FTDMRB
POP P,U ;RESTORE CURRENT UNIT
HRRM U,DEVUNI##(F) ;AND SAVE IN DDB
POPJ P, ;AND RETURN
;SUBROUTINE TO SET UP TO READ THE PRIME RIB
;RETURNS CPOPJ WITH DEVRIB SET UP IN DDB
PRMRIB::PUSHJ P,RIBAD ;COMPUTE ADR OF RIB
SETZM DEVRIB##(F) ;CLEAR DEVRIB FOR DPB'S
LDB T3,UNYLUN## ;GET RPIME RIB LOGICAL UNIT NUMBER
DPB T3,DEYRBU## ;DEPOSIT IN DDB
PUSH P,T2 ;SAVE T2
HRRZ T2,DEVACC##(F) ;GET ADDRESS OF A.T.
MOVE T2,ACCPT1##(T2) ;GET FIRST POINTER FOR FILE
LDB T3,STYCLP##(T4) ;GET CLUSTER ADDRESS
DPB T3,DEYRBA## ;DEPOSIT IN DDB
PJRST T2POPJ## ;RESTORE T2 AND RETURN
;SUBROUTINE TO SAVE CURRENT DEVUNI
;U CHANGED, ALL OTHER ACS RESPECTED
;SAVES THE CURRENT U ON THE PUSH DOWN LIST
;ENTER AT SAVUN TO SAVE CURRENT U, AT RIBUN JUST TO SET U FOR RIB UNIT
;CALLED WITH JSP T4
SAVUN:: PUSH P,U ;SAVE CURRENT U
RIBUN:: HLRZ U,DEVUNI##(F) ;UNIT OF RIB
HRRM U,DEVUNI##(F) ;SAVE AS CURRENT UNIT
JRST (T4) ;AND RETURN
;SUBROUTINE TO COMPUTE A RIB ADDRESS FROM THE ACCESS TABLE
;EXIT WITH T2=BLOCK NUMBER, T4=LOC OF STR DATA BLOCK
;T1 RESPECTED
RIBAD:: HRRZ T2,DEVACC##(F) ;LOC OF ACCESS TABLE
MOVE T2,ACCPT1##(T2) ;GET 1ST POINTER FROM A.T.
;SUBROUTINE TO COMPUTE A DISK ADDRESS FROM A GROUP POINTER
;ENTER WITH GROUP POINTER IN T2
;EXIT WITH T2=BLOCK NUMBER, T4=LOC OF STR DATA BLOCK
;T1 RESPECTED
GRPAD:: MOVE T4,UNISTR##(U) ;GET BYTE POINTER FOR ADDRESS OF CLUSTER
LDB T2,STYCLP##(T4) ;CLUSTER ADDRESS
LDB T3,UNYBPC## ;COMPUTE BLOCK ADDRESS (NOTE THAT
IMUL T2,T3 ; RELATIVE BLOCK 0 OF 1ST POINTER IS RIB LOC)
POPJ P, ;AND RETURN
;SUBROUTINE TO READ A UFD RIB
;ENTER WITH T2=BLOCK NUMBER
;RETURNS CPOPJ IF ERROR, CPOPJ1 NORMALLY WITH T1=IOWD, T2=BLOCK
UFDRED::MOVE T1,DEVMBF##(F) ;IOWD FOR MONITOR BUFFER
PUSHJ P,MONRED ;READ THE RIB
MOVE T4,T2 ;BLOCK NUMBER (FOR RIBERR)
JUMPN T3,RIBERR ;CHECK FOR ERRORS DETECTED IN MONRED
CAME T2,RIBSLF##+1(T1) ;CHECK BLOCK NUMBER
JRST RIBERR ;BAD
MOVEI T3,CODRIB## ;OK. CHECK RIB CODE WORD
CAME T3,RIBCOD##+1(T1)
JRST RIBERR ;BAD
HLRZ T3,RIBEXT##+1(T1) ;OK. CHECK EXT="UFD"
CAIN T3,(SIXBIT .UFD.)
SKIPA T3,RIBNAM##+1(T1) ;OK, GET PPN
JRST UFDRE2 ;NOT "UFD", CHECK IF "SFD"
CAME T3,DEVPPN(F) ;UFD, CHECK PPN
CAMN T3,DEVFIL(F)
SKIPA T3,RIBSTS##+1(T1) ;OK, GET STATUS WD
JRST RIBERR ;BAD
TROE T3,RIPABC## ;RIPABC ON?
PJRST CPOPJ1## ;YES
MOVEM T3,RIBSTS##+1(T1) ;NO, LIGHT IT (UFD WRITTEN BY OLDER MON)
PUSHJ P,MONWRT ;REWRITE THE UFD RIB
JRST CPOPJ1## ;OK, RETURN
;HERE IF EXTENSION ISN'T "UFD"
UFDRE2:
IFN FTSFD,<
CAIE T3,(SIXBIT .SFD.) ;IS IT "SFD"?
JRST RIBERR ;NO, BAD RIB
HRRZ T3,DEVSFD##(F) ;YES, GET NAME OF SFD
MOVE T3,NMBNAM##(T3) ; FROM ITS ACCESS TABLE
CAMN T3,RIBNAM##+1(T1) ;CHECK NAME IN RIB
JRST CPOPJ1## ;OK, SKIP-RETURN
>
JRST RIBERR ;ERROR
;SUBROUTINE TO READ A RIB INTO THE MONITOR BUFFER
;ENTER WITH T2 = DISK ADDRESS OF RIB
;RETURNS CPOPJ IF ERROR, CPOPJ1 NORMALLY
RIBRED::MOVE T1,DEVMBF(F) ;T1=IOWD FOR THE DATA
PUSHJ P,MONRED ;READ THE RIB
;AND FALL INTO RIBCHK
;SUBROUTINE TO CHECK THE VALIDITY OF A RIB
;ENTER WITH T1 = IOWD, T2 = DISK ADDRESS
;RETURNS CPOPJ IF ERROR, CPOPJ1 NORMALLY, T1=IOWD
RIBCHK: MOVE T4,T2 ;BLOCK (FOR RIBERR)
TRNN T3,IOIMPM+IODTER+IODERR+IOBKTL ;ERROR BIT FORM READ?
CAME T2,RIBSLF##+1(T1) ;BLOCK NUMBER IN RIB RIGHT?
JRST RIBERR ;NO. ERROR
MOVEI T2,CODRIB## ;YES. IS THIS BLOCK A RIB?
CAME T2,RIBCOD##+1(T1)
JRST RIBERR ;NO. ERROR
MOVE T2,DEVFIL(F) ;YES. FILE NAME
CAME T2,RIBNAM##+1(T1) ;MATCH NAME IN RIB?
JRST RIBERR ;NO. ERROR
HLLZ T2,DEVEXT(F) ;YES. FILE EXTENSION
HLLZ T3,RIBEXT##+1(T1) ;EXT STORED IN RIB
CAME T2,T3 ;MATCH?
JRST RIBERR ;NO. ERROR
HRRZ T3,RIBFIR##+1(T1) ;YES. REL LOC OF 1ST PNTR IN RIB
CAILE T3,BLKSIZ##-2 ;LEGAL?
JRST RIBERR ;NO. ERROR
MOVE T2,DEVPPN(F) ;YES. PPN OF FILE
CAME T2,RIBPPN##+1(T1) ;MATCH PPN IN RIB?
JRST RIBERR ;NO,RIB ERR
TLNN F,RENMB ;DON'T CHECK RIBUFD ON RENAME
; ONLY ON RENAME FOR EXTENDED RIBS
PUSHJ P,GTUFR## ;YES, COMPUTE RIBUFD WORD
PJRST CPOPJ1## ;NOT YET SET UP, ASSUME OK
CAMN T2,RIBUFD##+1(T1) ;MATCH?
PJRST CPOPJ1## ;YES, OK RETURN
ADDI T2,1 ;THE OLD REFRESHER PUT THIS NUMBER 1 TOO LOW
CAMN T2,RIBUFD##+1(T1) ; SO ALLOW THAT CASE TO WIN
PJRST CPOPJ1##
MOVE T4,RIBSLF##+1(T1) ;RESTORE BLOCK NUMBER
;HERE WHEN THE RIB INFORMATION IS NOT CORRECT.
RIBERR::MOVEI T1,UNPRER## ;SET UP A BIT
ADDM T1,UNIMCT##(U) ;COUNT IN SOFTWARE-ERROR WORD FOR UNIT
TDO S,[XWD IOSSCE##,IOIMPM] ;LIGHT ERROR BITS
IFN FTDAEM,<
SKIPGE UNIECT##(U) ;IF NOT IN ERROR RECOVERY
MOVEM T4,UNIHBN##(U) ; SAVE BLOCK FOR DAEMON
MOVEI T1,.ERDPE ;SAY DISK ERROR TO DAEMON
HRL T1,F ;PUT DDB ADDR IN LH FOR DAEMON
PUSHJ P,DAEERR## ;AND CALL DAEMON LOG ERROR
>
IFN FTDBAD,<
PUSHJ P,SAVE2## ;SLFND USES P-ACS
MOVE T1,UNISTR##(U) ;STR LOC
MOVE T1,STRNAM##(T1) ;STR NAME
MOVE P1,U ;SRSTR WIPES U
PUSHJ P,SRSTR## ;FIND STR NUMBER
JRST STRIOS ;WHAT???
MOVE U,P1 ;RESTORE U
HRRZ T1,T4 ;STR NUMBER
MOVE P2,SYSSRC## ;THIS STR IN SYS SEARCH-LIST?
PUSHJ P,SLFNA##
PJRST STRIOS ;NO
AOS T1,RIBTOT## ;YES, COUNT AN ERROR
CAMG T1,RIBECT## ;ABOVE THE THRESHOLD?
PJRST STRIOS ;NO, DONT TELL OPR
PUSH P,U ;YES, INFORM THE OPR
HRRZ U,OPRLDB## ;WHERE TO TYPE THE MESSAGE
PUSHJ P,INLMES##
ASCIZ /
RIB ERROR ON /
PUSHJ P,PRTDDB## ;"STR:NAM.EXT[PPN]"
PUSHJ P,PCRLF##
POP P,U ;RESTORE U
> ;END FTDBAD CONDITIONAL
PJRST STRIOS ;STORE S AND RETURN
;SUBROUTINE TO WRITE A BLOCK (OR SERIES OF BLOCKS)
;ENTER WITH T1= IOWD T2=BLOCK NUMBER
MONWRT::PUSH P,S ;SAVE S
TLO S,IO ;INDICATE WRITING
PJRST MONIO ;DO IO
;SUBROUTINE TO READ A BLOCK (OR SERIES OF BLOCKS)
;ENTER WITH T1 = IOWD FOR THE DATA T2= BLOCK NUMBER
;RETURNS WITH T2 = BLOCK NUMBER READ, T1 = IOWD,
; AND T3=ERROR BITS SET ON THIS OPERATION
;INSTEAD OF ERROR RETURN - S SET WITH ERROR BITS IN RH. U AND F SET UP
MONRED::PUSH P,S ;SAVE S
TLZ S,IO ;INDICATE READING
MONIO: TRNE S,IOACT ;IO IN PROGRESS?
STOPCD .,STOP,IIP, ;++ IO IN PROGRESS - ERROR
SKIPGE T2
STOPCD .,STOP,BIN, ;IO TO A NEGATIVE BLOCK #
MOVEI T3,IODTER ;SET FOR ERROR
CAML T2,UNIBPU##(U) ;REQUESTED BLOCK ABOVE HIGHEST ON UNIT?
JRST MONIO2 ;YES. ERROR RETURN
TRZ S,IOIMPM+IOBKTL+IODTER+IODERR ;ZERO THE TEMP ERR BITS
TLO S,IOSMON ;INDICATE MONITOR IO
MOVEM S,DEVIOS(F) ;STORE NEW S IN DDB
PUSH P,DEVDMP##(F) ;SAVE CURRENT DEVDMP
MOVEM T1,DEVDMP##(F) ;IOWD TO READ THE DATA
PUSH P,DEVBLK##(F) ;SAVE CURRENT BLOCK NUMBER
MOVEM T2,DEVBLK##(F) ;BLOCK WE WANT TO READ
PUSHJ P,UNICHK ;MAKE SURE UNIT IS OKAY.
JRST MONIO1 ;NO GOOD - WRITING FROM MON-BUF, UNIT OFF LINE
HRRZ J,UNIKON##(U) ;LOC OF KONTROLLER DATA BLOCK
PUSHJ P,UUOPWQ ;GO PUT IN QUEUE STRUCTURE
PUSHJ P,PWAIT1 ;WAIT T1LL DATA IS IN
MONIO1: ANDI S,IOIMPM+IODERR+IODTER+IOBKTL ;GET ERROR BITS FROM S
MOVE T3,S ;AND SAVE IN T3
MOVE T2,DEVBLK##(F) ;RESTORE BLOCK NO. TO T2
MOVE T1,DEVDMP##(F) ;AND IOWD TO T1
IFN FTDUAL,<
HRRZ U,DEVUNI##(F) ;RESET U
>
POP P,DEVBLK##(F) ;RESTORE THE CHANGED STUFF TO THE DDB
POP P,DEVDMP##(F)
MONIO2: POP P,S
OR S,T3 ;SET ANY ERROR BITS INTO S
STRIOS::MOVEM S,DEVIOS(F)
POPJ P, ;AND RETURN TO CALLER
IFN FTDHNG,<
;HERE IF THE MONITOR CALLS FILSER ON A HUNG DEVICE
HNGDSK::MOVE U,DEVUNI##(F) ;UNIT DB
JUMPE U,CPOPJ## ;JUST IN CASE F/S HAS BEEN JERKED OUT
IFN FTDUAL,<
HRRZ U,DEVCUR##(F)
>
IFN FTDHIA,<
LDB J,PJOBN## ;SET UP J
PUSHJ P,FNDPDB## ;SET UP W
JRST HNGDSD ;MUST BE SWAPPER
CAME F,.PDDIA##(W) ;IN DIAG?
JRST HNGDSD ;NO
SOSGE DEVDIA##(F) ;YES, HUNG TIME GONE?
JRST DIACLR ;YES, RESTART THE IO ON THE CHANNEL
AOS (P) ;NO, SET FOR OK RETURN
MOVEI T1,7 ;SET FOR 2-MINUTE INTERVALS
DPB T1,PDVTIM##
PJRST SETHNG## ;RESET HUNG TIMER AND KEEP ON GOING
HNGDSD:>
IFN FTVM,<
SKIPL DEVSWP(F) ;SWAPPER?
JRST HNGDSX
SETZ J, ;YES, INDICATE "UUO" LEVEL
PUSHJ P,SWPSCN## ;START IO AGAIN
AOSA (P)
PJRST CPOPJ1## ;OK, NO MSG
PJRST SWPHNG##
HNGDSX:>
HRRZ J,UNIKON##(U) ;KONTROLLER DB
SKIPE T1,UNISTS##(U) ;STATUS OF UNIT
CAIN T1,TWCOD## ;TW OR IDLE?
AOSE @KONCHN##(J) ;YES. IS CHAN FREE?
JRST HNGDS0 ;NO
AOS UNIHNG##(U) ;COUNT TOTAL NUMBER OF HUNG TIMEOUTS
AOS (P) ;SET FOR NO ERROR MESSAGE (WE RECOVERED)
HRRZ P1,KONCHN##(J) ;SET P1 TO CHAN DB LOC
JUMPE T1,UUOTWQ ;GO START I/O IF UNIT IS IDLE
PJRST PIKTRN ;TW- REMOVE FILE FROM CHAN Q AND START IO
;HERE IF UNIT NOT IN IDLE OR TW, OR CHAN NOT FREE
HNGDS0: CAIGE T1,OWCOD## ;UNIT IN OPERATOR WAIT?
JRST HNGDS7 ;NO, TEST FOR T OR P STATE
SKIPL KONPOS##(J) ;DOES KONTROLLER POSITION?
JRST HNGD0A ;YES
SETZM UNISTS##(U) ;NO, HENCE NO FREE INTERRUPT
CAIN T1,OWCOD## ; RETRY OPERATION IF OW
PUSHJ P,UUOPWQ ; SINCE UNIT MIGHT HAVE BEEN FIXED
PJRST CPOPJ1## ;AND RETURN WITH NO MESSAGE
HNGD0A:
LDB J,PJOBN## ;YES, GET JOB NUMBER
JUMPE J,CPOPJ1## ;SWAP I/O
PUSH P,F ;SAVE FILE IN QUESTION
HRRZ F,TTYTAB##(J) ;HUNG JOB'S TTY DDB
PUSHJ P,CTLJBD## ;FIND THE CONTROLLING JOB IF THERE IS ONE
POP P,F ;RESTORE POINTER TO DSK DDB
HRRZ U,DEVUNI##(F) ;RESTORE UNIT
SKIPG T1 ;SKIP IF THE CURRENT JOB IS CONTROLLED
TDZA T2,T2 ;IF NOT CONTROLLED, THE CONTROLLING JOB ISN'T SWAPPED
MOVE T2,JBTSTS##(T1) ;JOB STATUS OF THE CONTROLLING JOB
MOVE T1,JBTSTS##(J) ;STATUS OF JOB
TLNN T2,SWP ;IS IT SWAPPED OUT?
; IF YES, HUNG DEVICE MESSAGE (ELSE THIS IS
; DEADLY EMBRACES
TLNE T1,CNTRLC ;^C TYPED?
JRST HNGDS3 ;YES, GIVE HUNG DEVICE
TLNE S,IOSHMS ;NO, MESSAGE ALREADY TYPED?
JRST HNGDS2 ;YES, DONT TYPE IT AGAIN
PUSH P,F ;NO, SAVE F
PUSHJ P,TTYFND## ;FIND USER'S TTY
PUSHJ P,INLMES## ;TYPE THE MESSAGE:
ASCIZ /
DISK IS OFF-LINE, WAITING FOR OPERATOR ACTION
TYPE ^C TO GET HUNG MESSAGE (IN 15 SECONDS).
DON'T TYPE ANYTHING TO WAIT FOR THE OPERATOR TO FIX THE DRIVE.
/
PUSHJ P,FLSDR## ;DOES HE HAVE A RESOURCE?
JRST HNGDS1 ;NO
PUSHJ P,INLMES## ;YES, TELL HIM HE'S HUNG THE SYSTEM
ASCIZ /(THE SYSTEM WILL DO NO USEFUL WORK UNTIL THE
DRIVE IS FIXED OR YOU TYPE ^C)
/
HNGDS1: POP P,F ;RESTORE F
HNGDS2: MOVE S,DEVIOS(F) ;S BITS
TLO S,IOSHMS ;BIT INDICATING MSG WAS TYPED
PUSHJ P,STOIOS## ;SAVE S, RESET HUNG TIME
PJRST CPOPJ1## ;RETURN WITH NO ERROR MESSAGE
;HERE IF ^C WAS TYPED - FINISH OUT THE OPERATION
HNGDS3: DSKOFF
PUSHJ P,RETRES ;GIVE UP MONITOR RESOURCES
LDB T1,DEYCOD## ;STATE OF FILE (FORMER STATE OF UNIT)
CAIE T1,TCOD## ;TRANSFER
CAIN T1,PCOD## ; OR POSITION?
JRST HNGDS6 ;YES
HRRZ U,DEVUNI##(F) ;NO, IN A QUEUE
PUSHJ P,SAVE2## ;SAVE P1,P2
MOVEI P2,MINDVQ##(T3) ;P2=1ST POSSIBLE PRED
;HERE TO FIND ACTUAL PREDECESSOR
HNGDS5: HLRZ P1,DEVQUE##(P2) ;FILE THIS ONE POINTS TO
CAIN P1,(F) ;POINTING TO OUR FILE?
JRST HNGD6A ;YES, GO UNQUEUE
SKIPE P2,P1 ;NO, LOOK AT NEXT IN Q
JRST HNGDS5
MOVEI T1,O2COD## ;COULDNT FIND PREDECESSOR (SYSTEM ERROR?)
MOVEM T1,UNISTS##(U) ;SET UNIT INTO OPR WAIT
DSKON ;RESTORE PIS
POPJ P, ;AND RETURN
HNGD6A: MOVE J,UNIKON##(U)
SKIPA P1,KONCHN##(J)
;HERE IF FILE WAS IN POSITION OR TRANSFER STATE
HNGDS6: AOSA UNISTS##(U) ;CHANGE UNIT STATE TO O (FROM OW)
;(INDICATING DONT START IO ON FREE INTERRUPT)
PUSHJ P,UNQUEX ;UNQUEUE THE REQUEST
DSKON
POPJ P, ;AND GIVE ERROR MESSAGE RETURN
;STILL IN FTDHNG CONDITIONAL
;HERE IF UNIT MAY BE IN TRANSFER OR POSITION STATE
HNGDS7: PUSHJ P,SAVE1## ;SAVE P1
HRRZ T1,UNICDA##(U) ;DDB WE'RE TALKING TO
CAIE T1,(F) ;SAME AS THIS DDB?
JRST HNGD7B ;NO, IT MUST BE IN A QUEUE
AOS T1,UNIRCV##(U) ;YES, COUNT NUMBER OF ATTEMPTS AT RECOVERY
CAIG T1,10 ;TRIED ENOUGH?
JRST HNGD7A ;NO, TRY AGAIN
HNGDWN: HRRZ P1,KONCHN##(J) ;YES, SET UP P1
MOVEI T1,KOPOFL## ;TELL BADUNI TO GRUMBLE AT OPR
PUSH P,U
PUSHJ P,BADUNI ;GIVE UP, SHOUT AT OPR OVER THIS UNIT
POP P,U
MOVEI T1,OCOD##
MOVEM T1,UNISTS##(U) ;SET STATE TO OCOD
MOVSI S,IOSMON
ANDCAB S,DEVIOS(F) ;MAKE SURE IOSMON=0
HRLZS UNICDA##(U)
SKIPE DEVRHB##(F) ;IF NOT REREADING HOME BLOCKS,
POPJ P, ; LET USER GET HUNG DEVICE MESSAGE
PJRST CPOPJ1## ;AND DONT GIVE ERROR MESSAGE
HNGD7A: SKIPL UNIECT##(U) ;IN ERROR RECOVERY?
JRST HNGD7B ;YES, LEAVE REGS ALONE
SETZM UNIECT##(U) ;NO, SET SO DAEMON WILL BE CALLED ON RECOVERY
SETZM UNIERR##(U)
IFN FTRP04,<
PUSHJ P,@KONRRG##(J) ;GO READ ALL DRIVE REGISTERS NOW
PUSHJ P,FSTREG ;COPY REGISTERS TO UDB
>
HNGD7B: MOVE T1,UNISTS##(U) ;RESET T1=UNIT STATUS
CAIE T1,TCOD## ;IS UNIT IN TRANSFER STATE?
JRST HNGDS8 ;NO
MOVSI P1,UNPHRC## ;ASSUME WE'LL RECOVER
PUSHJ P,@KONSTP##(J) ;STOP UNIT AND CAUSE AN INTERRUPT
MOVSI P1,UNPHNR## ;COUNT NOT-RECOVERED INSTEAD
ADDM P1,UNIHNG##(U)
MOVEM T2,UNISOF##(U) ;STORE (BEFORE) CONI STATUS
MOVEM T3,UNISDI##(U) ;STORE (BEFORE) DATAI STATUS
PUSHJ P,SETHNG## ;RESTORE HUNG-TIME
PJRST CPOPJ1## ;SKIP RETURN - NO ERROR MESSAGE
;STILL IN FTDHNG CONDITIONAL
HNGDS8: LDB T2,DEYCOD## ;STATUS OF FILE
CAIE T2,PCOD## ;FILE IN TRANSFER?
PJRST CPOPJ1## ;NO, RETURN WITH NO ERROR MESSAGE
;(UNIT PROBABLY HUNG, THIS JOB IN QUEUE)
HRRZM T1,UNISTS##(U) ;RESET SIGN BIT IF ON
JUMPL T1,HNGDWN ;GIVE UP IF DIDN'T RECOVER
MOVEI T2,UNPHPS## ;YES, INCREMENT HUNG POSITION COUNT
ADDM T2,UNIHNG##(U)
DSKOFF
SKIPG KONTAB##(J) ;#KONTROLLER FREE?
JRST HNGDS9 ;#NO, PUT IN QUEUE
MOVSI T1,400000 ;#INDICATE PHUNG RECOVERY
IORM T1,UNISTS##(U)
PUSHJ P,SETHNG##
PUSHJ P,@KONRCL##(J) ;#TRY A RECAL
JRST [MOVEI T1,OWCOD## ;UNIT IS DOWN
MOVEM T1,UNISTS##(U) ; SET UNISTS FOR
JRST .+1] ; ONCE A MINUTE TYPEOUT
DSKON ;#
PJRST CPOPJ1## ;AND RETURN WITH NO ERR MESSAGE
HNGDS9: MOVEI T1,PWCOD## ;#INDICATE UNIT IN PW
MOVEM T1,UNISTS##(U) ;#
MOVEI T1,UNIQUE##(U) ;#PUT FILE IN POSITION QUEUE
IFN FTVM,<
HRRZ P1,UNICHN##(U) ;POINT P1 AT CHAN BLOCK
>
PUSHJ P,PUTQUE ;# FOR THE UNIT
PJRST CPOPJ1## ;AND RETURN W/ NO MSG
> ;END CONDITIONAL ON FTDHNG
IFE FTDHNG,<
HNGDSK==:CPOPJ1##
>
;ROUTINE TO RETURN ANY RESOURCES A DDB HAS
;ENTER WITH JOB NUMBER IN J; EXITS WITH IOACT OFF IN DDB
RETRES::SKIPE T1,DEVMBF##(F) ;HAVE MON BUF?
PUSHJ P,GVMNBF ;YES, RETURN IT
TLNE S,IOSDA ;HAVE DA?
PUSHJ P,DWNDA
TLNE S,IOSAU ;HAVE AU?
PUSHJ P,DWNAU
JUMPE J,CLRACT##
CAMN J,CBUSER## ;HAVE CB?
PUSHJ P,GVCBJ##
PJRST CLRACT## ;RESET IOACT + POPJ
;SUBROUTINE TO CALL WAIT1
;THIS IS NEEDED SINCE WAIT1 CLOBBERS P1-P4
;THIS INSTRUCTION IS USED TO RESTORE PWAIT1 AFTER RESTART (SEE WTSATS)
PWAITZ::PJRST WAIT1##
PWAIT1:: PJRST WAIT1## ;GO TO WAIT1
;HERE TO PUT A SWAPPING REQUEST IN THE QUEUE
;S,F,U SET, T2=IOWD FOR DATA, T1=LOGICAL BLOCK NUMBER
SWAPIO::
IFE FTVM,<
TLO S,IOSMON ;SET MONITOR I/O BIT
HRRM U,DEVUNI##(F) ;SAVE U (RESET IN UUOSET)
MOVEM T1,DEVBLK##(F) ;LOGICAL BLOCK NUMBER
MOVEM T2,DEVDMP##(F) ;IOWD
MOVEM S,DEVIOS(F) ;S
PJRST UUOPTR ;PUT REQUEST IN QUEUE AND RETURN
> ;END IFE FTVM
IFN FTVM,<
SKIPE SQREQ## ;ANY REQUESTS?
SKIPL DEVSWP(F) ;YES, IS THIS THE SWAPPER?
STOPCD .+1,DEBUG,SFU, ;++SWAPPER FOULED UP
MOVE S,[IOSMON,,IOACT] ;OK, INDICATE ACTIVE MONITOR IO
MOVEM S,DEVIOS(F)
MOVEI J,0 ;INDICATE ON UUO LEVEL
PJRST SWPSCN## ;GO CRANK THINGS UP
> ;END FTVM
SUBTTL UUO LEVEL MODULE
IFN FTDSEK,<
;HERE TO START A SEEK
UUOSEK::PUSHJ P,VALUUO ;DSK CHAN?
POPJ P, ;NO
HRRZ T1,DEVACC##(F) ;IF NOT SUPER I/O,
TLNN S,IOSUPR
JUMPE T1,CPOPJ## ; DO NOTHING IF LOOKUP/ENTER NOT DONE
PUSHJ P,UUOSET ;YES, SET UP U, J AND GET NEXT BLOCK
POPJ P, ;EOF - RETURN
SKIPL KONPOS##(J) ;DOES CONTROLLER POSITION?
SKIPE UNISTS##(U) ;YES. IS UNIT IDLE?
POPJ P, ;NO. IGNORE SEEK REQUEST
IFN FTDHIA,<
SKIPE DIADSK##
POPJ P,
>
NOSCHEDULE ;YES. THIS CODE MUST BE INTERLOCKED
DSKOFF ;SINCE WE ARE DIDDLING QUEUES
SKIPL KONTAB##(J) ;#IS CONTROL BUSY
JRST UUOSK1 ;#NO, START POSITION
MOVEI T1,SWCOD## ;#YES, PUT UNIT INTO SW
MOVEM T1,UNISTS##(U) ;#PUT FILE INTO SW/PW QUEUE
IFN FTVM,<
PUSHJ P,SAVE1## ;PUT QUE USES CHNWAT(P1)
MOVE P1,KONCHN##(J) ; SO SET P1 TO CHAN DB LOC
>
JRST UUOPW3 ;#FOR THIS UNIT AND RETURN
;HERE WHEN CONTROLLER IS FREE - START POSITION (UNLESS UNIT ALREADY POSITIONED)
UUOSK1: PUSHJ P,CYLCOM ;#COMPUTE DISTANCE TO TARGET CYLINDER
JUMPE T1,SEKRET ;#ALREADY THERE IF DISTANCE =0
MOVEI T1,SCOD## ;#NOT POSITIONED, PUT UNIT INTO S STATE
MOVEM T1,UNISTS##(U) ;#
PJRST STRPS1 ;#START POSITION AND RETURN
;HERE IF UNIT IS ALREADY POSITIONED - TURN ON PI AND RETURN
SEKRET: DSKON ;#
SCHEDULE
POPJ P, ;RETURN
> ;END CONDITIONAL ON FTDSEK
;SUBROUTINE TO ENSURE THAT A CALLI IS FOR A DSK CHAN
;RETURNS CPOPJ IF NOT, CPOPJ1 IF SO
VALUUO::LDB T1,PUUOAC##
CAMG T1,USRHCU## ;CHAN NO LEGAL?
SKIPN F,USRJDA##(T1) ;YES, DEVICE ASSIGNED?
JRST IOIERR## ;NO, "IO TO UNASSIGNED CHAN"
VALUUX: MOVE T1,DEVMOD(F) ;IS DEVICE A DSK?
TLNN T1,DVDSK
POPJ P, ;NO, IMMEDIATE RETURN
MOVE S,DEVIOS(F)
PJRST CPOPJ1## ;YES, SKIP RETURN
UUOPTR: PUSHJ P,UUOSET ;SET U, J AND GO TO NXTBLK
POPJ P, ;EOF OR QUOTA EXHAUSTED
UUOPWQ::JUMPN U,UUOPWR ;JUST IN CASE F/S HAS BEEN JERKED OUT
TRO S,IOIMPM ; LIGHT ERROR BIT
PJRST STOIOS##
UUOPWR:
IFN FTDPRI,<
MOVE T1,DEVPRI##(F) ;DISK - PRIORITY WORD
TRNE T1,DEPUUO ;PRIORITY SET BY UUO?
JRST UUOPWS ;YES
LDB T1,PJOBN## ;NO, GET DISK-PRIORITY OF JOB
LDB T1,JBXPRI##
DPB T1,DEYPRI## ;AND SAVE IT IN DDB
UUOPWS:>
IFN FTDSUP,< ;SUPER USETI/USETO
TLNN S,IOSUPR ;SUPER USETI/USETO?
>
SKIPE DEVBLK##(F) ;NO. 0 BLOCK REQUESTED?
JRST UUOPWZ ;NO. OK
IFN FTRP04,<
SKIPN DINITF## ;RP04 READS BLOCK 0 TO CHECK FORMAT ERROR
>
SKIPN DEVUNI##(F) ;STR BEEN REMOVED?
JRST UUOPWZ ;YES, DON'T HALT
MOVE T1,DEVPPN(F) ;REQUESTING BLOCK 0 -
CAME T1,SYSPPN## ;[1,4]?
JRST UUOPWY ;NO - ERROR
MOVE T2,DEVFIL(F) ;SINCE HOME BLOCKS ARE WRITTEN AT THE FRONT OF
CAME T2,[SIXBIT .HOME.] ; THE DISK, CHECK FOR HOME[1,4]
UUOPWY: STOPCD .,JOB,IBZ, ;++I/O TO BLOCK ZERO
UUOPWZ: PUSHJ P,SAVE1## ;SAVE P1
UUOPW0: HRRZ P1,KONCHN##(J) ;SET P1 = CHANNEL DB
TLZ S,IOSHMS ;RESET HUNG-DEVICE MESSAGE BIT
TRO S,IOACT ;SET FILE ACTIVE(IO ACT=1)
MOVEM S,DEVIOS(F) ;CANT USE SETACT-IOW MAY BE ON
NOSCHEDULE
DSKOFF ;#TURN OFF ALL DSK CONTROLLER PI CHANS.
SKIPG T2,KONPOS##(J) ;#DOES KONTROL POSITION?
PJRST UUOTWQ ;#NO, PUT FILE TW OR T
SKIPN T1,UNISTS##(U) ;#IS UNIT IDLE?
JRST UUOPW1 ;#YES,GO SET PW,P,TW, OR T
IFN FTMDA,<
LDB T3,PJOBN## ;NO, IS UNIT IN MDA WAIT?
CAIN T1,MDACOD## ; (AFTER FREE INTERRUPT, MDA SHOULD REREAD HOME BLOCKS)
CAME T3,MDCJOB## ;YES, IS THIS MDA?
CAIA ;NO
JRST UUOTWQ ;YES, LET IT HAPPEN
>
CAIE T1,SWCOD## ;#NO,IS UNIT SW?
JRST UUOPW2 ;#NO, SET FILE PW
SETZB T1,UNIQUE##(U) ;#YES. ERASE QUEUE FOR UNIT
;#CAN ONLY BE 1 SEEK WAITER IF ANY FILE IN SW/PW QUEUE
MOVEM T1,UNISTS##(U) ;#AND SET UNIT STATE=I
UUOPW1:
IFN FTDHIA,<
SKIPE DIADSK## ;TRYING TO SHUT DOWN IO?
CAME P1,DIACHN## ;YES, FOR THIS CHAN?
CAIA ;NO
JRST UUOPW2 ;YES, JUST QUEUE THIS REQUEST
>
IFN FTRP04,<
TLNN T2,KOPPWX## ;UNIT/KONTROLLER POSITION WHILE TRANFERRING?
>
SKIPL KONTAB##(J) ;#IS KONTROL BUSY?
JRST UUOPOS ;#NO, PUT FILE P, TW OR T
IFN FTDUAL,<
SKIPE T1,UNI2ND##(U) ;IS THERE AN ALTERNATE PATH?
SKIPE UNISTS##(T1) ;YES, IS THE UNIT USEABLE?
JRST UUOPW2 ;NO
MOVE T2,UNIKON##(T1) ;YES, T1=UNIT T2=KON
IFN FTDHIA,<
HRRZ T3,KONCHN## ;ALTERNATE CHAN
SKIPE DIADSK## ;DIAG IN PROGRESS FOR IT?
CAME T3,DIACHN##
>
SKIPGE KONTAB##(T2) ;KONTROLLER IDLE?
JRST UUOPW2 ;NO
HRRM F,UNICDA##(U) ;YES, 1ST UNIT ALSO IS THIS DDB
HRRZ U,T1 ;SET UP NEEDED ACS
HRRZ J,T2
HRRZ P1,KONCHN##(J)
JRST UUOPOS ;AND GO START SEEK (OR MAYBE XFER)
>
UUOPW2: MOVEI T1,PWCOD## ;#YES, PUT FILE PW
SKIPN UNISTS##(U) ;#UNIT IDLE?
MOVEM T1,UNISTS##(U) ;#YES SET UNIT PW
DPB T1,DEYCOD## ;#SET FILE PW
UUOPW3: MOVEI T1,UNIQUE##(U) ;#QUEUE ON WHICH TO INSERT FILE
PJRST PUTQUE ;#PUT FILE ON PWQ AND RETURN
UUOPOS: PUSHJ P,CYLCOM ;#IS UNIT ALREADY POSITIONED?
JUMPN T1,STRPOS ;#NO, START POSITIONING IF 0
PJRST UUOTWQ ;#YES, SET FILE+UNIT TO T, KONCHN TO B
;# OR FILE+UNIT TO TW AND ADD TO TWQ
;SUBROUTINE TO CHECK STATUS OF A UNIT
;RETURNS CPOPJ1 IF OK (MAY HAVE HAD TO CALL HNGSTP)
;RETURNS CPOPJ IF NG - UNIT OFF-LINE, DDB HAS MON-BUF TO WRITE
UNICHK: SKIPE T1,UNISTS##(U) ;STATUS OF UNIT
CAIGE T1,MDACOD## ;OKAY ?
PJRST CPOPJ1## ;YES
IFN FTMDA,<
MOVE J,.C0JOB## ;NO, IS UNIT IN MDA WAIT
CAMN J,MDCJOB## ;AND THIS JOB=MDA?
CAIE T1,MDACOD## ;IF SO, LET MDA DO THE IO
>
SKIPGE DEVSWP##(F) ;IS IT THE SWAPPER?
PJRST CPOPJ1 ;YES, LET IT TRY ANYWAY
PUSHJ P,SAVSTS ;SAVE A RECORD OF JOB'S RESOURCES
JRST UNICH1 ;HAS MON BUF AND WRITING - LOSE
PUSHJ P,HNGSTP## ;TELL OPR AND USER ABOUT THE PROBLEM
POP P,T3 ;RECORD OF RESOURCES
PUSHJ P,RESSTS ;GET THEM AGAIN
JRST UNICHK ;AND TRY AGAIN
UNICH1: TRO S,IODERR ;LOSE - LIGHT ERROR BIT
POPJ P, ;AND RETURN
;SUBROUTINE TO MAKE A RECORD OF RESOURCES JOB OWNS
;RETURNS WITH THE RECORD ON THE PD LIST AND NO RESOURCES
;RETURNS CPOPJ IF HAS MON-BUF, ELSE CPOPJ1
; (CANT CONTINUE IF THE JOB HAS A MON-BUF SINCE IT HAS ALREADY SET UP
; THE IOWD IN THE DDB, MIGHT GET THE OTHER BUFFER BACK ON CONTINUE)
SAVSTS::SKIPE T1,DEVMBF##(F) ;NO. HAVE MON BUF?
POPJ P, ;YES, THIS OPERATION LOSES
MOVE J,JOB## ;NO, OK TO ATTEMPT RECOVERY
HLR T1,S ;SAVE IOSDA,IOSAU BITS
CAMN J,CBUSER## ;JOB HAVE CB?
TLO T1,1 ;YES, LIGHT A BIT
PUSH P,T1 ;SAVE RECORD OF RESOURCES
PUSHJ P,RETRES ;GIVE UP ALL RESOURCES JOB OWNS
POP P,T1 ;RECORD
EXCH T1,(P) ;PUT BACK ON STACK
PJRST 1(T1) ;AND TAKE SKIP-RETURN
;SUBROUTINE TO GET BACK THE RESOURCES
;ENTER WITH T3 = THE RECORD AS SET BY SAVSTS
;PRESERVES T2
RESTS::
RESSTS: HRRZ U,DEVUNI##(F) ;RESET U
MOVE S,DEVIOS(F) ;AND S
TLNE T3,1 ;WANT CB?
PUSHJ P,GETCB##
TLNE T3,-200 ;WANT MON BUF?
PUSHJ P,GTMNB ;YES
TRNE T3,IOSDA ;GET DA IF NEEDED
PUSHJ P,UPDA
TRNE T3,IOSAU ;GET AU IF NEEDED
PUSHJ P,UPAU
POPJ P, ;AND RETURN TO CALLER
;SUBROUTINE TO SET UP U AND J, EXIT TO NXTBLK (WHICH RETURNS CPOPJ
;OR CPOPJ1 DEPENDING ON WHETHER ANOTHER BLOCK OF THE FILE EXISTS
UUOSET::HRRZ U,DEVUNI##(F) ;SET U(UNIT DATA BLOCK)
JUMPE U,CPOPJ## ;JUST IN CASE F/S HAS BEEN JERKED OUT
PUSHJ P,UNICHK ;MAKE SURE UNIT IS OKAY.
POPJ P, ;UNIT IS DOWN
HRRZ J,UNIKON##(U) ;SET J(KONTROL DATA BLOCK)
;FALL INTO NXTBLK, GET THE NEXT BLOCK OF THE FILE AND RETURN
;SUBROUTINE TO OBTAIN THE NEXT BLOCK ADDRESS
;ENTER WITH J,F,U,S SET UP
;INTERRUPT LEVEL CHECKS BEFORE CALLING TO MAKE SURE STILL POINTERS IN CORE SO NO IO
;RETURNS CPOPJ IF EOF OR DISK FULL OR PNTR BLOCK FULL
;RETURNS CPOPJ1 IF OK
NXTBLK: JUMPL S,CPOPJ1## ;*RETURN IF A MONITOR READ
MOVMS DEVREL##(F) ;NEGATIVE DEVREL IS FLAG FROM USETI -N
HRRZ T1,DEVLFT##(F) ;*NO, NUMBER OF BLOCKS LEFT IN CURRENT GROUP
HRRZ T2,DEVACC##(F) ;*YES. LOC OF A.T.
MOVE T2,ACCWRT##(T2) ;*ACTUAL NUMBER OF BLOCKS WRITTEN
TLNE S,IO+IOSUPR ;*READING?
JRST NXTBLA ;*NO
CAMGE T2,DEVREL##(F) ;*TRYING TO READ MORE THAN WAS WRITTEN?
POPJ P, ;*YES. TAKE EOF RETURN
JRST NXTBL0 ;NO, CONTINUE
NXTBLA: LDB T3,DEYFNC## ;WRITING, FILE APPEND ONLY ?
IFN FTDSUP,< ;SUPER USETI/USETO
TLNN S,IOSUPR ;AND NOT SUPER USETO ?
>
CAIE T3,FNCAPP##
JRST NXTBL0 ;NO
CAML T2,DEVREL##(F) ;YES. WRITING IN THE MIDDLE ?
POPJ P,0 ;YES, RETURN IOBKTL
NXTBL0: SOJGE T1,CPOPJ1## ;*NO, IF BLOCKS LEFT, RETURN (THIS PNTR OK)
IFN FTDSUP,< ;SUPER USETI/USETO
TLNE S,IOSUPR ;*SUPER USETI/USETO?
POPJ P, ;*YES - NO PNTRS TO READ
>
NXTBL1: MOVSI T2,1 ;*DECREMENT NO. OF POINTERS
SKIPG DEVRSU##(F) ;* IF RIB WASNT ALREADY FULL
ADDB T2,DEVRSU##(F) ;*RIB NOW FULL?
JUMPGE T2,NXTB1B ;*YES, LIGHT IOBKTL
AOS T1,DEVRET##(F) ;*STEP TO NEXT POINTER IN CORE
CAIG T1,DEVRBN##(F) ;*RUN OUT OF IN-CORE POINTERS?
JRST NXTBL5 ;*NO, CONTINUE
MOVE T1,DEVREL##(F) ;YES, UPDATE RELATIVE BLOCK NUMBER
MOVEM T1,DEVFLR##(F) ;OF LOWEST BLOCK IN DDB POINTERS
HRRM U,DEVFUN##(F) ;AND UNIT OF 1ST PNTR IN DDB
TLNN S,IO ;READING?
JRST NXTBL4 ;YES
;HERE WHEN WRITING AND THE DDB POINTER SPACE IS FULL
TLNN F,OCLOSB ;OUTPUT CALLED BY CLOSE?
JRST NXTBL2 ;NO. CONTINUE
;HERE IF THE OUTPUT IS FROM CLOSE
SKIPN DEVMBF##(F) ;HAVE MONITOR BUFFER?
PUSHJ P,GTMNBF ;GET MONITOR BUFFER
PUSHJ P,RIBCUR ;READ THE CURRENT RIB
JUMPN T3,CPOPJ## ;NON-ZERO T3 MEANS ERROR READING RIB
PUSHJ P,SPTRW ;SET AOBJN WORD FOR POINTERS
MOVE T4,T1 ;SAVE ORIGINAL POINTER WORD
PUSHJ P,DD2MN ;COPY POINTERS INTO MON BUF
JFCL
SUBM T1,T4 ;COMPUTE NEW FREE-POINTER COUNT
AOS T2,T4
DPB T4,DEYRLC## ;AND SAVE IT IN DDB
AOBJN T1,.+1 ;POINT T1 AT NEXT GROUP OF POINTERS
SKIPE (T1) ;MORE POINTERS? (PRE-ALLOCATED SPACE)
PUSHJ P,PTRBL1 ;YES, COPY THEM TO DDB
MOVEI T2,DEVRB1##(F) ;RESET DEVRET
HRRM T2,DEVRET##(F)
HLRE T2,DEVRSU##(F) ;NUMBER OF PNTRS LEFT
AOJGE T2,NXTB1A ;GO IF LAST PNTR
SKIPL DEVRIB##(F) ;NOT LAST, PRIME RIB?
TLOA S,IOSRIB ;YES, SET FLAG TO INDICATE TO CLOSE CODE
NXTB1A: TLZ S,IOSRIB ;RIB NOT PRIME, BETTER WRITE IT OUT
PUSHJ P,WRTRIB ;WRITE OUT THE RIB
JRST NXTBL3 ;AND CONTINUE
;HERE IF THE CURRENT RIB IS FULL
NXTB1B: TLNE S,IO ;READING?
JRST NXTB1D ;NO
PUSHJ P,PTRCUR ;READ THE CURRENT RIB
JUMPN T3,CPOPJ## ;ERROR IF T3 NON-ZERO
IFN FTDMRB,< ;IF MULTIPLE RIBS
PUSHJ P,PTRNXT ;YES, GET THE NEXT RIB
PJRST GVMNB0 ;RIB ERROR
PUSHJ P,CPYEXT## ;COPY FIRST POINTERS FROM EXTENDED RIB
POPJ P, ;RIB ERROR
HRRZ T1,DEVLFT##(F) ;GET COUNT OF NUMBER OF BLOCKS LEFT IN PNTR
JRST NXTBL7 ;AND CONTINUE
>
;IF NO MULTIPLE RIBS, FALL INTO POPJ AT NXTBID
;HERE WHEN WRITING AND WE'VE RUN OUT OF ROOM IN THE CURRENT RIB
NXTB1D:
IFE FTDMRB,<
POPJ P, ;RIB FULL - ERROR RETURN
>
IFN FTDMRB,< ;IF MULTIPLE RIBS
PUSHJ P,EXTRIB ;YES, ALLOCATE ANOTHER RIB
IFE FTDSIM,<
POPJ P, ;ERROR
>
IFN FTDSIM,<
PJRST DOWNIF ;ERROR- RETURN DA AND EXIT
PUSHJ P,DOWNIF ;OK - GIVE UP DA IF WE HAVE IT
>
PUSHJ P,CPYEXT## ;SET UP THE DDB
POPJ P, ;RIB ERROR
JRST NXTBLK ;USE THE NEW BLOCKS ACQUIRED
>
;HERE IF THE OUTPUT IS NOT FROM CLOSE
NXTBL2: PUSHJ P,PTRCUR ;GET THE RIB
JUMPN T3,GVMNB0 ;ERROR IN RIB KILLS US
PUSHJ P,PTRWRT ;SAVE POINTERS AND WRITE (KEEP THE MON BUF)
PUSHJ P,SPTRW ;SET T1 TO AN AOBJN WORD FOR ALL THE PNTRS
LDB T2,DEYRLC## ;NEW POINTER LOC
HRLS T2 ;SET TO ADD TO AOBJN WORD
ADD T1,T2
SKIPE (T1) ;POINTERS (PRE-ALLOCATED SPACE)?
PUSHJ P,PTRBLT ;COPY THE NEW POINTERS INTO DDB
PUSHJ P,GVMNB0 ;GIVE UP MONITOR BUFFER
NXTBL3: SETZM DEVBLK##(F) ;INDICATE NO CONTIGUITY
JRST NXTBL5 ;AND CONTINUE
;HERE WHEN THE POINTERS RAN OUT ON INPUT
NXTBL4: PUSHJ P,CLSNAM## ;MAKE SURE NAME IN DDB IS RIGHT
; (FILE MIGHT BE RENAMED)
PUSHJ P,PTRTST ;READ NEW POINTERS, WRITE THESE IF CHANGED (CHECKSUM)
JFCL
PUSHJ P,RDPTRA ;COPY NEW SET OF POINTERS INTO DDB
SKIPE DEVRB1##(F) ;GET NEW POINTERS?
JRST NXTBL5 ;YES
PUSHJ P,FNDPTR ;NO, FIND THEM IN A DDB
POPJ P, ;COULDN'T FIND THEM
;HERE WITH DEVRET POINTING TO NEXT RETRIEVAL POINTER (OR 0)
NXTBL5: SKIPE T2,@DEVRET##(F) ;*IS THERE ANOTHER POINTER?
JRST NXTBL6 ;*YES
MOVE T1,DEVACC##(F) ;NO, ARE WE UP TO END OF FILE?
IFN FTDSIM,<
TLNN S,IO ;IF READING,
SKIPA T1,ACCWRT##(T1) ;USE NUMBER OF BLOCKS WRITTEN
PUSHJ P,GETALC ;ELSE USE NUMBER OF BLOCKS ALLOCATED
>
IFE FTDSIM,<
MOVE T1,ACCWRT##(T1) ;ELSE USE NUMBER OF BLOCKS WRITTEN
>
CAMG T1,DEVREL##(F)
JRST NXTB5A ;YES, RETURN EOF OR ALLOCATE BLOCKS
TLNN S,IO ;NOT AT EOF. READING?
JRST FNDPTR ;YES, FIND POINTERS IN SOME DDB
IFE FTDSIM,<
JRST OUTGRP ;NO, ALLOCATE SOME MORE (SYSTEM ERROR?)
>
IFN FTDSIM,<
JRST FIXDDB ;NO, FIND NEWLY-ALLOCATED PNTRS IN RIB
>
;HERE IF WE ARE UP TO THE END OF THE FILE
NXTB5A: TLNN S,IO ;READING?
POPJ P, ;YES - EOF
;NO, FALL INTO OUTPUT ALLOCATION
;HERE TO ALLOCATE SOME MORE SPACE FOR AN OUTPUT FILE
OUTGRP: HRRZ T1,UNISTR##(U) ;LOC OF STR DATA BLOCK
HLRZ T2,UNIGRP##(U) ;YES, NUMBER OF CLUSTERS TO ALLOCATE
TLO T2,400000 ;TELL CHKQTA THAT THE CALL IS FROM OUTPUT
PUSHJ P,CHKQTA ;CHECK USERS QUOTA OR DISK FULL
IFN FTDQTA,<
JUMPLE T2,OUTG5A ;CAN'T GET ANY MORE RETURN (IOBKTL SET)
>
;HERE WITH T2 = AMOUNT TO ALLOCATE, STR HAS SOME SPACE
IFN FTDSIM,<
MOVE T1,DEVACC##(F) ;IF SIMULT UPDATE FILE
MOVE T1,ACCSMU##(T1) ; GET MON BUF NOW, BEFORE GETTING DA
TRNE T1,ACPSMU ; TO AVOID A DEADLY EMBRACE
PUSHJ P,GTMB2
>
SKIPN T1,DEVBLK##(F) ;CONTIGUITY ALLOWED?
JRST OUTGR3 ;NO. GET SPACE ANYWHERE
SOS DEVRET##(F) ;YES. POINT TO CURRENT RETRIEVAL PTR
PUSHJ P,CHKADD ;ROOM LEFT IN POINTER?
JUMPLE T2,OUTGR2 ;NO. GET SPACE ANYWHERE
MOVE T1,DEVBLK##(F) ;YES. GET SPACE PAST 1ST UNALLOCATED BLOCK
PUSHJ P,TAKBLK ;YES. GET SPACE AT END
JRST OUTGR2 ;TRY FOR SPACE ANYWHERE ON UNIT
PUSHJ P,ADDPTR ;ADD NEW BLOCKS TO CURRENT POINTER
MOVSI T2,-1 ;DEVRSU WAS INCREMENTED AT NXTBL1,
ADDM T2,DEVRSU##(F) ; SO DECREMENT IT BACK DOWN
IFN FTDSIM,<
TLNE S,IOSDA ;IF A SIM UPDATE FILE,
JRST OUTG4A ; REWRITE RIB WITH NEW POINTERS IN IT
>
JRST NXTBL8 ;STORE NEW DEVLFT AND CONTINUE
;HERE WHEN CANT GET SPACE CONTIGUOUS WITH THE OLD SPACE
OUTGR2: AOS DEVRET##(F) ;POINT TO NEW POINTER LOC
MOVE T2,T1 ;RESTORE AMOUNT TO GET
OUTGR3: MOVEI T4,TAKCHK ;SET TO TAKE N BLOCKS ON ANY UNIT
HLRE T1,DEVRSU##(F) ;UNLESS THERE IS ONLY ROOM FOR 1 POINTER
CAMLE T1,[EXP -2] ;IN WHICH CASE, SET TO STAY ON THIS UNIT
MOVEI T4,TAKBLK
AOJN T1,OUTG3A ;JUMP IF NOT LAST POINTER
MOVSI T1,DEPLPC## ;IS LAST, SET LAST POINTER IN CORE BIT
IORM T1,DEVLPC##(F) ;SO BLOCK WILL BE RESERVED FOR REDUNDANT RIB
OUTG3A: SETZ T1, ;GET BLOCKS ANYWHERE
PUSHJ P,(T4) ;GET SOME BLOCKS
JRST OUTGR5 ;ON NEW UNIT
OUTGR4: MOVEM T2,@DEVRET##(F) ;GOT SPACE ON SAME UNIT - SAVE POINTER IN DDB
IFN FTDSIM,<
TLNN S,IOSDA ;SIM UPDATE FILE?
JRST OUTG4B ;NO, CONTINUE
OUTG4A: PUSHJ P,WRTPTR ;YES, REWRITE RIB WITH NEW PNTRS
PUSHJ P,DWNDA ;GIVE UP DA NOW THAT RIB WRITTEN
PUSHJ P,GVMNB0
PJRST FIXDDB ;CALL USETI TO GET PNTRS BACK INTO CORE
; (DD2MN ZEROES THE DDB)
OUTG4B:>
HRRZ T3,DEVACC##(F) ;LOC OF A.T.
MOVEI T4,ACP1PT## ;MAKE SURE THAT 1PT
ANDCAM T4,ACCUN1##(T3) ; IS OFF IN THE UN1 WORD
JRST NXTBL6 ;AND CONTINUE
;GOT SOME SPACE ON A NEW UNIT, OR STR FULL
OUTGR5: JUMPE T3,OUTGR6 ;IF GOT ANY
MOVSI T1,1
ADDB T1,DEVRSU##(F) ;UPDATE DEVRSU
JUMPGE T1,OUTGR6 ;ERROR IF ALL SLOTS TAKEN (SHOULD NEVER HAPPEN)
AOBJN T1,OUTG5B
MOVSI T1,DEPLPC##
IORM T1,DEVLPC##(F)
OUTG5B: MOVEM T3,@DEVRET##(F) ;SAVE UNIT-CHANGE IN DDB
HRRM U,DEVUNI##(F) ;SAVE NEW UNIT IN DDB
AOS T1,DEVRET##(F) ;POINT TO NEXT PNTR SLOT
CAIG T1,DEVRBN##(F) ;DDB FULL?
JRST OUTGR4 ;NO, STORE REAL POINTER IN DDB
PUSH P,T2 ;YES, SAVE PNTR
PUSHJ P,WRTPTR ;COPY PNTRS TO RIB, WRITE
POP P,T2 ;RESTORE NEW RETRIEVAL POINTER
JUMPE T3,OUTGR4 ;CONTINUE IF NO RIB ERR
PUSHJ P,CNVPTR ;RIB ERR- GIVE BACK THE BLOCKS
JFCL ;BAD UNIT!
STOPCD OUTGR6,DEBUG,LNP, ;++LAST POINTER NOT A POINTER
MOVE T2,T1 ;SET TO GIVE BACK THE BLOCKS
MOVE T1,DEVBLK##(F)
PUSHJ P,GIVBLK ;RETURN THE BLOCKS, UPDATE COUNTS
JRST OUTGR6 ;UPDATE DEVRSO AND ERROR RETURN
;HERE WHEN STRUCTURE IS FULL, RETURN DEVRSU TO PRE-CALL STATE
OUTG5A:
MOVEI T4,.ERFUL ;STR FULL INTERCEPT
PUSHJ P,SETINJ ;LET JOB KNOW
JFCL ;DON'T CARE IF NOT ENABLED
;HERE WHEN THERE ARE NO FREE BLOCKS LEFT IN THE STR
OUTGR6: SOS DEVRET##(F) ;POINT DEVRET BACK TO LAST REAL POINTER
MOVSI T1,-1 ;DECR DEVSRU
ADDM T1,DEVRSU##(F) ;(INCREMENTED AT NXTBL1)
PJRST ERRFUL ;LIGHT AN ERROR BIT AND RETURN
;HERE WHEN WE HAVE A POINTER IN T2 (MAY BE UNIT CHANGE)
NXTBL6: PUSHJ P,CNVPTR ;*CONVERT POINTER TO BLK, COUNT
JRST OUTGR6 ;BAD UNIT-CHANGE PTR-LOSE
JRST NXTBL1 ;*WAS A UNIT-CHANGE. TRY NEXT
;HERE WITH T1=BLOCK COUNT, DEVBLK SET UP
TLO S,IOSFIR ;*INDICATE CHECKSUM MUST BE COMPUTED
MOVEM S,DEVIOS(F) ;*SAVE S IN DDB
TLNE S,IO ;*READING?
JRST NXTBL8 ;*NO, ANY ALLOCATED BLOCK IS OK
NXTBL7: HRRZ T2,DEVACC##(F) ;*YES, MAKE SURE THESE BLOCKS ARE ALL WRITTEN IN
MOVE T2,ACCWRT##(T2) ;*HIGHEST BLOCK WRITTEN
SUB T2,DEVREL##(F) ;*-1ST RELATIVE BLOCK OF GROUP
AOJLE T2,CPOPJ## ;*EOF IF NO BLOCKS LESS THAN HIGHEST WRITTEN
NXTBL8: MOVE T2,DEVLPC##(F) ;GET WORD TO TEST FOR LAST POINTER
TLNN T2,DEPLPC## ;LAST POINTER IN CORE?
JRST NXTBL9 ;NO, NO NEED TO WORRY
HRRZ T2,DEVRET##(F) ;GET ADDRESS OF CURRENT POINTER IN DDB
CAIE T2,DEVRBN##(F) ;POINTING TO LAST SLOT IN DDB?
SKIPN 1(T2) ;NO, NEXT SLOT EMPTY?
SOJE T1,NXTBL1 ;YES, JUMP IF CURRENT POINTER EXHAUSTED
NXTBL9: HRRM T1,DEVLFT##(F) ;*AND SAVE IN DDB
HRRZ J,UNIKON##(U) ;*RIB MIGHT BE ON ANOTHER KONTROLLER
JRST CPOPJ1## ;*AND TAKE SKIP RETURN
IFN FTDMRB,< ;IF MULTIPLE RIBS
;SUBROUTINE TO CREATE AN EXTENDED RIB
; RETURNS CPOPJ IF ERROR OR RIB NOT EXTENDABLE
; RETURNS CPOPJ1 WITH NEW RIB IN THE MONITOR BUFFER AND T1=NUMBER OF NEW BLOCKS ADDED
EXTRIB:
IFN FTDSIM,<
MOVE T1,DEVACC##(F)
MOVE T1,ACCSMU##(T1) ;SIM UPDATE FILE?
TLNN S,IOSDA ; AND DDB WITHOUT DA (OR MON BUF)?
TRNN T1,ACPSMU
JRST EXTRB0
PUSHJ P,GTMNBF ;YES, GET DA BEFORE READING RIB
PUSHJ P,UPDA ; CAUSE ANOTHER JOB MIGHT TRY TO EXTEND RIB
EXTRB0:>
PUSHJ P,PTRCUR ;GET CURRENT RIB INTO CORE
JUMPN T3,GVMNB0 ;RIB ERROR IF T3 NON-0
PUSHJ P,DD2MN ;COPY POINTERS FROM DDB TO RIB
STOPCD GVMNB0,DEBUG,NPD, ;++NO POINTERS IN DDB
MOVE T1,DEVMBF##(F) ;IOWD TO MONITOR BUFFER
SKIPG DEVRIB##(F) ;CURRENT RIB EXTENDED?
JRST EXTRB1 ;YES, CAN EXTEND AGAIN
SKIPE RIBFLR+1(T1) ;PRIME RIB. RIBFLR=0?
PJRST EXTRB3 ;NO, CANNOT EXTEND THIS RIB
EXTRB1: SKIPE T2,RIBXRA##+1(T1) ;RIB ALREADY EXTENDED?
JRST EXTRB2 ;YES, GO GET THE NEXT RIB
PUSHJ P,SAVE1##
MOVEI T1,DEPWRT## ;MAKE SURE FNDDDB DOESN'T
ANDCAM T1,DEVWRT##(F) ; FIND THIS DDB WHILE NUMBERS ARE CHANGING
PUSHJ P,GETALC ;GET "REAL" ACCALC
MOVE P1,T1 ;RIBFLR=ACCALC - 2 RIB BLOCKS
SUBI P1,2
PUSHJ P,GTLPT## ;GET LAST RIB POINTER
PUSHJ P,CNVPTR ;DECODE THE POINTER
JFCL
STOPCD GVMNB0,DEBUG,UPI, ;++UNIT POINTER ILL.
;STILL IN FTDMRB CONDITIONAL
SOJ T1, ;DECREMENT NUMBER OF BLOCKS LEFT
ADDM T1,DEVBLK##(F) ;NOW DEVBLK IS LAST BLOCK IN THE RIB
MOVE T1,DEVMBF##(F) ;IOWD FOR MONITOR BUFFER
MOVE T2,DEVBLK##(F) ;GET ABSOLUTE BLOCK NUMBER OF REDUNDANT RIB
MOVEM T2,RIBSLF##+1(T1) ;STORE IN THE RIB
MOVEI T2,CODRIB## ;RIB IDENTIFICATION CODE
MOVEM T2,RIBCOD##+1(T1) ;STORE IN THE RIB
SETZ T1, ;TELL TAKBLK TO GET BLOCKS ANYWHERE
HRROI T2,3 ;LOOK FOR 2 BLOCKS (RIBS + 1 DATA)
PUSHJ P,TAKBLK ;ALLOCATE SOME BLOCKS
PJRST EXTRB3 ;ERROR, COULDN'T GET THE BLOCKS
PUSH P,T1 ;SAVE NUMBER OF BLOCKS JUST TAKEN
MOVEM T2,DEVRB1##(F) ;STORE NEW POINTER IN DDB
PUSH P,DEVRIB##(F) ;SAVE CURRENT RIB POINTER
LDB T4,DEYRBC## ;NUMBER OF CURRENT RIB
MOVSI T1,400000 ;TURN ON BIT 0 IN T1
MOVEM T1,DEVRIB##(F) ;NEGATIVE DEVRIB MEANS CURRENT RIB IS EXTENDED
ADDI T4,1 ;INCREMENT RIB NUMBER
DPB T4,DEYRBC## ;AND SAVE IN DDB
MOVE T4,UNISTR##(U) ;GET ADDRESS OF SDB FOR CURRENT RIB UNIT
LDB T1,STYCLP##(T4) ;EXTRACT CLUSTER ADDRESS FROM POINTER
DPB T1,DEYRBA## ;SAVE IN DEVRIB
LDB T1,UNYLUN## ;GET CURRENT RIN LOGICAL UNIT NUMBER
DPB T1,DEYRBU## ;SAVE IN DEVRIB
;STILL IN FTDMRB CONDITIONAL
MOVE T1,DEVMBF##(F) ;IOWD TO MONITOR BUFFER
MOVE T2,DEVRIB##(F) ;POINTER TO NEXT RIB ON CHAIN
MOVEM T2,RIBXRA##+1(T1) ;SAVE IN CURRENT RIB
POP P,DEVRIB##(F) ;RESTORE POINTER TO CURRENT RIB
MOVE T2,RIBSLF##+1(T1) ;GET BLOCK NUMBER FOR REDUNDANT RIB WRITE
PUSHJ P,MONWRT ;WRITE THE REDUNDANT RIB
PUSHJ P,WRTRIB ;WRITE THE WORKING COPY OF THE RIB
MOVE T1,DEVMBF##(F) ;GET THAT IOWD AGAIN
MOVE T2,RIBXRA##+1(T1) ;POINTER TO EXTENDED RIB
MOVEM T2,DEVRIB##(F) ;NEW CURRENT RIB
PUSHJ P,SPTRW ;SET UP POINTER TO RIB
MOVE T4,T1 ;MOVE POINTER TO T4
SETZM (T1) ;CLEAR THE POINTER LOCATION
AOBJN T1,.-1 ;CLEAR ALL POINTERS IN THE RIB
MOVE T1,DEVMBF##(F) ;GET IOWD TO MONITOR BUFFER
MOVEM P1,RIBFLR##+1(T1) ;SET UP RIBFLR TO FIRST BLOCK NUMBER IN RIB
SETZM RIBXRA##+1(T1) ;CLEAR POINTER TO NEXT(NON-EXISTANT) RIB
MOVE T2,RIBFIR##+1(T1) ;GET AOBJN WORD
MOVE T2,DEVRB1##(F) ;GET FIRST POINTER IN RIB
MOVEM T2,(T4) ;SAVE FIRST POINTER IN RIB
SETZM DEVRB1##(F) ;CLEAR THE POINTER LOCATION
PUSHJ P,GRPAD ;COMPUTE DISK ADDRESS FROM POINTER
MOVEM T2,RIBSLF##+1(T1) ;SAVE IN THE RIB
PUSHJ P,WRTRIB ;WRITE THE RIB
MOVEI T1,DEPWRT## ;ITS OK FOR FNDDDB TO
IORM T1,DEVWRT##(F) ; SEE US AGAIN
JRST TPOPJ1## ;GOOD RETURN
;HERE WHEN THIS RIB ALREADY IS EXTENDED
EXTRB2: PUSH P,T2
PUSHJ P,WRTRIB ;WRITE CURRENT RIB (NEW CHECKSUMS
POP P,DEVRIB##(F) ;SET UP POINTER TO NEXT RIB
PUSHJ P,RIBCUR ;READ THE NEXT RIB
PJUMPN T3,CPOPJ## ;ERROR READING RIB IF T3 NON-ZERO
JRST CPOPJ1## ;HAPPY RETURN
;HERE WHEN THE RIB CAN'T BE EXTENDED
EXTRB3: PUSHJ P,WRTRIB ;WRITE THE RIB (WITH NEW PNTRS)
MOVEI T1,DEPWRT## ;ITS OK FOR FNDDDB TO
IORM T1,DEVWRT##(F) ; SEE US AGAIN
IFN FTDSIM,<
PUSHJ P,DOWNIF ;GIVE UP DA IF WE OWN IT
>
PJRST GVMNB0 ;GIVE UP MON-BUF AND RETURN
> ;END CONDITIONAL ON FTDMRB
;SUBROUTINE TO CONVERT A RETRIEVAL POINTER
;ENTER WITH T2=POINTER
;EXIT CPOPJ LF BAD UNIT-CHANGE PNTR
;EXIT CPOPJ1 (WITH NEW U SET UP) IF CHANGE-UNIT POINTER
;EXIT CPOPJ2 WITH DEVBLK SET AND T1=COUNT IF A REAL POINTER
CNVPTR::TLNE T2,-1 ;*REAL POINTER?
JRST CNVPT2 ;*YES
TRZ T2,RIPNUB## ;*CHANGE UNIT. REMOVE BIT 18
CNVPT1: PUSHJ P,NEWUNI ;*SET U, DEVUNI
TDZA T2,T2 ;*INVALID U - SET TO 0
JRST CPOPJ1## ;*OK - RETURN
SOS (P)
JRST CNVPT1 ;*SET U TO 1ST UNIT IN STR AND RETURN
CNVPT2: MOVE T4,UNISTR##(U) ;*STR DB LOCATION
LDB T1,STYCLP##(T4) ;*CLUSTER ADDRESS
LDB T3,UNYBPC## ;*BLOCKS PER CLUSTER
IMUL T1,T3 ;*BLOCK ADDR
MOVEM T1,DEVBLK##(F) ;*SAVE IN DDB
LDB T1,STYCNP##(T4) ;*GROUP COUNT FIELD
IMUL T1,T3 ;*BLOCK COUNT
JRST CPOPJ2## ;*RETURN
;SUBROUTINE TO RESET U, DEVUNI(F) TO A NEW VALUE
;ENTER WITH LOGICAL UNIT NUMBER IN T2
;EXIT WITH U, DEVUNI SET TO NEW VALUE
;RETURNS CPOPJ IF NO SUCH UNIT, CPOPJ1 NORMALLY
;NEWUX SAME, PRESERVES T1,T3,T4
NEWUX:
NEWUNI::HRRZ U,UNISTR##(U) ;*LOC OF FILE STRUCTURE DB
JUMPE U,CPOPJ## ;*RETURN NON-SKIP IF NOT IN A F/S
HLRZ U,STRUNI##(U) ;*LOC OF UNIT 0 IN STRUCTURE
;SUBROUTINE TO RESET U,DEVUNI(F) TO A NEW VALUE
;ENTER WITH POINTING TO 1ST UNIT IN STR
;EXIT WITH U,DEVUNI(F) SET TO NEW VALUE
;T1,T3,T4 RESPECTED
NEWUN:: SOJL T2,NEWUN2 ;*DONE IF T2 COUNTS OUT
HLRZ U,UNISTR##(U) ;*STEP TO NEXT UNIT OF STRUCTURE
JUMPN U,NEWUN ;*TRY NEXT
JRST NEWUN3 ;*DESIRED UNIT WAS HIGHER THAN LAST UNIT IN STRUCTURE
NEWUN2: HRRM U,DEVUNI##(F) ;*SET DDB
HRRZ J,UNIKON##(U) ;*SET UP J
JRST CPOPJ1## ;*AND EXIT
NEWUN3: TRO S,IOBKTL ;*ILLEGAL UNIT - LIGHT ERROR BIT
JRST STRIOS ;*SAVE S AND ERROR RETURN
SUBTTL FILINT - INTERRUPT HANDLING MODULE
FILINT::PUSHJ P,SAVE4## ;SAVE P1-P4
PUSH P,T3 ;SAVE DATAI WORD
PUSH P,T2 ;SAVE CONI WORD
PUSH P,T1 ;SAVE COMMUNICATION WORD
;SET UP P1 AS A GLOBAL IN FILINT MODULE = ADDRESS OF CHAN DATA BLOCK
HRRZ P1,KONCHN##(J) ;SET UP P1=LOC OF CHAN DB
HLLZ P2,T1 ;GET INTERRUPT BITS
TLZ P2,1777 ;MASK OUT UNIT
ANDCAM P2,(P) ;MASK OUT OF COMMUNICATION WORD
SETZM P4 ;P4 WILL KEEP THE DRIVE NUMBER
POSTST: JFFO P2,.+2 ;ANY MORE POSITIONS?
JRST POSDON ;NO, CLEAN UP TRANSFER
LSH P2,1(P3) ;YES. SET P2 FOR NEXT DRIVE TEST
ADDB P4,P3 ;COMPUTE THE DRIVE
SKIPN U,@KONPTR##(J) ;SET U=UNIT BLOCK ADR
JRST FINPS2 ;NO UNIT BLOCK - IGNORE THE INTERRUPT
IFN FTDUAL,<
MOVE T1,UNICYL##(U) ;IF THIS IS PART OF DUAL-PORT DRIVE
SKIPE T2,UNI2ND##(U)
MOVEM T1,UNICYL##(T2) ;THEN BOTH UNITS ARE ON SAME CYL
>
SKIPE T1,UNISTS##(U) ;GET STATE OF UNIT
CAIL T1,OWCOD## ;OPERATOR WAIT?
JRST FREINT ;IDLE OR OPR WAIT
JUMPL T1,RECALH ;IF NEGATIVE WE'RE IN PHUNG RECOVERY
CAIN T1,TCOD## ;IS UNIT IN TRANSFER STATE?
JRST RECALR ;YES - POSITIONING INT. FROM XFERRING DRIVE
CAIE T1,SCOD## ;NO, WAS UNIT S?
JRST FINPS1 ;NO,
SETZM UNISTS##(U) ;YES, SET STATUS =0 (IDLE)
SKIPN UNIQUE##(U) ;ANY POSITIONS WAITING?
AOJA P4,POSTST ;NO. TEST NEXT POSITION BIT
PUSH P,KONCUA##(J) ;YES, SAVE KONCUA (STARTING
; A POSITION WILL CHANGE IT)
PUSHJ P,UNIPOS ;START UNIT POSITIONING (FORGET SEEK)
POP P,T1 ;KONCUA
MOVE T2,UNISTS##(U) ;STATE OF UNIT
CAIE T2,TCOD## ; TRANSFER?
MOVEM T1,KONCUA##(J) ;NO, RESET KONCUA (CHANGED
; SINCE UNIT WAS POSITIONED, POSDON
; WILL CHECK STATE OF UNIT
AOJA P4,POSTST ;AND GO TEST NEXT POSITION BIT
;CALLED BY KON ROUTINE WHEN A UNIT GOES OFF-LINE
;PRESERVES T1
FILDN:: MOVEI T2,O2COD## ;IF IDLE SAY OFF-LINE, NO MESSAGE
SKIPN UNISTS##(U) ; (IF BUSY HNGDSK WILL CATCH IT)
MOVEM T2,UNISTS##(U) ; AND RETURN
POPJ P,
;HERE ON AN UNSOLICITED INTERRUPT, UNIT IDLE OR OPR WAIT
FREINT: SKIPG KONPOS##(J) ;DOES UNIT DO POSITIONING?
STOPCD FINPS2,DEBUG,FDP, ;++FIXED-HEAD DEVICE POSITION
MOVSI T2,UNPUNO## ;CLEAR 'OFF-LINE' AND FILE UNSAFE BITS
ANDCAM T2,UNIDES##(U)
OWRSET: CAIE T1,OW2COD##
CAIN T1,OWCOD## ;WAS THERE A FILE IN T STATE ON THIS UNIT?
IFE FTDUAL,<
JRST OWRSE1 ;YES, ASSUME POWERED UP IN RESPONSE TOO TELOPR
>
IFN FTDUAL,<
JRST OWRSED
>
SKIPE DINITF## ;IF STILL IN ONCE-ONLY
JRST FREIN2 ; DONT DO ANYTHING FANCY
SKIPE UNILOG##(U) ;IS THIS PACK KNOWN TO THE SYSTEM?
JRST FREIN1 ;YES, REREAD HOME BLOCKS
IFN FTMDA,<
PUSHJ P,CALMDA ;NO, TELL MOUNTABLE DEVICE ALLOCATOR
JRST FREIN2 ;NOT RUNNING, CONTINUE
PUSHJ P,SET4MD ;SET SO ONLY MDA CAN READ DRIVE
AOJA P4,POSTST ;AND TEST NEXT DRIVE
>
IFE FTMDA,<
JRST FREIN2
>
;HERE IF PACK KNOWN TO SYSTEM
FREIN1: MOVSI T2,UNPRHB## ;ALREADY GOTTEN AN INTERRUPT FROM DRIVE?
TDNN T2,UNIDES##(U)
AOS HOMFLG## ;NO, COUNT UP NO OF HOMES TO READ
IORM T2,UNIDES##(U) ;INDICATE THIS UNIT NEEDS REREADING
SETZM UNISTS##(U) ;MAKE IDLE SO WONT START A SEEK
AOJA P4,POSTST ;AND GO TEST NEXT UNIT
FREIN2: SKIPE T1,UNIQUE##(U) ;FILES WAITING TO START POSITIONING?
MOVEI T1,PWCOD## ;YES
MOVEM T1,UNISTS##(U) ;SET UNIT IDLE OR PW
JUMPE T1,FINPS2 ;TEST NEXT UNIT IF IDLE
PUSHJ P,SETID1 ;POSITIONERS WAITING - START ONE GOING
AOJA P4,POSTST ;AND TEST NEXT UNIT
;SUBROUTINE TO SET THINGS UP SUCH THAT ONLY MDA CAN READ A UNIT
SET4MD: MOVEI T1,MDACOD##
MOVEM T1,UNISTS##(U) ;SET SO ONLY MDA CAN READ THE DRIVE
MOVSI T1,UNPWMD## ;SET THAT UNIT IS WAITING FOR MDA
IORM T1,UNIDES##(U) ; SO WILL GO BACK TO MDA WAIT AFTER A READ
POPJ P, ;AND RETURN
;HERE FOR POSITION INTERRUPT, DRIVE IN TRANSFER STATE
;THIS HAPPENS ON RECALIBRATE AFTER DATA ERROR
RECALR: MOVE F,UNICDA##(U) ;CURRENT DDB
MOVE S,DEVIOS(F) ;S WORD (SO STARTE WILL KNOW IF READ OR WRITE)
SKIPL T1,CHNECT##(P1) ;RECALIBRATE INTERRUPT?
JRST RECAL1 ;NO - START DATA AGAIN
SETZM CHNECT##(P1) ;YES - REPOSITION DRIVE
IFN FTRP04,<
AOJN T1,RECAL1 ;IF CHNECT=-1,,0 OFFSET COMPLETED
>
RECALH: HRRZS UNISTS##(U) ;CLEAR SIGN BIT (PHUNG RECOVERY)
PUSHJ P,@KONPOS##(J) ;START POSITION
PUSHJ P,BADUNI ;DRIVE IS NOW DOWN
AOJA P4,POSTST ;GO HANDLE OTHER ATTENTION INTERRUPTS
RECAL1: PUSHJ P,STARTE ;START DATA TRANSFER
AOJA P4,POSTST ;AND TEST NEXT ATTENTION BIT
IFN FTDUAL,<
OWRSED: SKIPLE T2,UNI2ND##(U) ;IF THIS INTERRUPT ON A-PORT
CAME T1,UNISTS##(T2) ; AND B-PORT IS BUSY
CAIA
JRST FINPS2 ; WAIT FOR INTERRUPT ON B TO START IO
HRRZ T1,(P) ;COMMUNICATION OWRD
TRZE T1,IODTER+IODERR ;IS AN ERROR UP
CAIE T1,OPPOS## ; AND JUST A POSITION COMPLETE?
JRST OWRSE2 ;NO, "REAL" FREE INTERRUPT
LDB T1,[POINT 3,(P),17] ;YES, DID THIS DRIVE HAVE THE ERROR
CAIN T1,(P4)
JRST FINPS2 ;YES, IT ISN'T THE INTERRUPT WE WANT
OWRSE2: PUSH P,U ;NO, FREE INTERRUPT
JRST OWRSE1 ;GO REDO THE OPERATION
>
;HERE FOR POSITION INTERRUPT, DRIVE NOT IN S OR T STATE
FINPS1: TRNE T1,1 ;IS UNIT CURRENTLY IN A WAIT STATE?
JRST FINPS2 ;YES, SPURIOUS INTERRUPT
HRRZ F,UNICDA##(U)
JUMPE F,FINPS2 ;FORGET IT IF SPURIOUS
MOVE S,DEVIOS(F)
TLNE S,IOSMON ;MONITOR OR SWAPPER IO?
AOSA UNIMSC##(U) ;YES. COUNT 1 MORE MONITOR OR SWAP SEEK
AOS UNIUSC##(U) ;NO. COUNT 1 MORE USER SEEK
SKIPN DEVUNI##(F) ;UNIT BEEN YANKED?
JRST FINPS3 ;YES
IFN FTDUAL,<
PUSH P,U
>
CAIN T1,PCOD## ;IF UNIT WAS IN POSITION,
OWRSE1: PUSHJ P,SETTTW ;SET FILE TO TW, OR T AND START TRANSFER
IFN FTDUAL,<
POP P,U ;IF WE STARTED IO ON ALTERNATE UNIT
HRRZ J,UNIKON##(U) ;THEN U, J AND P1 WERE CHANGED
HRRZ P1,KONCHN##(J) ;SO RESET THEM
>
FINPS2: AOJA P4,POSTST ;GO TEST NEXT POSITION INTERRUPT
;HERE IF UNIT WAS YANKED DURING A SEEK
FINPS3: MOVEI U,0
PUSHJ P,UNIYNK ;FIX UP DDB,UDB
AOJA P4,POSTST ; AND TEST NEXT POSITION INT.
;SUBROUTINE TO PUT THE DRIVE POINTED TO BY U IN T OR TW
;P1= CHAN DATA BLOCK ADR.
;IF PUT THE DRIVE IN T1, START TRANSFERRING DATA
;UUOTWQ IS CALLED FROM UUO LEVEL
SETTTW: DSKOFF ;TURN ALL DISK PI'S OFF
HRRZ F,UNICDA##(U) ;*SET F TO FILE
IFN FTVM,<
MOVE T1,UNIBLK##(U) ;IN CASE OF SWAPPING ON 2 DF10'S
SKIPL DEVSWP##(F)
JRST UUOTWQ
MOVEM T1,DEVBLK##(F) ; DEVBLK MAY BE TIMESHARED
PUSHJ P,SWPCHK## ;CAN WE START IO NOW?
JRST SETTW0 ;NO, QUEUE TILL LATER
>
UUOTWQ:
IFN FTDHIA,<
SKIPE DIADSK## ;SHUTTING DOWN IO
CAME P1,DIACHN## ; FOR THIS CHANNEL?
JRST UUOTWR ;NO
CONSO PI,PIPROG## ;YES, ON UUO LEVEL?
JRST SETTW0 ;YES, PUT REQUEST IN TW QUEUE
UUOTWR:>
AOSG @KONCHN##(J) ;#CHAN AVAILABLE?
IFN FTDUAL,<
JRST UUOTWS ;#YES, START IO
SKIPE T1,UNI2ND##(U) ;#NO, IS THIS A DUAL-PORTED DRIVE?
SKIPE UNISTS##(T1) ;#YES, IS THE 2ND DRIVE USEABLE?
JRST SETTW0 ;#NO, QUEUE THE REQUEST ON 1ST CHAN
LDB T2,UNZUST## ;STATUS OF OTHER PORT
CAIN T2,UNVDWN## ;DOWN? (DETACH COMMAND)
JRST SETTW0 ;YES, DON'T TRY TO USE IT
MOVE T2,UNIKON##(T1) ;#YES, T1=UNIT T2=KON
SKIPL KONTAB##(T2) ;KONTROLLER IDLE?
AOSE @KONCHN##(T2) ;#YES, CHAN IDLE?
JRST SETTW0 ;CANT START IT NOW
HRRM F,UNICDA##(U) ;SAVE DDB IN MAIN UDB
MOVE T3,UNISTR##(U) ;WILL NEED UNISTR LATER
MOVEM T3,UNISTR##(T1) ;(CHKSUM), SO SET IT UP
MOVE T3,UNIBPC##(U) ;WILL NEED UNIBPL
MOVEM T3,UNIBPC##(T1) ; IF WE GO TO NXTBLK
HRRZ J,T2 ;WE CAN START THE IO ON THE ALTERNATE UNIT
HRRZ U,T1 ; SO SET UP NEEDED ACS
MOVE P1,KONCHN##(J)
UUOTWS:>
IFE FTDSIM,<
PJRST STRTIO ;#YES, START IO
>
IFN FTDSIM,<
PJRST SETBS1 ;HAVE TO CHECK FOR WRITE/USETO RACE
>
SETTW0: MOVEI T1,TWCOD## ;#TRANSFER WAIT STATE CODE
DPB T1,DEYCOD## ;MAKE DDB TW
IFN FTVM,<
SKIPL DEVSWP##(F) ;#IF THIS IS THE SWAPPER,
JRST SETTW1
SETZM UNISTS##(U)
JRST PUTQU3 ;# DONT PUT IN CHAN QUEUE (START IT AT SWPPIK ONLY)
SETTW1:
>
;#PUT IN TWQ AND RETURN
CONSO PI,PI.IPA-PI.IP7 ;IF ON UUO LEVEL,
SKIPN UNISTS##(U) ; DON'T CHANGE UNISTS IF NON-0
MOVEM T1,UNISTS##(U) ;ELSE MAKE UNIT TW STATE
IFN FTDUAL,<
PUSHJ P,SECCOD ;#MAKE OTHER UNIT STATE = THIS ONE
HRRM U,DEVCUR##(F) ;#SAVE CURRENT UNIT IN DDB
>
MOVEI T1,CHNQUE##(P1) ;#T1= QUEUE ON WHICH TO PUT FILE
IFN FTDUAL,<
REPEAT 0,< ;NOT NEEDED UNTIL WE GET A FIXED-HEAD DUAL-PORTED DRIVE
SKIPLE T2,UNI2ND##(U) ;IF A MAIN-UNIT
SKIPN UNISTS##(T2)
JRST PUTQUX ; WHICH HAS AN ALTERNATE UNIT
MOVE T1,UNICHN##(T2) ;IF THE ALTERNATE UNIT IS BUSY THE REQUEST
MOVEI T1,CHNQUE##(T1) ; SHOULD BE PUT ON THE ALTERNATE CHAN QUEUE
> ;END REPEAT 0
> ;END FTDUAL
;AND FALL INTO PUTQUX
;SUBROUTINE TO ENTER A FILE IN A QUEUE
; PUTQUX FOR CHANNEL (XFER WAIT) QUEUES
; P1=ADDR. CHAN. DATA BLOCK
; PUTQUE FOR UNIT (POSITION WAIT) QUEUES
; U=ADDR. UNIT DATA BLOCK
;C(T1)=LOC OF QUEUE POINTER F POINTS TO FILE
;LH OF QUEUE POINTER CONTAINS FIRST FILE (DDB) ADR., RH=0
;THESE ARE REALLY LISTS NOT QUEUES, SINCE ARE REMOVED IN RANDOM ORDER
PUTQUX:
IFN FTDSTT,< ;IF KEEPING DISK STATISTICS
AOSA CHNQUL##(P1) ; INCREMENT XFER QUEUE LENGTH
>
PUTQUE:
IFN FTDSTT,<
IFE FTVM,<
AOS UNIQUL##(U) ; OR POSITION QUEUE
>
IFN FTVM,<
AOSA UNIQUL##(U)
>
>
IFN FTVM,<
AOS CHNWAT##(P1)
>
IFE FTDPRI,<
MOVE T2,(T1) ;CURRENT FIRST IN "QUEUE"
HRLM F,(T1) ;MAKE THIS FIRST
HLLM T2,DEVQUE##(F) ;POINT THIS AT NEXT
; TO BE ADR. FILE(DDB)BEING ADDED
>
IFN FTDPRI,< ;IF DISK PRIORITY
MOVEI T4,MINDVQ##(T1) ;SET TO POINTER-OFFSET (FIRST PREDECESSOR)
SKIPN (T1) ;EMPTY?
JRST PUTQU2 ;YES, GO
PUSHJ P,SAVE1## ;NO, SAVE P1
PUSHJ P,DFPRI ;COMPUTE PRIORITY
MOVE P1,T2 ;IN P1
HLRZ T3,(T1) ;FIRST ITEM IN QUEUE
PUTQU1: PUSHJ P,D3PRI ;PRIORITY OF FILE IN QUEUE
CAMGE T2,P1 ;IS NEW ENTRY HIGHER?
JRST PUTQU2 ;YES. INSERT IT HERE
MOVE T4,T3 ;NO, SAVE AS PREDECESSOR
HLRZ T3,DEVQUE##(T3) ;STEP TO NEXT IN QUEUE
JUMPN T3,PUTQU1 ;TEST THAT ONE
PUTQU2: HLLZ T2,DEVQUE##(T4) ;POINTER TO NEXT IN QUEUE
HRLM F,DEVQUE##(T4) ;LINK THIS FILE TO IT
HLLM T2,DEVQUE##(F) ;INSERT NEW LINK TO NEXT
>
PUTQU3: DSKON ;#TURN ALL DISK PI'S BACK ON
SCHEDULE
SETZ T1, ;SET HUNG-TIME TO ZERO
DPB T1,PDVCNT## ; SO WONT TIME OUT WHILE IN A QUEUE
POPJ P, ;AND RETURN
IFN FTDPRI,<
;SUBROUTINE TO COMPUTE PRIORITY OF A DISK FILE
;ENTER AT D3PRI IF DDB ADR IN T3, AT DFPRI IF IN F
;EXIT WITH T2=PRIORITY. RESPECTS ALL ACS EXCEPT T2
D3PRI: LDB T2,DEZPRI## ;PNTR INDEX = T3
CAIA
DFPRI: LDB T2,DEYPRI## ;INDEX=
TRZE T2,MINDPR ;PRIORITY NEGATIVE?
MOVNS T2 ;YES, MAKE T2 NEGATIVE
POPJ P, ;AND RETURN
>
;ROUTINE TO SET UP R (POSSIBLY AN INTERRUPT LEVEL)
ADRINT: LDB T1,PJOBN## ;JOB NUMBER
SKIPN R,JBTADR##(T1) ;GET ADDRESS
SKIPG DEVSWP##(F) ;NONE - SWAPPER OR REREADING HOME BLOCKS?
POPJ P, ;HAVE R=ADDR, OR SWAPPER
SKIPE DINITF## ;NOTHING - ONCE CODE?
POPJ P, ;YES, OK RETURN
STOPCD CPOPJ##,DEBUG,JNC, ;++JOB NOT IN CORE
;SUBROUTINE TO START IO ON FILE POINTED TO BY F
;P1= CHAN. DATA BLOCK ADR.
;ALSO CALLED AT UUO LEVEL
STRTIO::MOVSI T1,400000 ;#SET KONTROL BUSY
ORM T1,KONTAB##(J) ;#
IFN FTPDBS,<
IOSMON==IOSMON ;FOR CREF. BIT IS ALREADY IN T1
SKIPGE DEVSWP##(F) ;IS THIS THE SWAPPING DDB?
IORM T1,DEVIOS(F) ;YES--MAKE SURE IOSMON IS SET IN DDB.
> ;END FTPDBS
MOVEI T1,TCOD## ;#SET FILE, UNIT TO T1
PUSHJ P,FILCOD ;SETPAR EXPECTS T1 TO STILL CONTAIN TCOD
; FOR METER POINT
PUSHJ P,SETPAR ;SET KONCUA,UNICDA,UNIBLK
IFN FTDUAL,<
HRRM U,DEVCUR##(F) ;SAVE CURRENT UNIT IN DDB
PUSHJ P,SECCOD ;MAKE OTHER UNIT STATE = THIS ONE
>
DSKON ;#TURN ALL DISK PI'S BACK ON
SCHEDULE
MOVE S,DEVIOS(F) ;S WORD
PUSHJ P,ADRINT ;SET UP R FROM JBTADR
PUSHJ P,SETLST ;SET UP IOLST, NUMBER OF BLOCKS
IFN FTKL10,<
PUSHJ P,CFDMP ;INVALIDATE CACHE, VALIDATE MEMORY
>
SETZM CHNECT##(P1) ;ZERO ERROR COUNT LOC
SETZM CHNRCT##(P1) ;ZERO RECALIBRATE-COUNT
SETZM UNIRCV##(U) ;ZERO HUNG UNIT RETRY-COUNTER
MOVSI T1,UNPFIR## ;SET SIGN BIT IN UNIECT
ORM T1,UNIECT##(U) ; TO INDICATE NO ERRORS YET
JUMPL S,STARTE ;NO CHECKSUMS IF MONITOR IO
TLNE S,IO ;DATA - READING?
TLZN S,IOSFIR ;WRITING DATA - CHECKSUM TIME?
JRST STARTE ;NO
MOVEM S,DEVIOS(F) ;YES, SAVE S IN DDB
PUSHJ P,CHKSUM ;COMPUTE CHECKSUM
SKIPN T2,@DEVRET##(F) ;GET RETRIEVAL PNTR
MOVE T2,DEVRB1##(F) ;JUST 1 PNTR IN DDB, STORED IN DEVRB1
MOVE T3,UNISTR##(U) ;LOC OF STR DB
MOVE T4,T2 ;SAVE OLD POINTER
DPB T1,STYCKP##(T3) ;SAVE CHKSUM IN PNTR
CAMN T4,T2 ;DID CHECKSUM CHANGE?
JRST STARTE ;NO, CRANK UP THE IO
SKIPN @DEVRET##(F) ;YES, JUST 1 PNTR IN DDB?
MOVEM T2,DEVRB1##(F) ;YES. SAVE NEW PNTR BACK OVER OLD
SKIPE @DEVRET##(F)
MOVEM T2,@DEVRET##(F) ;SAVE PNTR IN DDB
HRRZ T1,DEVACC##(F) ;ACCESS TABLE
MOVE T3,DEVREL##(F) ;RELATIVE BLOCK OF FILE
SOJN T3,STRTI1 ;IF 1ST BLOCK,
MOVEM T2,ACCPT1##(T1) ; SO ANOTHER JOB CAN READ THIS FILE (UPDATE)
STRTI1: MOVE T3,ACCSTS##(T1) ;FILE STATUS
TRNN T3,ACPUPD ;FILE BEING UPDATED?
JRST STARTE ;NO
MOVEI T4,ACPSBC## ; SO DDBS WHICH HAVE POINTER IN CORE ALREADY
ORM T4,ACCSBC##(T1) ; WON'T GET SPURIOUS CHECKSUM ERRORS
IFN FTDSIM,<
;MIGHT GET A WRONG CHECKSUM INTO THE RIB OF A SIM-UPDATE FILE IF WE
; LEAVE A BAD CHECKSUM IN SOME OTHER DDB, SO FIND ALL WRITERS OF THE FILE
; AND CHANGE THE CHKSUMS IN THEIR DDB'S
LDB T3,ACYWCT## ;IF WE'RE THE ONLY WRITER,
SOJLE T3,STARTE ; FORGET IT
PUSH P,T2 ;OTHER WRITERS - SAVE THE NEW PNTR
MOVE T3,UNISTR##(U) ;SET UP A MASK FOR PNTR ADDRESSES
SETO T2,
LDB T2,STYCLP##(T3)
MOVEM T2,MSKLOC ;AND SAVE IT
PUSHJ P,FNDDDB ;FIND A WRITING DDB
JRST STRTI6 ;NONE THERE (SYSTEM ERROR?)
HRRZ T3,DEVUNI##(T2) ;UNIT
CAIE T3,(U) ;SAME UNIT?
JRST STRTI5 ;NO, FIND ANOTHER DDB
;HERE WHEN WE FOUND A WRITING DDB
STRTI2: MOVEI T1,DEVRB1##(T2) ;SET A POINTER FOR THE
HRLI T1,MPTRLN## ;RETRIEVAL PNTRS OF FOUND DDB
STRTI3: SKIPN T3,(T1) ;GET A RET PNTR FROM THE DDB
JRST STRTI4
XOR T3,(P) ;DOES IT MATCH NEW PNTR?
TDNE T3,MSKLOC
JRST STRTI4 ;NO, TRY NEXT PNTR
MOVE T3,(P) ;YES, SUBSTITUTE NEW PNTR
MOVEM T3,(T1) ;IN THE FOUND DDB
JRST STRTI5 ;AND TEST FOR ANOTHER WRITER
STRTI4: AOBJN T1,STRTI3 ;TES NEXT PNTR IN FOUND DDB
STRTI5: PUSHJ P,FNDDDN ;LOOK FOR ANOTHER WRITER
JRST STRTI6 ;NONE THERE, DONE
JRST STRTI2 ;FOUND ONE, LOOK FOR PNTR IN THIS DDB
STRTI6: POP P,(P) ;DONE - REMOVE PNTR FROM STACK
> ;END FTDSIM
STARTE::MOVEI T1,KONRDS##(J) ;SET TO READ - STOP ON ERROR FOR F.S.
; ALL BUT LAST TRY SO KNOW BAD BLOCK+WORD
TLNE S,IO
MOVEI T1,KONWTS##(J) ; WRITE DATA - STOP ON ERROR FOR F.S.
; ALL BUT LAST TRY SO KNOW BAD BLOCK+WORD
IFN FTDSUP,<
TLNN S,IOSUPR ;SUPER I/O MODE?
JRST STARTF ;NO, START THE IO
TRNE S,UDSX ;YES, READ/WRITE HEADERS AND DATA?
AOJA T1,STARTF ;YES, RDF/WTF = RDS/WTS +1
IFN FTRP04,<
MOVSI T2,DEPCPT## ;NO, COMPATABILITY MODE?
TDNE T2,DEVCPT##(F)
ADDI T1,2 ;YES, ENTRY POINT = RDS(WTS)+2
>>
STARTF: MOVEM S,DEVIOS(F) ;SAVE S
DSKOFF ;MAKE SURE NO INTERRUPTS HAPPEN
PUSHJ P,@(T1) ;#START READ OR WRITE
JRST BADUNI ;#UNIT NOT OK
DSKON ;#
POPJ P,
;SUBROUTINE TO SET STATE OF FILE,UNIT TO C(T1)
;THIS ROUTINE RESPECTS AC T1
FILCOD: DPB T1,DEYCOD## ;#SET STATE OF FILE
MOVEM T1,UNISTS##(U) ;#SET STATE OF UNIT
POPJ P, ;#AND RETURN
IFN FTDUAL,<
;ROUTINE TO SET 2ND UNIT'S STATE CODE = THIS UNIT'S
SECCOD: SKIPL T1,UNI2ND##(U) ;IS THIS AN ALTERNATE UNIT?
POPJ P, ;NO
MOVE T2,UNISTS##(U) ;YES, GET STATE WORD OF UNIT
MOVEM T2,UNISTS##(T1) ;AND SAVE IN UDB FOR PRIME UNIT
POPJ P, ;AND RETURN
>
;SUBROUTINE TO SET UP AN IOLIST BLOCK
;ENTER WITH J,U,F AND S SET UP
;P1=CHAN.DATA BLOCK ADR.
;ALSO CALLED AT UUO LEVEL
;RETURNS WITH THE LIST IN CORE, POINTED TO BY @KONIOC(J)
SETLST: PUSHJ P,NXTBLK ;GET NEXT BLOCK OF FILE
STOPCD .,JOB,BWA, ;++BLOCK WENT AWAY
IFN FTKA10,<
MOVE T4,KONCOM##(J) ;AOBJN POINTER FOR IO LIST BLOCK
>
IFE FTVM,<
HRRZM T4,@KONIOC##(J) ;SET KONIOC TO LOC OF IOLIST BLOCK
>
IFN FTKI10!FTKL10,<
PUSHJ P,SAVE4## ;YES, SAVE P1-P4
HRLZ P3,J ;SAVE J
HRR P3,P1 ;SAVE CHAN DB ADR
TLO P3,400000 ;TELL MAPIO TO STORE EXPECTED TERM WD
LDB J,PJOBN## ;JOB NUMBER
IFN FTSWAP,<
IFE FTVM,<
SKIPGE DEVSWP##(F) ;SWAPPER?
MOVM J,FINISH## ;YES, GET THE SEGMENT NUMBER
>
IFN FTVM,<
SKIPLE DEVSWP##(F) ;"THIS" MAKES JOB ADDRESSABLE ITSELF
>
> ;END FTSWAP
PUSHJ P,SVEUB## ;SET UP UBR
SETZB P1,P2 ;P1=FREE CORE LOC
> ;END FTKI10
PUSH P,U ;SAVE U
MOVE T2,DEVBLK##(F) ;BLOCK TO START AT
HRRZ T1,UNIBPY##(U) ;NUMBER OF BLOCKS PER CYLINDER
IDIVI T2,(T1) ;T3=RELATIVE BLOCK IN CYL
SUBI T1,(T3) ;T1=DISTANCE TO END OF CYLINDER
CAMLE T1,MAXTRN## ;GTR THAN THE MAX NUMBER OF BLOCKS ALLOWED?
MOVE T1,MAXTRN## ;YES, REDUCE COUNT (SO HIGH PRIORITY REQUESTS
; WONT GET LOCKED OUT TOO LONG)
HRRZ T3,DEVLFT##(F) ;NO OF BLOCKS LEFT IN CURRENT GROUP
SKIPL S ;ASSUME ALL BLOCKS TO EOL OK IF MON MODE
CAMLE T3,T1 ;MORE THAN TO END OF CYLINDER?
MOVE T3,T1 ;YES, REDUCE COUNT
TLNE S,IO+IOSUPR+IOSMON ;READING DATA?
JRST SETLS0 ;NO. USE ALL OF GROUP
MOVE T2,DEVACC##(F) ;YES. GET LOC OF A.T.
MOVE T2,ACCWRT##(T2) ;NO OF DATA BLOCKS IN FILE
SUB T2,DEVREL##(F) ;NO OF DATA BLOCKS LEFT (FROM CURRENT POSITION)
CAIGE T2,-1(T3) ;MORE IN GROUP THAN HAVE DATA?
MOVEI T3,1(T2) ;YES, ONLY READ BLOCKS WITH DATA
SETLS0: JUMPL S,SETDMP ;SAME AS DUMP IF MONITOR MODE
LDB T1,PIOMOD##
CAIL T1,SD ;NO, DUMP MODE?
JRST SETDMP ;YES
HRLZ T1,T3 ;BUFFERRED MODE. NUMBER OF BLOCKS LEFT IN GROUP
MOVNS T1 ;SET FOR AOBJN POINTER
TLNN S,IO ;READING?
SKIPA T2,DEVIAD(F) ;YES. USE DEVIAD
MOVE T2,DEVOAD(F) ;WRITING, USE DEVOAD
HRLI T2,MBLKSZ## ;SET LH(T2) FOR AN IOWD
HLRZ U,R ;SET U TO USERS PROTECTION
SUBI U,BLKSIZ## ;-200 FOR ADRRESS CHECK
MOVEI T3,1(T2) ;STARTING BUFFER LOC
HRLM T3,DEVUVA##(F) ;SAVE UVA-1 OF CHKSUM WORD
SUBI T3,1
SETLS1: CAMLE T2,[XWD MBLKSZ##,JOBPFI##] ;IS BUFFER LOC ABOVE JOBPFI
CAIGE U,(T2) ;AND BELOW TOP OF USER'S SPACE?
JRST SETLS5 ;NO - LET UUOCON GIVE ADR ERROR MESSAGE
IFN FTKA10,<
ADDI T2,(R) ;YES, RELOCATE
>
IFN FTVM,<
EXCTUU <MAP P4,-1(T2)> ;IS THE PAGE IN CORE?
IFN FTKI10,<
TRNN P4,420000
>
IFN FTKL10,<
PUSHJ P,SFLTST ;CALL COPY OF FLTST FOR FILSER
>
EXCTUU <MAP P4,BLKSIZ+1(T2)>
IFN FTKI10,<
TRNN P4,420000
>
IFN FTKL10,<
PUSHJ P,SFLTST ;CALL COPY OF FLTST FOR FILSER
>
JRST SETL1B ;NO PAGE FAILURE, CONTINUE
JRST SETLS5 ;PAGE FAILURE WILL OCCURR, TERMINATE THE IO LIST
SETL1B:> ;END FTVM
TLNN S,IO
JRST SETLS2
EXCTUX <SKIPL (T2)> ;WRITING - IS NEXT BUFFER FULL?
JRST SETL5S ;NO, DONE
JRST SETLS3 ;YES. SET IOWORD FOR BUFFER
SETLS2: EXCTUX <SKIPG (T2)> ;READING - IS NEXT BUFFER FREE?
JRST SETL5S ;NO. DONE
SETLS3: ADDI T2,1 ;INCREMENT BY 1 FOR WORDCOUNT WORD
IFN FTKI10!FTKL10,<
JUMPGE S,SETL30 ;IF MONITOR I/O,
MOVE P4,T2 ;SAVE IOWD IN P4
IFN FTKL10,<
MAP T2,(T2) ;CONVERT TO UNMAPPED FOR MAPIO
HLL T2,P4 ;GET WORD COUNT BACK
>
SETL30: ADDI T2,(R)
HRLI T2,MBLKSZ##
MOVEI P4,0
PUSHJ P,MAPIO## ; STORE THE IOWD IN FREE-CORE
JRST PUNTB ;NOT ENOUGH MONITOR FREE-CORE
SUBI T2,(R)
HRLI T2,MBLKSZ##
> ;END FTKI10!FTKL10
IFN FTKA10,<
MOVEM T2,(T4) ;SAVE THE IOWD
>
TLNE S,IO ;READING?
JRST SETL3C ;NO
EXCTXU <HLREM T2,(T2)> ;YES, SAVE -WDCNT IN BUFFER WORD 0
EXCTUU <MOVNS (T2)> ;SET +WDCNT
SETL3C: EXCTUX <HRR T2,-1(T2)> ;STEP TO NEXT BUFFER
CAIE T3,(T2) ;BACK TO THE BUFFER WE STARTED AT?
TRNE S,IOCON ;OR DISCONTINUOUS MODE?
AOJA T4,SETLS4 ;YES, IOWDS ARE ALL SET
IFN FTKA10,<
AOBJP T4,SETLS4 ;COUNT THE IOWD, THROUGH IF NO MORE ROOM IN BLOCK
>
AOBJN T1,SETLS1 ;COUNT THE BLOCK AND TRY FOR ANOTHER IOWD
JRST SETLS5 ;NO BLOCKS LEFT IN THIS POINTER
IFN FTKL10&FTVM,<
;COPY OF FLTST FOR FILSER. CALL WITH RESULT OF MAP INSTRUCTION
; IN P4. SKIP RETURN IF FAULT WILL OCCUR, NON SKIP IF OK.
; (THIS IS BACKWARDS W.R.T. THE REAL FLTST)
SFLTST: TLNN P4,(1B8) ;PAGED REF?
POPJ P, ;NO, WILL NOT FAULT
TLNN P4,(1B1) ;BAD FAIL OR
TLNN P4,(1B2) ;NO ACCESS ALLOWED AND
JRST CPOPJ1## ;WE WILL FAULT
POPJ P, ;OTHERWISE WE WILL NOT
>;END IFN FTKL10&FTVM
;HERE WHEN THE IOLIST IS COMPLETELY SET UP
SETLS4: AOJA T1,SETLS5
SETL5S:
IFN FTKL10&FTMS,<
EXCH T1,T2 ;WE'VE TOUCHED THE T2-BUFFER,
PUSHJ P,OUCHE## ; WON'T GET IT OUT OF CACHE BELOW,
MOVE T1,T2 ; SO REMOVE IT FROM CACHE NOW
>
SETLS5:
IFN FTKI10!FTKL10,<
SETL5A: HLRZ J,P3 ;RESTORE KONTROLLER DB LOC
TRZ J,400000
IFN FTKL10,<
MOVEM T2,KONCNT##(J) ;SAVE ACTUAL WRDCNT IF DUMP-MODE
MAP P2,(P2) ;CONVERT VIRTUAL ADDRESS TO PHYSICAL ADDRESS
>
HRRZS P2 ;JUST ADDRESS (DF10-STYLE JUMP)
IFN FTKL10&FT22BIT,<
JUMPE P2,SETL5C ;ALL ZERO IF NO IOWDS
MOVE T2,CHB22B##(P3) ;RH20?
TLNE T2,CP.RH2##
TLO P2,RH2JMP## ;YES, MAKE IT THAT FLAVOR OF JUMP
>
SETL5C: MOVEM P2,@KONIOC##(J) ;SAVE LOC OF 1ST IOWD IN KONTROL BLOCK
IFN FTKL10&FT22BIT,<
IFN FTVM,<
SKIPGE DEVSWP##(F) ;SWAPPER?
JRST SETL5B ;YES, THIS ALREADY FIXED THINGS
>
TLNE P2,-1 ;RH20?
PUSHJ P,RH2ND## ;YES, FINISH THE LIST
>
SETL5B: MOVE P1,P3
SETLS6:> ;END FTKI10!FTKL10
IFN FTKA10,<
SETZM (T4) ;TERMINATE IO LIST
HLRE T2,-1(T4) ;GET NEGATIVE WORD COUNT
HRRZ T3,-1(T4) ;ISOLATE ADDRESS
SUB T3,T2 ;DETERMINE FINAL DATA ADDRESS
HRL T3,T4 ;GET IOWD LIST TERMINATION ADDRESS
MOVEM T3,CHNTCW##(P1) ;SAVE AS EXPECTED TERMINATION CONTROL WD
>
SETLS7: HRRZM T1,CHNNUM##(P1) ;NUMBER OF BLOCKS TO TRANSFER
; CLEAR LH (ERROR FLAG)
IFE FTKL10&FTMS,<
JRST UPOPJ##
>
IFN FTKL10&FTMS,<
JUMPE P2,UPOPJ## ;DONE IF NO LIST
SETLS8: SKIPN T1,(P2) ;GET NEXT IOWD
JRST UPOPJ## ;DONE IF 0
TLNN T1,577777 ;IS IT A GOTO?
JRST [HRR P2,T1 ;YES, POINT AT WHERE IT GOES
JRST SETLS8] ;AND GO TEST THAT IOWD
TLNN P2,-1 ;POINT T1 AT 1ST DATA LOC -2
SOSA T1 ; (POINTER TO NEXT BUFFER)
SUBI T1,2 ;GET THAT LOC OUT OF THE CACHE
PUSHJ P,OUCHE## ;SINCE WE WON'T SWEEP WE MUST CLEAR VALI
AOJA P2,SETLS8 ; FOR THAT CACHE LINE
>
;HERE TO SET UP A DUMP-MODE LIST
;ASSUMES THE CURRENT IOWD (RELOCATED) IS IN DEVDMP(F)
;UPDATES DEVDMP(F) BY NUMBER OF WORDS TRANSFERRED
;IF THE IOWD IS FINISHED, DEVDMP WILL BE POSITIVE
SETDMP:
IFN FTVM,<
SKIPGE DEVSWP##(F) ;IS IT THE SWAPPER?
JRST SETSWP ;YES, DO IT DIFFERENTLY
>
IFN FTKI10!FTKL10,<
HLRZ T4,P3
TRZ T4,400000
MOVEI T1,KONDMP##(T4)
PUSH P,T1
>
HLLZ T1,DEVDMP##(F) ;NUMBER OF WORDS IN IOWD
MOVNS T1 ;+N
HRLZ T2,T3 ;NUMBER OF BLOCKS LEFT IN GROUP
LSH T1,MBKLSH## ;NUMBER OF BLOCKS IN IOWD
JUMPE T2,SETDM2 ;0 IS OK
IFN FTDSUP,<
TDC S,[IOSUPR,,UDSX]
TDCN S,[IOSUPR,,UDSX] ;FORMATTING?
JRST SETDM2 ;YES, ASSUME HE REALLY KNOWS
>
CAMLE T1,T2 ;MORE REQUESTED THAN IN GROUP?
MOVE T1,T2 ;YES, TAKE LESSER AMOUNT
SETDM2: LSH T1,BLKLSH## ;CONVERT BACK TO WORDS
MOVNM T1,T3 ;SAVE N
MOVE T2,DEVDMP##(F) ;CURRENT IOWD
SKIPN DINITF## ;IF NOT IN ONCE-ONLY,
IFN FTKA10,<
MOVEM T2,KONDMP##(J) ;SAVE IN KONT DATA BLOCK IN CASE UNIT IS OFF-LINE
>
IFN FTKI10!FTKL10,<
MOVEM T2,@0(P)
POP P,(P) ;REMOVE THIS WORD FROM PD LIST
>
HLL T2,T3 ;SET LH OF IOWD
IFN FTKI10!FTKL10,<
MOVEI P4,0
PUSHJ P,MAPIO## ;SAVE THE IOWD IN FREE-CORE
JRST PUNTD ;NOT ENOUGH FREE CORE
SKIPA ;AND CONTINUE
>
MOVEM T2,(T4) ;STORE IOWD IN BLOCK
SETD2B: JUMPL S,SETDM3 ;GO IF MONITOR MODE
ADDM T1,DEVDMP##(F) ;UPDATE WDCNT
HLRZ T2,T1
ADD T2,DEVDMP##(F) ;UPDATE ADDRESS
HRRM T2,DEVDMP##(F) ; (DON'T PROPOGATE CARRY)
TLNE T3,BLKSIZ##-1 ;EVEN NUMBER OF WRDS FOR BLOCK?
TLNN S,IO ;NO. WRITING?
JRST SETDM3 ;NO. OK
SKIPN DEVMBF##(F) ;HAVE MON BUF?
JRST SETDM3 ;NO, HARDWARE MUST WRITE 0'S IN REST OF BLOCK
TLO T3,MBLKSZ## ;YES. MAKE SURE T3 .GT. -200
MOVNS T3 ;PARTIAL-BLOCK WORD COUNT
IFN FTKI10!FTKL10,<
ADD T3,[XWD MBLKSZ##,0] ;-NO OF WDS LEFT
HRRZ T2,DEVMBF##(F) ;MAKE AN IOWD FOR 0'S
ADD T2,T3
TLO S,IOSMON ;SO MAPIO WON'T UNRELOCATE THE IOWD
PUSHJ P,MAPIO## ;SAVE IT
JRST PUNTA
TLZ S,IOSMON
JRST SETDM3 ;AND CONTINUE
IFN FTVM,<
;HERE TO SET UP AN IO LIST FOR THE SWAPPER
SETSWP: PUSHJ P,SWPP1 ;POINT P1 AT THE SWPLST ENTRY
;T3=MAXIMUM NUMBER OF BLOCKS TO DO
PUSHJ P,THIS## ;GO SET THE IO LIST
;RETURNS T1= NO OF BLOCKS, P3=L(LIST)
JRST SETL5A ;GO FINISH UP THE IO LIST STUFF
;SUBROUTINE TO POINT P1 AT SWPLST ENTRY
;PRESERVES T1
SWPP1:
IFN FTDUAL,<
SKIPGE T2,UNI2ND##(U)
SKIPA P1,UNISWA##(T2)
>
MOVE P1,UNISWA##(U)
POPJ P,
> ;END FTVM
> ;END FTKI10
HLRZ T2,T3 ;SAVE IT
ADDB T3,(T4) ;MAKE LAST IOWD EVEN NO OF BLOCKS
HLRES T3 ;GOOD WDCNT OF LAST IOWD
MOVNS T3 ;+N
ADD T3,(T4) ;ADR-1 OF 1ST WORD IN PARTIAL BLOCK
HRL T3,DEVMBF##(F) ;ADR-1 OF MON BUF
MOVSS T3 ;SET FOR BLT
AOBJN T3,.+1
ADDI T2,-1(T3) ;LAST DATA WORD
BLT T3,(T2) ;COPY PARTIAL BLOCK TO MON BUF
MOVE T2,DEVMBF##(F) ;IOWD FOR PARTIAL BLOCK
SKIPGE (T4) ;IF LAST IOWD STILL POINTS TO DATA,
ADDI T4,1 ; POINT TO NEXT IOWD SLOT
MOVEM T2,(T4) ;STORE IOWD FOR WRITING TERMINATING 0'S
SETDM3:
IFN FTKL10,<
HLRZ T2,T1 ;SAVE ORIGINAL WRDCNT (RH20 MAY NEED IT)
>
ADD T1,[XWD BLKSIZ##-1,0] ;ACCOUNT FOR PARTIAL BLOCKS
LSH T1,MBKLSH##-^D18 ;CONVERT T1 TO BLOCKS
AOJA T4,SETLS5 ;ZERO IOLIST+1, SET CHNNUM AND RETURN
IFN FTKI10!FTKL10,<
;HERE WHEN NO MONITOR FREE-CORE LEFT
PUNTB:
JUMPN P1,SETLS5 ;AT NEW IOWD, JUST DO FEWER BUFS
PUNTA: TLZ S,IOSMON
PUNTC: JUMPE P1,PUNTXX ;IF GOT SOME CORE
SUB P1,[XWD 1,1] ; POINT TO LAST FULL IOWD
SETZM 1(P1) ;ZERO LAST (PARTIAL) IOWD
PUNTD: JUMPE P1,PUNTXX ;NO FREE-CORE IF 0
IFE FT22BIT,<
MOVEI T1,-1 ;GOT SOME, COUNT NO OF WORDS
>
IFN FT22BIT,<
MOVEI T4,0 ;SET INDEX INTO POINTER-TABLE
SKIPGE CHB22B##(P3) ;0 IF 18-BIT CHAN,
MOVEI T4,1 ;1 IF 22-BIT CHAN
MOVE T1,MSK22B##(T4) ;GET MASK FOR ADDRESS
>
MOVE T2,P2 ;INITIAL ADR OF IOWDS
PUNTE: SKIPN T3,(T2) ;GET NEXT IOWD
JRST PUNTG ;DONE-DO PARTIAL LIST
TLNN T3,-1
JRST PUNTF ;GO TO WORD IF POSITIVE
SUB T1,T3 ;COUNT THE WORDS
IFE FT22BIT,<
TROA T1,-1 ;IGNORE THE ADDRESS
>
IFN FT22BIT,<
TDOA T4,MSK22B##(T4)
>
PUNTF: SOS T2,T3 ;POINT TO NEW IOWD
AOJA T2,PUNTE ;GO DO NEXT IOWD
PUNTG:
IFN FT22BIT,<
SKIPE T4
ASH T1,-4
>
HLRS T1 ;SET TO UPDATE DEVDMP
JRST SETD2B ;FIX IOLIST AND START IO
;HERE IF COULDN'T GET ANY FREE CORE
PUNTXX: SETZB T1,P2 ;NO IO LIST, CHNNUM=0 (SO
JRST SETLS5 ; IOIMPM WILL LIGHT AT INTERRUPT)
>;END FTKI10 CONDITIONAL
;SUBROUTINE TO FIND A UNIT WHICH NEEDS HOME BLOCKS REREAD
;NON-SKIP RETURN IF NO SUCH UNITS
;SKIP RETURN IF FOUND, U POINTS AT UNIT
UNIRHB: SKIPG HOMFLG## ;ANY UNITS NEED REREADING?
POPJ P, ;NO
MOVEI T1,KONTAB##(J) ;YES, SET TO LOOK FOR UNIT
HRLI T1,-10
MOVSI T2,UNPRHB## ;BIT TO TEST
UNIRH1: SKIPE T3,(T1) ;GET A UNIT DATA BLOCK
TDNN T2,UNIDES##(T3);NEED REREADING?
AOBJN T1,UNIRH1 ;NO
JUMPGE T1,CPOPJ## ;GO IF DIDNT FIND ANY (DIFFERENT KONTROLLER)
MOVE U,T3 ;SET U TO UDB TO REREAD
JRST CPOPJ1## ;AND NON-SKIP
;SUBROUTINE TO SEE IF ANY UNIT NEEDS HOME BLOCKS READ
;STARTS IO TO READ HOME BLOCKS IF ANY UNITS ON THIS KONTROL NEED IT
;OTHERWISE RETURNS (DISMISSING INTERRUPT)
TSTRHB: PUSHJ P,UNIRHB ;ANY UNIT NEED REREADING?
JRST POSDIA ;NO
PUSHJ P,FAKDDX ;YES, GET A DDB
JRST POSDIA ;NO FREE CORE, TRY LATER
TSTRHX: HRRM U,DEVUNI##(F) ;SAVE U
SETZM DEVRHB##(F) ;WHOLE WORD=0 INDICATES REREADING HOME BLOCKS
MOVE T1,RHBIOW## ;IOWD TO READ 1ST 10 WORDS
MOVEM T1,DEVDMP##(F) ; INTO A BUFFER
MOVEI T1,1 ;BLOCK 1IS 1ST HOME BLOCK
MOVEM T1,DEVBLK##(F)
MOVE S,[IOSMON,,IOACT] ;SET S
MOVEM S,DEVIOS(F)
AOS @KONCHN##(J) ;MARK CHANNEL BUSY
PJRST STRTIO ;START THE READ AND RETURN
;SUBROUTINE TO GET A DDB
FAKDDX: PUSH P,J ;FAKDDB WIPES OUT J
PUSHJ P,FAKDDB## ;GET ONE
JRST JPOPJ## ;NO FREE CORE
SETZ J, ;CANT BLAME THIS READ ON ANY JOB,
DPB J,PJOBN## ; SO SET PJOBN = 0
JRST JPOPJ1## ;RESTORE J AND GOODNESS-RETURN
;HERE WHEN ALL POSITIONING INTERRUPTS HAVE BEEN HANDLED
POSDON: POP P,P2 ;LH=UNIT #, RH = FUNCTION
POP P,P4 ;CONI WORD
POP P,P3 ;DATAI WORD
HRRZ U,KONCUA##(J) ;SET U TO UNIT ADDRESS
JUMPE U,CPOPJ## ;MIGHT BE SPURIOUS INTERRUPT FROM WRONG KONTROLLER
HRRZ F,UNICDA##(U) ; SET F TO FILE ADDRESS
JUMPE F,TSTRHB ;FREE INT IF 0 OR UNSOLICITED
MOVE S,DEVIOS(F) ;AND S TO S
IFN FTVM,<
SKIPL DEVSWP##(F)
JRST POSDNS
HRRM U,DEVUNI##(F) ;IF MULTIPLE SWAPS ON DIFFERENT DF10'S
MOVE T1,UNIBLK##(U) ; DEVBLK AND DEVUNI MAY BE CHANGED
MOVEM T1,DEVBLK##(F) ; SO RESTORE THEM TO RIGHT VALUES
POSDNS:
>
IFN FT5UUO,<
; SKIPE UNISTS##(U) ;IF NOT A FREE INTERRUPT W/NO ACTIVE DRIVES,
MOVEM P4,DEVSTS(F) ;SAVE CONI WORD IN DDB
>
TRNE P2,OPPOS## ;POSITIONING INTERRUPT ONLY?
;THE FOLLOWING INSTRUCTION COUNTS ON THE FACT THAT RP10 DISKS
;DONT GET ERRORS ON SEEKS (EXCEPT SEEK INCOMPLETE, AND DPXKON LIES ABOUT THAT)
;AND THAT RP04 DISKS HAVE IMPLIED SEEKS IF OFF-CYLINDER WHEN IO IS STARTED
POSDIA:
IFE FTDHIA,<
POPJ P, ;YES
>
IFN FTDHIA,<
JRST [SKIPGE (P1)
JRST CHNIDL
POPJ P,]
>
SOS CHNCFT##(P1) ;NO, DECREASE FAIRNESS COUNTS
SOS CHNCFP##(P1)
PUSHJ P,ADRINT ;SET UP R
HRRZ T4,P2 ;ERROR BITS FROM DRIVER
LDB T1,UNYPUN## ;UNIT NUMBER
HLRZ T2,P2 ;UNIT NUMBER THE DEVICE CODE RETURNED
CAME T1,T2 ;DO THEY AGREE?
TROA P2,IOBKTL ;NO - LIGHT IOBKTL
TRZ P2,IOBKTL ;YES - ZERO IOBKTL
SKIPE CHNNUM##(P1) ;0 BLOCKS XFERRED?
JRST POSDN0 ;NO (ERROR SETS LH -1 EVEN IF RH IS 0)
IFN FTKI10!FTKL10,<
LDB T1,PIOMOD## ;IF DUMP MODE CHNNUM=0
CAIL T1,SD ; IFF WE RAN OUT OF LOW-CORE BLOCKS
JUMPGE S,POSDN3 ; AND WE WILL RECOVER AT DUMPG5
>
TRO S,IOBKTL ;YES, 1ST BUFFER HEADER ZAPPED BY USER
JRST POSER8 ;STORE S, GET NEXT IO
POSDN0: TRNE P2,IOIMPM+IODTER+IODERR+IOBKTL ;ERROR?
JRST POSERR ;YES, RECOVER
MOVE T1,KONIOC##(J) ;GET ADDR OF INAD PAIR
IFN FTKL10,<
MOVE T2,CHB22B##(P1)
TLNE T2,CP.RH2## ;IS IT AN RH20?
SKIPA T1,2(T1) ;YES, TERMINATION WORD IS ELSEWHERE
>
MOVE T1,1(T1) ;GET TERMINATION CONTROL WORD
CAMN T1,CHNTCW##(P1) ;IS IT WHAT WE EXPECT
JRST POSDN1 ;YES, CONTINUE
AOS UNICCT##(U) ;NO, INCREMENT TERMINATION ERROR COUNT
TRO P2,IODTER ;SET AS DATA ERROR TO GET INFO TO ERROR.SYS
;FALL INTO ERROR HANDLING
;HERE ON ANY ERROR RETURN FROM THE DEVICE DEPENDENT ROUTINE
POSERR: HRRZ T1,DEVDMP##(F) ;ADR OF (POSSIBLE) DUMP IOWD
TLNE S,IOSMON ;MONITOR CALL?
JUMPE T1,POSR5A ;YES. NO ERR RECOVERY IF CHAN SKIP IOWD
SKIPGE UNIECT##(U) ;FIRST ERROR (UNFIR)?
JRST FSTERR ;YES
IFN FTRP04,<
TRNE P2,IOECCX ;ECC-CRRECTABLE ERROR?
JRST POSERF ;YES, COMPUTE THE NUMBER OF GOOD BLOCKS PASSED
>
JRST POSERA ;NO
;HERE ON FIRST ERROR (DON'T KNOW WHETHER IT WILL BE SOFT OR HARD)
FSTERR: SETZM UNIERR##(U) ;YES, ZERO LAST ERROR CONI STATUS
MOVEM P4,UNISOF##(U) ;SAVE FIRST CONI IN UNIT DB
MOVEM P3,UNISDI##(U) ;SAVE FIRST DATAI
IFN FTRP04,<
TRNN T4,IOBKTL+IODTER+IODERR ;IF NOT DEVICE DETECTED,
PUSHJ P,@KONRRG##(J) ; READ THE MASSBUS REGISTERS
>
PUSHJ P,FSTREG ;COPY REGISTERS TO RH OF UDB
MOVEI T2,CHNDPE## ;ASSUME DATA PARITY ERR ON DEVICE
TRNE P2,IOCHMP ;IS IT A MEMORY PARITY ERROR?
MOVEI T2,CHNMPE## ;YES, SET FOR MEM PAR ERR
TRNE P2,IOCHNX ; IS IT A NXM ERROR
MOVEI T2,CHNNXM## ;YES, NXM ERROR
MOVE T1,KONIOC##(J) ;INITIAL CONTROL WORD ADR
TRNN P2,IOCHMP!IOCHNX ;CHANNEL DETECTED ERROR?
;(MPE, NXM, OR DATA LATE)
TRNE P2,IODTER ;NO, DEVICE ERROR (DO NOT COLLECT CHAN DATA)?
PUSHJ P,(T2) ;GO STORE CHANNEL DATA FOR F.S. ON SOFT+HARD ERRORS
SETZM UNIECT##(U) ;SET RETRY COUNT TO 0
;(FIRST ERROR FLAG -UNPFIR AND HARD ERROR FLAG -UNPHRD)
IFN FTRP04,<
TRC P2,IODTER!IODERR ;IODERR AND IODTER BOTH ON
TRCE P2,IODTER!IODERR ; MEAN FORMAT ERROR
JRST FSTER2 ;NOT FORMAT ERROR
HRRZ T1,DEVACC##(F) ;FORMAT - IF A FILE IS OPEN,
JUMPE T1,FMTERR
MOVE T1,DEVREL##(F) ; AND WE ARE NOT AT FIRST BLOCK OF FILE,
SOJLE T1,FMTERR
TRZ P2,IODERR ;THEN CALL IT A DATA ERROR,
JRST FSTER2 ; AND ATTEMPT RECOVERY
FMTERR: TRO S,IODERR!IODTER ;FORMAT ERR, NO FILE OR AT 1ST BLOCK
JRST POSER8 ;INDICATE FORMAT ERR AND DON'T RETRY
FSTER2:>
IFN FT5UUO,< ;5 SERIES UUOS?
MOVEI T1,DEPDER ;ERROR-RETRY DISABLED?
TDNN T1,DEVSTA(F) ;BY OPEN UUO
JRST POSERF ;NO
AOS UNISCT##(U) ;YES, COUNT A SOFT ERROR
PUSH P,CHNNUM##(P1) ;SAVE NO OF BLOCKS XFERRED
PUSHJ P,CTGOOD ;COMPUTE BAD BLOCK
ADD T1,DEVBLK##(F)
MOVEM T1,UNIHBN##(U) ;SAVE IN UDB
POP P,CHNNUM##(P1) ;RESTORE NO OF BLOCKS
JRST POSR1A ;NO RETRY
>
POSERF: PUSHJ P,CTGOOD ;COMPUTE THE NUMBER OF GOOD DATA BLOCKS
HRLM P2,CHNNUM##(P1) ;STORE ERROR BITS IN CASE CHN MEM PAR
; SO CAN CALL SWEEP AFTER LAST TRY
ADD T1,DEVBLK##(F) ;FIRST LOGICAL BLOCK OF THIS TRANSFER
MOVEM T1,UNIHBN##(U) ;STORE LOGICAL BLOCK OF ERROR FOR F.S.
; TELL LATER IF HARD OR SOFT
POSERA: MOVEI T4,1 ;SET BIT FOR ERR COUNTER, ASSUME DATA ERR
TRNE P2,IODERR ;DEVICE ERROR?
MOVSI T4,1 ;YES, COUNT AS DEVICE ERROR
ADDM T4,UNIECT##(U) ;ACCUMULATE ERROR COUNT
IFN FTRP04,<
TLNN S,IO ;IF READING
TRNN P2,IOECCX ;AND AN ECC-CORRECTABLE ERROR
JRST NOECC1
IFN FTKI10!FTKL10,<
SKIPGE DEVSWP##(F) ;IF SWAPPER
JRST NOECC1
PUSHJ P,SVEUF## ;MAKE JOB ADDRESSABLE
>
SOS T1,UNIHBN##(U) ;GET BLOCK BEFORE THE BAD ONE
SUB T1,DEVBLK##(F) ;COMPUTE NUMBER OF GOOD BLOCKS
PUSH P,DEVDMP##(F) ;SAVE DEVDMP
LDB T2,PIOMOD## ;IF DUMP MODE
CAIGE T2,SD
JUMPGE S,ECC1 ;OR MONITOR IO
HRRM T1,CHNNUM##(P1) ;SAVE NUMBER OF GOOD BLOCKS
AOS CHNNUM##(P1) ;+1 FOR THE BLOCK WE'LL CORRECT
LSH T1,BLKLSH## ;CONVERT TO WORDS
HRLS T1
ADD T1,KONDMP##(J) ;UPDATE THE IOWD
JUMPL S,ECC3 ;IF NOT MONITOR IO
MOVE T2,T1 ; UPDATE IOWD FOR THE DDB
ADD T2,[BLKSIZ,,BLKSIZ]
CAMLE T1,[MBLKSZ,,0] ;IF LESS THAN 1 BLOCK IN IOWD
TLOA T1,400000 ;USET A FLAG
TLZ T1,-1
MOVEM T2,DEVDMP##(F)
SUBI T1,(R) ;UNRELOCATE THE IOWD
JRST ECC3 ;AND CONTINUE
;HERE IF ERROR WAS IN BUFFERED MODE
ECC1: HRRZ T2,DEVIAD(F) ;LOC OF 1ST BUFFER
JUMPE T1,ECC2
IFN FTKA10,<
TLO T2,R
>
EXCTUX <HRR T2,@T2> ;ADVANCE THE NUMBER OF GOOD BUFFERS
SOJG T1,.-1
ECC2: MOVEI T1,1(T2) ;POINT T1 AT LOC-1 OF BAD BLOCK
;HERE WITH T1=UVA-1 OF START OF BAD BLOCK (OR EVA-1 IF MONITOR BUFFER)
ECC3: PUSH P,T1 ;SAVE LOC
HRRZ T1,CHNNUM##(P1)
JUMPE T1,NOECC ;CHAN GOOFED IF NO BLOCKS XFERRED
PUSHJ P,@KONECC##(J) ;GET RELATIVE POSITION OF BAD WORD
JRST NOECC ;OOPS
CAILE T1,BLKSIZ##-1 ;IF THE ERROR IS IN THE ECC BYTE,
JRST [POP P,T1
POP P,(P)
JUMPGE T1,POSDN1
HRRZS DEVDMP##(F)
JRST POSDN1] ;NO CORRECTION NEEDED
LDB T4,PIOMOD## ;IF DUMP MODE,
CAIGE T4,SD
JUMPGE S,ECC4
HLRE T4,DEVDMP##(F) ;WORD COUNT OF IOWD
SUBI T4,BLKSIZ## ; PRIOR TO BAD BLOCK
ADDI T4,(T1) ;+ POSITION OF ERROR BURST
SKIPGE (P) ;IF AT END OF IOWD
HRRZS DEVDMP##(F) ; MAKE SURE WE TERMINATE
JUMPL T4,ECC4 ;CONTINUE IF HE'S READING THAT PART OF BLOCK
MOVEI T3,0 ;NO CORRECTION FOR 2ND PART, HE'S NOT READING IT
SKIPE T4 ;READING 1ST WORD OF ERROR BURST?
MOVEI T2,0 ;NO, DON'T CORRECT 1ST PART EITHER
ECC4: POP P,T4 ;RESTORE START OF BLOCK
POP P,(P) ;REMOVE SAVED DEVDMP FROM LIST
ADDI T1,1(T4) ;POINT T1 AT 1ST BAD WORD
JUMPL S,ECC6 ;IF NOT MONITOR IO,
IFN FTKA10,<
TLO T1,R
>
JUMPE T2,ECC5 ;NO, 1ST PART IF T2=0
EXCTUX <MOVS T4,@T1> ;THIS WILL HAVE TO BE MODIFIED IF WE GET
; OTHER HARDWARE WHICH DOES ECC DIFFERENTLY
XOR T4,T2 ;APPLY MASK
EXCTXU <MOVSM T4,@T1> ;AND SAVE RESULT
ECC5: ADDI T1,1
JUMPE T3,POSDN1 ;NO 2ND PART IF T3=0
EXCTUX <MOVS T4,@T1> ;GET 2ND WORD
XOR T4,T3 ;APPLY MASK
EXCTXU <MOVSM T4,@T1> ;AND SAVE RESULT
JRST POSDN1 ;LOG A RECOVERED ERROR AND CONTINUE
;HERE IF ERROR IN MONITOR IO (INTO MON BUF)
ECC6: JUMPE T2,ECC7 ;NO 1ST PART IF T2=0
MOVS T4,(T1) ;GET 1ST BAD WORD
XOR T4,T2 ;APPLY MASK
MOVSM T4,(T1) ;AND SAVE
ECC7: JUMPE T3,POSDN1 ;NO 2ND PART IF T3=0
MOVS T4,1(T1) ;GET 2ND BAD WORD
XOR T4,T3 ;CORRECT
MOVSM T4,1(T1) ;AND SAVE
JRST POSDN1 ;AND CONTINUE
;HERE IF WE REALLY COULDNT RECOVER THOUGH WE THOUGHT WE COULD
NOECC: POP P,(P) ;REMOVE RELATIVE ADDR FROM LIST
POP P,DEVDMP##(F) ;RESTORE ORIGINAL DEVDMP TO THE DDB
MOVEI T4,1 ;COUNT DATA ERROR
NOECC1:>
AOS T1,CHNECT##(P1) ;UPDATE COUNT OF TRIES
CAIN T1,1 ;FIRST RETRY OR 1ST AFTER RECAL?
CAME T1,CHNRCT##(P1) ;YES, THIS 1ST RECAL?
JRST POSERB ;NO
MOVEM P4,UNIERR##(U) ;YES. SAVE 2ND ("HARD") CONI WORD
MOVEM P3,UNIHDI##(U) ;SAVE "HARD" DATAI WORD
PUSHJ P,LSTER ;SAVE THE DRIVE REGISTERS NOW
POSERB: MOVE T2,UNISTS##(U) ;STATE OF UNIT
CAIE T2,TCOD ;IGNORE IF POSITIONING,
POPJ P, ;RETRY IF XFER (ERROR RECOVERY)
MOVE P4,T4 ;SAVE ERROR-COUNT WORD
IFN FTRP04,<
TRNN T2,IOHDER ;HEADER ERROR
TRNN P2,IODERR ; OR DATA ERROR? (AND NOT FMT)
SKIPG T2,KONERR##(J) ;YES, WILL KONTROL ROUTINE TELL US WHAT TO DO?
JRST NOOFST ;NO
HLRZ T1,UNIECT##(U) ;USE LH(UNIECT) IF HEADER ERROR
ADD T1,UNIECT##(U) ; OR RH IF DATA ERROR
HRRZS T1 ;ONE HALF MUST HAVE COUNTED UP
PUSHJ P,(T2) ;ASK THE KONTROLLER ROUTINE
JRST @ERRTBL(T1) ;AND GO DO WHAT IT SAID
ERRTBL: STARTE ;(0) RETRY
OFFSET ;(1) OFFSET
POSERE ;(2) LAST TIME
POSERG ;(3) GIVE UP
POSERD ;(4) RECAL
NOOFST:>
HRRZ T2,UNISTR##(U) ;LOC OF STR DATA BLOCK
LDB T3,STYTRY## ;NO OF TIMES TO RETRY
TRNE P2,IODERR ;DEVICE (POSITIONING) ERROR?
LDB T3,STYSER## ;YES. USE A DIFFERENT PARAMETER
SKIPN T2 ;UNIT IN AN STR?
MOVEI T3,^D10 ;NO, USE 10
SUB T1,T3 ;HAVE WE TRIED ENOUGH?
JUMPL T1,STARTE ;RETRY IF NEGATIVE
TRNE P2,IOBKTL ;WRONG UNIT?
JRST POSERC ;YES, DONT TRY RECAL
AOS T4,CHNRCT##(P1) ;UPDATE RECALIBRATE-COUNTER
LDB T3,STYRCL## ;NO OF TIMES TO RECAL
SKIPN T2 ;UNIT IN AN STR?
MOVEI T3,^D10 ;NO, USE 10
SUB T4,T3 ;TRIED ENOUGH?
JUMPGE T4,POSER0 ;TRIED ENOUGH IF T4=0
POSERD: SETOM CHNECT##(P1) ;NO. SET A SWITCH FOR RECALR
PUSHJ P,@KONRCL##(J) ;DO A RECALIBRATE
JRST POSER0 ;NOT A PACK OR UNIT DOWN
PJRST STOIOS## ;RECALIBRATING - RESET HUNG TIME, DISMISS INTERRUPT
POSER0: JUMPG T1,POSER1 ;GO IF NOT LAST TIME
POSERE: MOVEI T1,KONRED##(J) ;LAST TIME - SET TO NO STOPPING
TLNE S,IO ; ON ERROR SO ALL OF DATA IS ATTEMPTED
; TO BE TRANSFERRED ON LAST RETRY
MOVEI T1,KONWRT##(J)
IFN FTRP04,<
MOVSI T2,DEPCPT## ;IF IN 10/11 COMPAT MODE,
TDNE T2,DEVCPT##(F)
ADDI T1,2 ;CALL THE COMPATABILITY ENTRY
>
PUSHJ P,@(T1) ;CALL DEVICE DEPENDENT ROUTINE
PJRST BADUNI ;UNIT NOT UP
POPJ P, ;DISMISS THE INTERRUPT
IFN FTRP04,<
;HERE WHEN ERROR IS DECLARED HARD FOR AN ERROR-PROGRAM KONTROLLER
POSERG: PUSHJ P,LSTER ;READ ALL DRIVE REGISTERS
JRST POSER1 ;AND GO DECLARE HARD ERROR
>
IFN FTRP04,<
OFFSET: HRROS CHNECT##(P1) ;CHNECT=-1,,N TO INDICATE OFFSET IN PROGRESS
PJRST STOIOS##
>
;SUBROUTINE TO COUNT THE NUMBER OF GOOD DATA BLOCKS
;UPDATES CHNNUM IF NOT DUMP-MODE
;RETURNS NO OF GOOD BLOCKS IN T1
CTGOOD: HRRZ T1,KONIOC##(J) ;INITIAL CONTROL WORD ADDRESS
PUSHJ P,WRDCNT## ;COMPUTE NO. OF GOOD WORDS TRANSFERRED
IFN FTRP04,<
TRNE P2,IOECCX ;ECC CORRECTABLE ERROR?
ADDI T1,BLKSIZ##-2 ;YES, MAKE SURE "BAD" BLOCK IS COUNTED
>
LSH T1,MBKLSH## ;CONVERT TO # OF GOOD BLOCKS
SKIPGE DEVSWP##(F) ;IF SWAPPER
POPJ P, ; DON'T CHANGE CHNNUM
LDB T2,PIOMOD## ;IS IT DUMP MODE?
CAIGE T2,SD ;(DONT CHANGE CHNNUM SINCE DEVDMD
; HAS BEEN UPDATED COUNTING ON IT
HRRZM T1,CHNNUM##(P1) ;STORE NO. OF GOOD BLOCKS TRANSFERRED
; ON FIRST TRY TO BE USED BY BUFAD
; IF THIS PROVES TO BE HARD ERROR
; FLAG ERROR SO NON-ZERO EVEN IF IN FIRST BLOCK
; SO DIFFERENT FROM USER ZAP FIRST BUFFER
POPJ P,
;SUBROUTINE TO COPY THE DRIVE REGISTERS INTO THE RH OF THE UDB
;CALLED ON FIRST ERROR, WIPES OUT LH OF UDB REG'S
FSTREG: SKIPN T2,KONREG##(J) ;GET NUMBER OF DRIVE REGISTERS TO STORE
POPJ P, ;NONE - NOT A MASSBUS DEVICE
ADDI T2,UNIEBK##(U) ;POINT TO TOP OF BLOCK
MOVSI T1,KONEBK##(J) ;WHERE THEY WERE SAVED
HRRI T1,UNIEBK##(U) ;WHERE THEY ARE TO GO
BLT T1,-1(T2) ;SAVE THEM
MOVE T1,UNILAS##(U) ;LAST DATAO
MOVEM T1,(T2) ;SAVE IN THE UDB
MOVE T1,KONECR##(J) ;GET KONTROLLER
MOVEM T1,UNISCR##(U) ; CONTROL REG & DATA REG
MOVE T1,KONEDB##(J) ;AND SAVE IN UDB
MOVEM T1,UNISDR##(U)
POPJ P, ;AND RETURN
;SUBROUTINE TO SAVE THE DRIVE REGISTERS IN THE UDB
; RESPECTS T1,T4
LSTER: PUSH P,T1
MOVN T1,KONREG##(J) ;NUMBER OF REGISTERS TO SAVE
JUMPE T1,TPOPJ##
HRLS T1 ;MAKE AN AOBJN WORD
HRRI T1,KONEBK##(J)
MOVEI T2,UNIEBK##(U) ;WHERE TO STORE
LSTER1: MOVE T3,(T1)
HRLM T3,(T2) ;SAVE A DRIVE REGISTER IN LH OF UDB WORD
ADDI T2,1
AOBJN T1,LSTER1 ;GET ANOTHER WORD
MOVE T1,UNILAS##(U) ;LAST DATAO TO THE DRIVE
HRLM T1,(T2) ;SAVE IN UDB
MOVE T1,KONECR##(J) ;SAVE KONTROLLER (RH10)
MOVEM T1,UNIHCR##(U) ; CONTROL REG & DATA REG
MOVE T1,KONEDB##(J) ;IN UDB
MOVEM T1,UNIHDR##(U)
JRST TPOPJ## ;AND RETURN
;HERE ON HARD WRONG-UNIT
POSERC: TLNE S,IO ;IF READING, CONTINUE
STOPCD .,JOB,HWU, ;++HARD WRONG UNIT
;HERE ON HARD DEVICE OR DATA ERRORS
POSER1: SOJE T1,STARTE ;LAST RETRY, STOP ON ERROR, IF 1
ADDM P4,UNIHCT##(U) ;UPDATE HARD-ERROR WORD
TLNE P4,-1 ;BEEN COUNTING IN LH (POSITION ERRS)?
HLRZS UNIECT##(U) ;YES. SAVE COUNT IN RH (UNIECT)
MOVSI T1,UNPHRD## ;HARD ERROR FLAG ON LAST ERROR ON THIS UNIT
IORM T1,UNIECT##(U) ;SET FOR DAEMON AND SYSERR
PUSHJ P,CHKCMP ;CHECK IF CHN MEM PAR ERR
; IF YES, FLAG CPU MEM SWEEP
;HERE IF USER AVOIDING ALL RETRIES
POSR1A:
IFN FTKL10,<
TLNN S,IO ;IF INPUT,
PUSHJ P,CFDMP ;FLUSH CACHE SO BAD DATA WILL BE SEEN
> ;END IFN FTKL10
JUMPL S,POSR5A ;ONLY 1 IOWD IF MONITOR IO
HRRZ P4,CHNNUM##(P1) ;GET # OF GOOD BLOCKS FOR BUFAD
PUSHJ P,BUFAD ;ADVANCE THE GOOD BUFFERS
JFCL
POSR5A: PUSH P,P2 ;SAVE ALL ERROR BITS
ANDI P2,IOIMPM+IODTER+IODERR+IOBKTL+IOCHMP+IOCHNX ;P2=ERROR BITS
TRNN P2,IOCHMP+IOCHNX ;CHAN-DETECTED ERROR?
JRST POSR5C ;NO
SKIPL DEVSWP##(F) ;SWAPPER?
TRZ P2,IOCHMP+IOCHNX ;NO, DON'T KEEP THESE ERR BITS
POSR5C: OR S,P2 ;STORE ERROR BITS IN S
POP P,P2
JUMPL S,POSER6 ;GO IF MONITOR IO
LDB T1,PIOMOD ;MODE
CAIL T1,SD ;WE DIDN'T STOP EARLY IF DUMP-MODE
JRST POSER6 ; SO DON'T FIDDLE WITH DEVBLK, ETC.
AOS DEVBLK##(F) ;UPDATE DB LOCS WHICH POINT TO BLOCK
AOS DEVREL##(F) ; UPDATE DEVREL
SOS DEVLFT##(F)
PUSHJ P,(P3) ;ADVANCE 1 MORE BUFFER (THE BAD ONE)
JFCL
POSER6: TRNN S,IODTER ;PARITY ERROR?
JRST POSER7 ;NO
TLNE S,IO ;YES. LIGHT ERR BIT IN LH(S)
TLOA S,IOSHWE## ; BECAUSE USER CAN CLEAR RH OF S (SETSTS)
TLO S,IOSHRE##
MOVE T1,KONIOC##(J) ;INITIAL CONTROL WORD ADR
PUSHJ P,CHNDPE## ;LOG THE ERROR
POSER7: TRNE S,IODERR ;DEVICE (POSITIONING) ERROR?
TLO S,IOSSCE## ;YES. LIGHT A BIT IN LH(S) SOFTWARE CHECKSUM
; OR DEVICE ERROR
IFN FTDBBK,<
SKIPE DEVELB##(F) ;IF NOT ALREADY A BAD BLOCK,
JRST POSER8
MOVE T1,UNIHBN##(U) ;BAD BLOCK NO. STORED ON FIRST ERROR
TRNE P2,IODTER ;GET ERROR CODE
TLO T1,BAPDTR## ; (DATA ERR,HEADER ERR, OR OTHER)
TRNE P2,IOHDER ;HEADER ERR?
TLO T1,BAPHDR## ;YES
TRNN P2,IODTER+IOHDER ;NOT HEADER OR DATA?
TLO T1,BAPOTR## ;"OTHER"
TRNN P2,IOCHNX+IOCHMP ;CHANNEL ERRORS?
MOVEM T1,DEVELB##(F) ;STORE BLOCK + CODE IN DDB
LDB T1,UNYLUN## ;AND SAVE THE LOGICAL UNIT NUMBER
DPB T1,DEYEUN## ;FOR ERRFIN
>
POSER8: PUSHJ P,STDIOD## ;YES. WAKE IT UP
IFN FTDAEM,< ;DAEMON CODE?
MOVEI T1,.ERDPE ;CODE FOR DISK ERROR
HRL T1,F ;PUT DDB ADDR IN LH FOR DAEMON
SKIPN DINITF ;DON'T TRY TO STOP THE JOB OR WAKE
; DAEMON IF IN ONCE-ONLY
PUSHJ P,DAEERR## ;STOP JOB, WAKE UP DAEMON
>
POSER9:
IFN FTKI10!FTKL10,<
SKIPE T1,@KONIOC##(J) ;ARE THERE ANY IOWDS?
PUSHJ P,RTNIOW## ;YES, GIVE THEM BACK
>
PJRST SETIDL ;SET THIS FILE IDLE AND LOOK FOR ANOTHER
;HERE WHEN THERE WAS NO HARDWARE ERROR ON THE DATA TRANSFER
POSDN1: SKIPG T1,UNIECT##(U) ;NO. IS THIS A RECOVERED ERROR (UNPFIR)?
JRST POSDN2 ;NO - NO ERROR AT ALL (USUAL)
MOVSI T2,1 ;YES, FOR POSSIBLE UPDATE OF LH
TLNN T1,-1 ;DEVICE ERROR?
AOSA UNISCT##(U) ;NO, DATA ERROR UPDATE RH(UNISCT)
ADDM T2,UNISCT##(U) ;YES, UPDATE LH(UNISCT)
TLNE T1,-1 ;WERE WE COUNTING IN LH (POSITION ERRS)?
HLRZS T1,UNIECT(U) ;YES, SET UNIECT= NUMBER IN RH
IFN FTDAEM,< ;DAEMON CODE?
MOVE T2,CHNNUM##(P1) ;IF AN OVERRUN
TLNE T2,IOVRUN ; RECOVERED ON 1ST RETRY,
SOJE T1,[SKIPN ALLOVR## ;DON'T CALL DAEMON
JRST POSD1A ; IF ALLOVR = 0
JRST .+1]
MOVEI T1,.ERDPE ;CODE FOR DISK ERROR
HRL T1,F ;PUT DDB ADDR IN LH FOR DAEMON
SKIPN DINITF ;DON'T TRY TO STOP THE JOB OR WAKE
; DAEMON IF IN ONCE-ONLY
PUSHJ P,DAEERR## ;STOP JOB, WAKE UP DAEMON
>
PUSHJ P,CHKCMP ;CHECK IF THIS WAS A CHN MEM PAR
; IF YES, FLAG FOR CPU SWEEP
POSD1A: PUSHJ P,CTGOOD ;RESET CHNNUM TO NO OF GOOD BLOCKS
SKIPE UNIERR##(U) ;IS THIS BEFORE FIRST RECAL?
JRST POSDN2 ;NO, "HARD" CONI STUFF ALREADY STORED
MOVEM P4,UNIERR##(U) ;YES, SAVE "HARD" CONI
MOVEM P3,UNIHDI##(U) ;SAVE "HARD" DATAI
PUSHJ P,LSTER ;SAVE THE DRIVE REGISTERS AT END (THEY PROBABLY ARE
; MEANINGLESS AT THIS POINT SINCE THEY ARE ONLY
; STORED AT ERROR TIME, BUT ITS BETTER THAN NOTHING
POSDN2: TLNN S,IO+IOSMON
TLZN S,IOSFIR ;TIME FOR CHECKSUMS?
IFN FTKL10,<
JRST .+2 ;NOT TIME FOR CHECKSUMS, SO WE DON'T HAVE
; TO SWEEP YET.
JRST POSD2A ;CHECKSUM TIME, SWEEP IMMEDIATELY
MOVSI T1,CP.SWF## ;GET READY TO SET THE BIT
TLNN S,IO ;READING?
IORM T1,CHB22B##(P1) ;YES, INDICATE THAT A SWEEP MUST BE
; DONE BEFORE INTERRUPT EXIT TIME.
JRST POSDN3 ;CONTINUE.
POSD2A: PUSHJ P,CFDMP ;CHECKSUM TIME, SO SWEEP THAT WE MAY SEE
; THE DATA NOW.
>;END IFN FTKL10
IFN FTKA10!FTKI10,<
JRST POSDN3 ;NO. CONTINUE
>;END IFN FTKA10!FTKI10
PUSHJ P,CHKSUM ;YES. COMPUTE CHECKSUM
SKIPN T2,@DEVRET##(F) ;PICK UP RETRIEVAL PNTR
MOVE T2,DEVRB1##(F) ;1ST PNTR, MORE IN RIB
HRRZ T3,UNISTR##(U) ;LOC OF STR DB
LDB T2,STYCKP##(T3) ;GET CHECKSUM
CAMN T2,T1 ;DOES IT MATCH COMPUTED CHECKSUM?
JRST POSDN3 ;YES. OK
MOVE T2,DEVACC##(F) ;LOC OF A.T.
MOVE T4,ACCNCK##(T2) ;ALWAYS-BAD-CHECKSUM WORD
TRNE T4,ACPNCK## ;FILE A DIRECTORY OR HAVE ABC?
JRST POSDN3 ;YES. IGNORE ERROR
MOVE T4,DEVREL##(F) ;NO. RELATIVE BLOCK NUMBER
MOVE T2,ACCPT1##(T2) ;CURRENT 1ST POINTER
LDB T2,STYCKP##(T3) ;CHECKSUM BYTE
CAMN T2,T1 ;MATCH?
SOJE T4,POSDN3 ;YES, IF 1ST BLOCK FILE IS A UFD WHOSE
;CHECKSUM HAS CHANGED BETWEEN LOOKUP AND INPUT
AOS UNIMCT##(U) ;REAL CHKSUM ERR. COUNT SOFTWARE ERROR
TDO S,[XWD IOSSCE##,IOIMPM];LIGHT ERROR BIT (LH SINCE USER CAN CLEAR IOIMPM)
PUSHJ P,CTGOOD ;T1= NO. OF GOOD BLOCKS TRANSFERRED
ADD T1,DEVBLK##(F) ;FIRST LOGICAL BLOCK OF TRANSFER
MOVEM T1,UNIHBN##(U) ;STORE BAD BLOCK NO. FOR ERROR REPORTING
IFN FTDAEM,< ;DAEMON CODE?
MOVEI T1,.ERDPE ;SAY DISK ERROR FOR DAEMON
HRL T1,F ;PUT DDB ADDR IN LH FOR DAEMON
PUSHJ P,DAEERR## ;STOP JOB, WAKE UP DAEMON
>
POSDN3: SKIPLE DEVSWP##(F) ;FILE IN IOWAIT?
PUSHJ P,STDIOD## ;YES. WAKE JOB UP
MOVEM S,DEVIOS(F) ;SAVE S IN DDB
IFN FTKI10!FTKL10,<
SKIPE T1,@KONIOC##(J) ;RETURN THE FREE-CORE BLOCKS
PUSHJ P,RTNIOW##
>
HRRZ P4,CHNNUM##(P1) ;NO OF BLOCKS TRANSFERRED
IFN FTDSTT,<
LDB T1,PJOBN## ;JOB NUMBER
; HRRZ T1,JBTPDB##(T1) ;ADDR OF PDB FOR JOB
TLNN S,IO ;READING?
ADDM P4,JBTRCT##(T1) ;YES. UPDATE JOB READ COUNT
TLNE S,IO ;WRITING?
ADDM P4,JBTWCT##(T1) ;YES, INCREMENT NO BLOCKS WRITTEN BY THIS JOB
>
JUMPL S,SETMDL ;MONITOR IO? YES IF S NEG
PUSHJ P,BUFAD ;NO. UPDATE DDB, ADVANCE BUFFERS
JRST SETIDL ;NEXT BUFFER NOT USABLE, OR DUMP MODE
;AT LEAST 1 BUFFER IS AVAILABLE
PUSHJ P,CHKNXT ;ANY MORE BLOCKS ON DISK NOW?
JRST SETIDL ;NO. SET FILE, UNIT TO IDLE
SKIPL KONPOS##(J) ;YES. DOES KONTROLLER POSITION?
JRST SETPW ;YES. SET FILE TO PW STATE
PUSHJ P,SETTW0 ;NO, SET FILE TO TW STATE
JRST PIKTRN ;AND LOOK FOR NEXT TRANSFER OPERATION
;ROUTINE TO INITIATE CPU MEM PAR SWEEP AFTER ALL RETRIES DONE
; IF ERROR WAS A MEM PARITY DETECTED BY THE CHANNEL
; MUST WAIT TILL END BECAUSE SWEEP CLEARS PARITY ERRORS
;CALL: MOVE P1,CHANNEL DATA BLOCK ADDRESS
; PUSHJ P,CHKCMP
; RETURN
CHKCMP: MOVE T1,CHNNUM##(P1) ;LH=IO STATUS ERR BITS ON RETRIES
HLLZ T2,.CHCSR(P1) ;FLAG FOR THIS CHANNEL TO REQUEST CPU0 SWEEP
TRO T2,UE.PEF ;CAUSE PARITY (NOT NXM) SWEEP
TLNE T1,IOCHMP ;DID CHAN DETECT MEM PAR?
IORM T2,.C0AEF## ;YES, FLAG CPU0 TO DO A CORE SWEEP
POPJ P, ;RETURN
;HERE IF FILE IS ON A POSITIONING DEVICE
;SET FILE TO PW STATE, ADD TO PWQ
SETPW: MOVEI T1,PWCOD## ;SET FILE, UNIT TO PW
PUSHJ P,FILCOD
IFN FTDUAL,<
PUSHJ P,SECCOD ;SET STATE OF PRIME UNIT IF THIS ALTERNATE
>
MOVEI T1,UNIQUE##(U) ;SET T1=UNI PWQ
PUSHJ P,PUTQUE ;PUT FILE ON Q
JRST SETID1 ;AND LOOK FOR ANY POSITIONING TO DO
;HERE WHEN MONITOR IO DONE - SET FILE TO IDLE
SETMDL: MOVEI T1,UNIMRC##(U) ;SET TO UPDATE UNIMRC (UNIMWC)
PUSHJ P,UPSTAT ;UPDATE STATISTICS FOR UNIT
;HERE TO SET FILE TO IDLE
SETIDL:
IFN FTVM,<
SKIPL DEVSWP##(F) ;IF NOT THE SWAPPER
>
TLZ S,IOSMON ;MAKE SURE IOSMON IS OFF
SKIPE DEVRHB##(F)
JRST SETIDA
SOSGE HOMFLG##
STOPCD RERED1,DEBUG,RHN, ;++REREAD-HOMEBLOCK-COUNT NEGATIVE
MOVSI T2,UNPRHB## ;CLEAR REREAD HOME BLOCKS
ANDCAM T2,UNIDES##(U) ; FROM UDB
TRNE S,IOIMPM ;READ GO OK?
JRST RERED1 ;WENT OFF-LINE AGAIN
MOVS T1,HOMBUF## ;IS 1ST WORD HOME,
CAIN T1,'HOM'
TRNE S,IODERR+IODTER; AND NO DATA ERRORS?
JRST RERED2 ;CANT READ HOME BLOCK, CALL MDA
MOVE T1,UNIHID##(U) ;READ OK, IS IT WHAT WE EXPECT?
CAME T1,HOMBID##
JRST RERED2 ;HOME BLOCKS DONT MATCH
RERED1: PUSHJ P,CLRDDB## ;ALL IS WELL, GIVE UP DDB
SETZM UNISTS##(U) ;SET UNIT IDLE
JRST SETIDD ;AND CONTINUE
;HERE IF CANT READ HOME BLOCKS OR NO MATCH
RERED2:
IFN FTMDA,<
SKIPGE UNIPTR##(U) ;IF UNIT HAS SWAPPING SPACE
JRST RERED3 ; COMPLAIN TO THE OPERATOR
PUSHJ P,CALMDA ;OTHERWISE LET MDA HANDLE IT
JRST RERED3 ;NO MDA, COMPLAIN
PUSHJ P,CLRDDB## ;MDA IS RUNNING, RETURN DDB
PUSHJ P,SET4MD ;SET SO ONLY MDA CAN READ DRIVE
JRST SETIDC ;AND GO FIND SOMETHING ELSE TO DO
RERED3:>
;HERE IF UNIT HAD SWAPPING SPACE OR IF MDE ISNT RUNNING. COMPLAIN ABOUT UNIT
PUSH P,F ;SAVE F AND U
PUSH P,U
HRRZ U,OPRLDB## ;TALK TO OPR
TRNN S,IODTER+IODERR ;ERROR
JRST RERED4 ;WRONG PACK
PUSHJ P,INLMES## ;WHO KNOWS
BYTE (7)7,7,7,7,7
ASCIZ /
ERROR TRYING TO REREAD HOME BLOCKS ON /
MOVSI T2,'???' ;WHAT WE READ
MOVEM T2,HOMBID##
JRST RERED5
RERED4: PUSHJ P,INLMES##
BYTE (7)7,7,7,7,7 ;SOME BELLS
ASCIZ /
WRONG PACK POWERED UP ON /
RERED5: MOVE T2,(P) ;UDB
PUSH P,UNIHID##(T2) ;SAVE WHAT WE THINK IT IS
MOVE T2,UNINAM##(T2);PHYSICAL NAME
PUSHJ P,PRNAME##
PUSHJ P,INLMES##
ASCIZ / IS /
MOVE T2,HOMBID## ;WHAT WE READ
PUSHJ P,PRNAME##
PUSHJ P,INLMES##
ASCIZ /, SHOULD BE /
POP P,T2 ;WHAT WE EXPECT
PUSHJ P,PRNAME##
PUSHJ P,INLMES##
ASCIZ /
PLEASE DISMOUNT IT AND MOUNT THE CORRECT PACK
/
POP P,U
POP P,F ;RESTORE ACS
MOVEI T1,OCOD## ;SET FOR ONCE-A-MINUTE GRUMP AT OPR
MOVEM T1,UNISTS##(U) ; AND NO IO TO START WHEN IT COMES UP
PUSHJ P,CLRDDB## ;RETURN THE FAKE DDB
JRST SETIDC ;AND FIND SOMETHING ELSE TO DO
IFN FTMDA,<
;SUBROUTINE TO SEND A MESSAGE TO THE MOUNTABLE DEVICE ALLOCATOR
CALMDA: MOVE T1,UNINAM##(U) ;SIXBIT /DEV NAME/
MOVEI T2,.TYDSK ;ITS A DSK
PUSHJ P,SNDMDC## ;TELL MDC
POPJ P, ;MDC ISNT THERE
JRST CPOPJ1## ;MDC IS THERE
>
;HERE FROM BADUNI TO SET THE FILE IDLE. DON'T CLEAR IOSMON
SETIDA: MOVEI T1,ICOD## ;YES. SET FILE, UNIT TO I
PUSHJ P,FILCOD
SETIDD:
IFN FTDUAL,<
SKIPN UNIQUE##(U) ;THIS UNIT GOING IDLE?
SKIPL T1,UNI2ND##(U) ;YES, IS THERE A PRIME UNIT?
JRST SETIDB ;NO
MOVE T2,UNIQUE##(T1) ;YES, PUT POS. QUEUE OF PRIME UNIT
HLLM T2,UNIQUE##(U) ; ONTO THIS UNIT INSTEAD, TO START SEEK
SETZM UNIQUE##(T1) ; ON THIS UNIT NOW. PRIME UNIT HAS NO QUEUE NOW
SETIDB:>
MOVEI T1,PWCOD##
SKIPE UNIQUE##(U) ;ARE POSITIONS WAITING FOR UNIT?
MOVEM T1,UNISTS##(U) ;YES. PUT UNIT IN PW
SETIDC:
IFN FTDUAL,<
SKIPN UNIQUE##(U)
PUSHJ P,SECCOD
SKIPGE T1,UNI2ND##(U) ;IF ALT PATH TO DRIVE
HRLZS UNICDA##(T1) ;CLEAR UNICDA FOR MAIN PORT
>
HRLZS UNICDA##(U) ;NO CURRENT DDB
IFN FTVM,<
SKIPGE DEVSWP##(F) ;SWAPPER?
JRST SETID0 ;YES, TELL VMSER ABOUT THE REQUEST
>
PUSHJ P,CLRACT## ;CLEAR IOACT
IFN FTSWAP,<
IFE FTVM,<
SKIPL DEVSWP##(F) ;IS IT THE SWAPPER?
>
JRST SETID1 ;NO, CONTINUE
IFE FTVM,<
PUSH P,J ;SAVE J
PUSHJ P,SWPINT## ;CALL SWPINT
POP P,J ;RESET J
> ;END IFE FTVM
IFN FTVM,<
SETID0: HRRZ T1,CHNNUM##(P1)
PUSHJ P,SWPP1 ;POINT PI AT SWPLST ENTRY
PUSH P,U
PUSHJ P,DONE##
MOVEI F,SWPDDB##
POP P,U
MOVE J,UNIKON##(U) ;DONE ZAPS EVERYBODY
HRRZ P1,KONCHN##(J)
MOVSI T1,UNPUNO## ;SINCE WE WILL TRY SWAP IF OFF-LINE
ANDCAM T1,UNIDES##(U) ; CLEAR UNPOFL,UNPUNS
> ;END FTVM
> ;END FTSWAP
SETID1:
IFN FTKI10!FTKL10,<
JUMPE P1,CHNIDX ;DONT START IO (CHANGE UBR) AT UUO LEVEL
>
IFN FTVM,<
MOVEI F,SWPDDB## ;MAKE SURE F POINTS TO THE SWAPPER
SKIPE SQREQ## ;SWAPPER GOING?
PUSHJ P,SWPSCN## ;YES, START SWAP SEEKS FIRST
>
SKIPG KONPOS##(J) ;DOES KONTROL POSITION?
JRST PIKTRX ;NO, LOOK FOR BEST TRANSFER
;YES. START ANY WAITING POSITIONS
;HERE TO PICK THE BEST FILE ON EACH UNIT TO START POSITIONING
PIKPOS:
IFN FTDHIA,<
SKIPLE DIADSK## ;IF WAITING FOR DIAG
CAME P1,DIACHN## ; FOR THIS CHANNEL
CAIA
JRST PIKTRX ;DON'T START ANOTHER SEEK
>
PUSH P,U ;SAVE U (LAST UNIT TO LOOK AT)
KONLUP: HLR U,UNIKON##(U) ;STEP TO NEXT UNIT IN RING
SKIPE T1,UNISTS##(U) ;GET STATE OF UNIT
CAIE T1,PWCOD## ;PW?
CAIN T1,SWCOD## ;OR SW?
PUSHJ P,UNIPOS ;YES, START UNIT POSITIONING
PIKPND: CAME U,(P) ;WAS THIS LAST UNIT ON KONTROL?
JRST KONLUP ;NO, STEP TO NEXT UNIT
POP P,T1 ;YES, REMOVE U FROM PD LIST
SKIPN SQREQ## ;DON'T REREAD IF SWAPPER WANTS SERVICE
PUSHJ P,UNIRHB ;SEE IF ANY UNIT NEEDS REREADING HOME BLOCKS
JRST PIKTRX ;NO, GO LOOK FOR FILE TO START DATA ON
PUSHJ P,FAKDDX ;YES, GET A DDB
JRST PIKTRX ;NO SPACE NOW, PUSH ON AND TRY LATER
PJRST TSTRHX ;GOT 1, GO START TO READ HOME BLOCK
IFN FTDHIA,<
;ROUTINE TO START POSITIONING DURING CRANK-UP
CRNPOS: MOVE J,UNIKON##(U) ;MAKE SURE J IS RIGHT
SETZM UNISTS##(U) ;SO RIGHT UNISTS WILL BE STORED (UUOTWQ)
>
;SUBROUTINE TO PICK A FILE ON A UNIT AND START POSITIONING FOR THAT FILE
;ENTER WITH U=LOC OF UNIT DATA BLOCK
;EXIT CPOPJ, A FILE IS NOW POSITIONING ON THE UNIT
UNIPOS: PUSHJ P,SAVE4##
IFE FTVM,<
SKIPN UNIQUE##(U) ;GET UNIQUE = PWQ FOR UNIT
STOPCD CPOPJ##,DEBUG,PQE, ;++POSITIONING QUEUE EMPTY
>
IFN FTVM,<
SKIPE UNIQUE##(U)
JRST UNIPS0
IFN FTDHIA&FTDUAL,<
SKIPN DIADSK## ;OK IF SHUTTING DOWN CHAN
>
SKIPL SCNCNT## ;IF NO DDB IN PWQ,
POPJ P, ; ITS OK IF SWPSCN PUT UNIT IN PW
STOPCD CPOPJ##,DEBUG,PQE,
UNIPS0:>
IFE FTDOPT,< ;NO DISK OPTIMIZATION
UNIPS1: HLRZ F,UNIQUE##(U) ;F=FIRST FILE IN UNIT POSITION QUEUE
MOVEI P2,DIFUDQ##(U) ;P2=PREDECESSOR
SKIPE T1,DEVUNI##(F) ;SKIP IF F/S HAS BEEN JERKED OUT
PUSHJ P,CYLCOM ;COMPUTE DISTANCE TO TARGET CYLINDER
JUMPE T1,PWQOK ;JUMP IF ON-CYLINDER, SET UNIT TO TW
> ;END CONDITIONAL ON FTDOPT
IFN FTDOPT,<
PUSH P,S ;SAVE S
UNIPS1: HRLOI P3,377777 ;SET P3 (BEST DISTANCE) TO PLUS INFINITY
HLRZ F,UNIQUE##(U) ;F WILL GO THROUGH FILES ON U
MOVEI S,DIFUDQ##(U) ;INITIALIZE S (PREDECESSOR FILE)
SKIPN T1,DEVUNI##(F) ;JUST IN CASE F/S HAS BEEN JERKED OUT
JRST PWQLP1 ;JERKED OUT - SET DISTANCE=0
; (SO WILL IMMEDIATELY GO TO TWQ)
IFN FTDPRI,<
PUSHJ P,DFPRI ;PRIORITY OF FIRST FILE IN QUEUE
MOVE R,T2 ;SAVE IN R
>
PWQLUP: PUSHJ P,CYLCOM ;COMPUTE DISTANCE TO TARGET CYLINDER
MOVMS T1 ;ABSOLUTE VALUE
SKIPL DEVSWP##(F) ;IS THIS SWAPPER REQUEST?
CAMGE T1,P3 ;NO, IS THIS BEST SO FAR?
PWQLP1: PUSHJ P,SVBST ;YES, SAVE POINTERS TO IT
;P4=BEST F,P3=DIST TO BEST P2=PRED. F
JUMPE T1,PWQOK ;PERFECT IF 0 (SET UNIT TO TW)
SKIPGE DEVSWP##(F) ;SWAPPER REQUEST?
JRST SETPOS ;YES. USE THIS FILE
MOVEM F,S ;SAVE F AS PREDECESSOR POINTER
HLRZ F,DEVQUE##(F) ;AND STEP TO NEXT FILE ON UNIT
IFN FTDPRI,<
JUMPE F,SETPOS ;START POSITION IF LOOKED AT ALL IN QUEUE
PUSHJ P,DFPRI ;PRIORITY OF THIS FILE
CAME T2,R ;SAME AS FIRST IN QUEUE?
JRST SETPOS ;NO, START POSITION OF BEST IN PRIORITY GROUP
JUMPG R,PWQLUP ;YES, CHECK IT IF PRIORITY IS POSITIVE
; (NEVER BY FAIR IF HI-PRI FILES IN QUEUE)
>
SKIPLE CHNCFP##(P1) ;IF TIME TO BE FAIR, FIRST IS BEST
JUMPN F,PWQLUP ;LOOP IF NOT END OF FILES, NOT FAIR TIME
SETPOS: POP P,S ;RESTORE S
;STILL IN FTDOPT CONDITIONAL
;HERE P4 HAS POINTER TO FILE, P2=PREDECESSOR
;REMOVE THIS FILE FROM PWQ, START IT POSITIONING
MOVE F,P4 ;SET F TO FILE
> ;END CONDITIONAL ON FTDOPT
PUSHJ P,UNQUE0 ;REMOVE FILE FROM Q
MOVE T1,UNISTS##(U) ;STATUS OF UNIT
CAIN T1,SWCOD## ;SEEK WAIT?
AOJA T1,STRPS0 ;YES, SET STATE=SEEK AND START POSITION GOING
PJRST STRPOS ;NO, SET STATE=POSITION AND START POSITION GOING
;HERE IF A FILE IS ALREADY ON CYLINDER
PWQOK: PUSHJ P,UNQUE0 ;REMOVE FILE FROM PWQ
DSKOFF ;#TURN ALL DISK PI'S OFF
MOVE T1,UNISTS##(U) ;#STATUS OF UNIT
CAIN T1,SWCOD## ;#UNIT IN SEEK WAIT?
JRST PWQOK2 ;#YES. DO SOMETHING ELSE
HRRM F,UNICDA##(U) ;SO SWAPPER WILL KNOW
IFN FTDUAL,<
PUSH P,U
>
PUSHJ P,UUOTWQ ;#PW STATE - ADD FILE TO TWQ,SET STATE TO TW
;(UNLESS THERE WAS A SEEK WHICH WE FORGOT
;BECAUSE OF A POSITION REQUEST, AND THE POSITION
;IS ALREADY ON-CYLINDER. IN THAT CASE, I/O
;WILL BE STARTED)
IFN FTDUAL,<
POP P,U ;IF WE STARTED IO ON ALTERNATE UNIT
HRRZ J,UNIKON##(U) ;THEN U, J AND P1 WERE CHANGED
HRRZ P1,KONCHN##(J) ;SO RESET THEM
>
PWQOK1:
IFN FTDOPT,<
POP P,S ;RESTORE S
> ;END CONDITIONAL ON FTDOPT
POPJ P, ;AND RETURN
;HERE IF POSITION REQUEST IS FOR A SEEK WHICH IS ON-CYLINDER
PWQOK2: SKIPE UNIQUE##(U) ;ANY POSITIONS WAITING?
JRST UNIPS1 ;YES, START POSITION (FORGET SEEK)
SETZM UNISTS##(U) ;NO, SET UNIT IDLE
JRST PWQOK1 ;RESTORE S AND RETURN
;SUBROUTINE TO UPDATE DDB, ADVANCE BUFFERS AFTER A DATA TRANSFER
;ENTER WITH P4=CHNNUM= NUMBER OF BUFFERS TO ADVANCE
;RETURNS WITH P3=ADVBFE OR ADVBFF DEPENDING ON INPUT OR OUTPUT
;CHANGES P3,P4
BUFAD: MOVNM P4,T1 ;DECREASE NUMBER OF BLOCKS LEFT
ADDM T1,DEVLFT##(F) ;BY NUMBER OF BLOCKS XFERRED
ADDM P4,DEVBLK##(F) ;UPDATE FILE CURRENT BLOCK NUMBER
IFN FTDSUP,< ;SUPER USETI/USETO
TLNE S,IOSUPR ;IO FROM SUPER USETI/USETO?
JRST BUFAD0 ;YES. DONT TOUCH A.T.
>
ADDB P4,DEVREL##(F) ;UPDATE CURRENT RELATIVE BLOCK NUMBER
SOJN P4,BUFAD4 ;-1=HIGHEST BLOCK WRITTEN
TLO S,IOSFIR ;NEXT BLOCK = 1 - SET FOR CHKSUM
MOVEM S,DEVIOS(F) ;SAVE S IN DDB
BUFAD4: HRRZ T1,DEVACC##(F) ;ACCESS TABLE LOC
MOVE T3,ACCWRT##(T1) ;SET T3 NEGATIVE IF NEW
SUB T3,P4 ; LAST BLOCK, 0 IF OLD LAST
CAMLE P4,ACCWRT##(T1) ;THIS BLOCK HIGHER THAN FORMER HIGHEST?
MOVEM P4,ACCWRT##(T1) ;YES, SAVE NEW HIGHEST WRITTEN
SKIPE ACCWRT##(T1) ;IF RIB, NOT LAST BLOCK
CAME P4,ACCWRT##(T1) ;LAST BLOCK?
TLZA P3,-1 ;NO, LH(P3)=0
HRL P3,T1 ;YES, LH(P3)=LOC OF A.T.
BUFAD0: HRRZ P4,CHNNUM##(P1) ;NUMBER OF BLOCKS TRANSFERRED AGAIN
LDB T2,PIOMOD## ;GET MODE
MOVEI T1,UNIDRC##(U) ;SET FOR DUMP-MODE STATS
CAIL T2,SD ;DUMP?
PJRST UPSTA ;YES. UPDATE UNIDRC(DWC) AND RETURN
MOVEI T1,UNIBRC##(U) ;NO. SET TO UPDATE BUFFERRED-MODE STATS
PUSHJ P,UPSTAT ;UPDATE UNIBRC(BWC)
HRRI P3,ADVBFE## ;SET TO ADVANCE BUFFERS
TLNN S,IO
HRRI P3,ADVBFF##
JUMPE P4,CPOPJ1## ;RETURN IF NO BUFFERS TO ADVANCE
BUFAD1: SOJN P4,BUFAD2 ;GO IF NOT LAST BUFFER
HLRZ T2,P3 ;LAST - LOC OF ACC IF READING LAST BLOCK OF FILE
JUMPE T2,BUFAD2 ;NOT LAST BLOCK IF 0
;HERE WHEN DOING I/O TO LAST BLOCK OF THE FILE
IFN FTKI10!FTKL10,<
PUSHJ P,SVEUF## ;SETUP USER BASE REGISTER SO CAN ADDRESS BUFFERS
>
TLNE S,IO ;WRITING?
JRST BUFAD3 ;YES, COMPUTE LBS
LDB T1,ACYLBS## ;READING LAST BLOCK - GET ITS SIZE
MOVE T2,DEVIAD(F)
IFN FTKA10,<
MOVEI T2,@T2
>
EXCTXU <MOVEM T1,1(T2)>
BUFAD2: PUSHJ P,(P3) ;ADVANCE BUFFERS
JUMPE P4,CPOPJ## ;RAN OUT - RETURN IF LAST BUFFER
JUMPG P4,BUFAD1 ;ADVANCE MORE IF NOT LAST BUFFER
JRST CPOPJ1## ;ANOTHER BUFFER AVAILABLE - SKIP RETURN
;HERE WHEN WRITING LAST BLOCK OF A FILE - COMPUTE LAST BLOCK SIZE
BUFAD3: MOVE T1,DEVOAD(F) ;ADDRESS OF LAST BUFFER
IFN FTKA10,<
MOVEI T1,@T1 ;RELOCATE IF A KA
>
EXCTUX <MOVE T1,1(T1)>;GET WRDCNT OF LAST BUFFER
CAILE T1,BLKSIZ## ;TOO MANY WORDS?
MOVEI T1,BLKSIZ## ;YES, REDUCE COUNT
JUMPL T3,BUFAD5 ;NEW LAST BLOCK IF NEGATIVE
MOVE T3,DEVACC##(F) ;LOC OF A.T.
MOVE T3,ACCSTS##(T3) ;STATUS WORD
TRNN T3,ACPUPD ;UPDATE MODE?
TDZA T3,T3 ;NO
LDB T3,ACYLBS## ;YES, GET PREVIOUS LAST-BLOCK LENGTH
CAILE T3,BLKSIZ## ;TOO MANY WORDS?
MOVEI T3,BLKSIZ## ;REDUCE COUNT
CAMLE T3,T1 ;CURRENT LENGTH LESS THAN PREVIOUS LENGTH?
MOVE T1,T3 ;YES, SET CURRENT=PREVIOUS LENGTH
BUFAD5: DPB T1,ACYLBS## ;SAVE IN ACC
JRST BUFAD2 ;AND GO ADVANCE BUFFERS
;SUBROUTINE TO COMPUTE A FOLDED CHECKSUM FROM THE FIRST DATA WORD
;CALL WHEN THE IO LIST HAS BEEN SET UP IN THE KONTROLLER DATA BLOCK
CHKSUM:
IFN FTKI10!FTKL10,<
PUSHJ P,SVEUF## ;MAKE ADDRESSABLE
PUSH P,J ;SAVE J
PUSH P,M ; AND M
LDB J,PJOBN## ;GET NEW JOB #
HLRZ M,DEVUVA##(F) ;GET L(1ST WORD)-1
MOVEI M,1(M) ;=ACTUAL I/O ADDRESS
PUSHJ P,GETWRD## ;GET WORD FROM USER AREA
STOPCD .,STOP,CSE, ;++CHECKSUM ERROR
MOVE T2,T1 ;MOVE IT
IFN FTKL10&FTMS,<
MOVE T1,M ;CHASE WORD FROM CACHE
PUSHJ P,OUCHE## ; IN CASE IT RUNS ON OTHER CPU
>
POP P,M ;RESTORE STUFF
POP P,J ;...
>
IFN FTKA10,<
HLRZ T2,DEVUVA##(F) ;ADDRESS OF THE CHECKSUM WORD
TLO T2,R ;RELOCATE
HRRI T2,1(T2)
MOVE T2,@T2 ;GET CHECKSUM WORD
>
;SUBROUTINE TO COMPUTE A CHECKSUM FROM T2
;ENTER WITH T2=WORD TO BE CHECKSUMMED
;EXIT WITH T1 = CHECKSUM
CHKST1::HRRZ T4,UNISTR##(U) ;LOC OF STR DB
MOVE T4,STYCKP##(T4) ;CHECKSUM POINTER
LDB T3,[POINT 6,T4,11] ;SIZE FIELD OF CHKSUM PNTR
MOVNS T3 ;SET FOR LSH
TLZA T4,770000 ;SET TO BIT 35
CHKSM1: ADD T2,T1 ;NOT DONE. ADD BYTE TO REST OF WORD (FOLD CHKSUM)
LDB T1,T4 ;GET A BYTE OF CHKSUM SIZE
LSH T2,(T3) ;THROW AWAY THE BYTE
JUMPN T2,CHKSM1 ;FINISHED WHEN NO MORE OF ORIGINAL WORD
POPJ P, ;DONE - RETURN
;SUBROUTINE TO UPDATE UNIT STATISTICS AFTER IO
;ENTER WITH P4=NUMBER OF BLOCKS, T1=UNIXRC(U) (X=M,B OR D)
;UPSTA = UPSTAT,PRESERVES T3
UPSTA:
UPSTAT: TLNE S,IO ;WRITING?
ADDI T1,1 ;YES. UNIXWC=UNIXRC+1
ADDM P4,(T1) ;UPDATE WORD IN UNIT BLOCK
IFN FTDMRB,< ;IF MULTIPLE RIBS
SKIPL DEVRIB##(F) ;IO TO FILE CURRENTLY USING EXTENDED RIB?
> ;END CONDITIONAL ON FTDMRB
POPJ P, ;NO, RETURN
IFN FTDMRB,< ;IF MULTIPLE RIBS
HRRZ T1,P4 ;NUMBER OF BLOCKS TRANSFERRED
TLNE S,IO ;WRITING?
HRLZ T1,T1 ;YES, MOVE NUMBER OF BLOCKS TO LEFT HALF
ADDM T1,UNIXRA##(U) ;AND STORE IN UNIT DATA BLOCK
POPJ P, ;AND RETURN
> ;END CONDITIONAL ON FTDMRB
;SUBROUTINE TO COMPUTE DISTANCE TO TARGET CYLINDER
;ENTER WITH TARGET BLOCK IN DEVBLK(F)
;ENTER AT CYLCM WITH T1=BLOCK NUMBER, T4 PRESERVED
;EXIT WITH T1 =DISTANCE FROM CURRENT CYLINDER TO TARGET CYLINDER
CYLCOM: MOVE T1,DEVBLK##(F) ;#TARGET BLOCK
CYLCM:: LDB T3,UNYBPY## ;#NUMBER OF BLOCKS PER CYLINDER
IDIV T1,T3 ;#COMPUTE CYLINDER
SUB T1,UNICYL##(U) ;#-PRESENT CYLINDER
POPJ P, ;#EXIT T1=DISTANCE
;SUBROUTINE TO REMOVE A FILE FROM A QUEUE
; UNQUEX FOR CHANNEL (XFER WAIT) QUEUES
; P1=ADDR. OF CHAN. DATA BLOCK
; UNQUE0 FOR UNIT (POSITION WAIT) QUEUES
; U=ADDR. OF UNIT DATA BLOCK
;F HAS FILE P2 HAS PREDECESSOR
UNQUEX:
IFN FTDSTT,< ;IF KEEPING DISK STATISTICS
SOSA CHNQUL##(P1) ; DECREMENT XFER QUEUE LENGTH
>
UNQUE0:
IFN FTDSTT,<
IFE FTVM,<
SOS UNIQUL##(U) ; OR POSITION WAIT QUEUE LENGTH
>
IFN FTVM,<
SOSA UNIQUL##(U)
>
>
IFN FTVM,<
SOS CHNWAT##(P1)
>
HLLZ T1,DEVQUE##(F) ;Q WORD OF FILE
HLLM T1,DEVQUE##(P2) ;SAVE IN Q WORD OF PREDECESSOR
POPJ P, ;AND RETURN
;SUBROUTINE TO START UNIT POSITIONING
STRPOS::MOVEI T1,PCOD## ;#PUT FILE AND UNIT INTO P STATE
STRPS0: PUSHJ P,FILCOD ;#
IFN FTDUAL,<
PUSHJ P,SECCOD ;SET MAIN UNIT TO SAME STATE
HRRM U,DEVCUR##(F)
>
STRPS1:
IFN FTMETR,<
MOVE T1,UNISTS##(U) ;GET STATE FOR METER POINT IN SETPAR
>
PUSHJ P,SETPRS ;#SET KONCUA, UNICDA, UNIBLK
PUSHJ P,@KONPOS##(J) ;#GO TO DEPENDENT ROUTINE
JRST STRPS2 ;UNIT NOT OK - PUT IT INTO T OR TW
; (UNLESS CALLED FROM HNGDSK)
;SO THAT STARTIO WILL CALL BADUNI
DSKON ;# TURN ALL DISK PI'S BACK ON
SCHEDULE
POPJ P, ;AND RETURN
;HERE IF UNIT NOT READY WHEN TRYING TO START POSITIONING
STRPS2: MOVE T1,UNISTS##(U)
SETZM UNISTS##(U) ;SO WE'LL STORE NEW UNISTS
CAIN T1,SCOD ;IF TRYING TO START A SEEK,
POPJ P, ; FORGET IT
IFE FTDUAL,<
PJRST SETTTW ;TRY TO START I/O
>
IFN FTDUAL,<
PUSH P,U
PUSHJ P,SETTTW
POP P,U ;IF WE STARTED IO ON ALTERNATE UNIT
HRRZ J,UNIKON##(U) ;THEN U, J AND P1 WERE CHANGED
HRRZ P1,KONCHN##(J) ;SO RESET THEM
POPJ P,
>
IFN FTDOPT,<
;SUBROUTINE TO SAVE POINTERS TO BEST FILE SO FAR
;P1-P3 CHANGED
SVBST: MOVEM F,P4 ;P4 = BEST FILE SO FAR
HRRZM T1,P3 ;P3 = DISTANCE TO BEST
MOVEM S,P2 ;P2 = PREDECESSOR IN QUEUE
POPJ P,
> ;END CONDITIONAL ON FTDOPT
;SUBROUTINE TO SET UP PARAMETERS FOR DEPENDENT ROUTINE
SETPRS:
IFN FTRP04,<
SKIPL KONTAB##(J) ;#DONT STORE NEW KONCUA IF THIS IS A POSITION
; STARTED WHILE XFER IS ACTIVE
>
SETPAR: HRRM U,KONCUA##(J) ;#SAVE CURRENT UNIT ADDR
HRRM F,UNICDA##(U) ;#SAVE FILE ADDR
IFN FTDUAL,<
SKIPGE T2,UNI2ND##(U) ;IF STARTING 2ND PORT,
HRRM F,UNICDA##(T2) ; MAKE SURE WORLD KNOWS ABOUT MAIN PORT
>
IFN FTMETR,
< ;SET UP T1 FOR METER POINT 3
;SHOULD ALREADY CONTAIN
; OPCODE 4 OR 6
ROT T1,-2 ;#GET THE DISTINGUISHING BIT INTO BIT 0
> ;END CONDITIONAL ON FTMETR
MOVE T2,DEVBLK##(F) ;# SAVE LOGICAL BLOCK NR WITHIN UNIT
MOVEM T2,UNIBLK##(U) ;#IN UNIT DATA BLOCK
HRR T1,T2
MOVE S,DEVIOS(F) ;#GET S
IFN FTMETR,
< SKIPL T2,MP3## ;# POINT INITIALIZED
PJRST STOIOS## ;#NO, RESET HUNG TIME AND RETURN
DPB F,[POINT 11,T1,11] ;# STORE DDB ADR
DPB U,[POINT 8,T1,17] ;# AND UDB ADR
PUSHJ P,@MPDPRA##(T2) ;# CALL METER POINT ROUTINE
> ;END CONDITIONAL ON FTMETR
PJRST STOIOS ;# RESET HUNG TIME AND EXIT
;SUBROUTINE TO DETERMINE IF A POINTER TO THE NEXT BLOCK OF A FILE IS IN CORE
;ENTER WITH J,U,F SET UP
;RETURN CPOPJ IF NEXT BLOCK NOT AVAILABLE
;RETURN CPOPJ1 IF THE NEXT BLOCK HAS A POINTER ALREADY IN CORE
CHKNXT: HRRZ T1,DEVLFT##(F) ;NUMBER OF BLOCKS LEFT IN CURRENT GROUP
SOJGE T1,CHKNX2 ;CPOPJ1 IF ANY LEFT AND WRITING
MOVE T1,DEVRET##(F) ;LOOK AT NEXT POINTER
CHKNX1: SKIPE T2,1(T1) ;IF ZERO (EOF)
CAIL T1,DEVRBN##(F) ;OR END OF POINTER BLOCK
POPJ P, ;NONE LEFT IN CORE
HLRE T1,DEVRSU##(F) ;DEVRSU
CAMGE T1,[-2] ;NO MORE IF MIGHT GET REDUNDANT RIB
TLNN T2,-1 ;CHANGE OF LOGICAL UNITS?
POPJ P, ;NOT AVAILABLE SINCE IT IS ON ANOTHER UNIT
;(CANT SWITCH UNITS ON INTERRUPT LEVEL SINCE STATE
; OF OTHER UNITIS INDEPENDENT OF THIS ONE)
PUSHJ P,NXTBLK ;STORE NEW DEVBLK IN CASE THIS UNIT PUT BACK
; IN PW AND NEW PNTR IS FOR A NEW CYLINDER
POPJ P, ;NEW BLOCK PAST WRITTEN DATA - RETURN
CHKNX2: HRRZ T2,DEVACC##(F) ;SAME UNIT. LOC OF A.T.
JUMPE T2,CPOPJ## ;UNAVAILABLE IF F/S WAS JERKED
MOVE T2,ACCWRT##(T2) ;NUMBER OF BLOCKS WRITTEN
TLNN S,IO ;IF READING,
CAML T2,DEVREL##(F) ;IS NEXT READ PAST THE EOF?
AOS (P) ;NO. OK
POPJ P, ;RETURN CPOPJ OR CPOPJ1
;HERE WHEN THE UNIT IS NOT READY
BADUNI: HRRZS UNISTS##(U) ;#CLEAR SIGN BIT (PHUNG)
IFN FTDAEM,<
MOVEM T2,UNIERR##(U) ;#YES, SAVE CONI, DATAI
MOVEM T3,UNIHDI##(U) ;#
>
IFN <FTKI10!FTKL10>,<
IFN FTVM,<
SKIPGE DEVSWP##(F) ;#DON'T RETURN IOWD'S IF THE SWAPPER
JRST BADUNA
>
MOVE T3,UNISTS##(U) ;IF THE UNIT IS IN T STATE
SKIPG UNIECT##(U) ; AND NOT IN ERROR-RECOVERY
CAIE T3,TCOD##
JRST BADUNA
PUSH P,T1 ;SAVE T1
SKIPE T1,@KONIOC##(J) ;GET LOC OF FIRST CORE BLOCK
PUSHJ P,RTNIOW## ;RETURN THE FREE-CORE BLOCKS
POP P,T1 ;RESTORE T1 AND CONTINUE
BADUNA:> ;END FTKI10!FTKL10
MOVSI T4,UNPOFL##
TRNE T1,KOPFUS## ;#UNIT GO FILE-UNSAFE?
TLO T4,UNPFUS## ;#YES, LIGHT ANOTHER BIT TOO
TRNE T1,KOPUSI##
TLO T4,UNPUSI##
TRNN T1,KOPOFL## ;#UNIT OFF-LINE?
JRST BADUN3 ;# NO, CALL HNGSTP
IORM T4,UNIDES##(U) ;#YES - LIGHT UNIDES BIT
IFN FTDBAD,<
SKIPE DEVRHB##(F) ;#IF REREADING HOME BLOCKS,
SKIPE DINITF## ;#OR IF IN ONCE-ONLY CODE,
JRST BADUN6 ;#DONT DO THIS OPR-WAIT STUFF
IFN FTDAEM,<
MOVE T2,UNIBLK##(U) ;#
MOVEM T2,UNIHBN##(U) ;#SAVE BLOCK NUMBER
AOS UNIHCT##(U) ;#BUMP ERROR COUNT
PUSH P,T1 ;SAVE STATUS
PUSHJ P,LSTER ;COPY REGISTERS FROM KDB TO UDB
MOVEI T1,.ERDPE ;SAY DISK ERROR TO DAEMON
HRL T1,F ;PUT DDB ADDR IN LH FOR DAEMON
PUSHJ P,DAEERR## ;#AND CALL DAEMAN
POP P,T1 ;#RESTORE STATUS
> ;END FTDAEM
;STILL IN FTBAD CONDITIONAL
;HERE IF JOB NOT INTERCEPTING UNIT OFF-LINE
TLO P1,-1 ;NO INTERCEPTS, INDICATE OFF-LINE
MOVE T1,[XWD TELOPC,1] ;#SET FOR CLOCK REQUEST
CONO PI,PIOFF## ;#SET TO STORE IN THE CLOCK QUEUE
CONSO PI,PIPROG## ;#ON PI LEVEL?
JRST BADLP2 ;#NO
IDPB T1,CLOCK## ;#YES, TELL OPR ON NEXT TICK
IDPB U,CLOCK## ;SAVE ADR OF UDB
BADLP2: CONO PI,PION## ;#IN ORDER TO TYPE MESSAGE TO OPR
CONSO PI,PIPROG## ;#DONT TELL OPR NOW IF ON PI LEVEL
PUSHJ P,TELOPR ;#NOT ON PI LEVEL - TYPE MESSAGE NOW
IFN FTVM,<
SKIPGE DEVSWP##(F) ;#IF THE SWAPPER,
JRST BADU4A ; LET SWPHNG RECOVER
>
MOVEI T1,OWCOD## ;#SET UNIT IN OPR WAIT STATE
MOVEM T1,UNISTS##(U) ;#(ONLY GETS OUT OF OW ON "FREE" INTERRRUPT)
BADUN3: JUMPL S,BADUN4 ;#IF NOT A MONITOR I/O OPERATION,
MOVE T1,KONDMP##(J) ;#GET IOWD BACK
LDB T2,PIOMOD## ;#MODE
CAIL T2,SD ;#DUMP?
MOVEM T1,DEVDMP##(F) ;#YES, RESET DEVDMP IN CASE "CONT" IS TYPED
JUMPGE P1,BADUN4 ;#GO IF NOT OFF-LINE
MOVEI T4,.EROFL ;DISK OFF-LINE ERROR
PUSHJ P,SETINJ ;IS HE INTERCEPTING?
JRST BADUN4 ;NO
MOVEI T1,OW2COD## ;YES, SET UNIT O2 (OPR WAIT, NO MESSAGE EACH MINUTE)
MOVEM T1,UNISTS##(U) ; (ONLY GETS OUT OF O2 ON "FREE" INTERRUPT)
;JOB IN IO WAIT?
PUSHJ P,STDIOD## ;YES, UNWAIT HIM
BADUN4:
IFN FTDUAL,<
MOVE T1,UNISTS##(U)
CAIN T1,TCOD## ;IF WRITE-LOCK
SETZ T1, ;MAKE OTHER PORT IDLE
SKIPGE T2,UNI2ND##(U) ;DUAL-PORTED UNIT?
MOVEM T1,UNISTS##(T2) ;YES, PUT OTHER PORT IN OPR WAIT TOO
>
JUMPGE P1,BADUN5 ;#IF UNIT WENT OFF-LINE,
PUSH P,F ;#SAVE F
IFN FTKI10!FTKL10,<
CONSO PI,PIPROG## ;IF ON UUO LEVEL AND A SEEK FINISHED
SETZ P1, ; CAN'T START IT - CANT CALL SVEUB
>
PUSHJ P,SETID1 ;#GO FIND SOMETHING ELSE TO DO
;#THIS (UNIT WONT BE USED TILL COMES OUT OF OW)
POP P,F ;#RESTORE F
BADU4A: DSKON
POPJ P, ;AND RETURN
;STILL IN FTDBAD CONDITIONAL
;HERE WHEN THE UNIT DID NOT GO OFF-LINE (PROBABLY WRITE-LOCK)
BADUN5: DSKON ;#TURN ON DSK PI
IFE FTVM,<
SKIPGE DEVSWP##(F) ;CAN'T CALL HNGSTP IF THE SWAPPER
JRST BADUN6 ;(P) = NULPDL
>
IFN FTVM,<
SKIPGE DEVSWP##(F) ;IF THE SWAPPER,
JRST BADU4A ; LET SWPHNG RECOVER
>
CONSZ PI,PIPROG## ;IS A PI IN PROGRESS (P-10 ONLY)
JRST BADU5A
PUSHJ P,SAVSTS ;MAKE A RECORD OF RESOURCES, RETURN THEM
JRST BADUN6 ;HAS MON-BUF OR PI IN PROGRESS, HE LOSES
HRRZ J,UNIKON##(U) ;SAVSTS CLOBBERS J
PUSH P,F ;NO, SAVE F
IFN FTKI10!FTKL10,<
SETZ P1, ; INTERRUPTED DONT START THE IO
; (CANT CHANGE UBR AT UUO LEVEL)
>
PUSHJ P,SETIDA ;SET UNIT IDLE (DON'T CLEAR IOSMON), START NEXT OPERATION
MOVE F,(P) ;RESET DDB ADDR FOR HNGSTP
PUSHJ P,HNGSTP## ;"DEVICE DEV OK?"
POP P,F ;RESTORE F,
POP P,T3
PUSHJ P,RESSTS ; U, AND ANY RESOURCES JOB OWNED
HRRZ J,UNIKON##(U) ; AND J
PJRST UUOPW0 ;HE SAID CONTINUE - TRY AGAIN
BADU5A: TRO S,IOIMPM
PUSHJ P,STDIOD##
PJRST SETIDA
> ;END CONDITIONAL ON FTDBAD
;HERE ON INTERRUPT LEVEL OR DURING ONCE-ONLY
;LIGHT AN ERROR BIT AND RETURN TO CALLER
BADUN6: DSKON ;#TURN ON DSK PI'S
MOVE S,DEVIOS(F) ;GET CURRENT DEVIOS
TRO S,IOIMPM ;LIGHT IOIMPM
;IN IO WAIT?
PUSHJ P,STDIOD## ;YES. TAKE OUT
JUMPE U,CLRACT## ;RETURN IF F/S WAS JERKED
PUSH P,F ;SAVE F
IFN FTKI10!FTKL10,<
CONSO PI,PIPROG## ;CANT CHANGE UBR IF
SETZ P1, ; AT UUO LEVEL
>
PUSHJ P,SETIDL ;SET UNIT IDLE,START NEXT OPERATION
POP P,F
POPJ P, ;RETURN
IFN FTDBAD,<
;ROUTINE TO TYPE A MESSAGE TO THE OPR WHEN A DRIVE FOES OFF-LINE
;ENTER AT TELOPC FROM THE CLOCK, T1=UDB
;ENTER AT TELOPR WITH U SET CORRECTLY
TELOPC: SETZ J, ;SET J=0 AS A SWITCH
MOVE U,T1
IFN FTVM,<
HRRZ T1,UNICDA##(U) ;IF THE SWAPPER
SKIPL DEVSWP##(T1) ; UNIT IS STILL IN TW
>
MOVE T1,UNISTS##(U) ;GET STATUS OF UNIT
CAIGE T1,OWCOD## ;HAS UNIT ALREADY BEEN POWERED DOWN AND UP?
POPJ P, ;YES, DONT TYPE ANYTHING
;NO, TELL OPR ABOUT BAD UNIT
JRST TELOP3 ;SAVE ONLY U, AND TYPE
TELOPR: PUSH P,F
TELOP3: PUSH P,U
HRRZ U,OPRLDB## ;TYPE MESSAGE ON OPR'S TTY
PUSHJ P,INLMES##
BYTE (7) 7,7,7,7,7 ;SOME BELLS
ASCIZ /
UNIT /
MOVE T2,(P) ;UNIT
MOVE T2,UNINAM##(T2) ;PHYSICAL UNIT NAME
PUSHJ P,PRNAME## ;TYPE IT
IFN FTDUAL,<
MOVE T2,(P) ;UDB ADDR
SKIPL T2,UNI2ND##(T2);IS THIS AN ALTERNATE PATH?
JRST TELO3A ;NO
PUSH P,T2 ;YES, SAVE ITS ADDR
PUSHJ P,INLMES## ;TELL OPR THE NAME HE KNOWS
ASCIZ / = /
POP P,T2
MOVE T2,UNINAM##(T2)
PUSHJ P,PRNAME##
TELO3A:>
MOVE T2,(P)
MOVSI T1,UNPUSI##
TDNE T1,UNIDES##(T2)
JRST TELO3B
PUSHJ P,INLMES## ;AND THE MESSAGE:
ASCIZ / WENT OFF-LINE/
JRST TELO3C
TELO3B: ANDCAM T1,UNIDES##(T2)
PUSHJ P,INLMES##
ASCIZ / STATUS INCONSISTENT/
TELO3C: MOVE T2,(P) ;UNIT
MOVE T2,UNIDES##(T2) ;UNIDES WORD
TLNN T2,UNPFUS## ;FILE UNSAFE?
JRST TELOP4
PUSHJ P,INLMES## ;YES
ASCIZ / (FILE UNSAFE)/
;STILL IN FTDBAD
TELOP4: MOVE T2,(P) ;UDB AGAIN
HRRZ T2,UNIKON##(T2);KDB
SKIPG KONPOS##(T2) ;DOES THIS UNIT POSITION?
JRST TELOP5 ;NO, HE CANT TURN IT OFF
PUSHJ P,INLMES## ;YES, ASK HIM TO
ASCIZ /
PLEASE POWER IT DOWN, THEN TURN IT ON AGAIN./
TELOP5: PUSHJ P,INLMES##
ASCIZ /
/
POP P,U ;RESTORE ACS
JUMPE J,CPOPJ## ;DIDN'T SAVE ACS IF J=0
POP P,F
MOVE S,DEVIOS(F) ;RESTORE S
POPJ P, ;AND RETURN
;ROUTINE CALLED ONCE A MINUTE
CHKUNI::SETZM F
SETOM J
HLRZ U,SYSUNI## ;FIRST UNIT IN SYSTEM
CHKUN1: MOVE T1,UNISTS##(U) ;STATUS OF UNIT
CAIE T1,OCOD##
CAIN T1,OWCOD## ;WAITING FOR OPR?
PUSHJ P,TELOPR ;YES, REMIND HIM AGAIN
HLRZ U,UNISYS##(U) ;STEP TO NEXT UNIT
JUMPN U,CHKUN1 ;AND TEST IT
POPJ P, ;DONE - RETURN
> ;END CONDITIONAL ON FTDBAD
;HERE TO START TRANSFER OF BEST (LATENCY DETERMINED) FILE IN TW
PIKTRN:
IFE FTVM,<
PIKTRX:
>
IFN FTVM,<
MOVEI F,SWPDDB ;MAKE SURE WE'RE POINTING AT SWAPPER
SKIPE SQREQ ;SWAPPER WANT SERVICE?
PUSHJ P,SWPSCN## ;YES, LET IT DO PART 1 (SET ASL UNITS IN TW STATE)
PIKTRX: SKIPN SQREQ## ;SWAPPER WAITING?
JRST PIKTR0 ;NO
MOVEI F,SWPDDB ;YES, POINT F AT SWAPPER DDB
PUSHJ P,SWPPIK## ;GO FIND SOMETHING TO DO
CAIA ;COULDN'T, DO A FILE IO
POPJ P, ;SWAPPER IS GOING, EXIT
PIKTR0:> ;END FTVM
PUSHJ P,SAVE4## ;#
DSKOFF
MOVSI T1,400000 ;#SET OLD KONTROL IDLE
ANDCAM T1,KONTAB##(J) ;#
PIKTR1: HLRZ F,CHNQUE##(P1) ;#FIRST FILE DDB IN TW QUEUE FOR THIS CHAN
JUMPE F,CHNIDL ;#IF QUEUE EMPT, SET CHAN IDLE AND EXIT
DSKON ;#
IFE FTDOPT,<
;HERE IF NO LATENCY OPTIMIZATION - PICK FIRST IN QUEUE
MOVEI P2,DIFCDQ##(P1) ;SET P2=PREDECESSOR
TWQLUP: HRRZ U,DEVUNI##(F) ;U=UNIT FILE IS ACCESSING
JUMPE U,SETBSY ;GO IMMEDIATELY IF F/S HAS BEEN JERKED OUT
SKIPE T2,UNISTS##(U) ;STATUS OF UNIT, SKIP IF IDLE
CAIN T2,TWCOD## ;SKIP IF NOT TRANSFER WAIT
JRST SETBSY ;IDLE OR TW, PICK THIS TRANSFER
MOVEM F,P2 ;UNIT IS BUSY, LOOK FOR NEXT
HLRZ F,DEVQUE##(F) ;F=NEXT FILE IN QUEUE, P2=PREDECESSOR
JUMPN F,TWQLUP ;TEST NEXT FILE IF ANY
PJRST CHNIDL ;SET CHANNEL IDLE, NO WORK TO DO
;HERE WHEN A FILE IS FOUND, START IO
SETBSY: MOVE J,UNIKON##(U) ;J=CONTROLLER FILE IS ACCESSING
> ;END CONDITIONAL ON FTDOPT
IFN FTDOPT,<
;HERE TO FIND THE BEST FILE TO START TRANSFERRING
HRLOI P3,377777 ;PRESET BEST LATENCY TIME
PUSH P,S ;SAVE S
MOVEI S,DIFCDQ##(P1) ;PRESET PREDECESSOR
IFN FTDPRI,<
PUSHJ P,DFPRI ;PRIORITY OF FIRST ITEM IN QUEUE
MOVE R,T2 ;SAVE IN R
>
;STILL IN FTDOPT CONDITIONAL
TWQLUP: MOVE T1,DEVBLK##(F) ;BLOCK FILE NEEDS
IFE FTDUAL,<
HRRZ U,DEVUNI##(F) ;SET UP U
>
IFN FTDUAL,<
SKIPE U,DEVUNI##(F)
HRRZ U,DEVCUR##(F)
>
JUMPE U,TWQLP0 ;JUST IN CASE F/S HAS BEEN JERKED OUT
SKIPE T2,UNISTS##(U) ;STATUS OF UNIT
CAIN T2,TWCOD## ;TW?
SKIPA J,UNIKON##(U) ;IDLE OR TW - SET J=KONTROLLER FOR UNIT
JRST TWQLP1 ;UNIT IN TW - DONT GET ANOTHER FILE FOR TW
IFN FTDUAL,<
SKIPGE T2,UNI2ND##(U) ;IF THIS IS AN ALTERNATE PATH
MOVE T2,UNISTS##(T2) ; AND THE MAIN UNIT IS BUSY
CAIN T2,TCOD##
JRST TWQLP1 ; THEN LEAVE ALTERNATE ALONE
>
PUSHJ P,@KONLTM##(J) ;COMPUTE LATENCY
TWQLP0: MOVEI T1,0 ;UNIT NOT OK- SET LATENCY=0
; SO UNIT WILL BE GOTTEN RID OF IMMEDIATELY
SKIPL DEVSWP##(F) ;SWAPPER REQUEST?
CAMG T1,P3 ;NO. BEST SO FAR?
PUSHJ P,SVBST ;YES, SAVE POINTERS TO FILE
; P4=BEST F, P3=DIST TO BEST, P2=PRED.F
JUMPE T1,SETBSY
SKIPGE DEVSWP##(F) ;SWAPPER REQUEST?
JRST SETBSY ;YES. USE THIS FILE
TWQLP1: MOVEM F,S ;SAVE F AS PREDECESSOR
HLRZ F,DEVQUE##(F) ;STEP TO NEXT FILE
IFN FTDPRI,<
JUMPE F,TWQLP2 ;START IO OR SET CHAN IDLE
PUSHJ P,DFPRI ;HAVE A QUEUE ENTRY - GET PRIORITY
CAME T2,R ;SAME AS FIRST IN QUEUE?
JRST TWQLP2 ;NO, START IO
JUMPG R,TWQLUP ;YES, DON'T BE FAIR IF HI PRIORITY
>
SKIPLE CHNCFT##(P1) ;TIME TO BE FAIR?
JUMPN F,TWQLUP ;NO, TEST NEXT FILE
TWQLP2: TLNN P3,-1 ;WAS ANY FILE FOUND TO START DATA?
PJRST SETBSY ;YES. GO
JUMPN F,TWQLUP ;NO. LOOK AGAIN IF THERE ARE MORE FILES
POP P,S ;NOTHING TO DO. RETORE S
PJRST CHNIDL ;SET CHAN IDLE
;STILL IN FTDOPT CONDITIONAL
;HERE WHEN BEST FILE IS FOUND START IO
SETBSY: POP P,S ;RESTORE S
MOVE F,P4 ;SET F TO FILE
> ;END CONDITIONAL ON FTDOPT
SKIPE U,DEVUNI##(F) ;SET U TO UNIT DB ADR. IT IS ACCESSING
IFN FTDUAL,<
HRRZ U,DEVCUR##(F)
SKIPE U
>
HRRZ J,UNIKON##(U) ;SET J TO KONTROLLER DB UNIT IS ON
DSKOFF ;#TURN ALL DISK PI'S OFF
PUSHJ P,UNQUEX ;#REMOVE FILE FROM TWQ
JUMPE U,UNIDWN ;JUST IN CASE F/S HAS BEEN JERKED OUT
IFN FTDSIM,<
SETBS1: MOVE T1,DEVUWZ##(F) ;ZEROING BLOCKS IN USETO CODE?
TLNN T1,DEPUWZ##
JRST SETBS2
MOVE T1,DEVACC##(F) ;YES, IS CURRENT BLOCK WRITTEN?
MOVE T1,ACCWRT##(T1) ; (IF USETO DECIDED TO ZERO BLOCK N WHILE
CAML T1,DEVREL##(F) ; ANOTHER JOB WAS IN DIOW FOR BLOCK N)
JRST NXTIO ;BLOCK NOW HAS GOOD DATA IN IT - GET OUT
SETBS2:>
PUSHJ P,STRTIO ;#SET FILE+UNIT TO T1, KONTROL TO B(CHAN ALREADY B)
;COMPUTE CHAN COMMAND LIST AND START TRANSFER
JRST INTXIT ;AND DISMISS THE INTERRUPT
;HERE FROM BADUNI ON UUO LEVEL
CHNIDX: MOVSI T1,400000
ANDCAM T1,KONTAB##(J)
;HERE WHEN ALL PROCESSING FOR DATA INTERRUPT IS DONE
;RESET FAIRNESS COUNTS IF THEY ARE NEGATIVE AND DISMISS
;P1 IS STILL CHAN. DB ADR. WHICH CAUSED THIS DATA INTERRUPT
CHNIDL: SETOB T2,@KONCHN##(J) ;#SET CHAN IDLE
DSKON ;#
IFN FTKI10!FTKL10,<
JUMPE P1,CPOPJ## ;DONE IF SETIDL CALLED BY
; BADUNI AT UUO LEVEL
>
IFN FTKL10,<
MOVSI T1,CP.SWF## ;GET SWEEP FLAG
TDNE T1,CHB22B##(P1) ;DO WE HAVE TO DO A SWEEP BEFORE DISMISSING?
PUSHJ P,CFDMP ;YES, SO DO IT NOW SINCE NO I/O WILL
; BE STARTED, AS WE ARE AT CHNIDL
>;END IFN FTKL10
PUSH P,U ;SAVE U FOR END TEST
CHNID1: MOVE T1,UNISTS##(U) ;STATUS OF UNIT
IFN FTDHIA,<
CAIE T1,TCOD## ;(PI'S ARE ON NOW)
CAIN T1,PCOD## ;IF ANY UNIT IS STILL DOING SOMETHING,
AOJA T2,CHNID2 ; BUMP T2
CAIN T1,SCOD##
AOJA T2,CHNID2
>
IFN FTVM,<
SKIPGE SCNCNT ;UNITS ARE IN FUNNY STATE IF
; IN SWPSCN ON UUO LEVEL
>
CAIE T1,TWCOD## ;IS UNIT IN TW?
JRST CHNID2 ;NO
IFN FTDUAL,<
SKIPE T1,UNI2ND##(U) ;IF AN ALTERNATE UNIT,
SKIPN UNISTS##(T1) ; WHICH ISNT IDLE
CAIA
JRST CHNID2 ;IT'S OK
>
SKIPE T1,UNIQUE##(U) ;YES (UNKNOWN BUG)
MOVEI T1,PWCOD## ;PUT UNIT INTO PW OR IDLE
MOVEM T1,UNISTS##(U) ; SO SYSTEM WONT HANG
AOS UNIHNG##(U) ;INCREMENT SOFTWARE-HUNG COUNT
CHNID2: HLR U,UNICHN##(U) ;STEP TO NEXT UNIT ON CHAN
CAME U,(P) ;BACK WHERE WE STARTED?
JRST CHNID1 ;NO, TEST THIS UNIT
POP P,U ;YES, REMOVE JUNK FROM PD LIST
IFN FTDHIA,<
SKIPLE F,DIADSK## ;TRYING TO SHUT DOWN IO?
CAME P1,DIACHN## ;YES, FOR THIS CHAN?
JRST CHNID4 ;NO
JUMPGE T2,INTXIT ;YES, DISMISS IF ANY UNIT BUSY
CHNID3: MOVE S,DEVIOS(F) ;NO UNITS BUSY, SET UP S
PUSHJ P,STDIOD## ;AND WAKE UP THE JOB (TO START DIAGNOSTIC)
PUSHJ P,CLRACT##
IFN FTDUAL,<
JRST INTXIT
CHNID4: JUMPLE F,INTXIT ;GO IF NO DIAG WAITING
MOVE U,KONCUA##(J) ;UNIT WHICH INTERRUPTS
SKIPL T1,UNI2ND##(U) ;DIAG - DUAL PORTED DRIVE ?
JRST INTXIT
HRRZ T2,UNICHN##(T1) ;YES, IS 2ND PORT ON RIGHT CHAN ?
SKIPGE (T2) ; AND IS THAT CHAN IDLE ?
CAME T2,DIACHN##
JRST INTXIT ;NO, CAN'T START THE DIAG JOB
HRRZ T2,T1
MOVEI T3,PCOD## ;YES, ARE ALL UNITS ON THE CHAN IDLE ?
MOVEI T4,SCOD##
CHNID5: CAME T3,UNISTS##(T1)
CAMN T4,UNISTS##(T1)
JRST INTXIT ;NO, CAN'T START THE DIAG JOB
HLRZ T1,UNICHN##(T1)
CAME T1,T2
JRST CHNID5
JRST CHNID3 ;YES, TAKE DIAG JOB OUT OF DIOW QUEUE
>
IFE FTDUAL,<
CHNID4:
>
>
INTXIT:
IFN FTKL10,<
MOVSI T1,CP.SWF## ;CLEAR SWEEP FLAG
ANDCAM T1,CHB22B##(P1) ;FOR NEXT INTERRUPT
>;END IFN FTKL10
HRRZ T1,CHNIFP##(P1) ;RESET FAIRNESS COUNTS IF THEY HAVE GONE NEGATIVE
SKIPG CHNCFP##(P1) ;CURRENT FAIRNESS COUNT FOR POSITIONING OPTIMIZATION
MOVEM T1,CHNCFP##(P1)
HRRZ T1,CHNIFT##(P1)
SKIPG CHNCFT##(P1) ;CURRENT FAIRNESS COUNT FOR TRANSFER OPTIMIZATION
MOVEM T1,CHNCFT##(P1)
POPJ P, ;AND DISMISS THE INTERRUPT
IFN FTKL10,<
;SUBROUTINE TO SWEEP CACHE
;CALL WITH P1=LOC OF CHANNEL DATA BLOCK
;PRESERVES ALL ACS EXCEPT T1
CFDMP:
IFN FTKL10,<
MOVE T1,CHB22B##(P1)
TLNN T1,CP.RH2## ;IS IT AN RH20?
>
SKIPE DINITF## ; OR ARE WE IN ONCE-ONLY
POPJ P, ;YES, NO NEED TO SWEEP
JRST CSDMP## ;NO, SWEEP CACHE
>
UNIDWN: PUSHJ P,UNIYNK
PJRST PIKTR1 ;START NEXT I/O IF CHAN NOT IDLE
; IF IDLE, EXIT THE INTERRUPT
UNIYNK:
IFN FTDBAD,<
PUSHJ P,BADUN6 ;FINISH THIS OPERATION, START NEXT
>
MOVE U,DEVFUN##(F) ;UNIT ADDR IS IN DEVFUN IF DEVUNI=0
SKIPE T1,UNIQUE##(U) ;MORE POSITIONS WAITING?
MOVEI T1,PWCOD## ;YES. RETURN UNIT TO PW STATE
MOVEM T1,UNISTS##(U) ;SET UNIT IDLE OR PW
POPJ P,
IFN FTDSIM,<
;HERE TO EXIT WITHOUT WRITING THE DISK
NXTIO: MOVE S,DEVIOS(F) ;SET UP S FOR STDIOD
PUSHJ P,STDIOD## ;WAKE UP JOB
PUSHJ P,CLRACT## ;NO LONGER IO ACTIVE
DSKOFF
CONSO PI,PIPROG## ;#IF ON UUO LEVEL,
JRST [SETOM @KONCHN##(J) ;#INDICATE CHAN IS IDLE
DSKON ;#TURN ON DSK PI
POPJ P,] ;#AND RETURN WITHOUT WIPING IT
IFN FTDUAL,<
SKIPGE T1,UNI2ND##(U) ;2ND PORT?
HRR U,T1 ;YES, GO BACK TO MAIN
>
SKIPE T1,UNIQUE##(U) ;UNIT IS IDLE OR PW
MOVEI T1,PWCOD##
MOVEM T1,UNISTS##(U)
JUMPE T1,PIKTR1 ;LOOK FOR A TRANSFER TO START IF NOT PW
SETOM UNICYL##(U) ;ENSURE A SEEK HAPPENS (ELSE COULD PDL OV)
JRST PIKPOS ; AND START A SEEK GOING
>
FILEND: END