

### **Cocotb Presentation**

Introduction to Cocotb

Timothée Charrier timothee.charrier@elsys-design.com

Elsys Design Advans Group

January 30, 2025









# **Agenda**

- 1. Introduction
- 2. Basic Example with Combinational Logic
- 3. Writing Testbenches
- 4. Example with Clocked Logic

# 1. Introduction

### 1.1 Introduction

References

- [1] Robin Müller. Python based FPGA verification using CocoTB. Tech. rep. University of Applied Sciences and Arts Northwestern Switzerland, July 2024. URL: https://github.com/m47812/CocoTb\_Example/tree/main.
- [2] FOSSi Foundation. CocoTB. Version 1.9.2. Oct. 26, 2024. URL: https://github.com/cocotb/cocotb.
- [3] Ben Rosser. Cocotb: a Python-based digital logic verification framework. Tech. rep. University of Pennsylvania, Dec. 1028. URL: https://indico.cern.ch/event/776422/attachments/1769690/2874927/cocotb\_talk.pdf.

### 1.2 Introduction

What is Cocotb?

#### Definition

- COroutine based CO-simulation TestBench (COCOTB) is an open-source Python library for digital design verification.
- Enables writing testbenches in Python as an alternative to traditional testbenches in VHDL, Verilog, or SystemVerilog.
- Provides a high-level, Pythonic interface to interact with and validate RTL designs.
- Allows seamless simulation of VHDL and Verilog designs entirely in Python.
- Similar in philosophy to Universal Verification Methodology (UVM), but leverages the simplicity and power of Python.
- Encourages reusable, modular, and scalable verification code.

### 1.3 Introduction

Verification in Microelectronics

#### **Key Points**

- Verification is a crucial step in the design of digital systems.
- Ensures that the design meets the specifications.
- Verification is a time-consuming task.
- Verification is a complex task.



Figure: V-Cycle in Microelectronics

### 1.4 Introduction

**Existing Solutions** 

#### Tools and Frameworks for Verification

- VHDL/Verilog/SystemVerilog testbenches.
- Universal Verification Methodology (UVM).
- Open Source VHDL Verification Methodology (OSVVM).
- C++ or SystemC.
- Python-based testbenches.

### 1.5 Introduction

Cocotb Architecture Overview (adapted from [1] and [2])



#### **Key Points**

- The simulator runs the RTL design.
- Uses the VHDL Procedural Interface (VHPI) or Verilog Procedural Interface (VPI) interfaces.

### 1.6 Introduction

Python and Simulator Interaction (adapted from [1])





### 1.7 Installation

Requirements

#### Requirements

- Python 3.6 or later.
- GNU Make 3 or later.
- VHDL/Verilog simulator (e.g. GHDL, NVC, Verilator, ModelSim, Cadence, Synopsys, etc.).

See the Simulator Support page for more information.

### 1.8 Installation

**Good Practices** 

#### Recommendations

- Use a Python virtual environment:
  - \$ python3 -m venv .venv
  - \$ source .venv/bin/activate
- Use a linter for any code you write:
  - Ruff, Black, Flake8, etc. for Python.
  - vhdl-ls for VHDL code, Verible for Verilog/SystemVerilog code.

# 2. Basic Example with Combinational Logic

## 2.1 Basic Example with Combinational Logic

HDL Design and Python Testbench Part 1

```
Verilog Adder (adder.sv)

module adder #(
    parameter integer DATA_WIDTH = 4
) (
    input logic unsigned [DATA_WIDTH-1:0] X,
    input logic unsigned [DATA_WIDTH-1:0] Y,
    output logic unsigned [DATA_WIDTH:0] SUM
);

assign SUM = X + Y;
endmodule
```

```
— Python Testbench (test adder.py) —
from random import randint
import cocotb
from cocotb.triggers import Timer
@cocotb.test()
async def test adder 10 random values(dut):
   dut X value = 0
   dut V value = 0
   await Timer(2, units="ns")
   assert dut.SUM.value == 0. "Error: 0 + 0 != 0"
    # Get generic value for DATA WIDTH
   DATA WIDTH = int(dut.DATA WIDTH.value)
   for i in range(10):
       X = randint(0, 2**DATA_WIDTH - 1)
       Y = randint(0, 2**DATA_WIDTH - 1)
       dut.X.value = X
       dut. V. value = V
       await Timer(2, units="ns")
       assert dut.SUM.value == X + Y. f"Error: {X} + {Y} != {X + Y}"
```

# 2.2 Basic Example with Combinational Logic

Some Explanations on the Testbench

See this repository for the complete example (under src/rtl/example and src/bench/example).

- @cocotb.test() is a decorator to mark a function as a test.
- async is used to define a coroutine.
- .value = some\_value is used to assign a value to a signal.
- value is used to read the value of a signal.
- await is used to wait for a coroutine to finish.
- assert is used to test a condition.

```
Python Testbench (test adder.py) _
from random import randint
import cocotb
from cocotb.triggers import Timer
@cocotb.test()
async def test_adder_10_random_values(dut):
   dut.X.value = 0
   dut.Y.value = 0
   await Timer(2, units="ns")
   assert dut.SUM.value == 0. "Error: 0 + 0 != 0"
    # Get generic value for DATA WIDTH
   DATA WIDTH = int(dut.DATA WIDTH.value)
   for i in range(10):
       X = randint(0. 2**DATA WIDTH - 1)
       Y = randint(0. 2**DATA WIDTH - 1)
       dut.X.value = X
       dut V value = V
       await Timer(2, units="ns")
       assert dut.SUM.value == X + Y, f"Error: {X} + {Y} != {X + Y}"
```

# 2.3 How to invoke the test (1/2)?

Using the Makefile

```
_ Makefile invoking the adder test _
# Makefile for simulating Verilog code with cocotb and Verilator
# Directory containing Verilog source files
                   := ../../hdl/adder
SRC DIR
# Source Files
VERTLOG SOURCES
                    := $(SRC DIR)/adder.v
# Cocotb and Verilator configuration
STM
                    := verilator
TOPLEVEL LANG
                   := verilog
TOPLEVEL
                   := adder
COCOTB_TEST_MODULES := test adder
                   := $(COCOTB TEST MODULES)
MODULE
EXTRA ARGS
                   += --trace --trace-fst --trace-structs -GDATA WIDTH=8
# Calling cocotb
                    := $(shell pwd)
DMD
                   := $(PWD)/../model:$(PYTHONPATH)
export PYTHONPATH
include $(shell cocotb-config --makefiles)/Makefile.sim
clean..
 rm -rf results.xml __pycache__ sim_build dump.fst
```

- VERILOG\_SOURCES is the list of Verilog files to compile.
- SIM is the simulator to use.
- TOPLEVEL\_LANG is the language of the top-level module.
- TOPLEVEL is the name of the top-level module.
- COCOTB\_TEST\_MODULE and MODULE are the name of the python test module.
- EXTRA\_ARGS are the extra arguments to pass to the simulator (generics, trace, etc.).

# 2.4 How to invoke the test (2/2)?

Using the Python test runner

```
— Python Runner Part 1 —
def test adder runner() -> None:
    # Define the simulator to use
   default simulator: str = "verilator"
    # Build Arguments
   build_args: list[str] = ["-j", "0", "-Wall"]
    # Define the library
   library: str = "LIB RTL"
    # Define the rtl path
   rtl_path: Path = (Path(__file__).parent.parent / "rtl/").resolve
   \hookrightarrow \stackrel{-}{\circ}
    # Define the sources
   sources: list[str] = [f"{rtl path}/adder.sv"]
    # Define the name of the top-level entity
   entity = "adder"
    # Parameters
   parameters: dict[str, int] = {"DATA WIDTH": 8}
```

```
Python Runner Part 2 ____
trv:
    # Get simulator name from environment
    simulator: str = os.environ.get("SIM", default=default simulator
   # Initialize the test runner
    runner: Simulator = get_runner(simulator_name=simulator)
    # Build HDL sources
    runner build(
        build dir="sim build".
        build args=build args.
        clean=True.
        hdl library=library.
        hdl toplevel=entity.
        parameters=parameters.
        sources=sources.
        waves=True.
```

# 2.5 How to invoke the test (3/3)?

Using the Python test runner (continued)

```
Python Runner Part 3 _____
        # Run tests
       runner.test(
           build dir="sim build".
           hdl_toplevel=entity,
           hdl toplevel library=library.
           test_module=f"test_{entity}".
           waves=True.
       # Log the wave file path
       wave file: Path = (Path("sim build") / "dump.vcd").resolve()
       sys.stdout.write(f"Wave file: {wave file}\n")
   except Exception as e:
       raise RuntimeError(f"Simulation failed: {str(e)}") from e
if name == " main ":
   test_adder_runner()
```

### 2.6 Build Flows Comparison

Makefile vs. Python Test Runner

#### Makefile Flow

- Widely used and well-documented.
- Involves writing a Makefile and executing make in the terminal.
- Conceptually similar to the Python test runner but requires additional tools and setup.

#### **Python Test Runner Flow**

Cocotb Presentation • Basic Example with Combinational Logic 0000000

- A newer and experimental approach.
- Allows running tests directly using Python scripts or pytest.
- Integrates seamlessly with Python testing frameworks such as pytest.
- Requires fewer tools (only Python and a simulator).
- Simplifies CI/CD pipeline integration by using pytest instead of navigating directories and executing make.

## 2.7 Running the Test

Output of the Adder Test

| C | Commands to Run the Test |                                                         |  |
|---|--------------------------|---------------------------------------------------------|--|
|   | Method                   | Command                                                 |  |
|   | Makefile                 | make                                                    |  |
|   | Python Test Runner       | <pre>python test_adder.py or pytest test_adder.py</pre> |  |

|                                                                                          |                                     | Terminal Output of the                                                                                                                                                                | Adder T                 | est    |                 |                |     |
|------------------------------------------------------------------------------------------|-------------------------------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|-------------------------|--------|-----------------|----------------|-----|
| 0.00ns INFO<br>0.00ns INFO<br>0.00ns INFO<br>0.00ns INFO<br>22.00ns INFO<br>22.00ns INFO | cocotb.regression cocotb.regression | Running on Verilator version 5.029 devel<br>Seeding Python random module with 17314215<br>Found test test_adder.test_adder_10_random<br>running test_adder (1/1)<br>test_adder passed | 38<br>_values<br>****** | *****  |                 |                |     |
|                                                                                          |                                     | ***********                                                                                                                                                                           |                         |        | TELLE TITLE (5) | 144110 (115/5) |     |
|                                                                                          |                                     | ** test_adder.test_adder_10_random_values  ***********************************                                                                                                        | PASS                    | 22.00  | 0.00            | 27744.70       |     |
|                                                                                          |                                     | ** TESTS=1 PASS=1 FAIL=0 SKIP=0                                                                                                                                                       |                         | 22.00  | 0.04            | 550.57         | **  |
|                                                                                          |                                     | ***********                                                                                                                                                                           | *****                   | ****** | *****           | *****          | *** |

- :0: Verilog \$finish

INFO: Results file: /home/tim/Project/verilog/sim\_build/results.xml Waveform file: /home/tim/Project/verilog/sim\_build/dump.fst

# 2.8 Waveform Debugging Overview

Generate and view waveforms for debugging:

| Enabling Waveform Generation |               |                                           |
|------------------------------|---------------|-------------------------------------------|
|                              | Method        | Configuration                             |
|                              | Makefile      | WAVES := 1                                |
|                              | Python Runner | <pre>runner.test(waves=True,\ldots)</pre> |

| Simulator-Specific FST Options |           |                             |
|--------------------------------|-----------|-----------------------------|
|                                | Simulator | Options                     |
| ·                              | Verilator | tracetrace-fsttrace-structs |
|                                | NVC       | wave=wave_dump.fst          |

Use tools like GTKWave or Surfer to view waveforms if using Open Source simulators.

# 3. Writing Testbenches

## 3.1 Time Management

Simulation Triggers (Part 1/2)

#### **Trigger Overview**

- Independent Simulation: The simulator runs the design and the testbench concurrently.
- Communication: Interaction occurs through VPI/VHPI interfaces, managed via cocotb triggers.
- Execution and Time:
  - During Python execution, simulation time is paused.
  - The testbench halts at a trigger, waiting for the specified condition to be met before continuing.
- To use a trigger, a coroutine should await it.

### **3.**2 Time Management Simulation Triggers (Part 2/2)

| Trigger                                         | Description                                 |  |  |
|-------------------------------------------------|---------------------------------------------|--|--|
| Edge(signal)                                    | Waits for any edge transition of the signal |  |  |
| RisingEdge(signal)                              | Waits for a rising edge of the signal       |  |  |
| FallingEdge(signal)                             | Waits for a falling edge of the signal      |  |  |
| ClockCycles(signal,<br>num_cycles, rising=True) | Waits for a number of clock cycles          |  |  |
| Timer(time, units)                              | Waits for a certain amount of time          |  |  |
| Trigger Usage Examples                          |                                             |  |  |
| wait RisingEdge(dut.clock)                      | # Wait for next clock edge                  |  |  |
| <pre>wait ClockCycles(dut.clock, 5)</pre>       | # Wait for 5 clock cycles                   |  |  |
| wait Edge(dut.s_axis_tready)                    | # Wait for any edge on tready               |  |  |
| <pre>wait Timer(100, units='ns')</pre>          | # Wait for 100 nanoseconds                  |  |  |

### 3.3 Concurrent and Sequential Execution

Coroutines in Python (Part 1/2)

#### Coroutines Overview

- Coroutines are functions that can pause and resume execution.
- Concurrency:
  - Multiple coroutines can run concurrently.
  - Use await to pause a coroutine until a trigger is satisfied.
  - Use cocotb.start or cocotb.start\_soon to run a coroutine concurrently, allowing the current coroutine to continue executing.

## 3.4 Concurrent and Sequential Execution

Coroutines in Python (2/2)

Following example from [2] shows how to run a coroutine concurrently:

```
Example of Concurrent and Sequential Testbench -
# A coroutine
async def reset_dut(reset_n, duration_ns):
   reset n.value = 0
   await Timer(duration ns. units="ns")
   reset n.value = 1
   reset_n._log.debug("Reset complete")
@cocotb.test()
async def parallel_example(dut):
   reset n = dut.reset
   # Execution will block until reset dut has completed
   await reset dut(reset n. 500)
   dut. log.debug("After reset")
   # Run reset dut concurrently
   reset thread = cocotb start soon(reset dut(reset n. duration ns=500))
   # This timer will complete before the timer in the concurrently executing "reset thread"
   await Timer(250, units="ns")
   dut. log.debug("During reset (reset n = %s)" % reset n.value)
   # Wait for the other thread to complete
   await reset thread
   dut. log.debug("After reset")
```

Page 21 / 38

### 3.5 Number Representation

How to correctly assign values to signals

#### Number Representation

- Cocotb allows assigning both signed and unsigned values to signals using a Python int.
- The range of values is determined by the width of the signal:

$$-2^{(Nbits-1)} \le \mathsf{value} \le 2^{Nbits} - 1$$

- Assigning out-of-range values raises an OverflowError.
- Example with our adder and DATA\_WIDTH = 4:

```
dut.X.value = 15  # valid
dut.X.value = -4  # valid (in range for 4 bits, even if X is unsigned)
dut.X.value = 16  # raises OverflowError
dut.X.value = -9  # raises OverflowError
```

# 3.6 Reading Values from Signals

Accessing and Interpreting Values

#### Value Types

- Access values in the DUT using the value property of a handle object.
- The Python type of a value depends on the handle's HDL type:
  - BinaryValue for arrays of logic and subtypes (e.g., std\_logic, std\_logic\_vector in VHDL, logic, bit, bit\_vector in SystemVerilog).
  - list for arrays and subtypes.
  - int for integer nets and constants.
  - float for floating point nets and constants.
  - bool for boolean nets and constants.
  - bytes for string nets and constants.
- For constrained or unconstrained arrays, cocotb creates a list of BinaryValue objects.

# 3.7 Finding elements in the design

#### **Finding Elements**

- To find elements of the DUT (for example, instances, signals, or constants) at a certain hierarchy level, we can use the dir() function.
- Here is the output of dir() for the DUT in the adder example:

```
Terminal Output of dir(dut) for the Adder

['DATA_WIDTH', 'SUM', 'X', 'Y', '_len', '_log', '_name',
...,
'_path', '_type', 'get_definition_file', 'get_definition_name']
```

 We can then access the elements directly using the names in the list (e.g., dut.SUM.value), even internal signals, instances or constants.

# 3.8 Forcing and Freezing Signals

Changing Signal Values (example from [2])

#### Forcing and Freezing Signals

Cocotb provides a way to force and freeze signals to specific values.

```
Examples of Forcing and Freezing Signals
# Deposit action
dut.mv signal.value = 12
dut.my signal.value = Deposit(12) # equivalent syntax
# Force action
dut.mv signal.value = Force(12)
                                 # my signal stays 12 until released
# Release action
dut.mv signal.value = Release() # Reverts any force/freeze assignments
# Freeze action
dut.my_signal.value = Freeze()

→ # mu signal staus at current value until released
```

# 3.9 Logging and Debugging

The logging library

#### Logging Levels

- Cocotb uses the Python logging library to manage logging.
- The logging level for cocotb logs is set based on the COCOTB\_LOG\_LEVEL environment variable.
- The default logging level is INFO.
- The logging levels are, in order of increasing severity:
  - DEBUG
  - INFO
  - WARNING
  - ERROR
  - CRITICAL
- It is very useful to anything from debugging to tracing the simulation (e.g. dut.\_log.info(f"DATA\_WIDTH={dut.DATA\_WIDTH.value})").

## 3.10 Using an HDL Library

Custom Libraries with a Makefile

#### **HDL** Libraries

- Use the TOPLEVEL\_LIBRARY variable to specify the library name.
- Most simulators will try to automatically determine the compilation order. However, it is recommended to specify the order of compilation explicitly:

### 3.11 Using an HDL Library

Custom Libraries with a Python Test Runner

#### **HDL** Libraries

- Use hdl\_library=LIBRARY\_NAME in the runner.build() and runner.test() methods.
- For example, to compile the ascon module, use:

```
Specifying the Compilation Order with Multiple Files sources: list[str] = [
    f"{rtl_path}/ascon_pkg.sv",
    f"{rtl_path}/addition_layer/addition_layer.sv",
    f"{rtl_path}/substitution_layer/sbox.sv",
    f"{rtl_path}/substitution_layer/substitution_layer.sv",
    f"{rtl_path}/diffusion_layer/diffusion_layer.sv",
    f"{rtl_path}/xor/xor_begin.sv",
    f"{rtl_path}/xor/xor_end.sv",
    f"{rtl_path}/permutation/permutation.sv",
    f"{rtl_path}/fsm/ascon_fsm.sv",
    f"{rtl_path}/fsm/ascon_scon.sv"]
```

# 4. Example with Clocked Logic

## 4.1 Testing a Clocked Design

An HDI Counter

```
_ Verilog Adder (counter.sv) _
`timescale 1ns / 1ps
module counter #(
   parameter integer DATA_WIDTH = 8, // Counter width
   parameter integer COUNT_FROM = 0,
                                                   // Initial value
   parameter integer COUNT_TO = 2 ** (DATA_WIDTH - 1), // Terminal value
   parameter integer STEP = 1 // Increment step
   input logic
                     clock, // Clock signal
   input logic reset_n, // Asynchronous reset (active low)
input logic count_enable, // Enable signal
   output logic [DATA_WIDTH-1:0] count // Counter output
   // Sequential logic
   always @(posedge clock or negedge reset n) begin
       if (!reset n) count <= COUNT FROM[DATA WIDTH-1:0]:
       else if (count enable) begin
          if (count >= COUNT TO DATA WIDTH-1:0]) count <= COUNT FROM DATA WIDTH-1:0]:
          else count <= count + STEP[DATA WIDTH-1:0]:
       end
   end
endmodule
```

### 4.2 Defining reusable functions

Common Operations for Clocked Designs

#### **Common Operations**

To maintain clean and reusable test code, we define standard functions for common operations:

| Function       | Description                                                      |
|----------------|------------------------------------------------------------------|
| setup_clock    | Configures the clock signal with specified period and duty cycle |
| reset_dut      | Performs asynchronous reset sequence of the design               |
| sys_enable_dut | Controls the system enable signal for the DUT                    |
| initialize_dut | Sets up initial conditions and default signal values             |
| toggle_signal  | Changes signal state from 0 to 1 or vice versa                   |

Note: These functions are not exhaustive and can be extended to suit the design requirements.

### 4.3 Breakdown of the basic functions

Function: setup\_clock

```
- Python Function (setup_clock) - Part 1 -
async def setup_clock(
   dut: cocotb.handle.HierarchvObject.
   period ns: int = 10.
   verbose: bool = True.
) -> None:
    Initialize and start the clock for the DUT.
    Parameters
    dut : cocotb.handle.HierarchyObject
       The Device Under Test (DUT)
   period ns : int
       Clock period in nanoseconds (default is 10).
   verbose : bool. optional
       If True, logs the clock operation (default is True).
```

### 4.4 Breakdown of the basic functions

#### Function: reset dut

```
Python Function (reset dut) - Part 1 -
asvnc def reset dut(
   dut: cocotb.handle.HierarchvObject.
   num cvcles: int = 5.
   reset high: int = 0.
   verbose: bool = True.
) -> None:
    Reset the DIIT
   This function assumes the reset signal is active low.
   It asserts the reset signal for 'num cycles' and then deasserts it.
    Parameters
   reset high : int. optional
       Indicates if the reset signal is active high (1) or active low (0).
       By default, the reset signal is active low (0).
   ... skipped for slide ...
   if reset high not in [0, 1]:
       error message: str = (
           f"Invalid reset high value: {reset high}".
           "Hint: reset high should be 0 or 1.".
       raise ValueError(error message)
```

```
Python Function (reset_dut) - Part 2 ___
trv:
    if reset_high == 0:
       dut.reset n.value = 0
    else:
       dut.reset h.value = 1
    await ClockCycles(signal=dut.clock, num_cycles=num_cycles)
    if reset_high == 0:
       dut.reset n.value = 1
    else:
       dut.reset h.value = 0
    await ClockCycles(signal=dut.clock, num_cycles=2)
    if not verbose:
        return
    dut._log.info(
       f"DUT reset for {num cycles} cycles with reset high={reset high}."
except Exception as e:
    error message: str = (
       f"Failed in reset dut with error: {e}".
        "Hint: DUT might not have reset n or reset h port."
    raise RuntimeError(error message) from e
```

### 4.5 Breakdown of the basic functions

Function: sys enable dut

```
Python Function (sys enable dut) -
async def sys_enable_dut(
   dut: cocotb.handle.HierarchyObject,
   verbose: bool = True.
) -> None:
    Enable the DUT.
    Parameters
    dut : SimHandleBase
        The device under test.
   verbose : bool, optional
        If True, logs the enable operation (default is True).
    m m m
    trv:
       dut.i svs enable.value = 1
        await RisingEdge(signal=dut.clock)
        if not werhoes:
            return
       dut._log.info("DUT enabled.")
    except Exception as e:
       error message: str = (
            f"Failed in sys_enable_dut with error: {e}",
            "Hint: DUT might not have i sys enable port or clock signal.".
       raise RuntimeError(error message) from e
```

### 4.6 Breakdown of the basic functions

Function: initialize dut

```
- Python Function (initialize dut) - Part 1 -
async def initialize dut(
   dut: cocotb.handle.HierarchvObject.
   inputs: dict.
   outputs: dict.
   clock period ns: int = 10.
   reset high: int = 0.
   verbose: bool = True,
) -> None:
   Initialize the DUT with default values.
    Parameters
    dut : SimHandleBase
        The device under test (DUT).
    inputs : dict
       A dictionary containing the input names and values.
    outputs : dict
       A dictionary containing the output names and expected values.
    ... skipped for slide ...
    Usage
   >>> inputs = f"i data": 0. "i valid": 0}
   >>> outputs = f"o data": 0. "o valid": 0}
   >>> await initialize dut(dut, inputs, outputs)
    trv:
```

```
Python Function (initialize dut) - Part 2 -
    # Setup the clock
    await setup clock(dut=dut, period ns=clock period ns, verbose=verbose)
    # Reset the DIT
    await reset dut(dut=dut, reset high=reset high, verbose=verbose)
    # Set the input values
    for key, value in inputs.items():
        getattr(dut, key).value = value
    # Wait a few clock cycles
    await ClockCycles(signal=dut.clock. num cycles=5)
    # Check the output values
    for key, value in outputs.items():
        assert getattr(dut, key) value == value, f"Output {key}

→ is incorrect"

    # Check if i sys enable is present
    if hasattr(dut, "i sys enable"):
        await sys_enable_dut(dut=dut, verbose=verbose)
        await ClockCycles(signal=dut.clock, num cycles=5)
    if not verbose:
        return
    dut. log.info("DUT initialized successfully.")
except Exception as e:
    error message: str = f"Failed in initialize dut with error: {e}"
    raise RuntimeError(error message) from e
```

### 4.7 Breakdown of the basic functions

Function: toggle\_signal

```
Python Function (toggle_signal) - Part 1 -
async def toggle signal (
   dut: cocotb handle HierarchyObject.
   signal_dict: dict,
   verbose: bool = True.
) -> None:
   Toggle a signal between high and low values.
    Parameters
    dut · SimHandleRase
       The device under test (DUT).
   signal dict : dict
       A dictionary containing the signal name and value.
       If the value is 1, the signal is toggled to 0: otherwise, it is toggled
   verbose : bool. optional
       If True, logs the signal toggling operation (default is True).
    Usage
   >>> signal dict = {"i valid": 0, "i ready": 0}
   >>> await toggle signal(dut, signal dict)
    .....
```

```
Python Function (toggle signal) - Part 2 _
    for key, value in signal_dict.items():
        getattr(dut, key).value = value
        await RisingEdge(signal=dut.clock)
        if value == 1:
            getattr(dut, kev).value = 0
        else.
            getattr(dut, kev).value = 1
        await RisingEdge(signal=dut.clock)
    if not verbose:
        return
    dut._log.info("Signal toggled successfully.")
except Exception as e:
    error message: str = (
       f"Failed to toggle signal {key}.",
       f"Error: {e}".
    raise RuntimeError(error message) from e
```

### 4.8 Breakdown of the basic functions

Function: get generics

#### Another useful function

If a design has generics, we can define a function to get the generics from the design:

```
- Python Function (get_generics)
def get_generics(dut: cocotb.handle.HierarchyObject) -> dict:
   Retrieve the generic parameters from the DUT.
    Parameters
    dut : cocotb.handle.HierarchyObject
       The device under test (DUT).
    Returns
    dict
       A dictionary containing the generic parameters.
   return {
        "DATA_WIDTH": int(dut.DATA_WIDTH.value),
        "COUNT FROM": int(dut.COUNT FROM.value).
        "COUNT TO": int(dut.COUNT TO.value).
        "STEP": int(dut.STEP.value).
```

# 4.9 Modeling the HDL Counter

Counter Model

#### Counter Model

- The CounterModel class is a simple model of the counter.
- It is used to verify the behavior of the DUT.
- The model is used to compare the output of the DUT with the expected output.

This way, we can verify the correctness of the DUT. The class feeds the DUT with random inputs to compute an expected output, and finally compares the module output to the expected for correctness.

With more complex designs, the model could monitor a streaming data/valid bus, sample the bus when a transaction occurs, and compare the output to the expected output.

## 4.10 Tying it all together

Complete Testbench Implementation

#### Testbench Organization

- All utility functions are consolidated in src/bench/cocotb\_utils.py
- Functions are designed to be modular and reusable across different testbenches
- Easy integration with both simple and complex designs

#### Complete Example

A full implementation example is available in this GitHub repository:

- Basic counter example: src/rtl/example/
- Associated testbenches: src/bench/example/
- Advanced implementation: Ascon-128 cryptographic core, using all the aspects discussed in this presentation.

### Timothée Charrier