Commit 4732f244 authored by Like Xu's avatar Like Xu Committed by Paolo Bonzini

KVM: x86: Making the module parameter of vPMU more common

The new module parameter to control PMU virtualization should apply
to Intel as well as AMD, for situations where userspace is not trusted.
If the module parameter allows PMU virtualization, there could be a
new KVM_CAP or guest CPUID bits whereby userspace can enable/disable
PMU virtualization on a per-VM basis.

If the module parameter does not allow PMU virtualization, there
should be no userspace override, since we have no precedent for
authorizing that kind of override. If it's false, other counter-based
profiling features (such as LBR including the associated CPUID bits
if any) will not be exposed.

Change its name from "pmu" to "enable_pmu" as we have temporary
variables with the same name in our code like "struct kvm_pmu *pmu".

Fixes: b1d66dad ("KVM: x86/svm: Add module param to control PMU virtualization")
Suggested-by : Jim Mattson <jmattson@google.com>
Signed-off-by: default avatarLike Xu <likexu@tencent.com>
Message-Id: <20220111073823.21885-1-likexu@tencent.com>
Signed-off-by: default avatarPaolo Bonzini <pbonzini@redhat.com>
parent ecebb966
...@@ -845,10 +845,10 @@ static inline int __do_cpuid_func(struct kvm_cpuid_array *array, u32 function) ...@@ -845,10 +845,10 @@ static inline int __do_cpuid_func(struct kvm_cpuid_array *array, u32 function)
perf_get_x86_pmu_capability(&cap); perf_get_x86_pmu_capability(&cap);
/* /*
* Only support guest architectural pmu on a host * The guest architecture pmu is only supported if the architecture
* with architectural pmu. * pmu exists on the host and the module parameters allow it.
*/ */
if (!cap.version) if (!cap.version || !enable_pmu)
memset(&cap, 0, sizeof(cap)); memset(&cap, 0, sizeof(cap));
eax.split.version_id = min(cap.version, 2); eax.split.version_id = min(cap.version, 2);
......
...@@ -101,7 +101,7 @@ static inline struct kvm_pmc *get_gp_pmc_amd(struct kvm_pmu *pmu, u32 msr, ...@@ -101,7 +101,7 @@ static inline struct kvm_pmc *get_gp_pmc_amd(struct kvm_pmu *pmu, u32 msr,
{ {
struct kvm_vcpu *vcpu = pmu_to_vcpu(pmu); struct kvm_vcpu *vcpu = pmu_to_vcpu(pmu);
if (!pmu) if (!enable_pmu)
return NULL; return NULL;
switch (msr) { switch (msr) {
......
...@@ -192,10 +192,6 @@ module_param(vgif, int, 0444); ...@@ -192,10 +192,6 @@ module_param(vgif, int, 0444);
static int lbrv = true; static int lbrv = true;
module_param(lbrv, int, 0444); module_param(lbrv, int, 0444);
/* enable/disable PMU virtualization */
bool pmu = true;
module_param(pmu, bool, 0444);
static int tsc_scaling = true; static int tsc_scaling = true;
module_param(tsc_scaling, int, 0444); module_param(tsc_scaling, int, 0444);
...@@ -957,7 +953,7 @@ static __init void svm_set_cpu_caps(void) ...@@ -957,7 +953,7 @@ static __init void svm_set_cpu_caps(void)
kvm_cpu_cap_set(X86_FEATURE_VIRT_SSBD); kvm_cpu_cap_set(X86_FEATURE_VIRT_SSBD);
/* AMD PMU PERFCTR_CORE CPUID */ /* AMD PMU PERFCTR_CORE CPUID */
if (pmu && boot_cpu_has(X86_FEATURE_PERFCTR_CORE)) if (enable_pmu && boot_cpu_has(X86_FEATURE_PERFCTR_CORE))
kvm_cpu_cap_set(X86_FEATURE_PERFCTR_CORE); kvm_cpu_cap_set(X86_FEATURE_PERFCTR_CORE);
/* CPUID 0x8000001F (SME/SEV features) */ /* CPUID 0x8000001F (SME/SEV features) */
...@@ -1093,7 +1089,7 @@ static __init int svm_hardware_setup(void) ...@@ -1093,7 +1089,7 @@ static __init int svm_hardware_setup(void)
pr_info("LBR virtualization supported\n"); pr_info("LBR virtualization supported\n");
} }
if (!pmu) if (!enable_pmu)
pr_info("PMU virtualization is disabled\n"); pr_info("PMU virtualization is disabled\n");
svm_set_cpu_caps(); svm_set_cpu_caps();
......
...@@ -32,7 +32,6 @@ ...@@ -32,7 +32,6 @@
extern u32 msrpm_offsets[MSRPM_OFFSETS] __read_mostly; extern u32 msrpm_offsets[MSRPM_OFFSETS] __read_mostly;
extern bool npt_enabled; extern bool npt_enabled;
extern bool intercept_smi; extern bool intercept_smi;
extern bool pmu;
/* /*
* Clean bits in VMCB. * Clean bits in VMCB.
......
...@@ -5,6 +5,7 @@ ...@@ -5,6 +5,7 @@
#include <asm/vmx.h> #include <asm/vmx.h>
#include "lapic.h" #include "lapic.h"
#include "x86.h"
extern bool __read_mostly enable_vpid; extern bool __read_mostly enable_vpid;
extern bool __read_mostly flexpriority_enabled; extern bool __read_mostly flexpriority_enabled;
...@@ -389,6 +390,9 @@ static inline u64 vmx_get_perf_capabilities(void) ...@@ -389,6 +390,9 @@ static inline u64 vmx_get_perf_capabilities(void)
{ {
u64 perf_cap = 0; u64 perf_cap = 0;
if (!enable_pmu)
return perf_cap;
if (boot_cpu_has(X86_FEATURE_PDCM)) if (boot_cpu_has(X86_FEATURE_PDCM))
rdmsrl(MSR_IA32_PERF_CAPABILITIES, perf_cap); rdmsrl(MSR_IA32_PERF_CAPABILITIES, perf_cap);
......
...@@ -487,7 +487,7 @@ static void intel_pmu_refresh(struct kvm_vcpu *vcpu) ...@@ -487,7 +487,7 @@ static void intel_pmu_refresh(struct kvm_vcpu *vcpu)
pmu->reserved_bits = 0xffffffff00200000ull; pmu->reserved_bits = 0xffffffff00200000ull;
entry = kvm_find_cpuid_entry(vcpu, 0xa, 0); entry = kvm_find_cpuid_entry(vcpu, 0xa, 0);
if (!entry) if (!entry || !enable_pmu)
return; return;
eax.full = entry->eax; eax.full = entry->eax;
edx.full = entry->edx; edx.full = entry->edx;
......
...@@ -187,6 +187,11 @@ module_param(force_emulation_prefix, bool, S_IRUGO); ...@@ -187,6 +187,11 @@ module_param(force_emulation_prefix, bool, S_IRUGO);
int __read_mostly pi_inject_timer = -1; int __read_mostly pi_inject_timer = -1;
module_param(pi_inject_timer, bint, S_IRUGO | S_IWUSR); module_param(pi_inject_timer, bint, S_IRUGO | S_IWUSR);
/* Enable/disable PMU virtualization */
bool __read_mostly enable_pmu = true;
EXPORT_SYMBOL_GPL(enable_pmu);
module_param(enable_pmu, bool, 0444);
/* /*
* Restoring the host value for MSRs that are only consumed when running in * Restoring the host value for MSRs that are only consumed when running in
* usermode, e.g. SYSCALL MSRs and TSC_AUX, can be deferred until the CPU * usermode, e.g. SYSCALL MSRs and TSC_AUX, can be deferred until the CPU
......
...@@ -336,6 +336,7 @@ extern u64 host_xcr0; ...@@ -336,6 +336,7 @@ extern u64 host_xcr0;
extern u64 supported_xcr0; extern u64 supported_xcr0;
extern u64 host_xss; extern u64 host_xss;
extern u64 supported_xss; extern u64 supported_xss;
extern bool enable_pmu;
static inline bool kvm_mpx_supported(void) static inline bool kvm_mpx_supported(void)
{ {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment