mirror of
https://github.com/torvalds/linux.git
synced 2026-05-27 16:44:58 +02:00
Use u32 instead of uint32_t to make the KVM selftests code more concise and more similar to the kernel (since selftests are primarily developed by kernel developers). This commit was generated with the following command: git ls-files tools/testing/selftests/kvm | xargs sed -i 's/uint32_t/u32/g' Then by manually adjusting whitespace to make checkpatch.pl happy. No functional change intended. Signed-off-by: David Matlack <dmatlack@google.com> Link: https://patch.msgid.link/20260420212004.3938325-7-seanjc@google.com Signed-off-by: Sean Christopherson <seanjc@google.com>
130 lines
4.0 KiB
C
130 lines
4.0 KiB
C
// SPDX-License-Identifier: GPL-2.0-only
|
|
/*
|
|
* VMX control MSR test
|
|
*
|
|
* Copyright (C) 2022 Google LLC.
|
|
*
|
|
* Tests for KVM ownership of bits in the VMX entry/exit control MSRs. Checks
|
|
* that KVM will set owned bits where appropriate, and will not if
|
|
* KVM_X86_QUIRK_TWEAK_VMX_CTRL_MSRS is disabled.
|
|
*/
|
|
#include <linux/bitmap.h>
|
|
#include "kvm_util.h"
|
|
#include "vmx.h"
|
|
|
|
static void vmx_fixed1_msr_test(struct kvm_vcpu *vcpu, u32 msr_index, u64 mask)
|
|
{
|
|
u64 val = vcpu_get_msr(vcpu, msr_index);
|
|
u64 bit;
|
|
|
|
mask &= val;
|
|
|
|
for_each_set_bit(bit, &mask, 64) {
|
|
vcpu_set_msr(vcpu, msr_index, val & ~BIT_ULL(bit));
|
|
vcpu_set_msr(vcpu, msr_index, val);
|
|
}
|
|
}
|
|
|
|
static void vmx_fixed0_msr_test(struct kvm_vcpu *vcpu, u32 msr_index, u64 mask)
|
|
{
|
|
u64 val = vcpu_get_msr(vcpu, msr_index);
|
|
u64 bit;
|
|
|
|
mask = ~mask | val;
|
|
|
|
for_each_clear_bit(bit, &mask, 64) {
|
|
vcpu_set_msr(vcpu, msr_index, val | BIT_ULL(bit));
|
|
vcpu_set_msr(vcpu, msr_index, val);
|
|
}
|
|
}
|
|
|
|
static void vmx_fixed0and1_msr_test(struct kvm_vcpu *vcpu, u32 msr_index)
|
|
{
|
|
vmx_fixed0_msr_test(vcpu, msr_index, GENMASK_ULL(31, 0));
|
|
vmx_fixed1_msr_test(vcpu, msr_index, GENMASK_ULL(63, 32));
|
|
}
|
|
|
|
static void vmx_save_restore_msrs_test(struct kvm_vcpu *vcpu)
|
|
{
|
|
vcpu_set_msr(vcpu, MSR_IA32_VMX_VMCS_ENUM, 0);
|
|
vcpu_set_msr(vcpu, MSR_IA32_VMX_VMCS_ENUM, -1ull);
|
|
|
|
vmx_fixed1_msr_test(vcpu, MSR_IA32_VMX_BASIC,
|
|
BIT_ULL(49) | BIT_ULL(54) | BIT_ULL(55));
|
|
|
|
vmx_fixed1_msr_test(vcpu, MSR_IA32_VMX_MISC,
|
|
BIT_ULL(5) | GENMASK_ULL(8, 6) | BIT_ULL(14) |
|
|
BIT_ULL(15) | BIT_ULL(28) | BIT_ULL(29) | BIT_ULL(30));
|
|
|
|
vmx_fixed0and1_msr_test(vcpu, MSR_IA32_VMX_PROCBASED_CTLS2);
|
|
vmx_fixed1_msr_test(vcpu, MSR_IA32_VMX_EPT_VPID_CAP, -1ull);
|
|
vmx_fixed0and1_msr_test(vcpu, MSR_IA32_VMX_TRUE_PINBASED_CTLS);
|
|
vmx_fixed0and1_msr_test(vcpu, MSR_IA32_VMX_TRUE_PROCBASED_CTLS);
|
|
vmx_fixed0and1_msr_test(vcpu, MSR_IA32_VMX_TRUE_EXIT_CTLS);
|
|
vmx_fixed0and1_msr_test(vcpu, MSR_IA32_VMX_TRUE_ENTRY_CTLS);
|
|
vmx_fixed1_msr_test(vcpu, MSR_IA32_VMX_VMFUNC, -1ull);
|
|
}
|
|
|
|
static void __ia32_feature_control_msr_test(struct kvm_vcpu *vcpu,
|
|
u64 msr_bit,
|
|
struct kvm_x86_cpu_feature feature)
|
|
{
|
|
u64 val;
|
|
|
|
vcpu_clear_cpuid_feature(vcpu, feature);
|
|
|
|
val = vcpu_get_msr(vcpu, MSR_IA32_FEAT_CTL);
|
|
vcpu_set_msr(vcpu, MSR_IA32_FEAT_CTL, val | msr_bit | FEAT_CTL_LOCKED);
|
|
vcpu_set_msr(vcpu, MSR_IA32_FEAT_CTL, (val & ~msr_bit) | FEAT_CTL_LOCKED);
|
|
vcpu_set_msr(vcpu, MSR_IA32_FEAT_CTL, val | msr_bit | FEAT_CTL_LOCKED);
|
|
vcpu_set_msr(vcpu, MSR_IA32_FEAT_CTL, (val & ~msr_bit) | FEAT_CTL_LOCKED);
|
|
vcpu_set_msr(vcpu, MSR_IA32_FEAT_CTL, val);
|
|
|
|
if (!kvm_cpu_has(feature))
|
|
return;
|
|
|
|
vcpu_set_cpuid_feature(vcpu, feature);
|
|
}
|
|
|
|
static void ia32_feature_control_msr_test(struct kvm_vcpu *vcpu)
|
|
{
|
|
u64 supported_bits = FEAT_CTL_LOCKED |
|
|
FEAT_CTL_VMX_ENABLED_INSIDE_SMX |
|
|
FEAT_CTL_VMX_ENABLED_OUTSIDE_SMX |
|
|
FEAT_CTL_SGX_LC_ENABLED |
|
|
FEAT_CTL_SGX_ENABLED |
|
|
FEAT_CTL_LMCE_ENABLED;
|
|
int bit, r;
|
|
|
|
__ia32_feature_control_msr_test(vcpu, FEAT_CTL_VMX_ENABLED_INSIDE_SMX, X86_FEATURE_SMX);
|
|
__ia32_feature_control_msr_test(vcpu, FEAT_CTL_VMX_ENABLED_INSIDE_SMX, X86_FEATURE_VMX);
|
|
__ia32_feature_control_msr_test(vcpu, FEAT_CTL_VMX_ENABLED_OUTSIDE_SMX, X86_FEATURE_VMX);
|
|
__ia32_feature_control_msr_test(vcpu, FEAT_CTL_SGX_LC_ENABLED, X86_FEATURE_SGX_LC);
|
|
__ia32_feature_control_msr_test(vcpu, FEAT_CTL_SGX_LC_ENABLED, X86_FEATURE_SGX);
|
|
__ia32_feature_control_msr_test(vcpu, FEAT_CTL_SGX_ENABLED, X86_FEATURE_SGX);
|
|
__ia32_feature_control_msr_test(vcpu, FEAT_CTL_LMCE_ENABLED, X86_FEATURE_MCE);
|
|
|
|
for_each_clear_bit(bit, &supported_bits, 64) {
|
|
r = _vcpu_set_msr(vcpu, MSR_IA32_FEAT_CTL, BIT(bit));
|
|
TEST_ASSERT(r == 0,
|
|
"Setting reserved bit %d in IA32_FEATURE_CONTROL should fail", bit);
|
|
}
|
|
}
|
|
|
|
int main(void)
|
|
{
|
|
struct kvm_vcpu *vcpu;
|
|
struct kvm_vm *vm;
|
|
|
|
TEST_REQUIRE(kvm_has_cap(KVM_CAP_DISABLE_QUIRKS2));
|
|
TEST_REQUIRE(kvm_cpu_has(X86_FEATURE_VMX));
|
|
|
|
/* No need to actually do KVM_RUN, thus no guest code. */
|
|
vm = vm_create_with_one_vcpu(&vcpu, NULL);
|
|
|
|
vmx_save_restore_msrs_test(vcpu);
|
|
ia32_feature_control_msr_test(vcpu);
|
|
|
|
kvm_vm_free(vm);
|
|
}
|