KVM: x86/pmu: Add macros to iterate over all PMCs given a bitmap
authorSean Christopherson <seanjc@google.com>
Fri, 10 Nov 2023 02:28:52 +0000 (18:28 -0800)
committerSean Christopherson <seanjc@google.com>
Thu, 1 Feb 2024 17:35:48 +0000 (09:35 -0800)
Add and use kvm_for_each_pmc() to dedup a variety of open coded for-loops
that iterate over valid PMCs given a bitmap (and because seeing checkpatch
whine about bad macro style is always amusing).

No functional change intended.

Link: https://lore.kernel.org/r/20231110022857.1273836-6-seanjc@google.com
Signed-off-by: Sean Christopherson <seanjc@google.com>
arch/x86/kvm/pmu.c
arch/x86/kvm/pmu.h
arch/x86/kvm/vmx/pmu_intel.c

index 81a0c67198639dbf80df6e77a3b79325c4001ca8..7b5563ff259cec7e56c10f965f09c61003431f69 100644 (file)
@@ -493,6 +493,7 @@ void kvm_pmu_handle_event(struct kvm_vcpu *vcpu)
 {
        DECLARE_BITMAP(bitmap, X86_PMC_IDX_MAX);
        struct kvm_pmu *pmu = vcpu_to_pmu(vcpu);
+       struct kvm_pmc *pmc;
        int bit;
 
        bitmap_copy(bitmap, pmu->reprogram_pmi, X86_PMC_IDX_MAX);
@@ -505,12 +506,7 @@ void kvm_pmu_handle_event(struct kvm_vcpu *vcpu)
        BUILD_BUG_ON(sizeof(bitmap) != sizeof(atomic64_t));
        atomic64_andnot(*(s64 *)bitmap, &pmu->__reprogram_pmi);
 
-       for_each_set_bit(bit, bitmap, X86_PMC_IDX_MAX) {
-               struct kvm_pmc *pmc = kvm_pmc_idx_to_pmc(pmu, bit);
-
-               if (unlikely(!pmc))
-                       continue;
-
+       kvm_for_each_pmc(pmu, pmc, bit, bitmap) {
                /*
                 * If reprogramming fails, e.g. due to contention, re-set the
                 * regprogram bit set, i.e. opportunistically try again on the
@@ -730,11 +726,7 @@ static void kvm_pmu_reset(struct kvm_vcpu *vcpu)
 
        bitmap_zero(pmu->reprogram_pmi, X86_PMC_IDX_MAX);
 
-       for_each_set_bit(i, pmu->all_valid_pmc_idx, X86_PMC_IDX_MAX) {
-               pmc = kvm_pmc_idx_to_pmc(pmu, i);
-               if (!pmc)
-                       continue;
-
+       kvm_for_each_pmc(pmu, pmc, i, pmu->all_valid_pmc_idx) {
                pmc_stop_counter(pmc);
                pmc->counter = 0;
                pmc->emulated_counter = 0;
@@ -806,10 +798,8 @@ void kvm_pmu_cleanup(struct kvm_vcpu *vcpu)
        bitmap_andnot(bitmask, pmu->all_valid_pmc_idx,
                      pmu->pmc_in_use, X86_PMC_IDX_MAX);
 
-       for_each_set_bit(i, bitmask, X86_PMC_IDX_MAX) {
-               pmc = kvm_pmc_idx_to_pmc(pmu, i);
-
-               if (pmc && pmc->perf_event && !pmc_speculative_in_use(pmc))
+       kvm_for_each_pmc(pmu, pmc, i, bitmask) {
+               if (pmc->perf_event && !pmc_speculative_in_use(pmc))
                        pmc_stop_counter(pmc);
        }
 
@@ -861,10 +851,8 @@ void kvm_pmu_trigger_event(struct kvm_vcpu *vcpu, u64 perf_hw_id)
        struct kvm_pmc *pmc;
        int i;
 
-       for_each_set_bit(i, pmu->all_valid_pmc_idx, X86_PMC_IDX_MAX) {
-               pmc = kvm_pmc_idx_to_pmc(pmu, i);
-
-               if (!pmc || !pmc_event_is_allowed(pmc))
+       kvm_for_each_pmc(pmu, pmc, i, pmu->all_valid_pmc_idx) {
+               if (!pmc_event_is_allowed(pmc))
                        continue;
 
                /* Ignore checks for edge detect, pin control, invert and CMASK bits */
index 56e8e665e1aff78f32fc56583b5fd6e5224e4011..fd18bc0b281cfe86ae98b51c65534f5d94b0214a 100644 (file)
@@ -83,6 +83,12 @@ static inline struct kvm_pmc *kvm_pmc_idx_to_pmc(struct kvm_pmu *pmu, int idx)
        return NULL;
 }
 
+#define kvm_for_each_pmc(pmu, pmc, i, bitmap)                  \
+       for_each_set_bit(i, bitmap, X86_PMC_IDX_MAX)            \
+               if (!(pmc = kvm_pmc_idx_to_pmc(pmu, i)))        \
+                       continue;                               \
+               else                                            \
+
 static inline u64 pmc_bitmask(struct kvm_pmc *pmc)
 {
        struct kvm_pmu *pmu = pmc_to_pmu(pmc);
index 845a964f22a622ed582b7e05f690d39e6594c618..73b9943ceb17d01ca47de90dc921b995e36b3dba 100644 (file)
@@ -704,11 +704,8 @@ void intel_pmu_cross_mapped_check(struct kvm_pmu *pmu)
        struct kvm_pmc *pmc = NULL;
        int bit, hw_idx;
 
-       for_each_set_bit(bit, (unsigned long *)&pmu->global_ctrl,
-                        X86_PMC_IDX_MAX) {
-               pmc = kvm_pmc_idx_to_pmc(pmu, bit);
-
-               if (!pmc || !pmc_speculative_in_use(pmc) ||
+       kvm_for_each_pmc(pmu, pmc, bit, (unsigned long *)&pmu->global_ctrl) {
+               if (!pmc_speculative_in_use(pmc) ||
                    !pmc_is_globally_enabled(pmc) || !pmc->perf_event)
                        continue;