



# CAVEAT – a Framework for Context-Aware Verification, Emulation, and Training

Torsten Reuschel, M. Ferris, P. Trottier

FPGA Conference Europe 2025

# Electronics in Remote and Adverse Surroundings @ UNB



Emerging questions and ties at Department of Physics

- Quantum Sensing & Ultracold Matter Lab
- Radio and Space Physics Laboratory

New **ERAS** lab as of 2025

- Highly reliable, high accuracy measurement instrumentation in unconventional environments
- Infrastructure for assembly, integration, testing, calibration

**RSPL**



**QSUM**



**ERAS**



Campus image: UNB media services; map adapted from Hogweard, Public domain, Wikimedia Commons

# Motivation

- Scientific remote sensing platform *sanimut* (currently ionospheric radar)
- Facilitate maintenance, reconfiguration and operating modes specific to experiments
- Operation subject to physical environment
  - radio environment
  - state of ionosphere

impacts includes configuration of

- analog frontend
- signal processing

...and diverse interests of physicists.

Map adapted from Hogweard, Public domain, via Wikimedia Commons



# Beyond Verified Function: Context!



Ionosphere sketch adapted from Trex2001, CC BY-SA 3.0 via Wikimedia Commons; emoticon (#2847) and hat (#53868) via clipartix.com; panel: Pearson Scott Foresman, Public domain, via Wikimedia Commons

# Beyond Verified Function: Involved People and Skills

- Verification is software
- Improve communication (via 'tangibles')
- Python is a popular language (ease of onboarding staff/students)
- Tremendous ecosystem (we are still a small group with limited budget)
- Cross-platform support (we opt for Linux)



Images: 'emoticon confused' (#2847), 'emoticon think' (#2982), 'hat' (#53868), 'computer' (#50654) via clipartix.com



| Jun 2025 | Jun 2024 | Change | Programming Language | Ratings |
|----------|----------|--------|----------------------|---------|
| 1        | 1        |        | Python               | 25.87%  |
| 2        | 2        |        | C++                  | 10.68%  |
| 3        | 3        |        | C                    | 9.47%   |
| 4        | 4        |        | Java                 | 8.84%   |
| 5        | 5        |        | C#                   | 4.69%   |
| 6        | 6        |        | JavaScript           | 3.24%   |

Source: [tiobe.com](https://www.tiobe.com/tiobe-index/), accessed on June 25, 2025

# What If... a Solution Concept

A digital twin ...

... built on existing tools

... contributing a physical verification framework to the community

## Context-Aware Verification

- Create virtual environment for concept studies
- Insight into interdependency of components and their surrounding
- Operational risk mitigation

## Emulation

- Facilitate root-cause analysis and predictive study beyond simulating a particular process
- Model-based core development and design of novel experiments
- Reduce time-to-field

## Training

- Sandbox for student and research training



# On Today's Agenda

---

- Introduction to pytest
- Beyond pytest: static verification of design and operating specification
- Step-by-step introduction to cocotb
- Beyond cocotb: context-aware dynamic verification
- More examples from our lab: caveat in emulation and training  
(we're scientists, technologists, and engineers with a knack for building tools)



- Automate testing
  - Increase confidence in design – in an uncertain environment
  - Raise acceptance of changes – iterative improvement
  - Offer documentation and traceability – students come and go
  - Reduce worries and distraction
- Scope of testing: unit, integration, system – **software, gateware, and hardware!**
- Failing test  $\neq$  faulty code ...it might be lack of understanding of DUT

## pytest offers ...

- Automatic test discovery
- Parametrization
- Ability to combine with packages from Python ecosystem

# Example Code and Test Definition



pytest

tests/test\_adder.py

```
1 import pytest
2
3 def adder(a: int, b: int) -> int:
4     """Add two integers"""
5     return a + b
6
7 @pytest.mark.parametrize(
8     "invals, expect",
9     [[(3, 5), 8], ([2, 4], 6), ([6, 9], 42)])
10 def test_adder(invals, expect):
11     assert adder(invals[0], invals[1]) == expect, \
12         "Unexpected result"
```

1. Package import(s)
2. Code subject to testing
3. Test definition  
optional: parametrization

# Initiating Tests



pytest

## Command line

```
$ pytest tests/ -k "adder and not multiplier"
```

## Programmatic call

```
pytest.main([
    'tests/',
    '-k adder and not multiplier',
])
```

## A matter of preference and workflow..

- Interchangeable and simultaneous integration into workflows
- Options to include/exclude certain tests during automatic collection, ...
- Functions and corresponding test definition in same or nearby file  
OR separate directories for application and test



# Example Test Output... Ooops!

```
$ python tests/test_adder.py
=====
tests/test_adder.py::test_adder[invals0-8] PASSED [ 33%]
tests/test_adder.py::test_adder[invals1-6] PASSED [ 66%]
tests/test_adder.py::test_adder[invals2-42] FAILED [100%]

=====
===== FAILURES =====
----- test_adder[invals2-42] -----
vals = [6, 9], expect = 42

    @pytest.mark.parametrize(
        "vals, expect",
        [([3, 5], 8), ([2, 4], 6), ([6, 9], 42))
)
def test_adder(vals, expect):
>     assert adder(vals[0], vals[1]) == expect, \
       "Unexpected result"
E     AssertionError: Unexpected result
E     assert 15 == 42
E     +   where 15 = adder(6, 9)

tests/test_adder.py:11: AssertionError
=====
===== short test summary info =====
FAILED tests/test_adder.py::test_adder[invals2-42] - AssertionError: Unexpected result
===== 1 failed, 2 passed in 0.02s =====
```

A failing test doesn't always point to faulty code.

# Example Fixed: All Pass



pytest

tests/test\_adder.py

```
5  [...]
6
7  @pytest.mark.parametrize(
8      "invals, expect",
9      [([3, 5], 8), ([2, 4], 6), ([6, 9], 15)])
10 def test_adder(invals, expect):
11     assert adder(invals[0], invals[1]) == expect, \
12         "Unexpected result"
```

```
python tests/test_adder.py
=====
test session starts =====
collected 3 items

tests/test_adder.py::test_adder[invals0-8] PASSED [ 33%]
tests/test_adder.py::test_adder[invals1-6] PASSED [ 66%]
tests/test_adder.py::test_adder[invals2-15] PASSED [100%]

===== 3 passed in 0.00s =====
```

# CAVEAT Static Verification

## Circuit design specification

(general definitions or project specific)

- Pin constraints
- Schematic or netlist
- Ohm's law



# CAVEAT Static Verification

## Circuit design specification

(general definitions or project specific)

- Pin constraints
- Schematic or netlist
- Ohm's law



## User input checking

(project specific)

- Syntax
- Valid operation sequence
- Radio license compatibility
- Concurrence of events
  - e.g. power amplifier data + gate



# CAVEAT Static Verification

## Circuit design specification

(general definitions or project specific)

- Pin constraints
- Schematic or netlist
- Ohm's law



## User input checking

(project specific)

- Syntax
- Valid operation sequence
- Radio license compatibility
- Concurrence of events
  - e.g. power amplifier data + gate



# CAVEAT Static Verification: Application Example

...with fixtures, docstring, and assertions.

caveat/tests/static/test\_io\_pin\_all\_constrained.py

---

```
1  def test_inout_pin_all_constrained(pin_constraints, net_specification):
2      """Verify that all physical pins are constrained. Check that:
3          - corresponding pin constraints exist, and
4          - no excess pin constraints exist, as this suggests an incomplete netlist.
5      """
6      pins_constrained = set([pin['name'] for pin in pin_constraints])
7      pins_netlisted = set(net_specification.keys())
8
9      excess_constrained = pins_constrained - pins_netlisted
10     assert excess_constrained == set([]), "constrained pin(s) unconnected or
11         → missing in netlist: {}".format(' '.join(excess_constrained))
12
13     excess_netlisted = pins_netlisted - pins_constrained
14     assert excess_netlisted == set([]), "netlist contains unconstrained pins:
15         → {}".format(' '.join(excess_netlisted))
```

# CAVEAT Static Verification: Example Report

- **Failed** pin constraints check due to incomplete netlist  
report: test description and failed assertion
- **Warning** due to implicit operation sequence \*

Start with predefined tests from framework... and add project-specific tests for past and future mishaps.

\* Mismatching pin events of power amplifier data and gate results in error.

## CAVEAT/sanimut static verification

|                |                                                                                                        |
|----------------|--------------------------------------------------------------------------------------------------------|
| Started        | 2025-06-24 18:01:09                                                                                    |
| Ended          | 2025-06-24 18:01:10                                                                                    |
| Duration       | 0:00:00.401646                                                                                         |
| Total run time | 0:00:00.072604                                                                                         |
| Python         | 3.11.2                                                                                                 |
| Platform       | Linux-6.1.0-35-amd64-x86_64-with-glibc2.36                                                             |
| Packages       | pytest: 7.3.1<br>pluggy: 1.3.0                                                                         |
| Plugins        | metadata: 3.1.1<br>reporter-html-dots: 0.11.0<br>reporter: 0.5.3<br>cocotb-test: 0.2.6<br>xdist: 3.7.0 |



### Warnings 2

**UserWarning** Command execution time is in relative past. Is this intended for RX\_NCO\_RST,0,0,1500 in line 3 of example/sanimut/cmdfile/timedcommand\_deadline\_violation.cmd?  
caveat/tests/static/test\_sanimut\_cmdfile\_deadline\_unique\_timedcommands.py:39

**UserWarning** Command execution time is in relative past. Is this intended for RX\_NCO\_RST,0,0,1500 in line 3 of example/sanimut/cmdfile/timedcommand\_deadline\_violation.cmd?  
caveat/tests/static/test\_sanimut\_cmdfile\_deadline\_grouped\_timedcommands.py:45

### caveat/tests/static/test\_io\_pin\_all\_constrained.py

#### FAILED test inout pin all constrained

Verify that all physical pins are constrained: starting with a list of netlisted pins, verify that

- corresponding pin constraints exist, and
- no excess pin constraints exist, as this suggests an incomplete netlist.

Requires the following fixtures:

pin\_constraints: dict=fileio.xdc\_parser.read\_xdc(...) net\_specification: dict=fileio.netlist\_parser.read\_netspec\_from\_csv(...)

# COroutine based COsimulation TestBench



**CO**routine cooperative scheduling, interacts with a simulator's event scheduler

**CO**-simulation cocotb's scheduler integrates with simulator's event scheduler via *triggers*

**TestBench** test environment and user interfacing via simulation *handles*



Image source: <https://docs.cocotb.org/en/stable/>

# COroutine based COsimulation TestBench (cont'd)



- Designed to reduce overhead when creating test  
Design re-use and randomized testing like UVM,  
implemented in Python rather than SystemVerilog
- With cocotb, VHDL/Verilog/SystemVerilog  
normally used for design, not the testbench...this is  
where Python comes in, e.g. for automatic test  
discovery
- It's free  
We're still a small group with limited budget,  
faced with vertical integration
- And yet: tremendous ecosystem for building  
testbench and models (coroutines)
  - General: math, plotting, science, ..
  - cocotb-specific: bus drivers/monitors, e.g.  
cocotbext



Image source: <https://docs.cocotb.org/en/stable/>

# Example Code and Test Definition



Let's define and test an adder in three distinct parts:

reference model .. cocotb 'glue' .. RTL code



Image source: <https://docs.cocotb.org/en/stable/>

# Example Code and Test Definition (cont'd)



Image adapted from <https://docs.cocotb.org/en/stable/>

# Example Code and Test Definition (cont'd)



Image adapted from <https://docs.cocotb.org/en/stable/>

```
import cocotb
from cocotb.clock import Clock
from cocotb.triggers import RisingEdge

def adder_model(val_a: int, val_b: int) -> int:
    """Adder reference model"""
    return val_a + val_b

@cocotb.test()
async def test_adder(dut):
    """Test coroutine"""
    # generate clock and start clock as coroutine
    cocotb.start_soon(Clock(dut.clk, 10, 'ns').start())
    # set input signals
    dut.i_val_a.value = 2
    dut.i_val_b.value = 3
    # compare output after two cycles with model
    expect = adder_model(2, 3)
    for _ in range(2):
        await RisingEdge(dut.clk)
    assert dut.o_result.value == expect, "Unexpected result"
```

# Alternative Ways of Running a Test



## Makefile

```
# source file(s)
SOURCES = rtl/adder.v
# name of the toplevel module
TOPLEVEL = adder
# basename of the Python test file
MODULE = adder_tb
# cocotb's make rules take care of simulator
→ setup
include $(shell cocotb-config
→ --makefiles)/Makefile.sim
```

## Programmatic call

```
from cocotb_test.simulator import run
run(
    verilog_sources = ['rtl/adder.v'],
    toplevel = 'adder',
    module = 'adder_tb',
)
```

As with pytest: interchangeable and concurrent integration into workflows

# Example Single Test Output: Pass



```
INFO cocotb: Running command: vvp -M /home/treuschel/Downloads/venv/cocotb/lib/python3.11/site-packages/cocotb/libs -m libcocotbvpi_icarus
INFO cocotb: ---ns INFO gpi ..mbed/gpi_embed.cpp:108 in set_program_name_in_venv Using Python virtual environment interpreter at /home/treuschel/Downloads/venv/cocotb/bin/python
INFO cocotb: ---ns INFO gpi ..gpi/GpiCommon.cpp:101 in gpi_print_registered_impl VPI registered
INFO cocotb: 0.00ns INFO cocotb Running on Icarus Verilog version 11.0 (stable)
INFO cocotb: 0.00ns INFO cocotb Running tests with cocotb v1.9.2 from /home/treuschel/Downloads/venv/cocotb/lib/python3.11/site-packages/cocotb
INFO cocotb: 0.00ns INFO cocotb Seeding Python random module with 1750987252
INFO cocotb: 0.00ns INFO cocotb.regression Found test adder_tb.test_adder
INFO cocotb: 0.00ns INFO cocotb.regression running test_adder (1/1)
INFO cocotb: Test coroutine
INFO cocotb: 10.00ns INFO cocotb.regression test_adder passed
INFO cocotb: 10.00ns INFO cocotb.regression **** TEST STATUS SIM TIME (ns) REAL TIME (s) RATIO (ns/s) ****
INFO cocotb: **** adder_tb.test_adder PASS 10.00 0.00 31610.58 **
INFO cocotb: **** TESTS=1 PASS=1 FAIL=0 SKIP=0 10.00 0.11 93.03 **
INFO cocotb: ****
INFO cocotb: Results file: /sim_build/hlg4b3vh_results.xml
```

- `vvp` is iverilog's simulation engine
- Outputs: terminal, xml-file (HTML reporting via `pytest-html`)

# cocotb Parametrized Test



- *TestFactory* generates parametrized tests
- cocotb 2.0 introduces decorator (pytest look and feel)

```
@cocotb.test
@cocotb.parametrize(
    value_a=[2,1,0],
    value_b=[3,15],
)
async def test_adder(
    dut,
    value_a,
    value_b):
    ...

```

```
from cocotb.regression import TestFactory

async def testfactory_adder(dut, value_a, value_b):
    """Testfactory for adder"""
    # generate clock and start clock as coroutine
    cocotb.start_soon(Clock(dut.clk, 10, 'ns').start())
    # update input signals
    dut.i_val_a.value = value_a
    dut.i_val_b.value = value_b
    #obtain reference result
    expect = adder_model(value_a, value_b)
    # expect correct output after two cycles
    for _ in range(2):
        await RisingEdge(dut.clk)
        assert dut.o_result.value == expect, "Unexpected result"

tf = TestFactory(testfactory_adder)
tf.add_option('value_a', [2, 1, 0])
tf.add_option('value_b', [3, 15])
tf.generate_tests()
```

## cocotb Parametrized Test Output.... Ooops!



- TestFactory combines all parameters:  
 $\text{value\_a} \times \text{value\_b} = 2, 1, 0 \times 4, 15 \mapsto (\textcolor{green}{2}, \textcolor{red}{4}), (\textcolor{red}{2}, \textcolor{blue}{15}), (\textcolor{blue}{1}, \textcolor{green}{4}), (\textcolor{blue}{1}, \textcolor{red}{15}), (\textcolor{green}{0}, \textcolor{green}{4}), (\textcolor{green}{0}, \textcolor{blue}{15})$
  - What's wrong with cases 2 and 4?

cocotb Parametrized Test Output.... Ooops!



```
INFO cocotb:    0.00ns INFO    cocotb.regression
INFO cocotb:    0.00ns INFO    cocotb.regression
INFO cocotb:    0.00ns INFO    cocotb.regression
INFO cocotb:
INFO cocotb:
INFO cocotb:
INFO cocotb:
INFO cocotb:    10.00ns INFO    cocotb.regression
INFO cocotb:    10.00ns INFO    cocotb.regression
INFO cocotb:
INFO cocotb:
INFO cocotb:
INFO cocotb:
INFO cocotb:    30.00ns INFO    cocotb.regression
INFO cocotb:
INFO cocotb:
INFO cocotb:
INFO cocotb:
INFO cocotb:
```

Found test adder\_tb2.testfactory\_adder\_005  
Found test adder\_tb2.testfactory\_adder\_006  
running testfactory\_adder\_001 (1/6)  
 Automatically generated test

value\_a: 2  
value\_b: 3  
testfactory\_adder\_001 passed  
running testfactory\_adder\_002 (2/6)  
 Automatically generated test

value\_a: 2  
value\_b: 15  
testfactory\_adder\_002 failed  
Traceback (most recent call last):  
  
 assert dut.o.result.value == expect, "Unexpected result"  
AssertionError: Unexpected result  
assert 0001 == 17

- TestFactory combines all parameters:  
 $\text{value\_a} \times \text{value\_b} = 2, 1, 0 \times 4, 15 \mapsto (\textcolor{red}{2}, \textcolor{blue}{4}), (\textcolor{red}{2}, \textcolor{blue}{15}), (\textcolor{blue}{1}, \textcolor{blue}{4}), (\textcolor{red}{1}, \textcolor{blue}{15}), (\textcolor{blue}{0}, \textcolor{blue}{4}), (\textcolor{blue}{0}, \textcolor{blue}{15})$
  - What's wrong with cases 2 and 4?

# Example Fixed: All Pass



```
1 module adder #(parameter c_WIDTH = 4) (
2     input wire clk,
3     input wire [c_WIDTH-1:0] i_val_a,
4     input wire [c_WIDTH-1:0] i_val_b,
5     output reg [c_WIDTH-1:0] o_result
6 );
```

```
INFO cocotb: 110.01ns INFO    cocotb.regression
INFO cocotb:
INFO cocotb: Results file: sim_build/5pl4e4ni_results.xml
```

| ** TEST                            | STATUS | SIM TIME (ns) | REAL TIME (s) | RATIO (ns/s) | ** |
|------------------------------------|--------|---------------|---------------|--------------|----|
| ** adder_tb2.testfactory_adder_001 | PASS   | 10.00         | 0.00          | 28209.30     | ** |
| ** adder_tb2.testfactory_adder_002 | PASS   | 20.00         | 0.00          | 95547.01     | ** |
| ** adder_tb2.testfactory_adder_003 | PASS   | 20.00         | 0.00          | 112003.04    | ** |
| ** adder_tb2.testfactory_adder_004 | PASS   | 20.00         | 0.00          | 121933.54    | ** |
| ** adder_tb2.testfactory_adder_005 | PASS   | 20.00         | 0.00          | 128469.03    | ** |
| ** adder_tb2.testfactory_adder_006 | PASS   | 20.00         | 0.00          | 142913.58    | ** |
| ** TESTS=6 PASS=6 FAIL=0 SKIP=0    |        | 110.01        | 0.12          | 923.08       | ** |

# CAVEAT

- Foundation: pytest + cocotb
- Attach physical environment (simulated or real)
- Interfacing: network socket, serial, ...
- Reporting (HTML)
- Forastero-based testbench available



*"Forastero is a library for writing better testbenches with cocotb, inspired by UVM but distilling it to just the most useful parts."*

—<https://github.com/Intuity/forastero>

# CAVEAT

- Foundation: pytest + cocotb
- Attach physical environment (simulated or real)
- Interfacing: network socket, serial, ...
- Reporting (HTML)
- Forastero-based testbench available

*"Forastero is a library for writing better testbenches with cocotb, inspired by UVM but distilling it to just the most useful parts."*

—<https://github.com/Intuity/forastero>



# CAVEAT Dynamic Verification

## Functional specification

(RTL, general or project specific)

- Interfaces
- State machines
- ...



## User-/soft-defined operation

(project specific, some overlap w/ static)

- Interaction with peripherals
  - Practicability of sequences
  - Concurrency of events
  - Over-/underflow recovery
- Meaningfulness of input/output signals
- Radio license compatibility
- Compatibility with deployed system bitstream, circuit boards, peripherals

Images adapted from: 'magnifier': <http://www.simpleicon.com/>, CC BY 3.0 via Wikimedia Commons; 'folder': Vijay Verma, CC0, via Wikimedia Commons, 'clipboard': Emoji One, CC BY-SA 4.0 via Wikimedia Commons

# CAVEAT Dynamic Verification: Example Report

- **Warning**
  - Truncated input: 16 bit → 12 bit
  - Code change, incl. user-defined command files, code files
- **XFail** ...because some things should not be viable, e.g. receive signal before physical propagation delay has passed

## Feedback to user

- Operational constraints
  - e.g. resource availability, concurrence, dead-times
- Physical limitations
  - e.g. voltage rating, delay, tx power, ..
- Illustrate sequence for ease of review



# CAVEAT Dynamic Verification: Example Report (cont'd)

- **Integrate validation** (shift to user) and verification (mostly dev)
- Example: review Ethernet I/O, timed commands, experiment sequence



(Time scales of RF-Tx/RF-RX were compressed to fit this screen; relative sequencing unchanged.)

# CAVEAT Functional Emulation

- Leverage developer tools for a (permanent) virtual instance
- Open-source framework compatible with closed-source code, e.g. ionospheric radar
- Simulate **RTL** + surrogate **environment**

## Use cases

- Concurrent, yet independent development of hardware and software (API, controls)
- Developing new experiments, operational modes
- Study design parameter including impact of original user level software



# Emulation Example: Adder with Network Interface



- Simple 'instrument': hardware adder  
Code is available in caveat repo as application example
- Custom, project specific parts:
  - User interfacing client
  - Module *axis\_adder*
  - CaveatBench object: connect parts, monitor interfaces and operation

Images adapted from: 'calculator': Breathe Icon Team, CC BY-SA 3.0 via Wikimedia Commons; 'cable': R.Spekking CC BY-SA 4.0 via Wikimedia Commons; 'magnifier': <http://www.simpleicon.com/>, CC BY 3.0 via Wikimedia Commons

# Emulation Example: Adder with Network Interface (cont'd)

```
1  from cocotb.clock import Clock
2  from cocotb.triggers import RisingEdge
3  from caveat import augmented_handle
4  from caveat.caveatbench import CaveatBench
5
6  @cocotb.test()
7  async def run_network_adder(dut):
8      tb_env = CaveatBench(dut)
9      cocotb.start_soon(
10          Clock(dut.clk, 8, units='ns').start())
11      await tb_env.init_monitor('s_axis_tdata',
12          'clk', callback=parse_input_frame)
12      await tb_env.init_monitor('m_axis_tdata',
13          'clk')
13      dut.create_interface_socket_to_axis(
14          remote_address = '127.0.0.1',
15          remote_port = 20000,
16          local_port = 20002,
17          axis_bus_module_input = 's_axis',
18          axis_bus_module_output = 'm_axis')
18
19      #start simulation
20      try:
21          while True:
22              await RisingEdge(dut.clk)
23              #exit on magic number
24              if dut.s_axis_tdata.value ==
25                  65535:
26                  break
27
28      finally:
29          if __name__ == "__main__":
30              run(module = 'run_adder_host',
31                  verilog_sources =
32                      ['rtl/axis_adder.v'],
33                  toplevel = "axis_adder",
34                  timescale = "1ns/1ps",
34          )
```

# Emulation Example: Adder with Network Interface (cont'd)

Client (User Interface)

```
Enter two integers to add: 0 1
Requesting 0 + 1
Recv: 9
Received sum: 1
Enter two integers to add: 7 39
Requesting 7 + 39
Received sum: 46
Enter two integers to add: 200 132
Requesting 200 + 132
Received sum: 76
Enter two integers to add: 255 7
Requesting 255 + 7
Received sum: 6
Enter two integers to add: 25 20
Requesting 25 + 20
Received sum: 45
Enter two integers to add: 255 255
Requesting 255 + 255
Received sum: 510
Shutting down..
```



Images adapted from: 'calculator': Breathe Icon Team, CC BY-SA 3.0 via Wikimedia Commons; 'cable': R.Spekking CC BY-SA 4.0 via Wikimedia Commons; 'magnifier': <http://www.simpleicon.com/>, CC BY 3.0 via Wikimedia Commons

# Emulation: Instrument Interaction

Read out hardware revision via user software

- Original software with command file  
(or interactive mode)
- Emulated design delivers code revision  
(actual information, adjusted to scope)



# Emulation: Instrument in Context

- Let's add some physically meaningful context..
- Simple example: tx → rx loopback



# Emulation: Instrument in Context



## Emulated Design + Ideal Cable

(constant range across frequencies, peak SNR)



## Real Hardware + Real Cable

(artifacts in freq-range diagram)



# Measurement: Instrument in Arctic

Real Hardware + Antennae + Ionosphere



# CAVEAT in Training

- Communicating technical design and function to disciplinary experts can be tough
- Regular onboarding of students  
project needs  $\neq$  need to match individual experience and background

## Our Approach

- Focus on experience, learning how to use an instrument in context  
e.g. investigating parameters of an experiment
- Adopt dev-tools in support of disciplinary research  
need to be mindful of overhead
- Bonus: guidance and tools for early project phase  
informing requirements engineering et al.



# Wrapping Up

---

- Address emerging entry barrier for research environments  
before: **availability** of technology, now: **accessibility** of technology
- Expand on variety of Python packages, added value through combination  
e.g. asserting and logging physical/mixed signal peripherals
  - Focus on building working tools with intend to give back to community
  - Anticipate to leverage (and support) development of **Forastero**
- Current use: ionospheric radar, learning/teaching framework  
anticipate expansion to: steered timing reference, environmental monitoring, laser control (QSUM)

# Thank you for your attention



Get the sources, get started, get in touch:

<https://github.com/ERAS-Research/>

**Electronics in Remote and Adverse Surroundings**

<https://ERAS.ca>

**Prof. Dr.-Ing. Torsten Reuschel**

✉ [torsten.reuschel@unb.ca](mailto:torsten.reuschel@unb.ca)

🌐 <https://eras.ca>

☎ +1-506-452-6031

University of New Brunswick, Department of Physics  
8 Bailey Dr. (Room 220), Fredericton, NB E3B 5A3, Canada

# Appendix

# Community Extensions



Efficient verification thanks to abstracted interfaces:

- Common interfaces available as `cocotbext`
- AXI(S), Wishbone, UART, I2C, .. integrate based on `cocotb.queue.Queue`

```
1  from cocotbext.axi import AxiStreamBus, AxiStreamSource, AxiStreamMonitor
2
3  @cocotb.test
4  async def test_interface(dut):
5      cocotb.start_soon(Clock(dut.clk, 5, units='ns').start())
6      src = AxiStreamSource(AxiStreamBus.from_prefix(dut, 'axis'), dut.clk, dut.rst)
7      mon = AxiStreamMonitor(AxiStreamBus.from_prefix(dut, 'axis'), dut.clk, dut.rst)
8      #initialize transfer and probe monitor
9      await src.send(bytes(test_data))
10     mon_data = await mon.recv()
11     #check transfer
12     assert mon_rx.tdata == test_data, "Invalid AXIS transfer detected"
```