]> git.baikalelectronics.ru Git - kernel.git/commitdiff
KVM: x86: Read and pass all CR0/CR4 role bits to shadow MMU helper
authorSean Christopherson <seanjc@google.com>
Tue, 22 Jun 2021 17:57:02 +0000 (10:57 -0700)
committerPaolo Bonzini <pbonzini@redhat.com>
Thu, 24 Jun 2021 22:00:39 +0000 (18:00 -0400)
Grab all CR0/CR4 MMU role bits from current vCPU state when initializing
a non-nested shadow MMU.  Extract the masks from kvm_post_set_cr{0,4}(),
as the CR0/CR4 update masks must exactly match the mmu_role bits, with
one exception (see below).  The "full" CR0/CR4 will be used by future
commits to initialize the MMU and its role, as opposed to the current
approach of pulling everything from vCPU, which is incorrect for certain
flows, e.g. nested NPT.

CR4.LA57 is an exception, as it can be toggled on VM-Exit (for L1's MMU)
but can't be toggled via MOV CR4 while long mode is active.  I.e. LA57
needs to be in the mmu_role, but technically doesn't need to be checked
by kvm_post_set_cr4().  However, the extra check is completely benign as
the hardware restrictions simply mean LA57 will never be _the_ cause of
a MMU reset during MOV CR4.

Signed-off-by: Sean Christopherson <seanjc@google.com>
Message-Id: <20210622175739.3610207-18-seanjc@google.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
arch/x86/kvm/mmu.h
arch/x86/kvm/mmu/mmu.c
arch/x86/kvm/x86.c

index 47131b92b9901c8e5479a70888d9e2b7697f1996..4e926f4935b0703e3eca0f4e9a57ee4be076a141 100644 (file)
 #define PT32_ROOT_LEVEL 2
 #define PT32E_ROOT_LEVEL 3
 
+#define KVM_MMU_CR4_ROLE_BITS (X86_CR4_PGE | X86_CR4_PSE | X86_CR4_PAE | \
+                              X86_CR4_SMEP | X86_CR4_SMAP | X86_CR4_PKE | \
+                              X86_CR4_LA57)
+
+#define KVM_MMU_CR0_ROLE_BITS (X86_CR0_PG | X86_CR0_WP)
+
 static __always_inline u64 rsvd_bits(int s, int e)
 {
        BUILD_BUG_ON(__builtin_constant_p(e) && __builtin_constant_p(s) && e < s);
index 588d789cc79f25d123747727fbbb3a888ad42ea3..51a48f17c80edeec367418538c26e2c29f0c689a 100644 (file)
@@ -4778,8 +4778,8 @@ static void init_kvm_softmmu(struct kvm_vcpu *vcpu)
        struct kvm_mmu *context = &vcpu->arch.root_mmu;
 
        kvm_init_shadow_mmu(vcpu,
-                           kvm_read_cr0_bits(vcpu, X86_CR0_PG),
-                           kvm_read_cr4_bits(vcpu, X86_CR4_PAE),
+                           kvm_read_cr0_bits(vcpu, KVM_MMU_CR0_ROLE_BITS),
+                           kvm_read_cr4_bits(vcpu, KVM_MMU_CR4_ROLE_BITS),
                            vcpu->arch.efer);
 
        context->get_guest_pgd     = get_cr3;
index 0b059698cd5c0324fb1480f834e38d6504a8818d..a7c7b2b28de781340975f880cde252bf7a1b68be 100644 (file)
@@ -851,14 +851,12 @@ EXPORT_SYMBOL_GPL(load_pdptrs);
 
 void kvm_post_set_cr0(struct kvm_vcpu *vcpu, unsigned long old_cr0, unsigned long cr0)
 {
-       unsigned long update_bits = X86_CR0_PG | X86_CR0_WP;
-
        if ((cr0 ^ old_cr0) & X86_CR0_PG) {
                kvm_clear_async_pf_completion_queue(vcpu);
                kvm_async_pf_hash_reset(vcpu);
        }
 
-       if ((cr0 ^ old_cr0) & update_bits)
+       if ((cr0 ^ old_cr0) & KVM_MMU_CR0_ROLE_BITS)
                kvm_mmu_reset_context(vcpu);
 
        if (((cr0 ^ old_cr0) & X86_CR0_CD) &&
@@ -1037,10 +1035,7 @@ EXPORT_SYMBOL_GPL(kvm_is_valid_cr4);
 
 void kvm_post_set_cr4(struct kvm_vcpu *vcpu, unsigned long old_cr4, unsigned long cr4)
 {
-       unsigned long mmu_role_bits = X86_CR4_PGE | X86_CR4_PSE | X86_CR4_PAE |
-                                     X86_CR4_SMEP | X86_CR4_SMAP | X86_CR4_PKE;
-
-       if (((cr4 ^ old_cr4) & mmu_role_bits) ||
+       if (((cr4 ^ old_cr4) & KVM_MMU_CR4_ROLE_BITS) ||
            (!(cr4 & X86_CR4_PCIDE) && (old_cr4 & X86_CR4_PCIDE)))
                kvm_mmu_reset_context(vcpu);
 }