OpenRAM/compiler/modules/dff_buf.py

190 lines
7.3 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.
#
2018-02-17 00:25:27 +01:00
import debug
import design
from tech import layer
from tech import cell_properties as props
2018-02-17 00:25:27 +01:00
from vector import vector
from globals import OPTS
2019-01-17 01:15:38 +01:00
from sram_factory import factory
2018-02-17 00:25:27 +01:00
2020-04-16 00:55:49 +02:00
2018-02-17 00:25:27 +01:00
class dff_buf(design.design):
"""
This is a simple buffered DFF. The output is buffered
with two inverters, of variable size, to provide q
and qbar. This is to enable driving large fanout loads.
"""
unique_id = 1
2020-11-03 15:29:17 +01:00
def __init__(self, inv1_size=2, inv2_size=4, name=""):
2018-02-17 00:25:27 +01:00
if name=="":
name = "dff_buf_{0}".format(dff_buf.unique_id)
dff_buf.unique_id += 1
2020-08-06 20:33:26 +02:00
super().__init__(name)
2018-02-17 00:25:27 +01:00
debug.info(1, "Creating {}".format(self.name))
self.add_comment("inv1: {0} inv2: {1}".format(inv1_size, inv2_size))
2020-11-03 15:29:17 +01:00
# This is specifically for SCMOS where the DFF vdd/gnd rails are more than min width.
# This causes a DRC in the pinv which assumes min width rails. This ensures the output
# contact does not violate spacing to the rail in the NMOS.
debug.check(inv1_size>=2, "Inverter must be greater than two for rail spacing DRC rules.")
debug.check(inv2_size>=2, "Inverter must be greater than two for rail spacing DRC rules.")
self.inv1_size=inv1_size
self.inv2_size=inv2_size
2020-11-03 15:29:17 +01:00
self.create_netlist()
if not OPTS.netlist_only:
self.create_layout()
def create_netlist(self):
self.add_modules()
self.add_pins()
2018-11-14 01:05:22 +01:00
self.create_instances()
def create_layout(self):
self.place_instances()
self.width = self.inv2_inst.rx()
self.height = self.dff.height
self.route_wires()
self.add_layout_pins()
self.add_boundary()
self.DRC_LVS()
2020-11-03 15:29:17 +01:00
def add_modules(self):
2019-01-17 01:15:38 +01:00
self.dff = factory.create(module_type="dff")
2018-02-17 00:25:27 +01:00
self.add_mod(self.dff)
2019-01-17 01:15:38 +01:00
self.inv1 = factory.create(module_type="pinv",
size=self.inv1_size,
height=self.dff.height)
2018-02-17 00:25:27 +01:00
self.add_mod(self.inv1)
2019-01-17 01:15:38 +01:00
self.inv2 = factory.create(module_type="pinv",
size=self.inv2_size,
height=self.dff.height)
2018-02-17 00:25:27 +01:00
self.add_mod(self.inv2)
2020-11-03 15:29:17 +01:00
2018-02-17 00:25:27 +01:00
def add_pins(self):
self.add_pin_list(props.dff_buf.port_names,
props.dff_buf.port_types)
2018-11-14 01:05:22 +01:00
def create_instances(self):
2018-02-17 00:25:27 +01:00
self.dff_inst=self.add_inst(name="dff_buf_dff",
mod=self.dff)
2020-04-05 12:58:26 +02:00
2020-11-14 00:55:55 +01:00
self.connect_inst([props.dff_buf.pin.D, "qint", props.dff_buf.pin.clk, props.dff_buf.pin.vdd, props.dff_buf.pin.gnd])
2018-02-17 00:25:27 +01:00
self.inv1_inst=self.add_inst(name="dff_buf_inv1",
mod=self.inv1)
2020-11-14 00:55:55 +01:00
self.connect_inst(["qint", "Qb", props.dff_buf.pin.vdd, props.dff_buf.pin.gnd])
2020-11-03 15:29:17 +01:00
2018-02-17 00:25:27 +01:00
self.inv2_inst=self.add_inst(name="dff_buf_inv2",
mod=self.inv2)
2020-11-14 00:55:55 +01:00
self.connect_inst(["Qb", props.dff_buf.pin.Q, props.dff_buf.pin.vdd, props.dff_buf.pin.gnd])
2018-11-14 01:05:22 +01:00
def place_instances(self):
# Add the DFF
self.dff_inst.place(vector(0, 0))
# Add INV1 to the right
# The INV needs well spacing because the DFF is likely from a library
# with different well construction rules
2020-04-16 00:55:49 +02:00
well_spacing = 0
try:
well_spacing = max(well_spacing, self.nwell_space)
except AttributeError:
pass
try:
well_spacing = max(well_spacing, self.pwell_space)
except AttributeError:
pass
try:
well_spacing = max(well_spacing, self.pwell_to_nwell)
except AttributeError:
pass
self.inv1_inst.place(vector(self.dff_inst.rx() + well_spacing + self.well_extend_active, 0))
2020-11-03 15:29:17 +01:00
# Add INV2 to the right
2020-04-16 00:55:49 +02:00
self.inv2_inst.place(vector(self.inv1_inst.rx(), 0))
2020-11-03 15:29:17 +01:00
def route_wires(self):
if "li" in layer:
self.route_layer = "li"
else:
self.route_layer = "m1"
2020-11-03 15:29:17 +01:00
2018-02-17 00:25:27 +01:00
# Route dff q to inv1 a
2020-11-14 00:55:55 +01:00
q_pin = self.dff_inst.get_pin(props.dff.pin.Q)
2018-02-17 00:25:27 +01:00
a1_pin = self.inv1_inst.get_pin("A")
mid1 = vector(a1_pin.cx(), q_pin.cy())
2020-06-13 00:23:51 +02:00
self.add_path(q_pin.layer, [q_pin.center(), mid1, a1_pin.center()], width=q_pin.height())
self.add_via_stack_center(from_layer=a1_pin.layer,
to_layer=q_pin.layer,
offset=a1_pin.center())
2018-02-17 00:25:27 +01:00
# Route inv1 z to inv2 a
z1_pin = self.inv1_inst.get_pin("Z")
a2_pin = self.inv2_inst.get_pin("A")
self.mid_qb_pos = vector(0.5 * (z1_pin.cx() + a2_pin.cx()), z1_pin.cy())
self.add_zjog(z1_pin.layer, z1_pin.center(), a2_pin.center())
2020-11-03 15:29:17 +01:00
2018-02-17 00:25:27 +01:00
def add_layout_pins(self):
# Continous vdd rail along with label.
2020-11-14 00:55:55 +01:00
vdd_pin=self.dff_inst.get_pin(props.dff.pin.vdd)
self.add_layout_pin(text=props.dff_buf.pin.vdd,
layer=vdd_pin.layer,
2018-02-17 00:25:27 +01:00
offset=vdd_pin.ll(),
width=self.width,
height=vdd_pin.height())
# Continous gnd rail along with label.
2020-11-14 00:55:55 +01:00
gnd_pin=self.dff_inst.get_pin(props.dff.pin.gnd)
self.add_layout_pin(text=props.dff_buf.pin.gnd,
layer=gnd_pin.layer,
2018-02-17 00:25:27 +01:00
offset=gnd_pin.ll(),
width=self.width,
height=vdd_pin.height())
2020-11-03 15:29:17 +01:00
2020-11-14 00:55:55 +01:00
clk_pin = self.dff_inst.get_pin(props.dff.pin.clk)
self.add_layout_pin(text=props.dff_buf.pin.clk,
2018-02-17 00:25:27 +01:00
layer=clk_pin.layer,
offset=clk_pin.ll(),
width=clk_pin.width(),
height=clk_pin.height())
2020-11-14 00:55:55 +01:00
din_pin = self.dff_inst.get_pin(props.dff_buf.pin.D)
self.add_layout_pin(text=props.dff_buf.pin.D,
2018-02-17 00:25:27 +01:00
layer=din_pin.layer,
offset=din_pin.ll(),
width=din_pin.width(),
height=din_pin.height())
dout_pin = self.inv2_inst.get_pin("Z")
2020-06-13 00:23:51 +02:00
mid_pos = dout_pin.center() + vector(self.m2_nonpref_pitch, 0)
q_pos = mid_pos - vector(0, 2 * self.m2_nonpref_pitch)
2020-11-14 00:55:55 +01:00
self.add_layout_pin_rect_center(text=props.dff_buf.pin.Q,
layer="m2",
offset=q_pos)
self.add_path(self.route_layer, [dout_pin.center(), mid_pos, q_pos])
self.add_via_stack_center(from_layer=dout_pin.layer,
to_layer="m2",
offset=q_pos)
2020-06-13 00:23:51 +02:00
qb_pos = self.mid_qb_pos + vector(0, 2 * self.m2_nonpref_pitch)
self.add_layout_pin_rect_center(text="Qb",
layer="m2",
offset=qb_pos)
self.add_path(self.route_layer, [self.mid_qb_pos, qb_pos])
a2_pin = self.inv2_inst.get_pin("A")
self.add_via_stack_center(from_layer=a2_pin.layer,
to_layer="m2",
offset=qb_pos)