blob: 18822aa7262983810e53f8240385136484ddfc35 [file] [log] [blame]
# Copyright 2023 Google LLC
# Copyright lowRISC contributors
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
import logging as log
import sys
import os
from CfgJson import load_hjson
import FormalCfg
import CdcCfg
import RdcCfg
import LintCfg
import dvsim_matcha.SimCfg as SimCfg
import SynCfg
def _load_cfg(path, initial_values):
'''Worker function for make_cfg.
initial_values is passed to load_hjson (see documentation there).
Returns a pair (cls, hjson_data) on success or raises a RuntimeError on
failure.
'''
# Set the `self_dir` template variable to the path of the currently
# processed Hjson file.
assert 'self_dir' in initial_values
initial_values['self_dir'] = os.path.dirname(path)
# Start by loading up the hjson file and any included files
hjson_data = load_hjson(path, initial_values)
# Look up the value of flow in the loaded data. This is a required field,
# and tells us what sort of FlowCfg to make.
flow = hjson_data.get('flow')
if flow is None:
raise RuntimeError('{!r}: No value for the "flow" key. Are you sure '
'this is a dvsim configuration file?'
.format(path))
classes = [
RdcCfg.RdcCfg,
CdcCfg.CdcCfg,
LintCfg.LintCfg,
SynCfg.SynCfg,
FormalCfg.FormalCfg,
SimCfg.SimCfg
]
found_cls = None
known_types = []
for cls in classes:
assert cls.flow is not None
known_types.append(cls.flow)
if cls.flow == flow:
found_cls = cls
break
if found_cls is None:
raise RuntimeError('{}: Configuration file sets "flow" to {!r}, but '
'this is not a known flow (known: {}).'
.format(path, flow, ', '.join(known_types)))
return (found_cls, hjson_data)
def _make_child_cfg(path, args, initial_values):
try:
cls, hjson_data = _load_cfg(path, initial_values)
except RuntimeError as err:
log.error(str(err))
sys.exit(1)
# Since this is a child configuration (from some primary configuration),
# make sure that we aren't ourselves a primary configuration. We don't need
# multi-level hierarchies and this avoids circular dependencies.
if 'use_cfgs' in hjson_data:
raise RuntimeError('{}: Configuration file has use_cfgs, but is '
'itself included from another configuration.'
.format(path))
# Call cls as a constructor. Note that we pass None as the mk_config
# argument: this is not supposed to load anything else.
return cls(path, hjson_data, args, None)
def make_cfg(path, args, proj_root, titan_root):
'''Make a flow config by loading the config file at path
args is the arguments passed to the dvsim.py tool and proj_root is the top
of the project.
'''
initial_values = {
'proj_root': proj_root,
'titan_root': titan_root,
'self_dir': os.path.dirname(path),
}
if args.tool is not None:
initial_values['tool'] = args.tool
try:
cls, hjson_data = _load_cfg(path, initial_values)
except RuntimeError as err:
log.error(str(err))
sys.exit(1)
def factory(child_path):
child_ivs = initial_values.copy()
child_ivs['flow'] = hjson_data['flow']
return _make_child_cfg(child_path, args, child_ivs)
return cls(path, hjson_data, args, factory)