CedarLoaderImpl.mesa
This module handles all of parts of loading that don't have anything to do with links:
copying the bcd into VM, creating global frames for each module,
associating code with each frame, and initializing the start trap cm's.
Linking is handled by LoaderOps.Export and LoaderOps.Bind.
Edited by Sandman on October 21, 1980 10:57 AM
Edited by Forrest on October 25, 1980 7:45 PM
Edited by John Maxwell on February 28, 1983 1:38 pm
Last Edited by: Birrell, August 24, 1983 5:13 pm
Last Edited by: Schroeder, August 3, 1983 8:56 am
Last Edited by: Levin, September 22, 1983 1:52 pm
Edited by Paul Rovner on August 17, 1983 4:34 pm
Edited by MBrown on September 11, 1983 12:52 pm
DIRECTORY
BasicTime USING [FromPupTime, GMT],
BcdDefs
USING [
Base, BcdBase, CTHandle, CTIndex, CTNull, FPHandle, FPIndex, FTHandle, FTIndex, FTSelf, MTHandle, MTIndex, NameRecord, NameString, PackedString, SGHandle, SGIndex, SpaceID, SPHandle, SPIndex, VersionID, VersionStamp, ModuleIndex
],
BcdOps USING [ProcessConfigs, ProcessSegs, ProcessModules, ProcessFramePacks, ProcessSpaces],
FS USING [GetInfo, OpenFile, Open, Read, Error],
Loader USING [ErrorType, IRItem],
LoaderOps USING [Bind, Export, LinkSegmentLength, FrameList],
LoadState
USING [
Acquire, ConfigID, ConfigInfo, GlobalFrameToModule, local, ModuleIndex, ModuleInfo, NewConfig, Release, SetGlobalFrame, SetType],
IO USING [STREAM, ROS, PutChar, RopeFromROS],
PrincOps
USING [
ControlModule, CSegPrefix, GlobalFrameHandle, LastAVSlot, MainBodyIndex, NullControl, wordsPerPage],
PrincOpsUtils
USING [
ZERO, BITAND, Codebase, FrameSize, GetReturnLink, GlobalFrame, MakeFsi, Alloc, LowHalf, Free],
Rope USING [ROPE, Find, Concat],
SafeStorage USING [Type],
SafeStorageOps USING [AcquireTypesAndLiterals],
SystemVersion USING [bootFileDate],
VM
USING [
AddressForPageNumber, Allocate, CantAllocate, Free, Interval, MakeReadOnly, PageNumber, PageNumberForAddress, PagesForWords, Pin, Unpin, WordsForPages];
CedarLoaderImpl: MONITOR
IMPORTS BasicTime, BcdOps, FS, IO, LoaderOps, LoadState, PrincOpsUtils, Rope, SafeStorageOps, SystemVersion, VM
EXPORTS Loader, LoaderOps =
BEGIN OPEN BcdDefs, BcdOps, Rope;
FrameList: TYPE = LoaderOps.FrameList;
GlobalFrameHandle: TYPE = PrincOps.GlobalFrameHandle;
ControlModule: TYPE = PrincOps.ControlModule;
SGList: TYPE = REF SGListObj;
SGItem:
TYPE =
RECORD [
sgh: SGHandle,
interval: VM.Interval,
file: FS.OpenFile,
bcdBase: INT
];
SGListObj: TYPE = RECORD[seq: SEQUENCE maxLength: CARDINAL OF SGItem];
CMMap: TYPE = REF CMMapObj;
CMMapObj: TYPE = RECORD [seq: SEQUENCE maxLength: NAT OF CMMapItem];
CMMapItem: TYPE = RECORD [cti: CTIndex, cm: PrincOps.ControlModule, level: CARDINAL];
PaintedTime: TYPE = RECORD[BasicTime.GMT];
**********************************************************
public procedures
**********************************************************
Error:
PUBLIC
ERROR[type: Loader.ErrorType, message:
ROPE ←
NIL] =
CODE;
Instantiate:
PUBLIC
PROC[file:
FS.OpenFile, pageOffset:
INT ← 0,
codeLinks:
BOOL ←
TRUE]
RETURNS[cm: PrincOps.ControlModule, unboundImports:
LIST
OF Loader.IRItem] = {
pageOffset: offset within the file of first page of the BCD, normally 0
msg: ROPE ← NIL;
t: Loader.ErrorType;
BEGIN
ENABLE {
Error => {t ← type; msg ← message; GOTO unwound};
VM.CantAllocate => {t ← insufficientVM; GOTO unwound};
};
[cm, unboundImports] ← LoaderEntry[file, pageOffset, codeLinks];
EXITS unwound => ERROR Error[t, msg];
END;
Start:
PUBLIC
PROC [cm: PrincOps.ControlModule] = {
IF cm # LOOPHOLE[PrincOps.NullControl] THEN START LOOPHOLE[cm, PROGRAM];
};
MakeProcedureResident:
PUBLIC
SAFE
PROC[proc:
PROC
ANY
RETURNS
ANY] =
TRUSTED {
LoadState.local.Acquire[]; -- acquire the lock on the loadstate
BEGIN
ENABLE {
UNWIND =>
NULL;
ANY => LoadState.local.Release[]};
VM.Pin[IntervalForProc[proc]];
END;
LoadState.local.Release[];
};
MakeProcedureSwappable:
PUBLIC
PROC[proc:
PROC
ANY
RETURNS
ANY] = {
LoadState.local.Acquire[]; -- acquire the lock on the loadstate
BEGIN
ENABLE {
UNWIND =>
NULL;
ANY => LoadState.local.Release[]};
VM.Unpin[IntervalForProc[proc]];
END;
LoadState.local.Release[];
};
MakeGlobalFrameResident:
PUBLIC
SAFE
PROC[proc:
PROC
ANY
RETURNS
ANY] =
TRUSTED {
VM.Pin[IntervalForGlobalFrame[proc]];
};
MakeGlobalFrameSwappable:
PUBLIC
PROC[proc:
PROC
ANY
RETURNS
ANY] = {
VM.Unpin[IntervalForGlobalFrame[proc]];
};
IntervalForProc:
PROC [proc:
PROC
ANY
RETURNS
ANY]
RETURNS [
VM.Interval] = {
gfh: PrincOps.GlobalFrameHandle = LOOPHOLE[PrincOpsUtils.GlobalFrame[proc]];
codeBase: LONG POINTER = PrincOpsUtils.Codebase[gfh];
bcd: BcdDefs.BcdBase;
config: LoadState.ConfigID;
module: LoadState.ModuleIndex;
mtb: BcdDefs.Base;
mth: BcdDefs.MTHandle;
sgb: BcdDefs.Base;
[config, module] ← LoadState.local.GlobalFrameToModule[gfh];
bcd ← LoadState.local.ConfigInfo[config].bcd;
sgb ← LOOPHOLE[bcd + bcd.sgOffset];
mtb ← LOOPHOLE[bcd + bcd.mtOffset];
mth ← @mtb[LoadState.local.ModuleInfo[config, module].mti];
IF ~mth.packageable OR mth.code.packed THEN ERROR; -- not yet implemented.
RETURN[[VM.PageNumberForAddress[codeBase], sgb[mth.code.sgi].pages]]
};
IntervalForGlobalFrame:
PROC [proc:
PROC
ANY
RETURNS
ANY]
RETURNS [
VM.Interval] = {
gfh: PrincOps.GlobalFrameHandle = LOOPHOLE[PrincOpsUtils.GlobalFrame[proc]];
codeBase: LONG POINTER TO PrincOps.CSegPrefix = PrincOpsUtils.Codebase[gfh];
pGFSize:
LONG
POINTER
TO
CARDINAL =
LOOPHOLE[codeBase + CARDINAL[codeBase.entry[PrincOps.MainBodyIndex].initialpc] - 1];
nLinks: CARDINAL = codeBase.header.info.nlinks;
linkspace:
CARDINAL =
IF gfh.codelinks THEN 0 ELSE
nLinks + PrincOpsUtils.BITAND[-LOOPHOLE[nLinks, INTEGER], 3B];
nWords: CARDINAL = pGFSize^ + linkspace;
RETURN[[VM.PageNumberForAddress[gfh-linkspace], VM.PagesForWords[nWords]]]
};
BCDBuildTime:
PUBLIC
SAFE
PROC[proc:
PROC
ANY
RETURNS
ANY ←
NIL]
RETURNS[BasicTime.GMT] = TRUSTED {
ref: REF ANY;
IF proc = NIL THEN proc ← LOOPHOLE[PrincOpsUtils.GetReturnLink[]];
LoadState.local.Acquire[];
BEGIN
ENABLE {
UNWIND =>
NULL;
ANY => LoadState.local.Release[]};
gfh: PrincOps.GlobalFrameHandle = LOOPHOLE[PrincOpsUtils.GlobalFrame[proc]];
ref ← LoadState.local.ConfigInfo[LoadState.local.GlobalFrameToModule[gfh].config].ref;
END;
LoadState.local.Release[];
IF ref = NIL THEN RETURN[BasicTime.FromPupTime[SystemVersion.bootFileDate]]
ELSE RETURN[NARROW[ref, REF PaintedTime]^]
};
**********************************************************
The central loading procedures. LoadBcd copies the bcd into VM,
and New calls the procedures that do the work.
**********************************************************
LoaderEntry:
ENTRY
PROC[
file: FS.OpenFile,
pageOffset: INT,
codeLinks: BOOL]
RETURNS [cm: PrincOps.ControlModule, unboundImports: LIST OF Loader.IRItem] = {
ENABLE UNWIND => NULL;
[cm, unboundImports] ← New[LoadBcd[file, pageOffset], file, pageOffset, ~codeLinks];
LoadBcd:
INTERNAL
PROC[file:
FS.OpenFile, pageOffset:
INT]
RETURNS [bcd: BcdBase] = {
bcdSpace: VM.Interval;
pages, rtPages: INT;
bcdSpace ← VM.Allocate[count: 1];
bcd ← VM.AddressForPageNumber[bcdSpace.page];
FS.Read[file: file, from: pageOffset, nPages: 1, to: bcd];
IF bcd.versionIdent # VersionID
OR bcd.definitions
OR ~bcd.spare1
THEN {VM.Free[bcdSpace]; ERROR Error[invalidBcd]};
pages ← bcd.nPages;
rtPages ← bcd.rtPages.pages;
IF pages > 1
THEN {
VM.Free[bcdSpace];
bcdSpace ← VM.Allocate[count: pages];
bcd ← VM.AddressForPageNumber[bcdSpace.page];
FS.Read[file: file, from: pageOffset, nPages: pages, to: bcd];
};
VM.MakeReadOnly[bcdSpace];
};
New:
INTERNAL
PROC[
bcd: BcdBase, file: FS.OpenFile, pageOffset: INT, framelinks: BOOL]
RETURNS[cm: PrincOps.ControlModule, unboundImports: LIST OF Loader.IRItem] = {
fl: FrameList ← NIL;
cid: LoadState.ConfigID;
LoadState.local.Acquire[exclusive]; -- acquire the lock on the loadstate
cid ← LoadState.local.NewConfig[bcd, NEW[PaintedTime ← [FS.GetInfo[file].created]]];
BEGIN
ENABLE {
UNWIND => NULL;
ANY => {
LoadState.local.Release[commit: FALSE];
ReleaseFrames[fl, cid];
};
};
moduleToGFH:
PROC[mx: ModuleIndex]
RETURNS[PrincOps.GlobalFrameHandle] =
{RETURN[LoadState.local.ModuleInfo[cid, mx].gfh]};
setType:
PROC[gfh: PrincOps.GlobalFrameHandle, type: SafeStorage.Type] =
{LoadState.local.SetType[gfh, type]};
fl ← CreateGlobalFrames[cid, framelinks];
AssignCodeToFrames[cid, file, pageOffset];
cm ← AssignControlModules[cid];
unboundImports ← LoaderOps.Bind[cid];
LoaderOps.Export[cid];
SafeStorageOps.AcquireTypesAndLiterals[bcd, moduleToGFH, setType];
END; -- ENABLE UNWIND
LoadState.local.Release[]; --XXX deadlock with TandS
**********************************************************
CreateGlobalFrames allocates and initializes global frames for the modules
in the bcd, constructs the loadstate info for these modules and returns the
list of frames.
**********************************************************
CreateGlobalFrames:
PUBLIC PROC[cid: LoadState.ConfigID, allframelinks:
BOOL]
RETURNS[fl: FrameList ← NIL] = {
bcd: BcdBase = LoadState.local.ConfigInfo[cid].bcd;
f: FrameList;
frames: POINTER;
space: CARDINAL;
single: BOOL ← (bcd.nModules = 1);
resident: BOOL;
GetFrameSizes:
PROC [mth: MTHandle] = {
IF allframelinks
OR mth.linkLoc = frame
OR ~mth.code.linkspace
THEN space ← space + LoaderOps.LinkSegmentLength[mth, bcd];
space ← NextMultipleOfFour[space] + mth.framesize;
resident ← resident OR mth.residentFrame;};
FrameInit:
PROC [mth: MTHandle] = {
frame: GlobalFrameHandle;
framelinks: BOOL ← allframelinks OR mth.linkLoc = frame OR ~mth.code.linkspace;
IF framelinks THEN frames ← frames + LoaderOps.LinkSegmentLength[mth, bcd];
frame ← NextMultipleOfFour[frames];
frames ← frame + mth.framesize;
LoadState.local.SetGlobalFrame[cid, mth.gfi, frame];
frame^
← [gfi: frame.gfi, copied:
FALSE, alloced: single, shared:
FALSE, started:
FALSE,
trapxfers: FALSE, codelinks: ~framelinks, global:, code:];};
DoFramePack:
PROC [fph: FPHandle, fpi: FPIndex]
RETURNS [stop:
BOOL ←
FALSE] = {
mtb: Base = LOOPHOLE[bcd + bcd.mtOffset];
space ← 0;
resident ← FALSE;
FOR i: CARDINAL IN [0..fph.length) DO GetFrameSizes[@mtb[fph.modules[i]]] ENDLOOP;
[f, frames]
← AllocateFrames[size: NextMultipleOfFour[space], single: single, resident: resident];
f.rest ← fl;
fl ← f;
FOR i: CARDINAL IN [0..fph.length) DO FrameInit[@mtb[fph.modules[i]]] ENDLOOP;};
OtherFrameSizes:
PROC [mth: MTHandle, mti: MTIndex]
RETURNS[stop: BOOL ← FALSE] = {
IF LoadState.local.ModuleInfo[cid, mth.gfi].gfh = NIL THEN GetFrameSizes[mth];};
OtherFrameInit:
PROC [mth: MTHandle, mti: MTIndex]
RETURNS[stop: BOOL ← FALSE] = {
IF LoadState.local.ModuleInfo[cid, mth.gfi].gfh = NIL THEN FrameInit[mth];};
Start CreateGlobalFrames HERE
BEGIN
ENABLE
UNWIND => ReleaseFrames[fl, cid];
[] ← BcdOps.ProcessFramePacks[bcd, DoFramePack];
space ← 0;
resident ← FALSE;
[] ← BcdOps.ProcessModules[bcd, OtherFrameSizes];
IF space # 0
THEN {
[f, frames]
← AllocateFrames[size: NextMultipleOfFour[space], single: single, resident: resident];
f.rest ← fl;
fl ← f;
[] ← BcdOps.ProcessModules[bcd, OtherFrameInit];
};
END; -- ENABLE UNWIND
};
-- end CreateGlobalFrames
**********************************************************
Frame allocation/deallocation
**********************************************************
AllocateFrames:
PUBLIC
PROC [size:
CARDINAL, single, resident:
BOOL]
RETURNS [fl: FrameList, frames: POINTER] = {
IF single
THEN {
index: CARDINAL;
FOR index
IN [0..PrincOps.LastAVSlot]
DO IF PrincOpsUtils.FrameSize[index] >= size THEN EXIT; ENDLOOP;
frames ← PrincOpsUtils.Alloc[index];
[] ← PrincOpsUtils.ZERO[LONG[frames], size];
fl ← CONS[[frame[ptr: frames]], NIL];
}
ELSE {
pages: INT = VM.PagesForWords[size];
interval: VM.Interval ← VM.Allocate[count: pages, partition: mds];
IF resident THEN VM.Pin[interval];
frames
← LOOPHOLE[PrincOpsUtils.LowHalf[VM.AddressForPageNumber[interval.page]], POINTER];
[] ← PrincOpsUtils.ZERO[LONG[frames], VM.WordsForPages[pages]];
fl ← CONS[[mdsInterval[interval: interval]], NIL];
};
};
NextMultipleOfFour: PROC [n: UNSPECIFIED] RETURNS [UNSPECIFIED] =
INLINE {RETURN[n + PrincOpsUtils.BITAND[-LOOPHOLE[n, INTEGER], 3B]]};
ReleaseFrames:
PUBLIC
PROC [fl: FrameList, cid: LoadState.ConfigID] = {
FOR f: FrameList ← fl, f.rest
UNTIL f =
NIL
DO
WITH v: f.first
SELECT
FROM
frame => PrincOpsUtils.Free[v.ptr];
mdsInterval => VM.Free[v.interval];
ENDCASE;
ENDLOOP;
**********************************************************
AssignCodeToFrames: Makes up a list of code segments, maps them in,
and associates each code segment with the appropriate global frame.
**********************************************************
AssignCodeToFrames:
PUBLIC
PROC [
config: LoadState.ConfigID, bcdCap: FS.OpenFile, pageOffset: INT--within file--] = {
sgList: SGList;
allCodeInSelf: BOOL;
bcd: BcdBase = LoadState.local.ConfigInfo[config].bcd;
GetCode:
PROC [mth: MTHandle, mti: MTIndex]
RETURNS [
BOOL] = {
MatchCSeg: PROC [mmth: MTHandle, mmti: MTIndex] RETURNS [BOOL] = {
IF mth # mmth
AND mth.code.sgi = mmth.code.sgi
AND mth.code.offset = mmth.code.offset
THEN
frame.shared ← LoadState.local.ModuleInfo[config, mmth.gfi].gfh.shared ← TRUE;
RETURN[FALSE];
};
frame: PrincOps.GlobalFrameHandle = LoadState.local.ModuleInfo[config, mth.gfi].gfh;
IF mth.altoCode THEN InvalidModule[bcd, mth];
frame.code.longbase ← SgiToLP[bcd, mth.code.sgi, sgList] + mth.code.offset;
frame.code.out ← TRUE;
Don't do the MatchCSeg stuff; Roy and I (Andrew) think it does no good, and it costs an n-squared enumeration of the modules!
[] ← BcdOps.ProcessModules[bcd, MatchCSeg];
RETURN[FALSE];
};
START AssignCodeToFrames HERE
[sgList, allCodeInSelf] ← FindSegments[bcd, bcdCap, pageOffset];
IF bcd.nModules = 1
THEN AllocateCodeSpace[bcd, sgList, pageOffset, bcdCap]
ELSE AllocateCodeSpaces[bcd, sgList, allCodeInSelf];
[] ← BcdOps.ProcessModules[bcd, GetCode ! UNWIND => ReleaseCode[sgList]];
};
-- end AssignCodeToFrames
ReleaseCode:
PROC [sgList: SGList] = {
FOR i: CARDINAL ← 0, i+1 UNTIL i = sgList.maxLength DO VM.Free[sgList[i].interval] ENDLOOP;
};
called only from AssignCodeToFrames
FindSegments:
PROC [bcd: BcdBase, bcdCap:
FS.OpenFile, pageOffset:
INT
--page number--]
RETURNS [sgList: SGList, allCodeInSelf: BOOL ← TRUE] =
BEGIN
n: CARDINAL;
sgItem: SGItem;
GetFileFromSGH:
PROC [sgh: SGHandle]
RETURNS [file:
FS.OpenFile ← [
NIL]] = {
fth: FTHandle = @LOOPHOLE[bcd + bcd.ftOffset, Base][sgh.file];
name: ROPE ← NameToRope[ssb: LOOPHOLE[bcd + bcd.ssOffset], name: fth.name];
dot: BOOL ← Rope.Find[name, "."] # -1;
bcd: BcdBase;
interval: VM.Interval;
IF ~dot THEN name ← Rope.Concat[name, ".bcd"];
file ← FS.Open[name: name ! FS.Error => CONTINUE];
IF file =
FS.OpenFile[
NIL]
THEN
ERROR Error[fileNotFound, name];
interval ← VM.Allocate[count: 1];
bcd ← LOOPHOLE[VM.AddressForPageNumber[interval.page]];
FS.Read[file: file, from: 0, nPages: 1, to: bcd];
IF fth.version # bcd.version
THEN {
VM.Free[interval];
ERROR Error[versionMismatch, name];
};
VM.Free[interval];
};
CountSegs:
PROC [sgh: SGHandle, sgi: SGIndex]
RETURNS [
BOOL] = {
IF sgh.class # code THEN RETURN[FALSE];
IF sgh.file # FTSelf THEN allCodeInSelf ← FALSE;
n ← n + 1;
RETURN[FALSE];
};
AddSeg:
PROC [sgh: SGHandle, sgi: SGIndex]
RETURNS [
BOOL] = {
file: FS.OpenFile;
base: INT;
IF sgh.class # code THEN RETURN[FALSE];
IF sgh.file = FTSelf
THEN {file ← bcdCap; base ← pageOffset}
ELSE {file ← GetFileFromSGH[sgh]; base ← 0};
sgList[n] ← [sgh: sgh, interval: [0, 0], file: file, bcdBase: base];
n ← n + 1;
RETURN[FALSE];
};
SiftUp:
PROC [low, high:
CARDINAL] = {
k, son: CARDINAL;
sgItem: SGItem;
k ← low;
DO
IF k*2 > high THEN EXIT;
IF k*2 + 1 > high
OR sgList[k*2 + 1 - 1].sgh.base < sgList[
k*2 - 1].sgh.base THEN son ← k*2
ELSE son ← k*2 + 1;
IF sgList[son - 1].sgh.base < sgList[k - 1].sgh.base THEN EXIT;
sgItem ← sgList[son - 1];
sgList[son - 1] ← sgList[k - 1];
sgList[k - 1] ← sgItem;
k ← son;
ENDLOOP;
RETURN;
};
START FindSegments HERE
n ← 0;
[] ← BcdOps.ProcessSegs[bcd, CountSegs];
sgList ← NEW[SGListObj[n]];
n ← 0;
[] ← BcdOps.ProcessSegs[bcd, AddSeg];
IF allCodeInSelf
THEN {
FOR n DECREASING IN [1..sgList.maxLength/2] DO SiftUp[n, sgList.maxLength] ENDLOOP;
FOR n
DECREASING
IN [1..sgList.maxLength)
DO
sgItem ← sgList[1 - 1];
sgList[1 - 1] ← sgList[n + 1 - 1];
sgList[n + 1 - 1] ← sgItem;
SiftUp[1, n];
ENDLOOP;
};
NameToRope:
PROC [ssb: NameString, name: NameRecord]
RETURNS[ans: ROPE] = {
s: IO.STREAM = IO.ROS[];
FOR i: NAT IN [0..ssb.size[name]) DO s.PutChar[ssb.string.text[name+i]] ENDLOOP;
ans ← s.RopeFromROS[];
AllocateCodeSpace:
PROC [bcd: BcdBase, sgList: SGList, pageOffset:
INT, bcdCap:
FS.OpenFile]
= {
this is a bcd produced by the compiler
interval: VM.Interval;
mth: MTHandle;
pages, biaspages: CARDINAL;
IF sgList.maxLength ~= 1 THEN ERROR;
mth ← @LOOPHOLE[bcd + bcd.mtOffset, Base][FIRST[MTIndex]];
non-0 if code links
biaspages ←
IF mth.linkLoc = code
THEN (LoaderOps.LinkSegmentLength[mth, bcd]/PrincOps.wordsPerPage)+1
ELSE 0;
pages ← sgList[0].sgh.pages;
ncodepages ← pages + biaspages;
interval ← VM.Allocate[count: pages + biaspages, in64K: TRUE]; -- won't cross a 64K VM bdry
if the bcd is being loaded with code links, the cedar compiler will have left room for them
FS.Read[
file: bcdCap,
from: pageOffset + sgList[0].sgh.base - 1,
nPages: pages + biaspages,
to: VM.AddressForPageNumber[interval.page]
];
VM.MakeReadOnly[interval];
sgList[0].interval ← interval;
AllocateCodeSpaces:
PROC [bcd: BcdBase, sgList: SGList, allCodeInSelf:
BOOL] = {
start, end, startBase, pages: CARDINAL;
pageOfInterval: VM.PageNumber;
interval: VM.Interval;
start ← end ← 0;
DO
startBase ← sgList[start].sgh.base;
pages ← sgList[start].sgh.pages;
DO
end ← end + 1;
IF end = sgList.maxLength
OR pages + sgList[end].sgh.pages > 255
OR ~ allCodeInSelf
THEN EXIT;
pages ← pages + sgList[end].sgh.pages;
ENDLOOP;
interval ← VM.Allocate[count: pages, in64K: TRUE];
FS.Read[
file: sgList[start].file,
from: sgList[start].bcdBase + sgList[start].sgh.base - 1,
nPages: pages,
to: VM.AddressForPageNumber[interval.page]
];
VM.MakeReadOnly[interval];
pageOfInterval ← interval.page;
FOR index:
CARDINAL
IN [start..end)
DO
sgh: SGHandle = sgList[index].sgh;
sph: SPHandle ← FindSPHandle[bcd, sgh];
subInterval: VM.Interval;
IF sph =
NIL
THEN subInterval ← [page: pageOfInterval, count: sgh.pages]
ELSE {
FOR sp:
CARDINAL
DECREASING
IN [0..sph.length)
DO
subInterval ←
[page: pageOfInterval + sph.spaces[sp].offset, count: sph.spaces[sp].pages];
Hmm. Does this work? who remembers subInterval; apparently only the last one is remembered (below) for this sgList entry. Is it the "first" one?
IF sph.spaces[sp].resident THEN VM.Pin[subInterval];
ENDLOOP;
};
sgList[index].interval ← subInterval;
pageOfInterval ← pageOfInterval + sgh.pages;
ENDLOOP;
IF end = sgList.maxLength THEN EXIT ELSE start ← end;
ENDLOOP;
};
-- end AllocateCodeSpaces
FindSPHandle:
PUBLIC
PROC [bcd: BcdBase, sgh: SGHandle]
RETURNS [sph: SPHandle] = {
sgb: Base = LOOPHOLE[bcd + bcd.sgOffset];
proc:
PROC [sph: SPHandle, spi: SPIndex]
RETURNS [stop: BOOL] = {RETURN[@sgb[sph.seg] = sgh]};
RETURN[BcdOps.ProcessSpaces[bcd: bcd, proc: proc].sph];
SgiToLP:
PROC[bcd: BcdBase, sgi: SGIndex, sgList: SGList]
RETURNS[LONG POINTER] = {
sgb: Base = LOOPHOLE[bcd + bcd.sgOffset];
FOR i:
CARDINAL
IN [0..sgList.maxLength)
DO
IF @sgb[sgi] = sgList[i].sgh
THEN RETURN[VM.AddressForPageNumber[sgList[i].interval.page]];
ENDLOOP;
RETURN[NIL];
InvalidModule:
PROC [bcd: BcdBase, mth: MTHandle] = {
name: ROPE = NameToRope[ssb: LOOPHOLE[bcd + bcd.ssOffset], name: mth.name];
ERROR Error[invalidBcd, name];
**********************************************************
AssignControlModules: determines start trap links for each global frame.
**********************************************************
AssignControlModules:
PUBLIC PROC[cid: LoadState.ConfigID]
RETURNS [cm: PrincOps.ControlModule] = { OPEN PrincOps;
bcd: BcdBase ← LoadState.local.ConfigInfo[cid].bcd;
ctb: Base ← LOOPHOLE[bcd + bcd.ctOffset];
mtb: Base ← LOOPHOLE[bcd + bcd.mtOffset];
cti: CTIndex;
mapIndex, maxLevel: NAT ← 0;
i: NAT;
cmMap: CMMap;
MapControls:
PROC[cth: CTHandle, cti: CTIndex]
RETURNS[stop:
BOOL ←
FALSE] = {
OPEN PrincOpsUtils;
cm: ControlModule;
level: CARDINAL ← 0;
IF cth.nControls = 0
THEN cm ← NullControl
ELSE {
cm.list ← PrincOpsUtils.Alloc[
MakeFsi[cth.nControls + SIZE[CARDINAL] + SIZE[ControlModule]]];
cm.list.nModules ← cth.nControls + 1;
FOR i:
CARDINAL
IN [0..cth.nControls)
DO
WITH ci: cth.controls[i]
SELECT
FROM
module =>
cm.list.frames[i+1] ← LoadState.local.ModuleInfo[cid, mtb[ci.mti].gfi].gfh;
config => ERROR;
ENDCASE => ERROR;
ENDLOOP;
cm.multiple ← TRUE;
};
FOR c: CTIndex ← ctb[cti].config, ctb[c].config
UNTIL c = CTNull
DO level ← level + 1 ENDLOOP;
cmMap[mapIndex] ← [cti: cti, cm: cm, level: level];
mapIndex ← mapIndex + 1;
maxLevel ← MAX[maxLevel, level];
GetControl:
PROC [mth: MTHandle, mti: MTIndex]
RETURNS [
BOOL] = {
OPEN PrincOps, PrincOpsUtils;
frame: GlobalFrameHandle ← LoadState.local.ModuleInfo[cid, mth.gfi].gfh;
IF mth.config # cti THEN RETURN[FALSE];
IF frame.global[0] = NullControl THEN frame.global[0] ← GetModule[cm];
RETURN[FALSE];
START AssignControlModules HERE
IF bcd.nModules = 1
THEN {
frame: GlobalFrameHandle ← LoadState.local.ModuleInfo[cid, 1].gfh;
frame.global[0] ← NullControl;
RETURN[[frame[frame]]];
};
cmMap ← NEW[CMMapObj[bcd.nConfigs]];
[] ← BcdOps.ProcessConfigs[bcd, MapControls];
FOR level:
CARDINAL
DECREASING
IN [0..maxLevel]
DO
FOR index:
CARDINAL
IN [0..mapIndex)
DO
list: ControlModule;
IF cmMap[index].level # level OR (cm ← cmMap[index].cm) = NullControl THEN LOOP;
list ← cm;
list.multiple ← FALSE;
list.list.frames[1] ← SetLink[cm, list.list.frames[1]].frame;
FOR i:
CARDINAL
IN [2..list.list.nModules)
DO
list.list.frames[i] ←
SetLink[GetModule[[frame[list.list.frames[1]]]], list.list.frames[i]].frame;
ENDLOOP;
cti ← cmMap[index].cti;
[] ← BcdOps.ProcessModules[bcd, GetControl];
ENDLOOP;
ENDLOOP;
FOR index:
CARDINAL
IN [0..mapIndex)
DO
parent: CARDINAL;
list: ControlModule;
IF (list ← cmMap[index].cm) = NullControl THEN LOOP;
list.multiple ← FALSE;
IF (cti ← ctb[cmMap[index].cti].config) = CTNull THEN cm ← NullControl
ELSE {
FOR parent IN [0..mapIndex) DO IF cmMap[parent].cti = cti THEN EXIT; ENDLOOP;
cm ← GetModule[cmMap[parent].cm]};
list.list.frames[0] ← cm.frame;
ENDLOOP;
FOR i
IN [0..mapIndex)
DO
IF ctb[cmMap[i].cti].config = CTNull
THEN {
cm ← GetModule[cmMap[i].cm]; EXIT};
ENDLOOP;
}; -- end AssignControlModules
SetLink:
PROC[cm: ControlModule, frame: GlobalFrameHandle]
RETURNS [ControlModule] = {
t: ControlModule = frame.global[0];
frame.global[0] ← cm;
RETURN[IF t = PrincOps.NullControl THEN [frame[frame]] ELSE t];
GetModule:
PROC [cm: ControlModule]
RETURNS [ControlModule] = {
list: ControlModule;
DO
IF ~cm.multiple THEN RETURN[cm];
list ← cm;
list.multiple ← FALSE;
cm.frame ← list.list.frames[1];
ENDLOOP;
END.