diff options
author | Sugandha Gupta <sugandha.gupta@ettus.com> | 2019-10-17 18:27:34 -0700 |
---|---|---|
committer | Martin Braun <martin.braun@ettus.com> | 2019-11-26 12:21:32 -0800 |
commit | 3e2cceeaed7c3a2a70318dab695d6adb9ee5602e (patch) | |
tree | 8967ee5d8470405630fc408d0d62156aaf01c781 | |
parent | 2da39536bb886f3f80858f531f86b95988e84bb3 (diff) | |
download | uhd-3e2cceeaed7c3a2a70318dab695d6adb9ee5602e.tar.gz uhd-3e2cceeaed7c3a2a70318dab695d6adb9ee5602e.tar.bz2 uhd-3e2cceeaed7c3a2a70318dab695d6adb9ee5602e.zip |
utils: image_builder: Fix ordering of noc_blocks, ports
The ports in the fpga need to be ordered to make correct
connections in verilog. This also keeps generated verilog
constant across runs.
-rwxr-xr-x | host/utils/rfnoc/image_builder.py | 31 |
1 files changed, 25 insertions, 6 deletions
diff --git a/host/utils/rfnoc/image_builder.py b/host/utils/rfnoc/image_builder.py index df3cfc3f1..92d8a909d 100755 --- a/host/utils/rfnoc/image_builder.py +++ b/host/utils/rfnoc/image_builder.py @@ -20,6 +20,24 @@ import mako.lookup import mako.template from ruamel import yaml +# Adapted from code found at +# https://stackoverflow.com/questions/5121931/ +# in-python-how-can-you-load-yaml-mappings-as-ordereddicts +# (Accessed 17 October 2019) +def ordered_load(stream, Loader=yaml.SafeLoader, object_pairs_hook=OrderedDict): + """ + In Python 3.5, element insertion order into dictionaries is not preserved. + This function uses an OrderedDict to read a YAML file, which does preserve order. + """ + class OrderedLoader(Loader): + pass + def construct_mapping(loader, node): + loader.flatten_mapping(node) + return object_pairs_hook(loader.construct_pairs(node)) + OrderedLoader.add_constructor( + yaml.resolver.BaseResolver.DEFAULT_MAPPING_TAG, + construct_mapping) + return yaml.load(stream, OrderedLoader) def split(iterable, function): """ @@ -90,6 +108,7 @@ class IOConfig: """ def __init__(self, config, signatures): # read configuration from config dictionary + # TODO: Is this guaranteed ordered? self.__dict__.update(**config) if hasattr(self, "io_ports"): expand_io_port_desc(getattr(self, "io_ports"), signatures) @@ -109,7 +128,7 @@ class ImageBuilderConfig: self.stream_endpoints = OrderedDict() self.connections = [] self.clk_domains = [] - self.block_ports = {} + self.block_ports = OrderedDict() self.io_ports = OrderedDict() self.clocks = OrderedDict() self.block_con = [] @@ -168,7 +187,7 @@ class ImageBuilderConfig: if not hasattr(desc, "parameters"): setattr(desc, "parameters", {}) if "parameters" not in block: - block["parameters"] = {} + block["parameters"] = OrderedDict() for key in block["parameters"].keys(): if key not in desc.parameters: logging.error("Unknown parameter %s for block %s", key, name) @@ -318,7 +337,7 @@ def load_config(filename): with open(filename) as stream: logging.info( "Using %s from %s.", basename, os.path.normpath(dirname)) - config = yaml.safe_load(stream) + config = ordered_load(stream) return config except IOError: logging.error("%s misses %s", os.path.normpath(dirname), basename) @@ -359,7 +378,7 @@ def read_grc_block_configs(path): for name in names: if re.match(r".*\.block\.yml", name): with open (os.path.join(root, name)) as stream: - config = yaml.safe_load(stream) + config = ordered_load(stream) result[config["id"]] = config return result @@ -463,13 +482,13 @@ def read_block_descriptions(signatures, *paths): :return: dictionary of noc blocks. Key is filename of the block, value is an IOConfig object """ - blocks = {} + blocks = OrderedDict() for path in paths: for root, dirs, files, in os.walk(path): for filename in files: if re.match(r".*\.yml$", filename): with open(os.path.join(root, filename)) as stream: - block = yaml.safe_load(stream) + block = ordered_load(stream) if "schema" in block and \ block["schema"] == "rfnoc_modtool_args": logging.info("Adding block description from " |