GetContact:
PUBLIC
PROC [layer1, layer2: Route.Layer]
RETURNS [
CD.Object] =
-- Create a big contact; may return the same object
-- on multiple calls; the contact object should be considered
-- to be read-only.
BEGIN
RETURN[CDSimpleRules.Contact[layer1, layer2]]
END;
GetBigContact:
PUBLIC
PROC [size: Route.Position, layer1, layer2: Route.Layer]
RETURNS [obj:
CD.Object ←
NIL] =
-- Create a big contact; may return NIL (if big contacts aren't defined
-- may return the same object
-- on multiple calls; the contact object should be considered
-- to be read-only.
BEGIN
l1a: ATOM ← CD.LayerKey[layer1];
l2a: ATOM ← CD.LayerKey[layer2];
tech: CD.Technology ← LToTech[layer1];
IF tech # LToTech[layer2] THEN Route.Error[callingError, "Invalid layers"];
SELECT tech.key
FROM
$cmos => NULL;
$cmosB => {
SELECT
TRUE
FROM
(l1a = $met2 AND l2a = $met) OR (l1a = $met AND l2a = $met2) => obj ← CDAtomicObjects.CreateAtomicOb[$C2LargeVia, size, tech];
(l1a = $pol AND l2a = $met) => obj tomicObjects.CreateAtomicOb[$C2LargeSimpleCon, size, tech, layer1];
(l1a = $met AND l2a = $pol) => obj tomicObjects.CreateAtomicOb[$C2LargeSimpleCon, size, tech, layer2];
ENDCASE => Route.Error[callingError, "Invalid layers"]};
ENDCASE => Route.Error[callingError, "Unsupported technology"]
END;
CmosDesignRules:
PUBLIC
PROC [dr: Route.DesignRules, trunkLayer, branchLayer: Route.Layer] =
Define the routing design rules for CMOS double or single layer metal.
BEGIN
cndObj: CD.Object ← NARROW[GetContact[trunkLayer, branchLayer], CD.Object];
dr.trunkLayer ← trunkLayer;
dr.branchLayer ← branchLayer;
dr.branchWidth ← GetWidth[dr, branchLayer];
dr.branchSpacing ← GetSpacing[dr, branchLayer];
dr.branchToContact ← GetMaterialToContact[dr, branchLayer, cndObj];
dr.branchToBranch ← MAX[dr.branchWidth + dr.branchSpacing, dr.branchToContact];
dr.branchOffset ← dr.branchSpacing;
dr.trunkWidth ← GetWidth[dr, trunkLayer];
dr.trunkSpacing ← GetSpacing[dr, trunkLayer];
dr.trunkToContact ← GetMaterialToContact[dr, trunkLayer, cndObj];
dr.trunkOffset ← dr.trunkSpacing;
dr.contactSize ← ContactSize[cndObj];
dr.contactToContact ← dr.contactSize + MAX[dr.trunkSpacing, dr.branchSpacing];
dr.trunkToTrunk ← MAX[dr.trunkWidth + dr.trunkSpacing, dr.trunkToContact, dr.contactToContact];
dr.pinSpacing ← dr.contactSize/2 + dr.branchWidth/2 + dr.branchSpacing;
-- dr.trunkToEdge ← MAX[dr.branchSpacing, dr.trunkSpacing] + MAX[dr.contactSize, dr.trunkWidth, dr.branchWidth]/2;
dr.trunkToEdge ← dr.trunkToTrunk;
END;