-- Copyright (C) 1980, 1981, 1983, 1984 by Xerox Corporation. All rights reserved. -- File: MDSWatcher.mesa -- HGM, 16-Dec-84 8:58:23 -- Ted Wobber 8-Aug-83 11:11:17 -- Randy Gobbel 7-Aug-81 16:11:44 -- Roy Levin 10-Sep-80 18:46:53 -- Andrew Birrell 16-Oct-80 18:30:01 -- Hankins 23-Aug-84 10:38:03 DIRECTORY Ascii USING [NUL], BufferOps USING [GetStatistics, Statistics], EnquiryDefs USING [DisplayStats, Histograms, RemoteServers, PolicyControls], GlassDefs USING [Handle, HandleObject, StringType], LogDefs USING [DisplayNumber], LogPrivateDefs USING [tty], Process USING [ Detach, InvalidProcess, Pause, Priority, SecondsToTicks, SetPriority, Ticks, ValidateProcess], ProcessPriorities USING [priorityFrameFault], ProcessOperations USING [ DisableInterrupts, EnableInterrupts, Notify, ReEnter, Wait], PSB USING [ Condition, FaultIndex, PDABase, PDA, PsbIndex, PsbNull, qPageFault, StartPsb], SpecialSpace USING [MakeProcedureResident], RestartDefs, TTY USING [GetChar, PutChar, PutDecimal, PutLongDecimal, PutString]; MDSWatcher: PROGRAM IMPORTS BufferOps, EnquiryDefs, LogDefs, LogPrivateDefs, Process, ProcessOperations, SpecialSpace, TTY EXPORTS RestartDefs = BEGIN screen: GlassDefs.Handle = @screenObject; screenObject: GlassDefs.HandleObject ← [ ReadChar: ReadCharNull, ReadString: ReadStringNull, WriteChar: MyWriteChar, WriteString: MyWriteString, WriteDecimal: MyWriteDecimal, WriteLongDecimal: MyWriteLongDecimal, SendNow: NullProc, CharsLeft: CardNullProc, LinesLeft: CardNullProc, SetWidth: SetWidthNull, SetHeight: SetHeightNull, DelTyped: BoolNullProc, Synch: NullProc, Flush: NullProc]; availablePages, availablePSBs, availablePups: CARDINAL ← 0; savedStorageFaults, storageFaults: CARDINAL ← 0; oneMinute: Process.Ticks = Process.SecondsToTicks[60]; tenSeconds: Process.Ticks = Process.SecondsToTicks[10]; StartKeyWatcher: PUBLIC PROCEDURE = BEGIN Process.Detach[FORK KeyboardWatcher[]]; END; Init: PROCEDURE = BEGIN -- LogDefs.DisplayNumber["Free MDS"L, [short[@availablePages]]]; LogDefs.DisplayNumber["Free PSBs"L, [short[@availablePSBs]]]; LogDefs.DisplayNumber["Free Pup Buffers"L, [short[@availablePups]]]; LogDefs.DisplayNumber["Page faults/min"L, [short[@savedStorageFaults]]]; END; WatchMDS: PROCEDURE = BEGIN << this is looking at MDS and seeing how much has been carved up for new, smaller, spaces. this is no longer possible to do in any clean way (could import global from of some Pilot impl and look at a variable therein (SpecialSpaceImpl?) so we'll just not calculate it any longer. firstPage: Space.PageNumber = Space.PageFromLongPointer[Space.MDS[].pointer]; lastPage: Space.PageNumber = firstPage + Environment.maxPagesInMDS; should also be equivalent to firstPage+Space.MDS[].count? >> DO available: CARDINAL ← 0; << FOR page: Space.PageNumber ← firstPage, SUCC[page] UNTIL page > lastPage DO IF Space.GetHandle[page] = Space.mds THEN available ← available + 1; ENDLOOP; availablePages ← available; available ← 0; >> FOR i: CARDINAL IN [PSB.StartPsb..PSB.StartPsb + PSB.PDA.count) DO Process.ValidateProcess[ LOOPHOLE[i, PROCESS] ! Process.InvalidProcess => {available ← available + 1; CONTINUE}]; ENDLOOP; availablePSBs ← available; savedStorageFaults ← storageFaults; storageFaults ← 0; Process.Pause[oneMinute]; ENDLOOP; END; WatchPupBuffers: PROCEDURE = BEGIN stats: ARRAY BufferOps.Statistics OF CARDINAL; DO stats ← BufferOps.GetStatistics[]; availablePups ← stats[available]; Process.Pause[tenSeconds]; ENDLOOP END; KeyboardWatcher: PROCEDURE = BEGIN DO c: CHAR ← TTY.GetChar[LogPrivateDefs.tty]; SELECT c FROM 'h, 'H => EnquiryDefs.Histograms[screen]; 'o, 'O => EnquiryDefs.RemoteServers[screen]; 'p, 'P => EnquiryDefs.PolicyControls[screen]; ENDCASE => EnquiryDefs.DisplayStats[screen]; ENDLOOP; END; -- Stuff stolen from BLyon's UsageMonitor tool priorityStorageFaultRecorders: Process.Priority ← ProcessPriorities.priorityFrameFault - 1; -- should be higher than Pilot's page and write fault handlers' priorities. pda: PSB.PDABase = PSB.PDA; qPageFault: PSB.FaultIndex = PSB.qPageFault; pPageFaultCondition: LONG POINTER TO PSB.Condition = @pda.fault[ qPageFault].condition; pPageFaultCONDITION: LONG POINTER TO CONDITION = LOOPHOLE[pPageFaultCondition]; PageFaultRecorder: --EXTERNAL-- PROCEDURE [] = BEGIN pageFaultLock: MONITORLOCK; shouldNotifyPilot: BOOLEAN ← FALSE; Process.SetPriority[priorityStorageFaultRecorders]; -- NO PAGE FAULTS or calls to swappable code from this point on! (else the state vector goblin will get ya) --FOREVER-- DO -- wait for a page fault.. ProcessOperations.Wait[@pageFaultLock, pPageFaultCONDITION, --timeout:-- 1]; UNTIL ProcessOperations.ReEnter[@pageFaultLock, pPageFaultCONDITION] DO NULL ENDLOOP; -- either a new page fault came along or we timed out.. IF pda.fault[qPageFault].queue.tail = PSB.PsbNull THEN -- no page fault, just timed out.. BEGIN IF shouldNotifyPilot AND pda.fault[qPageFault].condition.tail ~= PSB.PsbNull THEN { NakedNotify[pPageFaultCONDITION]; shouldNotifyPilot ← FALSE}; LOOP; -- go back and wait again. END; -- wake up the Pilot fault handler: ProcessOperations.DisableInterrupts[]; IF pPageFaultCondition↑.tail = PSB.PsbNull THEN shouldNotifyPilot ← TRUE -- Pilot not ready for this fault yet.. ELSE NakedNotify[pPageFaultCONDITION]; ProcessOperations.EnableInterrupts[]; -- log the fault: storageFaults ← SUCC[storageFaults]; ENDLOOP; -- Note: following code never executed in current implementation. -- Must get back to ordinary priority before exiting since Process.End is swappable.. -- Process.SetPriority[Process.priorityForeground]; END; -- PageFaultRecorder (process disappears) NakedNotify: PROCEDURE [pCondition: LONG POINTER TO CONDITION] = INLINE { -- Used ONLY to notify a condition from a high priority process outside the relevant monitor. pCond: LONG POINTER TO PSB.Condition = LOOPHOLE[pCondition]; ProcessOperations.DisableInterrupts[]; IF pCond↑.tail = PSB.PsbNull THEN { pCond↑.wakeup ← TRUE; ProcessOperations.EnableInterrupts[]} ELSE { ProcessOperations.EnableInterrupts[]; ProcessOperations.Notify[pCondition]}}; -- GlassDefs.HandleOject implementors NullProc: PROCEDURE = {}; BoolNullProc: PROCEDURE RETURNS [BOOLEAN] = {RETURN[FALSE]}; CardNullProc: PROCEDURE RETURNS [CARDINAL] = {RETURN[0]}; ReadCharNull: PROCEDURE RETURNS [CHARACTER] = {RETURN[Ascii.NUL]}; ReadStringNull: PROC [prompt, s: STRING, type: GlassDefs.StringType] RETURNS [CHARACTER] = {RETURN[Ascii.NUL]}; SetWidthNull, SetHeightNull: PROCEDURE [CARDINAL] = {}; MyWriteChar: PROCEDURE [c: CHARACTER] = {TTY.PutChar[LogPrivateDefs.tty, c]}; MyWriteString: PROC [s: STRING] = {TTY.PutString[LogPrivateDefs.tty, s]}; MyWriteDecimal: PROCEDURE [n: CARDINAL] = { TTY.PutDecimal[LogPrivateDefs.tty, n]}; MyWriteLongDecimal: PROCEDURE [n: LONG CARDINAL] = { TTY.PutLongDecimal[LogPrivateDefs.tty, n]}; Init[]; Process.Detach[FORK WatchMDS[]]; Process.Detach[FORK WatchPupBuffers[]]; SpecialSpace.MakeProcedureResident[PageFaultRecorder]; Process.Detach[FORK WatchPupBuffers[]]; END.