KVM: nVMX: Use fast PGD switch when emulating VMFUNC[EPTP_SWITCH]
Use __kvm_mmu_new_pgd() via kvm_init_shadow_ept_mmu() to emulate VMFUNC[EPTP_SWITCH] instead of nuking all MMUs. EPTP_SWITCH is the EPT equivalent of MOV to CR3, i.e. is a perfect fit for the common PGD flow, the only hiccup being that A/D enabling is buried in the EPTP. But, that is easily handled by bouncing through kvm_init_shadow_ept_mmu(). Explicitly request a guest TLB flush if VPID is disabled. Per Intel's SDM, if VPID is disabled, "an EPTP-switching VMFUNC invalidates combined mappings associated with VPID 0000H (for all PCIDs and for all EP4TA values, where EP4TA is the value of bits 51:12 of EPTP)". Note, this technically is a very bizarre bug fix of sorts if L2 is using PAE paging, as avoiding the full MMU reload also avoids incorrectly reloading the PDPTEs, which the SDM explicitly states are not touched: If PAE paging is in use, an EPTP-switching VMFUNC does not load the four page-directory-pointer-table entries (PDPTEs) from the guest-physical address in CR3. The logical processor continues to use the four guest-physical addresses already present in the PDPTEs. The guest-physical address in CR3 is not translated through the new EPT paging structures (until some operation that would load the PDPTEs). In addition to optimizing L2's MMU shenanigans, avoiding the full reload also optimizes L1's MMU as KVM_REQ_MMU_RELOAD wipes out all roots in both root_mmu and guest_mmu. Signed-off-by: Sean Christopherson <seanjc@google.com> Message-Id: <20210609234235.1244004-12-seanjc@google.com> Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
This commit is contained in:
parent
28f28d453f
commit
39353ab579
@ -351,16 +351,21 @@ static void nested_ept_inject_page_fault(struct kvm_vcpu *vcpu,
|
||||
vmcs12->guest_physical_address = fault->address;
|
||||
}
|
||||
|
||||
static void nested_ept_new_eptp(struct kvm_vcpu *vcpu)
|
||||
{
|
||||
kvm_init_shadow_ept_mmu(vcpu,
|
||||
to_vmx(vcpu)->nested.msrs.ept_caps &
|
||||
VMX_EPT_EXECUTE_ONLY_BIT,
|
||||
nested_ept_ad_enabled(vcpu),
|
||||
nested_ept_get_eptp(vcpu));
|
||||
}
|
||||
|
||||
static void nested_ept_init_mmu_context(struct kvm_vcpu *vcpu)
|
||||
{
|
||||
WARN_ON(mmu_is_nested(vcpu));
|
||||
|
||||
vcpu->arch.mmu = &vcpu->arch.guest_mmu;
|
||||
kvm_init_shadow_ept_mmu(vcpu,
|
||||
to_vmx(vcpu)->nested.msrs.ept_caps &
|
||||
VMX_EPT_EXECUTE_ONLY_BIT,
|
||||
nested_ept_ad_enabled(vcpu),
|
||||
nested_ept_get_eptp(vcpu));
|
||||
nested_ept_new_eptp(vcpu);
|
||||
vcpu->arch.mmu->get_guest_pgd = nested_ept_get_eptp;
|
||||
vcpu->arch.mmu->inject_page_fault = nested_ept_inject_page_fault;
|
||||
vcpu->arch.mmu->get_pdptr = kvm_pdptr_read;
|
||||
@ -5521,8 +5526,10 @@ static int nested_vmx_eptp_switching(struct kvm_vcpu *vcpu,
|
||||
return 1;
|
||||
|
||||
vmcs12->ept_pointer = new_eptp;
|
||||
nested_ept_new_eptp(vcpu);
|
||||
|
||||
kvm_make_request(KVM_REQ_MMU_RELOAD, vcpu);
|
||||
if (!nested_cpu_has_vpid(vmcs12))
|
||||
kvm_make_request(KVM_REQ_TLB_FLUSH_GUEST, vcpu);
|
||||
}
|
||||
|
||||
return 0;
|
||||
|
Loading…
x
Reference in New Issue
Block a user