All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.sun.electric.tool.drc.MTDRCLayoutTool Maven / Gradle / Ivy

/* -*- tab-width: 4 -*-
 *
 * Electric(tm) VLSI Design System
 *
 * File: Quick.java
 *
 * Copyright (c) 2004, Oracle and/or its affiliates. All rights reserved.
 *
 * Electric(tm) is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 3 of the License, or
 * (at your option) any later version.
 *
 * Electric(tm) is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with Electric(tm); see the file COPYING.  If not, write to
 * the Free Software Foundation, Inc., 59 Temple Place, Suite 330,
 * Boston, Mass 02111-1307, USA.
 */
package com.sun.electric.tool.drc;

import com.sun.electric.database.geometry.*;
import com.sun.electric.database.hierarchy.Cell;
import com.sun.electric.database.hierarchy.Export;
import com.sun.electric.database.network.Netlist;
import com.sun.electric.database.network.Network;
import com.sun.electric.database.prototype.NodeProto;
import com.sun.electric.database.prototype.PortOriginal;
import com.sun.electric.database.prototype.PortProto;
import com.sun.electric.database.text.TextUtils;
import com.sun.electric.database.topology.ArcInst;
import com.sun.electric.database.topology.Geometric;
import com.sun.electric.database.topology.NodeInst;
import com.sun.electric.database.topology.PortInst;
import com.sun.electric.database.variable.Variable;
import com.sun.electric.technology.*;
import com.sun.electric.technology.technologies.Generic;
import com.sun.electric.tool.Job;
import com.sun.electric.tool.Consumer;
import com.sun.electric.tool.user.ErrorLogger;
import com.sun.electric.util.ElapseTimer;
import com.sun.electric.util.math.DBMath;
import com.sun.electric.util.math.FixpTransform;
import com.sun.electric.util.math.MutableDouble;
import com.sun.electric.util.math.MutableInteger;

import java.awt.geom.Area;
import java.awt.geom.Point2D;
import java.awt.geom.Rectangle2D;
import java.util.*;

/**
 * This is the "quick" DRC which does full hierarchical examination of the circuit.
 * 

* The "quick" DRC works as follows: * It first examines every primitive node and arc in the cell * For each layer on these objects, it examines everything surrounding it, even in subcells * R-trees are used to limit search. * Where cell instances are found, the contents are examined recursively * It next examines every cell instance in the cell * All other instances within the surrounding area are considered * When another instance is found, the two instances are examined for interaction * A cache is kept of instance pairs in specified configurations to speed-up arrays * All objects in the other instance that are inside the bounds of the first are considered * The other instance is hierarchically examined to locate primitives in the area of consideration * For each layer on each primitive object found in the other instance, * Examine the contents of the first instance for interactions about that layer *

* Since Electric understands connectivity, it uses this information to determine whether two layers * are connected or not. However, if such global connectivity is propagated in the standard Electric * way (placing numbers on exports, descending into the cell, and pulling the numbers onto local networks) * then it is not possible to decompose the DRC for multiple processors, since two different processors * may want to write global network information on the same local networks at once. *

* To solve this problem, the "quick" DRC determines how many instances of each cell exist. For every * network in every cell, an array is built that is as large as the number of instances of that cell. * This array contains the global network number for that each instance of the cell. The algorithm for * building these arrays is quick (1 second for a million-transistor chip) and the memory requirement * is not excessive (8 megabytes for a million-transistor chip). It uses the CheckInst and CheckProto * objects. * @author Steve Rubin, Gilda Garreton */ public class MTDRCLayoutTool extends MTDRCTool { private boolean ignoreExtensionRules = true; public MTDRCLayoutTool(DRC.DRCPreferences dp, Cell c, boolean ignoreExtensionR, Consumer consumer) { super("Design-Rule Layout Check " + c, dp, c, consumer); this.ignoreExtensionRules = ignoreExtensionR; } @Override boolean checkArea() {return false;} // returns the number of errors found @Override public MTDRCResult runTaskInternal(Layer taskKey) { return (new Task(rules, this)).runTaskInternal(taskKey); } private class Task { private HashMap checkInsts; private HashMap checkProtos; private HashMap networkLists; private HashMap nodesMap = new HashMap(); // for node caching private Map od2Layers = new HashMap(3); /** to control OD2 combination in the same die according to foundries */ private List instanceInteractionList = new ArrayList(); // private List instanceInteractionList = new ArrayList(); private Map> instanceInteractionMap = new HashMap>(); /** * The DRCExclusion object lists areas where Generic:DRC-Nodes exist to ignore errors. */ private Map exclusionMap = new HashMap(); /** a NodeInst that is too tiny for its connection. */ private NodeInst tinyNodeInst; /** the other Geometric in "tiny" errors. */ private Geometric tinyGeometric; /** for tracking the time of good DRC. */ private HashSet goodSpacingDRCDate = new HashSet(); /** for tracking cells that need to clean good DRC vars */ private HashSet cleanSpacingDRCDate = new HashSet(); /** for tracking the time of good DRC. */ private HashSet goodAreaDRCDate = new HashSet(); /** for tracking cells that need to clean good DRC vars */ private HashSet cleanAreaDRCDate = new HashSet(); /** Miscellanous data for DRC */ private DRC.ReportInfo reportInfo; // To speed up the layer process ValidationLayers validLayers; // New for the MT code private Layer theLayer; private Layer.Function.Set thisLayerFunction; private DRCRules currentRules; private Job job; Task(DRCRules rules, Job j) { currentRules = rules; job = j; } /** * This is the entry point for DRC. *

* Method to do a hierarchical DRC check on cell "cell". * If "count" is zero, check the entire cell. * If "count" is nonzero, only check that many instances (in "nodesToCheck") and set the * entry in "validity" TRUE if it is DRC clean. */ // returns the number of errors found private MTDRCResult runTaskInternal(Layer taskKey) { String name; Technology tech = topCell.getTechnology(); if (taskKey != null) { name = "Layer " + taskKey.getName(); this.thisLayerFunction = Layer.getMultiLayersSet(taskKey); } else { name = "Node Min. Size"; } theLayer = taskKey; // if checking specific instances, adjust options and processor count Geometric[] geomsToCheck = null; // for now. int count = (geomsToCheck != null) ? geomsToCheck.length : 0; boolean[] validity = null; Rectangle2D bounds = null; ErrorLogger errorLogger = DRC.getDRCErrorLogger(true, ", " + name); reportInfo = new DRC.ReportInfo(errorLogger, tech, dp, (count > 0)); // caching bits System.out.println("Running DRC for " + name + " with " + DRC.explainBits(reportInfo.activeSpacingBits, dp)); // Check if there are DRC rules for particular tech // Nothing to check for this particular technology if (currentRules == null || currentRules.getNumberOfRules() == 0) return null; // cache valid layers for this technology // cacheValidLayers(tech); validLayers = new ValidationLayers(reportInfo.errorLogger, topCell, rules); // clean out the cache of instances instanceInteractionList.clear(); // determine if min area must be checked (if any layer got valid data) // cellsMap.clear(); nodesMap.clear(); // initialize all cells for hierarchical network numbering checkProtos = new HashMap(); checkInsts = new HashMap(); // initialize cells in tree for hierarchical network numbering Netlist netlist = topCell.getNetlist(); CheckProto cp = checkEnumerateProtos(topCell, netlist); // now recursively examine, setting information on all instances cp.hierInstanceCount = 1; reportInfo.checkTimeStamp = 0; checkEnumerateInstances(topCell); // now allocate space for hierarchical network arrays //int totalNetworks = 0; networkLists = new HashMap(); for (Map.Entry e : checkProtos.entrySet()) { Cell libCell = e.getKey(); CheckProto subCP = e.getValue(); if (subCP.hierInstanceCount > 0) { // allocate net number lists for every net in the cell for (Iterator nIt = subCP.netlist.getNetworks(); nIt.hasNext();) { Network net = nIt.next(); Integer[] netNumbers = new Integer[subCP.hierInstanceCount]; for (int i = 0; i < subCP.hierInstanceCount; i++) netNumbers[i] = new Integer(0); networkLists.put(net, netNumbers); //totalNetworks += subCP.hierInstanceCount; } } for (Iterator nIt = libCell.getNodes(); nIt.hasNext();) { NodeInst ni = nIt.next(); NodeProto np = ni.getProto(); if (!ni.isCellInstance()) continue; // ignore documentation icons if (ni.isIconOfParent()) continue; CheckInst ci = checkInsts.get(ni); CheckProto ocp = getCheckProto((Cell) np); ci.offset = ocp.totalPerCell; } reportInfo.checkTimeStamp++; for (Iterator nIt = libCell.getNodes(); nIt.hasNext();) { NodeInst ni = nIt.next(); NodeProto np = ni.getProto(); if (!ni.isCellInstance()) continue; // ignore documentation icons if (ni.isIconOfParent()) continue; CheckProto ocp = getCheckProto((Cell) np); if (ocp.timeStamp != reportInfo.checkTimeStamp) { CheckInst ci = checkInsts.get(ni); ocp.timeStamp = reportInfo.checkTimeStamp; ocp.totalPerCell += subCP.hierInstanceCount * ci.multiplier; } } } // now fill in the hierarchical network arrays reportInfo.checkTimeStamp = 0; reportInfo.checkNetNumber = 1; HashMap enumeratedNets = new HashMap(); for (Iterator nIt = cp.netlist.getNetworks(); nIt.hasNext();) { Network net = nIt.next(); enumeratedNets.put(net, new Integer(reportInfo.checkNetNumber)); reportInfo.checkNetNumber++; } checkEnumerateNetworks(topCell, cp, 0, enumeratedNets); if (count <= 0) System.out.println("Found " + reportInfo.checkNetNumber + " networks"); // now search for DRC exclusion areas exclusionMap.clear(); accumulateExclusion(topCell); // now do the DRC int logsFound = 0; if (count == 0) { // just do full DRC here checkThisCell(topCell, 0, bounds); // sort the errors by layer errorLogger.sortLogs(); } else { // check only these "count" instances (either an incremental DRC or a quiet one...from Array command) if (validity == null) { // not a quiet DRC, so it must be incremental logsFound = errorLogger.getNumLogs(); } // @TODO missing counting this number of errors. checkTheseGeometrics(topCell, count, geomsToCheck, validity); } if (errorLogger != null) { errorLogger.termLogging(true); logsFound = errorLogger.getNumLogs() - logsFound; } // -2 if cells and subcells are ok // Commentted out on May 2, 2006. Not sure why this condition is valid // If goodDRCDate contains information of DRC clean cells, it destroys that information -> wrong // if (totalErrors != 0 && totalErrors != -2) goodDRCDate.clear(); // some cells were sucessfully checked: save that information in the database // some cells don't have valid DRC date anymore and therefore they should be clean // This is only going to happen if job was not aborted. return new MTDRCResult(errorLogger.getNumErrors(), errorLogger.getNumWarnings(), !checkAbort(), goodSpacingDRCDate, cleanSpacingDRCDate, goodAreaDRCDate, cleanAreaDRCDate, null); } /*************************** QUICK DRC CELL EXAMINATION ***************************/ /** * Method to check the contents of cell "cell" with global network index "globalIndex". * Returns positive if errors are found, zero if no errors are found, negative on internal error. * * @param cell * @param globalIndex * @param bounds * @return positive number if errors are found, zero if no errors are found and check the cell, -1 if job was * aborted and -2 if cell is OK with DRC date stored. */ private int checkThisCell(Cell cell, int globalIndex, Rectangle2D bounds) { // Job aborted or scheduled for abort if (checkAbort()) return -1; // Cell already checked // if (cellsMap.get(cell) != null) if (getCheckProto(cell).cellChecked) return (0); // Previous # of errors/warnings int prevErrors = 0; int prevWarns = 0; if (reportInfo.errorLogger != null) { prevErrors = reportInfo.errorLogger.getNumErrors(); prevWarns = reportInfo.errorLogger.getNumWarnings(); } // cellsMap.put(cell, cell); // Check if cell doesn't have special annotation Variable drcVar = cell.getVar(DRC.DRC_ANNOTATION_KEY); if (drcVar != null && drcVar.getObject().toString().startsWith("black")) { // Skipping this one assert(reportInfo.totalSpacingMsgFound==0); return reportInfo.totalSpacingMsgFound; } // first check all subcells boolean allSubCellsStillOK = true; Area area = exclusionMap.get(cell); for (Iterator it = cell.getNodes(); it.hasNext();) { // Job aborted or scheduled for abort if (checkAbort()) return -1; NodeInst ni = it.next(); NodeProto np = ni.getProto(); if (!ni.isCellInstance()) continue; // ignore documentation icons if (ni.isIconOfParent()) continue; // Check DRC exclusion regions if (area != null && area.contains(ni.getBounds())) continue; // excluded // ignore if not in the area Rectangle2D subBounds = bounds; // sept30 if (subBounds != null) { if (!ni.getBounds().intersects(bounds)) continue; FixpTransform trans = ni.rotateIn(); FixpTransform xTrnI = ni.translateIn(); trans.preConcatenate(xTrnI); subBounds = new Rectangle2D.Double(); subBounds.setRect(bounds); DBMath.transformRect(subBounds, trans); } CheckProto cp = getCheckProto((Cell) np); if (cp.cellChecked && !cp.cellParameterized) continue; // recursively check the subcell CheckInst ci = checkInsts.get(ni); int localIndex = globalIndex * ci.multiplier + ci.localIndex + ci.offset; int retval = checkThisCell((Cell) np, localIndex, subBounds); if (retval < 0) return retval; // if cell is in goodDRCDate it means it changes its date for some reason. // This happen when a subcell was reloaded and DRC again. The cell containing // the subcell instance must be re-DRC to make sure changes in subCell don't affect // the parent one. if (retval > 0 || goodSpacingDRCDate.contains(np)) allSubCellsStillOK = false; // if (retval > 0) // allSubCellsStillOK = false; } // prepare to check cell CheckProto cp = getCheckProto(cell); cp.cellChecked = true; boolean skipLayer = skipLayerInvalidForMinArea(theLayer); boolean checkArea = (cell == topCell && !skipLayer && !dp.ignoreAreaCheck && reportInfo.errorTypeSearch != DRC.DRCCheckMode.ERROR_CHECK_CELL); // if the cell hasn't changed since the last good check, stop now Date lastSpacingGoodDate = DRC.getLastDRCDateBasedOnBits(cell, true, reportInfo.activeSpacingBits, !reportInfo.inMemory); Date lastAreaGoodDate = DRC.getLastDRCDateBasedOnBits(cell, false, -1, !reportInfo.inMemory); if (allSubCellsStillOK && DRC.isCellDRCDateGood(cell, lastSpacingGoodDate) && (!checkArea || DRC.isCellDRCDateGood(cell, lastAreaGoodDate))) { return 0; } // announce progress ElapseTimer timer = ElapseTimer.createInstance().start(); if (printLog) System.out.println("Checking " + cell + ", " + theLayer); // now look at every node and arc here reportInfo.totalSpacingMsgFound = 0; // Check the area first but only when is not incremental // Only for the most top cell if (checkArea) { assert(reportInfo.totalSpacingMsgFound == 0); int totalAreaMsgFound = MTDRCAreaTool.checkMinArea(theLayer, topCell, reportInfo, currentRules, cellLayersCon, job, null); if (totalAreaMsgFound == 0) goodAreaDRCDate.add(cell); else cleanAreaDRCDate.add(cell); } for (Iterator it = cell.getNodes(); it.hasNext();) { // Job aborted or scheduled for abort if (checkAbort()) return -1; NodeInst ni = it.next(); if (bounds != null) { if (!ni.getBounds().intersects(bounds)) continue; } if (area != null) { if (area.contains(ni.getBounds())) { continue; } } boolean ret = false; if (thisLayerFunction == null) // checking min size only { if (ni.isCellInstance()) continue; ret = checkNodeInst(ni, globalIndex); } else { ret = (ni.isCellInstance()) ? checkCellInst(ni, globalIndex) : checkNodeInst(ni, globalIndex); } if (ret) { reportInfo.totalSpacingMsgFound++; if (reportInfo.errorTypeSearch == DRC.DRCCheckMode.ERROR_CHECK_CELL) break; } } if (thisLayerFunction != null) // only when it is layer related { Technology cellTech = cell.getTechnology(); for (Iterator it = cell.getArcs(); it.hasNext();) { // Job aborted or scheduled for abort if (checkAbort()) return -1; ArcInst ai = it.next(); Technology tech = ai.getProto().getTechnology(); if (tech != cellTech) { DRC.createDRCErrorLogger(reportInfo, DRC.DRCErrorType.TECHMIXWARN, " belongs to " + tech.getTechName(), cell, 0, 0, null, null, ai, null, null, null, null); continue; } if (bounds != null) { if (!ai.getBounds().intersects(bounds)) continue; } if (checkArcInst(cp, ai, globalIndex)) { reportInfo.totalSpacingMsgFound++; if (reportInfo.errorTypeSearch == DRC.DRCCheckMode.ERROR_CHECK_CELL) break; } } } // If message founds, then remove any possible good date // !allSubCellsStillOK disconnected on April 18, 2006. totalMsgFound should // dictate if this cell is re-marked. if (reportInfo.totalSpacingMsgFound > 0) // || !allSubCellsStillOK) { cleanSpacingDRCDate.add(cell); } else { // Only mark the cell when it passes with a new version of DRC or didn't have // the DRC bit on // If lastGoodDate == null, wrong bits stored or no date available. if (lastSpacingGoodDate == null) goodSpacingDRCDate.add(cell); } // if there were no errors, remember that if (reportInfo.errorLogger != null && printLog) { int localErrors = reportInfo.errorLogger.getNumErrors() - prevErrors; int localWarnings = reportInfo.errorLogger.getNumWarnings() - prevWarns; timer.end(); if (localErrors == 0 && localWarnings == 0) { System.out.println("\tNo errors/warnings found"); } else { if (localErrors > 0) System.out.println("\tFOUND " + localErrors + " ERRORS"); if (localWarnings > 0) System.out.println("\tFOUND " + localWarnings + " WARNINGS"); } if (Job.getDebug()) System.out.println("\t(took " + timer + ")"); } return reportInfo.totalSpacingMsgFound; } /** * Check Poly for CIF Resolution Errors * * @param poly * @param cell * @param geom * @return true if an error was found. */ private boolean checkResolution(PolyBase poly, Cell cell, Geometric geom) { double minAllowedResolution = reportInfo.minAllowedResolution.getLambda(); if (minAllowedResolution == 0) return false; int count = 0; String resolutionError = ""; Point2D[] points = poly.getPoints(); for (Point2D point : points) { if (DBMath.hasRemainder(point.getX(), minAllowedResolution)) { count++; resolutionError = TextUtils.formatDistance(Math.abs(((point.getX() / minAllowedResolution) % 1) * minAllowedResolution)) + "(X=" + point.getX() + ")"; break; // with one error is enough } else if (DBMath.hasRemainder(point.getY(), minAllowedResolution)) { count++; resolutionError = TextUtils.formatDistance(Math.abs(((point.getY() / minAllowedResolution) % 1) * minAllowedResolution)) + "(Y=" + point.getY() + ")"; break; } } if (count == 0) return false; // no error // there was an error, for now print error Layer layer = poly.getLayer(); DRC.createDRCErrorLogger(reportInfo, DRC.DRCErrorType.RESOLUTION, " resolution of " + resolutionError + " less than " + reportInfo.minAllowedResolution + " on layer " + layer.getName(), cell, 0, 0, "Resolution", null, geom, null, null, null, null); return true; } /** * Private method to check if geometry is covered 100% by exclusion region * * @param geo * @return true if geometry is covered by exclusion */ private boolean coverByExclusion(Geometric geo) { Cell cell = geo.getParent(); Area area = exclusionMap.get(cell); if (area != null && area.contains(geo.getBounds())) { // System.out.println("DRC Exclusion found"); return true; } return false; } /** * Private method to check if geometry given by Poly object is inside an exclusion region * * @param poly * @param parent * @return true if Poly is inside an exclusion region */ private boolean coverByExclusion(Poly poly, Cell parent) { Area area = exclusionMap.get(parent); if (area == null) return false; Point2D[] pts = poly.getPoints(); for (Point2D point : pts) { if (!area.contains(point)) { return false; } } // System.out.println("DRC Exclusion found in coverByExclusion"); return true; } /** * Method to check the design rules about nodeinst "ni". * * @param ni * @param globalIndex * @return true if an error was found. */ private boolean checkNodeInst(NodeInst ni, int globalIndex) { Cell cell = ni.getParent(); Netlist netlist = getCheckProto(cell).netlist; NodeProto np = ni.getProto(); Technology tech = np.getTechnology(); FixpTransform trans = ni.rotateOut(); boolean errorsFound = false; // Skipping special nodes if (NodeInst.isSpecialNode(ni)) return false; // Oct 5; if (!np.getTechnology().isLayout()) return (false); // only layout nodes if (np.getFunction().isPin()) return false; // Sept 30 if (coverByExclusion(ni)) return false; // no errors // Already done if (nodesMap.get(ni) != null) return (false); nodesMap.put(ni, ni); if (thisLayerFunction != null) { // get all of the polygons on this node Poly[] nodeInstPolyList = tech.getShapeOfNode(ni, true, reportInfo.ignoreCenterCuts, thisLayerFunction); convertPseudoLayers(ni, nodeInstPolyList); int tot = nodeInstPolyList.length; boolean isTransistor = np.getFunction().isTransistor(); Technology.MultiCutData multiCutData = tech.getMultiCutData(ni); // examine the polygons on this node for (int j = 0; j < tot; j++) { Poly poly = nodeInstPolyList[j]; Layer layer = poly.getLayer(); if (layer == null) continue; if (coverByExclusion(poly, cell)) continue; // Checking combination boolean ret = DRC.checkOD2Combination(tech, ni, layer, od2Layers, reportInfo); if (ret) { // panic errors -> return regarless errorTypeSearch return true; } poly.transform(trans); // Checking resolution only after transformation ret = checkResolution(poly, cell, ni); if (ret) { if (reportInfo.errorTypeSearch == DRC.DRCCheckMode.ERROR_CHECK_CELL) return true; errorsFound = true; } // determine network for this polygon int netNumber = getDRCNetNumber(netlist, poly.getPort(), ni, globalIndex); ret = badBox(poly, layer, netNumber, tech, ni, trans, cell, globalIndex, multiCutData); if (ret) { if (reportInfo.errorTypeSearch == DRC.DRCCheckMode.ERROR_CHECK_CELL) return true; errorsFound = true; } ret = checkMinWidth(ni, layer, poly, tot == 1); if (ret) { if (reportInfo.errorTypeSearch == DRC.DRCCheckMode.ERROR_CHECK_CELL) return true; errorsFound = true; } // Check select over transistor poly // Assumes polys on transistors fulfill condition by construction // ret = !isTransistor && checkSelectOverPolysilicon(ni, layer, poly, cell); ret = !isTransistor && checkExtensionRules(ni, layer, poly, cell); if (ret) { if (reportInfo.errorTypeSearch == DRC.DRCCheckMode.ERROR_CHECK_CELL) return true; errorsFound = true; } // ret = checkExtensionRule(ni, layer, poly, cell); // if (ret) // { // if (errorTypeSearch == DRC.DRCCheckMode.ERROR_CHECK_CELL) return true; // errorsFound = true; // } if (validLayers.isABadLayer(tech, layer.getIndex())) { DRC.createDRCErrorLogger(reportInfo, DRC.DRCErrorType.BADLAYERERROR, null, cell, 0, 0, null, poly, ni, layer, null, null, null); if (reportInfo.errorTypeSearch == DRC.DRCCheckMode.ERROR_CHECK_CELL) return true; errorsFound = true; } } } else { // Node errors! if (DRC.checkNodeAgainstCombinationRules(ni, reportInfo)) { if (reportInfo.errorTypeSearch == DRC.DRCCheckMode.ERROR_CHECK_CELL) return true; errorsFound = true; } // if (np instanceof PrimitiveNode && DRC.isForbiddenNode(currentRules, ((PrimitiveNode) np).getPrimNodeIndexInTech(), -1, // DRCTemplate.DRCRuleType.FORBIDDEN)) // { // DRC.createDRCErrorLogger(reportInfo, DRC.DRCErrorType.FORBIDDEN, " is not allowed by selected foundry", cell, -1, -1, null, null, ni, null, null, null, null); // if (reportInfo.errorTypeSearch == DRC.DRCCheckMode.ERROR_CHECK_CELL) return true; // errorsFound = true; // } // check node for minimum size if (DRC.checkNodeSize(ni, cell, reportInfo)) errorsFound = true; } return errorsFound; } /** * Method to check the design rules about arcinst "ai". * Returns true if errors were found. */ private boolean checkArcInst(CheckProto cp, ArcInst ai, int globalIndex) { // Nothing to check if (thisLayerFunction == null) return false; // ignore arcs with no topology Network net = cp.netlist.getNetwork(ai, 0); if (net == null) return false; Integer[] netNumbers = networkLists.get(net); if (nodesMap.get(ai) != null) { return (false); } nodesMap.put(ai, ai); // Check if arc is contained in exclusion region // if (coverByExclusion(ai)) // return false; // no error boolean errorsFound = false; // Checking if the arc is horizontal or vertical Point2D from = ai.getHeadLocation(); Point2D to = ai.getTailLocation(); if (!DBMath.areEquals(from.getX(), to.getX()) && !DBMath.areEquals(from.getY(), to.getY())) { DRC.createDRCErrorLogger(reportInfo, DRC.DRCErrorType.CROOKEDERROR, null, ai.getParent(), -1, -1, null, null, ai, null, null, null, null); if (reportInfo.errorTypeSearch == DRC.DRCCheckMode.ERROR_CHECK_CELL) return true; errorsFound = true; } // get all of the polygons on this arc Technology tech = ai.getProto().getTechnology(); Poly[] arcInstPolyList = tech.getShapeOfArc(ai, thisLayerFunction); // Check resolution before cropping the for (Poly poly : arcInstPolyList) { Layer layer = poly.getLayer(); if (layer == null) continue; if (layer.isNonElectrical()) continue; // Checking resolution boolean ret = checkResolution(poly, ai.getParent(), ai); if (ret) { if (reportInfo.errorTypeSearch == DRC.DRCCheckMode.ERROR_CHECK_CELL) return true; errorsFound = true; } } cropActiveArc(ai, arcInstPolyList); int tot = arcInstPolyList.length; // examine the polygons on this arc for (int j = 0; j < tot; j++) { Poly poly = arcInstPolyList[j]; Layer layer = poly.getLayer(); if (layer == null) continue; if (layer.isNonElectrical()) continue; int layerNum = layer.getIndex(); int netNumber = netNumbers[globalIndex]; // autoboxing boolean ret = badBox(poly, layer, netNumber, tech, ai, DBMath.MATID, ai.getParent(), globalIndex, null); if (ret) { if (reportInfo.errorTypeSearch == DRC.DRCCheckMode.ERROR_CHECK_CELL) return true; errorsFound = true; } ret = checkMinWidth(ai, layer, poly, tot == 1); if (ret) { if (reportInfo.errorTypeSearch == DRC.DRCCheckMode.ERROR_CHECK_CELL) return true; errorsFound = true; } // Check select over transistor poly // ret = checkSelectOverPolysilicon(ai, layer, poly, ai.getParent()); ret = checkExtensionRules(ai, layer, poly, ai.getParent()); if (ret) { if (reportInfo.errorTypeSearch == DRC.DRCCheckMode.ERROR_CHECK_CELL) return true; errorsFound = true; } if (validLayers.isABadLayer(tech, layerNum)) // if (tech == layersValidTech && !layersValid[layerNum]) { DRC.createDRCErrorLogger(reportInfo, DRC.DRCErrorType.BADLAYERERROR, null, ai.getParent(), 0, 0, null, (tot == 1) ? null : poly, ai, layer, null, null, null); if (reportInfo.errorTypeSearch == DRC.DRCCheckMode.ERROR_CHECK_CELL) return true; errorsFound = true; } } return errorsFound; } /** * Method to check the design rules about cell instance "ni". Only check other * instances, and only check the parts of each that are within striking range. * Returns true if an error was found. */ private boolean checkCellInst(NodeInst ni, int globalIndex) { Cell thisCell = (Cell)ni.getProto(); if (!thisCell.isLayout()) return false; // skips non-layout cells. // look for other instances surrounding this one Rectangle2D nodeBounds = ni.getBounds(); // double worstInteractionDistance = reportInfo.worstInteractionDistance; MutableDouble mutableDist = new MutableDouble(0); if (!cellLayersCon.getWorstSpacingDistance(thisCell, mutableDist)) { // System.out.println("No worst spacing distance found in MTDRCLayoutTool:checkCellInst"); return false; } // get transformation out of the instance FixpTransform upTrans = ni.translateOut(ni.rotateOut()); // get network numbering for the instance CheckInst ci = checkInsts.get(ni); int localIndex = globalIndex * ci.multiplier + ci.localIndex + ci.offset; boolean errorFound = false; CheckProto cpNi = getCheckProto(thisCell); double worstInteractionDistance = mutableDist.doubleValue(); Rectangle2D searchBounds = new Rectangle2D.Double( nodeBounds.getMinX() - worstInteractionDistance, nodeBounds.getMinY() - worstInteractionDistance, nodeBounds.getWidth() + worstInteractionDistance * 2, nodeBounds.getHeight() + worstInteractionDistance * 2); instanceInteractionList.clear(); // part3 for (Iterator it = ni.getParent().searchIterator(searchBounds); it.hasNext();) { Geometric geom = it.next(); if (geom == ni) continue; // covered by checkInteraction? if (!(geom instanceof NodeInst)) continue; NodeInst oNi = (NodeInst) geom; // only check other nodes that are numerically higher (so each pair is only checked once) if (oNi.getNodeIndex() <= ni.getNodeIndex()) continue; if (!oNi.isCellInstance()) continue; // see if this configuration of instances has already been done CheckProto cpoNi = getCheckProto((Cell)oNi.getProto()); // assume oNi is a cell // see if this configuration of instances has already been done if (DRC.checkInteraction(instanceInteractionMap, reportInfo.errorTypeSearch, ni, ni, cpNi.cellParameterized, oNi, oNi, cpoNi.cellParameterized, ni, searchBounds)) continue; // if (checkInteraction(ni, null, oNi, null, null)) continue; // found other instance "oNi", look for everything in "ni" that is near it Rectangle2D nearNodeBounds = oNi.getBounds(); if (!cellLayersCon.getWorstSpacingDistance(oNi.getProto(), mutableDist)) { // System.out.println("No worst spacing distance found in Quick:checkThisCellPlease"); continue; } double worstInteractionDistanceLocal = mutableDist.doubleValue(); Rectangle2D subBounds = new Rectangle2D.Double( nearNodeBounds.getMinX() - worstInteractionDistanceLocal, nearNodeBounds.getMinY() - worstInteractionDistanceLocal, nearNodeBounds.getWidth() + worstInteractionDistanceLocal * 2, nearNodeBounds.getHeight() + worstInteractionDistanceLocal * 2); // recursively search instance "ni" in the vicinity of "oNi" boolean ret = checkCellInstContents(subBounds, ni, upTrans, localIndex, oNi, null, globalIndex, null); if (ret) errorFound = true; } return errorFound; } /** * Method to recursively examine the area "bounds" in cell "cell" with global index "globalIndex". * The objects that are found are transformed by "uptrans" to be in the space of a top-level cell. * They are then compared with objects in "oNi" (which is in that top-level cell), * which has global index "topGlobalIndex". */ private boolean checkCellInstContents(Rectangle2D bounds, NodeInst thisNi, FixpTransform upTrans, int globalIndex, NodeInst oNi, NodeInst oNiParent, int topGlobalIndex, NodeInst triggerNi) { // Job aborted or scheduled for abort if (checkAbort()) return true; Cell cell = (Cell) thisNi.getProto(); if (!cell.isLayout()) return false; // skips non-layout cells. CheckProto cpoNi = getCheckProto((Cell)oNi.getProto()); // assume oNi is a cell boolean logsFound = false; Netlist netlist = getCheckProto(cell).netlist; Technology cellTech = cell.getTechnology(); // Need to transform bounds coordinates first otherwise it won't // never overlap Rectangle2D bb = (Rectangle2D) bounds.clone(); FixpTransform downTrans = thisNi.transformIn(); DBMath.transformRect(bb, downTrans); for (Iterator it = cell.searchIterator(bb); it.hasNext();) { Geometric geom = it.next(); // Checking if element is covered by exclusion region // if (coverByExclusion(geom)) // continue; // skips this element if (geom instanceof NodeInst) { NodeInst ni = (NodeInst) geom; NodeProto np = ni.getProto(); if (NodeInst.isSpecialNode(ni)) continue; // Oct 5'04 if (ni.isCellInstance()) { CheckProto cpNi = getCheckProto((Cell)np); // see if this configuration of instances has already been done if (DRC.checkInteraction(instanceInteractionMap, reportInfo.errorTypeSearch, ni, thisNi, cpNi.cellParameterized, oNi, oNiParent, cpoNi.cellParameterized, triggerNi, bb)) continue; // Jan 27'05. Removed on May'05 // You can't discard by interaction becuase two cells could be visited many times // during this type of checking FixpTransform subUpTrans = ni.translateOut(ni.rotateOut()); subUpTrans.preConcatenate(upTrans); CheckInst ci = checkInsts.get(ni); int localIndex = globalIndex * ci.multiplier + ci.localIndex + ci.offset; // changes Sept04: subBound by bb boolean ret = checkCellInstContents(bb, ni, subUpTrans, localIndex, oNi, oNiParent, topGlobalIndex, (triggerNi == null) ? ni : triggerNi); if (ret) { if (reportInfo.errorTypeSearch == DRC.DRCCheckMode.ERROR_CHECK_CELL) return true; logsFound = true; } } else { Technology tech = np.getTechnology(); Poly[] primPolyList = tech.getShapeOfNode(ni, true, reportInfo.ignoreCenterCuts, thisLayerFunction); convertPseudoLayers(ni, primPolyList); int tot = primPolyList.length; if (tot == 0) continue; Technology.MultiCutData multiCutData = tech.getMultiCutData(ni); FixpTransform rTrans = ni.rotateOut(); rTrans.preConcatenate(upTrans); for (int j = 0; j < tot; j++) { Poly poly = primPolyList[j]; Layer layer = poly.getLayer(); if (layer == null) { assert (false); if (Job.LOCALDEBUGFLAG) System.out.println("When is this case?"); continue; } poly.transform(rTrans); // determine network for this polygon int net = getDRCNetNumber(netlist, poly.getPort(), ni, globalIndex); boolean ret = badSubBox(poly, layer, net, tech, ni, rTrans, globalIndex, oNi, topGlobalIndex, multiCutData); if (ret) { if (reportInfo.errorTypeSearch == DRC.DRCCheckMode.ERROR_CHECK_CELL) return true; logsFound = true; } } } } else { ArcInst ai = (ArcInst) geom; Technology tech = ai.getProto().getTechnology(); if (tech != cellTech) { // Leaving this tech conflict to the code in checkCell() // DRC.createDRCErrorLogger(reportInfo, DRC.DRCErrorType.TECHMIXWARN, " belongs to " + tech.getTechName(), cell, 0, 0, null, null, ai, null, null, null, null); continue; } Poly[] arcPolyList = tech.getShapeOfArc(ai, thisLayerFunction); int tot = arcPolyList.length; for (int j = 0; j < tot; j++) arcPolyList[j].transform(upTrans); cropActiveArc(ai, arcPolyList); for (int j = 0; j < tot; j++) { Poly poly = arcPolyList[j]; Layer layer = poly.getLayer(); if (layer == null) continue; if (layer.isNonElectrical()) { assert(false); // should this happen? continue; } Network jNet = netlist.getNetwork(ai, 0); int net = -1; if (jNet != null) { Integer[] netList = networkLists.get(jNet); net = netList[globalIndex].intValue(); } boolean ret = badSubBox(poly, layer, net, tech, ai, upTrans, globalIndex, oNi, topGlobalIndex, null); if (ret) { if (reportInfo.errorTypeSearch == DRC.DRCCheckMode.ERROR_CHECK_CELL) return true; logsFound = true; } } } } return logsFound; } /** * Method to examine polygon "poly" layer "layer" network "net" technology "tech" geometry "geom" * which is in cell "cell" and has global index "globalIndex". * The polygon is compared against things inside node "oNi", and that node's parent has global index "topGlobalIndex". */ private boolean badSubBox(Poly poly, Layer layer, int net, Technology tech, Geometric geom, FixpTransform trans, int globalIndex, NodeInst oNi, int topGlobalIndex, Technology.MultiCutData multiCutData) { // see how far around the box it is necessary to search double maxSize = poly.getMaxSize(); MutableDouble mutableDist = new MutableDouble(0); boolean found = currentRules.getMaxSurround(layer, maxSize, mutableDist); if (!found) return false; // get bounds Rectangle2D bounds = new Rectangle2D.Double(); bounds.setRect(poly.getBounds2D()); FixpTransform downTrans = oNi.rotateIn(); FixpTransform tTransI = oNi.translateIn(); downTrans.preConcatenate(tTransI); DBMath.transformRect(bounds, downTrans); FixpTransform upTrans = oNi.translateOut(oNi.rotateOut()); CheckInst ci = checkInsts.get(oNi); int localIndex = topGlobalIndex * ci.multiplier + ci.localIndex + ci.offset; // search in the area surrounding the box double bound = mutableDist.doubleValue(); bounds.setRect(bounds.getMinX() - bound, bounds.getMinY() - bound, bounds.getWidth() + bound * 2, bounds.getHeight() + bound * 2); return (badBoxInArea(poly, layer, tech, net, geom, trans, globalIndex, bounds, (Cell) oNi.getProto(), localIndex, oNi.getParent(), topGlobalIndex, upTrans, multiCutData, false)); // true in this case you don't want to check Geometric.objectsTouch(nGeom, geom) if nGeom and geom are in the // same cell because they could come from different cell instances. } /** * Method to examine a polygon to see if it has any errors with its surrounding area. * The polygon is "poly" on layer "layer" on network "net" from technology "tech" from object "geom". * Checking looks in cell "cell" global index "globalIndex". * Object "geom" can be transformed to the space of this cell with "trans". * Returns TRUE if a spacing error is found relative to anything surrounding it at or below * this hierarchical level. */ private boolean badBox(Poly poly, Layer layer, int net, Technology tech, Geometric geom, FixpTransform trans, Cell cell, int globalIndex, Technology.MultiCutData multiCutData) { // see how far around the box it is necessary to search double maxSize = poly.getMaxSize(); MutableDouble mutableDist = new MutableDouble(0); boolean found = currentRules.getMaxSurround(layer, maxSize, mutableDist); if (!found) return false; // get bounds Rectangle2D bounds = new Rectangle2D.Double(); bounds.setRect(poly.getBounds2D()); // determine if original object has multiple contact cuts // boolean baseMulti = false; // if (geom instanceof NodeInst) // baseMulti = tech.isMultiCutCase((NodeInst) geom); // search in the area surrounding the box double bound = mutableDist.doubleValue(); bounds.setRect(bounds.getMinX() - bound, bounds.getMinY() - bound, bounds.getWidth() + bound * 2, bounds.getHeight() + bound * 2); return badBoxInArea(poly, layer, tech, net, geom, trans, globalIndex, bounds, cell, globalIndex, cell, globalIndex, DBMath.MATID, multiCutData, true); } /** * Method to recursively examine a polygon to see if it has any errors with its surrounding area. * The polygon is "poly" on layer "layer" from technology "tech" on network "net" from object "geom" * which is associated with global index "globalIndex". * Checking looks in the area (lxbound-hxbound, lybound-hybound) in cell "cell" global index "cellGlobalIndex". * The polygon coordinates are in the space of cell "topCell", global index "topGlobalIndex", * and objects in "cell" can be transformed by "upTrans" to get to this space. * The base object, in "geom" can be transformed by "trans" to get to this space. * The minimum size of this polygon is "minSize" and "baseMulti" is TRUE if it comes from a multicut contact. * If the two objects are in the same cell instance (nonhierarchical DRC), then "sameInstance" is TRUE. * If they are from different instances, then "sameInstance" is FALSE. *

* Returns TRUE if errors are found. */ private boolean badBoxInArea(Poly poly, Layer layer, Technology tech, int net, Geometric geom, FixpTransform trans, int globalIndex, Rectangle2D bounds, Cell cell, int cellGlobalIndex, Cell topCell, int topGlobalIndex, FixpTransform upTrans, Technology.MultiCutData multiCutData, boolean sameInstance) { Rectangle2D rBound = new Rectangle2D.Double(); rBound.setRect(bounds); DBMath.transformRect(rBound, upTrans); // Step 1 Netlist netlist = getCheckProto(cell).netlist; Rectangle2D subBound = new Rectangle2D.Double(); //Sept 30 boolean foundError = false; boolean isLayerAContact = layer.getFunction().isContact(); boolean baseMulti = tech.isMultiCutInTechnology(multiCutData); // These nodes won't generate any DRC errors. Most of them are pins if (geom instanceof NodeInst && NodeInst.isSpecialNode(((NodeInst) geom))) return false; // Sept04 changes: bounds by rBound for (Iterator it = cell.searchIterator(bounds); it.hasNext();) { Geometric nGeom = it.next(); // I have to check if they are the same instance otherwise I check geometry against itself if (nGeom == geom && (sameInstance))// || nGeom.getParent() == cell)) continue; // Checking if element is covered by exclusion region if (coverByExclusion(nGeom)) continue; // skips this element if (nGeom instanceof NodeInst) { NodeInst ni = (NodeInst) nGeom; NodeProto np = ni.getProto(); if (NodeInst.isSpecialNode(ni)) continue; // Oct 5; // ignore nodes that are not primitive if (ni.isCellInstance()) { // instance found: look inside it for offending geometry FixpTransform rTransI = ni.rotateIn(); FixpTransform tTransI = ni.translateIn(); rTransI.preConcatenate(tTransI); subBound.setRect(bounds); DBMath.transformRect(subBound, rTransI); CheckInst ci = checkInsts.get(ni); int localIndex = cellGlobalIndex * ci.multiplier + ci.localIndex + ci.offset; FixpTransform subTrans = ni.translateOut(ni.rotateOut()); subTrans.preConcatenate(upTrans); //Sept 15 04 // compute localIndex if (badBoxInArea(poly, layer, tech, net, geom, trans, globalIndex, subBound, (Cell) np, localIndex, topCell, topGlobalIndex, subTrans, multiCutData, sameInstance)) { foundError = true; if (reportInfo.errorTypeSearch != DRC.DRCCheckMode.ERROR_CHECK_EXHAUSTIVE) return true; } } else { // don't check between technologies if (np.getTechnology() != tech) continue; // see if this type of node can interact with this layer if (!validLayers.checkLayerWithNode(layer, np)) continue; // see if the objects directly touch but they are not // coming from different NodeInst (not from checkCellInstContents // because they might below to the same cell but in different instances boolean touch = sameInstance && nGeom.isConnected(geom); // get the shape of each nodeinst layer Poly [] subPolyList = DRC.getShapeOfNodeBasedOnRules(tech, reportInfo, ni); int tot = subPolyList.length; if (tot == 0) System.out.println("Should i skup this one?"); // prepare to examine every layer in this nodeinst FixpTransform rTrans = ni.rotateOut(); rTrans.preConcatenate(upTrans); convertPseudoLayers(ni, subPolyList); for (int i = 0; i < tot; i++) subPolyList[i].transform(rTrans); /* Step 1 */ boolean multi = baseMulti; Technology.MultiCutData niMCD = tech.getMultiCutData(ni); if (tot==0 && niMCD != null) assert(false); // does it happen? // in case it is one via against many from another contact (3-contact configuration) if (!multi && isLayerAContact && niMCD != null) { multi = tech.isMultiCutInTechnology(niMCD); if (!multi) { // geom must be NodeInst NodeInst gNi = (NodeInst)geom; // in this case, both possible contacts are either 1xn or mx1 with n,m>=1 if (multiCutData == null || multiCutData.numCutsX() == 1) // it is 1xn { // Checking if they match at the Y axis. If yes, they are considered as a long 1xn array // so multi=false. The other element must be 1xM if (niMCD.numCutsY() != 1 && ni.getAnchorCenterX() != gNi.getAnchorCenterX()) multi = true; } else { // Checking if they match at the Y axis. If yes, they are considered as a long 1xn array // so multi=false if (niMCD.numCutsX() != 1 && ni.getAnchorCenterY() != gNi.getAnchorCenterY()) multi = true; } } } int multiInt = (multi) ? 1 : 0; for (int j = 0; j < tot; j++) { Poly npoly = subPolyList[j]; Layer nLayer = npoly.getLayer(); if (nLayer == null) continue; if (coverByExclusion(npoly, nGeom.getParent())) continue; //npoly.transform(rTrans); Rectangle2D nPolyRect = npoly.getBounds2D(); // can't do this because "lxbound..." is local but the poly bounds are global // On the corners? if (DBMath.isGreaterThan(nPolyRect.getMinX(), rBound.getMaxX()) || DBMath.isGreaterThan(rBound.getMinX(), nPolyRect.getMaxX()) || DBMath.isGreaterThan(nPolyRect.getMinY(), rBound.getMaxY()) || DBMath.isGreaterThan(rBound.getMinY(), nPolyRect.getMaxY())) continue; // determine network for this polygon int nNet = getDRCNetNumber(netlist, npoly.getPort(), ni, cellGlobalIndex); // see whether the two objects are electrically connected boolean con = false; if (nNet >= 0 && nNet == net) con = true; // Checking extension, it could be slow boolean ret = checkExtensionGateRule(geom, layer, poly, nLayer, npoly, netlist); if (ret) { foundError = true; if (reportInfo.errorTypeSearch != DRC.DRCCheckMode.ERROR_CHECK_EXHAUSTIVE) return true; } // check if both polys are cut and if the combine area doesn't excess cut sizes // regardless if they are connected or not ret = checkCutSizes(np, geom, layer, poly, nGeom, nLayer, npoly, topCell); if (ret) { foundError = true; if (reportInfo.errorTypeSearch != DRC.DRCCheckMode.ERROR_CHECK_EXHAUSTIVE) return true; } // if they connect electrically and adjoin, don't check if (con && touch) { // Check if there are minimum size defects boolean maytouch = DRC.mayTouch(tech, con, layer, nLayer); Rectangle2D trueBox1 = poly.getBox(); if (trueBox1 == null) trueBox1 = poly.getBounds2D(); Rectangle2D trueBox2 = npoly.getBox(); if (trueBox2 == null) trueBox1 = npoly.getBounds2D(); ret = checkMinDefects(cell, maytouch, geom, poly, trueBox1, layer, nGeom, npoly, trueBox2, nLayer, topCell); if (ret) { foundError = true; if (reportInfo.errorTypeSearch != DRC.DRCCheckMode.ERROR_CHECK_EXHAUSTIVE) return true; } continue; } boolean edge = false; // @TODO check if previous spacing rule was composed of same layers! DRCTemplate theRule = getSpacingRule(layer, poly, geom, nLayer, npoly, ni, con, multiInt); // Try edge rules if (theRule == null) { theRule = this.currentRules.getEdgeRule(layer, nLayer); edge = true; } if (theRule == null) continue; ret = checkDist(tech, topCell, topGlobalIndex, poly, layer, net, geom, trans, globalIndex, npoly, nLayer, nNet, nGeom, rTrans, cellGlobalIndex, con, theRule, edge); if (ret) { foundError = true; if (reportInfo.errorTypeSearch != DRC.DRCCheckMode.ERROR_CHECK_EXHAUSTIVE) return true; } } } } else { ArcInst ai = (ArcInst) nGeom; ArcProto ap = ai.getProto(); // don't check between technologies if (ap.getTechnology() != tech) continue; // see if this type of arc can interact with this layer if (!validLayers.checkLayerWithArc(layer, ap)) continue; // see if the objects directly touch boolean touch = sameInstance && nGeom.isConnected(geom); // see whether the two objects are electrically connected Network jNet = netlist.getNetwork(ai, 0); Integer[] netNumbers = networkLists.get(jNet); int nNet = netNumbers[cellGlobalIndex].intValue(); boolean con = false; if (net >= 0 && nNet == net) con = true; // if they connect electrically and adjoin, don't check // if (con && touch) continue; // get the shape of each arcinst layer Poly[] subPolyList = tech.getShapeOfArc(ai); int tot = subPolyList.length; for (int i = 0; i < tot; i++) subPolyList[i].transform(upTrans); cropActiveArc(ai, subPolyList); boolean multi = baseMulti; int multiInt = (multi) ? 1 : 0; for (int j = 0; j < tot; j++) { Poly nPoly = subPolyList[j]; Layer nLayer = nPoly.getLayer(); if (nLayer == null) continue; if (coverByExclusion(nPoly, nGeom.getParent())) continue; Rectangle2D nPolyRect = nPoly.getBounds2D(); // can't do this because "lxbound..." is local but the poly bounds are global if (nPolyRect.getMinX() > rBound.getMaxX() || nPolyRect.getMaxX() < rBound.getMinX() || nPolyRect.getMinY() > rBound.getMaxY() || nPolyRect.getMaxY() < rBound.getMinY()) continue; boolean ret = false; // if they connect electrically and adjoin, don't check // We must check if there are minor defects if they overlap regardless if they are connected or not if (con && touch) { // Check if there are minimum size defects boolean maytouch = DRC.mayTouch(tech, con, layer, nLayer); Rectangle2D trueBox1 = poly.getBox(); if (trueBox1 == null) trueBox1 = poly.getBounds2D(); Rectangle2D trueBox2 = nPoly.getBox(); if (trueBox2 == null) trueBox1 = nPoly.getBounds2D(); ret = checkMinDefects(cell, maytouch, geom, poly, trueBox1, layer, nGeom, nPoly, trueBox2, nLayer, topCell); if (ret) { foundError = true; if (reportInfo.errorTypeSearch != DRC.DRCCheckMode.ERROR_CHECK_EXHAUSTIVE) return true; } continue; } // see how close they can get boolean edge = false; DRCTemplate theRule = getSpacingRule(layer, poly, geom, nLayer, nPoly, ai, con, multiInt); if (theRule == null) { theRule = this.currentRules.getEdgeRule(layer, nLayer); edge = true; } if (theRule == null) continue; // check the distance ret = checkDist(tech, topCell, topGlobalIndex, poly, layer, net, geom, trans, globalIndex, nPoly, nLayer, nNet, nGeom, upTrans, cellGlobalIndex, con, theRule, edge); if (ret) { foundError = true; if (reportInfo.errorTypeSearch != DRC.DRCCheckMode.ERROR_CHECK_EXHAUSTIVE) return true; } // Checking extension, it could be slow ret = checkExtensionGateRule(geom, layer, poly, nLayer, nPoly, netlist); if (ret) { foundError = true; if (reportInfo.errorTypeSearch != DRC.DRCCheckMode.ERROR_CHECK_EXHAUSTIVE) return true; } } } } return foundError; } /** * Function to detect small edges due to overlapping polygons of the same material. It is valid only * for manhattan shapes * * @return true if error was found */ private boolean checkMinDefects(Cell cell, boolean maytouch, Geometric geom1, Poly poly1, Rectangle2D trueBox1, Layer layer1, Geometric geom2, Poly poly2, Rectangle2D trueBox2, Layer layer2, Cell topCell) { if (trueBox1 == null || trueBox2 == null) return false; if (!maytouch) return false; // manhattan double pdx = DBMath.round(Math.max(trueBox2.getMinX() - trueBox1.getMaxX(), trueBox1.getMinX() - trueBox2.getMaxX())); double pdy = DBMath.round(Math.max(trueBox2.getMinY() - trueBox1.getMaxY(), trueBox1.getMinY() - trueBox2.getMaxY())); double pd = Math.max(pdx, pdy); if (DBMath.areEquals(pdx, 0) && DBMath.areEquals(pdy, 0)) pd = 0; // touching boolean foundError = false; // They have to overlap if (DBMath.isGreaterThan(pd, 0)) return false; // they are electrically connected and they overlap: look for minimum size errors // of the overlapping region. DRCTemplate wRule = this.currentRules.getMinValue(layer1, DRCTemplate.DRCRuleType.MINWID); if (wRule == null) return false; // no rule double minWidth = wRule.getValue(0); double lxb = DBMath.round(Math.max(trueBox1.getMinX(), trueBox2.getMinX())); double hxb = DBMath.round(Math.min(trueBox1.getMaxX(), trueBox2.getMaxX())); double lyb = DBMath.round(Math.max(trueBox1.getMinY(), trueBox2.getMinY())); double hyb = DBMath.round(Math.min(trueBox1.getMaxY(), trueBox2.getMaxY())); Point2D lowB = new Point2D.Double(lxb, lyb); Point2D highB = new Point2D.Double(hxb, hyb); Rectangle2D bounds = new Rectangle2D.Double(lxb, lyb, hxb - lxb, hyb - lyb); // If resulting bounding box is identical to one of the original polygons // then one polygon is contained in the other one if (bounds.equals(trueBox1) || bounds.equals(trueBox2)) { return false; } //+-------------------+F //| | //| | //| |B E //| +-------+---------+ //| | | | //|C A| | | //+-----------+-------+ | // | | // D+-----------------+ // Checking A-B distance double actual = lowB.distance(highB); // !DBMath.areEquals(actual, 0) is on, it skips cases where A==B. // Condition off as of July 2nd 2008. Bugzilla 1745 if (/*!DBMath.areEquals(actual, 0) &&*/ DBMath.isGreaterThan(minWidth, actual) && foundSmallSizeDefect(cell, geom1, poly1, layer1, geom2, poly2, pd, lxb, lyb, hxb, hyb)) { // you can't pass geom1 or geom2 becuase they could be in different cells and therefore the message // could mislead DRC.createDRCErrorLogger(reportInfo, DRC.DRCErrorType.MINWIDTHERROR, null, topCell, minWidth, actual, wRule.ruleName, new Poly(bounds), null, layer1, null, null, layer2); foundError = true; } return foundError; } private boolean foundSmallSizeDefect(Cell cell, Geometric geom1, Poly poly1, Layer layer1, Geometric geom2, Poly poly2, double pd, double lxb, double lyb, double hxb, double hyb) { boolean foundError = false; boolean[] pointsFound = new boolean[2]; pointsFound[0] = pointsFound[1] = false; Point2D lowB = new Point2D.Double(lxb, lyb); Point2D highB = new Point2D.Double(hxb, hyb); Rectangle2D bounds = new Rectangle2D.Double(lxb, lyb, hxb - lxb, hyb - lyb); Poly rebuild = new Poly(bounds); Point2D pt1 = null; //new Point2D.Double(lxb - TINYDELTA, lyb - TINYDELTA); Point2D pt2 = null; //new Point2D.Double(lxb - TINYDELTA, hyb + TINYDELTA); Point2D pt1d = null; //(Point2D) pt1.clone(); Point2D pt2d = null; //(Point2D) pt2.clone(); // Search area should be bigger than bounding box otherwise it might not get the cells due to // rounding errors. Rectangle2D search = new Rectangle2D.Double(DBMath.round(lxb - DRC.TINYDELTA), DBMath.round(lyb - DRC.TINYDELTA), DBMath.round(hxb - lxb + 2 * DRC.TINYDELTA), DBMath.round(hyb - lyb + 2 * DRC.TINYDELTA)); // Looking for two corners not inside bounding boxes A and B if (pd == 0) // flat bounding box { pt1 = lowB; pt2 = highB; pt1d = lowB; pt2d = highB; } else { Point2D[] points = rebuild.getPoints(); int[] cornerPoints = new int[2]; int corners = 0; // if more than 3 corners are found -> bounding box is flat for (int i = 0; i < points.length && corners < 2; i++) { if (!DBMath.pointInsideRect(points[i], poly1.getBounds2D()) && !DBMath.pointInsideRect(points[i], poly2.getBounds2D())) { cornerPoints[corners++] = i; } } if (corners != 2) throw new Error("Wrong corners in Quick.foundSmallSizeDefect()"); pt1 = points[cornerPoints[0]]; pt2 = points[cornerPoints[1]]; double delta = ((pt2.getY() - pt1.getY()) / (pt2.getX() - pt1.getX()) > 0) ? 1 : -1; // getting point lightly out of the elements if (pt1.getX() < pt2.getX()) { // (y2-y1)/(x2-x1)(x1-tinydelta-x1) + y1 pt1d = new Point2D.Double(pt1.getX() - DRC.TINYDELTA, -delta * DRC.TINYDELTA + pt1.getY()); // (y2-y1)/(x2-x1)(x2+tinydelta-x2) + y2 pt2d = new Point2D.Double(pt2.getX() + DRC.TINYDELTA, delta * DRC.TINYDELTA + pt2.getY()); } else { pt1d = new Point2D.Double(pt2.getX() - DRC.TINYDELTA, -delta * DRC.TINYDELTA + pt2.getY()); pt2d = new Point2D.Double(pt1.getX() + DRC.TINYDELTA, delta * DRC.TINYDELTA + pt1.getY()); } if (DBMath.areEquals(pt1.getX(), pt2.getX()) || DBMath.areEquals(pt1.getY(), pt2.getY())) { return false; } } // looking if points around the overlapping area are inside another region // to avoid the error DRC.lookForLayerCoverage(geom1, poly1, geom2, poly2, cell, layer1, DBMath.MATID, search, pt1d, pt2d, null, pointsFound, false, thisLayerFunction, false, reportInfo.ignoreCenterCuts); // Nothing found if (!pointsFound[0] && !pointsFound[1]) { foundError = true; } return (foundError); } /** * Method to compare: * polygon "poly1" layer "layer1" network "net1" object "geom1" * with: * polygon "poly2" layer "layer2" network "net2" object "geom2" * The polygons are both in technology "tech" and are in the space of cell "cell" * which has global index "globalIndex". * Note that to transform object "geom1" to this space, use "trans1" and to transform * object "geom2" to this space, use "trans2". * They are connected if "con" is nonzero. * They cannot be less than "dist" apart (if "edge" is nonzero, check edges only) * and the rule for this is "rule". *

* Returns TRUE if an error has been found. */ private boolean checkDist(Technology tech, Cell cell, int globalIndex, Poly poly1, Layer layer1, int net1, Geometric geom1, FixpTransform trans1, int globalIndex1, Poly poly2, Layer layer2, int net2, Geometric geom2, FixpTransform trans2, int globalIndex2, boolean con, DRCTemplate theRule, boolean edge) { // turn off flag that the nodeinst may be undersized tinyNodeInst = null; Poly origPoly1 = poly1; Poly origPoly2 = poly2; Rectangle2D isBox1 = poly1.getBox(); Rectangle2D trueBox1 = isBox1; if (trueBox1 == null) trueBox1 = poly1.getBounds2D(); Rectangle2D isBox2 = poly2.getBox(); Rectangle2D trueBox2 = isBox2; if (trueBox2 == null) trueBox2 = poly2.getBounds2D(); boolean errorFound = false; // remember if there was a min defect error boolean maytouch = DRC.mayTouch(tech, con, layer1, layer2); // special code if both polygons are manhattan double pd = 0; boolean overlap = false; if (isBox1 != null && isBox2 != null) { // manhattan double pdx = DBMath.round(Math.max(trueBox2.getMinX() - trueBox1.getMaxX(), trueBox1.getMinX() - trueBox2.getMaxX())); double pdy = DBMath.round(Math.max(trueBox2.getMinY() - trueBox1.getMaxY(), trueBox1.getMinY() - trueBox2.getMaxY())); pd = Math.max(pdx, pdy); if (DBMath.areEquals(pdx, 0) && DBMath.areEquals(pdy, 0)) pd = 0; // touching overlap = DBMath.isLessThan(pd, 0); if (maytouch) { // they are electrically connected: see if they touch if (checkMinDefects(cell, maytouch, geom1, poly1, trueBox1, layer2, geom2, poly2, trueBox2, layer2, cell)) { if (reportInfo.errorTypeSearch != DRC.DRCCheckMode.ERROR_CHECK_EXHAUSTIVE) return true; errorFound = true; } } // crop out parts of any arc that is covered by an adjoining node trueBox1 = new Rectangle2D.Double(trueBox1.getMinX(), trueBox1.getMinY(), trueBox1.getWidth(), trueBox1.getHeight()); trueBox2 = new Rectangle2D.Double(trueBox2.getMinX(), trueBox2.getMinY(), trueBox2.getWidth(), trueBox2.getHeight()); // first to crop arcs since node cropping crops arcs too! boolean geom1IsNode = (geom1 instanceof NodeInst); boolean geom2IsNode = (geom2 instanceof NodeInst); if (!geom1IsNode) // arc -> crop it first { if (cropArcInst((ArcInst) geom1, layer1, trans1, trueBox1, overlap)) return errorFound; } if (!geom2IsNode) { if (cropArcInst((ArcInst) geom2, layer2, trans2, trueBox2, overlap)) return errorFound; } if (geom1IsNode) { if (cropNodeInst((NodeInst) geom1, globalIndex1, trans1, layer2, net2, geom2, trueBox2)) return errorFound; // } else // { // if (cropArcInst((ArcInst)geom1, layer1, trans1, trueBox1, overlap)) // return errorFound; } if (geom2 instanceof NodeInst) { if (cropNodeInst((NodeInst) geom2, globalIndex2, trans2, layer1, net1, geom1, trueBox1)) return errorFound; // } else // { // if (cropArcInst((ArcInst)geom2, layer2, trans2, trueBox2, overlap)) // return errorFound; } poly1 = new Poly(trueBox1); poly1.setStyle(Poly.Type.FILLED); poly2 = new Poly(trueBox2); poly2.setStyle(Poly.Type.FILLED); // compute the distance double lX1 = trueBox1.getMinX(); double hX1 = trueBox1.getMaxX(); double lY1 = trueBox1.getMinY(); double hY1 = trueBox1.getMaxY(); double lX2 = trueBox2.getMinX(); double hX2 = trueBox2.getMaxX(); double lY2 = trueBox2.getMinY(); double hY2 = trueBox2.getMaxY(); if (edge) { // calculate the spacing between the box edges double pdedge = DBMath.round(Math.min( Math.min(Math.min(Math.abs(lX1 - lX2), Math.abs(lX1 - hX2)), Math.min(Math.abs(hX1 - lX2), Math.abs(hX1 - hX2))), Math.min(Math.min(Math.abs(lY1 - lY2), Math.abs(lY1 - hY2)), Math.min(Math.abs(hY1 - lY2), Math.abs(hY1 - hY2))))); pd = Math.max(pd, pdedge); } else { pdx = DBMath.round(Math.max(lX2-hX1, lX1-hX2)); pdy = DBMath.round(Math.max(lY2-hY1, lY1-hY2)); if (DBMath.areEquals(pdx, 0) && DBMath.areEquals(pdy, 0)) pd = 0; // they are touching!! else { // They are overlapping if pdx < 0 && pdy < 0 pd = (Math.max(pdx, pdy)); if (pd < theRule.getValue(0) && pd > 0) { pd = poly1.separation(poly2); } } } } else { // nonmanhattan Poly.Type style = poly1.getStyle(); if (style != Poly.Type.FILLED && style != Poly.Type.CLOSED && style != Poly.Type.CROSSED && style != Poly.Type.OPENED && style != Poly.Type.OPENEDT1 && style != Poly.Type.OPENEDT2 && style != Poly.Type.OPENEDT3 && style != Poly.Type.VECTORS) return errorFound; style = poly2.getStyle(); if (style != Poly.Type.FILLED && style != Poly.Type.CLOSED && style != Poly.Type.CROSSED && style != Poly.Type.OPENED && style != Poly.Type.OPENEDT1 && style != Poly.Type.OPENEDT2 && style != Poly.Type.OPENEDT3 && style != Poly.Type.VECTORS) return errorFound; // make sure polygons don't intersect //@TODO combine this calculation otherwise Poly.intersects is called twice! double pd1 = poly1.separation(poly2); if (poly1.intersects(poly2)) pd = 0; else { // find distance between polygons pd = poly1.separation(poly2); } if (!DBMath.areEquals(pd1, pd)) System.out.println("Wrong case in non-nonmanhattan, Quick."); } DRC.DRCErrorType errorType = DRC.DRCErrorType.SPACINGERROR; if (theRule.ruleType == DRCTemplate.DRCRuleType.SURROUND) { if (DBMath.isGreaterThan(pd, 0)) // layers don't overlap -> no condition to check return errorFound; pd = Math.abs(pd); errorType = DRC.DRCErrorType.SURROUNDERROR; } // see if the design rule is met if (!DBMath.isGreaterThan(theRule.getValue(0), pd)) // default case: SPACING pd >= theRule.value1 { return errorFound; } if (theRule.ruleType != DRCTemplate.DRCRuleType.SURROUND) { /* * special case: ignore errors between two active layers connected * to either side of a field-effect transistor that is inside of * the error area. */ if (activeOnTransistor(poly1, layer1, net1, poly2, layer2, net2, cell, globalIndex)) return errorFound; /** * Special case: ignore spacing errors generated by VT layers if poly is already * part of a VTH-transistors. Not very elegant solution but should work for now. */ if (polyCoverByAnyVTLayer(cell, theRule, tech, new Poly[]{poly1, poly2}, new Layer[]{layer1, layer2}, new Geometric[]{geom1, geom2}, reportInfo.ignoreCenterCuts)) return errorFound; // special cases if the layers are the same if (tech.sameLayer(layer1, layer2)) { // special case: check for "notch" if (maytouch) { // if they touch, it is acceptable if (DBMath.isLessThanOrEqualTo(pd, 0)) return errorFound; // see if the notch is filled boolean newR = lookForCrossPolygons(geom1, poly1, geom2, poly2, layer1, cell, overlap); // if (Job.LOCALDEBUGFLAG) // { // Point2D pt1 = new Point2D.Double(); // Point2D pt2 = new Point2D.Double(); // int intervening = findInterveningPoints(poly1, poly2, pt1, pt2, false); // if (intervening == 0) // { // if (!newR) // { // System.out.println("DIfferent"); // lookForCrossPolygons(geom1, poly1, geom2, poly2, layer1, cell, overlap); // // } // //return false; // } // boolean needBoth = true; // if (intervening == 1) needBoth = false; // // if (lookForPoints(pt1, pt2, layer1, cell, needBoth)) // { // if (!newR) // { // System.out.println("DIfferent"); // lookForPoints(pt1, pt2, layer1, cell, needBoth); // lookForCrossPolygons(geom1, poly1, geom2, poly2, layer1, cell, overlap); // //return false; // } // } // // boolean oldR = lookForPoints(pt1, pt2, layer1, cell, needBoth); // if (oldR != newR) // System.out.println("DIfferent 2"); // } if (newR) return errorFound; // look further if on the same net and diagonally separate (1 intervening point) //if (net1 == net2 && intervening == 1) return false; errorType = DRC.DRCErrorType.NOTCHERROR; } } } String msg = null; if (tinyNodeInst != null) { // see if the node/arc that failed was involved in the error if ((tinyNodeInst == geom1 || tinyNodeInst == geom2) && (tinyGeometric == geom1 || tinyGeometric == geom2)) { msg = tinyNodeInst + " is too small for the " + tinyGeometric; } } DRC.createDRCErrorLogger(reportInfo, errorType, msg, cell, theRule.getValue(0), pd, theRule.ruleName, origPoly1, geom1, layer1, origPoly2, geom2, layer2); return true; } /*************************** QUICK DRC SEE IF GEOMETRICS CAUSE ERRORS ***************************/ /** * Method to examine, in cell "cell", the "count" instances in "geomsToCheck". * If they are DRC clean, set the associated entry in "validity" to TRUE. */ private void checkTheseGeometrics(Cell cell, int count, Geometric[] geomsToCheck, boolean[] validity) { CheckProto cp = getCheckProto(cell); // loop through all of the objects to be checked for (int i = 0; i < count; i++) { Geometric geomToCheck = geomsToCheck[i]; boolean errors = false; if (geomToCheck instanceof NodeInst) { NodeInst ni = (NodeInst) geomToCheck; if (ni.isCellInstance()) { errors = checkThisCellPlease(ni); } else { errors = checkNodeInst(ni, 0); } } else { ArcInst ai = (ArcInst) geomToCheck; errors = checkArcInst(cp, ai, 0); } if (validity != null) validity[i] = !errors; } } private boolean checkThisCellPlease(NodeInst ni) { Cell cell = ni.getParent(); Netlist netlist = getCheckProto(cell).netlist; int globalIndex = 0; // get transformation out of this instance FixpTransform upTrans = ni.translateOut(ni.rotateOut()); // get network numbering information for this instance CheckInst ci = checkInsts.get(ni); if (ci == null) return false; int localIndex = globalIndex * ci.multiplier + ci.localIndex + ci.offset; // look for other objects surrounding this one Rectangle2D nodeBounds = ni.getBounds(); // double worstInteractionDistance = reportInfo.worstInteractionDistance; MutableDouble mutableDist = new MutableDouble(0); if (!cellLayersCon.getWorstSpacingDistance(cell, mutableDist)) { System.out.println("No worst spacing distance found in Quick:checkThisCellPlease"); return false; } double worstInteractionDistance = mutableDist.doubleValue(); Rectangle2D searchBounds = new Rectangle2D.Double( nodeBounds.getMinX() - worstInteractionDistance, nodeBounds.getMinY() - worstInteractionDistance, nodeBounds.getWidth() + worstInteractionDistance * 2, nodeBounds.getHeight() + worstInteractionDistance * 2); for (Iterator it = cell.searchIterator(searchBounds); it.hasNext();) { Geometric geom = it.next(); if (geom == ni) continue; if (geom instanceof ArcInst) { if (checkGeomAgainstInstance(netlist, geom, ni)) return true; continue; } NodeInst oNi = (NodeInst) geom; if (oNi.getProto() instanceof PrimitiveNode) { // found a primitive node: check it against the instance contents if (checkGeomAgainstInstance(netlist, geom, ni)) return true; continue; } // found other instance "oNi", look for everything in "ni" that is near it Rectangle2D subNodeBounds = oNi.getBounds(); // double worstInteractionDistanceLocal = cellLayersCon.getWorstSpacingDistance(oNi.getProto()); if (!cellLayersCon.getWorstSpacingDistance(oNi.getProto(), mutableDist)) { System.out.println("No worst spacing distance found in Quick:checkThisCellPlease"); continue; } double worstInteractionDistanceLocal = mutableDist.doubleValue(); Rectangle2D subBounds = new Rectangle2D.Double( subNodeBounds.getMinX() - worstInteractionDistanceLocal, subNodeBounds.getMinY() - worstInteractionDistanceLocal, subNodeBounds.getWidth() + worstInteractionDistanceLocal * 2, subNodeBounds.getHeight() + worstInteractionDistanceLocal * 2); // recursively search instance "ni" in the vicinity of "oNi" if (checkCellInstContents(subBounds, ni, upTrans, localIndex, oNi, null, globalIndex, null)) return true; } return false; } /** * Method to check primitive object "geom" (an arcinst or primitive nodeinst) against cell instance "ni". * Returns TRUE if there are design-rule violations in their interaction. */ private boolean checkGeomAgainstInstance(Netlist netlist, Geometric geom, NodeInst ni) { NodeProto np = ni.getProto(); int globalIndex = 0; Cell subCell = geom.getParent(); // boolean baseMulti = false; Technology.MultiCutData multiCutData = null; Technology tech = null; Poly[] nodeInstPolyList = null; FixpTransform trans = DBMath.MATID; if (geom instanceof NodeInst) { // get all of the polygons on this node NodeInst oNi = (NodeInst) geom; if (NodeInst.isSpecialNode(oNi)) return false; // Nov 16, no need for checking pins or other special nodes; tech = oNi.getProto().getTechnology(); trans = oNi.rotateOut(); nodeInstPolyList = tech.getShapeOfNode(oNi, true, reportInfo.ignoreCenterCuts, thisLayerFunction); convertPseudoLayers(oNi, nodeInstPolyList); multiCutData = tech.getMultiCutData(oNi); // baseMulti = tech.isMultiCutCase(oNi); } else { ArcInst oAi = (ArcInst) geom; tech = oAi.getProto().getTechnology(); assert (false); // not sure about this nodeInstPolyList = tech.getShapeOfArc(oAi, this.thisLayerFunction); } if (nodeInstPolyList == null) return false; int tot = nodeInstPolyList.length; CheckInst ci = checkInsts.get(ni); if (ci == null) return false; int localIndex = globalIndex * ci.multiplier + ci.localIndex + ci.offset; // examine the polygons on this node MutableDouble mutableDist = new MutableDouble(0); for (int j = 0; j < tot; j++) { Poly poly = nodeInstPolyList[j]; Layer polyLayer = poly.getLayer(); if (polyLayer == null) continue; // see how far around the box it is necessary to search double maxSize = poly.getMaxSize(); boolean found = currentRules.getMaxSurround(polyLayer, maxSize, mutableDist); if (!found) continue; // determine network for this polygon int net; if (geom instanceof NodeInst) { net = getDRCNetNumber(netlist, poly.getPort(), (NodeInst) geom, globalIndex); } else { ArcInst oAi = (ArcInst) geom; Network jNet = netlist.getNetwork(oAi, 0); Integer[] netNumbers = networkLists.get(jNet); net = netNumbers[globalIndex].intValue(); } // determine area to search inside of cell to check this layer Rectangle2D polyBounds = poly.getBounds2D(); double bound = mutableDist.doubleValue(); Rectangle2D subBounds = new Rectangle2D.Double(polyBounds.getMinX() - bound, polyBounds.getMinY() - bound, polyBounds.getWidth() + bound * 2, polyBounds.getHeight() + bound * 2); FixpTransform tempTrans = ni.rotateIn(); FixpTransform tTransI = ni.translateIn(); tempTrans.preConcatenate(tTransI); DBMath.transformRect(subBounds, tempTrans); FixpTransform subTrans = ni.translateOut(ni.rotateOut()); // see if this polygon has errors in the cell if (badBoxInArea(poly, polyLayer, tech, net, geom, trans, globalIndex, subBounds, (Cell) np, localIndex, subCell, globalIndex, subTrans, multiCutData, false)) return true; } return false; } /*************************** QUICK DRC CACHE OF INSTANCE INTERACTIONS ***************************/ /** * Method to look for an interaction between instances "ni1" and "ni2". If it is found, * return TRUE. If not found, add to the list and return FALSE. */ private boolean checkInteraction(NodeInst ni1, NodeInst n1Parent, NodeInst ni2, NodeInst n2Parent, NodeInst triggerNi) { if (reportInfo.errorTypeSearch == DRC.DRCCheckMode.ERROR_CHECK_EXHAUSTIVE) return false; // must recheck parameterized instances always CheckProto cp = getCheckProto((Cell) ni1.getProto()); if (cp.cellParameterized) return false; cp = getCheckProto((Cell) ni2.getProto()); if (cp.cellParameterized) return false; // keep the instances in proper numeric order InstanceInter dii = new InstanceInter(); if (ni1.getNodeIndex() < ni2.getNodeIndex()) { NodeInst swapni = ni1; ni1 = ni2; ni2 = swapni; } else if (ni1 == ni2) { int node1Orientation = ni1.getAngle(); if (ni1.isMirroredAboutXAxis()) node1Orientation += 3600; if (ni1.isMirroredAboutYAxis()) node1Orientation += 7200; int node2Orientation = ni2.getAngle(); if (ni2.isMirroredAboutXAxis()) node2Orientation += 3600; if (ni2.isMirroredAboutYAxis()) node2Orientation += 7200; if (node1Orientation < node2Orientation) { NodeInst swapNI = ni1; ni1 = ni2; ni2 = swapNI; System.out.println("Check this case in Quick.checkInteraction"); } } // get essential information about their interaction dii.cell1 = (Cell) ni1.getProto(); dii.or1 = ni1.getOrient(); dii.cell2 = (Cell) ni2.getProto(); dii.or2 = ni2.getOrient(); // This has to be calculated before the swap dii.dx = ni2.getAnchorCenterX() - ni1.getAnchorCenterX(); dii.dy = ni2.getAnchorCenterY() - ni1.getAnchorCenterY(); dii.n1Parent = n1Parent; dii.n2Parent = n2Parent; dii.triggerNi = triggerNi; // if found, stop now if (findInteraction(dii)) return true; // insert it now instanceInteractionList.add(dii); return false; } /** * Method to look for the instance-interaction in "dii" in the global list of instances interactions * that have already been checked. Returns the entry if it is found, NOINSTINTER if not. */ private boolean findInteraction(InstanceInter dii) { for (InstanceInter thisII : instanceInteractionList) { if (thisII.cell1 == dii.cell1 && thisII.cell2 == dii.cell2 && thisII.or1.equals(dii.or1) && thisII.or2.equals(dii.or2) && thisII.dx == dii.dx && thisII.dy == dii.dy && thisII.n1Parent == dii.n1Parent && thisII.n2Parent == dii.n2Parent) { if (dii.triggerNi == thisII.triggerNi) { return true; } } } return false; } /************************* QUICK DRC HIERARCHICAL NETWORK BUILDING *************************/ /** * Method to recursively examine the hierarchy below cell "cell" and create * CheckProto and CheckInst objects on every cell instance. */ private CheckProto checkEnumerateProtos(Cell cell, Netlist netlist) { CheckProto cp = getCheckProto(cell); if (cp != null) return cp; cp = new CheckProto(); cp.instanceCount = 0; cp.timeStamp = 0; cp.hierInstanceCount = 0; cp.totalPerCell = 0; cp.cellChecked = false; cp.cellParameterized = false; cp.netlist = netlist; if (cell.hasParameters()) { cp.cellParameterized = true; } // for(Iterator vIt = cell.getVariables(); vIt.hasNext(); ) // { // Variable var = (Variable)vIt.next(); // if (var.isParam()) // { // cp.cellParameterized = true; // cp.treeParameterized = true; // break; // } // } checkProtos.put(cell, cp); for (Iterator nIt = cell.getNodes(); nIt.hasNext();) { NodeInst ni = nIt.next(); if (!ni.isCellInstance()) continue; // ignore documentation icons if (ni.isIconOfParent()) continue; Cell subCell = (Cell) ni.getProto(); CheckInst ci = new CheckInst(); checkInsts.put(ni, ci); CheckProto subCP = checkEnumerateProtos(subCell, netlist.getNetlist(ni)); // if (subCP.treeParameterized) // cp.treeParameterized = true; } return cp; } /** * Method to return CheckProto of a Cell. * * @param cell Cell to get its CheckProto. * @return CheckProto of a Cell. */ private final CheckProto getCheckProto(Cell cell) { return checkProtos.get(cell); } /** * Method to recursively examine the hierarchy below cell "cell" and fill in the * CheckInst objects on every cell instance. Uses the CheckProto objects * to keep track of cell usage. */ private void checkEnumerateInstances(Cell cell) { if (checkAbort()) return; // number all of the instances in this cell reportInfo.checkTimeStamp++; List subCheckProtos = new ArrayList(); for (Iterator it = cell.getNodes(); it.hasNext();) { NodeInst ni = it.next(); NodeProto np = ni.getProto(); if (!ni.isCellInstance()) continue; // ignore documentation icons if (ni.isIconOfParent()) continue; CheckProto cp = getCheckProto((Cell) np); if (cp.timeStamp != reportInfo.checkTimeStamp) { cp.timeStamp = reportInfo.checkTimeStamp; cp.instanceCount = 0; cp.nodesInCell = new ArrayList(); subCheckProtos.add(cp); } CheckInst ci = checkInsts.get(ni); ci.localIndex = cp.instanceCount++; cp.nodesInCell.add(ci); } // update the counts for this cell // for(Iterator it = subCheckProtos.iterator(); it.hasNext(); ) for (CheckProto cp : subCheckProtos) { // CheckProto cp = (CheckProto)it.next(); cp.hierInstanceCount += cp.instanceCount; // for(Iterator nIt = cp.nodesInCell.iterator(); nIt.hasNext(); ) for (CheckInst ci : cp.nodesInCell) { // CheckInst ci = (CheckInst)nIt.next(); ci.multiplier = cp.instanceCount; } } // now recurse for (Iterator it = cell.getNodes(); it.hasNext();) { NodeInst ni = it.next(); NodeProto np = ni.getProto(); if (!ni.isCellInstance()) continue; // ignore documentation icons if (ni.isIconOfParent()) continue; checkEnumerateInstances((Cell) np); } } private void checkEnumerateNetworks(Cell cell, CheckProto cp, int globalIndex, HashMap enumeratedNets) { // store all network information in the appropriate place for (Iterator nIt = cp.netlist.getNetworks(); nIt.hasNext();) { Network net = nIt.next(); Integer netNumber = enumeratedNets.get(net); Integer[] netNumbers = networkLists.get(net); netNumbers[globalIndex] = netNumber; } for (Iterator it = cell.getNodes(); it.hasNext();) { NodeInst ni = it.next(); NodeProto np = ni.getProto(); if (!ni.isCellInstance()) continue; // ignore documentation icons if (ni.isIconOfParent()) continue; // compute the index of this instance CheckInst ci = checkInsts.get(ni); int localIndex = globalIndex * ci.multiplier + ci.localIndex + ci.offset; // propagate down the hierarchy Cell subCell = (Cell) np; CheckProto subCP = getCheckProto(subCell); HashMap subEnumeratedNets = new HashMap(); for (Iterator pIt = ni.getPortInsts(); pIt.hasNext();) { PortInst pi = pIt.next(); Export subPP = (Export) pi.getPortProto(); Network net = cp.netlist.getNetwork(ni, subPP, 0); if (net == null) continue; Integer netNumber = enumeratedNets.get(net); Network subNet = subCP.netlist.getNetwork(subPP, 0); subEnumeratedNets.put(subNet, netNumber); } for (Iterator nIt = subCP.netlist.getNetworks(); nIt.hasNext();) { Network net = nIt.next(); if (subEnumeratedNets.get(net) == null) subEnumeratedNets.put(net, new Integer(reportInfo.checkNetNumber++)); } checkEnumerateNetworks(subCell, subCP, localIndex, subEnumeratedNets); } } /*********************************** QUICK DRC SUPPORT ***********************************/ /** * Function to look for conditional layers that will 100% overlap with the given region. * @return true if conditional layers are found for the test points */ private boolean searchForCondLayer(Geometric geom, Poly poly, Layer layer, Cell cell, boolean ignoreCenterCuts) { Rectangle2D polyBnd = poly.getBounds2D(); double midPointX = (polyBnd.getMinX() + polyBnd.getMaxX()) / 2; double midPointY = (polyBnd.getMinY() + polyBnd.getMaxY()) / 2; Point2D pt1 = new Point2D.Double(midPointX, polyBnd.getMinY()); Point2D pt2 = new Point2D.Double(midPointX, polyBnd.getMaxY()); Point2D pt3 = new Point2D.Double(midPointX, midPointY); // compute bounds for searching inside the given polygon Rectangle2D bnd = new Rectangle2D.Double(DBMath.round(polyBnd.getMinX() - DRC.TINYDELTA), DBMath.round(polyBnd.getMinY() - DRC.TINYDELTA), DBMath.round(polyBnd.getWidth() + 2 * DRC.TINYDELTA), DBMath.round(polyBnd.getHeight() + 2 * DRC.TINYDELTA)); // looking if points around the overlapping area are inside another region // to avoid the error boolean[] pointsFound = new boolean[3]; pointsFound[0] = pointsFound[1] = pointsFound[2] = false; // Test first with a vertical line boolean found = DRC.lookForLayerCoverage(geom, poly, null, null, cell, layer, DBMath.MATID, bnd, pt1, pt2, pt3, pointsFound, true, null, false, ignoreCenterCuts); if (!found) return found; // no need of checking horizontal line if the vertical test was not positive pt1.setLocation(polyBnd.getMinX(), midPointY); pt2.setLocation(polyBnd.getMaxX(), midPointY); pointsFound[0] = pointsFound[1] = false; pointsFound[2] = true; // no testing pt3 again found = DRC.lookForLayerCoverage(geom, poly, null, null, cell, layer, DBMath.MATID, bnd, pt1, pt2, null, pointsFound, true, null, false, ignoreCenterCuts); return found; } /** * Method to ensure that polygon "poly" on layer "layer" from object "geom" in * technology "tech" meets minimum width rules. If it is too narrow, other layers * in the vicinity are checked to be sure it is indeed an error. Returns true * if an error is found. */ private boolean checkMinWidth(Geometric geom, Layer layer, Poly poly, boolean onlyOne) { DRCTemplate minWidthRule = this.currentRules.getMinValue(layer, DRCTemplate.DRCRuleType.MINWID); boolean errorDefault = false; // Only if there is one default size if (minWidthRule != null) { errorDefault = DRC.checkMinWidthInternal(geom, layer, poly, onlyOne, minWidthRule, false, this.thisLayerFunction, reportInfo); if (!errorDefault) return false; // the default condition is the valid one. } DRCTemplate minWidthRuleCond = this.currentRules.getMinValue(layer, DRCTemplate.DRCRuleType.MINWIDCOND); // No appropiate overlapping condition if (minWidthRuleCond == null || !minWidthRuleCond.condition.startsWith("overlap(")) { // Now the error is reporte. Not very efficient if (errorDefault) DRC.checkMinWidthInternal(geom, layer, poly, onlyOne, minWidthRule, true, this.thisLayerFunction, reportInfo); return errorDefault; } // checking if geometry complains with overlap condition // Skipping "overlap" String[] layers = TextUtils.parseString(minWidthRuleCond.condition.substring(7), "(,)"); boolean found = false; for (String la : layers) { Layer l = layer.getTechnology().findLayer(la); found = searchForCondLayer(geom, poly, l, geom.getParent(), reportInfo.ignoreCenterCuts); if (!found) break; // no need to check the next layer } // If condition is met then the new rule applied. if (found) errorDefault = DRC.checkMinWidthInternal(geom, layer, poly, onlyOne, minWidthRuleCond, true, this.thisLayerFunction, reportInfo); else if (errorDefault) // report the errors here in case of default values DRC.checkMinWidthInternal(geom, layer, poly, onlyOne, minWidthRule, true, thisLayerFunction, reportInfo); return errorDefault; } private void traversePolyTree(Layer layer, PolyBase.PolyBaseTree obj, int level, DRCTemplate minAreaRule, DRCTemplate encloseAreaRule, DRCTemplate spacingRule, Cell cell, MutableInteger count) { for (PolyBase.PolyBaseTree son : obj.getSons()) { traversePolyTree(layer, son, level + 1, minAreaRule, encloseAreaRule, spacingRule, cell, count); } boolean minAreaCheck = level % 2 == 0; boolean checkMin = false, checkNotch = false; DRC.DRCErrorType errorType = DRC.DRCErrorType.MINAREAERROR; double minVal = 0; String ruleName = ""; if (minAreaCheck) { if (minAreaRule == null) return; // no rule minVal = minAreaRule.getValue(0); ruleName = minAreaRule.ruleName; checkMin = true; } else { // odd level checks enclose area and holes (spacing rule) errorType = DRC.DRCErrorType.ENCLOSEDAREAERROR; if (encloseAreaRule != null) { minVal = encloseAreaRule.getValue(0); ruleName = encloseAreaRule.ruleName; checkMin = true; } checkNotch = (spacingRule != null); } PolyBase poly = obj.getPoly(); if (checkMin) { double area = poly.getArea(); // isGreaterThan doesn't consider equals condition therefore negate condition is used if (!DBMath.isGreaterThan(minVal, area)) return; // larger than the min value count.increment(); DRC.createDRCErrorLogger(reportInfo, errorType, null, cell, minVal, area, ruleName, poly, null, layer, null, null, null); } if (checkNotch) { // Notches are calculated using the bounding box of the polygon -> this is an approximation Rectangle2D bnd = poly.getBounds2D(); if (bnd.getWidth() < spacingRule.getValue(0)) { count.increment(); DRC.createDRCErrorLogger(reportInfo, DRC.DRCErrorType.NOTCHERROR, "(X axis)", cell, spacingRule.getValue(0), bnd.getWidth(), spacingRule.ruleName, poly, null, layer, null, null, layer); } if (bnd.getHeight() < spacingRule.getValue(1)) { count.increment(); DRC.createDRCErrorLogger(reportInfo, DRC.DRCErrorType.NOTCHERROR, "(Y axis)", cell, spacingRule.getValue(1), bnd.getHeight(), spacingRule.ruleName, poly, null, layer, null, null, layer); } } } /** * Method to find two points between polygons "poly1" and "poly2" that can be used to test * for notches. The points are returned in (xf1,yf1) and (xf2,yf2). Returns zero if no * points exist in the gap between the polygons (becuase they don't have common facing edges). * Returns 1 if only one of the reported points needs to be filled (because the polygons meet * at a point). Returns 2 if both reported points need to be filled. */ private int findInterveningPoints(Poly poly1, Poly poly2, Point2D pt1, Point2D pt2, boolean newFix) { Rectangle2D isBox1 = poly1.getBox(); Rectangle2D isBox2 = poly2.getBox(); // Better to treat serpentine cases with bouding box if (newFix) { if (isBox1 == null) isBox1 = poly1.getBounds2D(); if (isBox2 == null) isBox2 = poly2.getBounds2D(); } if (isBox1 != null && isBox2 != null) { // handle vertical gap between polygons if (isBox1.getMinX() > isBox2.getMaxX() || isBox2.getMinX() > isBox1.getMaxX()) { // see if the polygons are horizontally aligned if (isBox1.getMinY() <= isBox2.getMaxY() && isBox2.getMinY() <= isBox1.getMaxY()) { double yf1 = Math.max(isBox1.getMinY(), isBox2.getMinY()); double yf2 = Math.min(isBox1.getMaxY(), isBox2.getMaxY()); if (isBox1.getMinX() > isBox2.getMaxX()) { double x = (isBox1.getMinX() + isBox2.getMaxX()) / 2; pt1.setLocation(x, yf1); pt2.setLocation(x, yf2); } else { double x = (isBox2.getMinX() + isBox1.getMaxX()) / 2; pt1.setLocation(x, yf1); pt2.setLocation(x, yf2); } return 2; } } else if (isBox1.getMinY() > isBox2.getMaxY() || isBox2.getMinY() > isBox1.getMaxY()) { // see if the polygons are horizontally aligned if (isBox1.getMinX() <= isBox2.getMaxX() && isBox2.getMinX() <= isBox1.getMaxX()) { double xf1 = Math.max(isBox1.getMinX(), isBox2.getMinX()); double xf2 = Math.min(isBox1.getMaxX(), isBox2.getMaxX()); if (isBox1.getMinY() > isBox2.getMaxY()) { double y = (isBox1.getMinY() + isBox2.getMaxY()) / 2; pt1.setLocation(xf1, y); pt2.setLocation(xf2, y); } else { double y = (isBox2.getMinY() + isBox1.getMaxY()) / 2; pt1.setLocation(xf1, y); pt2.setLocation(xf2, y); } return 2; } } else if ((isBox1.getMinX() == isBox2.getMaxX() || isBox2.getMinX() == isBox1.getMaxX()) || (isBox1.getMinY() == isBox2.getMaxY() || isBox2.getMinY() == isBox2.getMaxY())) { // handle touching at a point double xc = isBox2.getMinX(); if (isBox1.getMinX() == isBox2.getMaxX()) xc = isBox1.getMinX(); double yc = isBox2.getMinY(); if (isBox1.getMinY() == isBox2.getMaxY()) yc = isBox1.getMinY(); double xPlus = xc + DRC.TINYDELTA; double yPlus = yc + DRC.TINYDELTA; pt1.setLocation(xPlus, yPlus); pt2.setLocation(xPlus, yPlus); if ((xPlus < isBox1.getMinX() || xPlus > isBox1.getMaxX() || yPlus < isBox1.getMinY() || yPlus > isBox1.getMaxY()) && (xPlus < isBox2.getMinX() || xPlus > isBox2.getMaxX() || yPlus < isBox2.getMinY() || yPlus > isBox2.getMaxY())) return 1; pt1.setLocation(xc + DRC.TINYDELTA, yc - DRC.TINYDELTA); pt2.setLocation(xc - DRC.TINYDELTA, yc + DRC.TINYDELTA); return 1; } // handle manhattan objects that are on a diagonal if (isBox1.getMinX() > isBox2.getMaxX()) { if (isBox1.getMinY() > isBox2.getMaxY()) { pt1.setLocation(isBox1.getMinX(), isBox1.getMinY()); pt2.setLocation(isBox2.getMaxX(), isBox2.getMaxY()); return 1; } if (isBox2.getMinY() > isBox1.getMaxY()) { pt1.setLocation(isBox1.getMinX(), isBox1.getMaxY()); pt2.setLocation(isBox2.getMaxX(), isBox2.getMinY()); return 1; } } if (isBox2.getMinX() > isBox1.getMaxX()) { if (isBox1.getMinY() > isBox2.getMaxY()) { pt1.setLocation(isBox1.getMaxX(), isBox1.getMinY()); pt2.setLocation(isBox2.getMinX(), isBox2.getMaxY()); return 1; } if (isBox2.getMinY() > isBox1.getMaxY()) { pt1.setLocation(isBox1.getMaxX(), isBox1.getMaxY()); pt2.setLocation(isBox2.getMinX(), isBox2.getMinY()); return 1; } } } // boxes don't line up or this is a nonmanhattan situation isBox1 = poly1.getBounds2D(); isBox2 = poly2.getBounds2D(); pt1.setLocation((isBox1.getMinX() + isBox1.getMaxX()) / 2, (isBox2.getMinY() + isBox2.getMaxY()) / 2); pt2.setLocation((isBox2.getMinX() + isBox2.getMaxX()) / 2, (isBox1.getMinY() + isBox1.getMaxY()) / 2); return 1; } /** * Method to explore the points (xf1,yf1) and (xf2,yf2) to see if there is * geometry on layer "layer" (in or below cell "cell"). Returns true if there is. * If "needBoth" is true, both points must have geometry, otherwise only 1 needs it. */ private boolean lookForCrossPolygons(Geometric geo1, Poly poly1, Geometric geo2, Poly poly2, Layer layer, Cell cell, boolean overlap) { Point2D pt1 = new Point2D.Double(); Point2D pt2 = new Point2D.Double(); findInterveningPoints(poly1, poly2, pt1, pt2, true); // compute bounds for searching inside cells double flx = Math.min(pt1.getX(), pt2.getX()); double fhx = Math.max(pt1.getX(), pt2.getX()); double fly = Math.min(pt1.getY(), pt2.getY()); double fhy = Math.max(pt1.getY(), pt2.getY()); Rectangle2D bounds = new Rectangle2D.Double(DBMath.round(flx - DRC.TINYDELTA), DBMath.round(fly - DRC.TINYDELTA), DBMath.round(fhx - flx + 2 * DRC.TINYDELTA), DBMath.round(fhy - fly + 2 * DRC.TINYDELTA)); // Adding delta otherwise it won't consider points along edges. // Mind bounding boxes could have zero width or height // search the cell for geometry that fills the notch boolean[] pointsFound = new boolean[2]; pointsFound[0] = pointsFound[1] = false; boolean allFound = DRC.lookForLayerCoverage(geo1, poly1, geo2, poly2, cell, layer, DBMath.MATID, bounds, pt1, pt2, null, pointsFound, overlap, thisLayerFunction, false, reportInfo.ignoreCenterCuts); return allFound; } /** * Method to examine cell "cell" in the area (lx<=X<=hx, ly<=Y<=hy) for objects * on layer "layer". Apply transformation "moreTrans" to the objects. If polygons are * found at (xf1,yf1) or (xf2,yf2) or (xf3,yf3) then sets "p1found/p2found/p3found" to 1. * If all locations are found, returns true. */ private boolean lookForLayerWithPoints(Geometric geo1, Poly poly1, Geometric geo2, Poly poly2, Cell cell, Layer layer, FixpTransform moreTrans, Rectangle2D bounds, Point2D[] pts, boolean[] pointsFound, int length, boolean overlap) { assert (pts.length == pointsFound.length); int j; Rectangle2D newBounds = new Rectangle2D.Double(); // Sept 30 for (Iterator it = cell.searchIterator(bounds); it.hasNext();) { Geometric g = it.next(); // You can't skip the same geometry otherwise layers in the same Geometric won't // be tested. // if (ignoreSameGeometry && (g == geo1 || g == geo2)) // continue; // I can't skip geometries to exclude from the search if (g instanceof NodeInst) { NodeInst ni = (NodeInst) g; if (NodeInst.isSpecialNode(ni)) continue; // Nov 16, no need for checking pins or other special nodes; if (ni.isCellInstance()) { // compute bounding area inside of sub-cell FixpTransform rotI = ni.rotateIn(); FixpTransform transI = ni.translateIn(); rotI.preConcatenate(transI); newBounds.setRect(bounds); DBMath.transformRect(newBounds, rotI); // compute new matrix for sub-cell examination FixpTransform trans = ni.translateOut(ni.rotateOut()); trans.preConcatenate(moreTrans); if (lookForLayerWithPoints(geo1, poly1, geo2, poly2, (Cell) ni.getProto(), layer, trans, newBounds, pts, pointsFound, length, overlap)) return true; continue; } FixpTransform bound = ni.rotateOut(); bound.preConcatenate(moreTrans); Technology tech = ni.getProto().getTechnology(); // I have to ask for electrical layers otherwise it will retrieve one polygon for polysilicon // and poly.polySame(poly1) will never be true. CONTRADICTION! Poly[] layerLookPolyList = tech.getShapeOfNode(ni, false, reportInfo.ignoreCenterCuts, thisLayerFunction); // consistent change! // layerLookPolyList = tech.getShapeOfNode(ni, false, ignoreCenterCuts, null); int tot = layerLookPolyList.length; for (int i = 0; i < tot; i++) { Poly poly = layerLookPolyList[i]; // sameLayer test required to check if Active layer is not identical to thich actice layer if (!tech.sameLayer(poly.getLayer(), layer)) { // This happens when you have implant with VTH implant, Function.Set doesn't distinguish them if (Job.getDebug()) System.out.println("Wrong Function.Set with " + layer.getName() + " and " + poly.getLayer().getName()); // assert(false); // should this happen? continue; } // Should be the transform before? poly.transform(bound); if (poly1 != null && !overlap && poly.polySame(poly1)) continue; if (poly2 != null && !overlap && poly.polySame(poly2)) continue; for (j = 0; j < length; j++) { if (pointsFound[j]) continue; // point covered if (poly.isInside(pts[j])) pointsFound[j] = true; } for (j = 0; j < length && pointsFound[j]; j++) ; if (j == length) return true; // No need of checking rest of the layers break; // assuming only 1 polygon per layer (non-electrical) } } else { ArcInst ai = (ArcInst) g; Technology tech = ai.getProto().getTechnology(); Poly[] layerLookPolyList = tech.getShapeOfArc(ai, thisLayerFunction); // consistent change!); int tot = layerLookPolyList.length; for (int i = 0; i < tot; i++) { Poly poly = layerLookPolyList[i]; // sameLayer test required to check if Active layer is not identical to thich actice layer if (!tech.sameLayer(poly.getLayer(), layer)) { // This happens when you have implant with VTH implant, Function.Set doesn't distinguish them if (Job.getDebug()) System.out.println("Wrong Function.Set with " + layer.getName() + " and " + poly.getLayer().getName()); // assert(false); // should this happen? continue; } poly.transform(moreTrans); for (j = 0; j < length; j++) { if (pointsFound[j]) continue; // point covered if (poly.isInside(pts[j])) pointsFound[j] = true; } for (j = 0; j < length && pointsFound[j]; j++) ; if (j == length) return true; // No need of checking rest of the layers break; } } } return false; } /** * *************************** Cut Rule Functions ************************** */ private boolean checkCutSizes(NodeProto np, Geometric geom, Layer layer, Poly poly, Geometric nGeom, Layer nLayer, Poly nPoly, Cell topCell) { // None of them is cut if (!(np instanceof PrimitiveNode) || layer != nLayer || !layer.getFunction().isContact() || !nLayer.getFunction().isContact()) return false; PrimitiveNode nty = (PrimitiveNode) np; Technology.NodeLayer cutLayer = nty.findMulticut(); if (cutLayer == null) return false; // no cut layer double cutSizeX = cutLayer.getMulticutSizeX().getLambda(); double cutSizeY = cutLayer.getMulticutSizeY().getLambda(); // double[] specValues = nty.getSpecialValues(); // // 0 and 1 hold CUTSIZE // if (specValues == null) return false; // no values Rectangle2D box1 = poly.getBounds2D(); Rectangle2D box2 = nPoly.getBounds2D(); double pdx = DBMath.round(Math.max(box2.getMinX() - box1.getMaxX(), box1.getMinX() - box2.getMaxX())); double pdy = DBMath.round(Math.max(box2.getMinY() - box1.getMaxY(), box1.getMinY() - box2.getMaxY())); double pd = Math.max(pdx, pdy); if (DBMath.areEquals(pdx, 0) && DBMath.areEquals(pdy, 0)) pd = 0; // touching // They have to overlap if (DBMath.isGreaterThan(pd, 0)) return false; boolean foundError = false; double minX = Math.min(box1.getMinX(), box2.getMinX()); double minY = Math.min(box1.getMinY(), box2.getMinY()); double maxX = Math.max(box1.getMaxX(), box2.getMaxX()); double maxY = Math.max(box1.getMaxY(), box2.getMaxY()); Rectangle2D rect = new Rectangle2D.Double(minX, minY, maxX - minX, maxY - minY); DRCTemplate rule = this.currentRules.getRule(layer.getIndex(), DRCTemplate.DRCRuleType.MINWID); // DRCTemplate rule = DRC.getRules(nty.getTechnology()).getRule(nty.getPrimNodeIndexInTech(), // DRCTemplate.DRCRuleType.CUTSIZE); String ruleName = (rule != null) ? rule.ruleName : "for contacts"; if (DBMath.isGreaterThan(rect.getWidth(), cutSizeX)) { DRC.createDRCErrorLogger(reportInfo, DRC.DRCErrorType.CUTERROR, "along X", topCell, cutSizeX, rect.getWidth(), ruleName, new Poly(rect), null, layer, null, null, nLayer); foundError = true; } if (DBMath.isGreaterThan(rect.getHeight(), cutSizeY)) { DRC.createDRCErrorLogger(reportInfo, DRC.DRCErrorType.CUTERROR, "along Y", topCell, cutSizeY, rect.getHeight(), ruleName, new Poly(rect), null, layer, null, null, layer); foundError = true; } return foundError; } /** * *************************** Extension Rule Functions ************************** */ private boolean checkExtensionGateRule(Geometric geom, Layer layer, Poly poly, Layer nLayer, Poly nPoly, Netlist netlist) { if (ignoreExtensionRules) return false; return false; // DRCTemplate extensionRule = DRC.getExtensionRule(layer, nLayer, true); // Checking extension, it could be slow // if (extensionRule == null) return false; // // if (!layer.getName().equals(extensionRule.name1) || // !nLayer.getName().equals(extensionRule.name2)) // return false; // no match in names // Area firstArea = new Area(poly); // Area nPolyArea = new Area(nPoly); // Area inter = (Area)nPolyArea.clone(); // inter.intersect(firstArea); // nPolyArea.subtract(inter); // get original nPoly without the intersection // boolean found = true; // areas that don't insert are not subject to this test. // if (!nPolyArea.isEmpty()) // { // // Searching from the corresponding net on geom // Poly portPoly = null; // Poly[] primPolyList = getShapeOfGeometric(geom, nLayer); // for (int i = 0; i < primPolyList.length; i++) // { // // Look for Poly that overlaps with original nPoly to get the port // if (nPoly.intersects(primPolyList[i])) // { // portPoly = primPolyList[i]; // break; // } // } // assert portPoly != null; // Network net = getDRCNetNumber(netlist, geom, portPoly); // found = searchTransistor(net, geom); // } // // if (!found) // DRC.createDRCErrorLogger(reportInfo, DRCErrorType.FORBIDDEN, " does not comply with rule '" + extensionRule.ruleName + "'.", geom.getParent(), -1, -1, null, null, geom, null, null, null, null); // return false; } /** * Method to retrieve Poly objects from Geometric object (NodeInst, ArcInst) * @param geom * @param layer * @return Array of Geometric objects */ // private Poly[] getShapeOfGeometric(Geometric geom, Layer layer) // { // Poly [] primPolyList = null; // Layer.Function.Set drcLayers = new Layer.Function.Set(layer.getFunction()); // // if (geom instanceof NodeInst) // { // NodeInst ni = (NodeInst)geom; // NodeProto np = ni.getProto(); // Technology tech = np.getTechnology(); // primPolyList = tech.getShapeOfNode(ni, true, ignoreCenterCuts, drcLayers); // } // else if (geom instanceof ArcInst) // { // ArcInst ai = (ArcInst)geom; // primPolyList = ai.getProto().getTechnology().getShapeOfArc(ai, drcLayers); // } // return primPolyList; // } /** * Method to search for a transistor connected to this network that is not geom. This is to implement * special rule in 90nm technology. MOVE TO TECHNOLOGY class!! to be cleaner * @param network * @param geom * @return v */ // private boolean searchTransistor(Network network, Geometric geom) // { // // checking if any port in that network belongs to a transistor // // No need of checking arcs? // for (Iterator it = network.getPorts(); it.hasNext();) // { // PortInst pi = (PortInst)it.next(); // NodeInst ni = pi.getNodeInst(); // if (ni != geom && ni.getProto() instanceof PrimitiveNode) // { // PrimitiveNode pn = (PrimitiveNode)ni.getProto(); // if (pn.getFunction().isTransistor()) // return true; // found a transistor! // } // } // return false; // } /** * Method to check extension rules in general */ // private boolean checkExtensionRule(Geometric geom, Layer layer, Poly poly, Cell cell) // { // return false; //// if(DRC.isIgnoreExtensionRuleChecking()) return false; //// //// Rectangle2D polyBnd = poly.getBounds2D(); //// Set layerSet = new HashSet(); // to avoid repeated elements //// //// // Search among all possible neighbors and to get the layers //// for(Iterator sIt = cell.searchIterator(polyBnd); sIt.hasNext();) //// { //// Geometric g = (Geometric)sIt.next(); //// if (g == geom) continue; //// if ((g instanceof ArcInst)) //// { //// ArcProto ap = ((ArcInst)g).getProto(); //// for (int i = 0; i < ap.getLayers().length; i++) //// layerSet.add(pa.getLayers()[i].getLayer()); //// } //// else //// { //// NodeInst ni = (NodeInst)g; //// NodeProto np = ni.getProto(); //// if (NodeInst.isSkipSizeInPalette(ni)) continue; // Nov 4; //// if (np instanceof Cell) //// { //// System.out.println("Not implemented in checkExtensionRule"); //// } //// else //// { //// if (np instanceof PrimitiveNode) //// { //// PrimitiveNode pn = (PrimitiveNode)np; //// for (int i = 0; i < pn.getLayers().length; i++) //// layerSet.add(pn.getLayers()[i].getLayer()); //// } //// else //// System.out.println("When do we have this case"); //// } //// } //// } //// boolean error = false; //// //// Object[] layerArray = layerSet.toArray(); //// for (int i = 0; i < layerArray.length; i++) //// { //// Layer nLayer = (Layer)layerArray[i]; //// DRCTemplate extensionRule = DRC.getExtensionRule(layer, nLayer, techMode, false); //// // Checking extension, it could be slow //// if (extensionRule == null) continue; //// //// List list = extensionRule.getNodesInRule(); //// // Layers order is revelant, the first element is the one to check for the extension in the second one //// if (list != null && list.size() == 2) //// { //// // Not the correct order //// if (list.get(0) != layer || list.get(1) != nLayer) continue; //// } //// //// Area baseArea = new Area(poly); //// Area overlapArea = new Area(); // start empty //// Area extensionArea = new Area(); // start empty //// //// checkExtensionOverlapRule(geom, nLayer, cell, baseArea, extensionArea, overlapArea, polyBnd); //// //// // get the area outside nLayer //// if (!extensionArea.isEmpty()) //// { //// List polyList = PolyBase.getPointsInArea(extensionArea, layer, true, true, null); //// //// for (Iterator it = polyList.iterator(); it.hasNext(); ) //// { //// PolyBase nPoly = (PolyBase)it.next(); //// Rectangle2D rect = nPoly.getBounds2D(); //// int dir = 0; // X assume the intersection edge is Y oriented //// // Determine how they touch. This is to get the perpendicular direction to the common edge. //// if ((rect.getMaxY() == poly.getBounds2D().getMinY()) || //// (rect.getMinY() == poly.getBounds2D().getMaxY()) ) //// dir = 1; // Y //// //// // not easy to determine along which axis they are touching //// if ((dir == 1 && rect.getHeight() < extensionRule.value) || (dir == 0 && rect.getWidth() < extensionRule.value)) //// { //// DRC.createDRCErrorLogger(reportInfo, LAYERDRCErrorType.SURROUNDERROR, "No enough extension, ", cell, extensionRule.value, -1, extensionRule.ruleName, //// nPoly, geom, layer, null, null, nLayer); //// error = true; //// } //// } //// } //// //// // There is overlap //// if (!overlapArea.isEmpty()) //// { //// List polyList = PolyBase.getPointsInArea(overlapArea, layer, true, true, null); //// //// for (Iterator it = polyList.iterator(); it.hasNext(); ) //// { //// PolyBase nPoly = (PolyBase)it.next(); //// Rectangle2D rect = nPoly.getBounds2D(); //// // not easy to determine along which axis they are touching //// if (rect.getHeight() < extensionRule.value || rect.getWidth() < extensionRule.value) //// { //// DRC.createDRCErrorLogger(reportInfo, LAYERDRCErrorType.SURROUNDERROR, "No enough overlap, ", cell, extensionRule.value, -1, extensionRule.ruleName, //// nPoly, geom, layer, null, null, nLayer); //// error = true; //// } //// } //// } //// } //// return (error); // } /** * * @param geom * @param cell * @param extensionArea contains information about poly area outside of nLayer * @param overlapArea * @param polyBnd */ // private void checkExtensionOverlapRule(Geometric geom, Layer layer, Cell cell, Area baseArea, // Area extensionArea, Area overlapArea, // Rectangle2D polyBnd) // { // for(Iterator sIt = cell.searchIterator(polyBnd); sIt.hasNext(); ) // { // Geometric g = (Geometric)sIt.next(); // if (g == geom) continue; // if ((g instanceof NodeInst)) // { // NodeInst ni = (NodeInst)g; // NodeProto np = ni.getProto(); // if (NodeInst.isSpecialNode(ni)) continue; // Nov 4; // if (ni.isCellInstance()) // { // AffineTransform cellDownTrans = ni.transformIn(); // AffineTransform cellUpTrans = ni.transformOut(); // DBMath.transformRect(polyBnd, cellDownTrans); // extensionArea.transform(cellDownTrans); // overlapArea.transform(cellDownTrans); // checkExtensionOverlapRule(geom, layer, (Cell)np, baseArea, extensionArea, overlapArea, polyBnd); // DBMath.transformRect(polyBnd, cellUpTrans); // extensionArea.transform(cellUpTrans); // overlapArea.transform(cellUpTrans); // continue; // } // } // // Poly [] primPolyList = getShapeOfGeometric(g, layer); // int tot = primPolyList.length; // for(int j=0; j rules = this.currentRules.getRules(layer, DRCTemplate.DRCRuleType.SURROUND); //A AB B // +----------------------------+ // | | //A| | B //D| | C // | | // | | // +----------------------------+ //D DC C Point2D[] basePts = new Point2D[4]; // boolean[] baseFound = new boolean[4]; // all false at the beginning // NOTE: Enough to find a point matching 1 rule to be OK. Rectangle2D polyBnd = poly.getBounds2D(); basePts[0] = new Point2D.Double(polyBnd.getMinX(), polyBnd.getMinY()); basePts[1] = new Point2D.Double(polyBnd.getMinX(), polyBnd.getMaxY()); basePts[2] = new Point2D.Double(polyBnd.getMaxX(), polyBnd.getMaxY()); basePts[3] = new Point2D.Double(polyBnd.getMaxX(), polyBnd.getMinY()); for (DRCTemplate rule : rules) { if (rule.nodeName != null) continue; // ignore these ones, only use to resize primitives if (rule.condition == null) continue; // must be conditional if (!rule.name1.equals(layer.getName())) continue; // it has to be the first name String[] layersS = TextUtils.parseString(rule.condition.substring(2), "(,)"); // 2 to skip "or" Layer[] layers = new Layer[layersS.length]; for (int i = 0; i < layersS.length; i++) { layers[i] = layer.getTechnology().findLayer(layersS[i]); } for (int j = 0; j < 4; j++) { boolean[] basicFound = new boolean[3]; Point2D[] basicPts = new Point2D[3]; basicFound[0] = false; basicPts[0] = basePts[j]; Rectangle2D basicBnd = new Rectangle2D.Double(DBMath.round(basicPts[0].getX() - DRC.TINYDELTA), DBMath.round(basicPts[0].getY() - DRC.TINYDELTA), 2 * (DRC.TINYDELTA), 2 * (DRC.TINYDELTA)); boolean f = false; for (int i = 0; i < layers.length; i++) { f = lookForLayerWithPoints(geom, poly, null, null, cell, layers[i], DBMath.MATID, basicBnd, basicPts, basicFound, 1, true); if (f) break; // found } if (!f) continue; // no point in layers found // now testing the rule basicFound[0] = basicFound[1] = basicFound[2] = false; double cos45 = 1.0 / Math.sqrt(2); double xValue = rule.getValue(0); // on the diagonal, 45 degrees double yValue = rule.getValue(1); double x45Value = xValue * cos45; // on the diagonal, 45 degrees double y45Value = yValue * cos45; Point2D basicBndPoint = null; // Sample 3 points per vertex to detect corners. switch (j) { case 0: // Diagonal basicPts[0] = new Point2D.Double(polyBnd.getMinX() - x45Value, polyBnd.getMinY() - y45Value); // left basicPts[1] = new Point2D.Double(polyBnd.getMinX() - xValue, polyBnd.getMinY()); // bottom basicPts[2] = new Point2D.Double(polyBnd.getMinX(), polyBnd.getMinY() - yValue); basicBndPoint = new Point2D.Double(polyBnd.getMinX() - xValue, polyBnd.getMinY() - yValue); break; case 1: basicPts[0] = new Point2D.Double(polyBnd.getMinX() - x45Value, polyBnd.getMaxY() + y45Value); basicPts[1] = new Point2D.Double(polyBnd.getMinX() - xValue, polyBnd.getMaxY()); basicPts[2] = new Point2D.Double(polyBnd.getMinX(), polyBnd.getMaxY() + yValue); basicBndPoint = basicPts[1]; break; case 2: basicPts[0] = new Point2D.Double(polyBnd.getMaxX() + x45Value, polyBnd.getMaxY() + y45Value); basicPts[1] = new Point2D.Double(polyBnd.getMaxX() + xValue, polyBnd.getMaxY()); basicPts[2] = new Point2D.Double(polyBnd.getMaxX(), polyBnd.getMaxY() + yValue); basicBndPoint = new Point2D.Double(polyBnd.getMaxX(), polyBnd.getMaxY()); break; case 3: basicPts[0] = new Point2D.Double(polyBnd.getMaxX() + x45Value, polyBnd.getMinY() - y45Value); basicPts[1] = new Point2D.Double(polyBnd.getMaxX() + xValue, polyBnd.getMinY()); basicPts[2] = new Point2D.Double(polyBnd.getMaxX(), polyBnd.getMinY() - yValue); basicBndPoint = basicPts[2]; break; } basicBnd = new Rectangle2D.Double(DBMath.round(basicBndPoint.getX() - DRC.TINYDELTA), DBMath.round(basicBndPoint.getY() - DRC.TINYDELTA), DBMath.round(xValue + 2 * (DRC.TINYDELTA)), DBMath.round(yValue + 2 * (DRC.TINYDELTA))); for (int i = 0; i < layers.length; i++) { f = lookForLayerWithPoints(geom, poly, null, null, cell, layers[i], DBMath.MATID, basicBnd, basicPts, basicFound, 3, true); if (f) break; // found } if (!f) { DRC.createDRCErrorLogger(reportInfo, DRC.DRCErrorType.LAYERSURROUNDERROR, "Not enough surround of " + rule.condition + ", ", geom.getParent(), rule.getValue(0), -1, rule.ruleName, poly, geom, layer, null, null, null); if (reportInfo.errorTypeSearch != DRC.DRCCheckMode.ERROR_CHECK_EXHAUSTIVE) return true; // no need of checking other combinations break; // with next rule } } } return false; } /****************************** Select Over Polysilicon Functions ***************************/ /** * Method to check if non-transistor polysilicons are completed covered by N++/P++ regions */ // private boolean checkSelectOverPolysilicon(Geometric geom, Layer layer, Poly poly, Cell cell) // { // if(DRC.isIgnoreExtensionRuleChecking()) return false; // // if (!layer.getFunction().isPoly()) return false; // // One layer must be select and other polysilicon. They are not connected // // DRCTemplate minOverlapRule = DRC.getMinValue(layer, DRCTemplate.DRCRuleType.EXTENSION); // if (minOverlapRule == null) return false; // double minOverlapValue = minOverlapRule.getValue(0); // Rectangle2D polyBnd = poly.getBounds2D(); // Area polyArea = new Area(poly); // boolean found = polyArea.isEmpty(); // List checkExtraPoints = new ArrayList(); // // found = checkThisCellExtensionRule(geom, layer, poly, null, cell, polyArea, polyBnd, minOverlapRule, // found, checkExtraPoints); // // // error if the merged area doesn't contain 100% the search area. // if (!found) // { // List polyList = PolyBase.getPointsInArea(polyArea, layer, true, true, null); // // for (PolyBase nPoly : polyList) // { // DRC.createDRCErrorLogger(reportInfo, DRCErrorType.LAYERSURROUNDERROR, "Polysilicon not covered, ", cell, minOverlapValue, -1, // minOverlapRule.ruleName, nPoly, geom, layer, null, null, null); // } // } // else // { // // Checking if not enough coverage // Point2D[] extraPoints = new Point2D[checkExtraPoints.size()]; // checkExtraPoints.toArray(extraPoints); // boolean[] founds = new boolean[checkExtraPoints.size()]; // Arrays.fill(founds, false); // Rectangle2D ruleBnd = new Rectangle2D.Double(polyBnd.getMinX()-minOverlapValue, // polyBnd.getMinY()-minOverlapValue, // polyBnd.getWidth() + minOverlapValue*2, // polyBnd.getHeight() + minOverlapValue*2); // boolean foundAll = allPointsContainedInLayer(geom, cell, ruleBnd, null, extraPoints, founds); // // if (!foundAll) // DRC.createDRCErrorLogger(reportInfo, DRCErrorType.LAYERSURROUNDERROR, "Not enough surround, ", geom.getParent(), minOverlapValue, // -1, minOverlapRule.ruleName, poly, geom, layer, null, null, null); // } // return (!found); // } // private boolean checkThisCellExtensionRule(Geometric geom, Layer layer, Poly poly, Layer.Function.Set drcLayers, Cell cell, Area polyArea, // Rectangle2D polyBnd, DRCTemplate minOverlapRule, boolean found, // List checkExtraPoints) // { // boolean[] founds = new boolean[4]; // // for(Iterator sIt = cell.searchIterator(polyBnd); !found && sIt.hasNext(); ) // { // RTBounds g = sIt.next(); // if (g == geom) continue; // if (!(g instanceof NodeInst)) // { // if (Job.LOCALDEBUGFLAG && !DRC.isIgnoreExtensionRuleChecking()) // System.out.println("Skipping arcs!"); // continue; // Skipping arcs!!!! // } // NodeInst ni = (NodeInst)g; // NodeProto np = ni.getProto(); // if (NodeInst.isSpecialNode(ni)) continue; // Nov 4; // if (ni.isCellInstance()) // { // AffineTransform cellDownTrans = ni.transformIn(); // AffineTransform cellUpTrans = ni.transformOut(); // DBMath.transformRect(polyBnd, cellDownTrans); // polyArea.transform(cellDownTrans); // poly.transform(cellDownTrans); // List newExtraPoints = new ArrayList(); // found = checkThisCellExtensionRule(geom, layer, poly, drcLayers, (Cell)np, polyArea, polyBnd, minOverlapRule, // found, newExtraPoints); // for (Point2D point : newExtraPoints) //// for (Iterator it = newExtraPoints.iterator(); it.hasNext();) // { //// Point2D point = it.next(); // cellUpTrans.transform(point, point); // checkExtraPoints.add(point); // } // DBMath.transformRect(polyBnd, cellUpTrans); // polyArea.transform(cellUpTrans); // poly.transform(cellUpTrans); // } // else // { // Technology tech = np.getTechnology(); // Poly [] primPolyList = tech.getShapeOfNode(ni, true, ignoreCenterCuts, drcLayers); // int tot = primPolyList.length; // for(int j=0; j sIt = cell.searchIterator(ruleBnd); sIt.hasNext(); ) // { // RTBounds g = sIt.next(); // if (g == geom) continue; // if (!(g instanceof NodeInst)) continue; // NodeInst ni = (NodeInst)g; // NodeProto np = ni.getProto(); // if (NodeInst.isSpecialNode(ni)) continue; // Nov 4; // if (ni.isCellInstance()) // { // AffineTransform cellDownTrans = ni.transformIn(); // AffineTransform cellUpTrans = ni.transformOut(); // DBMath.transformRect(ruleBnd, cellDownTrans); // cellDownTrans.transform(points, 0, points, 0, points.length); // boolean allFound = allPointsContainedInLayer(geom, (Cell)np, ruleBnd, drcLayers, points, founds); // DBMath.transformRect(ruleBnd, cellUpTrans); // cellUpTrans.transform(points, 0, points, 0, points.length); // if (allFound) // return true; // } // else // { // Technology tech = np.getTechnology(); // Poly [] primPolyList = tech.getShapeOfNode(ni, true, ignoreCenterCuts, drcLayers); // int tot = primPolyList.length; // for(int j=0; j sIt = cell.searchIterator(bounds); sIt.hasNext();) { Geometric g = sIt.next(); if (!(g instanceof NodeInst)) continue; NodeInst ni = (NodeInst) g; NodeProto np = ni.getProto(); if (ni.isCellInstance()) { FixpTransform rTransI = ni.rotateIn(); FixpTransform tTransI = ni.translateIn(); rTransI.preConcatenate(tTransI); subBounds.setRect(bounds); DBMath.transformRect(subBounds, rTransI); CheckInst ci = checkInsts.get(ni); int localIndex = globalIndex * ci.multiplier + ci.localIndex + ci.offset; boolean ret = activeOnTransistorRecurse(subBounds, net1, net2, (Cell) np, localIndex, trans); if (ret) return true; continue; } // must be a transistor if (!ni.getFunction().isFET()) continue; // must be inside of the bounding box of the desired layers Rectangle2D nodeBounds = ni.getBounds(); double cX = nodeBounds.getCenterX(); double cY = nodeBounds.getCenterY(); if (cX < bounds.getMinX() || cX > bounds.getMaxX() || cY < bounds.getMinY() || cY > bounds.getMaxY()) continue; // determine the poly port (MUST BE BETTER WAY!!!!) PortProto badport = np.getPort(0); Network badNet = netlist.getNetwork(ni, badport, 0); boolean on1 = false, on2 = false; for (Iterator it = ni.getPortInsts(); it.hasNext();) { PortInst pi = (PortInst) it.next(); PortProto po = pi.getPortProto(); String name = po.getName(); boolean found = false; boolean oldFound = false; Network piNet = netlist.getNetwork(pi); // ignore connections on poly/gate if (name.indexOf("poly") != -1) { found = true; //continue; } if (piNet == badNet) oldFound = true; if (Job.LOCALDEBUGFLAG && oldFound != found) System.out.println("Here is different in activeOnTransistorRecurse"); if (found) continue; Integer[] netNumbers = networkLists.get(piNet); int net = netNumbers[globalIndex].intValue(); if (net < 0) continue; if (net == net1) on1 = true; if (net == net2) on2 = true; } // if either side is not connected, ignore this if (!on1 || !on2) continue; // transistor found that connects to both actives return true; } return false; } /** * Method to crop the box on layer "nLayer", electrical index "nNet" * and bounds (lx-hx, ly-hy) against the nodeinst "ni". The geometry in nodeinst "ni" * that is being checked runs from (nlx-nhx, nly-nhy). Only those layers * in the nodeinst that are the same layer and the same electrical network * are checked. Returns true if the bounds are reduced * to nothing. */ private boolean cropNodeInst(NodeInst ni, int globalIndex, FixpTransform trans, Layer nLayer, int nNet, Geometric nGeom, Rectangle2D bound) { Technology tech = ni.getProto().getTechnology(); assert(!ni.getProto().getFunction().isPin()); Layer.Function.Set set = Layer.getMultiLayersSet(nLayer); Poly[] cropNodePolyList = tech.getShapeOfNode(ni, true, reportInfo.ignoreCenterCuts, set); convertPseudoLayers(ni, cropNodePolyList); int tot = cropNodePolyList.length; if (tot < 0) return false; boolean[] rotated = new boolean[tot]; Arrays.fill(rotated, false); boolean isConnected = false; Netlist netlist = getCheckProto(ni.getParent()).netlist; for (int j = 0; j < tot; j++) { Poly poly = cropNodePolyList[j]; if (!tech.sameLayer(poly.getLayer(), nLayer)) { // This happens when you have implant with VTH implant, Function.Set doesn't distinguish them if (Job.getDebug()) System.out.println("Wrong Function.Set with " + nLayer.getName() + " and " + poly.getLayer().getName()); // assert(false); // should this happen? continue; } //only transform when poly is valid poly.transform(trans); // change 1 rotated[j] = true; if (nNet >= 0) { if (poly.getPort() == null) continue; // determine network for this polygon int net = getDRCNetNumber(netlist, poly.getPort(), ni, globalIndex); if (net >= 0 && net != nNet) continue; } isConnected = true; break; } if (!isConnected) return false; // get the description of the nodeinst layers boolean allgone = false; for (int j = 0; j < tot; j++) { Poly poly = cropNodePolyList[j]; if (!tech.sameLayer(poly.getLayer(), nLayer)) { // This happens when you have implant with VTH implant, Function.Set doesn't distinguish them if (Job.getDebug()) System.out.println("Wrong Function.Set with " + nLayer.getName() + " and " + poly.getLayer().getName()); // assert(false); // should this happen? continue; } if (!rotated[j]) poly.transform(trans); // change 1 // warning: does not handle arbitrary polygons, only boxes Rectangle2D polyBox = poly.getBox(); if (polyBox == null) continue; int temp = Poly.cropBox(bound, polyBox); if (temp > 0) { allgone = true; break; } if (temp < 0) { tinyNodeInst = ni; tinyGeometric = nGeom; } } return allgone; } /** * Method to crop away any part of layer "lay" of arcinst "ai" that coincides * with a similar layer on a connecting nodeinst. The bounds of the arcinst * are in the reference parameters (lx-hx, ly-hy). Returns false * normally, 1 if the arcinst is cropped into oblivion. */ private boolean cropArcInst(ArcInst ai, Layer lay, FixpTransform inTrans, Rectangle2D bounds, boolean overlap) { for (int i = 0; i < 2; i++) { // find the primitive nodeinst at the true end of the portinst PortInst pi = ai.getPortInst(i); PortOriginal fp = new PortOriginal(pi, inTrans); NodeInst ni = fp.getBottomNodeInst(); if (NodeInst.isSpecialNode(ni)) continue; // Dec 08 -> is a pin so ignore it NodeProto np = ni.getProto(); FixpTransform trans = fp.getTransformToTop(); Technology tech = np.getTechnology(); Poly[] cropArcPolyList = null; // No overlap means arc should be cropped to get away from other element // if (!overlap && np.getFunction() == PrimitiveNode.Function.PIN) // { // // Pins don't generate polygons // // Search for another arc and try to crop it with that geometry // List drcLayers = new ArrayList(1); // drcLayers.add(lay.getFunction()); // List arcPolys = new ArrayList(1); // int totalPolys = 0; // for (Iterator it = ni.getConnections(); it.hasNext(); ) // { // Connection con = (Connection)it.next(); // ArcInst arc = con.getArc(); // if (arc == ai) continue; // Poly[] polys = tech.getShapeOfArc(arc, null, null, drcLayers); // arcPolys.add(polys); // totalPolys += polys.length; // } // cropArcPolyList = new Poly[totalPolys]; // int destPos = 0; // for (Poly[] arcs : arcPolys) //// for (int j = 0; j < arcPolys.size(); j++) // { //// Poly[] arcs = arcPolys.get(j); // System.arraycopy(arcs, 0, cropArcPolyList, destPos, arcs.length); // destPos += arcs.length; // } // } // else Layer.Function.Set set = Layer.getMultiLayersSet(lay); cropArcPolyList = tech.getShapeOfNode(ni, false, reportInfo.ignoreCenterCuts, set); int tot = cropArcPolyList.length; for (int j = 0; j < tot; j++) { Poly poly = cropArcPolyList[j]; if (!tech.sameLayer(poly.getLayer(), lay)) { // This happens when you have implant with VTH implant, Function.Set doesn't distinguish them if (Job.getDebug()) System.out.println("Wrong Function.Set with " + lay.getName() + " and " + poly.getLayer().getName()); continue; } poly.transform(trans); // warning: does not handle arbitrary polygons, only boxes Rectangle2D polyBox = poly.getBox(); if (polyBox == null) continue; int temp = Poly.halfCropBox(bounds, polyBox); if (temp > 0) return true; if (temp < 0) { tinyNodeInst = ni; tinyGeometric = ai; } } } return false; } /** * Method to see if polygons in "pList" (describing arc "ai") should be cropped against a * connecting transistor. Crops the polygon if so. */ private void cropActiveArc(ArcInst ai, Poly[] pList) { // look for an active layer in this arc int tot = pList.length; int diffPoly = -1; for (int j = 0; j < tot; j++) { Poly poly = pList[j]; Layer layer = poly.getLayer(); if (layer == null) continue; Layer.Function fun = layer.getFunction(); if (fun.isDiff()) { diffPoly = j; break; } } if (diffPoly < 0) return; Poly poly = pList[diffPoly]; // must be manhattan Rectangle2D polyBounds = poly.getBox(); if (polyBounds == null) return; polyBounds = new Rectangle2D.Double(polyBounds.getMinX(), polyBounds.getMinY(), polyBounds.getWidth(), polyBounds.getHeight()); // search for adjoining transistor in the cell boolean cropped = false; boolean halved = false; for (int i = 0; i < 2; i++) { PortInst pi = ai.getPortInst(i); NodeInst ni = pi.getNodeInst(); if (!ni.getFunction().isFET()) continue; // crop the arc against this transistor FixpTransform trans = ni.rotateOut(); Technology tech = ni.getProto().getTechnology(); Poly[] activeCropPolyList = tech.getShapeOfNode(ni, false, reportInfo.ignoreCenterCuts, null); int nTot = activeCropPolyList.length; for (int k = 0; k < nTot; k++) { Poly nPoly = activeCropPolyList[k]; if (nPoly.getLayer() != poly.getLayer()) continue; nPoly.transform(trans); Rectangle2D nPolyBounds = nPoly.getBox(); if (nPolyBounds == null) continue; // @TODO Why only one half is half crop? // Should I change cropBox by cropBoxComplete? int result = (halved) ? Poly.cropBox(polyBounds, nPolyBounds) : Poly.halfCropBox(polyBounds, nPolyBounds); if (result == 1) { // remove this polygon from consideration poly.setLayer(null); return; } cropped = true; halved = true; } } if (cropped) { Poly.Type style = poly.getStyle(); Layer layer = poly.getLayer(); poly = new Poly(polyBounds); poly.setStyle(style); poly.setLayer(layer); pList[diffPoly] = poly; } } /** * Method to convert all Layer information in the Polys to be non-pseudo. * This is only done for pins that have exports but no arcs. * * @param ni the NodeInst being converted. * @param pList an array of Polys with Layer information for the NodeInst. */ private void convertPseudoLayers(NodeInst ni, Poly[] pList) { if (!ni.getProto().getFunction().isPin()) return; assert (false); // When do I get this? System.out.println("I should not get this"); if (ni.hasConnections()) return; // if (ni.getNumConnections() != 0) return; if (!ni.hasExports()) return; // if (ni.getNumExports() == 0) return; // for pins that are unconnected but have exports, convert them to real layers int tot = pList.length; for (int i = 0; i < tot; i++) { Poly poly = pList[i]; Layer layer = poly.getLayer(); if (layer != null) poly.setLayer(layer.getNonPseudoLayer()); } } /** * Method to determine which layers in a Technology are valid. */ // private void cacheValidLayers(Technology tech) // { // if (tech == null) return; // if (layersValidTech == tech) return; // // layersValidTech = tech; // // // determine the layers that are being used // int numLayers = tech.getNumLayers(); // layersValid = new boolean[numLayers]; // for (int i = 0; i < numLayers; i++) // layersValid[i] = false; // for (Iterator it = tech.getNodes(); it.hasNext();) // { // PrimitiveNode np = (PrimitiveNode) it.next(); // if (np.isNotUsed()) continue; // Technology.NodeLayer[] layers = np.getLayers(); // for (int i = 0; i < layers.length; i++) // { // Layer layer = layers[i].getLayer(); // layersValid[layer.getIndex()] = true; // } // } // for (Iterator it = tech.getArcs(); it.hasNext();) // { // ArcProto ap = (ArcProto) it.next(); // if (ap.isNotUsed()) continue; // for (Iterator lIt = ap.getLayerIterator(); lIt.hasNext();) // { // Layer layer = lIt.next(); // layersValid[layer.getIndex()] = true; // } // } // } /** * Method to determine the minimum distance between "layer1" and "layer2" in technology * "tech" and library "lib". If "con" is true, the layers are connected. Also forces * connectivity for same-implant layers. Returns conditional rules if not standard rules were found */ private DRCTemplate getSpacingRule(Layer layer1, Poly poly1, Geometric geo1, Layer layer2, Poly poly2, Geometric geo2, boolean con, int multi) { // if they are implant on the same layer, they connect if (!con && layer1 == layer2) { Layer.Function fun = layer1.getFunction(); // treat all wells as connected con = fun.isSubstrate(); } double[] values = layer1.getTechnology().getSpacingDistances(poly1, poly2); // try conditional or standard DRCTemplate theRule = (this.currentRules.getSpacingRule(layer1, geo1, layer2, geo2, con, multi, values[0], values[1])); if (theRule != null && theRule.condition != null) // conditional rules { String[] conds = TextUtils.parseString(theRule.condition, "{,}"); assert (conds.length == 2); // {layerName1, cond(layerName2)} String layerName1 = conds[0]; String[] parameters = TextUtils.parseString(conds[1], "()"); assert (parameters.length == 2); // cond(layerName2) assert (parameters[0].equals("no_overlap")); // only function implemented String layerName2 = parameters[1]; if (!layer1.getName().equals(layerName1)) theRule = null; // doesn't apply else { Layer l = layer1.getTechnology().findLayer(layerName2); boolean found = searchForCondLayer(geo1, poly1, l, geo1.getParent(), reportInfo.ignoreCenterCuts); if (found) theRule = null; // doesn't apply } } return theRule; } /** * Method to return the network number for port "pp" on node "ni", given that the node is * in a cell with global index "globalIndex". */ private int getDRCNetNumber(Netlist netlist, PortProto pp, NodeInst ni, int globalIndex) { if (pp == null) return -1; //@TODO REPLACE BY getNetwork(Nodable no, PortProto portProto, int busIndex) // or getNetIndex(Nodable no, PortProto portProto, int busIndex) // s0ee if there is an arc connected Network net = netlist.getNetwork(ni, pp, 0); Integer[] nets = networkLists.get(net); if (nets == null) return -1; return nets[globalIndex].intValue(); } /** * Method to recursively scan cell "cell" (transformed with "trans") searching * for DRC Exclusion nodes. Each node is added to the global list "exclusionList". */ private void accumulateExclusion(Cell cell) { Area area = null; for (Iterator it = cell.getNodes(); it.hasNext();) { NodeInst ni = it.next(); NodeProto np = ni.getProto(); if (np == Generic.tech().drcNode) { // Must get polygon from getNodeShape otherwise it will miss // rings FixpTransform trans = ni.rotateOut(); Poly[] list = cell.getTechnology().getShapeOfNode(ni, true, true, null); assert (list.length == 1); list[0].transform(trans); Area thisArea = new Area(list[0]); if (area == null) area = thisArea; else area.add(thisArea); continue; } if (ni.isCellInstance()) { // examine contents accumulateExclusion((Cell) np); } } ; exclusionMap.put(cell, area); } } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy