| # Copyright lowRISC contributors. | 
 | # Licensed under the Apache License, Version 2.0, see LICENSE for details. | 
 | # SPDX-License-Identifier: Apache-2.0 | 
 |  | 
 | '''A wrapper for loading hjson files as used by dvsim's FlowCfg''' | 
 |  | 
 | from utils import parse_hjson, subst_wildcards | 
 |  | 
 |  | 
 | # A set of fields that can be overridden on the command line and shouldn't be | 
 | # loaded from the hjson in that case. | 
 | _CMDLINE_FIELDS = {'tool'} | 
 |  | 
 |  | 
 | def load_hjson(path, initial_values): | 
 |     '''Load an hjson file and any includes | 
 |  | 
 |     Combines them all into a single dictionary, which is then returned. This | 
 |     does wildcard substitution on include names (since it might be needed to | 
 |     find included files), but not otherwise. | 
 |  | 
 |     initial_values is a starting point for the dictionary to be returned (which | 
 |     is not modified). It needs to contain values for anything needed to resolve | 
 |     include files (typically, this is 'proj_root' and 'tool' (if set)). | 
 |  | 
 |     ''' | 
 |     worklist = [path] | 
 |     seen = {path} | 
 |     ret = initial_values.copy() | 
 |     is_first = True | 
 |  | 
 |     # Figure out a list of fields that had a value from the command line. These | 
 |     # should have been passed in as part of initial_values and we need to know | 
 |     # that we can safely ignore updates. | 
 |     arg_keys = _CMDLINE_FIELDS & initial_values.keys() | 
 |  | 
 |     while worklist: | 
 |         next_path = worklist.pop() | 
 |         new_paths = _load_single_file(ret, next_path, is_first, arg_keys) | 
 |         paths_seen = set(new_paths) & seen | 
 |         if paths_seen: | 
 |             raise RuntimeError('The files {!r} appears more than once ' | 
 |                                'when processing include {!r} for {!r}.' | 
 |                                .format(list(paths_seen), next_path, path)) | 
 |         seen |= set(new_paths) | 
 |         worklist += new_paths | 
 |         is_first = False | 
 |  | 
 |     return ret | 
 |  | 
 |  | 
 | def _load_single_file(target, path, is_first, arg_keys): | 
 |     '''Load a single hjson file, merging its keys into target | 
 |  | 
 |     Returns a list of further includes that should be loaded. | 
 |  | 
 |     ''' | 
 |     hjson = parse_hjson(path) | 
 |     if not isinstance(hjson, dict): | 
 |         raise RuntimeError('{!r}: Top-level hjson object is not a dictionary.' | 
 |                            .format(path)) | 
 |  | 
 |     import_cfgs = [] | 
 |     for key, dict_val in hjson.items(): | 
 |         # If this key got set at the start of time and we want to ignore any | 
 |         # updates: ignore them! | 
 |         if key in arg_keys: | 
 |             continue | 
 |  | 
 |         # If key is 'import_cfgs', this should be a list. Add each item to the | 
 |         # list of cfgs to process | 
 |         if key == 'import_cfgs': | 
 |             if not isinstance(dict_val, list): | 
 |                 raise RuntimeError('{!r}: import_cfgs value is {!r}, but ' | 
 |                                    'should be a list.' | 
 |                                    .format(path, dict_val)) | 
 |             import_cfgs += dict_val | 
 |             continue | 
 |  | 
 |         # 'use_cfgs' is a bit like 'import_cfgs', but is only used for primary | 
 |         # config files (where it is a list of the child configs). This | 
 |         # shouldn't be used except at top-level (the first configuration file | 
 |         # to be loaded). | 
 |         # | 
 |         # If defined, check that it's a list, but then allow it to be set in | 
 |         # the target dictionary as usual. | 
 |         if key == 'use_cfgs': | 
 |             if not is_first: | 
 |                 raise RuntimeError('{!r}: File is included by another one, ' | 
 |                                    'but defines "use_cfgs".' | 
 |                                    .format(path)) | 
 |             if not isinstance(dict_val, list): | 
 |                 raise RuntimeError('{!r}: use_cfgs must be a list. Saw {!r}.' | 
 |                                    .format(path, dict_val)) | 
 |  | 
 |         # Otherwise, update target with this attribute | 
 |         set_target_attribute(path, target, key, dict_val) | 
 |  | 
 |     # Expand the names of imported configuration files as we return them | 
 |     return [subst_wildcards(cfg_path, | 
 |                             target, | 
 |                             ignored_wildcards=[], | 
 |                             ignore_error=False) | 
 |             for cfg_path in import_cfgs] | 
 |  | 
 |  | 
 | def set_target_attribute(path, target, key, dict_val): | 
 |     '''Set an attribute on the target dictionary | 
 |  | 
 |     This performs checks for conflicting values and merges lists / | 
 |     dictionaries. | 
 |  | 
 |     ''' | 
 |     old_val = target.get(key) | 
 |     if old_val is None: | 
 |         # A new attribute (or the old value was None, in which case it's | 
 |         # just a placeholder and needs writing). Set it and return. | 
 |         target[key] = dict_val | 
 |         return | 
 |  | 
 |     if isinstance(old_val, list): | 
 |         if not isinstance(dict_val, list): | 
 |             raise RuntimeError('{!r}: Conflicting types for key {!r}: was ' | 
 |                                '{!r}, a list, but loaded value is {!r}, ' | 
 |                                'of type {}.' | 
 |                                .format(path, key, old_val, dict_val, | 
 |                                        type(dict_val).__name__)) | 
 |  | 
 |         # Lists are merged by concatenation | 
 |         target[key] += dict_val | 
 |         return | 
 |  | 
 |     # The other types we support are "scalar" types. | 
 |     scalar_types = [(str, [""]), (int, [0, -1]), (bool, [False])] | 
 |     defaults = None | 
 |     for st_type, st_defaults in scalar_types: | 
 |         if isinstance(dict_val, st_type): | 
 |             defaults = st_defaults | 
 |             break | 
 |     if defaults is None: | 
 |         raise RuntimeError('{!r}: Value for key {!r} is {!r}, of ' | 
 |                            'unknown type {}.' | 
 |                            .format(path, key, dict_val, | 
 |                                    type(dict_val).__name__)) | 
 |     if not isinstance(old_val, st_type): | 
 |         raise RuntimeError('{!r}: Value for key {!r} is {!r}, but ' | 
 |                            'we already had the value {!r}, of an ' | 
 |                            'incompatible type.' | 
 |                            .format(path, key, dict_val, old_val)) | 
 |  | 
 |     # The types are compatible. If the values are equal, there's nothing more | 
 |     # to do | 
 |     if old_val == dict_val: | 
 |         return | 
 |  | 
 |     old_is_default = old_val in defaults | 
 |     new_is_default = dict_val in defaults | 
 |  | 
 |     # Similarly, if new value looks like a default, ignore it (regardless | 
 |     # of whether the current value looks like a default). | 
 |     if new_is_default: | 
 |         return | 
 |  | 
 |     # If the existing value looks like a default and the new value doesn't, | 
 |     # take the new value. | 
 |     if old_is_default: | 
 |         target[key] = dict_val | 
 |         return | 
 |  | 
 |     # Neither value looks like a default. Raise an error. | 
 |     raise RuntimeError('{!r}: Value for key {!r} is {!r}, but ' | 
 |                        'we already had a conflicting value of {!r}.' | 
 |                        .format(path, key, dict_val, old_val)) |