blob: 18e8181bc9586d3c1777bb161e779854b8401ac6 [file] [log] [blame]
# Copyright lowRISC contributors.
# Licensed under the Apache License, Version 2.0, see LICENSE for details.
# SPDX-License-Identifier: Apache-2.0
from pathlib import Path
from typing import Any, Dict, Optional, Union
import hjson # type: ignore
from reggen.lib import check_int, check_keys, check_list, check_name, check_str
from reggen.params import BaseParam, Params
class TemplateParseError(Exception):
pass
class TemplateParameter(BaseParam):
""" A template parameter. """
VALID_PARAM_TYPES = (
'int',
'string',
'object',
)
def __init__(self, name: str, desc: Optional[str], param_type: str,
default: str):
assert param_type in self.VALID_PARAM_TYPES
super().__init__(name, desc, param_type)
self.default = default
self.value = None
def as_dict(self) -> Dict[str, object]:
rd = super().as_dict()
rd['default'] = self.default
return rd
def _parse_template_parameter(where: str, raw: object) -> TemplateParameter:
rd = check_keys(raw, where, ['name', 'desc', 'type'], ['default'])
name = check_str(rd['name'], 'name field of ' + where)
r_desc = rd.get('desc')
if r_desc is None:
desc = None
else:
desc = check_str(r_desc, 'desc field of ' + where)
r_type = rd.get('type')
param_type = check_str(r_type, 'type field of ' + where)
if param_type not in TemplateParameter.VALID_PARAM_TYPES:
raise ValueError('At {}, the {} param has an invalid type field {!r}. '
'Allowed values are: {}.'.format(
where, name, param_type,
', '.join(TemplateParameter.VALID_PARAM_TYPES)))
r_default = rd.get('default')
if param_type == 'int':
default = check_int(
r_default,
'default field of {}, (an integer parameter)'.format(name))
elif param_type == 'string':
default = check_str(r_default, 'default field of ' + where)
elif param_type == 'object':
default = IpConfig._check_object(r_default, 'default field of ' + where)
else:
assert False, f"Unknown parameter type found: {param_type!r}"
return TemplateParameter(name, desc, param_type, default)
class TemplateParams(Params):
""" A group of template parameters. """
@classmethod
def from_raw(cls, where: str, raw: object) -> 'TemplateParams':
""" Produce a TemplateParams instance from an object as it is in Hjson.
"""
ret = cls()
rl = check_list(raw, where)
for idx, r_param in enumerate(rl):
entry_where = 'entry {} in {}'.format(idx + 1, where)
param = _parse_template_parameter(entry_where, r_param)
if param.name in ret:
raise ValueError('At {}, found a duplicate parameter with '
'name {}.'.format(entry_where, param.name))
ret.add(param)
return ret
class IpTemplate:
""" An IP template.
An IP template is an IP block which needs to be parametrized before it
can be transformed into an actual IP block (which can then be instantiated
in a hardware design).
"""
name: str
params: TemplateParams
template_path: Path
def __init__(self, name: str, params: TemplateParams, template_path: Path):
self.name = name
self.params = params
self.template_path = template_path
@classmethod
def from_template_path(cls, template_path: Path) -> 'IpTemplate':
""" Create an IpTemplate from a template directory.
An IP template directory has a well-defined structure:
- The IP template name (TEMPLATE_NAME) is equal to the directory name.
- It contains a file 'data/TEMPLATE_NAME.tpldesc.hjson' containing all
configuration information related to the template.
- It contains zero or more files ending in '.tpl'. These files are
Mako templates and rendered into an file in the same location without
the '.tpl' file extension.
"""
# Check if the directory structure matches expectations.
if not template_path.is_dir():
raise TemplateParseError(
"Template path {!r} is not a directory.".format(
str(template_path)))
if not (template_path / 'data').is_dir():
raise TemplateParseError(
"Template path {!r} does not contain the required 'data' directory."
.format(str(template_path)))
# The template name equals the name of the template directory.
template_name = template_path.stem
# Find the template description file.
tpldesc_file = template_path / 'data/{}.tpldesc.hjson'.format(
template_name)
# Read the template description from file.
try:
tpldesc_obj = hjson.load(open(tpldesc_file, 'r'), use_decimal=True)
except (OSError, FileNotFoundError) as e:
raise TemplateParseError(
"Unable to read template description file {!r}: {}".format(
str(tpldesc_file), str(e)))
# Parse the template description file.
where = 'template description file {!r}'.format(str(tpldesc_file))
if 'template_param_list' not in tpldesc_obj:
raise TemplateParseError(
f"Required key 'variables' not found in {where}")
try:
params = TemplateParams.from_raw(
f"list of parameters in {where}",
tpldesc_obj['template_param_list'])
except ValueError as e:
raise TemplateParseError(e) from None
return cls(template_name, params, template_path)
class IpConfig:
def __init__(self,
template_params: TemplateParams,
instance_name: str,
param_values: Dict[str, Union[str, int]] = {}):
self.template_params = template_params
self.instance_name = instance_name
self.param_values = IpConfig._check_param_values(
template_params, param_values)
@staticmethod
def _check_object(obj: object, what: str) -> object:
"""Check that obj is a Hjson-serializable object.
If not, raise a ValueError; the what argument names the object.
"""
try:
# Round-trip objects through the JSON encoder to get the
# same representation no matter if we load the config from
# file, or directly pass it on to the template. Also, catch
# encoding/decoding errors when setting the object.
json = hjson.dumps(obj,
ensure_ascii=False,
use_decimal=True,
for_json=True,
encoding='UTF-8')
obj_checked = hjson.loads(json,
use_decimal=True,
encoding='UTF-8')
except TypeError as e:
raise ValueError('{} cannot be serialized as Hjson: {}'
.format(what, str(e))) from None
return obj_checked
@staticmethod
def _check_param_values(template_params: TemplateParams,
param_values: Any) -> Dict[str, Union[str, int]]:
"""Check if parameter values are valid.
Returns the parameter values in typed form if successful, and throws
a ValueError otherwise.
"""
VALID_PARAM_TYPES = ('string', 'int', 'object')
param_values_typed = {}
for key, value in param_values.items():
if not isinstance(key, str):
raise ValueError(
f"The IP configuration has a key {key!r} which is not a "
"string.")
if key not in template_params:
raise ValueError(
f"The IP configuration has a key {key!r} which is a "
"valid parameter.")
param_type = template_params[key].param_type
if param_type not in VALID_PARAM_TYPES:
raise ValueError(
f"Unknown template parameter type {param_type!r}. "
"Allowed types: " + ', '.join(VALID_PARAM_TYPES))
if param_type == 'string':
param_value_typed = check_str(
value, f"the key {key} of the IP configuration")
elif param_type == 'int':
param_value_typed = check_int(
value, f"the key {key} of the IP configuration")
elif param_type == 'object':
param_value_typed = IpConfig._check_object(
value, f"the key {key} of the IP configuration")
else:
assert False, "Unexpected parameter type found, expand check"
param_values_typed[key] = param_value_typed
return param_values_typed
@classmethod
def from_raw(cls, template_params: TemplateParams, raw: object,
where: str) -> 'IpConfig':
""" Load an IpConfig from a raw object """
rd = check_keys(raw, 'configuration file ' + where, ['instance_name'],
['param_values'])
instance_name = check_name(rd.get('instance_name'),
"the key 'instance_name' of " + where)
if not isinstance(raw, dict):
raise ValueError(
"The IP configuration is expected to be a dict, but was "
"actually a " + type(raw).__name__)
param_values = IpConfig._check_param_values(template_params,
rd['param_values'])
return cls(template_params, instance_name, param_values)
@classmethod
def from_text(cls, template_params: TemplateParams, txt: str, where: str) -> 'IpConfig':
"""Load an IpConfig from an Hjson description in txt"""
raw = hjson.loads(txt, use_decimal=True, encoding="UTF-8")
return cls.from_raw(template_params, raw, where)
def to_file(self, file_path: Path, header: Optional[str] = ""):
obj = {}
obj['instance_name'] = self.instance_name
obj['param_values'] = self.param_values
with open(file_path, 'w') as fp:
if header:
fp.write(header)
hjson.dump(obj,
fp,
ensure_ascii=False,
use_decimal=True,
for_json=True,
encoding='UTF-8',
indent=2)
fp.write("\n")