This document specifies the processor debug system. The debug system implements run-control debug and bus access functionality according to the RISC-V Debug Specification 0.13.2. It can be accessed over JTAG (IEEE Std 1149.1-2013).
The debug system described in this document consists of two modules, which are implemented according to the RISC-V Debug Specification: The Debug Module (DM) implemented with a Program Buffer, and a JTAG Debug Transport Module (DTM).
The debug system is compatible with the RISC-V Debug Specification 0.13.2. The specification marks some features as optional. The status of these features is visible from the table below.
| Feature | Support | 
|---|---|
| External triggers (Section 3.6) | Not supported at this time. | 
| Halt groups (Section 3.6) | Not supported at this time. | 
| Hardware triggers | Not supported at this time. Triggers read 0 according to the specification. | 
| System Bus Access (SBA) (Section 3.10) | Supported with generic 32 or 64 bit wide interface. | 
| Authentication | Not supported at this time. | 
| Non-intrusive access to core registers | Not supported at this time. All registers must be accessed through the program buffer. | 
The functionality of the debug system is implemented according to the RISC-V Debug Specification. Refer to this document for more information.
An external debugger performs all interaction with the Debug Module through a register interface, accessed over a dedicated bus, the Debug Module Interface (DMI). The registers are called “Debug Module Registers” and defined in the RISC-V Debug Specification, Section 3.14. Our implementation only provides a single Debug Module on the DMI bus mapped to base address 0, hence all addresses below can be considered absolute.
| Address | Name | Implementation Notes | 
|---|---|---|
| 0x04 | Abstract Data 0 (data0) | |
| 0x0f | Abstract Data 11 (data11) | |
| 0x10 | Debug Module Control (dmcontrol) | see table below | 
| 0x11 | Debug Module Status (dmstatus) | see table below | 
| 0x12 | Hart Info (hartinfo) | |
| 0x13 | Halt Summary 1 (haltsum1) | |
| 0x14 | Hart Array Window Select (hawindowsel) | |
| 0x15 | Hart Array Window (hawindow) | |
| 0x16 | Abstract Control and Status (abstractcs) | |
| 0x17 | Abstract Command (command) | |
| 0x18 | Abstract Command Autoexec (abstractauto) | |
| 0x19 | Configuration String Pointer 0 (confstrptr0) | |
| 0x1a | Configuration String Pointer 1 (confstrptr1) | |
| 0x1b | Configuration String Pointer 2 (confstrptr2) | |
| 0x1c | Configuration String Pointer 3 (confstrptr3) | |
| 0x1d | Next Debug Module (nextdm) | |
| 0x1f | Custom Features (custom) | |
| 0x20 | Program Buffer 0 (progbuf0) | |
| 0x2f | Program Buffer 15 (progbuf15) | |
| 0x30 | Authentication Data (authdata) | |
| 0x32 | Debug Module Control and Status 2 (dmcs2) | |
| 0x34 | Halt Summary 2 (haltsum2) | |
| 0x35 | Halt Summary 3 (haltsum3) | |
| 0x37 | System Bus Address 127:96 (sbaddress3) | |
| 0x38 | System Bus Access Control and Status (sbcs) | |
| 0x39 | System Bus Address 31:0 (sbaddress0) | |
| 0x3a | System Bus Address 63:32 (sbaddress1) | |
| 0x3b | System Bus Address 95:64 (sbaddress2) | |
| 0x3c | System Bus Data 31:0 (sbdata0) | |
| 0x3d | System Bus Data 63:32 (sbdata1) | |
| 0x3e | System Bus Data 95:64 (sbdata2) | |
| 0x3f | System Bus Data 127:96 (sbdata3) | |
| 0x40 | Halt Summary 0 (haltsum0) | 
| Field | Access | (Reset) Value | Comment | 
|---|---|---|---|
| haltreq | WARZ | ||
| resumereq | W1 | ||
| hartreset | WARL | 0 | Not implemented, reads constant 0 | 
| ackhavereset | W1 | ||
| hasel | WARL | 0 | Hart array masks are not supported | 
| hartsello | R/W | ||
| hartselhi | R/W | ||
| setresethaltreq | W1 | - | Writing this register is a no-op. Halt-on-reset is not implemented, as indicated by dmstatus.hasresethaltreq. | 
| clrresethaltreq | W1 | - | Writing this register is a no-op. Halt-on-reset is not implemented, as indicated by dmstatus.hasresethaltreq. | 
| ndmreset | R/W | ||
| dmactive | R/W | 
| Field | Access | (Reset) Value | Comment | 
|---|---|---|---|
| impebreak | R | 0 | No implicit ebreak is inserted after the Program Buffer. | 
| allhavereset | R | Hart array masks are not supported; identical to anyhavereset. | |
| anyhavereset | R | Hart array masks are not supported; identical to allhavereset. | |
| allresumeack | R | Hart array masks are not supported; identical to anyresumeack. | |
| anyresumeack | R | Hart array masks are not supported; identical to allresumeack. | |
| allnonexistent | R | Hart array masks are not supported; identical to anynonexistent | |
| anynonexistent | R | Hart array masks are not supported; identical to allnonexistent | |
| allunavail | R | Hart array masks are not supported; identical to anyunavail. | |
| anyunavail | R | Hart array masks are not supported; identical to allunavail. | |
| allrunning | R | Hart array masks are not supported; identical to anyrunning. | |
| anyrunning | R | Hart array masks are not supported; identical to allrunning. | |
| allhalted | R | Hart array masks are not supported; identical to anyhalted. | |
| anyhalted | R | Hart array masks are not supported; identical to allhalted. | |
| authenticated | R | 1 | Authentication is not implemented, reads always 1. | 
| authbusy | R | 0 | Authentication is not implemented, reads always 0. | 
| hasresethaltreq | R | 0 | Halt-on-reset is not implemented, reads always 0. | 
| confstrptrvalid | R | 0 | Configuration strings are not supported. | 
| version | R | 2 | Specification version 0.13 | 
The debug system can be configured at synthesis time through parameters.
| Parameter Name | Valid values | Default | Description | 
|---|---|---|---|
| NrHarts | 1..2^20 | 1 | Number of connected harts | 
| BusWidth | 32, 64 | 32 | Bus width (for debug memory and SBA busses) | 
| SelectableHarts | 1 | Bitmask to select physically available harts for systems that don't use hart numbers in a contiguous fashion. | 
In addition to these parameters, additional configuration is provided through top-level signals, which are expected to be set to a fixed value at synthesis time.
| Signal Name | Data Type | Width | Description | 
|---|---|---|---|
| hartinfo | hartinfo_t | NrHarts | Value of the hartinfo DM register, see the RISC-V Debug Specification v0.13, Section 3.14.2 for details. nscratch: Number of debug scratch registers. Must be set to 2. dataaccess: Must be set to 1. (The data register are shadowed in the hart's memory.) datasize: Must be set to dm::DataCount. dataaddr: Must be set todm::DataAddr(0x380). | 
SystemVerilog definition of the hartinfo_t structure
typedef struct packed { logic [31:24] zero1; logic [23:20] nscratch; logic [19:17] zero0; logic dataaccess; logic [15:12] datasize; logic [11:0] dataaddr; } hartinfo_t;
The debug system interacts with multiple components, which are described in this section. The Debug Module interacts the debugged hart(s) through the core interface, with the system bus through its bus host and device, and with system management and reset functionality, and with the Debug Transport Module through the Debug Module Interface (DMI).
The JTAG Debug Transport Module interacts with the host PC through JTAG, and with the Debug Module through the Debug Module Interface (DMI).
| Signal | Direction | Description | 
|---|---|---|
| clk_i | input | clock. All components of the debug module and the DMI are synchronous to this clock, except for the JTAG components, which are clocked with the external TCLK. | 
| rst_ni | input | asynchronous, active low reset. | 
| testmode_i | input | not used currently | 
| ndmreset_o | output | Non-Debug Mode reset (ndmreset). ndmreset is triggered externally through JTAG, e.g. by a debugger, and should reset the whole system except for the debug system. See the RISC-V Debug Specification v0.13, Section 3.2 (Reset Control) for more details. | 
| dmactive_o | output | debug module is active | 
SystemVerilog interface definition
input logic clk_i, input logic rst_ni, input logic testmode_i, output logic ndmreset_o, output logic dmactive_o
The debug system is compatible with any RISC-V compliant CPU core, given that it support execution based debugging according to the RISC-V Debug Specification, Section A.2.
The debug request interrupt is a level-sensitive interrupt. It is issued by the Debug Module to the hart(s).
| Signal | Direction | Description | 
|---|---|---|
| debug_req_o[NrHarts-1:0] | output | Level-sensitive debug request interrupt | 
The Debug Module issues the debug request to request the core to enter its debug mode. In consequence, the core is expected to
| Signal | Direction | Description | 
|---|---|---|
| unavailable_i[NrHarts-1] | input | Set to 0 to mark the hart has unavailable (e.g.: power down). This information is made available to the debugger through the dmstatus.allunavail and dmstatus.anyavail fields. | 
Four debug-specific CSRs must be supported by the core, as described in the Debug Specification Section 4.8.
| Address | Name | Access | Description | 
|---|---|---|---|
| 0x7b0 | dcsr | field-dependent; check spec | Debug Control and Status | 
| 0x7b1 | dpc | RW from Debug Mode only | Debug PC | 
| 0x7b2 | dscratch0 | RW from Debug Mode only | Debug Scratch Register 0 | 
| 0x7b3 | dscratch1 | RW from Debug Mode only | Debug Scratch Register 1 (only required if DM is not located at address 0x0) | 
To return from debug mode the DRET instruction must be supported by the core according to the Debug Specification Section 4.7.
The JTAG Debug Transport Module (JTAG DTM) provides a standard IEEE 1149.1 JTAG connection.
| Signal | Direction | Description | 
|---|---|---|
| tck_i | input | Test Clock | 
| tms_i | input | Test Mode Select | 
| td_i | input | Test Data In (host to debug system) | 
| td_o | output | Test Data Out (debug system to host) | 
| tdo_oe_o | output | TDO output enable | 
| trst_ni | input | Test Reset (active low). Usage of TRST is optional, but recommended for reliable reset functionality. | 
The Debug Module connects to the system bus as device, exposing the debug memory (the Program Buffer and the Debug ROM), and as host for the System Bus Access (SBA) functionality. The same generic bus interface is used in both cases. The bus width is configurable to be 32 or 64 bit using the BusWidth parameter.
The bus host interface is compatible to the instruction and data interfaces of Ibex.
| Signal | Width (bit) | Direction | Description | 
|---|---|---|---|
| req | 1 | output | Request valid, must stay high until gnt is high for one cycle | 
| add | BusWidth | output | Address, word aligned | 
| we | BusWidth | output | Write Enable, high for writes, low for reads. Sent together with req | 
| be | 1 | output | Byte Enable. Is set for the bytes to write/read, sent together with req | 
| wdata | BusWidth | output | Data to be written to device, sent together with req | 
| gnt | 1 | input | The device accepted the request. Host outputs may change in the next cycle. | 
| r_valid | 1 | input | r_rdata hold valid data when r_valid is high. This signal will be high for exactly one cycle per request. | 
| r_rdata | BusWidth | input | Data read from the device | 
No error response is currently implemented.
SystemVerilog interface definition (host side)
output logic req_o, output logic [BusWidth-1:0] add_o, output logic we_o, output logic [BusWidth-1:0] wdata_o, output logic [BusWidth/8-1:0] be_o, input logic gnt_i, input logic r_valid_i, input logic [BusWidth-1:0] r_rdata_i
SystemVerilog interface definition (device side)
input logic slave_req_i, input logic slave_we_i, input logic [BusWidth-1:0] slave_addr_i, input logic [BusWidth/8-1:0] slave_be_i, input logic [BusWidth-1:0] slave_wdata_i, output logic [BusWidth-1:0] slave_rdata_o
A wrapper (called dm_obi_top) is provided which wraps the Debug Module (dm_top) and makes it OBI compliant. This wrapper can be ignored (and dm_top can be used directly instead) in case of non OBI compatible systems.
The OBI (Open Bus Interface) specification is at https://github.com/openhwgroup/core-v-docs/blob/master/cores/cv32e40p/.
The Debug Module connects to the system bus as device, exposing the debug memory (the Program Buffer and the Debug ROM), and as host for the System Bus Access (SBA) functionality. The bus interface is according to the OBI specification in both cases. The bus width is configurable to be 32 or 64 bit using the BusWidth parameter. The transfer identifier width is configurable using the IdWidth parameter.
Compared to dm_top the slave interface of dm_obi_top has the following additional signals: slave_gnt_o, slave_rvalid_o, slave_aid_i, slave_rid_o. Compared to dm_top the master interface of dm_obi_top has some renamed signals (master_addr_o, master_rvalid_i, master_rdata_i instead of master_add_o, master_r_valid_i, master_r_rdata_i).
Both interfaces are OBI compliant.
| Signal | Width (bit) | Direction | Description | 
|---|---|---|---|
| req | 1 | output | Request valid, must stay high until gnt is high for one cycle | 
| addr | BusWidth | output | Address, word aligned | 
| we | BusWidth | output | Write Enable, high for writes, low for reads. Sent together with req | 
| be | 1 | output | Byte Enable. Is set for the bytes to write/read, sent together with req | 
| wdata | BusWidth | output | Data to be written to device, sent together with req | 
| gnt | 1 | input | The device accepted the request. Host outputs may change in the next cycle. | 
| rvalid | 1 | input | r_rdata hold valid data when r_valid is high. This signal will be high for exactly one cycle per request. | 
| rdata | BusWidth | input | Data read from the device | 
No error response is currently implemented.
SystemVerilog interface definition (host side)
output logic master_req_o, output logic [BusWidth-1:0] master_addr_o, output logic master_we_o, output logic [BusWidth-1:0] master_wdata_o, output logic [BusWidth/8-1:0] master_be_o, input logic master_gnt_i, input logic master_rvalid_i, input logic [BusWidth-1:0] master_rdata_i
SystemVerilog interface definition (device side)
input logic slave_req_i, output logic slave_gnt_o, input logic slave_we_i, input logic [BusWidth-1:0] slave_addr_i, input logic [BusWidth/8-1:0] slave_be_i, input logic [BusWidth-1:0] slave_wdata_i, input logic [IdWidth-1:0] slave_aid_i, output logic slave_rvalid_o, output logic [BusWidth-1:0] slave_rdata_o, output logic [IdWidth-1:0] slave_rid_o
The Debug Module Interface is a bus connecting the JTAG Debug Transport Module (DTM) with the Debug Module (DM). The Debug Specification does not require a specific implementation or hardware interface of the DMI bus. Our implementation is inspired by the implementation used by the Rocket Chip Generator.
Transfers on the DMI are performed on two channels, a request and a response channel. A valid and a ready signal on each channel is used for handshaking.
A DMI request is issued from a DMI host to a DMI device.
| Signal | Width (bit) | Direction | Description | 
|---|---|---|---|
| addr | 7 | host -> device | Address of a debug module register (c.f. Section 3.14 of the RISC-V Debug Specification) | 
| op | 2 | host -> device | Performed operation: DTM_NOP = 2‘h0, DTM_READ = 2’h1, DTM_WRITE = 2'h2 | 
| data | 32 | host -> device | write data | 
| valid | 1 | host -> device | handshaking: request is valid | 
| ready | 1 | device -> host | handshaking: transaction processed | 
SystemVerilog interface definition (host side)
typedef enum logic [1:0] { DTM_NOP = 2'h0, DTM_READ = 2'h1, DTM_WRITE = 2'h2 } dtm_op_e; typedef struct packed { logic [6:0] addr; dtm_op_e op; logic [31:0] data; } dmi_req_t; output logic dmi_req_valid_o, input logic dmi_req_ready_i, output dmi_req_t dmi_req_o
| Signal | Width (bit) | Direction | Description | 
|---|---|---|---|
| resp | 2 | device -> host | Result of the operation. DTM_SUCCESS = 2'h0 | 
| data | 32 | device -> host | requested read data | 
| valid | 1 | device -> host | handshaking: response is valid | 
| ready | 1 | host -> device | handshaking: transaction processed | 
SystemVerilog interface definition (host side)
typedef struct packed { logic [31:0] data; logic [1:0] resp; } dmi_resp_t; input dmi_resp_t dmi_resp_i, output logic dmi_resp_ready_o, input logic dmi_resp_valid_i
A DMI transaction consists of a request and a response. A DMI host issues a request on the request channel. The request is transmitted as soon as both ready and valid are high. After some time, the device sends the answers the request on the response channel. The response is considered sent as soon as ready and valid of the response channel are both high. A transaction is completed as soon as the response is received by the host. Only one transaction may be active at any given time.
The following timing diagram shows two DMI transactions.
A READ transaction, reading address 0x12 successfully, the value 0x1234 is returned. A WRITE transaction, writing the value 0xabcd to address 0x34 successfully.
The system bus and all attached peripherals, including the memories, can be accessed from the host system through the System Bus Access (SBA) component. A typical use case is writing the program memory through this interface, or verifying its contents. The implementation conforms to the RISC-V Debug Specification v0.13, refer to this document for further information.
The Debug Module exposes a 16 kB memory over its device bus interface. This memory is called the Debug Memory. It consists of a ROM portion, containing the Debug ROM, multiple memory-mapped control and status registers, and a RAM portion, the Program Buffer. The Debug Memory should only be accessible from the CPU if it is in debug mode.
The memory map is an implementation detail of the Debug Module and should not be relied on when using the Debug Module.
| Address | Description | 
|---|---|
| 0x0 to 0x0ff | unused | 
| 0x100 | Halted. Write to this address to acknowledge that the core is halted. | 
| 0x104 | Going. Write to this address to acknowledge that the core is executing. | 
| 0x108 | Resuming. Write to this address to acknowledge that the core is resuming non-debug operation. | 
| 0x10c | Exception. An exception was triggered while the core was in debug mode. | 
| 0x300 | WhereTo | 
| 0x338 to 0x35f | AbstractCmd | 
| 0x360 to 0x37f | Program Buffer (8 words) | 
| 0x380 to 0x388 | DataAddr | 
| 0x400 to 0x7ff | Flags | 
| 0x800 to 0x1000 | Debug ROM | 
| 0x800 | HaltAddress. Entry point into the Debug Module. The core must jump to this address when it was requested to halt. | 
| 0x804 | ResumeAddress. Entry point into the Debug Module. Jumping to this address instructs the debug module to bring the core out of debug mode and back into normal operation mode. | 
| 0x808 | ExceptionAddress. Entry point into the Debug Module. The core must jump to this address when it receives an exception while being in debug mode. | 
(Note: The debug memory addressing scheme is adopted from the Rocket Chip Generator.)