diff options
-rw-r--r-- | src/arch/arm/pmu.cc | 30 | ||||
-rw-r--r-- | src/arch/arm/pmu.hh | 17 |
2 files changed, 45 insertions, 2 deletions
diff --git a/src/arch/arm/pmu.cc b/src/arch/arm/pmu.cc index b569b6967..90d9f1a08 100644 --- a/src/arch/arm/pmu.cc +++ b/src/arch/arm/pmu.cc @@ -214,7 +214,7 @@ PMU::setMiscReg(int misc_reg, MiscReg val) case MISCREG_PMOVSCLR_EL0: case MISCREG_PMOVSR: - reg_pmovsr &= ~val; + setOverflowStatus(reg_pmovsr & ~val); return; case MISCREG_PMSWINC_EL0: @@ -286,7 +286,7 @@ PMU::setMiscReg(int misc_reg, MiscReg val) case MISCREG_PMOVSSET_EL0: case MISCREG_PMOVSSET: - reg_pmovsr |= val; + setOverflowStatus(reg_pmovsr | val); return; default: @@ -645,6 +645,20 @@ PMU::setCounterTypeRegister(CounterId id, PMEVTYPER_t val) } void +PMU::setOverflowStatus(MiscReg new_val) +{ + const bool int_old = reg_pmovsr != 0; + const bool int_new = new_val != 0; + + reg_pmovsr = new_val; + if (int_old && !int_new) { + clearInterrupt(); + } else if (!int_old && int_new && (reg_pminten & reg_pmovsr)) { + raiseInterrupt(); + } +} + +void PMU::raiseInterrupt() { if (interrupt) { @@ -657,6 +671,18 @@ PMU::raiseInterrupt() } void +PMU::clearInterrupt() +{ + if (interrupt) { + DPRINTF(PMUVerbose, "Clearing PMU interrupt.\n"); + interrupt->clear(); + } else { + warn_once("Dropping PMU interrupt as no interrupt has " + "been specified\n"); + } +} + +void PMU::serialize(CheckpointOut &cp) const { DPRINTF(Checkpoint, "Serializing Arm PMU\n"); diff --git a/src/arch/arm/pmu.hh b/src/arch/arm/pmu.hh index 61516cffd..ee68272e7 100644 --- a/src/arch/arm/pmu.hh +++ b/src/arch/arm/pmu.hh @@ -219,6 +219,11 @@ class PMU : public SimObject, public ArmISA::BaseISADevice { void raiseInterrupt(); /** + * Clear a PMU interrupt. + */ + void clearInterrupt(); + + /** * Get the value of a performance counter. * * This method returns the value of a general purpose performance @@ -269,6 +274,18 @@ class PMU : public SimObject, public ArmISA::BaseISADevice { */ void setCounterTypeRegister(CounterId id, PMEVTYPER_t type); + /** + * Used for writing the Overflow Flag Status Register (SET/CLR) + * + * This method implements a write to the PMOVSSET/PMOVSCLR registers. + * It is capturing change of state in the register bits so that + * the overflow interrupt can be raised/cleared as a side effect + * of the write. + * + * @param new_val New value of the Overflow Status Register + */ + void setOverflowStatus(MiscReg new_val); + protected: /* Probe handling and counter state */ struct CounterState; |