-- BcdWrite.mesa -- Last edited by Satterthwaite on August 1, 1983 12:20 pm -- Last edited by Lewis on 30-Mar-81 16:51:35 DIRECTORY Alloc: TYPE USING [ AddNotify, Bounds, DropNotify, Handle, Notifier, Top, Trim, Words], BcdComData: TYPE USING [ aborted, bcdName, binderVersion, codeName, copyCode, copySymbols, nConfigs, nExports, nImports, nModules, nPages, objectStamp, op, outputFile, sourceName, sourceVersion, symbolName, table, typeExported, textIndex, zone], BcdControlDefs: TYPE USING [], BcdDefs: TYPE USING [ BCD, CTIndex, CTRecord, cttype, cxtype, EVIndex, evtype, EXPIndex, EXPRecord, exptype, FPIndex, FPRecord, fptype, FTIndex, FTNull, FTRecord, FTSelf, fttype, IMPIndex, IMPNull, imptype, LFNull, lftype, Link, MTIndex, MTNull, MTRecord, mttype, Namee, NameRecord, NTIndex, NTRecord, nttype, NullName, NullVersion, PackedString, rftype, SegClass, SGIndex, SGNull, SGRecord, sgtype, SpaceID, SPIndex, SPRecord, sptype, sstype, sttype, tftype, tmtype, treetype, typtype, VersionID, VersionStamp], BcdErrorDefs: TYPE USING [ErrorHti, ErrorName, ErrorNameBase, GetSti], BcdLiterals: TYPE USING [ EnterVersionFiles, LitSegSize, UpdateSegments, WriteLiterals], BcdOps: TYPE USING [BcdBase, NameString], BcdUtilDefs: TYPE USING [ BcdBases, ContextForTree, EnterExport, EnterFile, EnterImport, EnterName, GetGfi, MapName, MergeFile, NameForHti, SetFileVersion], Environment: TYPE USING [bytesPerPage, bytesPerWord, wordsPerPage], File: TYPE USING [Capability, nullCapability, SetSize], FileSegment: TYPE USING [Pages, nullPages], FileStream: TYPE USING [FileByteIndex, Create, GetIndex, SetIndex], --ExecOps: TYPE USING [CheckForAbort], HashOps: TYPE USING [Reset], Inline: TYPE USING [LongCOPY, LongMult], OSMiscOps: TYPE USING [FileError, FindFile, StampToTime], Space: TYPE USING [ Create, Delete, Handle, Map, LongPointer, Unmap, virtualMemory], Stream: TYPE USING [Handle, Delete, PutBlock, PutWord], Strings: TYPE USING [ String, SubString, SubStringDescriptor, AppendChar, AppendString], Symbols: TYPE USING [CXIndex, HTIndex, htNull, STIndex, stNull], Table: TYPE USING [Base, Selector], Tree: TYPE USING [Index, Link, Scan, null], TreeOps: TYPE USING [GetNode, ListLength, ScanList]; BcdWrite: PROGRAM IMPORTS Alloc, BcdErrorDefs, BcdLiterals, BcdUtilDefs, --ExecOps,-- File, FileStream, HashOps, Inline, OSMiscOps, Space, Stream, Strings, TreeOps, data: BcdComData EXPORTS BcdControlDefs = { OPEN BcdDefs; bytesPerWord: CARDINAL ~ Environment.bytesPerWord; nullFile: File.Capability ~ File.nullCapability; nullPages: FileSegment.Pages ~ FileSegment.nullPages; Alignment: CARDINAL ~ 4; -- Code Segments must start at 0 MOD Alignment BcdWriteError: PUBLIC ERROR ~ CODE; Error: PROC ~ {ERROR BcdWriteError}; UserAbort: ERROR ~ CODE; -- raised on ↑DEL during code or symbol copying table: Alloc.Handle; tb, stb, ctb, mtb, lfb, etb, itb, sgb, tyb, tmb, ftb, ntb, spb, fpb, cxb: Table.Base; ssb: BcdOps.NameString; Notifier: Alloc.Notifier ~ { tb ← base[treetype]; stb ← base[sttype]; ctb ← base[cttype]; mtb ← base[mttype]; lfb ← base[lftype]; tyb ← base[typtype]; tmb ← base[tmtype]; etb ← base[exptype]; itb ← base[imptype]; sgb ← base[sgtype]; ftb ← base[fttype]; spb ← base[sptype]; fpb ← base[fptype]; ntb ← base[nttype]; ssb ← base[sstype]; cxb ← base[cxtype]; IF bcd # NIL THEN { bcd.ctb ← ctb; bcd.mtb ← mtb; IF ~packing THEN bcd.sgb ← sgb; bcd.tyb ← tyb; bcd.tmb ← tmb; bcd.spb ← spb; bcd.fpb ← fpb}}; -- inline utilities Copy: PROC [from: LONG POINTER, nwords: CARDINAL, to: LONG POINTER] ~ Inline.LongCOPY; Zero: PROC [p: LONG POINTER, l: CARDINAL] ~ INLINE { IF l # 0 THEN {p↑ ← 0; Inline.LongCOPY[from~p, to~(p+1), nwords~(l-1)]}}; PagesForWords: PROC [nWords: CARDINAL] RETURNS [CARDINAL] ~ INLINE { RETURN [(nWords + Environment.wordsPerPage-1)/Environment.wordsPerPage]}; bcd: LONG POINTER TO BcdUtilDefs.BcdBases ← NIL; header: BcdOps.BcdBase; -- points to Bcd header and saved tables headerSpace: Space.Handle; WriteBcd: PUBLIC PROC [root: Tree.Link] ~ { saveIndex: CARDINAL ~ data.textIndex; node, subNode: Tree.Index; table ← data.table; table.AddNotify[Notifier]; node ← TreeOps.GetNode[root]; packing ← (tb[node].son[2] # Tree.null AND data.copyCode); Initialize[]; IF packing THEN { MakePackItem[tb[node].son[2]]; data.textIndex ← saveIndex; FillInSgMap[]}; CopyConfigs[]; CopyModules[]; CopyTypes[]; CopySpaces[]; CopyFramePacks[]; subNode ← TreeOps.GetNode[tb[node].son[3]]; data.textIndex ← tb[subNode].info; TreeOps.ScanList[tb[subNode].son[1], CopyImport]; TreeOps.ScanList[tb[subNode].son[2], CopyExport]; IF tb[subNode].attrs[$exportsALL] THEN ExportCx[BcdUtilDefs.ContextForTree[tb[subNode].son[4]]]; IF data.copySymbols THEN EnterMissingSymbolFiles[]; BcdLiterals.EnterVersionFiles[bcd.ftb, FtiForIndex[fileMap.length], MapFile]; TableOut[]; CloseOutputFile[]; Finalize[]; data.textIndex ← saveIndex; table.DropNotify[Notifier]; table ← NIL}; Initialize: PROC ~ { impSize, expSize, sgSize, fSize, nSize, ssSize: CARDINAL; nSgis: CARDINAL; b: Table.Base; desc: Strings.SubStringDescriptor ← [base~data.sourceName, offset~0, length~data.sourceName.length]; IF data.copyCode OR data.copySymbols THEN InitCodeSymbolCopy[]; impSize ← table.Bounds[imptype].size; expSize ← table.Bounds[exptype].size; sgSize ← table.Bounds[sgtype].size; nSgis ← sgSize/SGRecord.SIZE; IF ~packing THEN sgSize ← 0; fSize ← table.Bounds[fttype].size; nSize ← table.Bounds[nttype].size; ssSize ← table.Bounds[sstype].size; bcd ← (data.zone).NEW[BcdUtilDefs.BcdBases]; fileMap ← (data.zone).NEW[FileMap[fSize/FTRecord.SIZE]]; FOR i: CARDINAL IN [0..fileMap.length) DO fileMap[i] ← FTNull ENDLOOP; headerSpace ← Space.Create[ size~PagesForWords[ BCD.SIZE + impSize + expSize + sgSize + fSize + nSize + ssSize], parent~Space.virtualMemory]; headerSpace.Map[]; header ← LOOPHOLE[headerSpace.LongPointer, BcdOps.BcdBase]; b ← (LOOPHOLE[header, Table.Base] + BCD.SIZE); Copy[to~(bcd.etb ← b), from~etb, nwords~expSize]; b ← b + expSize; table.Trim[exptype,0]; Copy[to~(bcd.itb ← b), from~itb, nwords~impSize]; b ← b + impSize; table.Trim[imptype,0]; Copy[to~(bcd.ftb ← b), from~ftb, nwords~fSize]; b ← b + fSize; table.Trim[fttype,0]; Copy[to~(bcd.ntb ← b), from~ntb, nwords~nSize]; b ← b + nSize; table.Trim[nttype,0]; Copy[to~(bcd.ssb ← b), from~ssb, nwords~ssSize]; b ← b + ssSize; HashOps.Reset[]; IF packing THEN { -- save old segment table in heap Copy[to~(bcd.sgb ← b), from~sgb, nwords~sgSize]; b ← b + sgSize; table.Trim[sgtype,0]}; InitHeader[ header~header, objectVersion~OSMiscOps.StampToTime[data.objectStamp], source~BcdUtilDefs.EnterName[@desc], sourceVersion~data.sourceVersion]; bcd.ctb ← table.Bounds[cttype].base; bcd.mtb ← table.Bounds[mttype].base; bcd.tyb ← table.Bounds[typtype].base; bcd.tmb ← table.Bounds[tmtype].base; bcd.spb ← table.Bounds[sptype].base; bcd.fpb ← table.Bounds[fptype].base; IF data.copyCode OR data.copySymbols THEN {MapCodeSymbolFiles[]; InitCopyMap[nSgis]}; IF packing THEN InitSgMap[nSgis] ELSE { bcd.sgb ← table.Bounds[sgtype].base; IF ~data.copyCode THEN MapSegments[$code]; IF ~data.copySymbols THEN MapSegments[$symbols]}}; Finalize: PROC ~ { IF data.copyCode OR data.copySymbols THEN ReleaseCodeSymbolCopy[]; (data.zone).FREE[@fileMap]; (data.zone).FREE[@bcd]; Space.Delete[headerSpace]; FreePackItems[]; IF packing THEN FreeSgMap[]; IF data.copyCode OR data.copySymbols THEN FreeCopyMap[]}; -- BCD (re)construction CopyName: PROC [olditem, newitem: Namee] ~ { newNti: NTIndex ~ table.Words[nttype, NTRecord.SIZE]; FOR nti: NTIndex ← NTIndex.FIRST, nti+NTRecord.SIZE DO OPEN old~~bcd.ntb[nti]; IF old.item = olditem THEN { OPEN new~~ntb[newNti]; new.item ← newitem; new.name ← bcd.MapName[old.name]; RETURN}; ENDLOOP}; CopyConfigs: PROC ~ { -- configs are already copied, only map names and files cti: CTIndex ← CTIndex.FIRST; ctLimit: CTIndex ~ table.Top[cttype]; UNTIL cti = ctLimit DO header.nConfigs ← header.nConfigs + 1; ctb[cti].name ← bcd.MapName[ctb[cti].name]; ctb[cti].file ← MapFile[ctb[cti].file]; IF ctb[cti].namedInstance THEN CopyName[[config[cti]], [config[cti]]]; cti ← cti + (CTRecord.SIZE + ctb[cti].nControls*Namee.SIZE); ENDLOOP}; CopyModules: PROC ~ { -- modules are already copied, only map names and files MapOne: PROC [mti: MTIndex] RETURNS [BOOL ← FALSE] ~ { OPEN m~~mtb[mti]; header.nModules ← header.nModules + 1; m.name ← bcd.MapName[m.name]; m.file ← MapFile[m.file]; IF m.namedInstance THEN CopyName[[module[mti]], [module[mti]]]}; EnumerateModules[MapOne]}; EnumerateModules: PROC [p: PROC [MTIndex] RETURNS [BOOL]] ~ { mti: MTIndex ← MTIndex.FIRST; mtLimit: MTIndex ~ table.Top[mttype]; UNTIL mti = mtLimit DO IF p[mti] THEN EXIT; mti ← mti + MTRecord.SIZE; ENDLOOP}; CopyTypes: PROC ~ {}; -- types are already copied, nothing need be done (current typeIds) CopySpaces: PROC ~ { -- spaces are already copied, only map names (and segments?) MapOne: PROC [spi: SPIndex] RETURNS [BOOL ← FALSE] ~ { FOR i: CARDINAL IN [0..spb[spi].length) DO spb[spi].spaces[i].name ← bcd.MapName[spb[spi].spaces[i].name]; ENDLOOP}; EnumerateSpaces[MapOne]}; EnumerateSpaces: PROC [p: PROC [SPIndex] RETURNS [BOOL]] ~ { spi: SPIndex ← SPIndex.FIRST; spLimit: SPIndex ~ table.Top[sptype]; UNTIL spi = spLimit DO IF p[spi] THEN EXIT; spi ← spi + SPRecord.SIZE + spb[spi].length*SpaceID.SIZE; ENDLOOP}; CopyFramePacks: PROC ~ { -- framepacks are already copied, only map names MapOne: PROC [fpi: FPIndex] RETURNS [BOOL ← FALSE] ~ { fpb[fpi].name ← bcd.MapName[fpb[fpi].name]}; EnumerateFramePacks[MapOne]}; EnumerateFramePacks: PROC [p: PROC [FPIndex] RETURNS [BOOL]] ~ { fpi: FPIndex ← FPIndex.FIRST; fpLimit: FPIndex ~ table.Top[fptype]; UNTIL fpi = fpLimit DO IF p[fpi] THEN RETURN; fpi ← fpi + FPRecord.SIZE + fpb[fpi].length*MTIndex.SIZE; ENDLOOP}; CopyImport: Tree.Scan ~ { OPEN Symbols; sti: STIndex ← stNull; olditi, iti: IMPIndex; WITH t SELECT FROM symbol => sti ← index; subtree => sti ← NARROW[tb[index].son[1], Tree.Link.symbol].index; ENDCASE => Error[]; olditi ← stb[sti].impi; IF sti = stNull OR olditi = IMPNull THEN RETURN; iti ← bcd.EnterImport[olditi, TRUE]; itb[iti].file ← MapFile[itb[iti].file]; IF header.firstdummy = 0 THEN header.firstdummy ← itb[iti].gfi; header.nImports ← header.nImports + 1; header.nDummies ← header.nDummies + itb[iti].ngfi}; CopyExport: Tree.Scan ~ { OPEN Symbols; sti: STIndex ← stNull; hti: HTIndex ← htNull; neweti: EXPIndex; oldeti: EXPIndex; WITH t SELECT FROM symbol => sti ← index; subtree => { sti ← NARROW[tb[index].son[1], Tree.Link.symbol].index; hti ← stb[sti].hti}; ENDCASE => Error[]; WITH s~~stb[sti] SELECT FROM external => WITH m~~s.map SELECT FROM interface => { OPEN new~~etb[neweti]; oldeti ← m.expi; neweti ← bcd.EnterExport[oldeti, TRUE]; Copy[from~@bcd.etb[oldeti].links, to~@new.links, nwords~new.size]; new.file ← MapFile[new.file]}; module => [] ← NewExportForModule[m.mti, htNull]; ENDCASE => RETURN; ENDCASE => RETURN; header.nExports ← header.nExports + 1}; NewExportForModule: PROC [mti: MTIndex, name: Symbols.HTIndex] RETURNS [eti: EXPIndex] ~ { OPEN Symbols; eti ← table.Words[exptype, EXPRecord.SIZE+1*Link.SIZE]; etb[eti] ← [ name~mtb[mti].name, size~1, port~$module, namedInstance~name # htNull, typeExported~FALSE, file~mtb[mti].file, links~ ]; etb[eti].links[0] ← [variable[vgfi~mtb[mti].gfi, var~0, vtag~$var]]; IF name # htNull THEN { nti: NTIndex ~ table.Words[nttype, NTRecord.SIZE]; ntb[nti] ← [name~BcdUtilDefs.NameForHti[name], item~[module[mti]]]}; RETURN}; ExportCx: PROC [cx: Symbols.CXIndex] ~ { OPEN Symbols; neweti, oldeti: EXPIndex; FOR sti: STIndex ← cxb[cx].link, stb[sti].link UNTIL sti = stNull DO { IF ~stb[sti].filename THEN WITH s~~stb[sti] SELECT FROM external => WITH m~~s.map SELECT FROM interface => { OPEN old~~bcd.etb[oldeti], new~~etb[neweti]; -- first make sure that old is not already exported existingEti: EXPIndex ← EXPIndex.FIRST; etLimit: EXPIndex ~ table.Top[exptype]; oldeti ← m.expi; UNTIL existingEti = etLimit DO IF old = etb[existingEti] THEN GO TO AlreadyExported; existingEti ← existingEti + EXPRecord.SIZE+etb[existingEti].size; ENDLOOP; neweti ← bcd.EnterExport[oldeti, TRUE]; Copy[from~@old.links, to~@new.links, nwords~new.size]; new.file ← MapFile[new.file]; header.nExports ← header.nExports + 1}; ENDCASE; ENDCASE; EXITS AlreadyExported => NULL}; ENDLOOP}; -- file mapping FileMap: TYPE ~ RECORD [SEQUENCE length: CARDINAL OF FTIndex]; fileMap: LONG POINTER TO FileMap ← NIL; IndexForFti: PROC [fti: FTIndex] RETURNS [CARDINAL] ~ INLINE { RETURN [LOOPHOLE[fti,CARDINAL]/FTRecord.SIZE]}; FtiForIndex: PROC [n: CARDINAL] RETURNS [FTIndex] ~ INLINE { RETURN [FTIndex.FIRST + n*FTRecord.SIZE]}; MapFile: PROC [fti: FTIndex] RETURNS [FTIndex] ~ { SELECT fti FROM FTNull, FTSelf => RETURN [fti]; ENDCASE => { fileIndex: CARDINAL ~ IndexForFti[fti]; IF fileMap[fileIndex] = FTNull THEN fileMap[fileIndex] ← bcd.MergeFile[fti]; RETURN [fileMap[fileIndex]]}}; -- header processing InitHeader: PROC [ header: BcdOps.BcdBase, objectVersion: VersionStamp, source: NameRecord ← NullName, sourceVersion: VersionStamp ← NullVersion] ~ { Zero[header, BcdDefs.BCD.SIZE]; header.versionIdent ← BcdDefs.VersionID; header.version ← objectVersion; header.creator ← data.binderVersion; header.definitions ← (data.op = $conc); header.typeExported ← data.typeExported; header.source ← source; header.sourceVersion ← sourceVersion; header.repackaged ← table.Bounds[sptype].size # 0 OR table.Bounds[fptype].size # 0; header.tableCompiled ← FALSE; header.spare1 ← TRUE}; codeMap, symbolMap: LONG POINTER TO Map ← NIL; Map: TYPE ~ RECORD [ fti: FTIndex, type: SegClass, filename: Strings.String, filehandle: File.Capability]; InitCodeSymbolCopy: PROC ~ { Setup: PROC [file: Strings.String, type: SegClass] RETURNS [LONG POINTER TO Map] ~ { RETURN [(data.zone).NEW[Map ← [ type~type, filename~file, filehandle~nullFile, fti~IF file = NIL THEN FTSelf ELSE BcdUtilDefs.EnterFile[file]]]]}; IF data.copyCode THEN codeMap ← Setup[data.codeName, $code]; IF data.copySymbols THEN symbolMap ← Setup[data.symbolName, $symbols]}; MapCodeSymbolFiles: PROC ~ { IF data.copyCode THEN codeMap.fti ← MapFile[codeMap.fti]; IF data.copySymbols THEN symbolMap.fti ← MapFile[symbolMap.fti]}; ReleaseCodeSymbolCopy: PROC ~ { Release: PROC [p: LONG POINTER TO Map] ~ { p.filehandle ← nullFile}; IF codeMap # NIL THEN {Release[codeMap]; (data.zone).FREE[@codeMap]}; IF symbolMap # NIL THEN {Release[symbolMap]; (data.zone).FREE[@symbolMap]}}; EnumerateSegments: PROC [proc: PROC[SGIndex]] ~ { sgLimit: SGIndex ~ table.Top[sgtype]; FOR sgi: SGIndex ← SGIndex.FIRST, sgi + SGRecord.SIZE UNTIL sgi = sgLimit DO proc[sgi] ENDLOOP}; EnumerateOldSegments: PROC [proc: PROC[SGIndex]] ~ { IF ~packing THEN EnumerateSegments[proc] ELSE FOR i: NAT IN [0..sgMap.length) DO proc[SgiForIndex[i]] ENDLOOP}; MapSegments: PROC [type: SegClass] ~ { CopySegment: PROC [sgi: SGIndex] ~ { IF sgb[sgi].class = type THEN sgb[sgi].file ← MapFile[sgb[sgi].file]}; EnumerateSegments[CopySegment]}; InitFile: PROC [p: LONG POINTER TO Map, copiedPages: CARDINAL] RETURNS [stream: Stream.Handle, page: CARDINAL] ~ { lh: BcdDefs.BCD; bcdPages: CARDINAL ~ PagesForWords[BcdDefs.BCD.SIZE]; version: VersionStamp ~ BumpVersion[ OSMiscOps.StampToTime[data.objectStamp], (IF p.type=$code THEN 1 ELSE 2)]; BcdUtilDefs.SetFileVersion[p.fti, version]; p.filehandle ← OSMiscOps.FindFile[p.filename, $write]; (p.filehandle).SetSize[1 + bcdPages + copiedPages]; stream ← FileStream.Create[p.filehandle]; InitHeader[header~@lh, objectVersion~version]; lh.version ← ftb[p.fti].version; stream.PutBlock[[@lh, 0, BcdDefs.BCD.SIZE*bytesPerWord]]; page ← 1 + bcdPages}; BumpVersion: PROC [v: VersionStamp, n: CARDINAL] RETURNS [VersionStamp] ~ { v.time ← v.time + n; RETURN [v]}; MoveToPageBoundary: PROC [stream: Stream.Handle, page: CARDINAL] ~ { FileStream.SetIndex[stream, Inline.LongMult[page, Environment.bytesPerPage]]}; -- Code Packing PackHandle: TYPE ~ LONG POINTER TO PackItem; PackItem: TYPE ~ RECORD [ link: PackHandle, newsgi: SGIndex, -- in the new table item: SEQUENCE count: CARDINAL OF MTIndex]; packing: BOOL; phHead, phTail: PackHandle ← NIL; MakePackItem: Tree.Scan ~ { -- t is Tree.null, a list of ids, or a list of lists of ids ph: PackHandle ← NIL; i, nsons: CARDINAL; Item: Tree.Scan ~ { itemId: Symbols.HTIndex; WITH t SELECT FROM symbol => { itemId ← stb[index].hti; WITH stb[index] SELECT FROM external => WITH m~~map SELECT FROM module => { ph.item[i] ← m.mti; SELECT TRUE FROM ~mtb[m.mti].packageable => BcdErrorDefs.ErrorNameBase[ $error, "is packaged and cannot be PACKed"L, mtb[m.mti].name, bcd.ssb]; (ReadSgMap[mtb[m.mti].code.sgi] # SGNull) => BcdErrorDefs.ErrorNameBase[ $error, "cannot be PACKed twice"L, mtb[m.mti].name, bcd.ssb]; ENDCASE => SetSgMap[old~mtb[m.mti].code.sgi, new~ph.newsgi]}; ENDCASE => GOTO cant; ENDCASE => GOTO cant; EXITS cant => BcdErrorDefs.ErrorHti[$error, "cannot be PACKed"L, itemId ! BcdErrorDefs.GetSti => {RESUME [errorSti~Symbols.stNull]}]}; ENDCASE; i ← i+1}; IF t = Tree.null THEN RETURN; WITH t SELECT FROM subtree => { OPEN tt~~tb[index]; IF tt.name # $list THEN Error[]; data.textIndex ← tt.info; IF tt.son[1].tag = $subtree THEN {TreeOps.ScanList[t,MakePackItem]; RETURN}}; ENDCASE; nsons ← TreeOps.ListLength[t]; ph ← (data.zone).NEW[PackItem[nsons] ← [ link~NIL, newsgi~table.Words[sgtype, SGRecord.SIZE], item~]]; FOR j: CARDINAL IN [0..nsons) DO ph.item[j] ← MTNull ENDLOOP; sgb[ph.newsgi] ← [class~$code, file~codeMap.fti, base~0, pages~0, extraPages~0]; i ← 0; TreeOps.ScanList[t, Item]; IF phTail = NIL THEN phHead ← phTail ← ph ELSE {phTail.link ← ph; phTail ← ph}}; FreePackItems: PROC ~ { next: PackHandle ← phHead; UNTIL next = NIL DO p: PackHandle ← next; next ← next.link; (data.zone).FREE[@p] ENDLOOP; phHead ← phTail ← NIL}; WriteFromPages: PROC [ stream: Stream.Handle, pages: FileSegment.Pages, words: CARDINAL] ~ { bufferPages: CARDINAL ~ MIN[pages.span.pages, 16]; bufferSpace: Space.Handle ← Space.Create[size~bufferPages, parent~Space.virtualMemory]; base: CARDINAL ← pages.span.base; WHILE words # 0 DO wordsToTransfer: CARDINAL ~ MIN[words, bufferPages*Environment.wordsPerPage]; pagesToTransfer: CARDINAL ~ PagesForWords[wordsToTransfer]; bufferSpace.Map[[file~pages.file, base~base]]; stream.PutBlock[[bufferSpace.LongPointer, 0, wordsToTransfer*bytesPerWord]]; base ← (base + pagesToTransfer); words ← (words - wordsToTransfer); bufferSpace.Unmap[]; ENDLOOP; Space.Delete[bufferSpace]}; PackCodeSegments: PROC [out: Stream.Handle, startpage: CARDINAL] RETURNS [nextpage: CARDINAL] ~ { offset, validlength: CARDINAL; oldsgi: SGIndex; file: File.Capability; pages: FileSegment.Pages; FixUpOneModule: PROC [mti: MTIndex] RETURNS [BOOL ← FALSE] ~ { OPEN module~~mtb[mti]; length: CARDINAL; IF module.code.sgi = oldsgi THEN { length ← module.code.offset + module.code.length/2; module.code.offset ← module.code.offset + offset; module.code.packed ← TRUE; IF length > validlength THEN validlength ← length}; RETURN}; nextpage ← startpage; FOR ph: PackHandle ← phHead, ph.link UNTIL ph = NIL DO MoveToPageBoundary[stream~out, page~(nextpage-1)]; offset ← 0; sgb[ph.newsgi].base ← nextpage; FOR pi: CARDINAL IN [0..ph.count) DO { mti: MTIndex ~ ph.item[pi]; IF mtb[mti].linkLoc = $code AND ~mtb[mti].code.linkspace THEN { offset ← (AddLinksToCodeSegment[out, mti, offset, TRUE] + offset); GOTO ignore}; oldsgi ← mtb[mti].code.sgi; [file, pages] ← SegmentForOldCodeSgi[oldsgi]; IF file = nullFile THEN GOTO ignore; IF (offset MOD Alignment) # 0 THEN FOR i: CARDINAL IN [(offset MOD Alignment)..Alignment) DO out.PutWord[0]; offset ← offset + 1 ENDLOOP; validlength ← 0; EnumerateModules[FixUpOneModule]; WriteFromPages[stream~out, pages~pages, words~validlength]; offset ← offset + validlength; EXITS ignore => NULL}; ENDLOOP; sgb[ph.newsgi].pages ← PagesForWords[offset]; nextpage ← nextpage + sgb[ph.newsgi].pages; ENDLOOP; RETURN}; SegmentForOldCodeSgi: PROC [sgi: SGIndex] RETURNS [f: File.Capability ← nullFile, s: FileSegment.Pages ← nullPages] ~ { OPEN seg~~bcd.sgb[sgi]; IF Copied[sgi] OR seg.file = FTNull THEN RETURN; f ← FileForFti[seg.file]; IF f = nullFile THEN BcdErrorDefs.ErrorNameBase[ class~$error, s~"could not be opened to copy code"L, name~bcd.ftb[seg.file].name, base~bcd.ssb] ELSE { s ← [ file~f, span~[base~seg.base, pages~(seg.pages + seg.extraPages)]]; IF WrongOldSegVersion[s, bcd.ftb[seg.file].version] THEN { BcdErrorDefs.ErrorNameBase[ class~$error, s~"on disk has an incorrect version"L, name~bcd.ftb[seg.file].name, base~bcd.ssb]; f ← nullFile; s ← nullPages}}; IF s.file = File.nullCapability THEN header.versionIdent ← 0; SetCopied[sgi]}; WrongOldSegVersion: PROC [s: FileSegment.Pages, version: BcdDefs.VersionStamp] RETURNS [reply: BOOL] ~ { h: BcdOps.BcdBase; headerSpace: Space.Handle ← Space.Create[size~1, parent~Space.virtualMemory]; headerSpace.Map[[file~s.file, base~1]]; h ← headerSpace.LongPointer; reply ← (h.version # version); Space.Delete[headerSpace]; RETURN [reply]}; -- Segment Mapping SGMap: TYPE ~ RECORD [SEQUENCE length: CARDINAL OF SGIndex]; CopyMap: TYPE ~ RECORD [SEQUENCE length: CARDINAL OF BOOL]; sgMap: LONG POINTER TO SGMap ← NIL; copyMap: LONG POINTER TO CopyMap ← NIL; IndexForSgi: PROC [sgi: SGIndex] RETURNS [CARDINAL] ~ INLINE { RETURN [LOOPHOLE[sgi,CARDINAL]/SGRecord.SIZE]}; SgiForIndex: PROC [i: CARDINAL] RETURNS [SGIndex] ~ INLINE { RETURN [SGIndex.FIRST + i*SGRecord.SIZE]}; InitCopyMap: PROC [nsgis: CARDINAL] ~ { copyMap ← (data.zone).NEW[CopyMap[nsgis]]; FOR i: CARDINAL IN [0..nsgis) DO copyMap[i] ← FALSE ENDLOOP}; FreeCopyMap: PROC ~ { IF copyMap # NIL THEN (data.zone).FREE[@copyMap]}; SetCopied: PROC [sgi: SGIndex] ~ { copyMap[IndexForSgi[sgi]] ← TRUE}; Copied: PROC [sgi: SGIndex] RETURNS [BOOL] ~ { RETURN [copyMap[IndexForSgi[sgi]]]}; InitSgMap: PROC [nsgis: CARDINAL] ~ { sgMap ← (data.zone).NEW[SGMap[nsgis]]; FOR i: CARDINAL IN [0..nsgis) DO sgMap[i] ← BcdDefs.SGNull ENDLOOP}; FreeSgMap: PROC ~ {IF sgMap # NIL THEN (data.zone).FREE[@sgMap]}; SetSgMap: PROC [old, new: SGIndex] ~ { IF (packing AND old # SGNull) THEN sgMap[IndexForSgi[old]] ← new}; ReadSgMap: PROC [old: SGIndex] RETURNS [SGIndex] ~ { RETURN [IF (~packing OR old = SGNull) THEN old ELSE sgMap[IndexForSgi[old]]]}; FillInSgMap: PROC ~ { -- called only when packing (i.e. packing requested AND copyCode = TRUE) FOR i: CARDINAL IN [0..sgMap.length) DO IF sgMap[i] = SGNull THEN { oldsgi: SGIndex ~ SgiForIndex[i]; newsgi: SGIndex ~ table.Words[sgtype, SGRecord.SIZE]; sgb[newsgi] ← bcd.sgb[oldsgi]; sgb[newsgi].file ← (IF sgb[newsgi].class = $symbols THEN (IF data.copySymbols THEN symbolMap.fti ELSE MapFile[sgb[newsgi].file]) ELSE codeMap.fti); sgMap[i] ← newsgi}; ENDLOOP}; FixAllSgis: PROC ~ { -- replace all sgis with ReadSgMap[sgi] FixModule: PROC [mti: MTIndex] RETURNS [BOOL ← FALSE] ~ { OPEN m~~mtb[mti]; m.code.sgi ← ReadSgMap[m.code.sgi]; m.sseg ← ReadSgMap[m.sseg]}; FixSpace: PROC [spi: SPIndex] RETURNS [BOOL ← FALSE] ~ { OPEN sp~~spb[spi]; sp.seg ← ReadSgMap[sp.seg]}; EnumerateModules[FixModule]; EnumerateSpaces[FixSpace]}; -- Code Links LinkCount: PROC [mti: MTIndex] RETURNS [CARDINAL] ~ INLINE { RETURN [IF mtb[mti].links = LFNull THEN 0 ELSE lfb[mtb[mti].links].length]}; AlignOffset: PROC [offset: CARDINAL] RETURNS [CARDINAL] ~ INLINE { RETURN [((offset + (Alignment-1))/Alignment)*Alignment]}; AddLinksToCodeSegment: PROC [ stream: Stream.Handle, mti: MTIndex, offset: CARDINAL, packed: BOOL] RETURNS [CARDINAL] ~ { sgi: SGIndex ~ mtb[mti].code.sgi; codeLength: CARDINAL ~ mtb[mti].code.length/2; linkSpace: CARDINAL; f: File.Capability; s: FileSegment.Pages; prefixWords: CARDINAL ← 0; FixOffset: PROC [mti: MTIndex] RETURNS [BOOL ← FALSE] ~ { OPEN c~~mtb[mti].code; IF c.sgi = sgi THEN { c.linkspace ← TRUE; c.offset ← c.offset+offset; c.packed ← packed}}; [f, s] ← SegmentForOldCodeSgi[sgi]; IF f = nullFile THEN RETURN [0]; linkSpace ← LinkCount[mti]; IF offset = 0 AND linkSpace # 0 THEN { prefixWords ← 1; stream.PutWord[linkSpace + Alignment - (linkSpace MOD Alignment)]; offset ← offset+1}; IF (offset+linkSpace) MOD Alignment # 0 THEN linkSpace ← linkSpace + Alignment - ((offset+linkSpace) MOD Alignment); offset ← offset + linkSpace; EnumerateModules[FixOffset]; FOR i: CARDINAL IN [0..linkSpace) DO stream.PutWord[0] ENDLOOP; WriteFromPages[stream~stream, pages~s, words~codeLength]; RETURN [prefixWords + linkSpace + codeLength]}; -- code and symbol copying EstimateCopiedPages: PROC RETURNS [codePages, symbolPages: CARDINAL ← 0] ~ { -- estimates ignore possible packing of code packaged: BOOL ← FALSE; AddModule: PROC [mti: MTIndex] RETURNS [BOOL ← FALSE] ~ { OPEN m~~mtb[mti]; IF data.copyCode AND m.code.sgi # SGNull THEN { OPEN seg~~bcd.sgb[m.code.sgi]; IF ~m.packageable THEN packaged ← TRUE ELSE { IF m.linkLoc = $code AND ~m.code.linkspace THEN { nLinks: CARDINAL ~ LinkCount[mti]; offset: CARDINAL ~ AlignOffset[IF nLinks=0 THEN 0 ELSE 1+nLinks]; codePages ← codePages + PagesForWords[offset + m.code.length/2]} ELSE codePages ← codePages + seg.pages; codePages ← codePages + seg.extraPages}}; IF data.copySymbols AND m.sseg # SGNull THEN { OPEN seg~~bcd.sgb[m.sseg]; symbolPages ← symbolPages + seg.pages + seg.extraPages}; RETURN}; AddSegment: PROC [oldSgi: SGIndex] ~ { OPEN seg~~bcd.sgb[oldSgi]; IF seg.class = $code THEN { package: BOOL ← FALSE; TestModule: PROC [mti: MTIndex] RETURNS [BOOL] ~ { OPEN m~~mtb[mti]; IF ~m.packageable AND m.code.sgi = oldSgi THEN package ← TRUE; RETURN [m.code.sgi = oldSgi]}; EnumerateModules[TestModule]; IF package THEN codePages ← codePages + seg.pages + seg.extraPages}}; IF data.copyCode OR data.copySymbols THEN EnumerateModules[AddModule]; IF data.copyCode AND packaged THEN EnumerateOldSegments[AddSegment]; RETURN}; MoveCodeSegments: PROC [copiedPages: CARDINAL] ~ { stream: Stream.Handle; nextPage: CARDINAL; AddLinks: PROC [mti: MTIndex] RETURNS [BOOL ← FALSE] ~ { OPEN m~~mtb[mti]; wordsWritten, pagesWritten: CARDINAL; newSgi: SGIndex; -- IF ExecOps.CheckForAbort[] THEN ERROR UserAbort; IF m.linkLoc = $code AND ~m.code.linkspace AND m.packageable THEN { IF m.code.packed THEN BcdErrorDefs.ErrorName[ $error, "was previously PACKed and can not now have code links added"L, m.name] ELSE { MoveToPageBoundary[stream~stream, page~(nextPage-1)]; wordsWritten ← AddLinksToCodeSegment[ stream~stream, mti~mti, offset~0, packed~FALSE]; pagesWritten ← PagesForWords[wordsWritten]; newSgi ← ReadSgMap[m.code.sgi]; sgb[newSgi].file ← codeMap.fti; sgb[newSgi].base ← nextPage; sgb[newSgi].pages ← pagesWritten; nextPage ← nextPage + pagesWritten}}}; MoveOne: PROC [oldSgi: SGIndex] ~ { OPEN seg~~bcd.sgb[oldSgi]; -- IF ExecOps.CheckForAbort[] THEN ERROR UserAbort; IF seg.class = $code THEN { f: File.Capability; s: FileSegment.Pages; [f, s] ← SegmentForOldCodeSgi[oldSgi]; IF f # nullFile THEN { segPages: CARDINAL ~ s.span.pages; newSgi: SGIndex ~ ReadSgMap[oldSgi]; sgb[newSgi].file ← codeMap.fti; sgb[newSgi].base ← nextPage; MoveToPageBoundary[stream~stream, page~(nextPage-1)]; WriteFromPages[ stream~stream, pages~s, words~(segPages * Environment.wordsPerPage)]; nextPage ← nextPage + segPages}}}; IF codeMap.fti = FTSelf THEN {stream ← bcdStream; nextPage ← nextBcdPage} ELSE [stream, nextPage] ← InitFile[codeMap, copiedPages]; nextPage ← PackCodeSegments[stream, nextPage]; EnumerateModules[AddLinks]; EnumerateOldSegments[MoveOne]; IF codeMap.fti = FTSelf THEN nextBcdPage ← nextPage ELSE Stream.Delete[stream]}; EnterMissingSymbolFiles: PROC ~ { CheckOneSymbolsFileSeg: PROC [oldSgi: SGIndex] ~ { OPEN seg~~bcd.sgb[oldSgi]; f: File.Capability ← File.nullCapability; IF (seg.class = $symbols) AND ~Copied[oldSgi] AND (seg.file # FTNull) THEN -- insure that a file entry exists for this file [] ← MapFile[bcd.sgb[oldSgi].file]}; EnumerateOldSegments[CheckOneSymbolsFileSeg]}; MoveSymbolSegments: PROC [copiedPages: CARDINAL] ~ { stream: Stream.Handle; nextPage: CARDINAL; MoveOne: PROC [oldSgi: SGIndex] ~ { OPEN seg~~bcd.sgb[oldSgi]; f: File.Capability ← nullFile; newSgi: SGIndex; -- IF ExecOps.CheckForAbort[] THEN ERROR UserAbort; IF (seg.class # $symbols) OR Copied[oldSgi] OR (seg.file = FTNull) THEN RETURN; newSgi ← ReadSgMap[oldSgi]; f ← FileForFti[seg.file]; IF f = nullFile THEN { BcdErrorDefs.ErrorNameBase[ class~$warning, s~"could not be opened to copy symbols"L, name~bcd.ftb[seg.file].name, base~bcd.ssb]; sgb[newSgi] ← bcd.sgb[oldSgi]; sgb[newSgi].file ← MapFile[bcd.sgb[oldSgi].file]} ELSE { s: FileSegment.Pages ~ [ file~f, span~[base~seg.base, pages~(seg.pages + seg.extraPages)]]; IF WrongOldSegVersion[s, bcd.ftb[seg.file].version] THEN { BcdErrorDefs.ErrorNameBase[ class~$error, s~"on disk has incorrect version"L, name~bcd.ftb[seg.file].name, base~bcd.ssb]; header.versionIdent ← 0} ELSE { segPages: CARDINAL ~ s.span.pages; sgb[newSgi].file ← symbolMap.fti; sgb[newSgi].base ← nextPage; MoveToPageBoundary[stream~stream, page~(nextPage-1)]; WriteFromPages[ stream~stream, pages~s, words~(segPages * Environment.wordsPerPage)]; nextPage ← nextPage + segPages}}; SetCopied[oldSgi]}; IF symbolMap.fti = FTSelf THEN {stream ← bcdStream; nextPage ← nextBcdPage} ELSE [stream, nextPage] ← InitFile[symbolMap, copiedPages]; EnumerateOldSegments[MoveOne]; IF symbolMap.fti = FTSelf THEN nextBcdPage ← nextPage ELSE Stream.Delete[stream]}; FileForFti: PROC [oldFti: BcdDefs.FTIndex] RETURNS [f: File.Capability] ~ { name: BcdDefs.NameRecord ~ bcd.ftb[oldFti].name; ssd: Strings.SubStringDescriptor ← [ base~@bcd.ssb.string, offset~name, length~bcd.ssb.size[name]]; nameStr: Strings.String ← [100]; f ← nullFile; NormalizeFileName[in~@ssd, out~nameStr]; f ← OSMiscOps.FindFile[nameStr, $read ! OSMiscOps.FileError => {CONTINUE}]; RETURN}; NormalizeFileName: PROC [in: Strings.SubString, out: Strings.String] ~ { char: CHAR; dot: BOOL ← FALSE; out.length ← 0; FOR i: CARDINAL IN [in.offset .. in.offset+in.length) DO SELECT (char ← in.base[i]) FROM IN ['A..'Z] => char ← VAL[(char.ORD-'A.ORD) + 'a.ORD]; '. => dot ← TRUE; ENDCASE; Strings.AppendChar[out, char]; ENDLOOP; IF ~dot THEN Strings.AppendString[out, ".bcd"L]}; -- Bcd Output Routines bcdStream: Stream.Handle; nextBcdPage: CARDINAL; WriteSubTable: PROC [selector: Table.Selector] ~ { base: Table.Base; size: CARDINAL; [base, size] ← table.Bounds[selector]; bcdStream.PutBlock[[base, 0, size*bytesPerWord]]}; TableOut: PROC ~ { d, s: CARDINAL; bcdPages, codePages, symbolPages: CARDINAL; basePages: CARDINAL; rtPageCount: CARDINAL; saveNextPage: CARDINAL; saveIndex: FileStream.FileByteIndex; rtPageCount ← PagesForWords[BcdLiterals.LitSegSize[]]; BEGIN OPEN header; IF firstdummy = 0 THEN firstdummy ← BcdUtilDefs.GetGfi[0]; d ← BCD.SIZE; ssOffset ← d; d ← d + (ssLimit ← table.Bounds[sstype].size); ctOffset ← d; d ← d + (s ← table.Bounds[cttype].size); ctLimit ← LOOPHOLE[s]; mtOffset ← d; d ← d + (s ← table.Bounds[mttype].size); mtLimit ← LOOPHOLE[s]; impOffset ← d; d ← d + (s ← table.Bounds[imptype].size); impLimit ← LOOPHOLE[s]; expOffset ← d; d ← d + (s ← table.Bounds[exptype].size); expLimit ← LOOPHOLE[s]; evOffset ← d; d ← d + (s ← table.Bounds[evtype].size); evLimit ← LOOPHOLE[s, EVIndex]; sgOffset ← d; d ← d + (s ← table.Bounds[sgtype].size); sgLimit ← LOOPHOLE[s]; ftOffset ← d; d ← d + (s ← table.Bounds[fttype].size); ftLimit ← LOOPHOLE[s]; ntOffset ← d; d ← d + (s ← table.Bounds[nttype].size); ntLimit ← LOOPHOLE[s]; typOffset ← d; d ← d + (s ← table.Bounds[typtype].size); typLimit ← LOOPHOLE[s]; tmOffset ← d; d ← d + (s ← table.Bounds[tmtype].size); tmLimit ← LOOPHOLE[s]; spOffset ← d; d ← d + (s ← table.Bounds[sptype].size); spLimit ← LOOPHOLE[s]; fpOffset ← d; d ← d + (s ← table.Bounds[fptype].size); fpLimit ← LOOPHOLE[s]; lfOffset ← d; d ← d + (s ← table.Bounds[lftype].size); lfLimit ← LOOPHOLE[s]; rfOffset ← d; d ← d + (s ← table.Bounds[rftype].size); rfLimit ← LOOPHOLE[s]; tfOffset ← d; d ← d + (s ← table.Bounds[tftype].size); tfLimit ← LOOPHOLE[s]; basePages ← PagesForWords[d]; rtPages ← [relPageBase~basePages, pages~rtPageCount]; nPages ← bcdPages ← basePages + rtPageCount; END; [codePages, symbolPages] ← EstimateCopiedPages[]; IF data.copyCode AND codeMap.fti = FTSelf THEN bcdPages ← bcdPages + codePages; IF data.copySymbols AND symbolMap.fti = FTSelf THEN bcdPages ← bcdPages + symbolPages; OpenOutputFile[1 + bcdPages]; IF rtPageCount # 0 THEN { MoveToPageBoundary[stream~bcdStream, page~basePages]; IF packing THEN BcdLiterals.UpdateSegments[ReadSgMap]; BcdLiterals.WriteLiterals[bcdStream]; saveIndex ← FileStream.GetIndex[bcdStream]}; saveNextPage ← nextBcdPage ← header.nPages + 1; IF data.copyCode THEN MoveCodeSegments[codePages ! UserAbort => {GO TO AbortRequested}]; IF data.copySymbols THEN MoveSymbolSegments[symbolPages ! UserAbort => {GO TO AbortRequested}]; IF packing THEN FixAllSgis[]; FileStream.SetIndex[bcdStream, 0]; bcdStream.PutBlock[[header, 0, BCD.SIZE*bytesPerWord]]; WriteSubTable[sstype]; WriteSubTable[cttype]; WriteSubTable[mttype]; WriteSubTable[imptype]; WriteSubTable[exptype]; WriteSubTable[evtype]; WriteSubTable[sgtype]; WriteSubTable[fttype]; WriteSubTable[nttype]; WriteSubTable[typtype]; WriteSubTable[tmtype]; WriteSubTable[sptype]; WriteSubTable[fptype]; WriteSubTable[lftype]; WriteSubTable[rftype]; WriteSubTable[tftype]; IF nextBcdPage # saveNextPage THEN MoveToPageBoundary[stream~bcdStream, page~(nextBcdPage-1)] ELSE IF rtPageCount # 0 THEN FileStream.SetIndex[bcdStream, saveIndex]; data.nConfigs ← header.nConfigs; data.nModules ← header.nModules; data.nImports ← header.nImports; data.nExports ← header.nExports; data.nPages ← header.nPages; EXITS AbortRequested => data.aborted ← TRUE}; OpenOutputFile: PROC [initialPages: CARDINAL] ~ INLINE { file: File.Capability ← data.outputFile; IF file = File.nullCapability THEN file ← data.outputFile ← OSMiscOps.FindFile[data.bcdName, $write]; file.SetSize[initialPages]; bcdStream ← FileStream.Create[file]}; CloseOutputFile: PROC ~ INLINE {Stream.Delete[bcdStream]; bcdStream ← NIL}; }.