Customizing the CPU interface
Use your own existing SystemVerilog interface definition
This exporter comes pre-bundled with its own SystemVerilog interface declarations. What if you already have your own SystemVerilog interface declaration that you prefer?
Not a problem! As long as your interface definition is similar enough, it is easy to customize and existing CPUIF definition.
As an example, let’s use the SystemVerilog interface definition for AMBA AXI4-Lite that is bundled with this project. This interface uses the following style and naming conventions:
SystemVerilog interface type name is
axi4lite_intf
Defines modports named
master
andslave
Interface signals are all upper-case:
AWREADY
,AWVALID
, etc…
Lets assume your preferred SV interface definition uses a slightly different naming convention:
SystemVerilog interface type name is
axi4_lite_interface
Modports are capitalized and use suffixes
Master_mp
andSlave_mp
Interface signals are all lower-case:
awready
,awvalid
, etc…
Rather than rewriting a new CPU interface definition, you can extend and adjust the existing one:
from peakrdl_regblock.cpuif.axi4lite import AXI4Lite_Cpuif
class My_AXI4Lite(AXI4Lite_Cpuif):
@property
def port_declaration(self) -> str:
# Override the port declaration text to use the alternate interface name and modport style
return "axi4_lite_interface.Slave_mp s_axil"
def signal(self, name:str) -> str:
# Override the signal names to be lowercase instead
return "s_axil." + name.lower()
Then use your custom CPUIF during export:
exporter = RegblockExporter()
exporter.export(
root, "path/to/output_dir",
cpuif_cls=My_AXI4Lite
)
Custom CPU Interface Protocol
If you require a CPU interface protocol that is not included in this project, you can define your own.
Create a SystemVerilog CPUIF implementation template file.
This contains the SystemVerilog implementation of the bus protocol. The logic in this shall implement a translation between your custom protocol and the Internal CPUIF Protocol.
Reminder that this template will be preprocessed using Jinja, so you can use some templating tags to dynamically render content. See the implementations of existing CPU interfaces as an example.
Create a Python class that defines your CPUIF
Extend your class from
peakrdl_regblock.cpuif.CpuifBase
. Define the port declaration string, and provide a reference to your template file.Use your new CPUIF definition when exporting.
If you think the CPUIF protocol is something others might find useful, let me know and I can add it to PeakRDL!
Loading into the PeakRDL command line tool
There are two ways to make your custom CPUIF class visible to the PeakRDL command-line tool.
Via the PeakRDL TOML
The easiest way to add your cpuif is via the TOML config file. See the Configuring PeakRDL-regblock section for more details.
Via a package’s entry point definition
If you are publishing a collecxtion of PeakRDL plugins as an installable Python package, you can advertise them to PeakRDL using an entry point. This advertises your custom CPUIF class to the PeakRDL-regblock tool as a plugin that should be loaded, and made available as a command-line option in PeakRDL.
import setuptools
setuptools.setup(
name="my_package",
packages=["my_package"],
# ...
entry_points = {
"peakrdl_regblock.cpuif": [
'my-cpuif = my_package.__peakrdl_regblock__:MyCPUIF'
]
}
)
my_package
: The name of your installable Python modulepeakrdl-regblock.cpuif
: This is the namespace that PeakRDL-regblock will search. Any cpuif plugins you create must be enclosed in this namespace in order to be discovered.my_package.__peakrdl_regblock__:MyCPUIF
: This is the import path that points to your CPUIF class definition.my-cpuif
: The lefthand side of the assignment is your cpuif’s name. This text is what the end-user uses in the command line interface to select your CPUIF implementation.