KVM: nSVM: Track the physical cpu of the vmcb vmrun through the vmcb
authorCathy Avery <cavery@redhat.com>
Tue, 12 Jan 2021 16:43:12 +0000 (11:43 -0500)
committerPaolo Bonzini <pbonzini@redhat.com>
Mon, 15 Mar 2021 08:42:29 +0000 (04:42 -0400)
This patch moves the physical cpu tracking from the vcpu
to the vmcb in svm_switch_vmcb. If either vmcb01 or vmcb02
change physical cpus from one vmrun to the next the vmcb's
previous cpu is preserved for comparison with the current
cpu and the vmcb is marked dirty if different. This prevents
the processor from using old cached data for a vmcb that may
have been updated on a prior run on a different processor.

It also moves the physical cpu check from svm_vcpu_load
to pre_svm_run as the check only needs to be done at run.

Suggested-by: Paolo Bonzini <pbonzini@redhat.com>
Signed-off-by: Cathy Avery <cavery@redhat.com>
Message-Id: <20210112164313.4204-2-cavery@redhat.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
arch/x86/kvm/svm/svm.c
arch/x86/kvm/svm/svm.h

index aa7f6aee1d69cbe8dec8b13cc7726a7023ec3519..a6b02f6241a7dca3bc6e20ed6bd3d9fa57cd53a0 100644 (file)
@@ -1319,11 +1319,12 @@ void svm_switch_vmcb(struct vcpu_svm *svm, struct kvm_vmcb_info *target_vmcb)
        svm->asid_generation = 0;
 
        /*
-       * Workaround: we don't yet track the physical CPU that
-       * target_vmcb has run on.
+       * Track the physical CPU the target_vmcb is running on
+       * in order to mark the VMCB dirty if the cpu changes at
+       * its next vmrun.
        */
 
-       vmcb_mark_all_dirty(svm->vmcb);
+       svm->current_vmcb->cpu = svm->vcpu.cpu;
 }
 
 static int svm_create_vcpu(struct kvm_vcpu *vcpu)
@@ -1499,11 +1500,6 @@ static void svm_vcpu_load(struct kvm_vcpu *vcpu, int cpu)
        struct vcpu_svm *svm = to_svm(vcpu);
        struct svm_cpu_data *sd = per_cpu(svm_data, cpu);
 
-       if (unlikely(cpu != vcpu->cpu)) {
-               svm->asid_generation = 0;
-               vmcb_mark_all_dirty(svm->vmcb);
-       }
-
        if (sd->current_vmcb != svm->vmcb) {
                sd->current_vmcb = svm->vmcb;
                indirect_branch_prediction_barrier();
@@ -3435,6 +3431,17 @@ static void pre_svm_run(struct vcpu_svm *svm)
 {
        struct svm_cpu_data *sd = per_cpu(svm_data, svm->vcpu.cpu);
 
+       /*
+        * If the previous vmrun of the vmcb occurred on
+        * a different physical cpu then we must mark the vmcb dirty.
+        */
+
+        if (unlikely(svm->current_vmcb->cpu != svm->vcpu.cpu)) {
+               svm->asid_generation = 0;
+               vmcb_mark_all_dirty(svm->vmcb);
+               svm->current_vmcb->cpu = svm->vcpu.cpu;
+        }
+
        if (sev_guest(svm->vcpu.kvm))
                return pre_sev_run(svm, svm->vcpu.cpu);
 
index 818b37388d8c7919517569cfc3a0c1be91866569..a372810977512f59991e96a4fe490e75e21c86b9 100644 (file)
@@ -84,6 +84,7 @@ struct kvm_vcpu;
 struct kvm_vmcb_info {
        struct vmcb *ptr;
        unsigned long pa;
+       int cpu;
 };
 
 struct svm_nested_state {