KVM: x86: hyperv: optimize 'all cpus' case in kvm_hv_flush_tlb()
authorVitaly Kuznetsov <vkuznets@redhat.com>
Wed, 22 Aug 2018 10:18:29 +0000 (12:18 +0200)
committerPaolo Bonzini <pbonzini@redhat.com>
Tue, 16 Oct 2018 22:29:44 +0000 (00:29 +0200)
We can use 'NULL' to represent 'all cpus' case in
kvm_make_vcpus_request_mask() and avoid building vCPU mask with
all vCPUs.

Suggested-by: Radim Krčmář <rkrcmar@redhat.com>
Signed-off-by: Vitaly Kuznetsov <vkuznets@redhat.com>
Reviewed-by: Roman Kagan <rkagan@virtuozzo.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
arch/x86/kvm/hyperv.c
virt/kvm/kvm_main.c

index 0cd597b0f7546f7a9ce4169d3c7551ef27b30e14..b45ce136be2fe0cf57813a2fd6862f62677fd339 100644 (file)
@@ -1325,35 +1325,39 @@ static u64 kvm_hv_flush_tlb(struct kvm_vcpu *current_vcpu, u64 ingpa,
 
        cpumask_clear(&hv_current->tlb_lush);
 
+       if (all_cpus) {
+               kvm_make_vcpus_request_mask(kvm,
+                                   KVM_REQ_TLB_FLUSH | KVM_REQUEST_NO_WAKEUP,
+                                   NULL, &hv_current->tlb_lush);
+               goto ret_success;
+       }
+
        kvm_for_each_vcpu(i, vcpu, kvm) {
                struct kvm_vcpu_hv *hv = &vcpu->arch.hyperv;
                int bank = hv->vp_index / 64, sbank = 0;
 
-               if (!all_cpus) {
-                       /* Banks >64 can't be represented */
-                       if (bank >= 64)
-                               continue;
-
-                       /* Non-ex hypercalls can only address first 64 vCPUs */
-                       if (!ex && bank)
-                               continue;
+               /* Banks >64 can't be represented */
+               if (bank >= 64)
+                       continue;
 
-                       if (ex) {
-                               /*
-                                * Check is the bank of this vCPU is in sparse
-                                * set and get the sparse bank number.
-                                */
-                               sbank = get_sparse_bank_no(valid_bank_mask,
-                                                          bank);
+               /* Non-ex hypercalls can only address first 64 vCPUs */
+               if (!ex && bank)
+                       continue;
 
-                               if (sbank < 0)
-                                       continue;
-                       }
+               if (ex) {
+                       /*
+                        * Check is the bank of this vCPU is in sparse
+                        * set and get the sparse bank number.
+                        */
+                       sbank = get_sparse_bank_no(valid_bank_mask, bank);
 
-                       if (!(sparse_banks[sbank] & BIT_ULL(hv->vp_index % 64)))
+                       if (sbank < 0)
                                continue;
                }
 
+               if (!(sparse_banks[sbank] & BIT_ULL(hv->vp_index % 64)))
+                       continue;
+
                /*
                 * vcpu->arch.cr3 may not be up-to-date for running vCPUs so we
                 * can't analyze it here, flush TLB regardless of the specified
index f986e31fa68cceaf8b16385e9869dc5e2bf55e4a..587e1a0a87156b4c9b0ee4e9689f684ecf575f99 100644 (file)
@@ -219,7 +219,7 @@ bool kvm_make_vcpus_request_mask(struct kvm *kvm, unsigned int req,
        me = get_cpu();
 
        kvm_for_each_vcpu(i, vcpu, kvm) {
-               if (!test_bit(i, vcpu_bitmap))
+               if (vcpu_bitmap && !test_bit(i, vcpu_bitmap))
                        continue;
 
                kvm_make_request(req, vcpu);
@@ -243,12 +243,10 @@ bool kvm_make_all_cpus_request(struct kvm *kvm, unsigned int req)
 {
        cpumask_var_t cpus;
        bool called;
-       static unsigned long vcpu_bitmap[BITS_TO_LONGS(KVM_MAX_VCPUS)]
-               = {[0 ... BITS_TO_LONGS(KVM_MAX_VCPUS)-1] = ULONG_MAX};
 
        zalloc_cpumask_var(&cpus, GFP_ATOMIC);
 
-       called = kvm_make_vcpus_request_mask(kvm, req, vcpu_bitmap, cpus);
+       called = kvm_make_vcpus_request_mask(kvm, req, NULL, cpus);
 
        free_cpumask_var(cpus);
        return called;