Re: [PATCH v2 1/3] KVM: x86/pmu: Make part of the Intel v2 PMU MSRs handling x86 generic

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



On Mon, Sep 19, 2022 at 2:35 AM Like Xu <like.xu.linux@xxxxxxxxx> wrote:
>
> From: Like Xu <likexu@xxxxxxxxxxx>
>
> The AMD PerfMonV2 defines three registers similar to part of the
> Intel v2 PMU registers, including the GLOBAL_CTRL, GLOBAL_STATUS
> and GLOBAL_OVF_CTRL MSRs. For better code reuse, this specific
> part of the handling can be extracted to make it generic for X86.
>
> The new non-prefix pmc_is_enabled() works well as legacy AMD vPMU
> version is indexeqd as 1. Note that the specific *_is_valid_msr will
Nit: indexed
> continue to be used to avoid cross-vendor msr access.
>
> Signed-off-by: Like Xu <likexu@xxxxxxxxxxx>
> ---
>  arch/x86/include/asm/kvm-x86-pmu-ops.h |  1 -
>  arch/x86/kvm/pmu.c                     | 55 +++++++++++++++++++++---
>  arch/x86/kvm/pmu.h                     | 30 ++++++++++++-
>  arch/x86/kvm/svm/pmu.c                 |  9 ----
>  arch/x86/kvm/vmx/pmu_intel.c           | 58 +-------------------------
>  5 files changed, 80 insertions(+), 73 deletions(-)
>
> diff --git a/arch/x86/include/asm/kvm-x86-pmu-ops.h b/arch/x86/include/asm/kvm-x86-pmu-ops.h
> index c17e3e96fc1d..6c98f4bb4228 100644
> --- a/arch/x86/include/asm/kvm-x86-pmu-ops.h
> +++ b/arch/x86/include/asm/kvm-x86-pmu-ops.h
> @@ -13,7 +13,6 @@ BUILD_BUG_ON(1)
>   * at the call sites.
>   */
>  KVM_X86_PMU_OP(hw_event_available)
> -KVM_X86_PMU_OP(pmc_is_enabled)
>  KVM_X86_PMU_OP(pmc_idx_to_pmc)
>  KVM_X86_PMU_OP(rdpmc_ecx_to_pmc)
>  KVM_X86_PMU_OP(msr_idx_to_pmc)
> diff --git a/arch/x86/kvm/pmu.c b/arch/x86/kvm/pmu.c
> index fabbc43031b3..2643a3994624 100644
> --- a/arch/x86/kvm/pmu.c
> +++ b/arch/x86/kvm/pmu.c
> @@ -83,11 +83,6 @@ void kvm_pmu_ops_update(const struct kvm_pmu_ops *pmu_ops)
>  #undef __KVM_X86_PMU_OP
>  }
>
> -static inline bool pmc_is_enabled(struct kvm_pmc *pmc)
> -{
> -       return static_call(kvm_x86_pmu_pmc_is_enabled)(pmc);
> -}
> -
>  static void kvm_pmi_trigger_fn(struct irq_work *irq_work)
>  {
>         struct kvm_pmu *pmu = container_of(irq_work, struct kvm_pmu, irq_work);
> @@ -455,11 +450,61 @@ static void kvm_pmu_mark_pmc_in_use(struct kvm_vcpu *vcpu, u32 msr)
>
>  int kvm_pmu_get_msr(struct kvm_vcpu *vcpu, struct msr_data *msr_info)
>  {
> +       struct kvm_pmu *pmu = vcpu_to_pmu(vcpu);
> +       u32 msr = msr_info->index;
> +
> +       switch (msr) {
> +       case MSR_CORE_PERF_GLOBAL_STATUS:
> +               msr_info->data = pmu->global_status;
> +               return 0;
> +       case MSR_CORE_PERF_GLOBAL_CTRL:
> +               msr_info->data = pmu->global_ctrl;
> +               return 0;
> +       case MSR_CORE_PERF_GLOBAL_OVF_CTRL:
> +               msr_info->data = 0;
> +               return 0;
> +       default:
> +               break;
> +       }
> +
>         return static_call(kvm_x86_pmu_get_msr)(vcpu, msr_info);
>  }
>
>  int kvm_pmu_set_msr(struct kvm_vcpu *vcpu, struct msr_data *msr_info)
>  {
> +       struct kvm_pmu *pmu = vcpu_to_pmu(vcpu);
> +       u32 msr = msr_info->index;
> +       u64 data = msr_info->data;
> +       u64 diff;
> +
> +       switch (msr) {
> +       case MSR_CORE_PERF_GLOBAL_STATUS:
> +               if (msr_info->host_initiated) {
> +                       pmu->global_status = data;
> +                       return 0;
> +               }
> +               break; /* RO MSR */
Perhaps 'return 1'?
> +       case MSR_CORE_PERF_GLOBAL_CTRL:
> +               if (pmu->global_ctrl == data)
> +                       return 0;
> +               if (kvm_valid_perf_global_ctrl(pmu, data)) {
> +                       diff = pmu->global_ctrl ^ data;
> +                       pmu->global_ctrl = data;
> +                       reprogram_counters(pmu, diff);
> +                       return 0;
> +               }
> +               break;
Perhaps 'return 1'?
> +       case MSR_CORE_PERF_GLOBAL_OVF_CTRL:
> +               if (!(data & pmu->global_ovf_ctrl_mask)) {
> +                       if (!msr_info->host_initiated)
> +                               pmu->global_status &= ~data;
> +                       return 0;
> +               }
> +               break;
Perhaps 'return 1'?
> +       default:
> +               break;
> +       }
> +
>         kvm_pmu_mark_pmc_in_use(vcpu, msr_info->index);
>         return static_call(kvm_x86_pmu_set_msr)(vcpu, msr_info);
>  }
> diff --git a/arch/x86/kvm/pmu.h b/arch/x86/kvm/pmu.h
> index 847e7112a5d3..0275f1bff5ea 100644
> --- a/arch/x86/kvm/pmu.h
> +++ b/arch/x86/kvm/pmu.h
> @@ -26,7 +26,6 @@ struct kvm_event_hw_type_mapping {
>
>  struct kvm_pmu_ops {
>         bool (*hw_event_available)(struct kvm_pmc *pmc);
> -       bool (*pmc_is_enabled)(struct kvm_pmc *pmc);
>         struct kvm_pmc *(*pmc_idx_to_pmc)(struct kvm_pmu *pmu, int pmc_idx);
>         struct kvm_pmc *(*rdpmc_ecx_to_pmc)(struct kvm_vcpu *vcpu,
>                 unsigned int idx, u64 *mask);
> @@ -189,6 +188,35 @@ static inline void kvm_pmu_request_counter_reprogam(struct kvm_pmc *pmc)
>         kvm_make_request(KVM_REQ_PMU, pmc->vcpu);
>  }
>
> +/*
> + * Check if a PMC is enabled by comparing it against global_ctrl bits.
> + *
> + * If the current version of vPMU doesn't have global_ctrl MSR,
> + * all vPMCs are enabled (return TRUE).

The name of this function is a bit misleading. A PMC can be disabled
either by PERF_CLOBAL_CTRL or by its corresponding EVTSEL.

> + */
> +static inline bool pmc_is_enabled(struct kvm_pmc *pmc)
> +{
> +       struct kvm_pmu *pmu = pmc_to_pmu(pmc);
> +
> +       if (pmu->version < 2)
> +               return true;
> +
> +       return test_bit(pmc->idx, (unsigned long *)&pmu->global_ctrl);
> +}
> +
> +static inline void reprogram_counters(struct kvm_pmu *pmu, u64 diff)
> +{
> +       int bit;
> +
> +       if (!diff)
> +               return;
> +
> +       for_each_set_bit(bit, (unsigned long *)&diff, X86_PMC_IDX_MAX)
> +               __set_bit(bit, pmu->reprogram_pmi);

I see that you've dropped the index to pmc conversion and the test for
a valid pmc. Maybe this is okay, but I'm not sure what the caller
looks like, since this is all in global_ctrl_changed() upstream.
What's your diff base?

> +
> +       kvm_make_request(KVM_REQ_PMU, pmu_to_vcpu(pmu));

Why this new request? It's not in the Intel-specific version of these
function that you elide below.

Perhaps you could split up the semantic changes from the simple renamings?

> +}
> +
>  void kvm_pmu_deliver_pmi(struct kvm_vcpu *vcpu);
>  void kvm_pmu_handle_event(struct kvm_vcpu *vcpu);
>  int kvm_pmu_rdpmc(struct kvm_vcpu *vcpu, unsigned pmc, u64 *data);
> diff --git a/arch/x86/kvm/svm/pmu.c b/arch/x86/kvm/svm/pmu.c
> index f99f2c869664..3a20972e9f1a 100644
> --- a/arch/x86/kvm/svm/pmu.c
> +++ b/arch/x86/kvm/svm/pmu.c
> @@ -76,14 +76,6 @@ static bool amd_hw_event_available(struct kvm_pmc *pmc)
>         return true;
>  }
>
> -/* check if a PMC is enabled by comparing it against global_ctrl bits. Because
> - * AMD CPU doesn't have global_ctrl MSR, all PMCs are enabled (return TRUE).
> - */
> -static bool amd_pmc_is_enabled(struct kvm_pmc *pmc)
> -{
> -       return true;
> -}
> -
>  static bool amd_is_valid_rdpmc_ecx(struct kvm_vcpu *vcpu, unsigned int idx)
>  {
>         struct kvm_pmu *pmu = vcpu_to_pmu(vcpu);
> @@ -218,7 +210,6 @@ static void amd_pmu_reset(struct kvm_vcpu *vcpu)
>
>  struct kvm_pmu_ops amd_pmu_ops __initdata = {
>         .hw_event_available = amd_hw_event_available,
> -       .pmc_is_enabled = amd_pmc_is_enabled,
>         .pmc_idx_to_pmc = amd_pmc_idx_to_pmc,
>         .rdpmc_ecx_to_pmc = amd_rdpmc_ecx_to_pmc,
>         .msr_idx_to_pmc = amd_msr_idx_to_pmc,
> diff --git a/arch/x86/kvm/vmx/pmu_intel.c b/arch/x86/kvm/vmx/pmu_intel.c
> index 612c89ef5398..12eb2edfe9bc 100644
> --- a/arch/x86/kvm/vmx/pmu_intel.c
> +++ b/arch/x86/kvm/vmx/pmu_intel.c
> @@ -68,18 +68,6 @@ static struct kvm_pmc *intel_pmc_idx_to_pmc(struct kvm_pmu *pmu, int pmc_idx)
>         }
>  }
>
> -static void reprogram_counters(struct kvm_pmu *pmu, u64 diff)
> -{
> -       int bit;
> -       struct kvm_pmc *pmc;
> -
> -       for_each_set_bit(bit, (unsigned long *)&diff, X86_PMC_IDX_MAX) {
> -               pmc = intel_pmc_idx_to_pmc(pmu, bit);
> -               if (pmc)
> -                       kvm_pmu_request_counter_reprogam(pmc);
> -       }
> -}
> -
>  static bool intel_hw_event_available(struct kvm_pmc *pmc)
>  {
>         struct kvm_pmu *pmu = pmc_to_pmu(pmc);
> @@ -102,17 +90,6 @@ static bool intel_hw_event_available(struct kvm_pmc *pmc)
>         return true;
>  }
>
> -/* check if a PMC is enabled by comparing it with globl_ctrl bits. */
> -static bool intel_pmc_is_enabled(struct kvm_pmc *pmc)
> -{
> -       struct kvm_pmu *pmu = pmc_to_pmu(pmc);
> -
> -       if (!intel_pmu_has_perf_global_ctrl(pmu))
> -               return true;
> -
> -       return test_bit(pmc->idx, (unsigned long *)&pmu->global_ctrl);
> -}
> -
>  static bool intel_is_valid_rdpmc_ecx(struct kvm_vcpu *vcpu, unsigned int idx)
>  {
>         struct kvm_pmu *pmu = vcpu_to_pmu(vcpu);
> @@ -347,15 +324,6 @@ static int intel_pmu_get_msr(struct kvm_vcpu *vcpu, struct msr_data *msr_info)
>         case MSR_CORE_PERF_FIXED_CTR_CTRL:
>                 msr_info->data = pmu->fixed_ctr_ctrl;
>                 return 0;
> -       case MSR_CORE_PERF_GLOBAL_STATUS:
> -               msr_info->data = pmu->global_status;
> -               return 0;
> -       case MSR_CORE_PERF_GLOBAL_CTRL:
> -               msr_info->data = pmu->global_ctrl;
> -               return 0;
> -       case MSR_CORE_PERF_GLOBAL_OVF_CTRL:
> -               msr_info->data = 0;
> -               return 0;
>         case MSR_IA32_PEBS_ENABLE:
>                 msr_info->data = pmu->pebs_enable;
>                 return 0;
> @@ -404,29 +372,6 @@ static int intel_pmu_set_msr(struct kvm_vcpu *vcpu, struct msr_data *msr_info)
>                         return 0;
>                 }
>                 break;
> -       case MSR_CORE_PERF_GLOBAL_STATUS:
> -               if (msr_info->host_initiated) {
> -                       pmu->global_status = data;
> -                       return 0;
> -               }
> -               break; /* RO MSR */
> -       case MSR_CORE_PERF_GLOBAL_CTRL:
> -               if (pmu->global_ctrl == data)
> -                       return 0;
> -               if (kvm_valid_perf_global_ctrl(pmu, data)) {
> -                       diff = pmu->global_ctrl ^ data;
> -                       pmu->global_ctrl = data;
> -                       reprogram_counters(pmu, diff);
> -                       return 0;
> -               }
> -               break;
> -       case MSR_CORE_PERF_GLOBAL_OVF_CTRL:
> -               if (!(data & pmu->global_ovf_ctrl_mask)) {
> -                       if (!msr_info->host_initiated)
> -                               pmu->global_status &= ~data;
> -                       return 0;
> -               }
> -               break;
>         case MSR_IA32_PEBS_ENABLE:
>                 if (pmu->pebs_enable == data)
>                         return 0;
> @@ -783,7 +728,7 @@ void intel_pmu_cross_mapped_check(struct kvm_pmu *pmu)
>                 pmc = intel_pmc_idx_to_pmc(pmu, bit);
>
>                 if (!pmc || !pmc_speculative_in_use(pmc) ||
> -                   !intel_pmc_is_enabled(pmc) || !pmc->perf_event)
> +                   !pmc_is_enabled(pmc) || !pmc->perf_event)
>                         continue;
>
>                 hw_idx = pmc->perf_event->hw.idx;
> @@ -795,7 +740,6 @@ void intel_pmu_cross_mapped_check(struct kvm_pmu *pmu)
>
>  struct kvm_pmu_ops intel_pmu_ops __initdata = {
>         .hw_event_available = intel_hw_event_available,
> -       .pmc_is_enabled = intel_pmc_is_enabled,
>         .pmc_idx_to_pmc = intel_pmc_idx_to_pmc,
>         .rdpmc_ecx_to_pmc = intel_rdpmc_ecx_to_pmc,
>         .msr_idx_to_pmc = intel_msr_idx_to_pmc,
> --
> 2.37.3
>



[Index of Archives]     [KVM ARM]     [KVM ia64]     [KVM ppc]     [Virtualization Tools]     [Spice Development]     [Libvirt]     [Libvirt Users]     [Linux USB Devel]     [Linux Audio Users]     [Yosemite Questions]     [Linux Kernel]     [Linux SCSI]     [XFree86]

  Powered by Linux