Bit vector fields
Register fields can be of the type bit vector. Meaning, an array of logic bits.
This page will show you how the set up bit vector fields in a register, and will showcase all the code that can be generated from it.
Usage in TOML
The TOML file below shows how to set up a register with two bit vector fields. See comments for rules about the different properties.
 1[conf]
 2
 3mode = "r_w"
 4description = "Configuration register."
 5
 6# This will allocate a bit vector field named "tuser" in the "conf" register.
 7# The "type" property MUST be present and set to "bit_vector".
 8tuser.type = "bit_vector"
 9
10# The "width" property MUST be present for a bit vector field.
11# The value specified MUST be a positive integer.
12tuser.width = 4
13
14# The "description" property is OPTIONAL for a bit vector field.
15# Will default to "" if not specified.
16# The value specified MUST be a string.
17tuser.description = "Value to set for **TUSER** in the data stream."
18
19# The "default_value" property is OPTIONAL for a bit vector field.
20# Will default to all zeros if not specified.
21# The value specified MUST be either
22# 1. A string of 1's and 0's. The string length MUST be the same as the field width.
23# 2. An unsigned integer. The value's binary representation MUST fit within the field width.
24tuser.default_value = "0101"
25
26# The "numerical_interpretation" property is OPTIONAL for a bit vector field.
27# Will default to "unsigned" if not specified.
28# The value specified MUST be either
29# 1. "unsigned"
30# 2. "signed"
31# 3. "unsigned_fixed_point"
32# 4. "signed_fixed_point"
33tuser.numerical_interpretation = "unsigned"
34
35# The "min_bit_index" property is OPTIONAL for a bit vector field.
36# Will default to 0 if not specified.
37# This property is only used when "numerical_interpretation" is "unsigned_fixed_point"
38# or "signed_fixed_point".
39# The property specifies the bit index of the LSB of the fixed point number.
40# Set to a negative value to add fractional bits.
41tuser.min_bit_index = 0
42
43
44tid.type = "bit_vector"
45tid.width = 8
46tid.description = "Value to set for **TID** in the data stream."
47tid.default_value = 0xf3
48
49
50tdest.type = "bit_vector"
51tdest.width = 3
52tdest.description = "Value to set for **TDEST** in the data stream."
Note that the third field does not have any default value specified, meaning it will default to all zeros.
Below you will see how you can parse this TOML file and generate artifacts from it.
Usage with Python API
The Python code below shows
- How to parse the TOML file listed above. 
- How to create an identical register list when instead using the Python API. 
- 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.
 1import sys
 2from pathlib import Path
 3
 4from hdl_registers.generator.c.header import CHeaderGenerator
 5from hdl_registers.generator.cpp.implementation import CppImplementationGenerator
 6from hdl_registers.generator.cpp.interface import CppInterfaceGenerator
 7from hdl_registers.generator.html.page import HtmlPageGenerator
 8from hdl_registers.generator.vhdl.record_package import VhdlRecordPackageGenerator
 9from hdl_registers.generator.vhdl.register_package import VhdlRegisterPackageGenerator
10from hdl_registers.parser.toml import from_toml
11from hdl_registers.register_list import RegisterList
12from hdl_registers.register_modes import REGISTER_MODES
13
14THIS_DIR = Path(__file__).parent
15
16
17def parse_toml() -> RegisterList:
18    """
19    Create the register list by parsing a TOML data file.
20    """
21    return from_toml(name="caesar", toml_file=THIS_DIR.parent / "toml" / "field_bit_vector.toml")
22
23
24def create_from_api() -> RegisterList:
25    """
26    Alternative method: Create the register list by using the Python API.
27    """
28    register_list = RegisterList(name="caesar")
29
30    register = register_list.append_register(
31        name="conf", mode=REGISTER_MODES["r_w"], description="Configuration register."
32    )
33
34    register.append_bit_vector(
35        name="tuser",
36        description="Value to set for **TUSER** in the data stream.",
37        width=4,
38        default_value="0101",
39    )
40
41    register.append_bit_vector(
42        name="tid",
43        description="Value to set for **TID** in the data stream.",
44        width=8,
45        default_value=0xF3,
46    )
47
48    register.append_bit_vector(
49        name="tdest",
50        description="Value to set for **TDEST** in the data stream.",
51        width=3,
52        default_value=0,
53    )
54
55    return register_list
56
57
58def generate(register_list: RegisterList, output_folder: Path) -> None:
59    """
60    Generate the artifacts that we are interested in.
61    """
62    CHeaderGenerator(register_list=register_list, output_folder=output_folder).create()
63
64    CppImplementationGenerator(register_list=register_list, output_folder=output_folder).create()
65    CppInterfaceGenerator(register_list=register_list, output_folder=output_folder).create()
66
67    HtmlPageGenerator(register_list=register_list, output_folder=output_folder).create()
68
69    VhdlRegisterPackageGenerator(register_list=register_list, output_folder=output_folder).create()
70    VhdlRecordPackageGenerator(register_list=register_list, output_folder=output_folder).create()
71
72
73def main(output_folder: Path) -> None:
74    generate(register_list=parse_toml(), output_folder=output_folder / "toml")
75    generate(register_list=create_from_api(), output_folder=output_folder / "api")
76
77
78if __name__ == "__main__":
79    main(output_folder=Path(sys.argv[1]))
See Register.append_bit_vector() and the BitVector class
for more Python API details.
Generated code
See below for a description of the code that can be generated when using bit vector fields.
HTML page
See HTML file below for the human-readable documentation that is produced by the
generate() call in the Python example above.
Each bit vector field is documented with its range, default value and description.
See HTML generator for more details about the HTML generator and its capabilities.
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 generator for instructions on how it can be used in your VHDL project.
Base register package
Some interesting things to notice:
- There is only one register, at index 0. 
- The first field is four bits wide, occupying bits 3 down to 0, while the second one is eight bits wide, occupying but 11 down to 4. 
- For each bit vector field there is a named integer subtype that defines the fields’s bit range within the register. 
- In VHDL, slicing out a range from the register value will yield a value of type - std_ulogic_vector, meaning that typically no casting is needed. Hence there are no conversion functions for bit vector fields, the way there are for e.g. enumeration fields.
Record package
The caesar_regs_down_t type is a record with a member config, the only register in
this example.
The type of the config member is another record with the two bit vectors set up in
our example: tuser and tid.
These are of unsigned vector types defined in the base register package above.
In our VHDL code we can access a field value for example like this:
result_tuser <= regs_down.config.tuser;
C++
The C++ interface header and implementation code below is produced by the generate() call in
the Python example above.
Click the button to expand and view each code block.
The class header is skipped here, since its inclusion would make this page very long. See C++ generator for more details and an example of how the excluded file might look.
C++ interface header
Note the setters and getters for each individual field value.
C++ implementation
Note that each setter performs assertions that the supplied argument is withing the legal range of the field. This will catch calculation errors during testing and at run-time.
C header
The C code below is produced by the generate() call in the Python example above.
The range and mask of the each field are available as constants.