OpenRAM/compiler/router/router.py

1244 lines
47 KiB
Python
Raw Normal View History

# See LICENSE for licensing information.
#
2019-06-14 17:43:41 +02:00
# Copyright (c) 2016-2019 Regents of the University of California and The Board
# of Regents for the Oklahoma Agricultural and Mechanical College
# (acting for and on behalf of Oklahoma State University)
# All rights reserved.
#
2019-11-07 17:33:13 +01:00
import itertools
import math
2016-11-12 17:57:26 +01:00
import gdsMill
2019-11-07 17:33:13 +01:00
from tech import drc, GDS
from tech import layer as techlayer
2016-11-12 17:57:26 +01:00
import debug
from router_tech import router_tech
from pin_layout import pin_layout
from pin_group import pin_group
from vector import vector
2019-11-07 17:33:13 +01:00
from vector3d import vector3d
from globals import OPTS, print_time
import grid_utils
from datetime import datetime
2016-11-12 17:57:26 +01:00
2019-11-07 17:33:13 +01:00
class router(router_tech):
"""
A router class to read an obstruction map from a gds and plan a
route on a given layer. This is limited to two layer routes.
It populates blockages on a grid class.
2016-11-12 17:57:26 +01:00
"""
2021-01-15 22:25:57 +01:00
def __init__(self, layers, design, gds_filename=None, bbox=None, route_track_width=1):
2016-11-12 17:57:26 +01:00
"""
This will instantiate a copy of the gds file or the module at (0,0) and
route on top of this. The blockages from the gds/module will be
2019-11-07 17:33:13 +01:00
considered.
"""
router_tech.__init__(self, layers, route_track_width)
2020-11-03 15:29:17 +01:00
self.cell = design
# If didn't specify a gds blockage file, write it out to read the gds
# This isn't efficient, but easy for now
2019-11-07 17:33:13 +01:00
# start_time = datetime.now()
if not gds_filename:
gds_filename = OPTS.openram_temp+"temp.gds"
self.cell.gds_write(gds_filename)
# Load the gds file and read in all the shapes
self.layout = gdsMill.VlsiLayout(units=GDS["unit"])
2016-11-12 17:57:26 +01:00
self.reader = gdsMill.Gds2reader(self.layout)
self.reader.loadFromFile(gds_filename)
self.top_name = self.layout.rootStructureName
2019-11-07 17:33:13 +01:00
# print_time("GDS read",datetime.now(), start_time)
2020-11-03 15:29:17 +01:00
2019-11-07 17:33:13 +01:00
# The pin data structures
# A map of pin names to a set of pin_layout structures
# (i.e. pins with a given label)
self.pins = {}
2019-11-07 17:33:13 +01:00
# This is a set of all pins (ignoring names) so that can quickly
# not create blockages for pins
# (They will be blocked when we are routing other
# nets based on their name.)
self.all_pins = set()
2020-11-03 15:29:17 +01:00
2019-11-07 17:33:13 +01:00
# The labeled pins above categorized into pin groups
# that are touching/connected.
self.pin_groups = {}
2020-11-03 15:29:17 +01:00
2019-11-07 17:33:13 +01:00
# The blockage data structures
# A list of metal shapes (using the same pin_layout structure)
# that could be blockages.
# This will include the pins above as well.
2019-11-07 17:33:13 +01:00
self.blockages = []
# The corresponding set of blocked grids for above blockage pin_layout shapes
# It is a cached set of grids that *could* be blocked, but may be unblocked
# depending on which pin we are routing.
self.blocked_grids = set()
2020-11-03 15:29:17 +01:00
2019-11-07 17:33:13 +01:00
# The routed data structures
# A list of paths that have been "routed"
self.paths = []
# A list of path blockages (they might be expanded for wide metal DRC)
self.path_blockages = []
2021-01-15 22:25:57 +01:00
self.init_bbox(bbox)
def init_bbox(self, bbox=None):
"""
Initialize the ll,ur values with the paramter or using the layout boundary.
"""
if not bbox:
# The boundary will determine the limits to the size
# of the routing grid
self.boundary = self.layout.measureBoundary(self.top_name)
# These must be un-indexed to get rid of the matrix type
self.ll = vector(self.boundary[0][0], self.boundary[0][1])
self.ur = vector(self.boundary[1][0], self.boundary[1][1])
else:
self.ll, self.ur = bbox
self.bbox = (self.ll, self.ur)
size = self.ur - self.ll
debug.info(1, "Size: {0} x {1}".format(size.x, size.y))
def get_bbox(self):
return self.bbox
def create_routing_grid(self, router_type, bbox=None):
"""
Create a sprase routing grid with A* expansion functions.
"""
self.init_bbox(bbox)
self.rg = router_type(self.ll, self.ur, self.track_width)
def clear_pins(self):
"""
Convert the routed path to blockages.
Keep the other blockages unchanged.
"""
self.pins = {}
self.all_pins = set()
self.pin_groups = {}
# DO NOT clear the blockages as these don't change
self.rg.reinit()
2020-11-03 15:29:17 +01:00
2019-11-07 17:33:13 +01:00
def set_top(self, top_name):
""" If we want to route something besides the top-level cell."""
self.top_name = top_name
2020-11-03 15:29:17 +01:00
2019-11-07 17:33:13 +01:00
def is_wave(self, path):
"""
2019-11-07 17:33:13 +01:00
Determines if this is a multi-track width wave (True)
# or a normal route (False)
"""
2019-11-07 17:33:13 +01:00
return len(path[0]) > 1
2020-11-03 15:29:17 +01:00
2019-11-07 17:33:13 +01:00
def retrieve_pins(self, pin_name):
"""
Retrieve the pin shapes on metal 3 from the layout.
"""
debug.info(2, "Retrieving pins for {}.".format(pin_name))
2019-11-07 17:33:13 +01:00
shape_list = self.layout.getAllPinShapes(str(pin_name))
pin_set = set()
for shape in shape_list:
2019-11-07 17:33:13 +01:00
(layer, boundary) = shape
# GDSMill boundaries are in (left, bottom, right, top) order
# so repack and snap to the grid
2019-11-07 17:33:13 +01:00
ll = vector(boundary[0], boundary[1]).snap_to_grid()
ur = vector(boundary[2], boundary[3]).snap_to_grid()
rect = [ll, ur]
pin = pin_layout(pin_name, rect, layer)
pin_set.add(pin)
2019-11-07 17:33:13 +01:00
debug.check(len(pin_set) > 0,
"Did not find any pin shapes for {0}.".format(str(pin_name)))
self.pins[pin_name] = pin_set
self.all_pins.update(pin_set)
for pin in self.pins[pin_name]:
2019-11-07 17:33:13 +01:00
debug.info(3, "Retrieved pin {}".format(str(pin)))
2016-11-12 17:57:26 +01:00
def find_blockages(self):
"""
Iterate through all the layers and write the obstacles to the routing grid.
2019-11-07 17:33:13 +01:00
This doesn't consider whether the obstacles will be pins or not.
They get reset later if they are not actually a blockage.
"""
debug.info(1, "Finding blockages.")
for lpp in [self.vert_lpp, self.horiz_lpp]:
self.retrieve_blockages(lpp)
2020-11-03 15:29:17 +01:00
def find_pins_and_blockages(self, pin_list):
"""
Find the pins and blockages in the design
"""
2019-11-07 17:33:13 +01:00
# This finds the pin shapes and sorts them into "groups" that
# are connected. This must come before the blockages, so we
# can not count the pins themselves
# as blockages.
start_time = datetime.now()
for pin_name in pin_list:
self.retrieve_pins(pin_name)
2019-11-07 17:33:13 +01:00
print_time("Retrieving pins", datetime.now(), start_time, 4)
2020-11-03 15:29:17 +01:00
start_time = datetime.now()
for pin_name in pin_list:
self.analyze_pins(pin_name)
2019-11-07 17:33:13 +01:00
print_time("Analyzing pins", datetime.now(), start_time, 4)
# This will get all shapes as blockages and convert to grid units
# This ignores shapes that were pins
start_time = datetime.now()
self.find_blockages()
2019-11-07 17:33:13 +01:00
print_time("Finding blockages", datetime.now(), start_time, 4)
# Convert the blockages to grid units
start_time = datetime.now()
self.convert_blockages()
2019-11-07 17:33:13 +01:00
print_time("Converting blockages", datetime.now(), start_time, 4)
2020-11-03 15:29:17 +01:00
# This will convert the pins to grid units
2019-11-07 17:33:13 +01:00
# It must be done after blockages to ensure no DRCs
# between expanded pins and blocked grids
start_time = datetime.now()
for pin in pin_list:
self.convert_pins(pin)
2019-11-07 17:33:13 +01:00
print_time("Converting pins", datetime.now(), start_time, 4)
2018-12-06 23:29:06 +01:00
# Combine adjacent pins into pin groups to reduce run-time
# by reducing the number of maze routes.
# This algorithm is > O(n^2) so remove it for now
# start_time = datetime.now()
# for pin in pin_list:
# self.combine_adjacent_pins(pin)
# print_time("Combining adjacent pins",datetime.now(), start_time, 4)
2019-11-07 17:33:13 +01:00
# Separate any adjacent grids of differing net names
# that overlap
# Must be done before enclosing pins
start_time = datetime.now()
self.separate_adjacent_pins(0)
2019-11-07 17:33:13 +01:00
print_time("Separating adjacent pins", datetime.now(), start_time, 4)
2020-11-03 15:29:17 +01:00
2019-11-07 17:33:13 +01:00
# Enclose the continguous grid units in a metal
# rectangle to fix some DRCs
start_time = datetime.now()
self.enclose_pins()
2019-11-07 17:33:13 +01:00
print_time("Enclosing pins", datetime.now(), start_time, 4)
# MRG: Removing this code for now. The later compute enclosure code
# assumes that all pins are touching and this may produce sets of pins
# that are not connected.
# def combine_adjacent_pins(self, pin_name):
# """
# Find pins that have adjacent routing tracks and merge them into a
2020-11-03 15:29:17 +01:00
# single pin_group. The pins themselves may not be touching, but
# enclose_pins in the next step will ensure they are touching.
2020-11-03 15:29:17 +01:00
# """
# debug.info(1,"Combining adjacent pins for {}.".format(pin_name))
# # Find all adjacencies
# adjacent_pins = {}
# for index1,pg1 in enumerate(self.pin_groups[pin_name]):
# for index2,pg2 in enumerate(self.pin_groups[pin_name]):
2020-11-03 15:29:17 +01:00
# # Cannot combine with yourself, also don't repeat
# if index1<=index2:
# continue
# # Combine if at least 1 grid cell is adjacent
# if pg1.adjacent(pg2):
# if not index1 in adjacent_pins:
# adjacent_pins[index1] = set([index2])
# else:
# adjacent_pins[index1].add(index2)
# # Make a list of indices to ensure every group gets in the new set
# all_indices = set([x for x in range(len(self.pin_groups[pin_name]))])
2020-11-03 15:29:17 +01:00
# # Now reconstruct the new groups
# new_pin_groups = []
# for index1,index2_set in adjacent_pins.items():
# # Remove the indices if they are added to the new set
# all_indices.discard(index1)
# all_indices.difference_update(index2_set)
# # Create the combined group starting with the first item
# combined = self.pin_groups[pin_name][index1]
# # Add all of the other items that overlapped
# for index2 in index2_set:
# pg = self.pin_groups[pin_name][index2]
# combined.add_group(pg)
# debug.info(3,"Combining {0} {1}:".format(pin_name, index2))
# debug.info(3, " {0}\n {1}".format(combined.pins, pg.pins))
# debug.info(3," --> {0}\n {1}".format(combined.pins,combined.grids))
# new_pin_groups.append(combined)
# # Add the pin groups that weren't added to the new set
# for index in all_indices:
# new_pin_groups.append(self.pin_groups[pin_name][index])
2020-11-03 15:29:17 +01:00
# old_size = len(self.pin_groups[pin_name])
# # Use the new pin group!
# self.pin_groups[pin_name] = new_pin_groups
# removed_pairs = old_size - len(new_pin_groups)
2019-11-07 17:33:13 +01:00
# debug.info(1,
# "Combined {0} pin groups for {1}".format(removed_pairs,pin_name))
2020-11-03 15:29:17 +01:00
# return removed_pairs
2020-11-03 15:29:17 +01:00
def separate_adjacent_pins(self, separation):
2018-10-29 19:18:12 +01:00
"""
2020-11-03 15:29:17 +01:00
This will try to separate all grid pins by the supplied
2019-11-07 17:33:13 +01:00
number of separation tracks (default is to prevent adjacency).
"""
pin_names = self.pin_groups.keys()
for (pin_name1, pin_name2) in itertools.combinations(pin_names, 2):
self.separate_adjacent_pin(pin_name1, pin_name2, separation)
2020-11-03 15:29:17 +01:00
def separate_adjacent_pin(self, pin_name1, pin_name2, separation):
"""
2019-11-07 17:33:13 +01:00
Go through all of the pin groups and check if any other pin group is
2018-10-29 19:18:12 +01:00
within a separation of it.
If so, reduce the pin group grid to not include the adjacent grid.
Try to do this intelligently to keep th pins enclosed.
"""
debug.info(2,
2019-11-07 17:33:13 +01:00
"Comparing {0} and {1} adjacency".format(pin_name1,
pin_name2))
2018-12-06 23:11:15 +01:00
removed_grids = 0
2019-11-07 17:33:13 +01:00
for index1, pg1 in enumerate(self.pin_groups[pin_name1]):
for index2, pg2 in enumerate(self.pin_groups[pin_name2]):
adj_grids = pg1.adjacent_grids(pg2, separation)
2018-12-06 23:11:15 +01:00
removed_grids += len(adj_grids)
# These should have the same length, so...
2019-11-07 17:33:13 +01:00
if len(adj_grids) > 0:
debug.info(3,
"Adjacent grids {0} {1} adj={2}".format(index1,
index2,
adj_grids))
self.remove_adjacent_grid(pg1, pg2, adj_grids)
2020-11-03 15:29:17 +01:00
debug.info(2, "Removed {} adjacent grids.".format(removed_grids))
def remove_adjacent_grid(self, pg1, pg2, adj_grids):
"""
Remove one of the adjacent grids in a heuristic manner.
2019-11-07 17:33:13 +01:00
This will try to keep the groups similar sized by
removing from the bigger group.
"""
2019-11-07 17:33:13 +01:00
if pg1.size() > pg2.size():
bigger = pg1
smaller = pg2
else:
bigger = pg2
smaller = pg1
2020-11-03 15:29:17 +01:00
for adj in adj_grids:
2020-11-03 15:29:17 +01:00
2019-11-07 17:33:13 +01:00
# If the adjacent grids are a subset of the secondary
# grids (i.e. not necessary) remove them from each
if adj in bigger.secondary_grids:
2019-11-07 17:33:13 +01:00
debug.info(3,"Removing {} from bigger secondary {}".format(adj,
bigger))
bigger.grids.remove(adj)
bigger.secondary_grids.remove(adj)
self.blocked_grids.add(adj)
2018-12-05 02:08:22 +01:00
elif adj in smaller.secondary_grids:
2019-11-07 17:33:13 +01:00
debug.info(3,"Removing {} from smaller secondary {}".format(adj,
smaller))
2018-12-05 02:08:22 +01:00
smaller.grids.remove(adj)
smaller.secondary_grids.remove(adj)
self.blocked_grids.add(adj)
2018-12-05 02:08:22 +01:00
else:
2019-11-07 17:33:13 +01:00
# If we couldn't remove from a secondary grid,
# we must remove from the primary
2018-12-05 02:08:22 +01:00
# grid of at least one pin
if adj in bigger.grids:
2019-11-07 17:33:13 +01:00
debug.info(3,"Removing {} from bigger primary {}".format(adj,
bigger))
bigger.grids.remove(adj)
elif adj in smaller.grids:
2019-11-07 17:33:13 +01:00
debug.info(3,"Removing {} from smaller primary {}".format(adj,
smaller))
smaller.grids.remove(adj)
def set_supply_rail_blocked(self, value):
# This is just a virtual function
pass
def prepare_blockages(self):
"""
Reset and add all of the blockages in the design.
"""
2019-11-07 17:33:13 +01:00
debug.info(3, "Preparing blockages.")
2020-11-03 15:29:17 +01:00
# Start fresh. Not the best for run-time, but simpler.
self.clear_all_blockages()
# This adds the initial blockges of the design
# which includes all blockages due to non-pin shapes
# print("BLOCKING:", self.blocked_grids)
2019-11-07 17:33:13 +01:00
self.set_blockages(self.blocked_grids, True)
2019-11-07 17:33:13 +01:00
# Block all of the supply rails
# (some will be unblocked if they're a target)
try:
self.set_supply_rail_blocked(True)
except AttributeError:
# If function doesn't exist, it isn't a supply router
pass
2020-11-03 15:29:17 +01:00
# Now go and block all of the blockages due to pin shapes.
# Some of these will get unblocked later if they are the source/target.
for name in self.pin_groups:
# This should be a superset of the grids...
blockage_grids = {y for x in self.pin_groups[name] for y in x.blockages}
2019-11-07 17:33:13 +01:00
self.set_blockages(blockage_grids, True)
# If we have paths that were recently routed, add them as blockages as well.
# We might later do rip-up and reroute so they might not be metal shapes in the design yet.
# Also, this prevents having to reload an entire GDS and find the blockage shapes.
2018-11-02 22:57:40 +01:00
self.set_blockages(self.path_blockages)
def convert_shape_to_units(self, shape):
2019-11-07 17:33:13 +01:00
"""
Scale a shape (two vector list) to user units
"""
unit_factor = [GDS["unit"][0]] * 2
2019-11-07 17:33:13 +01:00
ll = shape[0].scale(unit_factor)
ur = shape[1].scale(unit_factor)
return [ll, ur]
2020-11-03 15:29:17 +01:00
def min_max_coord(self, coord):
2019-11-07 17:33:13 +01:00
"""
Find the lowest and highest corner of a Rectangle
"""
coordinate = []
minx = min(coord[0][0], coord[1][0], coord[2][0], coord[3][0])
maxx = max(coord[0][0], coord[1][0], coord[2][0], coord[3][0])
miny = min(coord[0][1], coord[1][1], coord[2][1], coord[3][1])
maxy = max(coord[0][1], coord[1][1], coord[2][1], coord[3][1])
coordinate += [vector(minx, miny)]
coordinate += [vector(maxx, maxy)]
return coordinate
2019-11-07 17:33:13 +01:00
def get_inertia(self, p0, p1):
"""
Sets the direction based on the previous direction we came from.
"""
2016-11-17 22:26:03 +01:00
# direction (index) of movement
2019-11-07 17:33:13 +01:00
if p0.x != p1.x:
return 0
2019-11-07 17:33:13 +01:00
elif p0.y != p1.y:
return 1
2016-11-17 22:26:03 +01:00
else:
# z direction
return 2
def clear_blockages(self, pin_name):
"""
This function clears a given pin and all of its components from being blockages.
"""
blockage_grids = {y for x in self.pin_groups[pin_name] for y in x.blockages}
self.set_blockages(blockage_grids, False)
def clear_all_blockages(self):
2019-11-07 17:33:13 +01:00
"""
Clear all blockages on the grid.
"""
2019-11-07 17:33:13 +01:00
debug.info(3, "Clearing all blockages")
self.rg.clear_blockages()
2020-11-03 15:29:17 +01:00
def set_blockages(self, blockages, value=True):
""" Flag the blockages in the grid """
self.rg.set_blocked(blockages, value)
def convert_to_tracks(self, ll, ur, z):
debug.info(3, "Converting ll={0} ur={1} z={2}".format(str(ll),str(ur),z))
grid_list = []
2019-11-07 17:33:13 +01:00
for x in range(int(ll[0]), int(ur[0])+1):
for y in range(int(ll[1]), int(ur[1])+1):
grid_list.append(vector3d(x, y, z))
return set(grid_list)
def convert_blockage(self, blockage):
2019-11-07 17:33:13 +01:00
"""
Convert a pin layout blockage shape to routing grid tracks.
"""
# Inflate the blockage by half a spacing rule
[ll, ur] = self.convert_shape_to_tracks(blockage.inflate())
zlayer = self.get_zindex(blockage.lpp)
blockage_tracks = self.convert_to_tracks(ll, ur, zlayer)
return blockage_tracks
2020-11-03 15:29:17 +01:00
def convert_blockages(self):
""" Convert blockages to grid tracks. """
2019-11-07 17:33:13 +01:00
debug.info(1, "Converting blockages.")
for blockage in self.blockages:
2019-11-07 17:33:13 +01:00
debug.info(3, "Converting blockage {}".format(str(blockage)))
blockage_list = self.convert_blockage(blockage)
self.blocked_grids.update(blockage_list)
2020-11-03 15:29:17 +01:00
def get_blocked_grids(self):
"""
Return the blocked grids with their flag set
"""
#return set([x for x in self.blocked_grids if self.rg.is_blocked(x)])
# These are all the non-pin blockages
return self.blocked_grids
def retrieve_blockages(self, lpp):
"""
Recursive find boundaries as blockages to the routing grid.
"""
shapes = self.layout.getAllShapes(lpp)
for boundary in shapes:
2019-11-07 17:33:13 +01:00
ll = vector(boundary[0], boundary[1])
ur = vector(boundary[2], boundary[3])
rect = [ll, ur]
new_pin = pin_layout("blockage{}".format(len(self.blockages)),
rect,
lpp)
# If there is a rectangle that is the same in the pins,
# it isn't a blockage!
if new_pin not in self.all_pins:
self.blockages.append(new_pin)
def convert_point_to_units(self, p):
2020-11-03 15:29:17 +01:00
"""
2016-11-17 22:26:03 +01:00
Convert a path set of tracks to center line path.
"""
pt = vector3d(p)
2019-11-07 17:33:13 +01:00
pt = pt.scale(self.track_widths[0], self.track_widths[1], 1)
return pt
2016-11-17 22:26:03 +01:00
def convert_wave_to_units(self, wave):
2019-11-07 17:33:13 +01:00
"""
Convert a wave to a set of center points
"""
return [self.convert_point_to_units(i) for i in wave]
def convert_shape_to_tracks(self, shape):
2019-11-07 17:33:13 +01:00
"""
Convert a rectangular shape into track units.
"""
2019-11-07 17:33:13 +01:00
(ll, ur) = shape
ll = snap_to_grid(ll)
ur = snap_to_grid(ur)
# to scale coordinates to tracks
2019-11-07 17:33:13 +01:00
debug.info(3, "Converting [ {0} , {1} ]".format(ll, ur))
ll = ll.scale(self.track_factor)
ur = ur.scale(self.track_factor)
# We can round since we are using inflated shapes
# and the track points are at the center
ll = ll.round()
ur = ur.round()
2019-11-07 17:33:13 +01:00
return [ll, ur]
def convert_pin_to_tracks(self, pin_name, pin, expansion=0):
2019-11-07 17:33:13 +01:00
"""
Convert a rectangular pin shape into a list of track locations,layers.
2019-11-07 17:33:13 +01:00
If no pins are "on-grid" (i.e. sufficient overlap)
it makes the one with most overlap if it is not blocked.
If expansion>0, expamine areas beyond the current pin
when it is blocked.
"""
2019-11-07 17:33:13 +01:00
(ll, ur) = pin.rect
debug.info(3, "Converting pin [ {0} , {1} ]".format(ll, ur))
2020-11-03 15:29:17 +01:00
# scale the size bigger to include neaby tracks
2019-11-07 17:33:13 +01:00
ll = ll.scale(self.track_factor).floor()
ur = ur.scale(self.track_factor).ceil()
# Keep tabs on tracks with sufficient and insufficient overlap
sufficient_list = set()
insufficient_list = set()
zindex = self.get_zindex(pin.lpp)
for x in range(int(ll[0]) - expansion, int(ur[0]) + 1 + expansion):
for y in range(int(ll[1] - expansion), int(ur[1]) + 1 + expansion):
2019-11-07 17:33:13 +01:00
(full_overlap, partial_overlap) = self.convert_pin_coord_to_tracks(pin,
vector3d(x,
y,
zindex))
if full_overlap:
sufficient_list.update([full_overlap])
if partial_overlap:
insufficient_list.update([partial_overlap])
debug.info(3,
2019-11-07 17:33:13 +01:00
"Converting [ {0} , {1} ] full={2}".format(x,
y,
full_overlap))
# Return all grids with any potential overlap (sufficient or not)
2019-11-07 17:33:13 +01:00
return (sufficient_list, insufficient_list)
def get_all_offgrid_pin(self, pin, insufficient_list):
2019-11-07 17:33:13 +01:00
"""
Find a list of all pins with some overlap.
"""
2019-11-07 17:33:13 +01:00
# print("INSUFFICIENT LIST",insufficient_list)
# Find the coordinate with the most overlap
any_overlap = set()
for coord in insufficient_list:
full_pin = self.convert_track_to_pin(coord)
# Compute the overlap with that rectangle
2019-11-07 17:33:13 +01:00
overlap_rect = pin.compute_overlap(full_pin)
# Determine the max x or y overlap
max_overlap = max(overlap_rect)
2019-11-07 17:33:13 +01:00
if max_overlap > 0:
any_overlap.update([coord])
2020-11-03 15:29:17 +01:00
return any_overlap
def get_best_offgrid_pin(self, pin, insufficient_list):
2019-11-07 17:33:13 +01:00
"""
Find a list of the single pin with the most overlap.
"""
# Find the coordinate with the most overlap
best_coord = None
best_overlap = -math.inf
for coord in insufficient_list:
2018-10-25 01:13:07 +02:00
full_pin = self.convert_track_to_pin(coord)
# Compute the overlap with that rectangle
2019-11-07 17:33:13 +01:00
overlap_rect = pin.compute_overlap(full_pin)
# Determine the min x or y overlap
min_overlap = min(overlap_rect)
2019-11-07 17:33:13 +01:00
if min_overlap > best_overlap:
best_overlap = min_overlap
best_coord = coord
2020-11-03 15:29:17 +01:00
return set([best_coord])
2020-11-03 15:29:17 +01:00
def get_furthest_offgrid_pin(self, pin, insufficient_list):
2019-11-07 17:33:13 +01:00
"""
Get a grid cell that is the furthest from the blocked grids.
"""
2020-11-03 15:29:17 +01:00
# Find the coordinate with the most overlap
best_coord = None
best_dist = math.inf
for coord in insufficient_list:
min_dist = grid_utils.distance_set(coord, self.blocked_grids)
2019-11-07 17:33:13 +01:00
if min_dist < best_dist:
best_dist = min_dist
best_coord = coord
2020-11-03 15:29:17 +01:00
return set([best_coord])
def get_nearest_offgrid_pin(self, pin, insufficient_list):
2019-11-07 17:33:13 +01:00
"""
Given a pin and a list of grid cells (probably non-overlapping),
return the nearest grid cell (center to center).
"""
# Find the coordinate with the most overlap
best_coord = None
best_dist = math.inf
for coord in insufficient_list:
track_pin = self.convert_track_to_pin(coord)
min_dist = pin.distance(track_pin)
2019-11-07 17:33:13 +01:00
if min_dist < best_dist:
best_dist = min_dist
best_coord = coord
2020-11-03 15:29:17 +01:00
return set([best_coord])
2020-11-03 15:29:17 +01:00
def convert_pin_coord_to_tracks(self, pin, coord):
2019-11-07 17:33:13 +01:00
"""
Return all tracks that an inflated pin overlaps
"""
2019-11-07 17:33:13 +01:00
# This is using the full track shape rather
# than a single track pin shape
# because we will later patch a connector if there isn't overlap.
track_pin = self.convert_track_to_shape_pin(coord)
# This is the normal pin inflated by a minimum design rule
2019-11-07 17:33:13 +01:00
inflated_pin = pin_layout(pin.name,
pin.inflate(0.5 * self.track_space),
pin.layer)
2020-11-03 15:29:17 +01:00
overlap_length = pin.overlap_length(track_pin)
debug.info(4,"Check overlap: {0} {1} . {2} = {3}".format(coord,
2019-11-07 17:33:13 +01:00
pin.rect,
track_pin,
overlap_length))
inflated_overlap_length = inflated_pin.overlap_length(track_pin)
debug.info(4,"Check overlap: {0} {1} . {2} = {3}".format(coord,
2019-11-07 17:33:13 +01:00
inflated_pin.rect,
track_pin,
inflated_overlap_length))
# If it overlaps with the pin, it is sufficient
2019-11-07 17:33:13 +01:00
if overlap_length == math.inf or overlap_length > 0:
debug.info(4," Overlap: {0} >? {1}".format(overlap_length, 0))
2019-11-07 17:33:13 +01:00
return (coord, None)
# If it overlaps with the inflated pin, it is partial
2019-11-07 17:33:13 +01:00
elif inflated_overlap_length == math.inf or inflated_overlap_length > 0:
2020-11-03 15:29:17 +01:00
debug.info(4," Partial overlap: {0} >? {1}".format(inflated_overlap_length, 0))
2019-11-07 17:33:13 +01:00
return (None, coord)
else:
debug.info(4, " No overlap: {0} {1}".format(overlap_length, 0))
2019-11-07 17:33:13 +01:00
return (None, None)
2016-11-19 01:16:19 +01:00
def convert_track_to_pin(self, track):
2019-11-07 17:33:13 +01:00
"""
Convert a grid point into a rectangle shape that is centered
track in the track and leaves half a DRC space in each direction.
"""
2019-11-07 17:33:13 +01:00
# calculate lower left
x = track.x * self.track_width - 0.5 * self.track_width + 0.5 * self.track_space
y = track.y * self.track_width - 0.5 * self.track_width + 0.5 * self.track_space
ll = snap_to_grid(vector(x,y))
2020-11-03 15:29:17 +01:00
# calculate upper right
2019-11-07 17:33:13 +01:00
x = track.x * self.track_width + 0.5 * self.track_width - 0.5 * self.track_space
y = track.y * self.track_width + 0.5 * self.track_width - 0.5 * self.track_space
ur = snap_to_grid(vector(x, y))
2018-10-25 01:13:07 +02:00
p = pin_layout("", [ll, ur], self.get_layer(track[2]))
return p
def convert_track_to_shape_pin(self, track):
2019-11-07 17:33:13 +01:00
"""
Convert a grid point into a rectangle shape
that occupies the entire centered track.
"""
# to scale coordinates to tracks
x = track[0]*self.track_width - 0.5*self.track_width
y = track[1]*self.track_width - 0.5*self.track_width
# offset lowest corner object to to (-track halo,-track halo)
2019-11-07 17:33:13 +01:00
ll = snap_to_grid(vector(x, y))
ur = snap_to_grid(ll + vector(self.track_width, self.track_width))
p = pin_layout("", [ll, ur], self.get_layer(track[2]))
return p
def convert_track_to_shape(self, track):
2019-11-07 17:33:13 +01:00
"""
Convert a grid point into a rectangle shape
that occupies the entire centered track.
2016-11-19 01:16:19 +01:00
"""
# to scale coordinates to tracks
2019-05-31 17:43:37 +02:00
try:
x = track[0]*self.track_width - 0.5*self.track_width
except TypeError:
2019-11-07 17:33:13 +01:00
print(track[0], type(track[0]), self.track_width, type(self.track_width))
y = track[1]*self.track_width - 0.5*self.track_width
2016-11-19 01:16:19 +01:00
# offset lowest corner object to to (-track halo,-track halo)
2019-11-07 17:33:13 +01:00
ll = snap_to_grid(vector(x, y))
ur = snap_to_grid(ll + vector(self.track_width, self.track_width))
2016-11-19 01:16:19 +01:00
2019-11-07 17:33:13 +01:00
return [ll, ur]
2020-11-03 15:29:17 +01:00
def convert_track_to_inflated_pin(self, track):
"""
2019-11-07 17:33:13 +01:00
Convert a grid point into a rectangle shape
that is inflated by a half DRC space.
"""
# calculate lower left
x = track.x*self.track_width - 0.5*self.track_width - 0.5*self.track_space
y = track.y*self.track_width - 0.5*self.track_width - 0.5*self.track_space
ll = snap_to_grid(vector(x,y))
2020-11-03 15:29:17 +01:00
# calculate upper right
x = track.x*self.track_width + 0.5*self.track_width + 0.5*self.track_space
y = track.y*self.track_width + 0.5*self.track_width + 0.5*self.track_space
2019-11-07 17:33:13 +01:00
ur = snap_to_grid(vector(x, y))
p = pin_layout("", [ll, ur], self.get_layer(track[2]))
return p
2020-11-03 15:29:17 +01:00
def analyze_pins(self, pin_name):
"""
2019-11-07 17:33:13 +01:00
Analyze the shapes of a pin and combine
them into pin_groups which are connected.
"""
2020-11-03 15:29:17 +01:00
debug.info(2, "Analyzing pin groups for {}.".format(pin_name))
pin_set = self.pins[pin_name]
2018-10-29 21:49:29 +01:00
# This will be a list of pin tuples that overlap
overlap_list = []
# Sort the rectangles into a list with lower/upper y coordinates
bottom_y_coordinates = [(x.by(), x, "bottom") for x in pin_set]
top_y_coordinates = [(x.uy(), x, "top") for x in pin_set]
y_coordinates = bottom_y_coordinates + top_y_coordinates
y_coordinates.sort(key=lambda x: x[0])
# Map the pins to the lower indices
2019-11-07 17:33:13 +01:00
bottom_index_map = {x[1]: i for i, x in enumerate(y_coordinates) if x[2] == "bottom"}
# top_index_map = {x[1]: i for i, x in enumerate(y_coordinates) if x[2] == "bottom"}
# Sort the pin list by x coordinate
pin_list = list(pin_set)
pin_list.sort(key=lambda x: x.lx())
# for shapes in x order
for pin in pin_list:
# start at pin's lower y coordinate
bottom_index = bottom_index_map[pin]
compared_pins = set()
2019-11-07 17:33:13 +01:00
for i in range(bottom_index, len(y_coordinates)):
compare_pin = y_coordinates[i][1]
# Don't overlap yourself
2019-11-07 17:33:13 +01:00
if pin == compare_pin:
continue
# Done when we encounter any shape above the pin
if compare_pin.by() > pin.uy():
break
# Don't double compare the same pin twice
if compare_pin in compared_pins:
continue
compared_pins.add(compare_pin)
# If we overlap, add them to the list
if pin.overlaps(compare_pin):
2019-11-07 17:33:13 +01:00
overlap_list.append((pin, compare_pin))
# Initial unique group assignments
group_id = {}
gid = 1
for pin in pin_list:
group_id[pin] = gid
gid += 1
2020-11-03 15:29:17 +01:00
for p in overlap_list:
2019-11-07 17:33:13 +01:00
(p1, p2) = p
for pin in pin_list:
if group_id[pin] == group_id[p2]:
group_id[pin] = group_id[p1]
2020-11-03 15:29:17 +01:00
# For each pin add it to it's group
group_map = {}
for pin in pin_list:
gid = group_id[pin]
if gid not in group_map:
2019-11-07 17:33:13 +01:00
group_map[gid] = pin_group(name=pin_name,
pin_set=[],
router=self)
group_map[gid].add_pin(pin)
self.pin_groups[pin_name] = list(group_map.values())
2020-11-03 15:29:17 +01:00
def convert_pins(self, pin_name):
2019-11-07 17:33:13 +01:00
"""
Convert the pin groups into pin tracks and blockage tracks.
"""
debug.info(2, "Converting pins for {}.".format(pin_name))
for pg in self.pin_groups[pin_name]:
pg.convert_pin()
2020-11-03 15:29:17 +01:00
def enclose_pins(self):
"""
This will find the biggest rectangle enclosing some grid squares and
put a rectangle over it. It does not enclose grid squares
that are blocked by other shapes.
"""
for pin_name in self.pin_groups:
debug.info(2, "Enclosing pins for {}".format(pin_name))
for pg in self.pin_groups[pin_name]:
self.clear_blockages(pin_name)
pg.enclose_pin()
pg.add_enclosure(self.cell)
def add_source(self, pin_name):
2019-11-07 17:33:13 +01:00
"""
This will mark the grids for all pin components as a source.
Marking as source or target also clears blockage status.
"""
for i in range(self.num_pin_components(pin_name)):
self.add_pin_component_source(pin_name, i)
# Clearing the blockage of this pin requires the inflated pins
self.clear_blockages(pin_name)
def add_target(self, pin_name):
2019-11-07 17:33:13 +01:00
"""
This will mark the grids for all pin components as a target.
Marking as source or target also clears blockage status.
"""
for i in range(self.num_pin_components(pin_name)):
self.add_pin_component_target(pin_name, i)
2020-11-03 15:29:17 +01:00
# Clearing the blockage of this pin requires the inflated pins
self.clear_blockages(pin_name)
def add_perimeter_target(self, side="all"):
"""
This will mark all the cells on the perimeter of the original layout as a target.
"""
self.rg.add_perimeter_target(side=side)
def num_pin_components(self, pin_name):
2019-11-07 17:33:13 +01:00
"""
This returns how many disconnected pin components there are.
"""
return len(self.pin_groups[pin_name])
2020-11-03 15:29:17 +01:00
def add_pin_component_source(self, pin_name, index):
2019-11-07 17:33:13 +01:00
"""
2020-11-03 15:29:17 +01:00
This will mark only the pin tracks
2019-11-07 17:33:13 +01:00
from the indexed pin component as a source.
It also unsets it as a blockage.
"""
2019-11-07 17:33:13 +01:00
debug.check(index<self.num_pin_components(pin_name),
"Pin component index too large.")
2020-11-03 15:29:17 +01:00
pin_in_tracks = self.pin_groups[pin_name][index].grids
debug.info(3,"Set source: " + str(pin_name) + " " + str(pin_in_tracks))
self.rg.add_source(pin_in_tracks)
def add_path_target(self, paths):
2019-11-07 17:33:13 +01:00
"""
Set all of the paths as a target too.
"""
for p in paths:
self.rg.set_target(p)
2019-11-07 17:33:13 +01:00
self.rg.set_blocked(p, False)
def add_pin_component_target(self, pin_name, index):
2019-11-07 17:33:13 +01:00
"""
This will mark only the pin tracks
from the indexed pin component as a target.
It also unsets it as a blockage.
"""
2019-05-31 17:43:37 +02:00
debug.check(index<self.num_pin_components(pin_name),"Pin component index too large.")
2020-11-03 15:29:17 +01:00
pin_in_tracks = self.pin_groups[pin_name][index].grids
debug.info(3, "Set target: " + str(pin_name) + " " + str(pin_in_tracks))
self.rg.add_target(pin_in_tracks)
2019-05-28 02:38:59 +02:00
def add_pin_component_target_except(self, pin_name, index):
2019-11-07 17:33:13 +01:00
"""
2019-05-28 02:38:59 +02:00
This will mark the grids for all *other* pin components as a target.
Marking as source or target also clears blockage status.
"""
for i in range(self.num_pin_components(pin_name)):
if i != index:
self.add_pin_component_target(pin_name, i)
2020-11-03 15:29:17 +01:00
def set_component_blockages(self, pin_name, value=True):
2019-11-07 17:33:13 +01:00
"""
Block all of the pin components.
"""
2019-11-07 17:33:13 +01:00
debug.info(3, "Setting blockages {0} {1}".format(pin_name,value))
for pg in self.pin_groups[pin_name]:
self.set_blockages(pg.grids, value)
def prepare_path(self,path):
2019-11-07 17:33:13 +01:00
"""
Prepare a path or wave for routing ebedding.
2020-11-03 15:29:17 +01:00
This tracks the path, simplifies the path and
2019-11-07 17:33:13 +01:00
marks it as a path for debug output.
"""
2019-11-07 17:33:13 +01:00
debug.info(4, "Set path: " + str(path))
# This is marked for debug
path.set_path()
# For debugging... if the path failed to route.
2019-11-07 17:33:13 +01:00
# if False or path == None:
# self.write_debug_gds()
# First, simplify the path for
# debug.info(3, str(self.path))
contracted_path = self.contract_path(path)
2019-11-07 17:33:13 +01:00
debug.info(3, "Contracted path: " + str(contracted_path))
2020-11-03 15:29:17 +01:00
return contracted_path
2020-11-03 15:29:17 +01:00
2019-11-07 17:33:13 +01:00
def add_route(self, path):
"""
Add the current wire route to the given design instance.
"""
2019-11-07 17:33:13 +01:00
path = self.prepare_path(path)
2020-11-03 15:29:17 +01:00
debug.info(4, "Adding route: {}".format(str(path)))
# If it is only a square, add an enclosure to the track
2019-11-07 17:33:13 +01:00
if len(path) == 1:
self.add_single_enclosure(path[0][0])
else:
# convert the path back to absolute units from tracks
# This assumes 1-track wide again
abs_path = [self.convert_point_to_units(x[0]) for x in path]
# Otherwise, add the route which includes enclosures
2019-11-07 17:33:13 +01:00
if len(self.layers) > 1:
2019-05-31 17:43:37 +02:00
self.cell.add_route(layers=self.layers,
coordinates=abs_path,
layer_widths=self.layer_widths)
else:
self.cell.add_path(layer=self.layers[0],
coordinates=abs_path,
width=self.layer_widths[0])
2020-11-03 15:29:17 +01:00
def add_single_enclosure(self, track):
"""
2019-11-07 17:33:13 +01:00
Add a metal enclosure that is the size of
the routing grid minus a spacing on each side.
"""
2018-10-25 01:13:07 +02:00
pin = self.convert_track_to_pin(track)
2019-11-07 17:33:13 +01:00
(ll, ur) = pin.rect
self.cell.add_rect(layer=self.get_layer(track.z),
offset=ll,
width=ur.x-ll.x,
height=ur.y-ll.y)
2020-11-03 15:29:17 +01:00
2019-11-07 17:33:13 +01:00
def add_via(self, loc, size=1):
"""
Add a via centered at the current location
"""
2019-11-07 17:33:13 +01:00
loc = self.convert_point_to_units(vector3d(loc[0], loc[1], 0))
self.cell.add_via_center(layers=self.layers,
2019-11-07 17:33:13 +01:00
offset=vector(loc.x, loc.y),
size=(size, size))
def compute_pin_enclosure(self, ll, ur, zindex, name=""):
"""
Enclose the tracks from ll to ur in a single rectangle that meets
2019-11-07 17:33:13 +01:00
the track DRC rules.
"""
layer = self.get_layer(zindex)
2020-11-03 15:29:17 +01:00
2019-11-07 17:33:13 +01:00
# This finds the pin shape enclosed by the
# track with DRC spacing on the sides
2018-10-25 01:13:07 +02:00
pin = self.convert_track_to_pin(ll)
2019-11-07 17:33:13 +01:00
(abs_ll, unused) = pin.rect
2018-10-25 01:13:07 +02:00
pin = self.convert_track_to_pin(ur)
2019-11-07 17:33:13 +01:00
(unused, abs_ur) = pin.rect
2020-11-03 15:29:17 +01:00
pin = pin_layout(name, [abs_ll, abs_ur], layer)
2020-11-03 15:29:17 +01:00
return pin
2019-11-07 17:33:13 +01:00
def contract_path(self, path):
"""
Remove intermediate points in a rectilinear path or a wave.
"""
# Waves are always linear, so just return the first and last.
if self.is_wave(path):
2019-11-07 17:33:13 +01:00
return [path[0], path[-1]]
# Make a list only of points that change inertia of the path
newpath = [path[0]]
2019-11-07 17:33:13 +01:00
for i in range(1, len(path) - 1):
prev_inertia = self.get_inertia(path[i-1][0], path[i][0])
next_inertia = self.get_inertia(path[i][0], path[i+1][0])
# if we switch directions, add the point, otherwise don't
2019-11-07 17:33:13 +01:00
if prev_inertia != next_inertia:
newpath.append(path[i])
# always add the last path unless it was a single point
2019-11-07 17:33:13 +01:00
if len(path) > 1:
newpath.append(path[-1])
return newpath
2020-11-03 15:29:17 +01:00
def run_router(self, detour_scale):
"""
2019-11-07 17:33:13 +01:00
This assumes the blockages, source, and target are all set up.
"""
2019-05-31 17:43:37 +02:00
# Double check source and taget are not same node, if so, we are done!
2019-11-07 17:33:13 +01:00
for k, v in self.rg.map.items():
2019-05-31 17:43:37 +02:00
if v.source and v.target:
return True
2020-11-03 15:29:17 +01:00
# returns the path in tracks
2019-11-07 17:33:13 +01:00
(path, cost) = self.rg.route(detour_scale)
if path:
debug.info(2, "Found path: cost={0} {1}".format(cost, str(path)))
2019-05-31 17:43:37 +02:00
self.paths.append(grid_utils.flatten_set(path))
2018-11-02 22:57:40 +01:00
self.add_route(path)
self.path_blockages.append(self.paths[-1])
return True
else:
return False
def annotate_pin_and_tracks(self, pin, tracks):
""""
Annotate some shapes for debug purposes
"""
2019-11-07 17:33:13 +01:00
debug.info(0, "Annotating\n pin {0}\n tracks {1}".format(pin, tracks))
for coord in tracks:
2019-11-07 17:33:13 +01:00
(ll, ur) = self.convert_track_to_shape(coord)
self.cell.add_rect(layer="text",
offset=ll,
width=ur[0] - ll[0],
height=ur[1] - ll[1])
# (ll, ur) = self.convert_track_to_pin(coord).rect
# self.cell.add_rect(layer="boundary",
# offset=ll,
# width=ur[0] - ll[0],
# height=ur[1] - ll[1])
2019-11-07 17:33:13 +01:00
(ll, ur) = pin.rect
self.cell.add_rect(layer="text",
offset=ll,
width=ur[0] - ll[0],
height=ur[1] - ll[1])
def write_debug_gds(self, gds_name="debug_route.gds", stop_program=True):
"""
2019-11-07 17:33:13 +01:00
Write out a GDS file with the routing grid and
search information annotated on it.
"""
debug.info(0, "Writing annotated router gds file to {}".format(gds_name))
self.add_router_info()
self.cell.gds_write(gds_name)
self.del_router_info()
if stop_program:
import sys
sys.exit(1)
def annotate_grid(self, g):
"""
Display grid information in the GDS file for a single grid cell.
"""
shape = self.convert_track_to_shape(g)
partial_track = vector(0, self.track_width / 6.0)
self.cell.add_rect(layer="text",
offset=shape[0],
width=shape[1].x - shape[0].x,
height=shape[1].y - shape[0].y)
2019-11-07 17:33:13 +01:00
t = self.rg.map[g].get_type()
2020-11-03 15:29:17 +01:00
# midpoint offset
off = vector((shape[1].x + shape[0].x) / 2,
(shape[1].y + shape[0].y) / 2)
if t:
2019-11-07 17:33:13 +01:00
if g[2] == 1:
2018-11-02 22:57:40 +01:00
# Upper layer is upper right label
2019-11-07 17:33:13 +01:00
type_off = off + partial_track
2018-11-02 22:57:40 +01:00
else:
# Lower layer is lower left label
2019-11-07 17:33:13 +01:00
type_off = off - partial_track
self.cell.add_label(text=str(t),
layer="text",
offset=type_off)
2018-11-02 22:57:40 +01:00
2019-11-07 17:33:13 +01:00
t = self.rg.map[g].get_cost()
partial_track = vector(self.track_width/6.0, 0)
if t:
if g[2] == 1:
2018-11-02 22:57:40 +01:00
# Upper layer is right label
2019-11-07 17:33:13 +01:00
type_off = off + partial_track
2018-11-02 22:57:40 +01:00
else:
# Lower layer is left label
2019-11-07 17:33:13 +01:00
type_off = off - partial_track
2018-11-02 22:57:40 +01:00
self.cell.add_label(text=str(t),
layer="text",
offset=type_off)
2020-11-03 15:29:17 +01:00
2019-11-07 17:33:13 +01:00
self.cell.add_label(text="{0},{1}".format(g[0], g[1]),
2018-10-26 18:25:10 +02:00
layer="text",
offset=shape[0])
2019-05-31 17:43:37 +02:00
def del_router_info(self):
"""
Erase all of the comments on the current level.
"""
debug.info(2, "Erasing router info")
lpp = techlayer["text"]
self.cell.objs = [x for x in self.cell.objs if x.lpp != lpp]
2020-11-03 15:29:17 +01:00
def add_router_info(self):
"""
2019-11-07 17:33:13 +01:00
Write the routing grid and router cost, blockage, pins on
the boundary layer for debugging purposes. This can only be
called once or the labels will overlap.
"""
debug.info(2, "Adding router info")
2018-11-02 22:57:40 +01:00
show_blockages = False
show_blockage_grids = False
show_enclosures = False
show_all_grids = True
2020-11-03 15:29:17 +01:00
if show_all_grids:
for g in self.rg.map:
self.annotate_grid(g)
2020-11-03 15:29:17 +01:00
if show_blockages:
# Display the inflated blockage
for blockage in self.blockages:
2019-11-07 17:33:13 +01:00
debug.info(1, "Adding {}".format(blockage))
(ll, ur) = blockage.inflate()
self.cell.add_rect(layer="text",
offset=ll,
width=ur.x - ll.x,
height=ur.y - ll.y)
if show_blockage_grids:
2019-11-07 17:33:13 +01:00
self.set_blockages(self.blocked_grids, True)
for g in self.rg.map:
self.annotate_grid(g)
if show_enclosures:
for key in self.pin_groups:
2018-10-26 18:25:10 +02:00
for pg in self.pin_groups[key]:
if not pg.enclosed:
continue
2018-10-26 18:25:10 +02:00
for pin in pg.enclosures:
2019-11-07 17:33:13 +01:00
# print("enclosure: ",
# pin.name,
# pin.ll(),
# pin.width(),
# pin.height())
2018-10-26 18:25:10 +02:00
self.cell.add_rect(layer="text",
offset=pin.ll(),
width=pin.width(),
height=pin.height())
def get_perimeter_pin(self):
""" Return the shape of the last routed path that was on the perimeter """
for v in self.paths[-1]:
if self.rg.is_target(v):
return self.convert_track_to_pin(v)
return None
def get_pin(self, pin_name):
""" Return the lowest, leftest pin group """
keep_pin = None
for index,pg in enumerate(self.pin_groups[pin_name]):
for pin in pg.enclosures:
if not keep_pin:
keep_pin = pin
else:
if pin.lx() <= keep_pin.lx() and pin.by() <= keep_pin.by():
keep_pin = pin
return keep_pin
def check_all_routed(self, pin_name):
"""
Check that all pin groups are routed.
"""
for pg in self.pin_groups[pin_name]:
if not pg.is_routed():
return False
2019-11-07 17:33:13 +01:00
# FIXME: This should be replaced with vector.snap_to_grid at some point
def snap_to_grid(offset):
"""
Changes the coodrinate to match the grid settings
"""
xoff = snap_val_to_grid(offset[0])
2019-11-07 17:33:13 +01:00
yoff = snap_val_to_grid(offset[1])
return vector(xoff, yoff)
2019-11-07 17:33:13 +01:00
def snap_val_to_grid(x):
grid = drc("grid")
xgrid = int(round(round((x / grid), 2), 0))
xoff = xgrid * grid
return xoff