OpenRAM/compiler/sram_factory.py

103 lines
3.8 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-01-17 02:04:28 +01:00
from globals import OPTS
2019-10-03 01:26:02 +02:00
2019-01-17 02:04:28 +01:00
class sram_factory:
"""
This is a factory pattern to create modules for usage in an SRAM.
Since GDSII has a flat namespace, it requires modules to have unique
names if their layout differs. This module ensures that any module
with different layouts will have different names. It also ensures that
identical layouts will share the same name to reduce file size and promote
hierarchical sharing.
"""
def __init__(self):
2019-10-03 01:26:02 +02:00
# A dictionary of modules indexed by module type
2019-01-17 02:04:28 +01:00
self.modules = {}
# These are the indices to append to name to make unique object names
self.module_indices = {}
# A dictionary of instance lists indexed by module type
self.objects = {}
def reset(self):
"""
Clear the factory instances for testing.
"""
self.__init__()
def create(self, module_type, **kwargs):
"""
2019-10-03 01:26:02 +02:00
A generic function to create a module with a given module_type.
The args are passed directly to the module constructor.
2019-01-17 02:04:28 +01:00
"""
# if name!="":
# # This is a special case where the name and type don't match
# # Can't be overridden in the config file
# module_name = name
if hasattr(OPTS, module_type):
# Retrieve the name from OPTS if it exists,
# otherwise just use the name
module_type = getattr(OPTS, module_type)
2019-01-17 02:04:28 +01:00
# Either retrieve the already loaded module or load it
try:
mod = self.modules[module_type]
except KeyError:
2019-05-31 17:43:37 +02:00
import importlib
2019-06-26 01:55:50 +02:00
c = importlib.reload(__import__(module_type))
mod = getattr(c, module_type)
2019-01-17 02:04:28 +01:00
self.modules[module_type] = mod
self.module_indices[module_type] = 0
self.objects[module_type] = []
# Either retreive a previous object or create a new one
for obj in self.objects[module_type]:
(obj_kwargs, obj_item) = obj
# Must have the same dictionary exactly (conservative)
if obj_kwargs == kwargs:
return obj_item
# Use the default name if there are default arguments
2019-10-03 01:26:02 +02:00
# This is especially for library cells so that the
# spice and gds files can be found.
if len(kwargs) > 0:
2019-01-17 02:04:28 +01:00
# Create a unique name and increment the index
2019-10-03 01:26:02 +02:00
module_name = "{0}_{1}".format(module_type,
self.module_indices[module_type])
2019-01-17 02:04:28 +01:00
self.module_indices[module_type] += 1
else:
module_name = module_type
2019-10-03 01:26:02 +02:00
# type_str = "type={}".format(module_type)
# name_str = "name={}".format(module_name)
# kwargs_str = "kwargs={}".format(str(kwargs))
# import debug
# debug.info(0, "New module:" + type_str + name_str + kwargs_str)
obj = mod(name=module_name, **kwargs)
self.objects[module_type].append((kwargs, obj))
2019-01-17 02:04:28 +01:00
return obj
def get_mods(self, module_type):
"""Returns list of all objects of module name's type."""
if hasattr(OPTS, module_type):
# Retrieve the name from OPTS if it exists,
# otherwise just use the input
module_type = getattr(OPTS, module_type)
try:
mod_tuples = self.objects[module_type]
2019-10-03 01:26:02 +02:00
mods = [mod for kwargs, mod in mod_tuples]
except KeyError:
mods = []
return mods
2019-10-03 01:26:02 +02:00
2019-01-17 02:04:28 +01:00
# Make a factory
factory = sram_factory()