Skip to content

Commit

Permalink
Merge pull request #257 from Timmmm/dev/fix_mcountinhibit
Browse files Browse the repository at this point in the history
Fix minstret off-by-one when mcountinhibit is set
  • Loading branch information
ptomsich authored May 29, 2023
2 parents f78532a + 9d9527e commit c57aebc
Show file tree
Hide file tree
Showing 4 changed files with 18 additions and 12 deletions.
4 changes: 2 additions & 2 deletions model/riscv_insts_zicsr.sail
Original file line number Diff line number Diff line change
Expand Up @@ -217,9 +217,9 @@ function writeCSR (csr : csreg, value : xlenbits) -> unit = {

/* machine mode counters */
(0xB00, _) => { mcycle[(sizeof(xlen) - 1) .. 0] = value; Some(value) },
(0xB02, _) => { minstret[(sizeof(xlen) - 1) .. 0] = value; minstret_written = true; Some(value) },
(0xB02, _) => { minstret[(sizeof(xlen) - 1) .. 0] = value; minstret_increment = false; Some(value) },
(0xB80, 32) => { mcycle[63 .. 32] = value; Some(value) },
(0xB82, 32) => { minstret[63 .. 32] = value; minstret_written = true; Some(value) },
(0xB82, 32) => { minstret[63 .. 32] = value; minstret_increment = false; Some(value) },

/* trigger/debug */
(0x7a0, _) => { tselect = value; Some(tselect) },
Expand Down
8 changes: 7 additions & 1 deletion model/riscv_step.sail
Original file line number Diff line number Diff line change
Expand Up @@ -73,7 +73,13 @@ function step(step_no : int) -> bool = {
/* for step extensions */
ext_pre_step_hook();

minstret_written = false; /* see note for minstret */
// This records whether or not minstret should be incremented when the
// instruction is retired. Since retirement occurs before CSR writes we
// initialise it based on mcountinhibit here, before it is potentially
// changed. This is also set to false if minstret is written.
// See the note near the minstret declaration for more information.
minstret_increment = mcountinhibit.IR() == 0b0;

let (retired, stepped) : (Retired, bool) =
match dispatchInterrupt(cur_privilege) {
Some(intr, priv) => {
Expand Down
2 changes: 1 addition & 1 deletion model/riscv_sys_control.sail
Original file line number Diff line number Diff line change
Expand Up @@ -594,7 +594,7 @@ function init_sys() -> unit = {
mcounteren->bits() = EXTZ(0b0);

minstret = EXTZ(0b0);
minstret_written = false;
minstret_increment = true;

init_pmp();

Expand Down
16 changes: 8 additions & 8 deletions model/riscv_sys_regs.sail
Original file line number Diff line number Diff line change
Expand Up @@ -490,7 +490,7 @@ register mepc : xlenbits
* When misa.C is writable, it zeroes only xepc[0].
*/
function legalize_xepc(v : xlenbits) -> xlenbits =
/* allow writing xepc[1] only if misa.C is enabled or could be enabled
/* allow writing xepc[1] only if misa.C is enabled or could be enabled
XXX specification says this legalization should be done on read */
if (sys_enable_writable_misa() & sys_enable_rvc()) | misa.C() == 0b1
then [v with 0 = bitzero]
Expand Down Expand Up @@ -557,17 +557,17 @@ register mtime : bits(64)
* simulation loop, we need to execute an instruction to find out
* whether it retired, and hence can only increment instret after
* execution. To avoid doing this in the case minstret was explicitly
* written to, we track writes to it in a separate model-internal
* register.
* written to, we track whether it should increment in a separate
* model-internal register.
*/
register minstret : bits(64)
register minstret_written : bool

// Should minstret be incremented when the instruction is retired.
register minstret_increment : bool

function retire_instruction() -> unit = {
if minstret_written == true
then minstret_written = false
else if mcountinhibit.IR() == 0b0
then minstret = minstret + 1
if minstret_increment
then minstret = minstret + 1;
}

/* informational registers */
Expand Down

0 comments on commit c57aebc

Please sign in to comment.