Add double-buffer UDP definitions

This commit is contained in:
Alex Mykyta
2022-10-03 21:36:49 -07:00
parent 1aa9d8b603
commit 21a4e5a41c
7 changed files with 160 additions and 3 deletions

View File

@@ -38,11 +38,12 @@ Below is a simple example that demonstrates how to generate a SystemVerilog
implementation from SystemRDL source. implementation from SystemRDL source.
.. code-block:: python .. code-block:: python
:emphasize-lines: 2-3, 23-27 :emphasize-lines: 2-4, 29-33
from systemrdl import RDLCompiler, RDLCompileError from systemrdl import RDLCompiler, RDLCompileError
from peakrdl_regblock import RegblockExporter from peakrdl_regblock import RegblockExporter
from peakrdl_regblock.cpuif.apb3 import APB3_Cpuif from peakrdl_regblock.cpuif.apb3 import APB3_Cpuif
from peakrdl_regblock.udps import ALL_UDPS
input_files = [ input_files = [
"PATH/TO/my_register_block.rdl" "PATH/TO/my_register_block.rdl"
@@ -50,6 +51,11 @@ implementation from SystemRDL source.
# Create an instance of the compiler # Create an instance of the compiler
rdlc = RDLCompiler() rdlc = RDLCompiler()
# Register all UDPs that 'regblock' requires
for udp in ALL_UDPS:
rdlc.register_udp(udp)
try: try:
# Compile your RDL files # Compile your RDL files
for input_file in input_files: for input_file in input_files:

View File

@@ -24,7 +24,7 @@ setuptools.setup(
include_package_data=True, include_package_data=True,
python_requires='>=3.6', python_requires='>=3.6',
install_requires=[ install_requires=[
"systemrdl-compiler>=1.23.0", "systemrdl-compiler>=1.25.0",
"Jinja2>=2.11", "Jinja2>=2.11",
], ],
entry_points = { entry_points = {

View File

@@ -2,6 +2,7 @@ from typing import TYPE_CHECKING
from .exporter import RegblockExporter from .exporter import RegblockExporter
from .cpuif import apb3, apb4, axi4lite, passthrough from .cpuif import apb3, apb4, axi4lite, passthrough
from .udps import ALL_UDPS
if TYPE_CHECKING: if TYPE_CHECKING:
import argparse import argparse
@@ -23,6 +24,8 @@ CPUIF_DICT = {
class Exporter: class Exporter:
short_desc = "Generate a SystemVerilog control/status register (CSR) block" short_desc = "Generate a SystemVerilog control/status register (CSR) block"
udp_definitions = ALL_UDPS
def add_exporter_arguments(self, arg_group: 'argparse.ArgumentParser') -> None: def add_exporter_arguments(self, arg_group: 'argparse.ArgumentParser') -> None:
arg_group.add_argument( arg_group.add_argument(
"--cpuif", "--cpuif",

View File

@@ -146,3 +146,25 @@ class DesignScanner(RDLListener):
self.out_of_hier_signals[path] = value self.out_of_hier_signals[path] = value
else: else:
self.in_hier_signal_paths.add(path) self.in_hier_signal_paths.add(path)
# 10.6.1-f: Any field that is software-writable or clear on read shall
# not span multiple software accessible sub-words (e.g., a 64-bit
# register with a 32-bit access width may not have a writable field with
# bits in both the upper and lower half of the register).
#
# Interpreting this further - this rule applies any time a field is
# software-modifiable by any means, including rclr, rset, ruser
# TODO: suppress this check for registers that have the appropriate
# buffer_writes/buffer_reads UDP set
parent_accesswidth = node.parent.get_property('accesswidth')
parent_regwidth = node.parent.get_property('regwidth')
if ((parent_accesswidth < parent_regwidth)
and (node.lsb // parent_accesswidth) != (node.msb // parent_accesswidth)
and (node.is_sw_writable or node.get_property('onread') is not None)):
# Field spans across sub-words
self.msg.error(
"Software-modifiable field '%s' shall not span multiple software-accessible subwords."
% node.inst_name,
node.inst.inst_src_ref
)

View File

@@ -0,0 +1,120 @@
from typing import Any
from systemrdl.udp import UDPDefinition
from systemrdl.component import Reg
from systemrdl.rdltypes.references import RefType, PropertyReference
from systemrdl.node import Node, RegNode, VectorNode
class xBufferTrigger(UDPDefinition):
valid_components = {Reg}
valid_type = RefType
def validate(self, node: Node, value: Any) -> None:
# TODO: Reference shall not cross an internal/external boundary
if isinstance(value, VectorNode):
# Trigger can reference a vector, but only if it is a single-bit
if value.width != 1:
self.msg.error(
"%s '%s' references %s '%s' but it's width is not 1"
% (
type(node.inst).__name__.lower(), node.inst_name,
type(value.inst).__name__.lower(), value.inst_name
),
self.get_src_ref(node)
)
elif isinstance(value, PropertyReference) and value.width is not None:
# Trigger can reference a property, but only if it is a single-bit
if value.width != 1:
self.msg.error(
"%s '%s' references property '%s->%s' but it's width is not 1"
% (
type(node.inst).__name__.lower(), node.inst_name,
value.node.inst_name, value.name,
),
self.get_src_ref(node)
)
elif isinstance(value, RegNode):
# Trigger can reference a register, which implies access of the
# 'correct half' of the register is the trigger.
# For buffered writes, it is the upper-half.
# For buffered reads, it is the lower-half.
pass
else:
# All other reference types are invalid
self.msg.error(
"Reference to a %s component is incompatible with the '%s' property."
% (type(node.inst).__name__.lower(), self.name),
self.get_src_ref(node)
)
#-------------------------------------------------------------------------------
class BufferWrites(UDPDefinition):
name = "buffer_writes"
valid_components = {Reg}
valid_type = bool
default_assignment = True
def validate(self, node: 'Node', value: Any) -> None:
assert isinstance(node, RegNode)
if value:
if not node.has_sw_writable:
self.msg.error(
"'buffer_writes' is set to true, but this register does not contain any writable fields.",
self.get_src_ref(node)
)
# TODO: Should I limit the use of other properties on double-buffered registers?
def get_unassigned_default(self, node: 'Node') -> Any:
return False
class WBufferTrigger(xBufferTrigger):
name = "wbuffer_trigger"
def get_unassigned_default(self, node: 'Node') -> Any:
# If buffering is enabled, trigger is the register itself
if node.get_property('buffer_writes'):
return node
return None
class BufferReads(UDPDefinition):
name = "buffer_reads"
valid_components = {Reg}
valid_type = bool
default_assignment = True
def validate(self, node: 'Node', value: Any) -> None:
assert isinstance(node, RegNode)
if value:
if not node.has_sw_readable:
self.msg.error(
"'buffer_reads' is set to true, but this register does not contain any readable fields.",
self.get_src_ref(node)
)
# TODO: Should I limit the use of other properties on double-buffered registers?
def get_unassigned_default(self, node: 'Node') -> Any:
return False
class RBufferTrigger(xBufferTrigger):
name = "rbuffer_trigger"
def get_unassigned_default(self, node: 'Node') -> Any:
# If buffering is enabled, trigger is the register itself
if node.get_property('buffer_reads'):
return node
return None
ALL_UDPS = [
BufferWrites,
WBufferTrigger,
BufferReads,
RBufferTrigger,
]

View File

@@ -10,6 +10,8 @@ import pytest
from systemrdl import RDLCompiler from systemrdl import RDLCompiler
from peakrdl_regblock import RegblockExporter from peakrdl_regblock import RegblockExporter
from peakrdl_regblock.udps import ALL_UDPS
from .cpuifs.base import CpuifTestMode from .cpuifs.base import CpuifTestMode
from .cpuifs.apb4 import APB4 from .cpuifs.apb4 import APB4
@@ -84,6 +86,9 @@ class BaseTestCase(unittest.TestCase):
rdl_file = glob.glob(os.path.join(this_dir, "*.rdl"))[0] rdl_file = glob.glob(os.path.join(this_dir, "*.rdl"))[0]
rdlc = RDLCompiler() rdlc = RDLCompiler()
for udp in ALL_UDPS:
rdlc.register_udp(udp)
rdlc.compile_file(rdl_file) rdlc.compile_file(rdl_file)
root = rdlc.elaborate(cls.rdl_elab_target, "regblock", cls.rdl_elab_params) root = rdlc.elaborate(cls.rdl_elab_target, "regblock", cls.rdl_elab_params)

View File

@@ -98,7 +98,8 @@ disable=
abstract-method, abstract-method,
protected-access, protected-access,
duplicate-code, duplicate-code,
unused-argument unused-argument,
consider-using-f-string
# Enable the message, report, category or checker with the given id(s). You can # Enable the message, report, category or checker with the given id(s). You can
# either give multiple identifier separated by comma (,) or put this option # either give multiple identifier separated by comma (,) or put this option