Creating a design¶
This chapter explains how to create a design in Topwrap, including a detailed overview of how Topwrap design files are structured.
Design description¶
To create a complete and fully synthesizable design, a design file is needed. It is used for:
specifying interconnects and IP cores
setting parameter values and describing hierarchies for the project
connecting the IPs and hierarchies
picking external ports (those which will be connected to the physical I/O).
You can see example design files in the examples
directory. The structure of the design file is shown below:
ips:
# specify relations between IPs instance names in the
# design yaml and IP cores description YAMLs
{ip1_instance_name}:
file: {resource_path} # see "Resource path syntax" section for more information
...
design:
name: {design_name} # optional name of the toplevel
hierarchies:
# see "Hierarchies" below for a detailed description of the format
...
parameters: # specify IP parameter values to be overridden
{ip_instance_name}:
{parameters_name} : {parameters_value}
...
ports:
# specify the incoming ports connections of an IP named `ip1_name`
{ip1_name}:
{port1_name} : [{ip2_name}, {port2_name}]
...
# specify the incoming ports connections of a hierarchy named `hier_name`
{hier_name}:
{port1_name} : [{ip_name}, {port2_name}]
...
# specify the external port connections
{ip_instance_name}:
{port_name} : ext_port_name
...
interfaces:
# specify the incoming interface connections of the `ip1_name` IP
{ip1_name}:
{interface1_name} : [{ip2_name}, {interface2_name}]
...
# specify the incoming interface connections of the `hier_name` hierarchy
{hier_name}:
{interface1_name} : [{ip_name}, {interface2_name}]
...
# specify the external interface connections
{ip_instance_name}:
{interface_name} : ext_interface_name
...
interconnects:
# see the "Interconnect generation" page for a detailed description of the format
...
external: # specify the names of external ports and interfaces of the top module
ports:
out:
- {ext_port_name}
inout:
- [{ip_name/hierarchy_name, port_name}]
interfaces:
in:
- {ext_interface_name}
# note that `inout:` is invalid in the interfaces section
inout
ports are handled differently than the in
and out
ports. When an IP has an inout port or when a hierarchy has an inout port specified in its external.ports.inout
section, it must be included in the external.ports.inout
section of the parent design. It is required to specify the name of the IP/hierarchy and the port name that contains it. The name of the external port is identical to the one in the IP core. In case of duplicate names, a suffix $n
is added (where n
is a natural number) to the name of the second and subsequent duplicate names. inout
ports cannot be connected to each other.
The design description YAML format allows for creating hierarchical designs. In order to create a hierarchy, add its name as a key in the design
section and describe the hierarchy design “recursively” by using the same keys and values (ports
, parameters
etc.) as in the top-level design (see above). Hierarchies can be nested recursively, which means that you can create a hierarchy inside another one.
Note that IPs and hierarchies names cannot be duplicated on the same hierarchy level. For example, the design
section cannot contain two identical keys, but it is possible to have ip_name
key in this section and ip_name
in the design
section of a separate hierarchy.
Hierarchies¶
Hierarchies allow for creating designs with subgraphs in them. The subgraphs can contain multiple IP cores and other subgraphs, allowing for the creation of nested designs in Topwrap.
Format¶
Hierarchies are specified in the design description. The hierarchies
key must be a direct descendant of the design
key.
The format is as follows:
hierarchies:
{hierarchy_name_1}:
ips: # ips that are used on this hierarchy level
{ip_name}:
...
design:
parameters:
...
ports: # ports connections internal to this hierarchy.
# note that also you have to connect port to it's external port equivalent (if exists).
{ip1_name}:
{port1_name} : [{ip2_name}, {port2_name}]
{port2_name} : {port2_external_equivalent} # connection to external port equivalent. Note that it has to be the parent port.
...
hierarchies:
{nested_hierarchy_name}:
# structure here will be the same as for {hierarchy_name_1}
...
external:
# external ports and/or interfaces of this hierarchy; these can be
# referenced in the upper-level `ports`, `interfaces` or `external` section
ports:
in:
- {port2_external_equivalent}
...
{hierarchy_name_2}:
...
A more complex example of a hierarchy can be found in the examples/hierarchy directory.
IP description files¶
Every IP wrapped by Topwrap needs a description file in the YAML format.
The ports of an IP should be placed in the global signals
key, followed by the direction - in
, out
or inout
. The module name of an IP should be placed in the global name
key, and it should be consistent with the definition in the HDL file.
As an example, this is the description of ports in the Clock Crossing
IP:
# file: clock_crossing.yaml
name: cdc_flag
signals:
in:
- clkA
- A
- clkB
out:
- B
The previous example can be used with any IP. However, in order to benefit from connecting entire interfaces simultaneously, the ports must belong to a named interface as in this example:
#file: axis_width_converter.yaml
name: axis_width_converter
interfaces:
s_axis:
type: AXIStream
mode: subordinate
signals:
in:
TDATA: [s_axis_tdata, 63, 0]
TKEEP: [s_axis_tkeep, 7, 0]
TVALID: s_axis_tvalid
TLAST: s_axis_tlast
TID: [s_axis_tid, 7, 0]
TDEST: [s_axis_tdest, 7, 0]
TUSER: s_axis_tuser
out:
TREADY: s_axis_tready
m_axis:
type: AXIStream
mode: manager
signals:
in:
TREADY: m_axis_tready
out:
TDATA: [m_axis_tdata, 31, 0]
TKEEP: [m_axis_tkeep, 3, 0]
TVALID: m_axis_tvalid
TLAST: m_axis_tlast
TID: [m_axis_tid, 7, 0]
TDEST: [m_axis_tdest, 7, 0]
TUSER: m_axis_tuser
signals: # These ports do not belong to an interface
in:
- clk
- rst
The names s_axis
and m_axis
will be used to group the selected ports.
Each signal in an interface has a name which must match with the signal that it is connected to, for example TDATA: port_name
must connect to TDATA: other_port_name
.
To speed up the generation of YAMLs, Topwrap’s parse
command (see Generating IP core description YAMLs) can be used to generate YAMLs from HDL source files.
Port widths¶
You can specify the port width in the following format:
signals:
in:
- [port_name, upper_limit, lower_limit]
port_name
- name of the port.upper_limit
andlower_limit
define the bit range, where[upper_limit, lower_limit]
determines the number of bits for the port (e.g.[63, 0]
for 64 bits).
As an example:
signals:
in:
- [gpio_io_i, 31, 0] # 32 bits
If the bit range is omitted, as in the example below, then the default width of port_name
is 1 bit.
signals:
in:
- port_name
You can also specify the signal width within interfaces.
interfaces:
s_axis:
type: AXIStream
mode: subordinate
signals:
in:
TDATA: [s_axis_tdata, 63, 0] # 64 bits
...
TVALID: s_axis_tvalid # defaults to 1 bit
TDATA
is assigned tos_axis_tdata
and is 64 bits wide, defined by[63, 0]
.TVALID
is assigned tos_axis_tvalid
and, without a specified range, defaults to1 bit
.
Parameterization¶
Port widths don’t have to be hardcoded, as parameters can describe an IP core in a generic way, and values specified in IP core YAMLs can be overridden in a design description file (see Design description).
parameters:
DATA_WIDTH: 8
KEEP_WIDTH: (DATA_WIDTH+7)/8
ID_WIDTH: 8
DEST_WIDTH: 8
USER_WIDTH: 1
interfaces:
s_axis:
type: AXI4Stream
mode: subordinate
signals:
in:
TDATA: [s_axis_tdata, DATA_WIDTH-1, 0]
TKEEP: [s_axis_tkeep, KEEP_WIDTH-1, 0]
...
TID: [s_axis_tid, ID_WIDTH-1, 0]
TDEST: [s_axis_tdest, DEST_WIDTH-1, 0]
TUSER: [s_axis_tuser, USER_WIDTH-1, 0]
The parameter values can be integers or math expressions.
Port slicing¶
Ports can be sliced for using some parts of the port as a signal that belongs to a defined interface.
As an example:
Port m_axi_bid
of the IP core is 36 bits wide. Use bits 23..12
as the BID
signal of the m_axi_1
AXI manager
m_axi_1:
type: AXI
mode: manager
signals:
in:
BID: [m_axi_bid, 35, 0, 23, 12]
Interface description files¶
Topwrap can use predefined interfaces, as illustrated in YAML files that come packaged with the tool. The currently supported interfaces are AXI3, AXI4, AXI Lite, AXI Stream and Wishbone.
An example file looks as follows:
name: AXI4Stream
port_prefix: AXIS
signals:
# The convention assumes the AXI Stream transmitter (manager) perspective
required:
out:
TVALID: tvalid
TDATA: tdata
TLAST: tlast
in:
TREADY: tready
optional:
out:
TID: tid
TDEST: tdest
TKEEP: tkeep
TSTRB: tstrb
TUSER: tuser
TWAKEUP: twakeup
The name
of an interface must be unique.
Signals are either required or optional, and their direction is described from the perspective of the manager (i.e. the direction of signals in the subordinate are flipped).
Note that clock
and reset
are not included as these are usually inputs to both the manager and subordinate, so they are not supported in the interface specification.
Every signal is a key-value pair, where the key is a generic signal name (normally taken from the interface specification) and used to identify it in other parts of Topwrap (i.e. IP core description files), and the value is a regex used to deduce which port defined in the HDL sources represents this signal.
Interface deduction¶
During IP core parsing, you can use the --iface-deduce
flag to enable automatic pairing of raw ports from HDL sources to interface signals.
This feature matches signal regexes from all available interface descriptions with raw port names of the IP core in order to discover possible interface instances.
The pairing is performed on port names that are transformed to lowercase and have the common port_prefix
removed, which means that the regexes must also be written in lowercase.
Interface compliance¶
During the build process, an optional verification of whether the interface instances used in IP cores are compliant with their respective descriptions can be enabled. The verification consists of checking in the instance if:
all signals designated as required in the description are included.
no additional signals beyond those defined in the description are included.
This feature is controlled by the --iface-compliance
CLI flag or the force_interface_compliance
key in the configuration file and is turned off by default.
Resource path syntax¶
Fields specified in the YAML file as a “resource path” support extended functionality and have their own specific syntax.
This field type is used for example in the Design Description for specifying an IP Core description location:
ips:
ip_inst_name:
file: {resource path}
...
The syntax is as follows:
SCHEME[ARG1|ARG2...]:SCHEME_PATH
SCHEME
is the scheme of this path (e.g.get
for remote resources)ARGS
are|
-separated positional arguments for the specific scheme (e.g. the user repo name for therepo
scheme)If there are no arguments to supply you can omit the square brackets entirely
SCHEME_PATH
is the path to the resource interpreted by the specific scheme (e.g. the URL for theget
scheme)
Available schemes¶
file
SCHEME_ARGS
: NoneSCHEME_PATH
: A filesystem path relative from the currently edited YAML file to the resource
repo
SCHEME_ARGS
: Repository nameSCHEME_PATH
: A path from the root of the user repository given by the name
get
SCHEME_ARGS
: NoneSCHEME_PATH
: The URL address of the remote resource. Onlyhttp(s)://
URLs are currently supported.
Examples¶
file:./my_directory/file.txt
A path to the file on the filesystem.
repo[builtin]:cores/axi_protocol_converter/core.yaml
This loads the axi_protocol_converter
core located in the builtin user repository.
repo[my_repo]:res.txt
This loads the res.txt
file inside the my_repo
loaded user repository.
get:https://raw.githubusercontent.com/antmicro/topwrap/refs/heads/main/pyproject.toml
This loads the remote resource. When necessary, it’s automatically downloaded into a temporary directory.