Ipgen is a tool to produce IP blocks from IP templates.
IP templates are highly customizable IP blocks which need to be pre-processed before they can be used in a hardware design. In the pre-processing ("rendering"), which is performed by the ipgen tool, templates of source files, written in the Mako templating language, are converted into "real" source files. The templates can be customized through template parameters, which are available within the templates.
Ipgen is a command-line tool and a library. Users wishing to instantiate an IP template or query it for template parameters will find the command-line application useful. For use in higher-level scripting, e.g. within topgen using ipgen as Python library is recommended.
An IP template is a directory with a well-defined directory layout, which mostly mirrors the standard layout of IP blocks.
An IP template directory has a well-defined structure:
- The IP template name (
<templatename>
) equals the directory name. - The directory contains a file
data/<templatename>.tpldesc.hjson
containing all configuration information related to the template. - The directory also 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.
Each IP template comes with a description itself.
This description is contained in the data/<templatename>.tpldesc.hjson
file in the template directory.
The file is written in Hjson.
It contains a top-level dictionary, the keys of which are documented next.
Keys within template_param_list
:
name
(string): Name of the template parameter.desc
(string): Human-readable description of the template parameter.type
(string): Data type of the parameter. Valid values:int
,str
default
(string|int): The default value of the parameter. The data type should match thetype
argument. As convenience, strings are converted into integers on demand (if possible).
An exemplary template description file with two parameters, src
and target
is shown below.
// data/<templatename>.tpldesc.hjson
{
"template_param_list": [
{
"name": "src",
"desc": "Number of Interrupt Source",
"type": "int",
"default": "32"
},
{
"name": "target",
"desc": "Number of Interrupt Targets",
"type": "int",
"default": "32"
},
],
}
Templates are written in the Mako templating language.
All template parameters are available in the rendering context.
For example, a template parameter src
can be used in the template as {{ src }}
.
Furthermore, the following functions are available:
instance_vlnv(vlnv)
: Transform a FuseSoC core name, expressed as VLNV string, into an instance-specific name. Thevendor
is set tolowrisc
, thelibrary
is set toopentitan
, and thename
is prefixed with the instance name. The optional version segment is retained. Use this function on thename
of all FuseSoC cores which contain sources generated from templates and which export symbols into the global namespace.
FuseSoC core files can be templated just like any other file.
Especially handy is the instance_vlnv()
template function, which transforms a placeholder VLNV (a string in the form vendor:library:name:version
) into a instance-specific one.
For example, a rv_plic.core.tpl
file could look like this:
CAPI=2:
name: ${instance_vlnv("lowrisc:ip:rv_plic")}
After processing, the name
key is set to e.g. lowrisc:opentitan:top_earlgrey_rv_plic
.
The following rules should be applied when creating IP templates:
- Template and use an instance-specific name for all FuseSoC cores which reference templated source files (e.g. SystemVerilog files).
- Template and use an instance-specific name at least the top-level FuseSoC core.
- If a FuseSoC core with an instance-specific name exposes a well-defined public interface (see below), add a
provides: lowrisc:ip_interfaces:<name>
line to the core file to allow other cores to refer to it without knowing the actual core name.
FuseSoC core files should be written in a way that upholds the principle "same name, same public interface", i.e. if a FuseSoC core has the same name as another one, it must also provide the same public interface.
Since SystemVerilog does not provide no strong control over which symbols become part of the public API developers must be carefully evaluate their source code. At least, the public interface is comprised of
- module header(s), e.g. parameter names, ports (names, data types),
- package names, and all identifiers within it, including enum values (but not the values assigned to them),
- defines
If any of those aspects of a source file are templated, the core name referencing the files must be made instance-specific.
Ipgen can be used as Python library by importing from the ipgen
package.
Refer to the comments within the source code for usage information.
The following example shows how to produce an IP block from an IP template.
from pathlib import Path
from ipgen import IpConfig, IpTemplate, IpBlockRenderer
# Load the template
ip_template = IpTemplate.from_template_path(Path('a/ip/template/directory'))
# Prepare the IP template configuration
params = {}
params['src'] = 17
ip_config = IpConfig("my_instance", params)
# Produce an IP block
renderer = IpBlockRenderer(ip_template, ip_config)
renderer.render(Path("path/to/the/output/directory"))
The output produced by ipgen is determined by the chosen renderer.
For most use cases the IpBlockRenderer
is the right choice, as it produces a full IP block directory.
Refer to the ipgen.renderer
module for more renderers available with ipgen.
The ipgen command-line tool lives in util/ipgen.py
.
The first argument is typically the action to be executed.
$ cd $REPO_TOP
$ util/ipgen.py --help
usage: ipgen.py [-h] ACTION ...
optional arguments:
-h, --help show this help message and exit
actions:
Use 'ipgen.py ACTION --help' to learn more about the individual actions.
ACTION
describe Show details about an IP template
generate Generate an IP block from an IP template
$ cd $REPO_TOP
$ util/ipgen.py generate --help
usage: ipgen.py generate [-h] [--verbose] -C TEMPLATE_DIR -o OUTDIR [--force] [--config-file CONFIG_FILE]
Generate an IP block from an IP template
optional arguments:
-h, --help show this help message and exit
--verbose More info messages
-C TEMPLATE_DIR, --template-dir TEMPLATE_DIR
IP template directory
-o OUTDIR, --outdir OUTDIR
output directory for the resulting IP block
--force, -f overwrite the output directory, if it exists
--config-file CONFIG_FILE, -c CONFIG_FILE
path to a configuration file
$ cd $REPO_TOP
$ util/ipgen.py generate --help
usage: ipgen.py describe [-h] [--verbose] -C TEMPLATE_DIR
Show all information available for the IP template.
optional arguments:
-h, --help show this help message and exit
--verbose More info messages
-C TEMPLATE_DIR, --template-dir TEMPLATE_DIR
IP template directory
Every IP block has a name, which is reflected in many places: in the name of the directory containing the block, in the base name of various files (e.g. the Hjson files in the data
directory), in the name
key of the IP description file in data/<ipname>.hjson
, and many more.
To "rename" an IP block, the content of multiple files, and multiple file names, have to be adjusted to reflect the name of the IP block, while keeping cross-references intact. Doing that is possible but a non-trivial amount of work, which is currently not implemented. This limitation is of lesser importance, as each template may be used only once in every toplevel design (see below).
What is supported and required for most IP templates is the modification of the FuseSoC core name, which can be achieved by templating relevant .core
files (see above).
Each template may be used to generate only once IP block for each top-level design. The generated IP block can still be instantiated multiple times from SystemVerilog, including with different SystemVerilog parameters passed to it. It is not possible, however, to, for example, use one IP block template to produce two different flash controllers with different template parameters.
IP templates generally contain code which exports symbols into the global namespace of the design: names of SystemVerilog modules and packages, defines, names of FuseSoC cores, etc. Such names need to be unique for each design, i.e. we cannot have multiple SystemVerilog modules with the same name in one design. To produce multiple IP blocks from the same IP template within a top-level design, exported symbols in generated IP blocks must be made "unique" within the design. Making symbols unique can be either done manually by the author of the template or automatically.
In the manual approach, the template author writes all global identifiers in templates in a unique way, e.g. module {prefix}_flash_ctrl
in SystemVerilog code, and with similar approaches wherever the name of the IP block is used (e.g. in cross-references in top_<toplevelname>.hjson
).
This is easy to implement in ipgen, but harder on the IP template authors, as it is easy to miss identifiers.
In the automated approach, a tool that understands the source code transforms all identifiers. C++ name mangling uses a similar approach. The challenge is, however, to create a tool which can reliably do such source code transformations on a complex language like SystemVerilog.
Finally, a third approach could be a combination of less frequently used/supported SystemVerilog language features like libraries, configs, and compilation units.