# See LICENSE for licensing information. # #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. # import hierarchy_layout import hierarchy_spice import globals import verify import debug import os from globals import OPTS import graph_util total_drc_errors = 0 total_lvs_errors = 0 class hierarchy_design(hierarchy_spice.spice, hierarchy_layout.layout): """ Design Class for all modules to inherit the base features. Class consisting of a set of modules and instances of these modules """ name_map = [] def __init__(self, name): self.gds_file = OPTS.openram_tech + "gds_lib/" + name + ".gds" self.sp_file = OPTS.openram_tech + "sp_lib/" + name + ".sp" self.name = name hierarchy_spice.spice.__init__(self, name) hierarchy_layout.layout.__init__(self, name) self.init_graph_params() def get_layout_pins(self,inst): """ Return a map of pin locations of the instance offset """ # find the instance for i in self.insts: if i.name == inst.name: break else: debug.error("Couldn't find instance {0}".format(inst_name),-1) inst_map = inst.mod.pin_map return inst_map def DRC_LVS(self, final_verification=False, top_level=False): """Checks both DRC and LVS for a module""" # Final verification option does not allow nets to be connected by label. # Unit tests will check themselves. if OPTS.is_unit_test: return if not OPTS.check_lvsdrc: return # Do not run if disabled in options. if (OPTS.inline_lvsdrc or top_level): global total_drc_errors global total_lvs_errors tempspice = "{0}/{1}.sp".format(OPTS.openram_temp,self.name) tempgds = "{0}/{1}.gds".format(OPTS.openram_temp,self.name) self.sp_write(tempspice) self.gds_write(tempgds) num_drc_errors = verify.run_drc(self.name, tempgds, final_verification) num_lvs_errors = verify.run_lvs(self.name, tempgds, tempspice, final_verification) debug.check(num_drc_errors == 0,"DRC failed for {0} with {1} error(s)".format(self.name,num_drc_errors)) debug.check(num_lvs_errors == 0,"LVS failed for {0} with {1} errors(s)".format(self.name,num_lvs_errors)) total_drc_errors += num_drc_errors total_lvs_errors += num_lvs_errors os.remove(tempspice) os.remove(tempgds) def DRC(self, final_verification=False): """Checks DRC for a module""" # Unit tests will check themselves. # Do not run if disabled in options. if (not OPTS.is_unit_test and OPTS.check_lvsdrc and (OPTS.inline_lvsdrc or final_verification)): global total_drc_errors tempgds = "{0}/{1}.gds".format(OPTS.openram_temp,self.name) self.gds_write(tempgds) num_errors = verify.run_drc(self.name, tempgds, final_verification) total_drc_errors += num_errors debug.check(num_errors == 0,"DRC failed for {0} with {1} error(s)".format(self.name,num_error)) os.remove(tempgds) def LVS(self, final_verification=False): """Checks LVS for a module""" # Unit tests will check themselves. # Do not run if disabled in options. if (not OPTS.is_unit_test and OPTS.check_lvsdrc and (OPTS.inline_lvsdrc or final_verification)): global total_lvs_errors tempspice = "{0}/{1}.sp".format(OPTS.openram_temp,self.name) tempgds = "{0}/{1}.gds".format(OPTS.openram_temp,self.name) self.sp_write(tempspice) self.gds_write(tempgds) num_errors = verify.run_lvs(self.name, tempgds, tempspice, final_verification) total_lvs_errors += num_errors debug.check(num_errors == 0,"LVS failed for {0} with {1} error(s)".format(self.name,num_errors)) os.remove(tempspice) os.remove(tempgds) #Example graph run # import graph_util # graph = graph_util.graph() # pins = ['A','Z','vdd','gnd'] # d.build_graph(graph,"Xpdriver",pins) # graph.remove_edges('vdd') # graph.remove_edges('gnd') # debug.info(1,"{}".format(graph)) # graph.print_all_paths('A', 'Z') def init_graph_params(self): """Initializes parameters relevant to the graph creation""" #Only initializes a set for checking instances which should not be added self.graph_inst_exclude = set() def build_graph(self, graph, inst_name, port_nets): """Recursively create graph from instances in module.""" #Translate port names to external nets if len(port_nets) != len(self.pins): debug.error("Port length mismatch:\nExt nets={}, Ports={}".format(port_nets,self.pins),1) port_dict = {pin:port for pin,port in zip(self.pins, port_nets)} debug.info(3, "Instance name={}".format(inst_name)) for subinst, conns in zip(self.insts, self.conns): if subinst in self.graph_inst_exclude: continue subinst_name = inst_name+'.X'+subinst.name subinst_ports = self.translate_nets(conns, port_dict, inst_name) subinst.mod.build_graph(graph, subinst_name, subinst_ports) def translate_nets(self, subinst_ports, port_dict, inst_name): """Converts connection names to their spice hierarchy equivalent""" converted_conns = [] for conn in subinst_ports: if conn in port_dict: converted_conns.append(port_dict[conn]) else: converted_conns.append("{}.{}".format(inst_name, conn)) return converted_conns def add_graph_edges(self, graph, port_nets): """For every input, adds an edge to every output. Only intended to be used for gates and other simple modules.""" #The final pin names will depend on the spice hierarchy, so #they are passed as an input. pin_dict = {pin:port for pin,port in zip(self.pins, port_nets)} input_pins = self.get_inputs() output_pins = self.get_outputs() inout_pins = self.get_inouts() for inp in input_pins+inout_pins: for out in output_pins+inout_pins: if inp != out: #do not add self loops graph.add_edge(pin_dict[inp], pin_dict[out]) def __str__(self): """ override print function output """ pins = ",".join(self.pins) insts = [" {}".format(x) for x in self.insts] objs = [" {}".format(x) for x in self.objs] s = "********** design {0} **********".format(self.name) s += "\n pins ({0})={1}\n".format(len(self.pins), pins) s += "\n objs ({0})=\n{1}\n".format(len(self.objs), "\n".join(objs)) s += "\n insts ({0})=\n{1}\n".format(len(self.insts), "\n".join(insts)) return s def __repr__(self): """ override print function output """ text="( design: " + self.name + " pins=" + str(self.pins) + " " + str(self.width) + "x" + str(self.height) + " )\n" for i in self.objs: text+=str(i)+",\n" for i in self.insts: text+=str(i)+",\n" return text