CDDirectoryOpsImpl.mesa (part of ChipNDale)
Copyright © 1985, 1986, 1987 by Xerox Corporation. All rights reserved.
Created by Christian Jacobi, June 3, 1985 7:06:54 pm PDT
Last edited by: Christian Jacobi, October 19, 1987 2:36:41 pm PDT
DIRECTORY
Atom,
CD,
CDCells,
CDDirectory,
CDDirectoryOps,
CDOps,
CDProperties,
CDSequencer,
CDValue,
CedarProcess,
Process,
RefTab,
Rope,
RuntimeError USING [UNCAUGHT],
TerminalIO,
TextFind;
CDDirectoryOpsImpl: CEDAR MONITOR
IMPORTS Atom, CD, CDCells, CDDirectory, CDOps, CDProperties, CDSequencer, CDValue, CedarProcess, Process, RefTab, Rope, RuntimeError, TerminalIO, TextFind
EXPORTS CDDirectoryOps
SHARES CD, CDDirectory =
BEGIN
QuitOnData: CDDirectory.EachObjectProc = {
quit ← me=data
};
DontQuit: CDDirectory.EachObjectProc = {
quit ← FALSE
};
RemoveIfUnused: PUBLIC PROC [design: CD.Design, ob: CD.Object] RETURNS [done: BOOLFALSE, msg: Rope.ROPENIL] = {
EachDirObject: CDDirectory.EachObjectProc = {
quit ← CDDirectory.EnumerateObject[ob: me, proc: QuitOnData, data: data, recurse: TRUE, visited: seen];
};
found: BOOL;
seen: RefTab.Ref ← RefTab.Create[];
name: Rope.ROPE ← CDDirectory.Name[ob, design];
toDeleteOb: CD.Object ← CDDirectory.Fetch[design, name];
IF toDeleteOb=NIL THEN RETURN [done ← FALSE, msg ← "object was not in directory"];
IF toDeleteOb#ob THEN RETURN [done ← FALSE, msg ← "naming problem"];
IF ~CDDirectory.IsIncluded[design, ob] THEN
RETURN [done ← FALSE, msg ← "object was not in directory"];
found ← CDDirectory.EnumerateDesign[design: design, proc: QuitOnData, visited: seen, data: toDeleteOb, dir: FALSE];
IF found THEN
RETURN [done ← FALSE, msg ← "is used in design"];
found ← CDDirectory.EnumerateDesign[design: design, proc: EachDirObject, visited: seen, data: toDeleteOb, dir: TRUE, recurse: FALSE, dummy: FALSE];
IF found THEN
RETURN [done ← FALSE, msg ← "is used by other object in directory of design"];
CDOps.FlushRemember[design];
[] ← CDDirectory.Remove[design, name, toDeleteOb];
RETURN [done ← TRUE, msg ← NIL];
};
CompletelyDestroy: PROC [design: CD.Design, list: LIST OF CD.Object] = {
cnt: INT ← 0;
tem: LIST OF CD.Object;
IF list#NIL THEN {
CDOps.FlushRemember[design];
WHILE list#NIL DO
IF list.first.class.composed AND ~list.first.immutable THEN list.first^.properties ← NIL;
list.first ← NIL; tem ← list; list ← list.rest; tem.rest ← NIL;
IF (cnt𡤌nt+1)>1000 THEN RETURN; --protect zct table
ENDLOOP
}
};
PruneDirectory: PUBLIC PROC [design: CD.Design, askFirst: BOOLFALSE, pattern: Rope.ROPENIL] = {
RemoveUnusedObs: PROC [name: Rope.ROPE, ob: CD.Object] RETURNS [quit: BOOLFALSE] = {
IF ~RefTab.Fetch[instanced, ob].found THEN {
ob ← CDDirectory.Remove[design: design, name: name, expectObject: ob];
IF ob#NIL THEN removed ← CONS[ob, removed];
}
};
ListUnusedObs: PROC [name: Rope.ROPE, ob: CD.Object] RETURNS [quit: BOOLFALSE] = {
IF ~RefTab.Fetch[instanced, ob].found THEN {
TerminalIO.PutF1[" - %g\n", [rope[name]]];
cnt ← cnt + 1
}
};
VisitNamedObs: PROC [name: Rope.ROPE, ob: CD.Object] RETURNS [quit: BOOLFALSE] = {
IF ~TextFind.SearchRope[finder: finder, rope: name].found THEN {
[] ← CDDirectory.EnumerateObject[ob: ob, proc: DontQuit, visited: instanced];
[] ← RefTab.Insert[instanced, ob, $x];
}
};
finder: TextFind.Finder ← NIL;
instanced: RefTab.Ref ← RefTab.Create[];
removed: LIST OF CD.Object ← NIL;
cnt: NAT ← 0;
IF pattern#NIL THEN finder ← TextFind.CreateFromRope[pattern: pattern, ignoreCase: TRUE];
[] ← CDDirectory.EnumerateDesign[design: design, proc: DontQuit, visited: instanced, dir: FALSE];
IF finder#NIL THEN {
[] ← CDDirectory.Enumerate[design: design, action: VisitNamedObs];
};
IF askFirst THEN {
TerminalIO.PutRope[" list of matching objects not used:\n"];
[] ← CDDirectory.Enumerate[design: design, action: ListUnusedObs];
TerminalIO.PutF1[" %g objects not used in design\n", [integer[cnt]]];
};
IF cnt>0 THEN {
IF askFirst AND ~TerminalIO.Confirm["delete listed objects"] THEN {
removed ← NIL;
TerminalIO.PutRope[" not done\n"];
RETURN
};
CDSequencer.MarkChanged[design];
CDOps.FlushRemember[design];
[] ← CDDirectory.Enumerate[design: design, action: RemoveUnusedObs];
CompletelyDestroy[design, removed];
IF askFirst THEN TerminalIO.PutRope[" deleted\n"];
};
removed ← NIL;
};
RenameNRemove: PUBLIC PROC [design: CD.Design, ob: CD.Object, name: Rope.ROPE] = {
ob1: CD.Object;
ob1 ← CDDirectory.Fetch[design, name].object;
IF ob1#NIL THEN {
IF ob1=ob THEN RETURN;
[] ← CDDirectory.Rename[design: design, object: ob1, newName: Rope.Cat[name, "@old"], fiddle: TRUE];
};
[] ← CDDirectory.Rename[design: design, object: ob, newName: name, fiddle: TRUE];
IF ob1#NIL THEN [] ← RemoveIfUnused[design: design, ob: ob1];
};
ReplaceFunnys: PROC [design: CD.Design, rootOb: CD.Object, key: REF] = {
--Replaces all object with funny classes in place,
--hopes that this is allowed... but it does not know funny classes and their restrictions
--IF rootOb#NIL only rootOb and its descendants are handled
--Of course, there shouldn't be any funny classes...
--key: a property key which will be used freely... NIL is ok
realKey: REF ~ IF key=NIL THEN NEW[INT] ELSE key;
DirtyExpand: PROC [ob: CD.Object] = {
WHILE ob.class.internalWrite=NIL AND ~ob.immutable DO
--real dirty, but it never happens because internalWrite is never NIL, except...
new: CD.Object; ta, ca: BOOL; own: REF;
[new, ta, ca] ← CDDirectory.Expand1[ob]; --we ignore modes; class is fishy
own ← CDProperties.GetObjectProp[ob, $OwnerDesign];
IF new=NIL THEN {
TerminalIO.PutF["**** %g is fishy\n", [rope[CD.Describe[ob, NIL, design]]]];
EXIT;
};
CDProperties.AppendProps[looser: new.properties, winner: ob.properties, putOnto: new];
TerminalIO.PutF["* %g has been modified in all designs\n", [rope[CD.Describe[ob, NIL, design]]]];
ob^ ← new^; --real dirty--
CDProperties.PutObjectProp[ob, $OwnerDesign, own];
CDCells.SetSimplificationTreshhold[ob, 50];
ENDLOOP;
};
DoIt: CDDirectory.EachObjectProc = {
--uses globals: realKey
IF me.class.composed AND CDProperties.GetObjectProp[me, realKey]#data THEN {
CDProperties.PutProp[me, realKey, data];
DirtyExpand[me];
[] ← CDDirectory.EnumerateChildObjects[me, DoIt, data];
};
};
IF rootOb#NIL THEN [] ← DoIt[rootOb, NEW[INT]]
ELSE [] ← CDDirectory.EnumerateDesign[design, DoIt, NEW[INT]];
IF key=NIL THEN RemoveProperties[design, realKey];
};
PreTag: PROC [design: CD.Design, rootOb: CD.Object, tag: REF, key: REF] = {
--figures out whether objects have funny classes and should be replaced
--or whether objects should be replaced because they are in different designs:
--such objects will be taken, expanded or copied, included in the directory and put
--onto the tag property
DoIt: CDDirectory.EachObjectProc = {
--uses globals: design, tag, key
IF me.class.composed AND CDProperties.GetObjectProp[me, key]#data THEN {
useOb, new: CD.Object; ta, ca: BOOL;
UseNewAndTmToFinish: PROC [includeIfNecessary: BOOL] RETURNS [failed: BOOLFALSE] = {
--uses new, tm, useOb, me and tag
IF new=NIL OR new.class.internalWrite=NIL THEN {
new ← CDDirectory.Expand1ByDraw[useOb, CDDirectory.LeaveNextLevel];
};
IF new#NIL THEN {
[new, ca] ← CDDirectory.Another1[me: new, into: design, friendly: TRUE];
IF new=NIL OR new.class.internalWrite=NIL THEN {
new ← CDDirectory.Expand1ByDraw[useOb, CDDirectory.LeaveNextLevel];
};
};
IF new=NIL THEN {
TerminalIO.PutF["**** %g is fishy\n", [rope[CD.Describe[me, NIL, design]]]];
RETURN [failed←TRUE]
};
--ca: childs are copied or included when going down the main recursion
useOb ← new;
CDProperties.PutProp[me, tag, useOb];
};
CDProperties.PutProp[me, key, data]; --don't visit me a second time
new ← useOb ← me;
--fix funny classes
IF useOb.class.internalWrite=NIL THEN {--it never happens, except...
WHILE new#NIL AND new.class.internalWrite=NIL DO
[new, ta, ca] ← CDDirectory.Expand1[me: new, into: design, friendly: TRUE];
ENDLOOP;
IF UseNewAndTmToFinish[FALSE].failed THEN RETURN;
};
--fix funny designs
IF ~CDDirectory.CompatibleOwner[design, useOb] THEN {
[new, ca] ← CDDirectory.Another[me: useOb, into: design, friendly: TRUE];
IF UseNewAndTmToFinish[TRUE].failed THEN RETURN;
ERROR; --on Friday February 13, 1987 Brian, Bertrand and Rick agreed that ChipNDale might call an error in this case
};
CDDirectory.SetOwner[design, useOb, FALSE];
--go down hierarchy
[] ← CDDirectory.EnumerateChildObjects[useOb, DoIt, data];
};
};
IF rootOb#NIL THEN [] ← DoIt[rootOb, NEW[INT]]
ELSE [] ← CDDirectory.EnumerateDesign[design, DoIt, NEW[INT]];
};
ReplaceTaggedOneLevel: PROC [design: CD.Design, ob: CD.Object, tag: REF] = {
ForCells: PROC [cell: CD.Object, tag: REF] = INLINE {
EachInst: CDCells.InstEnumerator = {
WITH CDProperties.GetObjectProp[inst.ob, tag] SELECT FROM
ob: CD.Object => inst.ob ← ob;
ENDCASE => NULL;
};
[] ← CDCells.EnumerateInstances[cell, EachInst];
}; --ForCells
ForNonCells: PROC [design: CD.Design, ob: CD.Object, tag: REF] = INLINE --gfi--{
replaces: CDDirectory.ReplaceList ← NIL;
FindReplaces: CDDirectory.EachObjectProc = {
WITH CDProperties.GetObjectProp[me, tag] SELECT FROM
new: CD.Object => {
FOR l: CDDirectory.ReplaceList ← replaces, l.rest WHILE l#NIL DO
IF l.first.old=me THEN RETURN;
ENDLOOP;
replaces ← CONS[NEW[CDDirectory.ReplaceRec ← [old: me, new: new]], replaces];
};
ENDCASE => NULL;
}; --FindReplaces
--ForNonCells
[] ← CDDirectory.EnumerateChildObjects[ob, FindReplaces, NIL];
IF replaces#NIL THEN
[] ← CDDirectory.ReplaceDirectChild[ob, design, replaces];
}; --ForNonCells
--ReplaceTaggedOneLevel
IF CDCells.IsCell[ob] THEN ForCells[ob, tag] ELSE ForNonCells[design, ob, tag];
}; --ReplaceTaggedOneLevel
ReplaceTagged: PUBLIC PROC [design: CD.Design, replaceBy: REF, rename: BOOL TRUE, key: REF] = {
myVal: REF ~ NEW[INT];
realKey: REF ~ IF key=NIL THEN NEW[INT] ELSE key;
DoChilds: CDDirectory.EachObjectProc = {
--globals: design, realKey
--data: visit key
IF me.class.composed AND CDProperties.GetObjectProp[me, realKey]#data THEN {
CDProperties.PutProp[me, realKey, data];
ReplaceTaggedOneLevel[design, me, replaceBy];
[] ← CDDirectory.EnumerateChildObjects[me, DoChilds, data];
};
}; --DoChilds
EachEntry: CDDirectory.EachObjectProc = {
WITH CDProperties.GetObjectProp[me, replaceBy] SELECT FROM
new: CD.Object => {
name: Rope.ROPE ← CDDirectory.Name[me, design];
removed: CD.Object ← CDDirectory.Remove[design, name, me];
IF removed#NIL THEN {
[] ← DoChilds[me, myVal];
TerminalIO.PutRopes["** problem with replacing ", name, "\n"];
};
IF name#NIL THEN {
IF rename THEN [] ← CDDirectory.Rename[design: design, object: new, newName: name, fiddle: FALSE, removeFirst: TRUE];
};
me ← new;
};
ENDCASE => NULL;
[] ← DoChilds[me, myVal];
};
--ReplaceTagged
FOR pushed: LIST OF CD.PushRec ← design.actual, pushed.rest WHILE pushed#NIL DO
FOR list: CD.InstanceList ← pushed.first.specific.contents, list.rest WHILE list#NIL DO
WITH CDProperties.GetObjectProp[list.first.ob, replaceBy] SELECT FROM
ob: CD.Object => {
list.first.ob ← ob;
[] ← DoChilds[ob, myVal];
}
ENDCASE => NULL;
ENDLOOP;
ENDLOOP;
[] ← CDDirectory.EnumerateDesign[design, EachEntry];
IF key=NIL THEN RemoveProperties[design, realKey];
}; --ReplaceTagged
CleanUp: PUBLIC PROC [design: CD.Design, rootOb: CD.Object] = {
myKey: REF ATOM ~ NEW[ATOM←$UsedForCleanup];
replaceBy: REF ATOM ~ NEW[ATOM←$UsedForCleanup];
CDOps.FlushRemember[design];
PreTag[design, rootOb, replaceBy, myKey];
ReplaceTagged[design, replaceBy, TRUE, myKey];
RemoveProperties[design, myKey];
RemoveProperties[design, replaceBy];
myKey^ ← $MayBeRemoved;
replaceBy^ ← $MayBeRemoved;
};
--==== property removing =====
RemoveRecord: TYPE = RECORD [ --monitored; used to remember what properties to forget
running: BOOLFALSE, --a forget loop process is running right now
forgetNext: LIST OF REF ANYNIL --use next time in forget loop
];
DepositAndStart: ENTRY PROC [design: CD.Design, rr: REF RemoveRecord, key: REF] = {
--deposit property key in forget queue; starts forget loop process if necessary
IF rr#NIL THEN {
rr.forgetNext ← CONS[key, rr.forgetNext];
IF ~rr.running THEN {
rr.running ← TRUE;
TRUSTED {Process.Detach[FORK RemoveLoop[design, rr]]}
};
};
};
FetchAndStop: ENTRY PROC [rr: REF RemoveRecord] RETURNS [forgetNow: LIST OF REF ANY NIL] = {
--fetch and remove property keys from the forget queue
--if queue is empty: mark forget loop process as NOT running
IF rr#NIL THEN {
forgetNow ← rr.forgetNext; rr.forgetNext ← NIL;
IF forgetNow=NIL THEN rr.running ← FALSE;
}
};
RemoveLoop: PROC [design: CD.Design, rr: REF RemoveRecord] = {
--loops and forgets properties, as long as there are some
forgetNow: LIST OF REF ANY;
RemoveInner: PROC [design: CD.Design] = {
RPEachOb: CDDirectory.EachObjectProc = {
IF me.class.composed THEN
FOR l: LIST OF REF ANY ← forgetNow, l.rest WHILE l#NIL DO
CDProperties.PutObjectProp[onto: me, prop: l.first, val: NIL];
Process.Yield[];
ENDLOOP;
};
[] ← CDDirectory.EnumerateDesign[design: design, proc: RPEachOb];
};
CedarProcess.SetPriority[CedarProcess.Priority[background]];
DO
forgetNow ← FetchAndStop[rr];
IF forgetNow=NIL THEN EXIT;
RemoveInner[design ! RuntimeError.UNCAUGHT => CONTINUE];
ENDLOOP;
};
RemoveProperties: PUBLIC PROC [design: CD.Design, key: REF] = {
GetRemoveRecord: PROC [d: CD.Design] RETURNS [rr: REF RemoveRecord] = {
WITH CDValue.Fetch[d, $CDDirectoryOpsPrivate, design] SELECT FROM
r: REF RemoveRecord => RETURN [r];
ENDCASE => NULL;
[] ← CDValue.StoreConditional[d, $CDDirectoryOpsPrivate, NEW[RemoveRecord]];
rr ← GetRemoveRecord[d];
};
rr: REF RemoveRecord ~ GetRemoveRecord[design];
DepositAndStart[design, rr, key];
};
IncludeDescribedObjects: PUBLIC PROC [design: CD.Design, ob: CD.Object, visited: RefTab.Ref←NIL] = {
EachObject: CDDirectory.EachObjectProc = {
IF me.class.composed THEN {
design: CD.Design = NARROW[data];
name: Rope.ROPE ← CDDirectory.Name[me, design];
IF name#NIL THEN RETURN;
WITH CDProperties.GetObjectProp[me, $Describe] SELECT FROM
n: Rope.ROPE => name ← n;
a: ATOM => name ← Atom.GetPName[a];
ENDCASE => RETURN;
[] ← CDDirectory.Include[design, me, name, TRUE];
}
};
[] ← CDDirectory.EnumerateObject[ob: ob, proc: EachObject, visited: visited, data: design, recurse: TRUE]
};
END.