# 2024 IEEE NPSS Real Time Conference

Workshop on open-source tools for FPGA and ASIC design

Introduction to Cocotb Marc-André Tétrault, Eng., PhD.

20/04/2024





https://github.com/mtetrault/RT2024\_CocotbWorkshopFiles.git

Please « git clone » these files inside the virtual box machine

Includes a pdf file with step-by-step lab instructions.

## RT2024 Workshop

Coding block

- 1- Cocotb
- 2- Surf

Hardware block

- 1- FABulous eFPGAs
- 2- Open-source ASIC Design (openLANE, Skywater, Caravel)



#### Cocotb

Coroutine cosimulation testbench

Overview in 2018 by **Benjamin John Rosser** <a href="https://indico.cern.ch/event/776422/">https://indico.cern.ch/event/776422/</a>

Uses Python instead of HDL/tailored language

- Python widely used by students and scientists
- Access to Python packages
- Object Oriented Programming support

- . . .



#### Cocotb

Many updates to the package/syntax since 2018

- Changed Makefile approach to pure Python
- Changed Python keywords to manage asynchronous behaviour
- New features merged as community ressources allow
- Open-source



#### **Cocotb** – Link with simulator

- Simulator compiles HDL
- Simulator is main thread
- Simulator imports Python code
- Flow control is swapped between simulator and Python code.



From https://indico.cern.ch/event/776422/



#### **Supported simulators**

Commercial

Open-source

Incisive/Xcellium

Questa/Modelsim

VCS (Synopsys)

Xsim (Vivado)

Verilator

**GHDL** 

Icarus Verilog

. . .

More at https://en.wikipedia.org/wiki/List\_of\_HDL\_simulators



#### Opensource vs commercial simulators

Commercial

Open-source

FPGA editions free

Mixed langage

Bug support

Advanced features (SVA)

**Includes Waveform viewer** 

Large designs might lock down performance

Single language

No design size limit

Community support

External waveform viewer

Some HDL features not

supported



## Survey – Participant experience?

How often have you:

- Used a Linux and command line terminal?
- Used Python?
- Used git (to download examples)?
- Used HDL simulators (Vivado, Modelsim, etc)?
- Used VHDL or Verilog?



## **Cocotb - Workshop**

Workshop relies on basic experience

Goal: provide a first contact with the Cocotb module and use flow.

Uses VHDL, with the GHDL simulator

Uses GtkWave to view waveforms

Basic text editors (nano, vim, gedit) for file edition

VSCodium for interactive debugging



3!T

# Lab 1

First contact

#### Lab 1

#### **Objectives**

- Launch a cocotb simulation.
- Add command line arguments to the underlying simulator (ghdl in this case).
- View waveforms using gtkwave.

- Learn from error messages



#### Lab 1 - Design

Simple adder from the official cocotb git repository (v1.8) cocotb/example/adder

#### Three files:

- Simple HDL Adder
- Python model (addition)
- Cocotb test and runner
  - \* first part is testbench
  - \* second part is runner/simulator call



#### Lab 1 - Design

Changes and simplifications

- Removed simulator configurability: GHDL simulator
- Removed langage configurability: VHDL only
- Simplified options
- Added comments



## Lab 1 - Design

```
cocotb.regression
                                                      Found test test_adder_solution.adder_randomised_test
    0.00ns INFO
                    cocotb.regression
                                                      running adder randomised test (1/1)
    0.00ns INFO
                                                        Test for adding 2 mandom numbers multiple times
../../src/openieee/v93/numeric_std-body.vhdl:398:9:@0ms:(assertion warring): NUMERIC_5.0."+": non logical value detected
   20.00ns INFO
                                                      adder randomi ed test passed
                    cocotb.regression
                    cocotb.regression
   20.00ns INFO
                                                      ** TEST
                                                      ** test_adder_solution.adder_randomised_test
                                                                                                              ***************************
                                                         TESTS=1 PASS=1 FAIL=0 SKIP=0
```



#### Lab 1 - GtkWave





3!T

# Lab 2

Customizing a cocotb template

#### Lab 2

#### **Objectives**

- Write your first customized Cocotb runner.
- Write your first Cocotb test.
- Automate verification (confirm design function without waveforms)



#### Lab 2 - Design

Square root arithmetic core from

https://vhdlguru.blogspot.com/2020/12/synthesizable-clocked-square-root.html

#### Two files:

- Square root arithmetic core (provided)
- Cocotb test and runner
  - \* first part is testbench (edit second)
  - \* second part is runner/simulator call (edit first)



#### Lab 2 - Design

Interface: clk, reset, input interface, output interface

Square Root Core, 32-bit integer input

```
clk
reset
arg_valid sqrt_valid
arg(31:0) sqrt_res(15:0)
```



#### Lab 2 - Work

- Modify the runner to match the provided design
- Learn Cocotb relevant Python keywords
- Add very simple test for arithmetic core

**Note**: Python "async" and "await" keywords (yield in older versions)

**Use Snippets File!** 

- Faster than google ©



## Lab 2 – Expected outcome





### Lab 2 – GtkWave Tip





3!T

# Lab 3

Interactive debugging

## Lab 3 - Debugging

Error messages may come from

- Starting the simulator (like in lab 1)
- HDL compile errors
- Python syntax (during execution)
- Testbench assertion failures

Using « print » functions is very inefficient



## Lab 3 – Why not directly from a GUI?

With Cocotb, the simulator calls Python.

Need to add a hook in Cocotb tests, where the IDE can connect.

Supported in PyCharm Pro, but not PyCharm Community https://blog.patfarley.org/pages/cocotb-pycharm.html

Supported in VSCode/VSCodium (free)



## Lab 3 - Objectives

- Use an IDE to graphically debug a cocotb test.

- Configure the cocotb test to support debug.
- Configure VSCodium to attach to the cocotb test.
- Add break points in the cocotb test.
- Inspect variables and dut signals within the IDE.



## Lab 3 – Test project

Copy of solution from Lab 2, already provided

```
clk
reset
arg_valid sqrt_valid
arg(31:0) sqrt_res(15:0)
```













And then press « enter » twice for the server name and port

- localhost
- 5678











**3!T** 

# Lab 4

Code reuse 1 - functions and drivers

#### Lab 4

Custom designs → custom testbench

Some parts are standard, like busses. Some are simple (UART), others more detailed (PCIe).

They might already exist somewhere in another project...?



#### Lab 4 – Cocotb extensions

Mainly bus/communication protocols

Many available through python/pip3: ethernet, spi, uart, ... <a href="https://pypi.org/search/?q=cocotbext">https://pypi.org/search/?q=cocotbext</a>

Others available from private repositories: ahb, ...

SURF workshop will use the AXI cocotb extension



## Lab 4 - Objectives

- Encapsulate reusable sequences in functions
- Use a cocotb extension to control a UART standard interface
- Get familiar with the data format used by the UART extension, and how to make conversions.



## Lab 4 - Design





## Lab 4 - Work

Update the runner to include the multi-file design

- add all VHDL files

Encapsulate init sequence and end-of-sim time

- use a function, not forgetting the special keywords

Use cocotbext-UART

- Add and use a driver and a sink object

Use native Python functions for conversion from/to bytearray



## Lab 4 – Expected outcome





3!T

# Lab 5

Code reuse 2 – object oriented programming

#### Lab 5

DAQ designs are complex: the verification code is not simpler

Universal Verification Methodology (UVM)

- Leverages Object Oriented Programming (OOP)
- Not supported by VHDL/Verilog; typically SystemVerilog
- Standardize structure and methods; excellent when buying a verification code

Steep ramp-up, requires simulation-specific SystemVerilog training, few students/scientists have basics on this topic.

#### Lab 5 – Cocotb with OOP

Python supports OOP, so cocotb does as well

UVM not required for small/medium sized designs

Planning a base class ahead will save a lot of time

- This is software programming, not firwmare/HDL programming
- Larger pool of trained students and scientists can contribute



## Lab 5 - Objectives

- Get familiar with a simple object oriented testbench structure.

 Populate the provided template with code prepared in previous labs.

- Write two different cocotb tests sharing the same base class.



## Lab 5 – Design (same as lab 4)





#### constructor (\_\_init\_\_)

- save dut pointer
- initialize logging utility

build environment
Init I/Os, clock and reset
configure dut
start environment
target test (pure virtual function)
post-test sequences
- wait for ongoing transactions

#### run function

- executes these steps one after the other



constructor (\_\_init\_\_\_)

- save dut pointer
- initialize logging utility

build environment
Init I/Os, clock and reset

configure dut start environment target test (pure virtual function) post-test sequences

- wait for ongoing transactions

run function

- executes these steps one after the other

Constructor – dut and logs pointers

Build the environment

- connect drivers, sinks, checkers (not previously covered)

Start clock and reset dut (same as lab 4)



constructor (\_\_init\_\_)

- save dut pointer
- initialize logging utility

build environment

Init I/Os, clock and reset

configure dut

start environment

target test (pure virtual function)

post-test sequences

- wait for ongoing transactions

run function

- executes these steps one after the other

Configure DUT – enable channels, set thresholds, set bias, etc.

Start the environment

- enable drivers, waveform generators, enable checkers...

Post test – wait for unfinished transactions or packets



constructor (\_\_init\_\_)

- save dut pointer
- initialize logging utility

build environment
Init I/Os, clock and reset
configure dut
start environment

target test (pure virtual function)

post-test sequences

- wait for ongoing transactions

run function

- executes these steps one after the other

Test: undefined in base class, forces designers to derive the class and override the test.

Run: executes steps in order for all tests.



#### Lab 5 - Child Classes

constructor (\_\_init\_\_)

- save dut pointer
- initialize logging utility

build environment
Init I/Os, clock and reset
configure dut
start environment
target test (pure virtual function)
post-test sequences

- wait for ongoing transactions

#### run function

- executes these steps one after the other



Contributors can focus their efforts on the test, relying on the environment



## Lab 5 – Template and work

constructor (\_\_init\_\_)

- save dut pointer (done)
- initialize logging utility (done)

build environment
Init I/Os, clock and reset
configure dut
start environment
target test (pure virtual function)
post-test sequences
- wait for ongoing transactions

run function

- executes these steps one after the other





## Lab 5 – Expected outcome

- Same waveform patterns as in lab 4.

Note: the two simulations will be appended in the same VCD file. Raising the reset at the end of a test (i.e. in the post-test sequence) helps to see this



3!T

# Lab 6

Code reuse 3 – Monitors, Models and Checkers

#### Lab 6

Labs 4 and 5 see the DUT as a black box

When an error occurs, it is not always clear where the problem originates from. The designer needs to read the waveforms to find the issue.

Localized tests accelerate bug localization System Verilog Assertions:

- industry standard, but...
- not supported by free/open source simulators
- Except perhaps Modelsim for Intel? To be confirmed.



Monitor(s), Model and Checker

Monitors are probes, only recovering useful data from signals

- Conditions to record data depends on the interface
- Example: AXI bus needs to consider address, valid, ready and data signals

- Most simple interface : enable + data (sqrt core)



Monitor(s), Model and Checker

Monitors are threads, basically while(true) loop.

In some cases, should not run while design is not in a known state

- For example, during the reset sequence

Monitor threads thus often have start/stop methods.



Monitor(s), Model and Checker

Models generate the expected result from the HDL module. For example, CRC core, square root, FIFO, Data packet, compressed packet, etc.

Models have no notion of clock or signals. Only data

- Ensures model is different from HDL, improving error detection
- Should make model easier to code compared to HDL



Monitor(s), Model and Checker

Checkers compare the result from the model and the HDL module

- Declares an error when differences are found
- Log utility provides instance location within the dut



Monitor(s), Model and Checker





## Lab 6 – MMC construction overview

How to write a checker (unit test) class

- 1- Create a monitor on the HDL input interface, connecting with its signals, in the constructor.
- 2- Create a monitor on the HDL output interface, connecting with its signals, in the constructor
- 3- Write a model method
- 4- Write a checker/test method
- 5- Write a "start" and a "stop method, launching and stopping the threads for the two monitors and the checker

## Lab 6 – MMC insertion in base class

- 1- Add an MMC instance in the "BuildEnvironment" method
- 2- Add a "StartEnvironment" method, where the MMC.start() is called
- 3- In the post-simulation method, call the MMC.stop() method
- 4- Run the existing test(s)



## Lab 6 - Objectives

- Reuse a monitor class from the cocotb main repository

- Adapt MMC class to the sqrt core.

- Attach MMC object to the base environment from lab 5

Warning: the template class from the cocotb repo uses efficient but less easy to understand native python constructs. Read the added comments for an initial explanations on these if they are not familiar to you.

## Lab 6 – Bench split in 2 files

- Monitor and MMC class in separate file for clarity

- File with base environment class must import MMC class



## Lab 6 – Expected outcome

- Same waveform patterns as in lab 4.

- Error messages will pinpoint the error location Introduce an error in the model to generate a failure

- The assertion in the test is still relevant: would indicate that something is wrong after the sqrt core



## Lab 6 – Interesting « bug »

If test fails, it stops at the assertion

Does not start a new simulation, but continues where the previous one stopped.

Notice here the UART modules have no reset signal.

If the simulation failed while the UART is transmitting, it will do so regardless that the first test stopped, making the next test fail.

Fix 1 – add reset to uart in HDL

Fix 2 – make reset time much longer, and clear the uart\_sink after the long reset.