OpenRAM/compiler/modules/write_driver_array.py

272 lines
11 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.
#
2016-11-08 18:57:35 +01:00
import design
import debug
from tech import drc
2019-01-17 01:15:38 +01:00
from sram_factory import factory
2016-11-08 18:57:35 +01:00
from vector import vector
from globals import OPTS
2020-03-31 19:15:46 +02:00
2016-11-08 18:57:35 +01:00
class write_driver_array(design.design):
"""
Array of tristate drivers to write to the bitlines through the column mux.
Dynamically generated write driver array of all bitlines.
"""
2020-06-07 18:27:25 +02:00
def __init__(self, name, columns, word_size, num_spare_cols=None, write_size=None, column_offset=0):
2019-01-17 01:15:38 +01:00
design.design.__init__(self, name)
2016-11-08 18:57:35 +01:00
debug.info(1, "Creating {0}".format(self.name))
self.add_comment("columns: {0}".format(columns))
self.add_comment("word_size {0}".format(word_size))
2016-11-08 18:57:35 +01:00
self.columns = columns
self.word_size = word_size
2019-07-19 22:17:55 +02:00
self.write_size = write_size
self.column_offset = column_offset
self.words_per_row = int(columns / word_size)
2020-04-14 05:09:10 +02:00
if not num_spare_cols:
self.num_spare_cols = 0
else:
self.num_spare_cols = num_spare_cols
2016-11-08 18:57:35 +01:00
if self.write_size:
2020-03-31 19:15:46 +02:00
self.num_wmasks = int(self.word_size / self.write_size)
self.create_netlist()
if not OPTS.netlist_only:
self.create_layout()
def get_bl_name(self):
bl_name = "bl"
return bl_name
def get_br_name(self):
br_name = "br"
return br_name
@property
def data_name(self):
return "data"
@property
def en_name(self):
return "en"
def create_netlist(self):
self.add_modules()
self.add_pins()
self.create_write_array()
def create_layout(self):
if self.bitcell.width > self.driver.width:
2020-04-14 05:09:10 +02:00
self.width = (self.columns + self.num_spare_cols) * self.bitcell.width
self.width_regular_cols = self.columns * self.bitcell.width
self.single_col_width = self.bitcell.width
else:
2020-04-14 05:09:10 +02:00
self.width = (self.columns + self.num_spare_cols) * self.driver.width
self.width_regular_cols = self.columns * self.driver.width
self.single_col_width = self.driver.width
self.height = self.driver.height
self.place_write_array()
self.add_layout_pins()
self.add_boundary()
self.DRC_LVS()
2016-11-08 18:57:35 +01:00
def add_pins(self):
2020-04-14 05:09:10 +02:00
for i in range(self.word_size + self.num_spare_cols):
self.add_pin(self.data_name + "_{0}".format(i), "INPUT")
2020-04-14 05:09:10 +02:00
for i in range(self.word_size + self.num_spare_cols):
self.add_pin(self.get_bl_name() + "_{0}".format(i), "OUTPUT")
self.add_pin(self.get_br_name() + "_{0}".format(i), "OUTPUT")
2019-08-23 00:02:52 +02:00
if self.write_size:
2020-05-16 12:09:03 +02:00
for i in range(self.num_wmasks + self.num_spare_cols):
self.add_pin(self.en_name + "_{0}".format(i), "INPUT")
2020-05-16 12:09:03 +02:00
elif self.num_spare_cols and not self.write_size:
2020-04-14 05:09:10 +02:00
for i in range(self.num_spare_cols + 1):
self.add_pin(self.en_name + "_{0}".format(i), "INPUT")
2019-07-19 22:17:55 +02:00
else:
self.add_pin(self.en_name, "INPUT")
2019-08-06 23:14:09 +02:00
self.add_pin("vdd", "POWER")
self.add_pin("gnd", "GROUND")
2016-11-08 18:57:35 +01:00
def add_modules(self):
2019-01-17 01:15:38 +01:00
self.driver = factory.create(module_type="write_driver")
self.add_mod(self.driver)
# This is just used for measurements,
# so don't add the module
2019-01-17 01:15:38 +01:00
self.bitcell = factory.create(module_type="bitcell")
2016-11-08 18:57:35 +01:00
def create_write_array(self):
self.driver_insts = {}
w = 0
windex=0
2020-03-31 19:15:46 +02:00
for i in range(0, self.columns, self.words_per_row):
2018-11-27 21:02:53 +01:00
name = "write_driver{}".format(i)
2020-03-31 19:15:46 +02:00
index = int(i / self.words_per_row)
self.driver_insts[index]=self.add_inst(name=name,
2018-08-27 20:13:34 +02:00
mod=self.driver)
if self.write_size:
self.connect_inst([self.data_name + "_{0}".format(index),
self.get_bl_name() + "_{0}".format(index),
self.get_br_name() + "_{0}".format(index),
self.en_name + "_{0}".format(windex), "vdd", "gnd"])
w+=1
# when w equals write size, the next en pin can be connected since we are now at the next wmask bit
if w == self.write_size:
w = 0
windex+=1
2020-04-14 05:09:10 +02:00
2020-05-16 12:09:03 +02:00
elif self.num_spare_cols and not self.write_size:
2020-04-14 05:09:10 +02:00
self.connect_inst([self.data_name + "_{0}".format(index),
self.get_bl_name() + "_{0}".format(index),
self.get_br_name() + "_{0}".format(index),
self.en_name + "_{0}".format(0), "vdd", "gnd"])
2019-07-19 22:17:55 +02:00
else:
self.connect_inst([self.data_name + "_{0}".format(index),
self.get_bl_name() + "_{0}".format(index),
self.get_br_name() + "_{0}".format(index),
self.en_name, "vdd", "gnd"])
2016-11-08 18:57:35 +01:00
2020-04-14 05:09:10 +02:00
for i in range(self.num_spare_cols):
index = self.word_size + i
2020-05-16 12:09:03 +02:00
if self.write_size:
offset = self.num_wmasks
else:
offset = 1
name = "write_driver{}".format(self.columns + i)
2020-04-14 05:09:10 +02:00
self.driver_insts[index]=self.add_inst(name=name,
mod=self.driver)
self.connect_inst([self.data_name + "_{0}".format(index),
self.get_bl_name() + "_{0}".format(index),
self.get_br_name() + "_{0}".format(index),
2020-05-16 12:09:03 +02:00
self.en_name + "_{0}".format(i + offset), "vdd", "gnd"])
2020-04-14 05:09:10 +02:00
2018-08-27 20:13:34 +02:00
def place_write_array(self):
from tech import cell_properties
if self.bitcell.width > self.driver.width:
self.driver_spacing = self.bitcell.width
else:
self.driver_spacing = self.driver.width
2020-03-31 19:15:46 +02:00
for i in range(0, self.columns, self.words_per_row):
index = int(i / self.words_per_row)
xoffset = i * self.driver_spacing
if cell_properties.bitcell.mirror.y and (i + self.column_offset) % 2:
mirror = "MY"
xoffset = xoffset + self.driver.width
else:
mirror = ""
base = vector(xoffset, 0)
self.driver_insts[index].place(offset=base, mirror=mirror)
2018-08-27 20:13:34 +02:00
2020-04-14 05:09:10 +02:00
# place spare write drivers (if spare columns are specified)
for i in range(self.num_spare_cols):
index = self.word_size + i
xoffset = (self.columns + i) * self.driver_spacing
2020-06-07 18:27:25 +02:00
if cell_properties.bitcell.mirror.y and (i + self.column_offset) % 2:
2020-04-14 05:09:10 +02:00
mirror = "MY"
xoffset = xoffset + self.driver.width
else:
mirror = ""
base = vector(xoffset, 0)
self.driver_insts[index].place(offset=base, mirror=mirror)
def add_layout_pins(self):
2020-04-14 05:09:10 +02:00
for i in range(self.word_size + self.num_spare_cols):
inst = self.driver_insts[i]
din_pin = inst.get_pin(inst.mod.din_name)
self.add_layout_pin(text=self.data_name + "_{0}".format(i),
layer=din_pin.layer,
offset=din_pin.ll(),
width=din_pin.width(),
height=din_pin.height())
bl_pin = inst.get_pin(inst.mod.get_bl_names())
self.add_layout_pin(text=self.get_bl_name() + "_{0}".format(i),
layer=bl_pin.layer,
offset=bl_pin.ll(),
width=bl_pin.width(),
height=bl_pin.height())
br_pin = inst.get_pin(inst.mod.get_br_names())
self.add_layout_pin(text=self.get_br_name() + "_{0}".format(i),
layer=br_pin.layer,
offset=br_pin.ll(),
width=br_pin.width(),
height=br_pin.height())
2018-04-11 18:29:54 +02:00
for n in ["vdd", "gnd"]:
2018-04-17 01:15:35 +02:00
pin_list = self.driver_insts[i].get_pins(n)
for pin in pin_list:
2020-03-31 19:15:46 +02:00
self.add_power_pin(name=n,
loc=pin.center(),
directions=("V", "V"),
start_layer=pin.layer)
2019-08-23 00:02:52 +02:00
if self.write_size:
for bit in range(self.num_wmasks):
2020-03-31 19:15:46 +02:00
inst = self.driver_insts[bit * self.write_size]
en_pin = inst.get_pin(inst.mod.en_name)
# Determine width of wmask modified en_pin with/without col mux
2020-03-31 19:15:46 +02:00
wmask_en_len = self.words_per_row * (self.write_size * self.driver_spacing)
if (self.words_per_row == 1):
en_gap = self.driver_spacing - en_pin.width()
else:
en_gap = self.driver_spacing
self.add_layout_pin(text=self.en_name + "_{0}".format(bit),
2019-08-08 17:46:58 +02:00
layer=en_pin.layer,
offset=en_pin.ll(),
2020-03-31 19:15:46 +02:00
width=wmask_en_len - en_gap,
2019-08-08 17:46:58 +02:00
height=en_pin.height())
2020-05-16 12:09:03 +02:00
for i in range(self.num_spare_cols):
inst = self.driver_insts[self.word_size + i]
en_pin = inst.get_pin(inst.mod.en_name)
2020-05-16 12:09:03 +02:00
self.add_layout_pin(text=self.en_name + "_{0}".format(i + self.num_wmasks),
layer="m1",
offset=en_pin.lr() + vector(-drc("minwidth_m1"),0))
2020-04-14 05:09:10 +02:00
2020-05-16 12:09:03 +02:00
elif self.num_spare_cols and not self.write_size:
2020-04-14 05:09:10 +02:00
# shorten enable rail to accomodate those for spare write drivers
inst = self.driver_insts[0]
en_pin = inst.get_pin(inst.mod.en_name)
2020-04-14 05:09:10 +02:00
self.add_layout_pin(text=self.en_name + "_{0}".format(0),
layer="m1",
offset=en_pin.ll(),
width=self.width_regular_cols - self.words_per_row * en_pin.width())
2020-04-14 05:09:10 +02:00
# individual enables for every spare write driver
for i in range(self.num_spare_cols):
inst = self.driver_insts[self.word_size + i]
en_pin = inst.get_pin(inst.mod.en_name)
2020-04-14 05:09:10 +02:00
self.add_layout_pin(text=self.en_name + "_{0}".format(i + 1),
layer="m1",
offset=en_pin.lr() + vector(-drc("minwidth_m1"),0))
else:
inst = self.driver_insts[0]
self.add_layout_pin(text=self.en_name,
layer="m1",
2020-03-31 19:15:46 +02:00
offset=inst.get_pin(inst.mod.en_name).ll().scale(0, 1),
width=self.width)
2016-11-08 18:57:35 +01:00
def get_w_en_cin(self):
"""Get the relative capacitance of all the enable connections in the bank"""
2020-03-31 19:15:46 +02:00
# The enable is connected to a nand2 for every row.
return self.driver.get_w_en_cin() * len(self.driver_insts)