Register Tool
The register tool is used to construct register documentation, register RTL and header files. It is either used stand-alone or by being invoked as part of Markdown processing.
Running standalone regtool.py
The standalone regtool.py
is a Python 3 tool to read configuration and register descriptions in Hjson and generate various output formats.
Currently it can output HTML documentation, standard JSON, compact standard JSON (whitespace removed), Hjson, Verilog RTL and various forms of C header files.
The standard --help
and --version
command line flags are supported to print the usage and version information.
Because the version includes information on libraries (which may be different between systems) reporting the version output is sometimes useful when issues are reported.
Setup and Examples
Setup and examples of the tool are given in the README.md file in the util/reggen
directory.
Configuration and Register Definition File Format
The tool input is an Hjson file containing the Comportable description of the IP block and its registers.
A description of Hjson (a variant of JSON) and the recommended style is in the Hjson Usage and Style Guide.
The tables below describe valid keys for each context. It is an error if required keys are missing from the input JSON. Optional keys may be provided in the input file as needed, as noted in the tables the tool may insert them (with default or computed values) during validation so the output generators do not have to special case them. Keys marked as “inserted by tool” should not be in the input JSON (they will be silently overwritten if they are there), they are derived by the tool during validation of the input and available to the output generators.
For more detail on the non-register entries of the Hjson configuration file, see this section of the Comportability Specification.
The tables describe each key and the type of the value. The following types are used:
Type | Description |
---|---|
int | integer (binary 0b, octal 0o, decimal, hex 0x) |
xint | x for undefined otherwise int |
bitrange | bit number as decimal integer, or bit-range as decimal integers msb:lsb |
list | comma separated list enclosed in [] |
name list | comma separated list enclosed in [] of one or more groups that have just name and dscr keys. e.g. { name: "name", desc: "description"} |
name list+ | name list that optionally contains a width |
parameter list | parameter list having default value optionally |
group | comma separated group of key:value enclosed in {} |
list of group | comma separated group of key:value enclosed in {} the second entry of the list is the sub group format |
string | string, typically short |
text | string, may be multi-line enclosed in ''' may use **bold** , *italic* or !!Reg markup |
tuple | tuple enclosed in () |
python int | Native Python type int (generated) |
python Bool | Native Python type Bool (generated) |
python list | Native Python type list (generated) |
python enum | Native Python type enum (generated) |
Register fields are tagged using the swaccess key to describe the permitted access and side-effects. This key must have one of these values:
Key | Description |
---|---|
none | No access |
ro | Read Only |
rc | Read Only, reading clears |
rw | Read/Write |
r0w1c | Read zero, Write with 1 clears |
rw1s | Read, Write with 1 sets |
rw1c | Read, Write with 1 clears |
rw0c | Read, Write with 0 clears |
wo | Write Only |
Register fields are tagged using the hwaccess key to describe the permitted access from hardware logic and side-effects. This key must have one of these values:
Key | Description |
---|---|
hro | Read Only |
hrw | Read/Write |
hwo | Write Only |
none | No Access Needed |
The top level of the JSON is a group containing the following keys:
Key | Kind | Type | Description of Value |
---|---|---|---|
name | required | string | name of the component |
cip_id | required | int | unique comportable IP identifier |
clocking | required | list | clocking for the device |
bus_interfaces | required | list | bus interfaces for the device |
registers | required | list | list of register definition groups and offset control groups |
human_name | optional | string | human-readable name of the component |
one_line_desc | optional | string | one-line description of the component |
one_paragraph_desc | optional | string | one-paragraph description of the component |
revisions | optional | list | list with revision records |
design_spec | optional | string | path to the design specification, relative to repo root |
dv_doc | optional | string | path to the DV document, relative to repo root |
hw_checklist | optional | string | path to the hw_checklist, relative to repo root |
sw_checklist | optional | string | path to the sw_checklist, relative to repo root |
design_stage | optional | string | design stage of module |
dif_stage | optional | string | DIF stage of module |
verification_stage | optional | string | verification stage of module |
notes | optional | string | random notes |
version | optional | string | module version |
life_stage | optional | string | life stage of module |
commit_id | optional | string | commit ID of last stage sign-off |
alert_list | optional | name list+ | list of peripheral alerts |
available_inout_list | optional | name list+ | list of available peripheral inouts |
available_input_list | optional | name list+ | list of available peripheral inputs |
available_output_list | optional | name list+ | list of available peripheral outputs |
expose_reg_if | optional | python Bool | if set, expose reg interface in reg2hw signal |
interrupt_list | optional | name list+ | list of peripheral interrupts |
inter_signal_list | optional | list | list of inter-module signals |
no_auto_alert_regs | optional | string | Set to true to suppress automatic generation of alert test registers. Defaults to true if no alert_list is present. Otherwise this defaults to false. |
no_auto_intr_regs | optional | string | Set to true to suppress automatic generation of interrupt registers. Defaults to true if no interrupt_list is present. Otherwise this defaults to false. |
param_list | optional | parameter list | list of parameters of the IP |
regwidth | optional | int | width of registers in bits (default 32) |
reset_request_list | optional | list | list of signals requesting reset |
scan | optional | python Bool | Indicates the module have scanmode_i |
scan_reset | optional | python Bool | Indicates the module have scan_rst_ni |
scan_en | optional | python Bool | Indicates the module has scan_en_i |
SPDX-License-Identifier | optional | string | License ientifier (if using pure json) Only use this if unable to put this information in a comment at the top of the file. |
wakeup_list | optional | name list+ | list of peripheral wakeups |
countermeasures | optional | name list | list of countermeasures in this block |
features | optional | name list | list of functional features in this block |
The basic structure of a register definition file is thus:
{
name: "GP",
regwidth: "32",
registers: [
// register definitions...
]
}
The list of registers includes register definition groups containing the following keys:
Key | Kind | Type | Description of Value |
---|---|---|---|
name | required | string | name of the register |
desc | required | text | description of the register. This field supports the markdown syntax. |
fields | required | list | list of register field description groups |
alias_target | optional | string | name of the register to apply the alias definition to. |
async | optional | string | indicates the register must cross to a different clock domain before use. The value shown here should correspond to one of the module’s clocks. |
sync | optional | string | indicates the register needs to be on another clock/reset domain.The value shown here should correspond to one of the module’s clocks. |
swaccess | optional | string | software access permission to use for fields that don’t specify swaccess |
hwaccess | optional | string | hardware access permission to use for fields that don’t specify hwaccess |
hwext | optional | string | ‘true’ if the register is stored outside of the register module |
hwqe | optional | string | ‘true’ if hardware uses ‘q’ enable signal, which is latched signal of software write pulse. |
hwre | optional | string | ‘true’ if hardware uses ‘re’ signal, which is latched signal of software read pulse. |
regwen | optional | string | if register is write-protected by another register, that register name should be given here. empty-string for no register write protection |
resval | optional | int | reset value of full register (default 0) |
tags | optional | string | tags for the register, following the format ‘tag_name:item1:item2…’ |
shadowed | optional | string | ‘true’ if the register is shadowed |
update_err_alert | optional | string | alert that will be triggered if this shadowed register has update error |
storage_err_alert | optional | string | alert that will be triggered if this shadowed register has storage error |
writes_ignore_errors | optional | bitrange | This register may update on a TL write that causes an error response. |
The basic register definition group will follow this pattern:
{ name: "REGA",
desc: "Description of register",
swaccess: "rw",
resval: "42",
fields: [
// bit field definitions...
]
}
The name and brief description are required. If the swaccess key is provided it describes the access pattern that will be used by all bitfields in the register that do not override with their own swaccess key. This is a useful shortcut because in most cases a register will have the same access restrictions for all fields. The reset value of the register may also be provided here or in the individual fields. If it is provided in both places then they must match, if it is provided in neither place then the reset value defaults to zero for all except write-only fields when it defaults to x.
In the fields list each field definition is a group itself containing the following keys:
Key | Kind | Type | Description of Value |
---|---|---|---|
bits | required | bitrange | bit or bit range (msb:lsb) |
name | optional | string | name of the field |
desc | optional | text | description of field (required if the field has a name). This field supports the markdown syntax. |
alias_target | optional | string | name of the field to apply the alias definition to. |
swaccess | optional | string | software access permission, copied from register if not provided in field. (Tool adds if not provided.) |
hwaccess | optional | string | hardware access permission, copied from register if not provided in field. (Tool adds if not provided.) |
hwqe | optional | bitrange | ‘true’ if hardware uses ‘q’ enable signal, which is latched signal of software write pulse. Copied from register if not provided in field. (Tool adds if not provided.) |
resval | optional | xint | reset value, comes from register resval if not provided in field. Zero if neither are provided and the field is readable, x if neither are provided and the field is wo. Must match if both are provided. |
enum | optional | list | list of permitted enumeration groups |
tags | optional | string | tags for the field, followed by the format ‘tag_name:item1:item2…’ |
mubi | optional | bitrange | boolean flag for whether the field is a multi-bit type |
auto_split | optional | bitrange | boolean flag which determines whether the field should be automatically separated into 1-bit sub-fields.This flag is used as a hint for automatically generated software headers with register description. |
Field names should be relatively short because they will be used frequently (and need to fit in the register layout picture!) The field description is expected to be longer and will most likely make use of the Hjson ability to include multi-line strings. An example with three fields:
fields: [
{ bits: "15:0",
name: "RXS",
desc: '''
Last 16 oversampled values of RX. These are captured at 16x the baud
rate clock. This is a shift register with the most recent bit in
bit 0 and the oldest in bit 15. Only valid when ENRXS is set.
'''
}
{ bits: "16",
name: "ENRXS",
desc: '''
If this bit is set the receive oversampled data is collected
in the RXS field.
'''
}
{bits: "20:19", name: "TXILVL",
desc: "Trigger level for TX interrupts",
resval: "2",
enum: [
{ value: "0", name: "txlvl1", desc: "1 character" },
{ value: "1", name: "txlvl4", desc: "4 characters" },
{ value: "2", name: "txlvl8", desc: "8 characters" },
{ value: "3", name: "txlvl16", desc: "16 characters" }
]
}
]
In all of these the swaccess parameter is inherited from the register level, and will be added so this key is always available to the backend. The RXS and ENRXS will default to zero reset value (unless something different is provided for the register) and will have the key added, but TXILVL expicitly sets its reset value as 2.
The missing bits 17 and 18 will be treated as reserved by the tool, as will any bits between 21 and the maximum in the register.
The TXILVL is an example using an enumeration to specify all valid values for the field. In this case all possible values are described, if the list is incomplete then the field is marked with the rsvdenum key so the backend can take appropriate action. (If the enum field is more than 7 bits then the checking is not done.)
Definitions in an enumeration group contain:
Key | Kind | Type | Description of Value |
---|---|---|---|
name | required | string | name of the member of the enum |
desc | required | text | description when field has this value |
value | required | int | value of this member of the enum |
The list of registers may include single entry groups to control the offset, open a window or generate registers:
Key | Kind | Type | Description of Value |
---|---|---|---|
reserved | optional | int | number of registers to reserve space for |
skipto | optional | int | set next register offset to value |
window | optional | group | group defining an address range for something other than standard registers |
multireg | optional | group | group defining registers generated from a base instance. |
Registers can protect themselves from software writes by using the
register attribute regwen. When not an empty string (the default
value), regwen indicates that another register must be true in order
to allow writes to this register. This is useful for the prevention
of software modification. The register-enable register (call it
REGWEN) must be either one bit or 4 bit in width to support a FI-protected
multi-bit REGWEN. The default value should be true and be rw1c for
preferred security control. This allows all writes to proceed
until at some point software disables future modifications by clearing
REGWEN. An error is reported if REGWEN does not exist, contains more
than one bit, is not rw1c
or does not default to true. One REGWEN can
protect multiple registers. The REGWEN register must precede those
registers that refer to it in the .hjson register list. An example:
{ name: "REGWEN",
desc: "Register write enable for a bank of registers",
swaccess: "rw1c",
fields: [ { bits: "0", resval: "1" } ]
}
{ name: "REGA",
swaccess: "rw",
regwen: "REGWEN",
...
}
{ name: "REGB",
swaccess: "rw",
regwen: "REGWEN",
...
}
A window defines an open region of the register space that can be used for things that are not registers (for example access to a buffer ram).
Key | Kind | Type | Description of Value |
---|---|---|---|
name | required | string | name of the window |
desc | required | text | description of the window |
items | required | int | size in fieldaccess width words of the window |
swaccess | required | string | software access permitted |
data-intg-passthru | optional | string | True if the window has data integrity pass through. Defaults to false if not present. |
byte-write | optional | string | True if byte writes are supported. Defaults to false if not present. |
validbits | optional | int | Number of valid data bits within regwidth sized word. Defaults to regwidth. If smaller than the regwidth then in each word of the window bits [regwidth-1:validbits] are unused and bits [validbits-1:0] are valid. |
unusual | optional | string | True if window has unusual parameters (set to prevent Unusual: errors).Defaults to false if not present. |
The multireg expands on the register required fields and will generate a list of the generated registers (that contain all required and generated keys for an actual register).
Key | Kind | Type | Description of Value |
---|---|---|---|
name | required | string | base name of the registers |
desc | required | text | description of the registers |
count | required | string | number of instances to generate. This field can be integer or string matching from param_list. |
cname | required | string | base name for each instance, mostly useful for referring to instance in messages. |
fields | required | list | list of register field description groups. Describes bit positions used for base instance. |
alias_target | optional | string | name of the register to apply the alias definition to. |
async | optional | string | indicates the register must cross to a different clock domain before use. The value shown here should correspond to one of the module’s clocks. |
sync | optional | string | indicates the register needs to be on another clock/reset domain.The value shown here should correspond to one of the module’s clocks. |
swaccess | optional | string | software access permission to use for fields that don’t specify swaccess |
hwaccess | optional | string | hardware access permission to use for fields that don’t specify hwaccess |
hwext | optional | string | ‘true’ if the register is stored outside of the register module |
hwqe | optional | string | ‘true’ if hardware uses ‘q’ enable signal, which is latched signal of software write pulse. |
hwre | optional | string | ‘true’ if hardware uses ‘re’ signal, which is latched signal of software read pulse. |
regwen | optional | string | if register is write-protected by another register, that register name should be given here. empty-string for no register write protection |
resval | optional | int | reset value of full register (default 0) |
tags | optional | string | tags for the register, following the format ‘tag_name:item1:item2…’ |
shadowed | optional | string | ‘true’ if the register is shadowed |
update_err_alert | optional | string | alert that will be triggered if this shadowed register has update error |
storage_err_alert | optional | string | alert that will be triggered if this shadowed register has storage error |
writes_ignore_errors | optional | bitrange | This register may update on a TL write that causes an error response. |
regwen_multi | optional | python Bool | If true, regwen term increments along with current multireg count. |
compact | optional | python Bool | If true, allow multireg compacting.If false, do not compact. |
cdc | optional | string | indicates the register must cross to a different clock domain before use. The value shown here should correspond to one of the module’s clocks. |
(end of output generated by regtool.py --doc
)
The tool will normally generate the register address offset by starting from 0 and allocating the registers in the order they are in the input file.
Between each register the offset is incremented by the number of bytes in the regwidth
(4 bytes for the default 32-bit regwidth
), so the registers end up packed into the smallest space.
Space may be held for future registers (or to match some other layout) by reserving register slots. A group containing just the reserved key can be inserted in the list of registers to reserve space. For example to reserve space for four registers between REGA and REGB (thus make REGB offset be REGA offset plus 5 times the size in bytes of a register):
{ name: "REGA",
...register definition...
}
{ reserved: "4" }
{ name: "REGB",
...register definition...
}
In other cases, such as separating functional groups of registers, the absolute offset can be specified. The next register will have the offset specified. It is an error if the requested offset is less than the current offset. For example to place ITCR at offset 0x100:
{ skipto: "0x100" }
{ name: "ITCR",
...register definition...
}
The tool can reserve an area of the memory space for something that is not a simple register, for example access to a buffer memory.
This is done with a window
declaration.
The window size is specified as items:
where each item is a regwidth
wide word.
The size in bytes is thus (items * (regwidth/8))
bytes.
If byte writes are supported the byte-write: "True"
flag can be given.
The tool will normally increment the offset to align the region based on its size.
{window: {
name: "win1"
items: "64"
swaccess: "rw"
desc: '''
A simple 256 byte window that will be aligned.
'''
}
},
The tool will give a warning if the size is not a power of 2.
The tool will also give a warning if the window has software access other than read-only, write-only or read-write.
Both of these warnings are suppressed if the description acknowledges there is something special about this window by setting unusual: "True"
in the window declaration.
The tool will increment the offset to align the region based on its size. The start address is aligned such that the base item in the window is at an address with all zeros in the low bits. For instance, if the current offset is 0x104, and the window size in 32-bit words is between 0x11 and 0x20 (inclusive) (i.e. 65-128 bytes), the window base will be set to 0x180. The next register will immediately follow the window, so will be at the window base address plus the window size in bytes.
Sometimes the window may need to map a structure that is not a full word wide (for example providing debug access to a the memory in a 12-bit wide FIFO). In this case it may be convenient to have only the low bits of each word valid and use the word address directly as an index (rather than presenting a “packed” structure with the sub-word items packed into as few words as possible). The window declaration can be annotated to document this. For example debug access to a 64 entry 12-bit wide FIFO could use a window:
{window: {
name: "fifodebug"
items: "64"
validbits: "12"
swaccess: "ro"
desc: '''
The 64 entry FIFO is mapped into the low 12-bits
of each regwidth bit wide word.
'''
}
},
The tool can generate registers that follow a base pattern, for example when there are configuration fields for multiple instances. The base pattern defines the bits (which need not be contiguous) used for the first instance and the tool uses this to pack the required number of instances into one or more registers.
For example a fancy GPIO interrupt configuration may have 4 bits per GPIO to allow generation on rising and falling edge and a two bit enum to determine the interrupt severity.
In this case the multireg can be used to build the multiple registers needed.
The description below shows the fields given for GPIO0 and requests generation of 32 instances.
If the registers are 32 bits wide then the tool will pack the four bit instances into four registers INT_CTRL_0
, INT_CTRL_1
, INT_CTRL_2
and INT_CTRL_3
.
{ multireg: {
name: "INT_CTRL",
desc: "GPIO Interrupt control",
count: "32",
cname: "GPIO",
swaccess: "rw",
fields: [
{ bits: "0", name: "POS", resval: "0",
desc: "Set to interrupt on rising edge"
}
{ bits: "1", name: "NEG", resval: "0",
desc: "Set to interrupt on falling edge"
}
{ bits: "3:2", name: "TYPE", resval: "0",
desc: "Type of interrupt to raise"
enum: [
{value: "0", name: "none", desc: "no interrupt, only log" },
{value: "1", name: "low", desc: "low priotiry interrupt" },
{value: "2", name: "high", desc: "high priotiry interrupt" },
{value: "3", name: "nmi", desc: "non maskable interrupt" }
]
}
]
}
},
Note that the definition bits for the base instance need not be contiguous. In this case the tool will match the pattern for the other instances. For example the data bits and mask bits could be in the lower and upper parts of a register:
{ multireg: {
name: "WDATA",
desc: "Write with mask to GPIO out register",
count: "32",
cname: "GPIO",
swaccess: "rw",
fields: [
{ bits: "0", name: "D", resval: "0",
desc: "Data to write if mask bit is 1"
}
{ bits: "16", name: "M", resval: "0",
desc: "Mask, set to allow data write"
}
]
}
}
In this case instance 1 will use bits 1 and 17, instance 2 will use 2 and 18 and so on. Instance 16 does not fit, so will start a new register.
Verification Tags Definition and Format
This section documents the usage of tags in the register Hjson file.
Tags
is a list of strings that could add into a register, field, or memory.
It can store special information such as csr register/field exclusion, memory exclusion, reset test exclusion, etc.
Adding a tag follows the string format "tag_name:item1:item2..."
.
For example:
tags: [// don't write to wdata - it affects several other csrs
"excl:CsrNonInitTests:CsrExclWrite"]
Current tags
supports:
- CSR tests exclusions:
Simulation based verification will run four CSR tests (if applicable to the module) through automation.
Detailed description of this methodology is documented in CSR utilities.
The tag name is
excl
, and items are enum values for what CSR tests to exclude from, and what type of exclusions. The enum types for exclusion test are:
The enum types for exclusion type are:// csr test types typedef enum bit [NUM_CSR_TEST-1:0] { CsrNonTest = 5'h0, // elementary test types CsrHwResetTest = 5'h1, CsrRwTest = 5'h2, CsrBitBashTest = 5'h4, CsrAliasingTest = 5'h8, // combinational test types (combinations of the above), used for exclusion tagging CsrNonInitTests = 5'he, // all but HwReset test CsrAllTests = 5'hf // all tests } csr_test_type_e;
typedef enum bit[2:0] { CsrNoExcl = 3'b000, // no exclusions CsrExclInitCheck = 3'b001, // exclude csr from init val check CsrExclWriteCheck = 3'b010, // exclude csr from write-read check CsrExclCheck = 3'b011, // exclude csr from init or write-read check CsrExclWrite = 3'b100, // exclude csr from write CsrExclAll = 3'b111 // exclude csr from init or write or write-read check } csr_excl_type_e;
Register Tool Hardware Generation
This section details the register generation for hardware instantiation.
The input to the tool for this generation is the same .hjson
file described above.
The output is two Verilog files that can be instantiated by a peripheral that follows the Comportability Guidelines.
The register generation tool will generate the RTL if it is invoked with the -r
flag.
The -t <directory>
flag is used to specify the output directory where the two files will be written.
As an example the tool can be invoked from the top project directory to generate the uart registers with:
$ ./util/regtool.py -r -t hw/ip/uart/rtl hw/ip/uart/data/uart.hjson
The first created file (name_reg_pkg.sv
, from name.hjson
) contains a SystemVerilog package definition that includes type definitions for two packed structures that have details of the registers and fields (all names are converted to lowercase).
The name_reg2hw_t
structure contains the signals that are driven from the register module to the rest of the hardware (this contains any required .q, .qe
, and .re
signals described below).
The name_hw2reg_t
structure contains the signals that are driven from the rest of the hardware to the register module (this contains any required .d
and .de
signals described below).
The file also contains parameters giving the byte address offsets of the registers (these are prefixed with the peripheral name
and converted to uppercase).
The second file (name_reg_top.sv
) is a SystemVerilog file that contains a module (name_reg_top
) that instantiates the registers.
This module connects to the TL-UL system bus interface and provides the register connections to the rest of the hardware.
If the register definition contains memory windows then there will be subordinate TL-UL bus connections for each window.
The module signature is:
module name_reg_top (
input clk_i,
input rst_ni,
// Below Regster interface can be changed
input tlul_pkg::tl_h2d_t tl_i,
output tlul_pkg::tl_d2h_t tl_o,
// This section is only provided if the definition includes
// 1 or more "window" definitions and contains an array of
// secondary TL-UL bus connectors for each window
// Output port for window
output tlul_pkg::tl_h2d_t tl_win_o [1],
input tlul_pkg::tl_d2h_t tl_win_i [1],
// To HW
output uart_reg_pkg::uart_reg2hw_t reg2hw, // Write
input uart_reg_pkg::uart_hw2reg_t hw2reg // Read
);
The sections below describe the hardware functionality of each register type both in terms of the RTL created, and the wires in the structures that will come along with the register.
Overall block diagram
The diagram below gives an overview of the register module, name_reg_top
.
In this diagram, the TL-UL bus is shown on the left. Logic then breaks down individual write requests and read requests based upon the assigned address of the bus requests. Writes that match an address create an internal write enable to an individual register (or collection of registers in the case of a field), and return a successful write response. Reads that match an address return the associated data content for that register. See the section below on requests that don’t match any register address.
In the middle are the collections of registers, which are a function of the hjson
input, and a definition of the functionality of each register (read-only, read-write, etc), detailed below.
These are instantiations of the primitives prim_subreg
, prim_subreg_ext
and prim_subreg_shadow
found in the lowRISC primitive library (lowrisc:prim:all).
These take as inputs the write requests from the bus as well as the hardware struct inputs associated with that register.
They create as output the current state of the register and a potential write enable.
The prim_subreg
module takes a parameter SwAccess
that is used to adjust the implementation to the access type required.
On the right are the typedef structs that gather the q
and qe
s into one output bundle, and receive the bundled d
and de
inputs.
The address decode and TL-UL 1:N adapter shown at the bottom are created only if the register definition includes one or more window:
descriptions.
Each window is given its own TL-UL connection and the implementation must provide a device interface.
It is notable that in the current definition, each field of a register has its own register instantiation.
This is required because the definitions allow unique swaccess
and hwaccess
values per field, but could be done at the register level otherwise.
The individual bundled wires are associated with the fields rather than the full register, so the designer of the rest of the peripheral does not need to know the bit range association of the individual fields.
Error responses
Writes and reads that target addresses that are not mapped within the register list may return an error. Whether an error is returned depends on the target address and the number of address bits used internally:
- Accesses to unmapped target addresses that can be fully represented with the internally used address bits typically return an error.
- Since excess address bits are stripped off during address decoding, other accesses are mapped to the internally available address space. They typically return an error if the resulting address is not mapped. If the resulting address is mapped, the request is forwarded to the corresponding register and no error is returned.
For example, consider a module with registers in the address range 0x0
to 0x84
and that hence uses 8 address bits internally.
- A request to address
0x88
can be represented using the 8 internal address bits but there is no register mapped at this address. Typical implementations return an error for such a request. - In contrast, a request to address
0x184
is mapped to address0x84
internally which is a valid address. No error is returned for this request.
Other error responses include for the following reasons:
- TL-UL
a_opcode
illegal value - TL-UL writes of size smaller than register size
- I.e. writes of size 8b to registers > 8b will cause error (explicitly: if it has field bits within
[31:08]
) - I.e. writes of size 16b to registers > 16b will cause error (explicitly: if it has field bits within
[31:16]
)
- I.e. writes of size 8b to registers > 8b will cause error (explicitly: if it has field bits within
- TL-UL writes of size smaller than 32b that are not word-aligned
- I.e. writes of size 8b or 16b that are not to an address that is 4B aligned return in error.
Reads of size smaller than full word (32b) return the full register content and do not signal error.
Reads response data is always in its byte-channel, i.e. a one-byte read to address 0x3
will return the full word with the correct MSB in bits [31:24]
on the TL-UL response bus (as well as the not-asked-for bytes 2:0 in [23:0]
).
Note with the windowing option, a new TL-UL bus (or more) is spawned and managed outside of this register module.
Any window that makes use of the byte masks will include the byte-write: "true"
keyword in their definition.
Error handling by that TL-UL bus is completely under the control of the logic that manages this bus.
It is recommended to follow the above error rules based on the declared number of validbits
: for the window, but there are some cases where this might be relaxed.
For example, if the termination of the TL-UL bus is a memory that handles byte and halfword writes via masking, errors do not need be returned for unaligned sub-word writes.
Register definitions per type
The definition of what exactly is in each register type is described in this section.
As shown above, the maximally featured register has inputs and outputs to/from both the bus interface side of the design as well as the hardware interface side.
Some register types don’t require all of these inputs and outputs.
For instance, a read-only register does not require write data from the bus interface (this is configured by the SwAccess
parameter to the prim_subreg
module).
The maximally defined inputs to this register block (termed the subreg
from here forward) are given in the table below.
Note that these are instantiated per field, not per register, so the width is the width of the field.
The direction is the Verilog signal definition of subreg
for that type.
name | direction | description |
we
|
input | Write enable from the bus interface, scalar |
wd[]
|
input | Write data from the bus interface, size == field size |
qs[]
|
output | Output to read response data mux, size == field_size. This is typically the same as q[] except for hwext registers.
|
de
|
input | Hardware data enable from peripheral logic, scalar, should be set when the hardware wishes to update the register field |
d[]
|
input | Hardware data from peripheral logic, size == field size |
qe
|
output | Output register enable, scalar, true when bus interface writes to subreg |
q[]
|
output | Output register content, size == field size. This output typically goes to both the hardware bundle output q as well as the software read response mux output qs[] .
|
re
|
output | Indicates to hwext registers that the bus is reading the register. |
Type RW
The first register type is the read-write register, invoked with an hjson
attribute swaccess
type of rw
.
There is a variant of this below, this is the default variant.
This uses the prim_subreg
with the connections shown.
The connectivity to the hardware struct bundles are a function of the hwaccess
and hwqe
attributes, and will be discussed here as well.
In this diagram, the maximum connection for subreg_rw is shown.
Coming in from the left (bus) are the software write enable and write data, which has the highest priority in modifying the register contents.
These are present for all RW types.
The “final answer” for the register content is stored in the subreg module, and presented to the peripheral hardware as the output q
and to bus reads as the output qs
.
Optionally, if the hwaccess
attribute allows writes from the hardware, the hardware can present updated values in the form of data enable (de
) and update data (d
).
If the data enable is true, the register content is updated with the update data.
If both software and hardware request an update in the same clock cycle (i.e. both de
and we
are true), the software updated value is used, as shown in the diagram.
The hwaccess
attribute value does not change the contents of the subreg, but the connections are potentially modified.
The attribute hwaccess
has four potential values, as shown earlier in the document: hrw, hro, hwo, none
.
A hwaccess
value of hrw
means that the hardware wants the ability to update the register content (i.e. needs connection to d
and de
), as well as see the updated output (q
).
hwo
doesn’t care about the output q
, but wants to update the register value.
This is the default for registers marked for software read-only access.
hro
conversely indicates the hardware doesn’t need to update the content, but just wants to see the value written by software.
This is the default for fields where the software access is read-write or write-only.
Finally an attribute value of none
asks for no interface to the hardware, and might be used for things like scratch registers or DV test registers where only software can modify the value, or informational registers like version numbers that are read-only by the software.
Another attribute in the register description hwqe
, when true indicates that the hardware wants to see the software write enable exported to the peripheral logic.
This is just a registered version of the bus side write-enable we
so that its rising edge aligns with the change in the q
output.
There only needs to be one instantiated qe
flop per register, but it is provided in the reg2hw structure for each field.
Type RW HWExt
There is an attribute called hwext
which indicates, when true, that the register value will be maintained outside the auto-generated logic.
It is up to the external logic to implement the correct functionality as indicated by the swaccess
attribute.
In other words, there is no guarantee that the custom logic will correctly implement rw
, or whichever attribute is included in the register definition.
It is expected that this functionality is only needed for custom features outside the scope of the list of supported swaccess features, such as masked writes or access to FIFOs.
Note that these could be done non-hwext as well, with swaccess==rw
and hwaccess=rw
, but could lose atomicity due to the register included within the autogenerated region.
The block diagram below shows the maximally functional hwext
RW
register, with some assumption of the implementation of the register on the outside.
This is implemented by the prim_subreg_ext
module which is implemented with assign
statements just as the wiring shown suggests.
In this diagram the q
is the q
output to the hardware, while the qs
is the output to the read response mux on the software side.
The storage register is shown in the custom portion of the logic.
Finally, note that no de
input is required from the rest of the peripheral hardware, only the d
is added to the struct bundle.
Note the timing of qe
is one cycle earlier in this model than in the non-hwext model.
Type RO, with hwext and zero-gate options
Read-only type registers can be thought of as identical as RW
types with no wd
and we
input.
They are implemented as prim_subreg
with those inputs disabled.
Similarly hwext RO
registers simply pass the d input from the outside world to the data mux for software read response.
There is one special case here [not yet implemented] where swaccess
is ro
and hwaccess
is none
or hro
and hwext
is true.
In this case, a hardwired value is returned for a software read equal to the default value assigned to the register this can be useful for auto-generated register values with no storage register required.
Type RC
Registers of software access type rc
are special cases of RO
, but require an additional signal from the address decode logic.
This signal re
indicates that this register is being read, in which case the contents should be set to zero.
Note this register is not recommended but might be required for backwards compatibility to other IP functionality.
At the moment hwext
is not allowed to be true for RC
since there is no exporting of the re
signal.
If this is required, please add a feature request.
Type WO
Write only registers are variants of prim_subreg
where there is no output back to the software read response mux, so the d
and de
pins are tied off.
If there is no storage required, only an indication of the act of writing, then hwext
should be set to true and the outside hardware can handle the write event.
Type R0W1C, RW1S, RW1C and RW0C
Certain RW
register types must be implemented with special configuration of prim_subreg
since the act of writing causes the values to be set in unique ways.
These types are shown in the block diagrams below.
Type R0W1C
not shown is just a special case of RW1C
where the q output is not sent back to the software read response mux, but the value 0
is sent instead.
Note the qe
is removed for readability but is available with the hwqe attribute.
Simultaneous SW and HW access
As shown in the module descriptions, the subreg needs to handle the case when both hardware and software attempt to write at the same time. As is true with the RW type, the software has precedence, but it is more tricky here. The goal for these types of registers is to have software clear or set certain bits at the same time hardware is clearing or setting other bits. So in theory what software is clearing, hardware is setting, or vice-versa. An example would be where hardware is setting interrupt status bits, and software is clearing them, using RW1C. The logic for RW1C shows how this is implemented in the module:
q <= (de ? d : q) & (we ? ~wd : '1)
In this description if the hardware is writing, its value is sent to the logic that potentially clears that value or the stored value.
So if the hardware accidentally clears fields that the software hasn’t cleared yet, there is a risk that events will not be seen by software.
The recommendation is that the hardware feed the q
value back into d
, only setting bits with new events.
Then there will be no “collision” between hardware setting events and software clearing events.
The HW could have chosen to simply treat d
and de
as set-only, but the preference is to leave the subreg
simple and allow the hardware to do either “the right thing” or whatever it feels is appropriate for its needs.
(Perhaps it is a feature to clear all events in the hardware.)
The one “conflict” that is common and worth mentioning is RW1C
on an interrupt vector.
This is the typical scenario where hardware sets bits (representing an interrupt event), and software clears bits (indicating the event has been handled).
The assumption is that between the hardware setting and software clearing, software has cleaned up whatever caused the event in the first place.
But if the event is still true (the HW d
input is still 1
) then the clear should still have effect for one cycle in order to create a new interrupt edge.
Since d
is still 1
the q
will return to 1
after one cycle, since the clean up was not successful.
HWExt RW1C etc.
It is legal to create RW1C
, RW1S
, etc. with hwext
true.
In these cases the auto-generated hardware is simply the same as the hwext RW
register shown earlier.
This causes all of the implementation to be done outside of the generated register block.
There is no way to guarantee that hardware is doing the right thing, but at least the RW1C
conveys the notion to software the intended effect.
Similarly it is legal to set hwqe
true for any of these register types if the clearing wants to be monitored by the hardware outside.
Shadow Registers
If in the .hjson
file the optional key shadowed
is set to true
for a particular register group, the register tool will implement the corresponding registers including all fields as shadow registers using the prim_subreg_shadow
module.
The following section describes the concept in detail.
Overview
For Comportable designs used in sensitive security environments, for example AES or HMAC peripherals, there may be a need to guard against fault injection attacks. One of the surfaces under threat are critical configuration settings that may be used to control keys, region access control and system security configuration. These registers do not typically contain secret values, but can be vulnerable to fault attacks if a fault can alter the state of a register to a specific state.
The fault attacks targeting these registers can come in a few forms:
- A fault attack directly on the storage element.
- A fault attack on peripheral logic (for example resets) that cause the registers to revert to a default state.
- A fault attack during the process of the write to disrupt the actual written value.
Shadow registers are a mechanism to guard sensitive registers against this specific type of attack. They come at a cost of increased area, and a modified SW interaction, see below. Usage of shadow registers should be evaluated with this impact in mind.
Description
The diagram below visualizes the internals of the prim_subreg_shadow
module used to implement shadow registers.
When compared to a normal register, the prim_subreg_shadow
module is comprised of two additional storage components, giving three storage flops in total.
The staged register holds a written value before being committed.
The shadow register holds a second copy after being committed.
And the standard storage register holds the committed value.
Under the hood, these registers are implemented using the prim_subreg
module.
This means a shadow register can implement any of the types described in this section.
The general behavior of updating a shadow register is as follows:
-
Software performs a first write to the register.
The staged register takes on the new value. The committed register is not updated, so the hardware still sees the old value.
-
Software performs a second write to the register (same address):
- If the data of the second write matches the value in the staged register, both the shadow register and the committed register are updated.
- If the data of the second write does not match the contents stored in the staged register, an update error is signaled to the hardware with a pulse on
err_update
.
The phase is tracked internally to the module. If needed, software can put the shadow register back into the first phase by performing a read operation. If the register is of type
RO
, software cannot interfere with the update process and read operations do not clear the phase tracker. If the register is of typeRW1S
orRW0C
, the staged register will latch the raw data value coming from the bus side. The raw data value of the second write is compared with the raw data value in the staged register before being filtered by theRW1S
orRW0C
set/clear logic. -
If the value of the shadow register and the committed register are ever mismatched, a storage error is signaled to the hardware using the
err_storage
signal.
For added security, the staged and shadow registers store the ones’ complement values of the write data (if the write data is 0x33
, 0xCC
is stored in the staged and shadow register).
This is done such that it is more difficult for a localized glitch to cause shadow and committed registers to become the same value.
Programmer’s model
The programmer needs to be aware of the functional behavior when dealing with shadow registers.
To this end, the naming of registers which contain a shadowed copy is suffixed by _SHADOWED
in the generated collateral.
This is ensured by the register tool.
Software should ensure that shadow registers are always accessed exclusively. If the two write operations required for an update are for some reason interleaved by a third write operation, this can trigger an update error depending on the written data. Such situations can be avoided by using some form of mutex at the driver level for example. Alternatively, software can enforce exclusive shadow register access by temporarily disabling interrupts with a sequence as follows:
- Disable interrupt context.
- Perform 1st write to shadow register.
- Perform 2nd write to shadow register.
- Read back the shadow register to ensure data matches the expected value. This step is not strictly required. It helps to identify update errors and can unveil potential fault attacks on the write data lines during process of the write.
- Restore interrupt context.
By performing a read operation, software can at any time put the shadow register back into the first phase, thereby preventing that a subsequent write triggers an update error, and recovering from situations where it is not clear in which phase the register is.
However, if the register is of type RO
, meaning if only the hardware can update the register, read operations do not clear the phase tracker.
In addition, error status registers can be added to inform software about update errors.
Design integration
The following aspects need to be considered when integrating shadow registers into the system.
-
Non-permissive reset values:
A fault attack on the reset line can reset the staged, shadow and committed registers altogether and not generate an error. It may remain undetected by the system. It is thus highly recommended to use non-permissive reset values and use software for configuring more permissive values.
-
Error hook-ups:
- Storage errors signaled to the hardware in
err_storage
are fatal, and should be hooked up directly into the alert system. - Update errors signaled to the hardware in
err_update
may be software errors and do not affect the current committed value; thus they can be considered informational bugs that are collected into error status or interrupts, and optionally alerts. They do not generate error responses on the TL-UL system bus interface.
- Storage errors signaled to the hardware in
-
Hardware and software cost:
Shadow registers are costly as they imply considerable overheads in terms of circuit area as well as software complexity and performance. They should not be used lightly nor extensively. For example, if a register is reconfigured frequently, using restrictive reset values, possibly combined with parity bits and/or with verification of the register value after writing, might be more suitable. In contrast, for registers that are more static the performance overhead of shadow registers is negligible. Also, for some registers it is not possible to define restrictive reset values, e.g., due to process variation. The use of shadow registers is more justified in such cases. Example use cases for shadow registers are:
- Main control registers of cryptographic accelerators such as AES and HMAC where fault attacks could enforce potentially less secure operating modes.
- Critical and security-related infrastructure such as:
- Alert escalation and sensor configuration registers, and
- Countermeasure tuning and functional/bandgap calibration registers.
DV shadow register alert test automation
In DV, the shadow_reg_errors
automated test will check if shadow registers’ update and storage errors trigger the correct alerts.
This alert automation test requires the user to add the following items in .hjson
file under each shadow register:
- Update_err_alert: Alert triggered by a shadow register’s update error
- Storage_err_alert: Alert triggered by a shadow register’s storage error
Future enhancements
The following features are currently not implemented but might be added in the future.
-
Lightweight implementations:
Depending on the use case and security requirements, lightweight shadow register implementations can be more suitable. Possible options include:
- Storing only a hash value or parity bits in the staged and/or shadow register instead of the full data word. This can help to reduce circuit area overheads but may weaken security.
- Removing the staged register and relying on software to verify the written value. This helps to reduce circuit area but increases software complexity. If lower security can be tolerated, avoiding the read back after a write can make this implementation transparent to software.
- A combination of the two options. For example, removing the staged register and storing parity bits only in the shadow register may be suitable for physical memory protection (PMP) registers.
-
Ones’ complement inversion in software:
The ones’ complement inversion could be done in software. This means software would first write the ones’ complement of the data and only the second write would be the real data. This may make following bus dumps easier because the data values will be different.
-
Shadow resets:
Staged and shadow registers could use a separate shadow reset from the committed register. This would imply that each module which contains shadow registers will receive two reset lines from the reset controller. The benefit is that a localized reset glitch attack will need to hit both reset lines, otherwise a storage error is triggered.
Reset skew may need to be handled separately. Depending on the design, the reset propagation may be multi-cycle, and thus it is possible for the shadow register and committed register to not be reset at the same time when resets are asserted. This may manifest as an issue if the reset skews are not balanced, and may cause register outputs to change on the opposite sides of a clock edge for the receiving domain. It is thus important to ensure the skews of the committed and shadow reset lines to be roughly balanced and/or downstream logic to correctly ignore errors when this happens. Note that a module may need to be reset during operation of the system, i.e., when the alert system is running.
Software-controllable resets should be avoided. If a single reset bit drives both normal and shadow resets, then the problem is simply moved to a different place. If multiple reset bits drive the resets, then it becomes a question of what resets those reset bits, which may itself have the same issue.
Generating C Header Files
The register tool can be used to generate C header files. It is intended that there will be several generators to output different formats of header file.
Simple hello_world test headers
The register generation tool will generate simple headers if it is invoked with the -D
flag.
The -o <file.h>
flag may be used to specify the output file.
As an example the tool can be invoked from the top project directory to generate the uart headers with:
$ ./util/regtool.py -D -o hw/ip/uart.h hw/ip/uart/data/uart.hjson
This format assumes that there is a base address NAME
n_BASE_ADDR
defined where n is an identifying number to allow for multiple instantiations of peripherals.
It provides a definition NAME_REG(n)
that provides the address of the register in instantiation n.
Single-bit fields have a define with their bit offset.
Multi-bit fields have a define for the bit offset and an mask and may have defines giving the enumerated names and values.
For example:
// UART control register
#define UART_CTRL(id) (UART ## id ## _BASE_ADDR + 0x0)
# define UART_CTRL_TX 0
# define UART_CTRL_RX 1
# define UART_CTRL_NF 2
# define UART_CTRL_SLPBK 4
# define UART_CTRL_LLPBK 5
# define UART_CTRL_PARITY_EN 6
# define UART_CTRL_PARITY_ODD 7
# define UART_CTRL_RXBLVL_MASK 0x3
# define UART_CTRL_RXBLVL_OFFSET 8
# define UART_CTRL_RXBLVL_BREAK2 0
# define UART_CTRL_RXBLVL_BREAK4 1
# define UART_CTRL_RXBLVL_BREAK8 2
# define UART_CTRL_RXBLVL_BREAK16 3
Generating documentation
The register tool can be used standalone to generate HTML documentation of the registers. However, this is normally done as part of the Markdown documentation using the special tags to include the register definition file and insert the configuration and register information.