jtag modules for cocotb
Project description
JTAG interface modules for Cocotb
GitHub repository: https://github.com/daxzio/cocotbext-jtag
Introduction
JTAG simulation models for cocotb.
Installation
Installation from pip (release version, stable):
$ pip install cocotbext-jtag
Installation from git (latest development version, potentially unstable):
$ pip install https://github.com/daxzio/cocotbext-jtag/archive/main.zip
Installation for active development:
$ git clone https://github.com/daxzio/cocotbext-jtag
$ pip install -e cocotbext-jtag
Documentation and usage examples
See the tests directory for complete testbenches using these modules.
OpenOCD socket test (local)
To run the test where OpenOCD connects to the cocotb JTAG socket (remote bitbang):
-
Prerequisites: Icarus Verilog, OpenOCD, Python deps:
pip install -r requirements.txt pip install -e . # Optional: sudo apt-get install openocd # for full automated test
-
Full test (sim in background, then OpenOCD):
./run_openocd_socket_test.sh
-
Sim only (you run OpenOCD in another terminal):
./run_openocd_socket_test.sh --sim-only # In another terminal: cd tests/test_socket && openocd -f openocd_remote_bitbang.cfg -c "init; exit"
JTAG Bus
The JTAGBus is used to map to a JTAG interface on the dut. These hold instances of bus objects for the individual channels, which are currently extensions of cocotb_bus.bus.Bus. Class methods from_entity and from_prefix are provided to facilitate signal default name matching.
Required:
- tck
- tms
- tdi
- tdo
Optional:
- trst
JTAG Driver
The JTAGDriver class implements a JTAG driver and is capable of generating read and write operations against JTAG devices, either singularly or in a chain.
To use these modules, import the one you need and connect it to the DUT:
from cocotbext.jtag import JTAGDriver, JTAGBus, JTAGDevice
bus = JTAGBus(dut)
jtag_driver = JTAGDriver(bus)
jtag_driver.add_device(JTAGDevice())
The first argument to the constructor accepts an JTAGBus object. These objects are containers for the interface signals and include class methods to automate connections.
Once the module is instantiated, read and write operations can be initiated in a couple of different ways.
Additional optional arguments for JTAGDriver
- period: Clock frequency period of
tck, default100 - units: Clock units, default
ns - logging_enabled: Logging enable, default
True
Methods
add_device(device): Add device to jtag chain, must be of typeJTAGDeviceset_reset(num): Reset for num if trst is present inJTAGBus, raise warning if trst is not presentreset_finished(): Async wait until reset is finishedreset_fsm(): Send 5 tck pulses while tms is held high inJTAGBus, this resets the finite state machine inside a JTAG TAPsend_val(addr, val, device, write): Send addr to device (default:0). The val is used to write if write is True or verify against if write is Falsewrite(addr, val, device=0): Write val to addr of device (default:0).read(addr, val=None, device=0): Read from addr of device (default:0). If val present verify against returned value.read_idcode(device=0, retry=0): Read the IDCODE register for device and check it against the value configured on that device’sJTAGDevice. Withretry=0(default), a single read is performed. With a positiveretry, the driver may repeat an IDCODE shift and TAP reset until the captured value matches or the attempt limit is reached (see below).capture_ir(): Return the Instruction Register value captured during the most recent IR shift operationcapture_dr(): Return the Data Register value captured during the most recent DR shift operation
IDCODE retries (read_idcode(..., retry=N))
Use a positive retry when the first IDCODE read after reset or bring-up is unreliable. That often happens when the TAP or chain needs a few cycles to settle: power-on, asynchronous reset release, clock-domain crossing into the JTAG logic, or simulator startup ordering can produce a wrong or unknown value on the first DR capture even though the DUT is otherwise correct.
With a positive retry, the driver repeatedly loads the IDCODE instruction, captures the DR, compares the result to the expected idcode on the active JTAGDevice, and runs a short TAP reset between attempts. If a match is found before the limit, it continues with a normal verified read; if not, it raises after N failed attempts. When the TAP is already stable, keep the default retry=0 to avoid extra cycles.
JTAG Device
JTAGDriver needs to be told what devices are in the JTAG chain it is communicating with inside the dut. A JTAGDevice needs to be defined for this purpose or a device that inherits from this base class.
A JTAGDevice needs a name, idcode and ir_len. It also needs to have the IR Register map defined and the width of the register. 'BYPASS' is prepopulated in the base class at last address in the IRLEN address map.
First inherit from the base class, and then add all the other IR registers using the add_jtag_reg method.
from cocotbext.jtag import JTAGDevice
class J1JTAGDevice(JTAGDevice):
def __init__(self, name='jtaglet1', idcode=0x53817905, ir_len=5):
super().__init__(name, idcode, ir_len)
self.add_jtag_reg("IDCODE", 32, 0x1e)
self.add_jtag_reg('USERDATA', 32, 0x8)
self.add_jtag_reg('USEROP', 8, 0x9)
self.idle_delay = 6
add_jtag_reg(name, width, address): Add an IR register toJTAGDevice. name is a string, width is the DR shift width of the register and the address, which should be within the range of the ir_len ofJTAGDevice
This results in a JTAGDevice that has 4 IR register defined:
0x08: USERDATA[31:0]
0x09: USEROP[7:0]
0x1e: IDCODE[31:0]
0x1f: BYPASS[0:0]
Multiple devices inside the JTAG chain:
from cocotbext.jtag import JTAGDriver
from cocotbext.jtag import JTAGBus
class testbench:
def __init__(self, dut):
bus = JTAGBus(dut)
self.jtag = JTAGDriver(bus)
self.jtag.add_device(J1JTAGDevice())
self.jtag.add_device(J2JTAGDevice())
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
File details
Details for the file cocotbext_jtag-0.4.0.tar.gz.
File metadata
- Download URL: cocotbext_jtag-0.4.0.tar.gz
- Upload date:
- Size: 19.8 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
3c83a0353c79c8fc360af553d8540da0dd3375bac8a2cd8803e132da84d5b09b
|
|
| MD5 |
77938da7cc53b18d888abc133179d050
|
|
| BLAKE2b-256 |
5402eee07b7e0762c15f7090dd4acc7653320e995eaf9c6fd0e754c92566885d
|
Provenance
The following attestation bundles were made for cocotbext_jtag-0.4.0.tar.gz:
Publisher:
test_checkin.yml on daxzio/cocotbext-jtag
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
cocotbext_jtag-0.4.0.tar.gz -
Subject digest:
3c83a0353c79c8fc360af553d8540da0dd3375bac8a2cd8803e132da84d5b09b - Sigstore transparency entry: 1271691646
- Sigstore integration time:
-
Permalink:
daxzio/cocotbext-jtag@c8f6e9c556cce8cca1f8c29af28bcc5ea8da79bc -
Branch / Tag:
refs/tags/v0.4.0 - Owner: https://github.com/daxzio
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
test_checkin.yml@c8f6e9c556cce8cca1f8c29af28bcc5ea8da79bc -
Trigger Event:
push
-
Statement type: