RecognizerImpl.mesa
Last Edited by: Gbier, July 11, 1985 12:18:45 pm PDT
Last Edited by: Gbier, July 26, 1985 1:41:17 pm PDT
Jacobi, January 9, 1986 12:27:25 pm PST
DIRECTORY
CD,
CDBasics,
CornerStitching,
CDOrient,
CDCells,
Recognizer,
TerminalIO;
RecognizerImpl:
CEDAR
MONITOR
IMPORTS CD, CDBasics, CornerStitching, CDCells, TerminalIO
EXPORTS Recognizer =
BEGIN
RegionList: TYPE = LIST OF REF CornerStitching.Region;
Plane: TYPE = REF CornerStitching.Tesselation;
PossibleProcedureRec:
TYPE =
RECORD [
technology: CD.Technology,
rectCount: INT,
instProc: Recognizer.PossibleInstanceProc --the procedure that will return possible instances
];
procRegistration: LIST OF REF PossibleProcedureRec ← NIL; --Contains registered possible procedures
ActiveLayersRec:
TYPE =
RECORD [
technology: CD.Technology,
active1: CD.Layer,
active2: CD.Layer,
layerList: LIST OF CD.Layer
];
layerRegistration: LIST OF REF ActiveLayersRec ← NIL; --Contains registered active layer pairs
Handle: TYPE = REF HandleRep;
HandleRep:
TYPE =
RECORD [
design: CD.Design,
cellOb: CD.Object,
active1: CD.Layer ← CD.combined,
active2: CD.Layer ← CD.combined,
overlapPlane: Plane ← NIL,
active1CheckTess: Plane ← NIL,
active2CheckTess: Plane ← NIL,
coveredList: LIST OF REF CornerStitching.Region ← NIL,
matches: BOOL ← TRUE,
layerPlanes: ARRAY CD.Layer OF Plane ← ALL[NIL]
];
Free:
PROC [p: Plane] = {
CornerStitching.FreeTesselation[p, FALSE];
};
RecognizeObjects:
PUBLIC
PROC [design:
CD.Design, cellOb:
CD.Object] =
BEGIN
FOR activeList:
LIST
OF
REF ActiveLayersRec ← layerRegistration, activeList.rest
WHILE activeList#NIL
DO
IF activeList.first.technology = design.technology
THEN {
h: Handle ← NEW[HandleRep ← [design, cellOb]];
h.active1 ← activeList.first.active1;
h.active2 ← activeList.first.active2;
h.layerPlanes[h.active1] ← CornerStitching.NewTesselation[];
h.layerPlanes[h.active2] ← CornerStitching.NewTesselation[];
FOR layerList:
LIST
OF
CD.Layer ← activeList.first.layerList, layerList.rest
WHILE layerList#
NIL
DO
h.layerPlanes[layerList.first] ← CornerStitching.NewTesselation[];
ENDLOOP;
FillPlanesByDrawingRects[design, h];
h.overlapPlane ← CornerStitching.NewTesselation[];
FormOverlapPlane[plane1: h.layerPlanes[h.active1], plane2: h.layerPlanes[h.active2], overlapPlane: h.overlapPlane];
CornerStitching.EnumerateArea[plane: h.overlapPlane, rect: CDBasics.universe, skipValue: $none, perTile: GrowTransistorTile, data: h];
--help the garbage collector
Free[h.overlapPlane];
FOR layerList:
LIST
OF
CD.Layer ← activeList.first.layerList, layerList.rest
WHILE layerList#
NIL
DO
Free[h.layerPlanes[layerList.first]];
ENDLOOP;
Free[h.layerPlanes[h.active1]];
Free[h.layerPlanes[h.active2]];
};
[] ← CDCells.RepositionCell[cellOb, design];
ENDLOOP;
END;
FillPlanesByDrawingRects:
PROC [design:
CD.Design, h: Handle] =
BEGIN
dummyInst: CD.Instance ~ NEW[CD.InstanceRep ← [ob: h.cellOb]];
pr: CD.DrawRef ← CD.CreateDrawRef[h.design];
pr.devicePrivate ← h;
pr.drawRect ← DrawRectIntoPlane;
pr.drawChild ← DrawChildIntoPlane;
h.cellOb.class.drawMe[inst: dummyInst, pos: [0, 0], orient: CD.original, pr: pr];
END;
DrawChildIntoPlane:
CD.DrawProc =
BEGIN
IF inst.ob.class.wireTyped THEN inst.ob.class.drawMe[inst, pos, orient, pr]
END;
DrawRectIntoPlane:
PROC [r:
CD.Rect, l:
CD.Layer, pr:
CD.DrawRef] =
BEGIN
h: Handle ~ NARROW[pr.devicePrivate];
IF h.layerPlanes[l]#
NIL
THEN
CornerStitching.ChangeRect[plane: h.layerPlanes[l], rect: r, newValue: $cover];
END;
FormOverlapPlane:
PUBLIC
PROC [plane1, plane2, overlapPlane: Plane] =
--Overlapping regions between plane1 and plane2 will be added to the overlapPlane
BEGIN
FOR rl1: RegionList ← CornerStitching.ListArea[plane1, CDBasics.universe, $none], rl1.rest
WHILE rl1#
NIL
DO
IF rl1.first.value=$cover
THEN
FOR ovL: RegionList ← CornerStitching.ListArea[plane2, rl1.first.rect, $none], ovL.rest
WHILE ovL#
NIL
DO
IF ovL.first.value=$cover
THEN
CornerStitching.ChangeRect[plane: overlapPlane, rect: ovL.first.rect, newValue: $cover];
ENDLOOP;
ENDLOOP;
END;
GrowTransistorTile:
PROC [tile:
REF CornerStitching.Tile, data:
REF
ANY] =
--A new tesselation called currentTess is formed which contains the maximum connected region that has the same value as tile and contains tile.
BEGIN
IF tile.value = $cover
THEN {
currentTess: Plane ~ CornerStitching.NewTesselation[];
h: Handle ← NARROW[data];
tileList: LIST OF REF CornerStitching.Tile ← LIST[tile];
WHILE tileList#
NIL
DO
tile: REF CornerStitching.Tile ← tileList.first;
tileList ← tileList.rest;
IF tile.value#$done
THEN {
tileList ← GetNeighborsSameAsMe[tile, h.overlapPlane, tileList];
CornerStitching.ChangeRect[currentTess, CornerStitching.Area[tile], $cover];
tile.value ← $done;
};
ENDLOOP;
AnalyzeTransistor[currentTess, h];
Free[currentTess];
}
END;
GetNeighborsSameAsMe:
PROC [tile:
REF CornerStitching.Tile, tess: Plane, tileList:
LIST
OF
REF CornerStitching.Tile]
RETURNS [newList:
LIST
OF
REF CornerStitching.Tile] =
--Returns all tiles adjacent to tile that have the same value as tile
BEGIN
cur: REF CornerStitching.Tile;
newList ← tileList;
cur ← CornerStitching.ENorthNeighbour[tile];
WHILE CornerStitching.WSouthNeighbour[cur] = tile
DO
IF cur.value = tile.value
THEN
newList ← CONS[cur, newList];
cur ← CornerStitching.SWestNeighbour[cur];
ENDLOOP;
IF cur.value = tile.value THEN newList ← CONS[cur, newList];
cur ← CornerStitching.SWestNeighbour[tile];
WHILE CornerStitching.NEastNeighbour[cur] = tile
DO
IF cur.value = tile.value
THEN
newList ← CONS[cur, newList];
cur ← CornerStitching.ENorthNeighbour[cur];
ENDLOOP;
IF cur.value = tile.value THEN newList ← CONS[cur, newList];
cur ← CornerStitching.WSouthNeighbour[tile];
WHILE CornerStitching.ENorthNeighbour[cur] = tile
DO
IF cur.value = tile.value
THEN
newList ← CONS[cur, newList];
cur ← CornerStitching.NEastNeighbour[cur];
ENDLOOP;
IF cur.value = tile.value THEN newList ← CONS[cur, newList];
cur ← CornerStitching.NEastNeighbour[tile];
WHILE CornerStitching.SWestNeighbour[cur] = tile
DO
IF cur.value = tile.value
THEN
newList ← CONS[cur, newList];
cur ← CornerStitching.WSouthNeighbour[cur];
ENDLOOP;
IF cur.value = tile.value THEN newList ← CONS[cur, newList];
END;
AnalyzeTransistor:
PROC [plane: Plane, h: Handle] =
--Enumerates plane, counting the covered regions and storing the covered regions on h.coveredList. The count and technology are used to select the proper Possible Instance Procedures to call. The Possible Instance Procedures return a list of possible instances which are then searched for a match.
BEGIN
currentList: RegionList ← CornerStitching.ListArea[plane, CDBasics.universe, $none];
count: INT ← 0;
possibleInstListPtr: REF Recognizer.InstAndForbidLayerList ← NEW[Recognizer.InstAndForbidLayerList ← NIL];
h.coveredList ← NIL;
FOR countList: RegionList ← currentList, countList.rest
WHILE countList#
NIL
DO
IF countList.first.value#
NIL
THEN {
count ← count + 1;
h.coveredList ← CONS[countList.first, h.coveredList];
};
ENDLOOP;
FOR procList:
LIST
OF
REF PossibleProcedureRec ← procRegistration, procList.rest
WHILE procList#
NIL
DO
IF procList.first.technology = h.design.technology
AND procList.first.rectCount = count
THEN
procList.first.instProc[h.active1, h.active2, h.coveredList, possibleInstListPtr];
ENDLOOP;
WHILE possibleInstListPtr^#
NIL
DO
IF InstanceMatches[possibleInstListPtr^.first.instance, h]
AND NoForbiddenOverlap[possibleInstListPtr^.first.forbidLayerList, h]
THEN {
newInst:
CD.Instance ← CDCells.IncludeOb[
design: h.design,
cell: h.cellOb,
ob: possibleInstListPtr^.first.instance.ob,
position: possibleInstListPtr^.first.instance.location,
orientation: possibleInstListPtr^.first.instance.orientation,
cellCSystem: cdCoords,
obCSystem: cdCoords,
mode: dontPropagate
].newInst;
newInst.selected ← TRUE;
EXIT;
}
ELSE
possibleInstListPtr^ ← possibleInstListPtr^.rest;
ENDLOOP;
END;
InstanceMatches:
PROC [instance:
CD.Instance, h: Handle]
RETURNS [
BOOL ←
FALSE] =
--Draws the instance using the instances drawMe procedure
BEGIN
pr: CD.DrawRef ← CD.CreateDrawRef[h.design];
pr.devicePrivate ← h;
pr.drawRect ← DrawRectAgainstTess;
h.matches ← TRUE;
h.active1CheckTess ← CornerStitching.NewTesselation[];
h.active2CheckTess ← CornerStitching.NewTesselation[];
instance.ob.class.drawMe[inst: instance, pos: instance.location, orient: instance.orientation, pr: pr];
IF h.matches
THEN
RETURN[OverlapRegionsMatch[h]];
Reaching here means that the instance drawn against the design matched. However it is still neccessary to make sure that the overlapping region in the instance matches what is on the coveredList. This guarantees that the matching instance is exactly the same size as the overlapping region in question in the design.
END;
DrawRectAgainstTess:
PROC [r:
CD.Rect, l:
CD.Layer, pr:
CD.DrawRef] =
--The rect is enumerated against its layer plane, if it is not covered then h.matches is set to false.
BEGIN
h: Handle ← NARROW[pr.devicePrivate];
checkPlane: Plane ← h.layerPlanes[l];
IF CDBasics.NonEmpty[r]
THEN {
IF l = h.active1 THEN CornerStitching.ChangeRect[h.active1CheckTess, r, $cover]
ELSE IF l = h.active2 THEN CornerStitching.ChangeRect[h.active2CheckTess, r, $cover];
IF checkPlane = NIL THEN {
h.matches ←
FALSE;
TerminalIO.WriteRope["Unexpected Layer Encountered\n"];
RETURN
};
FOR checkList: RegionList ← CornerStitching.ListArea[checkPlane, r, $none], checkList.rest
WHILE checkList#NIL
DO
IF checkList.first.value =
NIL
THEN {
h.matches ← FALSE;
RETURN
}
ENDLOOP;
}
END;
OverlapRegionsMatch:
PROC [h: Handle]
RETURNS [b:
BOOL ←
TRUE] =
--Forms the overlap of what is stored in active1CheckTess and active2CheckTess and makes sure that everthing on the coveredList is really covered by the overlap
BEGIN
checkTess: Plane ← CornerStitching.NewTesselation[];
FOR list1: RegionList ← CornerStitching.ListArea[plane: h.active1CheckTess, rect: CDBasics.universe, skipValue: $none], list1.rest
WHILE list1#
NIL
DO
IF list1.first.value#
NIL
THEN
FOR overlapList: RegionList ← CornerStitching.ListArea[plane: h.active2CheckTess, rect: list1.first.rect, skipValue: $none], overlapList.rest
WHILE overlapList#
NIL
DO
IF overlapList.first.value#
NIL
THEN
CornerStitching.ChangeRect[checkTess, overlapList.first.rect, $cover];
ENDLOOP;
ENDLOOP;
b ← RegionAndTessOverlapCompletely[h.coveredList, checkTess];
Free[checkTess];
END;
NoForbiddenOverlap:
PROC [layerList:
LIST
OF
CD.Layer, h: Handle]
RETURNS [
BOOL ←
TRUE] =
--h.coveredList is enumerated against the layerList. Returns FALSE if any piece of the coveredList is covered on any of the layers.
BEGIN
WHILE layerList#
NIL
DO
IF h.layerPlanes[layerList.first] =
NIL
THEN {
TerminalIO.WriteRope["Unexpected Layer Encountered\n"];
RETURN [FALSE];
}
ELSE
IF RegionAndTessIntersect[h.coveredList, h.layerPlanes[layerList.first]] THEN RETURN [FALSE]
ELSE layerList ← layerList.rest;
ENDLOOP;
END;
RegionAndTessOverlapCompletely:
PROC [checkList: RegionList, tess: Plane]
RETURNS [
BOOL ←
TRUE] =
--Returns true if everything on checkList is covered on the Tesselation
BEGIN
FOR list: RegionList ← checkList, list.rest
WHILE list#
NIL
DO
FOR list2: RegionList ← CornerStitching.ListArea[plane: tess, rect: list.first.rect, skipValue: $none], list2.rest
WHILE list2#NIL
DO
IF list2.first.value = NIL THEN RETURN[FALSE]
ENDLOOP;
ENDLOOP;
END;
RegionAndTessIntersect:
PROC [checkList: RegionList, tess: Plane]
RETURNS [
BOOL ←
FALSE] =
--Returns true if the intersection between items on the checklist and the tesselation is non-empty
BEGIN
WHILE checkList#
NIL
DO
FOR intersectList: RegionList ← CornerStitching.ListArea[tess, checkList.first.rect, $none], intersectList.rest
WHILE intersectList#
NIL
DO
IF intersectList.first.value#NIL THEN RETURN [TRUE];
ENDLOOP;
checkList ← checkList.rest;
ENDLOOP;
END;
RegisterPossibleInstanceProc:
PUBLIC
ENTRY PROC [technology:
CD.Technology, rectCount:
INT, instProc: Recognizer.PossibleInstanceProc] =
BEGIN
procRegistration ← CONS[NEW[PossibleProcedureRec ← [technology, rectCount, instProc]], procRegistration];
END;
RegisterLayerList:
PUBLIC
ENTRY PROC [technology:
CD.Technology, active1, active2:
CD.Layer, layerList:
LIST
OF
CD.Layer] =
BEGIN
layerRegistration ← CONS[NEW[ActiveLayersRec ← [technology, active1, active2, layerList]], layerRegistration];
END;
END.