Trailing-Edge
-
PDP-10 Archives
-
BB-J713A-BM
-
language-sources/glxpfh.mac
There are 7 other files named glxpfh.mac in the archive. Click here to see a list.
TITLE GLXPFH - Page fault handler for the GALAXY library on the DECsystem-10
SUBTTL Dave Cornelius 16-Aug-79
;
;
; COPYRIGHT (c) 1979
; DIGITAL EQUIPMENT CORPORATION
;
; THIS SOFTWARE IS FURNISHED UNDER A LICENSE AND MAY BE USED
; AND COPIED ONLY IN ACCORDANCE WITH THE TERMS OF SUCH LICENSE
; AND WITH THE INCLUSION OF THE ABOVE COPYRIGHT NOTICE. THIS
; SOFTWARE OR ANY OTHER COPIES THEREOF MAY NOT BE PROVIDED OR
; OTHERWISE MADE AVAILABLE TO ANY OTHER PERSON. NO TITLE TO
; AND OWNERSHIP OF THE SOFTWARE IS HEREBY TRANSFERRED.
;
; THE INFORMATION IN THIS SOFTWARE IS SUBJECT TO CHANGE
; WITHOUT NOTICE AND SHOULD NOT BE CONSTRUED AS A COMMITMENT
; BY DIGITAL EQUIPMENT CORPORATION.
;
; DIGITAL ASSUMES NO RESPONSIBILITY FOR THE USE OR RELIABILITY
; OF ITS SOFTWARE ON EQUIPMENT WHICH IS NOT SUPPLIED BY
; DIGITAL.
;This module is loaded in the low segment of any program
; using the GALAXY library code, GLXLIB.
SEARCH GLXMAC ;SEARCH GLXMAC FOR SYMBOL DEFS
PROLOG(GLXPFH,PFH)
PFHEDT==2 ;EDIT LEVEL FOR MODULE
SUBTTL Table of Contents
SUBTTL Revision History
COMMENT \
Revision History
EDIT GCO DESCRIPTION
1 Create GLXPFH from the remains of GLXINI
2 Don't throw out symbols until first timer trap
And when we do throw 'em out, do it in one or two UUOs,
not one PAGE. for each page thrown out
\ ;end of revision history
SUBTTL Fault handler data space
TOPS10 < ;THIS CODE EXISTS ONLY FOR THE TOPS10 MONITOR
COMMENT \
The fault handler will live in the realm of the library.
The code which handles the faults and determines what to do
is sharable, and part of the library high segment.
Since the monitor check to see if we are executing in the fault
handler is a range check of our PC agains the limits in .JBPFH,
the code should be as low as possible in the high segment.
We load it right after the entry vector.
Included as part of the fault handler (ie inside the high and low
bounds of the monitor's range check) is 10 words of
dirty space. Since the library is sharable, these 10 words
can't live in the high segment.
We put them in a page which is just below the library high segment.
Unfortunately, LINK doesn't support relocatable PSECTs, so
alot of this 'page manipulation' magic is done at link time.
The PSECTs whose origins must be defined (at LINK time) are:
If the library is impure (ie it will live in the lowseg), then
all of this PSECT nonsense is unneccessary, and not used.
Name Location Contents
.HIGH. 400000 The library sharable, reentrant code.
PFHDAT 377000 Fault handler data page (mostly blank)
DATA 600000 Local storage for library. Library symbol table
\
IFN GLXPURE,< ;Only if we're going to be sharable...
.PSECT PFHDAT ;Into dirty PFH data space
PFHEST==32 ;Estimated size of fault handler data
;If this is wrong, assembly won't go.
;If it is too high, we'll return the
;space anyway.
BLOCK PAGSIZ-PFHEST ;Get to near top of page
>;End IFN GLXPURE
PFHDST: ;Start of the data spce
PFHBGN: JRST PFHSTA ;ENTERED HERE
PFH.PC: 0 ;PC AT TIME OF TRAP
PFH.PN: 0 ;PAGE FAULT WORD
PFH.TI: 0 ;VIRTUAL AGE OF THE PAGE FAULT HANDLER
PFH.RA: 0 ;CURRENT PAGING RATE
PFH.PV: 0 ;Highest PSI,,Adr of PSI vec
BLOCK 3 ;ROOM FOR FUTURE EXPANSION
PFHS.P: BLOCK 1 ;SAVED P
PF.PDL: BLOCK ^D12 ;PDL FOR USE IN THE PAGE FAULT HANDLER
PF.PSZ==.-PF.PDL ;SIZE OF THE LIST
NICUCN: BLOCK 1
NICCNT: BLOCK 1
GVACNT: BLOCK 1
TIMCNT: BLOCK 1
IFN GLXPURE,< ;... Only for high segments
PFHDSZ==.-PFHDST ;Size of the data space
PFHDEN: ;End of PFH data space
IFG <PFHDSZ-PFHEST>,<PRINTX ?PFH DATA SPACE TOO BIG, INCREASE PFHEST DEFINITION>
.ENDPS ;Back to the library
>;End IFN GLXPURE
>;End TOPS10
SUBTTL The Page Fault Handler (TOPS-10 Only)
;THIS IS THE ROUTINE CALLED BY THE TOPS10 SYSTEM SO THAT A USER MAY PROCESS
;HIS OWN PAGE FAULTS. .JBPFH CONTAINS THE STARTING ADDRESS OF THE ROUTINE
;AND IS CALLED WHEN EITHER A PAGE FAULT OCCURS OR THE .STTVM TIMER HAS EXPIRED
TOPS10 <
PFHSTA:
MOVEM P,PFHS.P ;SAVE SOMEBODY'S P
;*** Temporary
MOVEM P,PFHACS+P
MOVEI P,PFHACS
BLT P,PFHACS+P-1
;*** End Temporary
MOVE P,[IOWD PF.PSZ,PF.PDL] ;GET MY OWN PDL
PUSH P,P1 ;BEGIN TO SAVE SEVERAL REGISTERS
PUSH P,P2 ;CAUSE I'M GOING TO CALL SUBROUTINES
PUSH P,P3 ;FOR SWAPPING AND SEARCHING
PUSH P,P4 ;AND THESE ARE THE ARGUMENTS
LOAD P1,PFH.PC,RHMASK ;Get addr of fault
CAIL P1,PFHDST ;Skip if before fault handler
CAILE P1,PFHEND ;or was fault off end of handler
SKIPA ;Fault outside of handle
HALT . ;Oops! We're very confused
LOAD P2,PFH.PN,PF.HFC ;THE FAULT TYPE
CAIG P2,PFHDCT ;RANGE CHECK
JRST PFHDSP(P2) ;DISPATCH THE FAULT
PFHDSP: JRST PFHILL ;PAGE FAULT CODE = 0 OR TOO BIG
JRST PFHGVA ;.PFHNA = 1 = NO ACCESS
JRST PFHSWI ;.PFHNI = 2 = NOT IN CORE
JRST PFHSWU ;.PFHUU = 3 = NOT IN CORE FOR UUO
JRST PFHTIM ;.PFHTI = 4 = VM TIMER TRAP
PFHDCT==.-PFHDSP-1 ;DISPATCH LENGTH (0-n)
PFHILL: $STOP(UPF,Unknown page fault type)
PFHSWU: AOSA NICUCN ;Count 'not in core for UUO' faults
PFHSWI: AOS NICCNT ;Count 'not in core' faults
LOAD P3,PFH.PN,PF.HPN ;GET THE PAGE THAT CAUSED THE FAULT
MOVEI P2,1 ;ONLY 1 PAGE
MOVE P1,[.PAGIO,,P2] ;FUNCTION PAGE IO,,ARGUMENT BLOCK
PAGE. P1, ;SWAP IT IN
JRST PFHSWF ;FAILED, SEE WHY
MOVX P1,PT.WRK ;THE WORKING SET BIT
IORM P1,PAGTBL##(P3) ;PAGE IS NOW IN CORE
JRST PFHXIT ;DISMISS THE FAULT
PFHSWF: CAIE P1,PAGLE% ;IS MY CORE LIMIT EXCEEDED
$STOP(PIF,Page in failure)
PUSHJ P,ANYOUT ;SWAP OUT SOMETHING (ANYTHING)
JRST PFHSWI ;TRY THE SWAPIN NOW
;;;CODE IS CONTINUED ON THE NEXT PAGE AND STILL UNDER TOPS10 CONDITIONAL
;HERE TO GIVE ACCESS TO A PAGE (CAN ONLY HAPPEN IF LOADED VIA GET.SHR)
PFHGVA: AOS GVACNT
LOAD P3,PFH.PN,PF.HPN ;GET THE PAGE NUMBER WE CAN'T ACCESS
TXO P3,1B0 ;SET TO ALLOW ACCESS
MOVEI P2,1 ;ONLY ONE ARG
MOVE P1,[.PAGAA,,P2] ;GIVE ACCESS,,ARGS
PAGE. P1, ;SET ACCESS ALLOWED ON THAT PAGE
$STOP(CGA,Cannot give access to page)
JRST PFHXIT ;AND EXIT FROM THE PAGE FAULT HANDLER
;HERE ON A TIMER TRAP, SEE IF WE CAN CLEAN UP SOME CORE
PFHTIM: AOS TIMCNT
SKIPE INMEMF## ;SOMEBODY FOOLING WITH THE PAGE TABLE?
JRST PFHXIT ;Yes, don't screw 'em up
PUSHJ P,PGSDES ;NO, DESTROY ANY JUNK PAGES
PUSHJ P,PGSSWP ;And dump out any swappable pages
PFHXIT: POP P,P4 ;RESTORE ACS
POP P,P3 ;...
POP P,P2 ;...
POP P,P1 ;...
MOVE P,PFHS.P ;RESTORE THE OLD AC P
;*** Temporary
$DATA (PFHACS,20)
PFH..X==0
REPEAT 20,<
CAME PFH..X,PFHACS+PFH..X
PUSHJ P,PFHDIE
PFH..X==PFH..X+1
>
;*** End Temporary
JRSTF @PFH.PC ;DISMISS THE FAULT
;*** Temporary
PFHDIE: OUTSTR [ASCIZ/
?GLXPFH is confused. Please save this core image
/]
$STOP (PFH,GLXPFH IS CONFUSED)
EXIT 1,
POPJ P,
;*** End Temporary
SUBTTL PGSDES - Routine to destroy unused pages
;CALL IS: NO ARGUMENTS
;
;TRUE RETURN: ALWAYS
PGSDES: MOVE P4,PAGSTA## ;THE FIRST AVAILABLE PAGE
PGSD.1: CAIL P4,MEMSIZ ;OFF THE END OF THE WORLD
POPJ P, ;YES, RETURN
MOVE P1,PAGTBL##(P4) ;GET TABLE ENTRY FOR PAGE IN P4
TXC P1,PT.ADR ;NEED CHECK FOR BOTH SO FLIP
TXNN P1,PT.USE!PT.ADR ;USED OR NOT ADDRESSABEL
PUSHJ P,DESTRY ;DESTROY THE PAGE (COULD BE PAGED OUT)
AOJA P4,PGSD.1 ;AND CONTINUE LOOPING
DESTRY: MOVEI P3,(P4) ;WANT IT IN P3
TXO P3,1B0 ;SET TO DESTROY
MOVEI P2,1 ;1 ARGUMENT
MOVE P1,[.PAGCD,,P2] ;CREATE/DESTROY,,ARGUMENT
PAGE. P1, ;TRY TO DESTROY IT
$STOP(PDF,Page destroy failed)
ZERO P1 ;CLEAR A REG
EXCH P1,PAGTBL##(P4) ;CLEAR PAGE TABLE ENTRY, GET OLD
TXNE P1,PT.WRK ;JUST DESTROY AN IN-CORE PAGE
SOS AVBPGS## ;YES, REDUCE COUNTER ACCORDINGLY
POPJ P, ;RETURN
SUBTTL PGSSWP - Swap out swappable pages
;PGSSWP - This routine will try to page out any pages
; which are marked as swappable in PAGTAB but have managed to
; crawl back into the working set. This can happen for instance,
; in the case of symbols thrown out during initialization
; but faulted back in at a breakpoint by a DDT search of the symbol
; table.
PGSSWP:
PUSHJ P,INDDT ;Fault from DDT itself?
POPJ P, ;Yes, don't throw out symbols
SETZB P4,SWPTBL ;Start at page 0, With none thrown out
PGSS.1: CAIL P4,MEMSIZ ;Off the end?
PJRST PGSS.2 ;Yes, dump out any we found, and return
MOVE P1,PAGTBL(P4) ;Get entry for this page
TXNE P1,PT.WRK ;Is this one out already?
TXNN P1,PT.SWP ;No, is it swappable?
AOJA P4,PGSS.1 ;Swapped, or not a candidate, skip it
;Found a page to throw out, add it to the table
AOS P3,SWPTBL ;Get next slot number
MOVEM P4,SWPTBL(P3) ;Save page number to go out
MOVX P1,1B0 ;Set to page OUT, not in!
IORM P1,SWPTBL(P3) ;Light in arg word for UUO
MOVX P1,PT.WRK ;Get the working set bit
ANDCAM P1,PAGTBL(P4) ;This page is out! (hopefully)
CAIL P3,SWPTLN ;Just filled up table?
PUSHJ P,PGSS.2 ;Yes, throw out what we have
AOJA P4,PGSS.1 ;Then try the next one
;A routine to page out whatever has been found as swappable
PGSS.2: SKIPN SWPTBL ;Anything to do?
POPJ P, ;No, just return
MOVX P1,<XWD .PAGIO,SWPTBL> ;Aim at the block we've built
PAGE. P1, ;Dump these pages out!
CAXN P1,PAGMI% ;Lost because we're confused?
SKIPA ;Wins, keep going
$STOP (SPF,<Page out for symbols failed, error code ^O/P1/>)
SETZM SWPTBL ;Clear UUO counter
POPJ P, ;Return
;Data space for the symbol pager
SWPTLN==^D25 ;At most (usually) 25 pages of symbols
$DATA (SWPTBL,SWPTLN+1) ;UUO block, counted
SUBTTL INDDT- Routine to determine if fault occurred inside DDT
;Returns .+1 if fault happened in DDT
;Returns .+2 if fault was outside DDT
INDDT: SKIPN P1,.JBDDT ;Get end,, start of DDT
JRST INDSKP ;No DDT, skip return
HLRZ P2,P1 ;Isolate ending adrs
HRRZS P1 ;Isolate starting adrs
LOAD P3,PFH.PC,RHMASK ;Get fault PC
CAIL P3,(P1) ;Below start of DDT
CAILE P3,(P2) ;Or above end
INDSKP: AOS (P) ;Not in DDT
POPJ P, ;Give skip return
SUBTTL ANYOUT - Routine to swap out anything possible
; CALL IS: NO ARGUMENTS
;
ANYOUT: MOVE P4,PAGSTA## ;FIRST MAKE A PASS LOOKING FOR ANY FREE
ANYO.1: PUSHJ P,ANYO.4 ;PAGES THAT CAN BE DESTROYED
JUMPE P4,ANYO.2 ;NONE, PAGE SOMETHING OUT
TXNE P1,PT.USE ;IS THIS IN-CORE PAGE USED
AOJA P4,ANYO.1 ;YES, LOOK FOR ANOTHER
PJRST DESTRY ;NO, DESTROY IT NOW AN SAVE A SWAP
ANYO.2: AOS P4,SWPSTA## ;START WHERE WE LEFT OFF
PUSHJ P,ANYO.4 ;FIND A SWAPPED IN PAGE
JUMPN P4,ANYO.3 ;SWAP IT OUT
MOVE P4,PAGSTA## ;RE-START AT THE FIRST
PUSHJ P,ANYO.4 ;TRY NOW
JUMPN P4,ANYO.3 ;FOUND ONE, SWAP IT OUT
$STOP(NTS,Nothing to swap out)
ANYO.3: MOVEM P4,SWPSTA## ;REMEMBER FOR LATER
MOVE P3,P4 ;COPY THE PAGE NUMBER
TXO P3,1B0 ;SET TO SWAP OUT INSTEAD OF SWAP IN
MOVEI P2,1 ;ONLY 1 PAGE FOR NOW
MOVE P1,[.PAGIO,,P2] ;FUNCTION SWAP,,ARGUMENT BLOCK
PAGE. P1, ;OUT IT GOES
$STOP(POF,Page out failure)
MOVX P1,PT.WRK ;GET THE WORKING SET BIT
ANDCAM P1,PAGTBL##(P3) ;NOT THERE ANY MORE
POPJ P, ;AND RETURN FOR ANY RETRIES
ANYO.4: CAIL P4,MEMSIZ ;OVER THE TOP
JRST ANYO.5 ;YES, RETURN A ZERO
MOVE P1,PAGTBL##(P4) ;GET THE STATUS FLAGS FOR PAGE IN P4
TXNE P1,PT.ADR ;IS IT ADDRESSABLE
TXNN P1,PT.WRK ;YES, IS IT IN THE WORKING SET
AOJA P4,ANYO.4 ;TRY THE NEXT PAGE
POPJ P, ;YES, A CANDIDATE FOR SWAP OUT
ANYO.5: ZERO P4 ;RETURN 0 IF NONE FOUND
POPJ P, ;RETURN
PFHEND: ;THE LAST LOCATION WITHIN THE PAGE FAULT HANDLER
> ;END TOPS10 CONDITIONAL
SUBTTL SETPFH - Set up private page fault handler
;This is a no-op for TOPS-20 systems, for TOPS-10, we install
; our PFH.
;CALL IS:
; M%INIT MUST be called before coming into SETPFH, since
; we depend on PAGSTA being setup
; No arguments
;
;TRUE RETURN: Always
SETPFH::
TOPS10 <
$SAVE <P1,P2>
SETZM INMEMF## ;CLEAR FLAG
MOVE S1,[PFHEND,,PFHBGN] ;GET START AND END OF PFH
MOVES PFHBGN ;Get it in core
MOVEM S1,.JBPFH## ;STORE LOCATION OF OUR PFH
PUSHJ P,MAPCOR ;Read the state of the working set
MOVE S1,[.STTVM,,^D1000] ;SET VIRTUAL TIMER TRAP (1 SEC)
SETUUO S1, ;
JFCL ;IGNORE ANY FAILURES
MOVE S1,PAGSTA## ;GET START OF FREE SPACE
SUBI S1,1 ;BACK OFF BY ONE
MOVEM S1,SWPSTA## ;AND STORE IT FOR LATER USE
IFN GLXPURE,<
MOVEI S1,PFHDAT/PAGSIZ ;Page # of data page
MOVX S2,PT.INI ;Page in initial core image bit
ANDCAM S2,PAGTBL(S1) ;Allow returns of pieces of this page
MOVEI S1,PFHDST-PFHDAT ;Size of space at start of page
MOVEI S2,PFHDAT ;Get addr of dirty page
PUSHJ P,RETMEM ;Give back that wasted space
IFN PFHEST-PFHDSZ,<
MOVEI S1,PFHEST-PFHDSZ ;Size of that slot
MOVEI S2,PFHDEN ;Addr of small chunk above fault handler
PUSHJ P,RETMEM ;Give that back too
>
SKIPN S1,.HIGH.+.JBHSM ;Load high seg sym pointer
JRST SETP.1 ;No high seg syms, try low seg
PUSHJ P,PAGSYM ;Page out the high seg symbols
HLRE S1,.HIGH.+.JBHSM ;Get neg length of high syms
HRRZ S2,.HIGH.+.JBHSM ;And get start adr
SUB S2,S1 ;Compute the adr of first free
TXNN S2,PAGSIZ-1 ;Symbols fit exactly on a page?
JRST SETP.1 ;Just fit, don't return anything
LOAD S1,S2,PAGSIZ-1 ;Get number used in this page
SUBI S1,PAGSIZ ;Make neg number free
MOVNS S1 ;And now positive
PUSHJ P,RETMEM ;Give back that space
SETP.1:
>;End IFN GLXPURE
SKIPE S1,.JBSYM ;Load low seg symbol pointer
PUSHJ P,PAGSYM ;Page out the low seg symbols
> ;END TOPS10 CONDITIONAL
$RETT ;All done
TOPS10<
;RETMEM is used to return pieces of memory which may have been part of
; the initial core image
;Call with S1 - length of block
; S2 - adr of block
;Returns true, memory returned, page marked as not in initial allocation
RETMEM:
PUSH P,S2 ;Save adr of block
ADR2PG S2 ;Get page number
MOVX TF,PT.INI ;Get 'in initial core image' bit
ANDCAM TF,PAGTBL(S2) ;Clear, so we can RMEM
POP P,S2 ;Get back start of block
PJRST M%RMEM ;Give back the space
>;END TOPS10
TOPS10<
;PAGSYM takes a symbol table pointer in S1 <-len,,adr> and marks
; all pages which completely reside in the symbol table
; as swappable on a timer trap
PAGSYM:
$SAVE <P1,P2>
HRRZ P2,S1 ;Copy adr of block
HLRE P1,S1 ;Make negative length of table
MOVNS P1 ;And now length of table
ADD P1,P2 ;Figure first free after syms
ANDI P2,-PAGSIZ ;Get page number of first page of syms
PAGS.1: ADDI P2,PAGSIZ ;Bump to next page
CAILE P2,(P1) ;Off end of symbols?
$RETT ;Yes, stop the page out war
MOVE S1,P2 ;Get addr of page to move out
ADR2PG S1 ;Make it a page number
MOVX S2,PT.SWP ;Get 'swappable anytime' bit
IORM S2,PAGTBL(S1) ;Mark it as such
JRST PAGS.1 ;Try the next page
> ;END TOPS10 CONDITIONAL
SUBTTL MAPCOR - Read the page map bits from the monitor
;This routine will ask the monitor for the state of the working
; set and will set or clear the PT.WRK bits in PAGTBL
TOPS10<
MAPCOR:
$SAVE <P1,P2>
MOVX S1,MAPLEN ;Length of table
MOVEM S1,BITMAP ;Count it for the monitor
MOVE S1,[XWD .PAGWS,BITMAP] ;Read working set into BITMAP
PAGE. S1, ;Ask the monitor
$STOP (CDW,<Can't determine working set error = ^O/S1/>)
MOVE S1,[POINT 1,BITMAP+1] ;Aim at the 512 bits
MOVSI S2,-^D512 ;Start at page 0
MOVX P1,PT.WRK ;Get our working set bit
MAPC.1: ILDB P2,S1 ;Get this page's bit
XCT [ANDCAM P1,PAGTBL(S2) ;Set or clear our bit
IORM P1,PAGTBL(S2)](P2) ;To match monitor's bit
AOBJN S2,MAPC.1 ;Do 'em all
$RETT
MAPLEN==<<^D512+^D35>/^D36>
$DATA (BITMAP,MAPLEN+1) ;Bit block, counted
>;END TOPS10
SUBTTL NODDT - Remove DDT.VMX routine for TOPS10
;This routine is intended to be used as an emergency tool
; in case a library gets saved with DDT in its high segment.
; In this case, any lowseg program which also has DDT.VMX in page 700
; will get page overlap errors when getting the library.
; It is not the intention that the library be saved in this state,
; however, if it happens, then this routine will be useful.
; The procedure is as follows:
; .GET GLXLIB ;The bad one, with DDT.VMX in it
; .DDT ;get into DDT.VMX
; NODDT$G ;Start it at this routine
; .SAVE ;Save the library without the symbols
TOPS10<
EXTERN .JBDDT
NODDT: SKIPN S1,.JBDDT ;Get the end,,start adr of DDT
JRST NODD.2 ;No DDT to begin with!
HRRZ T2,S1 ;Get start adr of DDT
HLRZ P1,S1 ;And ending adr
ADR2PG T2 ;Make starting page number
ADR2PG P1 ;Make ending page number
SUB P1,T2 ;Get number of pages in DDT
TXO T2,1B0 ;Light the destroy page bit
MOVEI T1,1 ;Count the args to PAGE. UUO
MOVE S2,[XWD .PAGCD,T1] ;Create/Destroy,, adr of arg list
NODD.1: PAGE. S2, ;Get rid of a page
JRST NODDFA ;Complain!
AOS T2 ;Bump to next page
SOJGE P1,NODD.1 ;And try the rest of the pages
SETZ S1, ;Store 0 in protected DDT loc
SETDDT S1, ;And tell the monitor
EXIT ;And go back to monitor
;Here is trying to destroy DDT, and it isn't here at all!
NODD.2: OUTSTR [ASCIZ/
% DDT not loaded
/] ;Inform the dumb user
EXIT ;And leave 'em alone
;Here if the PAGE. UUO fails
NODDFA: OUTSTR [ASCIZ/
?Can't destroy DDT page
/]
EXIT ;And die
>;End TOPS10
PFH%L:
END