Pārlūkot izejas kodu

target/riscv: Fix mcycle/minstret increment behavior

The mcycle/minstret counter's stop flag is mistakenly updated on a copy
on stack. Thus the counter increments even when the CY/IR bit in the
mcountinhibit register is set. This commit corrects its behavior.

Fixes: 3780e33732f88 (target/riscv: Support mcycle/minstret write operation)
Signed-off-by: Xu Lu <luxu.kernel@bytedance.com>
Reviewed-by: Daniel Henrique Barboza <dbarboza@ventanamicro.com>
Signed-off-by: Michael Tokarev <mjt@tls.msk.ru>
Xu Lu 1 gadu atpakaļ
vecāks
revīzija
5cb0e7abe1
1 mainītis faili ar 7 papildinājumiem un 7 dzēšanām
  1. 7 7
      target/riscv/csr.c

+ 7 - 7
target/riscv/csr.c

@@ -907,11 +907,11 @@ static int write_mhpmcounterh(CPURISCVState *env, int csrno, target_ulong val)
 static RISCVException riscv_pmu_read_ctr(CPURISCVState *env, target_ulong *val,
 static RISCVException riscv_pmu_read_ctr(CPURISCVState *env, target_ulong *val,
                                          bool upper_half, uint32_t ctr_idx)
                                          bool upper_half, uint32_t ctr_idx)
 {
 {
-    PMUCTRState counter = env->pmu_ctrs[ctr_idx];
-    target_ulong ctr_prev = upper_half ? counter.mhpmcounterh_prev :
-                                         counter.mhpmcounter_prev;
-    target_ulong ctr_val = upper_half ? counter.mhpmcounterh_val :
-                                        counter.mhpmcounter_val;
+    PMUCTRState *counter = &env->pmu_ctrs[ctr_idx];
+    target_ulong ctr_prev = upper_half ? counter->mhpmcounterh_prev :
+                                         counter->mhpmcounter_prev;
+    target_ulong ctr_val = upper_half ? counter->mhpmcounterh_val :
+                                        counter->mhpmcounter_val;
 
 
     if (get_field(env->mcountinhibit, BIT(ctr_idx))) {
     if (get_field(env->mcountinhibit, BIT(ctr_idx))) {
         /*
         /*
@@ -919,12 +919,12 @@ static RISCVException riscv_pmu_read_ctr(CPURISCVState *env, target_ulong *val,
          * stop the icount counting. Just return the counter value written by
          * stop the icount counting. Just return the counter value written by
          * the supervisor to indicate that counter was not incremented.
          * the supervisor to indicate that counter was not incremented.
          */
          */
-        if (!counter.started) {
+        if (!counter->started) {
             *val = ctr_val;
             *val = ctr_val;
             return RISCV_EXCP_NONE;
             return RISCV_EXCP_NONE;
         } else {
         } else {
             /* Mark that the counter has been stopped */
             /* Mark that the counter has been stopped */
-            counter.started = false;
+            counter->started = false;
         }
         }
     }
     }