Default registers

Many projects use a few default registers in standard locations that shall be present in all modules. For example, very commonly the first register of a module is an interrupt status register and the second one is an interrupt mask. In order to handle this, without having to duplicate names and descriptions in many places, there is a default_registers flag to the from_toml() function. Passing a list of Register objects will insert these registers first in the RegisterList.

Usage in TOML

The TOML file below is used to showcase insertion of default registers.

TOML for showcasing default registers.
 1# Adapt the "interrupt_status" default register for the needs of this register list.
 2[interrupt_status]
 3
 4# The "mode" property MUST NOT be present for a default register.
 5# The mode of the register is set in the default register object in Python.
 6
 7# The "description" property is OPTIONAL for a default register.
 8# If specified, it will only be set in this register list, not other register lists that also
 9# use this default register.
10# If not specified, the description of the default register object in Python will be used.
11description = "Interrupt status for my module."
12
13# Register fields can be added freely to a default register.
14# Will only be added to this register list, not other register lists that also use
15# this default register.
16overflow.type = "bit"
17overflow.description = "Too high data rate."
18
19underflow.type = "bit"
20underflow.description = "Too low data rate."
21
22# Note that the "interrupt_mask" default register is not adapted here.
23# It will be included in this register list at its default configuration.
24
25# Further registers and register arrays can be added freely after the default registers.
26[config]
27
28mode = "r_w"
29description = "Generic configuration register."

Usage with Python API

The Python code below shows

  1. How to parse the TOML file listed above.

  2. How to create an identical register list when instead using the Python API.

  3. How to generate register artifacts.

Note that the result of the create_from_api call is identical to that of the parse_toml call. Meaning that using a TOML file or using the Python API is completely equivalent. You choose yourself which method you want to use in your code base.

Python code to showcase default registers.
 1# Standard libraries
 2import sys
 3from pathlib import Path
 4
 5# First party libraries
 6from hdl_registers.generator.c.header import CHeaderGenerator
 7from hdl_registers.generator.cpp.interface import CppInterfaceGenerator
 8from hdl_registers.generator.html.page import HtmlPageGenerator
 9from hdl_registers.generator.vhdl.register_package import VhdlRegisterPackageGenerator
10from hdl_registers.parser.toml import from_toml
11from hdl_registers.register import Register
12from hdl_registers.register_list import RegisterList
13from hdl_registers.register_modes import REGISTER_MODES
14
15THIS_DIR = Path(__file__).parent
16
17
18DEFAULT_REGISTERS = [
19    Register(
20        name="interrupt_status",
21        index=0,
22        mode=REGISTER_MODES["r_wpulse"],
23        description="Interrupt status. Clear interrupt(s) by writing the corresponding bitmask.",
24    ),
25    Register(
26        name="interrupt_mask",
27        index=1,
28        mode=REGISTER_MODES["r_w"],
29        description="Enable or disable interrupts by setting bitmask.",
30    ),
31]
32
33
34def parse_toml() -> RegisterList:
35    """
36    Create the register list by parsing a TOML data file.
37    """
38    return from_toml(
39        name="caesar",
40        toml_file=THIS_DIR.parent / "toml" / "basic_feature_default_registers.toml",
41        default_registers=DEFAULT_REGISTERS,
42    )
43
44
45def create_from_api() -> RegisterList:
46    """
47    Alternative method: Create the register list by using the Python API.
48    """
49    register_list = RegisterList.from_default_registers(
50        name="caesar", source_definition_file=None, default_registers=DEFAULT_REGISTERS
51    )
52
53    interrupt_status = register_list.get_register(register_name="interrupt_status")
54    interrupt_status.description = "Interrupt status for my module."
55    interrupt_status.append_bit(
56        name="overflow", description="Too high data rate.", default_value="0"
57    )
58    interrupt_status.append_bit(
59        name="underflow", description="Too low data rate.", default_value="0"
60    )
61
62    register_list.append_register(
63        name="config", mode=REGISTER_MODES["r_w"], description="Generic configuration register."
64    )
65
66    return register_list
67
68
69def generate(register_list: RegisterList, output_folder: Path):
70    """
71    Generate the artifacts that we are interested in.
72    """
73    CHeaderGenerator(register_list=register_list, output_folder=output_folder).create()
74    CppInterfaceGenerator(register_list=register_list, output_folder=output_folder).create()
75    HtmlPageGenerator(register_list=register_list, output_folder=output_folder).create()
76    VhdlRegisterPackageGenerator(register_list=register_list, output_folder=output_folder).create()
77
78
79def main(output_folder: Path):
80    generate(register_list=parse_toml(), output_folder=output_folder / "toml")
81    generate(register_list=create_from_api(), output_folder=output_folder / "api")
82
83
84if __name__ == "__main__":
85    main(output_folder=Path(sys.argv[1]))

See RegisterList.from_default_registers() for more Python API details.

Generated code

See below for a description of the code that can be generated when using register arrays.

HTML page

See HTML file below for the human-readable documentation that is produced by the generate() call in the Python example above.

See HTML code generator for more details about the HTML generator and its capabilities.

HTML page

VHDL package

The VHDL code below is produced by the generate() call in the Python example above. Click the button to expand and view the code. See VHDL code generator for instructions on how it can be used in your VHDL project.

Click to expand/collapse code.
Generated VHDL code.
 1-- -----------------------------------------------------------------------------
 2-- This file is automatically generated by hdl-registers version 7.0.2-dev.
 3-- Code generator VhdlRegisterPackageGenerator version 2.0.0.
 4-- Generated 2025-01-21 20:52 at commit 3c3e6c67d817.
 5-- Register hash 8c0e8391ad087478739e47ddc075f056e1c63641.
 6-- -----------------------------------------------------------------------------
 7
 8library ieee;
 9use ieee.std_logic_1164.all;
10use ieee.numeric_std.all;
11use ieee.fixed_pkg.all;
12
13library register_file;
14use register_file.register_file_pkg.all;
15
16
17package caesar_regs_pkg is
18
19  -- ---------------------------------------------------------------------------
20  -- The valid range of register indexes.
21  subtype caesar_register_range is natural range 0 to 2;
22
23  -- ---------------------------------------------------------------------------
24  -- The number of bits needed to address all 3 registers on a register bus.
25  -- Note that this figure includes the lowest two address bits that are assumed zero, since
26  -- registers are 32-bit and unaligned accesses are not supported.
27  constant caesar_address_width : positive := 4;
28
29  -- Register indexes, within the list of registers.
30  constant caesar_interrupt_status : natural := 0;
31  constant caesar_interrupt_mask : natural := 1;
32  constant caesar_config : natural := 2;
33
34  -- Declare 'register_map' and 'regs_init' constants here but define them in
35  -- the package body (deferred constants).
36  -- So that functions have been elaborated when they are called.
37  -- Needed for ModelSim compilation to pass.
38
39  -- To be used as the 'registers' generic of 'axi_lite_register_file.vhd'.
40  constant caesar_register_map : register_definition_vec_t(caesar_register_range);
41
42  -- To be used for the 'regs_up' and 'regs_down' ports of 'axi_lite_register_file.vhd'.
43  subtype caesar_regs_t is register_vec_t(caesar_register_range);
44  -- To be used as the 'default_values' generic of 'axi_lite_register_file.vhd'.
45  constant caesar_regs_init : caesar_regs_t;
46
47  -- To be used for the 'reg_was_read' and 'reg_was_written' ports of 'axi_lite_register_file.vhd'.
48  subtype caesar_reg_was_accessed_t is std_ulogic_vector(caesar_register_range);
49
50  -- -----------------------------------------------------------------------------
51  -- Fields in the 'interrupt_status' register.
52  -- Range of the 'overflow' field.
53  constant caesar_interrupt_status_overflow : natural := 0;
54  -- Default value of the 'overflow' field.
55  constant caesar_interrupt_status_overflow_init : std_ulogic := '0';
56
57  -- Range of the 'underflow' field.
58  constant caesar_interrupt_status_underflow : natural := 1;
59  -- Default value of the 'underflow' field.
60  constant caesar_interrupt_status_underflow_init : std_ulogic := '0';
61
62end package;
63
64package body caesar_regs_pkg is
65
66  constant caesar_register_map : register_definition_vec_t(caesar_register_range) := (
67    0 => (index => caesar_interrupt_status, mode => r_wpulse, utilized_width => 2),
68    1 => (index => caesar_interrupt_mask, mode => r_w, utilized_width => 32),
69    2 => (index => caesar_config, mode => r_w, utilized_width => 32)
70  );
71
72  constant caesar_regs_init : caesar_regs_t := (
73    0 => "00000000000000000000000000000000",
74    1 => "00000000000000000000000000000000",
75    2 => "00000000000000000000000000000000"
76  );
77
78end package body;

C++ interface header

The C++ interface header code below is produced by the generate() call in the Python example above. Click the button to expand and view the code.

Click to expand/collapse code.
Generated C++ interface class code.
  1// -----------------------------------------------------------------------------
  2// This file is automatically generated by hdl-registers version 7.0.2-dev.
  3// Code generator CppInterfaceGenerator version 1.0.0.
  4// Generated 2025-01-21 20:52 at commit 3c3e6c67d817.
  5// Register hash 8c0e8391ad087478739e47ddc075f056e1c63641.
  6// -----------------------------------------------------------------------------
  7
  8#pragma once
  9
 10#include <sstream>
 11#include <cstdint>
 12#include <cstdlib>
 13
 14namespace fpga_regs
 15{
 16
 17  // Attributes for the 'overflow' field in the 'interrupt_status' register.
 18  namespace caesar::interrupt_status::overflow
 19  {
 20    static const auto width = 1;
 21    static const auto default_value = 0b0;
 22  }
 23  // Attributes for the 'underflow' field in the 'interrupt_status' register.
 24  namespace caesar::interrupt_status::underflow
 25  {
 26    static const auto width = 1;
 27    static const auto default_value = 0b0;
 28  }
 29
 30  class ICaesar
 31  {
 32  public:
 33    // Number of registers within this register list.
 34    static const size_t num_registers = 3uL;
 35
 36    virtual ~ICaesar() {}
 37
 38    // -------------------------------------------------------------------------
 39    // Methods for the 'interrupt_status' register. Mode 'Read, Write-pulse'.
 40
 41    // Getter that will read the whole register's value over the register bus.
 42    virtual uint32_t get_interrupt_status() const = 0;
 43
 44    // Setter that will write the whole register's value over the register bus.
 45    virtual void set_interrupt_status(
 46      uint32_t register_value
 47    ) const = 0;
 48
 49    // Getter for the 'overflow' field in the 'interrupt_status' register,
 50    // which will read register value over the register bus.
 51    virtual uint32_t get_interrupt_status_overflow() const = 0;
 52    // Getter for the 'overflow' field in the 'interrupt_status' register,
 53    // given a register value.
 54    virtual uint32_t get_interrupt_status_overflow_from_value(
 55      uint32_t register_value
 56    ) const = 0;
 57    // Setter for the 'overflow' field in the 'interrupt_status' register,
 58    // which will set the field to the given value, and everything else to default.
 59    virtual void set_interrupt_status_overflow(
 60      uint32_t field_value
 61    ) const = 0;
 62    // Setter for the 'overflow' field in the 'interrupt_status' register,
 63    // given a register value, which will return an updated value.
 64    virtual uint32_t set_interrupt_status_overflow_from_value(
 65      uint32_t register_value,
 66      uint32_t field_value
 67    ) const = 0;
 68
 69    // Getter for the 'underflow' field in the 'interrupt_status' register,
 70    // which will read register value over the register bus.
 71    virtual uint32_t get_interrupt_status_underflow() const = 0;
 72    // Getter for the 'underflow' field in the 'interrupt_status' register,
 73    // given a register value.
 74    virtual uint32_t get_interrupt_status_underflow_from_value(
 75      uint32_t register_value
 76    ) const = 0;
 77    // Setter for the 'underflow' field in the 'interrupt_status' register,
 78    // which will set the field to the given value, and everything else to default.
 79    virtual void set_interrupt_status_underflow(
 80      uint32_t field_value
 81    ) const = 0;
 82    // Setter for the 'underflow' field in the 'interrupt_status' register,
 83    // given a register value, which will return an updated value.
 84    virtual uint32_t set_interrupt_status_underflow_from_value(
 85      uint32_t register_value,
 86      uint32_t field_value
 87    ) const = 0;
 88
 89    // -------------------------------------------------------------------------
 90    // Methods for the 'interrupt_mask' register. Mode 'Read, Write'.
 91
 92    // Getter that will read the whole register's value over the register bus.
 93    virtual uint32_t get_interrupt_mask() const = 0;
 94
 95    // Setter that will write the whole register's value over the register bus.
 96    virtual void set_interrupt_mask(
 97      uint32_t register_value
 98    ) const = 0;
 99
100    // -------------------------------------------------------------------------
101    // Methods for the 'config' register. Mode 'Read, Write'.
102
103    // Getter that will read the whole register's value over the register bus.
104    virtual uint32_t get_config() const = 0;
105
106    // Setter that will write the whole register's value over the register bus.
107    virtual void set_config(
108      uint32_t register_value
109    ) const = 0;
110
111  };
112
113} /* namespace fpga_regs */

C header

The C code below is produced by the generate() call in the Python example above. Click the button to expand and view the code.

Click to expand/collapse code.
Generated C code.
 1// -----------------------------------------------------------------------------
 2// This file is automatically generated by hdl-registers version 7.0.2-dev.
 3// Code generator CHeaderGenerator version 1.0.0.
 4// Generated 2025-01-21 20:52 at commit 3c3e6c67d817.
 5// Register hash 8c0e8391ad087478739e47ddc075f056e1c63641.
 6// -----------------------------------------------------------------------------
 7
 8#ifndef CAESAR_REGS_H
 9#define CAESAR_REGS_H
10
11
12// Number of registers within this register list.
13#define CAESAR_NUM_REGS (3u)
14
15// Type for this register list.
16typedef struct caesar_regs_t
17{
18  // Mode "Read, Write-pulse".
19  uint32_t interrupt_status;
20  // Mode "Read, Write".
21  uint32_t interrupt_mask;
22  // Mode "Read, Write".
23  uint32_t config;
24} caesar_regs_t;
25
26// Address of the 'interrupt_status' register.
27// Mode 'Read, Write-pulse'.
28#define CAESAR_INTERRUPT_STATUS_INDEX (0u)
29#define CAESAR_INTERRUPT_STATUS_ADDR (4u * CAESAR_INTERRUPT_STATUS_INDEX)
30// Attributes for the 'overflow' field in the 'interrupt_status' register.
31#define CAESAR_INTERRUPT_STATUS_OVERFLOW_SHIFT (0u)
32#define CAESAR_INTERRUPT_STATUS_OVERFLOW_MASK (0b1u << 0u)
33#define CAESAR_INTERRUPT_STATUS_OVERFLOW_MASK_INVERSE (~CAESAR_INTERRUPT_STATUS_OVERFLOW_MASK)
34// Attributes for the 'underflow' field in the 'interrupt_status' register.
35#define CAESAR_INTERRUPT_STATUS_UNDERFLOW_SHIFT (1u)
36#define CAESAR_INTERRUPT_STATUS_UNDERFLOW_MASK (0b1u << 1u)
37#define CAESAR_INTERRUPT_STATUS_UNDERFLOW_MASK_INVERSE (~CAESAR_INTERRUPT_STATUS_UNDERFLOW_MASK)
38
39// Address of the 'interrupt_mask' register.
40// Mode 'Read, Write'.
41#define CAESAR_INTERRUPT_MASK_INDEX (1u)
42#define CAESAR_INTERRUPT_MASK_ADDR (4u * CAESAR_INTERRUPT_MASK_INDEX)
43
44// Address of the 'config' register.
45// Mode 'Read, Write'.
46#define CAESAR_CONFIG_INDEX (2u)
47#define CAESAR_CONFIG_ADDR (4u * CAESAR_CONFIG_INDEX)
48
49#endif // CAESAR_REGS_H