if new KVM_*_SREGS2 ioctls are used, the PDPTRs are
part of the migration state and thus are loaded
by those ioctls.

Signed-off-by: Maxim Levitsky <mlevi...@redhat.com>
---
 arch/x86/kvm/vmx/nested.c | 12 +++++++++++-
 1 file changed, 11 insertions(+), 1 deletion(-)

diff --git a/arch/x86/kvm/vmx/nested.c b/arch/x86/kvm/vmx/nested.c
index b44f1f6b68db..f2291165995e 100644
--- a/arch/x86/kvm/vmx/nested.c
+++ b/arch/x86/kvm/vmx/nested.c
@@ -1115,7 +1115,7 @@ static int nested_vmx_load_cr3(struct kvm_vcpu *vcpu, 
unsigned long cr3, bool ne
         * must not be dereferenced.
         */
        if (!nested_ept && is_pae_paging(vcpu) &&
-           (cr3 != kvm_read_cr3(vcpu) || pdptrs_changed(vcpu))) {
+           (cr3 != kvm_read_cr3(vcpu) || !kvm_register_is_available(vcpu, 
VCPU_EXREG_PDPTR))) {
                if (CC(!load_pdptrs(vcpu, vcpu->arch.walk_mmu, cr3))) {
                        *entry_failure_code = ENTRY_FAIL_PDPTE;
                        return -EINVAL;
@@ -3110,6 +3110,14 @@ static bool nested_get_vmcs12_pages(struct kvm_vcpu 
*vcpu)
        struct page *page;
        u64 hpa;
 
+       if (vcpu->arch.reload_pdptrs_on_nested_entry) {
+               /* if legacy KVM_SET_SREGS API was used, it might have loaded
+                * wrong PDPTRs from memory so we have to reload them here
+                * (which is against x86 spec)
+                */
+               kvm_register_clear_available(vcpu, VCPU_EXREG_PDPTR);
+       }
+
        if (nested_vmx_load_cr3(vcpu, vmcs12->guest_cr3, 
nested_cpu_has_ept(vmcs12),
                                &entry_failure_code))
                return false;
@@ -3357,6 +3365,7 @@ enum nvmx_vmentry_status 
nested_vmx_enter_non_root_mode(struct kvm_vcpu *vcpu,
        }
 
        if (from_vmentry) {
+               kvm_register_clear_available(vcpu, VCPU_EXREG_PDPTR);
                if (nested_vmx_load_cr3(vcpu, vmcs12->guest_cr3,
                    nested_cpu_has_ept(vmcs12), &entry_failure_code))
                        goto vmentry_fail_vmexit_guest_mode;
@@ -4195,6 +4204,7 @@ static void load_vmcs12_host_state(struct kvm_vcpu *vcpu,
         * Only PDPTE load can fail as the value of cr3 was checked on entry and
         * couldn't have changed.
         */
+       kvm_register_clear_available(vcpu, VCPU_EXREG_PDPTR);
        if (nested_vmx_load_cr3(vcpu, vmcs12->host_cr3, false, &ignored))
                nested_vmx_abort(vcpu, VMX_ABORT_LOAD_HOST_PDPTE_FAIL);
 
-- 
2.26.2

Reply via email to