CDMEBESGeomFns:
CEDAR
PROGRAM
IMPORTS Atom, CDBasics, CDMEBES, CStitching, IO =
BEGIN OPEN CDMEBES;
**** S U P P O R T R O U T I N E S ****
Direction: TYPE = {N, E, S, W} ← N;
reverseDirection: ARRAY Direction OF Direction ← [N: S, E: W, S: N, W: E];
Juxtaposition:
PROC [ a, b: TadRect ]
RETURNS [ aVsB: Direction ← N, biggestGap: Tad ← -1 ] = {
tGap: Tad;
IF (tGap ← b.x1-a.x2) > biggestGap THEN {aVsB ← W; biggestGap ← tGap};
IF (tGap ← a.x1-b.x2) > biggestGap THEN {aVsB ← E; biggestGap ← tGap};
IF (tGap ← b.y1-a.y2) > biggestGap THEN {aVsB ← S; biggestGap ← tGap};
IF (tGap ← a.y1-b.y2) > biggestGap THEN {aVsB ← N; biggestGap ← tGap};
};
ScanNeighborhood:
PROC [ proc:
PROC [z1, z2, depth: Tad], mother: CStitching.Tile, p1, p2: TadPair, depthLimit: Tad, continueValue:
REF ←
NIL ] = {
InnerDepthScanS:
PROC [ tile: CStitching.Tile, z1, z2: Tad ] = {
-- z1 < z2
curDepth: Tad = p1.y - tile.NEdge;
IF curDepth >= depthLimit THEN proc[z1, z2, depthLimit]
ELSE {
FOR tileSouth: CStitching.Tile ← tile.WS, tileSouth.NE WHILE tileSouth.WEdge < z2 DO
IF tileSouth.EEdge > z1
THEN {
IF (curDepth > 0) AND (tileSouth.value # continueValue) THEN proc[z1, z2, curDepth]
ELSE InnerDepthScanS[tileSouth, MAX[z1, tileSouth.WEdge], MIN[z2, tileSouth.EEdge]];
};
ENDLOOP;
};
};
InnerDepthScanN:
PROC [ tile: CStitching.Tile, z1, z2: Tad ] = {
-- z1 < z2
curDepth: Tad = tile.SEdge - p1.y;
IF curDepth >= depthLimit THEN proc[z1, z2, depthLimit]
ELSE {
FOR tileNorth: CStitching.Tile ← tile.
EN, tileNorth.
SW
WHILE tileNorth.EEdge > z1
DO
IF tileNorth.WEdge < z2
THEN {
IF (curDepth > 0) AND (tileNorth.value # continueValue) THEN proc[z1, z2, curDepth]
ELSE InnerDepthScanN[tileNorth, MAX[z1, tileNorth.WEdge], MIN[z2, tileNorth.EEdge]];
};
ENDLOOP;
};
};
InnerDepthScanE:
PROC [ tile: CStitching.Tile, z1, z2: Tad ] = {
-- z1 < z2
curDepth: Tad = tile.WEdge - p1.x;
IF curDepth >= depthLimit THEN proc[z1, z2, depthLimit]
ELSE {
FOR tileEast: CStitching.Tile ← tile.NE, tileEast.WS WHILE tileEast.NEdge > z1 DO
IF tileEast.SEdge < z2
THEN {
IF (curDepth > 0) AND (tileEast.value # continueValue) THEN proc[z1, z2, curDepth]
ELSE InnerDepthScanE[tileEast, MAX[z1, tileEast.SEdge], MIN[z2, tileEast.NEdge]];
};
ENDLOOP;
};
};
InnerDepthScanW:
PROC [ tile: CStitching.Tile, z1, z2: Tad ] = {
-- z1 < z2
curDepth: Tad = p1.x - tile.EEdge;
IF curDepth >= depthLimit THEN proc[z1, z2, depthLimit]
ELSE {
FOR tileWest: CStitching.Tile ← tile.SW, tileWest.EN WHILE tileWest.SEdge < z2 DO
IF tileWest.NEdge > z1
THEN {
IF (curDepth > 0) AND (tileWest.value # continueValue) THEN proc[z1, z2, curDepth]
ELSE InnerDepthScanW[tileWest, MAX[z1, tileWest.SEdge], MIN[z2, tileWest.NEdge]];
};
ENDLOOP;
};
};
By convention, the depth scan proceeds to the right from the line p1 -> p2.
SELECT (p2.x - p1.x)
FROM
> 0 => {IF p2.y # p1.y THEN ERROR; InnerDepthScanS[mother, p1.x, p2.x]};
< 0 => {IF p2.y # p1.y THEN ERROR; InnerDepthScanN[mother, p2.x, p1.x]};
ENDCASE =>
SELECT (p2.y - p1.y)
FROM
> 0 => InnerDepthScanE[mother, p1.y, p2.y];
< 0 => InnerDepthScanW[mother, p2.y, p1.y];
ENDCASE => ERROR;
};
***** M A S K O P E R A T I O N R O U T I N E S *****
NOTFn:
PROC [ ms: MaskState, spec:
LIST
OF
REF
ANY, extInfluenceDiameter: Tad ← 0 ]
RETURNS [ result: Tesselation ]
-- CoverProc -- =
BEGIN
ComplementTile:
PROC [tile: CStitching.Tile, data:
REF
ANY]
-- CStitching.PerTileProc -- =
{result.ChangeRect[CDBasics.Intersection[CDBasics.universe, tile.Area], $covered]};
basis: Tesselation ← GenerateCover[ms, spec.first, extInfluenceDiameter];
result ← NewTesselation[NIL];
[] ← basis.EnumerateArea[rect: CDBasics.universe, eachTile: ComplementTile, skip: $covered];
basis ← DisposeTesselation[basis];
END;
ANDFn:
PROC [ ms: MaskState, spec:
LIST
OF
REF
ANY, extInfluenceDiameter: Tad ← 0 ]
RETURNS [ result: Tesselation ]
-- CoverProc -- =
BEGIN
NilTile:
PROC [tile: CStitching.Tile, data:
REF
ANY]
-- CStitching.PerTileProc -- =
{result.ChangeRect[CDBasics.Intersection[CDBasics.universe, tile.Area], NIL]};
result ← GenerateCover[ms, spec.first, extInfluenceDiameter];
FOR rest:
LIST
OF
REF
ANY ← spec.rest, rest.rest
WHILE rest #
NIL
DO
term: Tesselation ← GenerateCover[ms, rest.first, extInfluenceDiameter];
[] ← term.EnumerateArea[rect: CDBasics.universe, eachTile: NilTile, skip: $covered];
term ← DisposeTesselation[term];
ENDLOOP;
END;
ANDNOTFn:
PROC [ ms: MaskState, spec:
LIST
OF
REF
ANY, extInfluenceDiameter: Tad ← 0 ]
RETURNS [ result: Tesselation ]
-- CoverProc -- =
BEGIN
NilTile:
PROC [tile: CStitching.Tile, data:
REF
ANY]
-- CStitching.PerTileProc -- =
{result.ChangeRect[CDBasics.Intersection[CDBasics.universe, tile.Area], NIL]};
result ← GenerateCover[ms, spec.first, extInfluenceDiameter];
FOR rest:
LIST
OF
REF
ANY ← spec.rest, rest.rest
WHILE rest #
NIL
DO
term: Tesselation ← GenerateCover[ms, rest.first, extInfluenceDiameter];
[] ← term.EnumerateArea[rect: CDBasics.universe, eachTile: NilTile, skip: NIL];
term ← DisposeTesselation[term];
ENDLOOP;
END;
ORFn:
PROC [ ms: MaskState, spec:
LIST
OF
REF
ANY, extInfluenceDiameter: Tad ← 0 ]
RETURNS [ result: Tesselation ]
-- CoverProc -- =
BEGIN
This is a very common case, so we speed it up by separating it into two lists: CDLayer's with nonnegative deltaDiameter's, and other things. We compute the other things first, and then take a single pass at the design for the CDLayer's with nonnegative deltaDiameter's.
CoverTile:
PROC [tile: CStitching.Tile, data:
REF
ANY]
-- CStitching.PerTileProc -- =
{result.ChangeRect[tile.Area, $covered]};
layers: LIST OF CDLayer ← NIL;
otherThings: LIST OF REF ANY ← NIL;
FOR subList:
LIST
OF
REF
ANY ← spec, subList.rest
WHILE subList #
NIL
DO
WITH subList.first
SELECT
FROM
layer: CDLayer =>
IF layer.deltaDiameter >= 0 THEN layers ← CONS[layer, layers]
ELSE otherThings ← CONS[layer, otherThings];
ENDCASE => otherThings ← CONS[subList.first, otherThings];
IF otherThings = NIL THEN result ← NewTesselation[NIL]
ELSE
BEGIN
result ← GenerateCover[ms, otherThings.first, extInfluenceDiameter];
FOR rest:
LIST
OF
REF
ANY ← otherThings.rest, rest.rest
WHILE rest #
NIL
DO
term: Tesselation ← GenerateCover[ms, rest.first, extInfluenceDiameter];
[] ← term.EnumerateArea[rect: CDBasics.universe, eachTile: CoverTile];
term ← DisposeTesselation[term];
ENDLOOP;
END;
IF layers #
NIL
THEN
DrawLayersWithPosDeltas[ms: ms, rects: result, layers: layers, extInfluenceDiameter: extInfluenceDiameter];
END;
RunRec:
TYPE =
RECORD [
desc: ROPE,
areaInSqMeters: REAL ← 0,
perimInMeters: Meters ← 0.0,
runsInMeters:
ARRAY Sense
OF
RECORD [
sense: ROPE,
r: REF RunArray
]
];
RunArray: TYPE = RECORD [ ra: SEQUENCE size: NAT OF Meters ];
MeasureRuns:
PROC [ ms: MaskState, spec:
LIST
OF
REF
ANY, extInfluenceDiameter: Tad ← 0 ]
RETURNS [ result: Tesselation ]
-- CoverProc -- =
BEGIN
PixelRunArray: TYPE = RECORD [ ra: SEQUENCE size: NAT OF MEBESPixels ];
TadCoverRec: TYPE = RECORD [ sense: ROPE, maxInterestDist, minFeature: Tad ← 0, r: REF PixelRunArray ];
areaInSqPixels: INT ← 0;
perimInPixels: MEBESPixels ← 0;
table: ARRAY Sense OF REF TadCoverRec ← ALL[NIL];
runRef: REF RunRec ← NIL; -- for external consumption
mebesPixelPitch: Meters ← ms.mebesPixelPitch -- nm -- *1.0E-9;
tadsPerPixel: Tad = ms.mebesPixelPitch*ms.tadsPerNm;
LookForRuns:
PROC [tile: CStitching.Tile, data:
REF
ANY]
-- CStitching.PerTileProc -- =
BEGIN
ComplainAboutFeature:
PROC [ dia: Tad, pos: TadPosition ] =
BEGIN
chose:
NAT = ComplainAt[ms: ms, pos: pos,
explanation: IO.PutFR["Too-small %g", IO.rope[c.sense]],
choice: LIST ["Reduce min feature", "Proceed"]];
SELECT chose
FROM
1 => c.minFeature ← dia;
ENDCASE => NULL;
END;
EWRuns:
PROC [ cov: CStitching.Tile, x1, x2: Tad ] =
BEGIN
IF cov.NEdge < keyTileArea.y2+c.maxInterestDist
THEN
BEGIN
covHeightTads: Tad = cov.NEdge-keyTileArea.y2;
covHeightPixels: MEBESPixels = covHeightTads/tadsPerPixel;
FOR tile: CStitching.Tile ← cov.EN, tile.SW
WHILE x1 < tile.EEdge
DO
tx1: Tad = MAX[x1, tile.WEdge];
tx2: Tad = MIN[x2, tile.EEdge];
dxPixels: MEBESPixels = (tx2-tx1)/tadsPerPixel;
SELECT
TRUE
FROM
tx2 <= tx1 => NULL;
tile.value # keyTilevalue =>
BEGIN
IF covHeightTads = 0
THEN
perimInPixels ← perimInPixels+dxPixels;
EWRuns[cov: tile, x1: tx1, x2: tx2];
END;
covHeightTads >= c.minFeature =>
c.r[covHeightPixels] ← c.r[covHeightPixels]+dxPixels;
covHeightTads > 0 =>
-- IN (0..minFeature) --
BEGIN
ComplainAboutFeature[covHeightTads, [x: (tx1+tx2)/2, y: (cov.NEdge+keyTileArea.y2)/2]];
c.r[covHeightPixels] ← c.r[covHeightPixels]+dxPixels;
END;
ENDCASE => NULL;
ENDLOOP;
END;
END; -- of EWRuns
NSRuns:
PROC [ cov: CStitching.Tile, y1, y2: Tad ] =
BEGIN
IF cov.EEdge < keyTileArea.x2+c.maxInterestDist
THEN
BEGIN
covWidthTads: Tad = cov.EEdge-keyTileArea.x2;
covWidthPixels: MEBESPixels = covWidthTads/tadsPerPixel;
FOR tile: CStitching.Tile ← cov.NE, tile.WS
WHILE y1 < tile.NEdge
DO
ty1: MEBESPixels = MAX[y1, tile.SEdge];
ty2: MEBESPixels = MIN[y2, tile.NEdge];
dyPixels: MEBESPixels = (ty2-ty1)/tadsPerPixel;
SELECT
TRUE
FROM
ty2 <= ty1 => NULL;
tile.value # keyTilevalue =>
BEGIN
IF covWidthTads = 0 THEN perimInPixels ← perimInPixels+dyPixels;
NSRuns[cov: tile, y1: ty1, y2: ty2];
END;
covWidthTads >= c.minFeature =>
c.r[covWidthPixels] ← c.r[covWidthPixels]+dyPixels;
covWidthTads > 0 =>
-- IN (0..minFeature) --
BEGIN
ComplainAboutFeature[covWidthTads, [x: (cov.EEdge+keyTileArea.x2)/2, y: (ty1+ty2)/2]];
c.r[covWidthPixels] ← c.r[covWidthPixels]+dyPixels;
END;
ENDCASE => NULL;
ENDLOOP;
END;
END; -- of NSRuns
keyTilevalue: REF ANY = tile.value;
c: REF TadCoverRec = table[(IF keyTilevalue = NIL THEN cover ELSE space)];
keyTileArea: TadRect = tile.Area;
clippedKeyTileArea: TadRect = CDBasics.Intersection[ms.tadStripeClip, keyTileArea];
IF keyTilevalue #
NIL
THEN
areaInSqPixels ← areaInSqPixels+
((clippedKeyTileArea.x2-clippedKeyTileArea.x1)/tadsPerPixel)*
((clippedKeyTileArea.y2-clippedKeyTileArea.y1)/tadsPerPixel);
We examine first the north and then the east neighbors of this key rectangle, noting the nearest rectangle of the same kind in the direction of view.
IF keyTileArea.y2
IN (ms.tadStripeClip.y1 .. ms.tadStripeClip.y2]
THEN
EWRuns[cov: tile, x1: clippedKeyTileArea.x1, x2: clippedKeyTileArea.x2];
IF keyTileArea.x2
IN (ms.tadStripeClip.x1 .. ms.tadStripeClip.x2]
THEN
NSRuns[cov: tile, y1: clippedKeyTileArea.y1, y2: clippedKeyTileArea.y2];
END; -- of LookForRuns
GetCoverRec:
PROC [sense:
ROPE, l:
LIST
OF
REF
ANY]
RETURNS [c:
REF TadCoverRec] =
BEGIN
c ← NEW[TadCoverRec ← [sense: sense]];
c.maxInterestDist ← NARROW[l.first, REF Nm]^*ms.tadsPerNm;
c.minFeature ← NARROW[l.rest.first, REF Nm]^*ms.tadsPerNm;
c.r ← NEW[PixelRunArray[c.maxInterestDist/(ms.mebesPixelPitch*ms.tadsPerNm)]];
FOR width: MEBESPixels
IN [0..c.r.size)
DO
c.r[width] ← 0;
ENDLOOP;
END;
table[cover] ← GetCoverRec["cover", spec.rest.rest];
table[space] ← GetCoverRec["space", spec.rest.rest.rest.rest];
result ← GenerateCover[ms, spec.rest.first, extInfluenceDiameter+MAX[table[cover].maxInterestDist, table[space].maxInterestDist]];
[] ← result.EnumerateArea[rect: ms.tadStripeClip, eachTile: LookForRuns, skip: $none];
ms.data ← runRef ←
NEW[RunRec ← [
desc: NARROW[spec.first],
areaInSqMeters: areaInSqPixels*mebesPixelPitch*mebesPixelPitch,
perimInMeters: perimInPixels*mebesPixelPitch,
runsInMeters: [
space: [sense: "space", r: NEW[RunArray[table[space].r.size]]],
cover: [sense: "cover", r: NEW[RunArray[table[cover].r.size]]]
]
]];
FOR sense: Sense
IN Sense
DO
FOR width: MEBESPixels
IN [0..table[sense].r.size)
DO
runRef.runsInMeters[sense].r[width] ← table[sense].r[width]*mebesPixelPitch;
ENDLOOP;
ENDLOOP;
END;
RestrictedEnlarge:
PROC [ ms: MaskState, spec:
LIST
OF
REF
ANY, extInfluenceDiameter: Tad ← 0 ]
RETURNS [ result: Tesselation ]
-- CoverProc -- =
BEGIN
PlaceNImplant:
PROC [tile: CStitching.Tile, data:
REF
ANY]
-- CStitching.PerTileProc -- =
BEGIN
InteractWithPImplant:
PROC [tile: CStitching.Tile, data:
REF
ANY]
-- CStitching.PerTileProc -- =
BEGIN
pTile: CStitching.Tile = tile;
pTileArea: TadRect = pTile.Area;
aVsB: Direction;
biggestGap: Tad ← -1;
gaps: ARRAY Direction OF Tad ← ALL[maxRadialExtension];
deInsertedArea: TadRect;
[aVsB, biggestGap] ← Juxtaposition[a: pTileArea, b: nTileArea];
IF biggestGap<0 THEN ERROR; -- design rule violation
IF
NOT anyInteractingPImplant
THEN
BEGIN
anyInteractingPImplant ← TRUE;
nImplPieces.ChangeRect[CDBasics.universe, NIL];
nImplPieces.ChangeRect[desiredNImplant, $covered];
END;
gaps[aVsB] ← biggestGap/2;
deInsertedArea ← [x1: pTileArea.x1-gaps[E], y1: pTileArea.y1-gaps[N], x2: pTileArea.x2+gaps[W], y2: pTileArea.y2+gaps[S]];
IF CDBasics.NonEmpty[deInsertedArea]
THEN
nImplPieces.ChangeRect[deInsertedArea, NIL];
END;
CoverNegotiatedPiece:
PROCEDURE [tile: CStitching.Tile, data:
REF
ANY]
-- CStitching.PerTileProc -- =
{area: TadRect = tile.Area; result.ChangeRect[area, $covered]};
nTile: CStitching.Tile = tile;
nTileArea: TadRect = nTile.Area;
desiredNImplant: TadRect = CDBasics.Extend[nTileArea, maxRadialExtension];
anyInteractingPImplant: BOOL ← FALSE;
[] ← pDif.EnumerateArea[rect: CDBasics.Extend[nTileArea, 2*maxRadialExtension], eachTile: InteractWithPImplant];
IF NOT anyInteractingPImplant THEN result.ChangeRect[desiredNImplant, $covered]
ELSE [] ← nImplPieces.EnumerateArea[desiredNImplant, CoverNegotiatedPiece];
END;
maxRadialExtension: Tad = NARROW[spec.first, REF Nm]^*ms.tadsPerNm;
nDif: Tesselation ← GenerateCover[ms, spec.rest.first, extInfluenceDiameter+2*maxRadialExtension];
pDif: Tesselation ← GenerateCover[ms, spec.rest.rest.first, extInfluenceDiameter+2*maxRadialExtension];
nImplPieces: Tesselation ← NewTesselation[];
result ← NewTesselation[];
[] ← nDif.EnumerateArea[rect: CDBasics.universe, eachTile: PlaceNImplant];
nDif ← DisposeTesselation[nDif];
pDif ← DisposeTesselation[pDif];
nImplPieces ← DisposeTesselation[nImplPieces];
END;
EnlargeSparse:
PROC [ ms: MaskState, spec:
LIST
OF
REF
ANY, extInfluenceDiameter: Tad ← 0 ]
RETURNS [ result: Tesselation ]
-- CoverProc -- =
BEGIN
EnlargeToThe:
PROC [ curDir: Direction, sideSpacing: Tad ] = {
Enlarge:
PROC [tile: CStitching.Tile, data:
REF]
-- CStitching.TileProc -- = {
tileArea: TadRect = tile.Area;
enlargedArea: TadRect = [
x1: tile.WEdge-enlargeTable[W],
y1: tile.SEdge-enlargeTable[S],
x2: tile.EEdge+enlargeTable[E],
y2: tile.NEdge+enlargeTable[N]];
result.ChangeRect[enlargedArea, $covered];
};
Clip:
PROC [tile: CStitching.Tile, data:
REF]
-- CStitching.TileProc -- = {
ClipS:
PROC [tile: CStitching.Tile, data:
REF]
-- CStitching.TileProc -- = {
tile and clipBasisTile adjoin and form a material edge
x1: Tad = MAX[tile.WEdge, clipBasisTile.WEdge];
x2: Tad = MIN[tile.EEdge, clipBasisTile.EEdge];
IF x1 < x2
THEN
result.ChangeRect[[x1: x1-sideSpacing, y1: clipBasisTile.SEdge-minSpacing, x2: x2+sideSpacing, y2: clipBasisTile.SEdge], NIL];
};
ClipN:
PROC [tile: CStitching.Tile, data:
REF]
-- CStitching.TileProc -- = {
tile and clipBasisTile adjoin and form a material edge
x1: Tad = MAX[tile.WEdge, clipBasisTile.WEdge];
x2: Tad = MIN[tile.EEdge, clipBasisTile.EEdge];
IF x1 < x2
THEN
result.ChangeRect[[x1: x1-sideSpacing, y1: clipBasisTile.NEdge, x2: x2+sideSpacing, y2: clipBasisTile.NEdge+minSpacing], NIL];
};
ClipW:
PROC [tile: CStitching.Tile, data:
REF]
-- CStitching.TileProc -- = {
tile and clipBasisTile adjoin and form a material edge
y1: Tad = MAX[tile.SEdge, clipBasisTile.SEdge];
y2: Tad = MIN[tile.NEdge, clipBasisTile.NEdge];
IF y1 < y2
THEN
result.ChangeRect[[x1: clipBasisTile.WEdge-minSpacing, y1: y1-sideSpacing, x2: clipBasisTile.WEdge, y2: y2+sideSpacing], NIL];
};
ClipE:
PROC [tile: CStitching.Tile, data:
REF]
-- CStitching.TileProc -- = {
tile and clipBasisTile adjoin and form a material edge
y1: Tad = MAX[tile.SEdge, clipBasisTile.SEdge];
y2: Tad = MIN[tile.NEdge, clipBasisTile.NEdge];
IF y1 < y2
THEN
result.ChangeRect[[x1: clipBasisTile.EEdge, y1: y1-sideSpacing, x2: clipBasisTile.EEdge+minSpacing, y2: y2+sideSpacing], NIL];
};
clipBasisTile: CStitching.Tile = tile;
basis.EnumerateNeighborhood[tile: tile,
eachTileS: (IF curDir = N THEN ClipS ELSE NIL),
eachTileW: (IF curDir = E THEN ClipW ELSE NIL),
eachTileN: (IF curDir = S THEN ClipN ELSE NIL),
eachTileE: (IF curDir = W THEN ClipE ELSE NIL),
skip: $covered];
};
IncludeOriginal:
PROC [tile: CStitching.Tile, data:
REF]
-- CStitching.TileProc -- = {
tileArea: TadRect = tile.Area;
result.ChangeRect[tile.Area, $covered];
};
enlargeTable: ARRAY Direction OF Tad;
FOR d: Direction
IN Direction
DO
enlargeTable[d] ← (IF d = curDir THEN maxRadialEnlargement ELSE 0);
ENDLOOP;
result ← CStitching.NewTesselation[];
basis.EnumerateArea[rect: CDBasics.universe, eachTile: Enlarge];
basis.EnumerateArea[rect: CDBasics.universe, eachTile: Clip];
basis.EnumerateArea[rect: CDBasics.universe, eachTile: IncludeOriginal];
basis ← DisposeTesselation[basis];
basis ← result;
};
maxRadialEnlargement: Tad = NARROW[spec.first, REF Nm]^*ms.tadsPerNm;
minSpacing: Tad = NARROW[spec.rest.first, REF Nm]^*ms.tadsPerNm;
basis: Tesselation ← GenerateCover[ms, spec.rest.rest.first, extInfluenceDiameter+2*maxRadialEnlargement+minSpacing];
EnlargeToThe[N, minSpacing+2*maxRadialEnlargement];
EnlargeToThe[S, minSpacing+2*maxRadialEnlargement];
EnlargeToThe[E, minSpacing];
EnlargeToThe[W, minSpacing];
basis ← NIL;
END;
EnlargeFn:
PROC [ ms: MaskState, spec:
LIST
OF
REF
ANY, extInfluenceDiameter: Tad ← 0 ]
RETURNS [ result: Tesselation ]
-- CoverProc -- =
BEGIN
deltaDiameterNm: Nm = NARROW[spec.first, REF Nm]^;
deltaDiameterTads: MEBESPixels = deltaDiameterNm*ms.tadsPerNm;
SELECT deltaDiameterTads
FROM
=0 => result ← GenerateCover[ms, spec.rest.first, extInfluenceDiameter];
>0 =>
BEGIN
EnlargeTile:
PROC [tile: CStitching.Tile, data:
REF
ANY]
-- CStitching.PerTileProc -- =
{result.ChangeRect[Bloat[tile.Area, deltaDiameterTads], tile.value]};
basis: Tesselation ← GenerateCover[ms, spec.rest.first, extInfluenceDiameter+deltaDiameterTads];
result ← NewTesselation[NIL];
[] ← basis.EnumerateArea[rect: CDBasics.universe, eachTile: EnlargeTile];
basis ← DisposeTesselation[basis];
END;
<0 =>
BEGIN
EnlargeGap:
PROC [tile: CStitching.Tile, data:
REF
ANY]
-- CStitching.PerTileProc -- =
{result.ChangeRect[CDBasics.Intersection[CDBasics.universe, Bloat[tile.Area, -deltaDiameterTads]], tile.value]};
basis: Tesselation ← GenerateCover[ms, spec.rest.first, extInfluenceDiameter-deltaDiameterTads];
result ← NewTesselation[NIL];
result.ChangeRect[CDBasics.universe, $covered];
[] ← basis.EnumerateArea[rect: CDBasics.universe, eachTile: EnlargeGap, skip: $covered];
basis ← DisposeTesselation[basis];
END;
ENDCASE => ERROR;
END;
DeNotchFn:
PROC [ ms: MaskState, spec:
LIST
OF
REF
ANY, extInfluenceDiameter: Tad ← 0 ]
RETURNS [ result: Tesselation ]
-- CoverProc -- =
BEGIN
deltaDiams: ARRAY [-1..2] OF Nm ← ALL[0];
ndd: [0..2];
newSpec: REF;
FOR ndd𡤀, ndd+1
WHILE ndd<2
DO
WITH spec.first
SELECT
FROM
rnm: REF Nm => {deltaDiams[ndd] ← rnm^; spec ← spec.rest};
ENDCASE => EXIT;
ENDLOOP;
newSpec ← spec.first;
FOR i: [0..2]
IN [0..ndd]
DO
IF deltaDiams[i] # deltaDiams[i-1]
THEN
newSpec ← LIST[$Enlarge, NEW[Nm ← (deltaDiams[i]-deltaDiams[i-1])], newSpec];
ENDLOOP;
result ← GenerateCover[ms, newSpec, extInfluenceDiameter];
END;
SizeLEqFn:
PROC [ ms: MaskState, spec:
LIST
OF
REF
ANY, extInfluenceDiameter: Tad ← 0 ]
RETURNS [ result: Tesselation ]
--
CoverProc -- =
SizeLEq returns a tesselation of all isolated $covered rectangles from input tesselation L3 which are no larger than L1 x L2 nanometers, where L1, L2, and L3 are the elements of spec. Shapes larger than L1 x L2 nanometers, as well as non-rectangular shapes, are omitted. This was added February 27, 1986 to handle VTI's requirement to enlarge minimum size vias over diffusion. JHH.
BEGIN
sieveWidth: Nm = NARROW[spec.first, REF Nm]^;
sieveHeight: Nm = NARROW[spec.rest.first, REF Nm]^;
sieveWidthTads: MEBESPixels = MAX[sieveWidth, sieveHeight]*ms.tadsPerNm;
sieveHeightTads: MEBESPixels = MIN[sieveWidth, sieveHeight]*ms.tadsPerNm;
BEGIN
SizeFilterTile:
PROC [tile: CStitching.Tile, data:
REF
ANY]
-- CStitching.PerTileProc -- =
{
tileWidth, tileHeight: Tad;
IsolatedTile:
PROC [tile: CStitching.Tile]
RETURNS [b:
BOOLEAN] ~ {
Returns TRUE iff tile doesn't touch other tiles of the same value.
Assumes the cornerstitched plane has tiles of only two values.
en: CStitching.Tile ← tile.EN;
ws: CStitching.Tile ← tile.WS;
b ← (
(en.value # tile.value) AND (en.WEdge < tile.WEdge) AND
(ws.value # tile.value) AND (ws.EEdge > tile.EEdge)
);
};
tileWidth ← MAX[(tile.EEdge - tile.WEdge), (tile.NEdge - tile.SEdge)];
tileHeight ← MIN[(tile.EEdge - tile.WEdge), (tile.NEdge - tile.SEdge)];
IF (
IsolatedTile[tile] AND
(sieveWidthTads >= tileWidth) AND
(sieveHeightTads >= tileHeight)
) THEN
result.ChangeRect[rect: tile.Area, new: $covered];
result ← CStitching.NewTesselation[];
[] ← CStitching.EnumerateArea[
plane: GenerateCover[ms, spec.rest.rest.first, extInfluenceDiameter],
rect: CDBasics.universe,
eachTile: SizeFilterTile];
END;
END;
Init:
PROC =
BEGIN
Atom.PutProp[$CDMEBESCoverProcs, $NOT, NEW[CoverProc ← NOTFn]];
Atom.PutProp[$CDMEBESCoverProcs, $AND, NEW[CoverProc ← ANDFn]];
Atom.PutProp[$CDMEBESCoverProcs, $ANDNOT, NEW[CoverProc ← ANDNOTFn]];
Atom.PutProp[$CDMEBESCoverProcs, $OR, NEW[CoverProc ← ORFn]];
Atom.PutProp[$CDMEBESCoverProcs, $MeasureRuns, NEW[CoverProc ← MeasureRuns]];
Atom.PutProp[$CDMEBESCoverProcs, $RestrictedEnlarge, NEW[CoverProc ← RestrictedEnlarge]];
Atom.PutProp[$CDMEBESCoverProcs, $EnlargeSparse, NEW[CoverProc ← EnlargeSparse]];
Atom.PutProp[$CDMEBESCoverProcs, $Enlarge, NEW[CoverProc ← EnlargeFn]];
Atom.PutProp[$CDMEBESCoverProcs, $DeNotch, NEW[CoverProc ← DeNotchFn]];
Atom.PutProp[$CDMEBESCoverProcs, $SizeLEq, NEW[CoverProc ← SizeLEqFn]];
END;
Init[];
END. -- of CDMEBESGeomFns