opentitanlib/debug/
dmi.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
// Copyright lowRISC contributors (OpenTitan project).
// Licensed under the Apache License, Version 2.0, see LICENSE for details.
// SPDX-License-Identifier: Apache-2.0

use std::ops::{Deref, DerefMut};
use std::time::Duration;

use anyhow::{bail, ensure, Result};
use thiserror::Error;

use super::openocd::OpenOcd;
use crate::test_utils::poll::poll_until;

/// Constants defined by RISC-V Debug Specification 0.13.
pub mod consts {
    // JTAG registers.
    pub const DTMCS: u32 = 0x10;
    pub const DMI: u32 = 0x11;

    pub const DTMCS_VERSION_SHIFT: u32 = 0;
    pub const DTMCS_ABITS_SHIFT: u32 = 4;
    pub const DTMCS_DMIRESET_SHIFT: u32 = 16;

    pub const DTMCS_VERSION_MASK: u32 = 0xf << DTMCS_VERSION_SHIFT;
    pub const DTMCS_ABITS_MASK: u32 = 0x3f << DTMCS_ABITS_SHIFT;
    pub const DTMCS_DMIRESET_MASK: u32 = 1 << DTMCS_DMIRESET_SHIFT;

    pub const DTMCS_VERSION_0_13: u32 = 1;

    pub const DMI_ADDRESS_SHIFT: u32 = 34;
    pub const DMI_DATA_SHIFT: u32 = 2;

    pub const DMI_OP_READ: u64 = 0x1;
    pub const DMI_OP_WRITE: u64 = 0x2;

    // Debug module registers.
    pub const DATA0: u32 = 0x04;
    pub const DATA1: u32 = 0x05;
    pub const DMCONTROL: u32 = 0x10;
    pub const DMSTATUS: u32 = 0x11;
    pub const HARTINFO: u32 = 0x12;
    pub const ABSTRACTCS: u32 = 0x16;

    pub const DMSTATUS_ANYHALTED_MASK: u32 = 1 << 8;
    pub const DMSTATUS_ANYRUNNING_MASK: u32 = 1 << 10;
    pub const DMSTATUS_ANYUNAVAIL_MASK: u32 = 1 << 12;
    pub const DMSTATUS_ANYNONEXISTENT_MASK: u32 = 1 << 14;
    pub const DMSTATUS_ANYRESUMEACK_MASK: u32 = 1 << 16;
    pub const DMSTATUS_ANYHAVERESET_MASK: u32 = 1 << 18;
    pub const DMSTATUS_ALLHAVERESET_MASK: u32 = 1 << 19;

    pub const DMCONTROL_HASEL_SHIFT: u32 = 26;
    pub const DMCONTROL_HARTSELHI_SHIFT: u32 = 6;
    pub const DMCONTROL_HARTSELLO_SHIFT: u32 = 16;

    pub const DMCONTROL_DMACTIVE_MASK: u32 = 1 << 0;
    pub const DMCONTROL_NDMRESET_MASK: u32 = 1 << 1;
    pub const DMCONTROL_ACKHAVERESET_MASK: u32 = 1 << 28;
    pub const DMCONTROL_RESUMEREQ_MASK: u32 = 1 << 30;
    pub const DMCONTROL_HALTREQ_MASK: u32 = 1 << 31;

    pub const ABSTRACTCS_CMDERR_MASK: u32 = (1 << 11) - (1 << 8);
    pub const ABSTRACTCS_BUSY_MASK: u32 = 1 << 12;

    pub const ABSTRACTCS_CMDERR_SHIFT: u32 = 8;

    pub const ABSTRACTCS_CMDERR_NONE: u32 = 0;
}

use consts::*;

/// Debug module interface (DMI) abstraction.
pub trait Dmi {
    /// Read a DMI register.
    fn dmi_read(&mut self, addr: u32) -> Result<u32>;

    /// Write a DMI register.
    fn dmi_write(&mut self, addr: u32, data: u32) -> Result<()>;
}

impl<T: Dmi> Dmi for &mut T {
    fn dmi_read(&mut self, addr: u32) -> Result<u32> {
        T::dmi_read(self, addr)
    }

    fn dmi_write(&mut self, addr: u32, data: u32) -> Result<()> {
        T::dmi_write(self, addr, data)
    }
}

/// DMI interface via OpenOCD.
pub struct OpenOcdDmi {
    openocd: OpenOcd,
    tap: String,
    abits: u32,
}

impl OpenOcdDmi {
    /// Create a new DMI interface via OpenOCD.
    ///
    /// This should be an OpenOCD instance with JTAG scan chain already set up,
    /// but not with target set up. If target has been set up, OpenOCD will access
    /// DMI registers on its own, which will interfere with raw DMI operations.
    pub fn new(mut openocd: OpenOcd, tap: &str) -> Result<Self> {
        let target_names = openocd.execute("target names")?;
        ensure!(
            target_names.is_empty(),
            "Target must not be setup when accessing DMI directly"
        );

        openocd.irscan(tap, DTMCS)?;
        let res = openocd.drscan(tap, 32, DTMCS_DMIRESET_MASK)?;
        let version = (res & DTMCS_VERSION_MASK) >> DTMCS_VERSION_SHIFT;
        let abits = (res & DTMCS_ABITS_MASK) >> DTMCS_ABITS_SHIFT;

        ensure!(
            version == DTMCS_VERSION_0_13,
            "DTMCS indicates version other than 0.13"
        );

        openocd.irscan(tap, DMI)?;
        Ok(Self {
            openocd,
            tap: tap.to_owned(),
            abits,
        })
    }

    fn dmi_op(&mut self, op: u64) -> Result<u64> {
        let res = self
            .openocd
            .drscan(&self.tap, self.abits + DMI_ADDRESS_SHIFT, op)?;

        // We just scanned into the DMI register, so the scanned result should be empty.
        ensure!(res == 0, "Unexpected DMI initial response {res:#x}");

        // Run the DMI operation.
        // TODO: The proper way is to run for a small number of cycles, and then try to read the result.
        // If an error occurs indicating that the number of cycles are not sufficient, then increase that number
        // and try again. Here we just use a large enough number to avoid having to implement the retry logic,
        // which is good enough for now.
        self.openocd.execute("runtest 10")?;

        // Read the result.
        let res = self
            .openocd
            .drscan(&self.tap, self.abits + DMI_ADDRESS_SHIFT, 0)?;
        ensure!(res & 3 == 0, "DMI operation failed with {res:#x}");

        // Double check the address matches.
        ensure!(
            res >> DMI_ADDRESS_SHIFT == op >> DMI_ADDRESS_SHIFT,
            "DMI operation address mismatch {res:#x}"
        );

        Ok(res)
    }
}

impl Dmi for OpenOcdDmi {
    fn dmi_read(&mut self, addr: u32) -> Result<u32> {
        let output = (self.dmi_op((addr as u64) << DMI_ADDRESS_SHIFT | DMI_OP_READ)?
            >> DMI_DATA_SHIFT) as u32;
        log::info!("DMI read {:#x} -> {:#x}", addr, output);
        Ok(output)
    }

    fn dmi_write(&mut self, addr: u32, value: u32) -> Result<()> {
        self.dmi_op(
            (addr as u64) << DMI_ADDRESS_SHIFT | (value as u64) << DMI_DATA_SHIFT | DMI_OP_WRITE,
        )?;
        log::info!("DMI write {:#x} <- {:#x}", addr, value);
        Ok(())
    }
}

#[derive(Debug, Error)]
pub enum DmiError {
    #[error("Hart does not exist")]
    Nonexistent,
    #[error("Hart is not currently available")]
    Unavailable,
    #[error("Timeout waiting for hart to halt")]
    WaitTimeout,
}

/// A debugger that communicates with the target via RISC-V Debug Module Interface (DMI).
pub struct DmiDebugger<D> {
    dmi: D,
    hartsel_mask: Option<u32>,
}

impl<D> Deref for DmiDebugger<D> {
    type Target = D;

    fn deref(&self) -> &Self::Target {
        &self.dmi
    }
}

impl<D> DerefMut for DmiDebugger<D> {
    fn deref_mut(&mut self) -> &mut Self::Target {
        &mut self.dmi
    }
}

impl<D: Dmi> DmiDebugger<D> {
    pub fn new(dmi: D) -> Self {
        Self {
            dmi,
            hartsel_mask: None,
        }
    }

    /// Obtain bits valid in hartsel as a bitmask.
    pub fn hartsel_mask(&mut self) -> Result<u32> {
        if self.hartsel_mask.is_none() {
            // Write all 1s to hartsel.
            let dm_control = 0 << DMCONTROL_HASEL_SHIFT
                | 0x3ff << DMCONTROL_HARTSELLO_SHIFT
                | 0x3ff << DMCONTROL_HARTSELHI_SHIFT
                | DMCONTROL_DMACTIVE_MASK;
            self.dmi.dmi_write(DMCONTROL, dm_control)?;

            // This is a WARL register so after writing 1, the readback value would be
            // the mask for valid bits in the register.
            let dm_control = self.dmi.dmi_read(DMCONTROL)?;
            let hart_select = (dm_control >> DMCONTROL_HARTSELLO_SHIFT) & 0x3ff
                | ((dm_control >> DMCONTROL_HARTSELHI_SHIFT) & 0x3ff) << 10;

            self.hartsel_mask = Some(hart_select);
        }

        Ok(self.hartsel_mask.unwrap())
    }

    /// Selects a hart to debug.
    pub fn select_hart(&mut self, hartid: u32) -> Result<DmiHart<'_, D>> {
        // The hart selection is up to 20 bits.
        if hartid >= (1 << 20) {
            bail!("Invalid hartid: {hartid}");
        }

        // When selecting non-zero hart, ensure written bit to HARTSEL is legal.
        if hartid != 0 {
            let mask = self.hartsel_mask()?;
            if (hartid & mask) != hartid {
                bail!(DmiError::Nonexistent);
            }
        }

        let hart_select = 0 << DMCONTROL_HASEL_SHIFT
            | (hartid & 0x3ff) << DMCONTROL_HARTSELLO_SHIFT
            | (hartid >> 10) << DMCONTROL_HARTSELHI_SHIFT
            | DMCONTROL_DMACTIVE_MASK;
        self.dmi.dmi_write(DMCONTROL, hart_select)?;

        let mut hart = DmiHart {
            debugger: self,
            hart_select,
        };

        let dmstatus = hart.dmstatus()?;
        if dmstatus & DMSTATUS_ANYNONEXISTENT_MASK != 0 {
            bail!(DmiError::Nonexistent);
        }
        if dmstatus & DMSTATUS_ANYUNAVAIL_MASK != 0 {
            bail!(DmiError::Unavailable);
        }

        Ok(hart)
    }

    /// Read a data register from DMI.
    pub fn data(&mut self, idx: u32) -> Result<u32> {
        ensure!(idx < 12, "data register index out of range {:#x}", idx);
        self.dmi_read(DATA0 + idx)
    }

    /// Write a data register from DMI.
    pub fn set_data(&mut self, idx: u32, data: u32) -> Result<()> {
        ensure!(idx < 12, "data register index out of range {:#x}", idx);
        self.dmi_write(DATA0 + idx, data)
    }
}

/// A DMI debugger with specific hart selected.
pub struct DmiHart<'a, D> {
    debugger: &'a mut DmiDebugger<D>,

    /// The value of DMCONTROL with hasel, hartsello and hartselhi set.
    hart_select: u32,
}

impl<D> Deref for DmiHart<'_, D> {
    type Target = DmiDebugger<D>;

    fn deref(&self) -> &Self::Target {
        self.debugger
    }
}

impl<D> DerefMut for DmiHart<'_, D> {
    fn deref_mut(&mut self) -> &mut Self::Target {
        self.debugger
    }
}

/// State of the hart.
///
/// If both `running` and `halted` are false, then the hart is in the process of transitioning between
/// the two states (i.e. resuming or halting).
pub struct HartState {
    pub running: bool,
    pub halted: bool,
}

impl<D: Dmi> DmiHart<'_, D> {
    /// Read `dmstatus` for the selected hart.
    pub fn dmstatus(&mut self) -> Result<u32> {
        let dmstatus = self.debugger.dmi_read(DMSTATUS)?;

        // `dmstatus` register have fields for any hart and all harts. If only a single hart is selected,
        // then the "all" and "any" values should match. This performs a sanity check.
        if (dmstatus ^ (dmstatus >> 1))
            & (DMSTATUS_ANYHALTED_MASK
                | DMSTATUS_ANYRUNNING_MASK
                | DMSTATUS_ANYUNAVAIL_MASK
                | DMSTATUS_ANYNONEXISTENT_MASK
                | DMSTATUS_ANYRESUMEACK_MASK
                | DMSTATUS_ANYHAVERESET_MASK)
            != 0
        {
            bail!(
                "Invalid dmstatus {:#x}: any and all bits mismatch",
                dmstatus
            );
        }

        Ok(dmstatus)
    }

    /// Write to dmcontrol without affecting hart selection.
    pub fn set_dmcontrol(&mut self, value: u32) -> Result<()> {
        self.debugger.dmi_write(DMCONTROL, value | self.hart_select)
    }

    /// Read hart info of the selected hart.
    pub fn hartinfo(&mut self) -> Result<u32> {
        self.debugger.dmi_read(HARTINFO)
    }

    /// Return the state of the hart.
    pub fn state(&mut self) -> Result<HartState> {
        let dmstatus = self.dmstatus()?;
        let running = dmstatus & DMSTATUS_ANYRUNNING_MASK != 0;
        let halted = dmstatus & DMSTATUS_ANYHALTED_MASK != 0;
        assert!(!(running && halted));
        Ok(HartState { running, halted })
    }

    /// Set the halt request bit.
    pub fn set_halt_request(&mut self, active: bool) -> Result<()> {
        self.set_dmcontrol(if active { DMCONTROL_HALTREQ_MASK } else { 0 })
    }

    /// Wait for the hart to halt.
    pub fn wait_halt(&mut self) -> Result<()> {
        // Per RISC-V debug specification, harts must respond within 1 second of receiving a halt or
        // resume request.
        poll_until(Duration::from_secs(1), Duration::from_millis(50), || {
            Ok(self.state()?.halted)
        })
    }

    /// Set the resume request bit.
    pub fn set_resume_request(&mut self, active: bool) -> Result<()> {
        self.set_dmcontrol(if active { DMCONTROL_RESUMEREQ_MASK } else { 0 })
    }

    /// Wait for the hart to resume.
    pub fn wait_resume(&mut self) -> Result<()> {
        // Per RISC-V debug specification, harts must respond within 1 second of receiving a halt or
        // resume request.
        poll_until(Duration::from_secs(1), Duration::from_secs(1), || {
            Ok(self.state()?.running)
        })
    }
}