OpenRAM/compiler/sram/sram_1bank.py

620 lines
29 KiB
Python
Raw Normal View History

# See LICENSE for licensing information.
#
2021-01-22 20:23:28 +01:00
# Copyright (c) 2016-2021 Regents of the University of California and The Board
2019-06-14 17:43:41 +02:00
# of Regents for the Oklahoma Agricultural and Mechanical College
# (acting for and on behalf of Oklahoma State University)
# All rights reserved.
#
from vector import vector
from sram_base import sram_base
2020-01-30 02:45:33 +01:00
from contact import m2_via
from channel_route import channel_route
from globals import OPTS
class sram_1bank(sram_base):
"""
Procedures specific to a one bank SRAM.
"""
def __init__(self, name, sram_config):
sram_base.__init__(self, name, sram_config)
2020-11-03 15:29:17 +01:00
def create_modules(self):
2020-03-06 01:13:49 +01:00
"""
This adds the modules for a single bank SRAM with control
2020-03-06 01:13:49 +01:00
logic.
"""
2020-11-03 15:29:17 +01:00
self.bank_inst=self.create_bank(0)
2018-11-14 01:05:22 +01:00
self.control_logic_insts = self.create_control_logic()
2018-11-14 01:05:22 +01:00
self.row_addr_dff_insts = self.create_row_addr_dff()
if self.col_addr_dff:
2018-11-14 01:05:22 +01:00
self.col_addr_dff_insts = self.create_col_addr_dff()
2019-07-03 22:17:12 +02:00
if self.write_size:
self.wmask_dff_insts = self.create_wmask_dff()
self.data_dff_insts = self.create_data_dff()
else:
self.data_dff_insts = self.create_data_dff()
2019-07-04 20:14:32 +02:00
if self.num_spare_cols:
self.spare_wen_dff_insts = self.create_spare_wen_dff()
else:
self.num_spare_cols = 0
def place_instances(self):
2020-03-06 01:13:49 +01:00
"""
This places the instances for a single bank SRAM with control
logic and up to 2 ports.
"""
2020-11-03 15:29:17 +01:00
# No orientation or offset
self.place_bank(self.bank_inst, [0, 0], 1, 1)
2018-08-14 19:36:35 +02:00
# The control logic is placed such that the vertical center (between the delay/RBL and
# the actual control logic is aligned with the vertical center of the bank (between
# the sense amps/column mux and cell array)
2018-08-14 19:36:35 +02:00
# The x-coordinate is placed to allow a single clock wire (plus an extra pitch)
# up to the row address DFFs.
self.control_pos = [None] * len(self.all_ports)
self.row_addr_pos = [None] * len(self.all_ports)
# DFFs are placd on their own
self.col_addr_pos = [None] * len(self.all_ports)
self.wmask_pos = [None] * len(self.all_ports)
self.spare_wen_pos = [None] * len(self.all_ports)
self.data_pos = [None] * len(self.all_ports)
# These positions utilize the channel route sizes.
# FIXME: Auto-compute these rather than manual computation.
# If a horizontal channel, they rely on the vertical channel non-preferred (contacted) pitch.
# If a vertical channel, they rely on the horizontal channel non-preferred (contacted) pitch.
# So, m3 non-pref pitch means that this is routed on the m2 layer.
2020-06-27 17:23:12 +02:00
self.data_bus_gap = self.m4_nonpref_pitch * 2
2020-06-29 14:54:30 +02:00
# Spare wen are on a separate layer so not included
# Start with 1 track minimum
self.data_bus_size = [1] * len(self.all_ports)
self.col_addr_bus_size = [1] * len(self.all_ports)
2020-06-29 14:54:30 +02:00
for port in self.all_ports:
# The column address wires are routed separately from the data bus and will always be smaller.
2020-06-29 14:54:30 +02:00
# All ports need the col addr flops
self.col_addr_bus_size[port] = self.col_addr_size * self.m4_nonpref_pitch
2020-06-29 14:54:30 +02:00
# Write ports need the data input flops and write mask flops
if port in self.write_ports:
self.data_bus_size[port] += self.num_wmasks + self.word_size
# This is for the din pins that get routed in the same channel
# when we have dout and din together
if port in self.readwrite_ports:
self.data_bus_size[port] += self.word_size
2020-06-29 14:54:30 +02:00
# Convert to length
self.data_bus_size[port] *= self.m4_nonpref_pitch
# Add the gap in unit length
self.data_bus_size[port] += self.data_bus_gap
2018-11-29 18:54:29 +01:00
# The control and row addr flops are independent of any bus widths.
self.place_control()
self.place_row_addr_dffs()
# Place with an initial wide channel (from above)
self.place_dffs()
# Route the channel and set to the new data bus size
# We need to temporarily add some pins for the x offsets
# but we'll remove them so that they have the right y
# offsets after the DFF placement.
self.add_layout_pins(add_vias=False)
self.route_dffs(add_routes=False)
self.remove_layout_pins()
# Re-place with the new channel size
self.place_dffs()
def place_row_addr_dffs(self):
"""
Must be run after place control logic.
"""
port = 0
# The row address bits are placed above the control logic aligned on the right.
x_offset = self.control_logic_insts[port].rx() - self.row_addr_dff_insts[port].width
# It is above the control logic and the predecoder array
y_offset = max(self.control_logic_insts[port].uy(), self.bank.predecoder_top)
self.row_addr_pos[port] = vector(x_offset, y_offset)
self.row_addr_dff_insts[port].place(self.row_addr_pos[port])
2018-11-29 18:54:29 +01:00
if len(self.all_ports)>1:
port = 1
# The row address bits are placed above the control logic aligned on the left.
x_offset = self.control_pos[port].x - self.control_logic_insts[port].width + self.row_addr_dff_insts[port].width
# If it can be placed above the predecoder and below the control logic, do it
y_offset = self.bank.predecoder_bottom
self.row_addr_pos[port] = vector(x_offset, y_offset)
self.row_addr_dff_insts[port].place(self.row_addr_pos[port], mirror="XY")
2020-11-03 15:29:17 +01:00
def place_control(self):
port = 0
2020-11-03 15:29:17 +01:00
2018-11-29 18:54:29 +01:00
# This includes 2 M2 pitches for the row addr clock line.
# The delay line is aligned with the bitcell array while the control logic is aligned with the port_data
# using the control_logic_center value.
self.control_pos[port] = vector(-self.control_logic_insts[port].width - 2 * self.m2_pitch,
self.bank.bank_array_ll.y - self.control_logic_insts[port].mod.control_logic_center.y)
self.control_logic_insts[port].place(self.control_pos[port])
if len(self.all_ports) > 1:
port = 1
# This includes 2 M2 pitches for the row addr clock line
# The delay line is aligned with the bitcell array while the control logic is aligned with the port_data
# using the control_logic_center value.
self.control_pos[port] = vector(self.bank_inst.rx() + self.control_logic_insts[port].width + 2 * self.m2_pitch,
self.bank.bank_array_ur.y
+ self.control_logic_insts[port].height
- self.control_logic_insts[port].height
+ self.control_logic_insts[port].mod.control_logic_center.y)
self.control_logic_insts[port].place(self.control_pos[port], mirror="XY")
def place_dffs(self):
"""
Place the col addr, data, wmask, and spare data DFFs.
This can be run more than once after we recompute the channel width.
"""
port = 0
2020-06-27 17:23:12 +02:00
# Add the col address flops below the bank to the right of the control logic
x_offset = self.control_logic_insts[port].rx() + self.dff.width
# Place it a data bus below the x-axis, but at least as low as the control logic to not block
# the control logic signals
y_offset = min(-self.data_bus_size[port] - self.dff.height,
self.control_logic_insts[port].by())
2020-06-27 17:23:12 +02:00
if self.col_addr_dff:
self.col_addr_pos[port] = vector(x_offset,
y_offset)
self.col_addr_dff_insts[port].place(self.col_addr_pos[port])
2020-06-27 17:23:12 +02:00
x_offset = self.col_addr_dff_insts[port].rx()
else:
self.col_addr_pos[port] = vector(x_offset, 0)
2020-11-03 15:29:17 +01:00
2020-06-27 17:23:12 +02:00
if port in self.write_ports:
if self.write_size:
# Add the write mask flops below the write mask AND array.
self.wmask_pos[port] = vector(x_offset,
y_offset)
self.wmask_dff_insts[port].place(self.wmask_pos[port])
2020-06-27 17:23:12 +02:00
x_offset = self.wmask_dff_insts[port].rx()
2018-11-29 18:54:29 +01:00
# Add the data flops below the write mask flops.
self.data_pos[port] = vector(x_offset,
y_offset)
self.data_dff_insts[port].place(self.data_pos[port])
x_offset = self.data_dff_insts[port].rx()
# Add spare write enable flops to the right of data flops since the spare columns
# will be on the right
2020-06-27 17:23:12 +02:00
if self.num_spare_cols:
self.spare_wen_pos[port] = vector(x_offset,
y_offset)
self.spare_wen_dff_insts[port].place(self.spare_wen_pos[port])
2020-06-27 17:23:12 +02:00
x_offset = self.spare_wen_dff_insts[port].rx()
2020-06-27 17:23:12 +02:00
else:
self.wmask_pos[port] = vector(x_offset, y_offset)
self.data_pos[port] = vector(x_offset, y_offset)
self.spare_wen_pos[port] = vector(x_offset, y_offset)
if len(self.all_ports) > 1:
port = 1
2020-11-03 15:29:17 +01:00
2020-06-27 17:23:12 +02:00
# Add the col address flops below the bank to the right of the control logic
x_offset = self.control_logic_insts[port].lx() - 2 * self.dff.width
# Place it a data bus below the x-axis, but at least as high as the control logic to not block
# the control logic signals
y_offset = max(self.bank.height + self.data_bus_size[port] + self.dff.height,
self.control_logic_insts[port].uy() - self.dff.height)
2020-06-27 17:23:12 +02:00
if self.col_addr_dff:
self.col_addr_pos[port] = vector(x_offset,
y_offset)
self.col_addr_dff_insts[port].place(self.col_addr_pos[port], mirror="XY")
2020-06-27 17:23:12 +02:00
x_offset = self.col_addr_dff_insts[port].lx()
else:
self.col_addr_pos[port] = vector(x_offset, y_offset)
2020-11-03 15:29:17 +01:00
2020-06-27 17:23:12 +02:00
if port in self.write_ports:
# Add spare write enable flops to the right of the data flops since the spare
# columns will be on the left
if self.num_spare_cols:
self.spare_wen_pos[port] = vector(x_offset - self.spare_wen_dff_insts[port].width,
y_offset)
self.spare_wen_dff_insts[port].place(self.spare_wen_pos[port], mirror="MX")
x_offset = self.spare_wen_dff_insts[port].lx()
2019-07-16 18:04:58 +02:00
2020-06-27 17:23:12 +02:00
if self.write_size:
# Add the write mask flops below the write mask AND array.
self.wmask_pos[port] = vector(x_offset - self.wmask_dff_insts[port].width,
y_offset)
self.wmask_dff_insts[port].place(self.wmask_pos[port], mirror="MX")
2020-06-27 17:23:12 +02:00
x_offset = self.wmask_dff_insts[port].lx()
# Add the data flops below the write mask flops.
self.data_pos[port] = vector(x_offset - self.data_dff_insts[port].width,
y_offset)
self.data_dff_insts[port].place(self.data_pos[port], mirror="MX")
2020-06-27 17:23:12 +02:00
else:
self.wmask_pos[port] = vector(x_offset, y_offset)
self.data_pos[port] = vector(x_offset, y_offset)
self.spare_wen_pos[port] = vector(x_offset, y_offset)
def add_layout_pins(self, add_vias=True):
"""
Add the top-level pins for a single bank SRAM with control.
"""
for port in self.all_ports:
2020-12-23 02:08:53 +01:00
# Hack: If we are escape routing, set the pin layer to
# None so that we will start from the pin layer
# Otherwise, set it as the pin layer so that no vias are added.
# Otherwise, when we remove pins to move the dff array dynamically,
# we will leave some remaining vias when the pin locations change.
if add_vias:
pin_layer = None
else:
pin_layer = self.pwr_grid_layer
# Connect the control pins as inputs
for signal in self.control_logic_inputs[port]:
2020-08-25 23:42:21 +02:00
if signal.startswith("rbl"):
continue
self.add_io_pin(self.control_logic_insts[port],
signal,
2020-12-23 02:08:53 +01:00
signal + "{}".format(port),
start_layer=pin_layer)
if port in self.write_ports:
for bit in range(self.word_size + self.num_spare_cols):
self.add_io_pin(self.data_dff_insts[port],
"din_{}".format(bit),
2020-12-23 02:08:53 +01:00
"din{0}[{1}]".format(port, bit),
start_layer=pin_layer)
2020-11-03 15:29:17 +01:00
if port in self.readwrite_ports or port in self.read_ports:
for bit in range(self.word_size + self.num_spare_cols):
self.add_io_pin(self.bank_inst,
"dout{0}_{1}".format(port, bit),
2020-12-23 02:08:53 +01:00
"dout{0}[{1}]".format(port, bit),
start_layer=pin_layer)
for bit in range(self.col_addr_size):
self.add_io_pin(self.col_addr_dff_insts[port],
"din_{}".format(bit),
2020-12-23 02:08:53 +01:00
"addr{0}[{1}]".format(port, bit),
start_layer=pin_layer)
for bit in range(self.row_addr_size):
self.add_io_pin(self.row_addr_dff_insts[port],
"din_{}".format(bit),
2020-12-23 02:08:53 +01:00
"addr{0}[{1}]".format(port, bit + self.col_addr_size),
start_layer=pin_layer)
2020-11-03 15:29:17 +01:00
if port in self.write_ports:
2019-08-23 00:02:52 +02:00
if self.write_size:
for bit in range(self.num_wmasks):
self.add_io_pin(self.wmask_dff_insts[port],
"din_{}".format(bit),
2020-12-23 02:08:53 +01:00
"wmask{0}[{1}]".format(port, bit),
start_layer=pin_layer)
if port in self.write_ports:
2020-05-14 12:30:29 +02:00
for bit in range(self.num_spare_cols):
self.add_io_pin(self.spare_wen_dff_insts[port],
"din_{}".format(bit),
2020-12-23 02:08:53 +01:00
"spare_wen{0}[{1}]".format(port, bit),
start_layer=pin_layer)
def route_layout(self):
""" Route a single bank SRAM """
2018-07-17 23:24:44 +02:00
self.route_clk()
2020-11-03 15:29:17 +01:00
2018-07-17 23:24:44 +02:00
self.route_control_logic()
2020-11-03 15:29:17 +01:00
2018-07-17 23:24:44 +02:00
self.route_row_addr_dff()
self.route_dffs()
# We add the vias to M3 before routing supplies because
# they might create some blockages
self.add_layout_pins()
# Route the pins to the perimeter
if OPTS.perimeter_pins:
self.route_escape_pins()
# Route the supplies first since the MST is not blockage aware
# and signals can route to anywhere on sides (it is flexible)
self.route_supplies()
def route_dffs(self, add_routes=True):
2020-11-03 15:29:17 +01:00
2020-06-27 17:23:12 +02:00
for port in self.all_ports:
self.route_dff(port, add_routes)
def route_dff(self, port, add_routes):
2020-06-27 17:23:12 +02:00
# This is only done when we add_routes because the data channel will be larger
# so that can be used for area estimation.
if add_routes:
self.route_col_addr_dffs(port)
self.route_data_dffs(port, add_routes)
def route_col_addr_dffs(self, port):
2020-06-27 17:23:12 +02:00
route_map = []
2020-11-03 15:29:17 +01:00
# column mux dff is routed on it's own since it is to the far end
# decoder inputs are min pitch M2, so need to use lower layer stack
2020-06-27 17:23:12 +02:00
if self.col_addr_size > 0:
dff_names = ["dout_{}".format(x) for x in range(self.col_addr_size)]
dff_pins = [self.col_addr_dff_insts[port].get_pin(x) for x in dff_names]
bank_names = ["addr{0}_{1}".format(port, x) for x in range(self.col_addr_size)]
bank_pins = [self.bank_inst.get_pin(x) for x in bank_names]
route_map.extend(list(zip(bank_pins, dff_pins)))
if len(route_map) > 0:
layer_stack = self.m1_stack
if port == 0:
offset = vector(self.control_logic_insts[port].rx() + self.dff.width,
- self.data_bus_size[port] + 2 * self.m3_pitch)
cr = channel_route(netlist=route_map,
offset=offset,
layer_stack=layer_stack,
parent=self)
# This causes problem in magic since it sometimes cannot extract connectivity of isntances
# with no active devices.
self.add_inst(cr.name, cr)
self.connect_inst([])
#self.add_flat_inst(cr.name, cr)
else:
offset = vector(0,
self.bank.height + self.m3_pitch)
cr = channel_route(netlist=route_map,
offset=offset,
layer_stack=layer_stack,
parent=self)
# This causes problem in magic since it sometimes cannot extract connectivity of isntances
# with no active devices.
self.add_inst(cr.name, cr)
self.connect_inst([])
#self.add_flat_inst(cr.name, cr)
def route_data_dffs(self, port, add_routes):
route_map = []
2020-06-27 17:23:12 +02:00
# wmask dff
if self.num_wmasks > 0 and port in self.write_ports:
dff_names = ["dout_{}".format(x) for x in range(self.num_wmasks)]
dff_pins = [self.wmask_dff_insts[port].get_pin(x) for x in dff_names]
bank_names = ["bank_wmask{0}_{1}".format(port, x) for x in range(self.num_wmasks)]
bank_pins = [self.bank_inst.get_pin(x) for x in bank_names]
route_map.extend(list(zip(bank_pins, dff_pins)))
if port in self.write_ports:
# synchronized inputs from data dff
2020-06-27 17:23:12 +02:00
dff_names = ["dout_{}".format(x) for x in range(self.word_size + self.num_spare_cols)]
dff_pins = [self.data_dff_insts[port].get_pin(x) for x in dff_names]
bank_names = ["din{0}_{1}".format(port, x) for x in range(self.word_size + self.num_spare_cols)]
bank_pins = [self.bank_inst.get_pin(x) for x in bank_names]
route_map.extend(list(zip(bank_pins, dff_pins)))
2020-11-03 15:29:17 +01:00
# spare wen dff
if self.num_spare_cols > 0 and port in self.write_ports:
dff_names = ["dout_{}".format(x) for x in range(self.num_spare_cols)]
dff_pins = [self.spare_wen_dff_insts[port].get_pin(x) for x in dff_names]
bank_names = ["bank_spare_wen{0}_{1}".format(port, x) for x in range(self.num_spare_cols)]
bank_pins = [self.bank_inst.get_pin(x) for x in bank_names]
route_map.extend(list(zip(bank_pins, dff_pins)))
2020-11-03 15:29:17 +01:00
2020-06-27 17:23:12 +02:00
if len(route_map) > 0:
2020-11-03 15:29:17 +01:00
# The write masks will have blockages on M1
if self.num_wmasks > 0 and port in self.write_ports:
layer_stack = self.m3_stack
else:
layer_stack = self.m1_stack
2020-11-03 15:29:17 +01:00
if port == 0:
offset = vector(self.control_logic_insts[port].rx() + self.dff.width,
- self.data_bus_size[port] + 2 * self.m3_pitch)
cr = channel_route(netlist=route_map,
offset=offset,
layer_stack=layer_stack,
parent=self)
if add_routes:
# This causes problem in magic since it sometimes cannot extract connectivity of isntances
# with no active devices.
self.add_inst(cr.name, cr)
self.connect_inst([])
#self.add_flat_inst(cr.name, cr)
else:
self.data_bus_size[port] = max(cr.height, self.col_addr_bus_size[port]) + self.data_bus_gap
else:
offset = vector(0,
self.bank.height + self.m3_pitch)
cr = channel_route(netlist=route_map,
offset=offset,
layer_stack=layer_stack,
parent=self)
if add_routes:
# This causes problem in magic since it sometimes cannot extract connectivity of isntances
# with no active devices.
self.add_inst(cr.name, cr)
self.connect_inst([])
#self.add_flat_inst(cr.name, cr)
else:
self.data_bus_size[port] = max(cr.height, self.col_addr_bus_size[port]) + self.data_bus_gap
2020-11-03 15:29:17 +01:00
2018-07-17 23:24:44 +02:00
def route_clk(self):
""" Route the clock network """
# This is the actual input to the SRAM
for port in self.all_ports:
# Connect all of these clock pins to the clock in the central bus
# This is something like a "spine" clock distribution. The two spines
# are clk_buf and clk_buf_bar
2018-11-29 00:30:52 +01:00
control_clk_buf_pin = self.control_logic_insts[port].get_pin("clk_buf")
control_clk_buf_pos = control_clk_buf_pin.center()
2020-11-03 15:29:17 +01:00
# This uses a metal2 track to the right (for port0) of the control/row addr DFF
# to route vertically. For port1, it is to the left.
2018-11-14 01:05:22 +01:00
row_addr_clk_pin = self.row_addr_dff_insts[port].get_pin("clk")
if port % 2:
control_clk_buf_pos = control_clk_buf_pin.lc()
row_addr_clk_pos = row_addr_clk_pin.lc()
mid1_pos = vector(self.row_addr_dff_insts[port].lx() - self.m2_pitch,
row_addr_clk_pos.y)
else:
control_clk_buf_pos = control_clk_buf_pin.rc()
row_addr_clk_pos = row_addr_clk_pin.rc()
mid1_pos = vector(self.row_addr_dff_insts[port].rx() + self.m2_pitch,
row_addr_clk_pos.y)
2018-11-29 00:30:52 +01:00
# This is the steiner point where the net branches out
clk_steiner_pos = vector(mid1_pos.x, control_clk_buf_pos.y)
self.add_path(control_clk_buf_pin.layer, [control_clk_buf_pos, clk_steiner_pos])
self.add_via_stack_center(from_layer=control_clk_buf_pin.layer,
to_layer="m2",
offset=clk_steiner_pos)
2020-11-03 15:29:17 +01:00
# Note, the via to the control logic is taken care of above
self.add_wire(self.m2_stack[::-1],
2020-03-06 01:13:49 +01:00
[row_addr_clk_pos, mid1_pos, clk_steiner_pos])
2020-11-03 15:29:17 +01:00
2018-11-29 00:30:52 +01:00
if self.col_addr_dff:
dff_clk_pin = self.col_addr_dff_insts[port].get_pin("clk")
dff_clk_pos = dff_clk_pin.center()
mid_pos = vector(clk_steiner_pos.x, dff_clk_pos.y)
self.add_wire(self.m2_stack[::-1],
2020-03-06 01:13:49 +01:00
[dff_clk_pos, mid_pos, clk_steiner_pos])
2020-06-27 17:23:12 +02:00
elif port in self.write_ports:
2018-11-29 00:30:52 +01:00
data_dff_clk_pin = self.data_dff_insts[port].get_pin("clk")
data_dff_clk_pos = data_dff_clk_pin.center()
mid_pos = vector(clk_steiner_pos.x, data_dff_clk_pos.y)
# In some designs, the steiner via will be too close to the mid_pos via
# so make the wire as wide as the contacts
2020-03-06 01:13:49 +01:00
self.add_path("m2",
[mid_pos, clk_steiner_pos],
width=max(m2_via.width, m2_via.height))
self.add_wire(self.m2_stack[::-1],
2020-03-06 01:13:49 +01:00
[data_dff_clk_pos, mid_pos, clk_steiner_pos])
2018-07-17 23:24:44 +02:00
def route_control_logic(self):
"""
Route the control logic pins that are not inputs
"""
2019-07-16 18:04:58 +02:00
for port in self.all_ports:
for signal in self.control_logic_outputs[port]:
2018-11-29 00:30:52 +01:00
# The clock gets routed separately and is not a part of the bank
if "clk" in signal:
continue
2018-11-14 01:05:22 +01:00
src_pin = self.control_logic_insts[port].get_pin(signal)
2020-03-06 01:13:49 +01:00
dest_pin = self.bank_inst.get_pin(signal + "{}".format(port))
self.connect_vbus(src_pin, dest_pin)
2019-07-16 18:04:58 +02:00
2019-08-10 21:53:07 +02:00
for port in self.all_ports:
2019-07-16 18:04:58 +02:00
# Only input (besides pins) is the replica bitline
src_pin = self.control_logic_insts[port].get_pin("rbl_bl")
2020-08-25 20:50:44 +02:00
dest_pin = self.bank_inst.get_pin("rbl_bl_{0}_{0}".format(port))
2021-05-05 01:42:42 +02:00
self.add_wire(self.m3_stack,
2020-06-04 20:03:39 +02:00
[src_pin.center(), vector(src_pin.cx(), dest_pin.cy()), dest_pin.rc()])
self.add_via_stack_center(from_layer=src_pin.layer,
2021-05-05 01:42:42 +02:00
to_layer="m4",
offset=src_pin.center())
self.add_via_stack_center(from_layer=dest_pin.layer,
2021-05-05 01:42:42 +02:00
to_layer="m3",
offset=dest_pin.center())
2020-11-03 15:29:17 +01:00
2018-07-17 23:24:44 +02:00
def route_row_addr_dff(self):
2020-11-03 15:29:17 +01:00
"""
Connect the output of the row flops to the bank pins
"""
for port in self.all_ports:
for bit in range(self.row_addr_size):
flop_name = "dout_{}".format(bit)
2020-03-06 01:13:49 +01:00
bank_name = "addr{0}_{1}".format(port, bit + self.col_addr_size)
2018-11-14 01:05:22 +01:00
flop_pin = self.row_addr_dff_insts[port].get_pin(flop_name)
bank_pin = self.bank_inst.get_pin(bank_name)
flop_pos = flop_pin.center()
bank_pos = bank_pin.center()
2020-03-06 01:13:49 +01:00
mid_pos = vector(bank_pos.x, flop_pos.y)
self.add_via_stack_center(from_layer=flop_pin.layer,
to_layer="m3",
offset=flop_pos)
2020-06-26 00:34:18 +02:00
self.add_path("m3", [flop_pos, mid_pos])
self.add_via_stack_center(from_layer=bank_pin.layer,
to_layer="m3",
offset=mid_pos)
self.add_path(bank_pin.layer, [mid_pos, bank_pos])
def add_lvs_correspondence_points(self):
2020-03-06 01:13:49 +01:00
"""
This adds some points for easier debugging if LVS goes wrong.
These should probably be turned off by default though, since extraction
will show these as ports in the extracted netlist.
"""
2020-08-25 23:42:21 +02:00
return
for n in self.control_logic_outputs[0]:
2018-11-14 01:05:22 +01:00
pin = self.control_logic_insts[0].get_pin(n)
self.add_label(text=n,
layer=pin.layer,
offset=pin.center())
2020-11-03 15:29:17 +01:00
def graph_exclude_data_dff(self):
"""
Removes data dff and wmask dff (if applicable) from search graph.
"""
2020-03-06 01:13:49 +01:00
# Data dffs and wmask dffs are only for writing so are not useful for evaluating read delay.
for inst in self.data_dff_insts:
self.graph_inst_exclude.add(inst)
if self.write_size:
for inst in self.wmask_dff_insts:
self.graph_inst_exclude.add(inst)
if self.num_spare_cols:
for inst in self.spare_wen_dff_insts:
self.graph_inst_exclude.add(inst)
2020-11-03 15:29:17 +01:00
def graph_exclude_addr_dff(self):
"""
Removes data dff from search graph.
"""
2020-03-06 01:13:49 +01:00
# Address is considered not part of the critical path, subjectively removed
for inst in self.row_addr_dff_insts:
self.graph_inst_exclude.add(inst)
2020-11-03 15:29:17 +01:00
if self.col_addr_dff:
for inst in self.col_addr_dff_insts:
self.graph_inst_exclude.add(inst)
def graph_exclude_ctrl_dffs(self):
"""
Exclude dffs for CSB, WEB, etc from graph
"""
2020-03-06 01:13:49 +01:00
# Insts located in control logic, exclusion function called here
for inst in self.control_logic_insts:
inst.mod.graph_exclude_dffs()
2020-11-03 15:29:17 +01:00
2019-05-21 03:35:52 +02:00
def get_cell_name(self, inst_name, row, col):
"""
Gets the spice name of the target bitcell.
"""
2020-03-06 01:13:49 +01:00
# Sanity check in case it was forgotten
if inst_name.find("x") != 0:
inst_name = "x" + inst_name
return self.bank_inst.mod.get_cell_name(inst_name + ".x" + self.bank_inst.name, row, col)
2020-11-03 15:29:17 +01:00
def get_bank_num(self, inst_name, row, col):
return 0