162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * KVM L1 hypervisor optimizations on Hyper-V. 462306a36Sopenharmony_ci */ 562306a36Sopenharmony_ci#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt 662306a36Sopenharmony_ci 762306a36Sopenharmony_ci#include <linux/kvm_host.h> 862306a36Sopenharmony_ci#include <asm/mshyperv.h> 962306a36Sopenharmony_ci 1062306a36Sopenharmony_ci#include "hyperv.h" 1162306a36Sopenharmony_ci#include "kvm_onhyperv.h" 1262306a36Sopenharmony_ci 1362306a36Sopenharmony_cistruct kvm_hv_tlb_range { 1462306a36Sopenharmony_ci u64 start_gfn; 1562306a36Sopenharmony_ci u64 pages; 1662306a36Sopenharmony_ci}; 1762306a36Sopenharmony_ci 1862306a36Sopenharmony_cistatic int kvm_fill_hv_flush_list_func(struct hv_guest_mapping_flush_list *flush, 1962306a36Sopenharmony_ci void *data) 2062306a36Sopenharmony_ci{ 2162306a36Sopenharmony_ci struct kvm_hv_tlb_range *range = data; 2262306a36Sopenharmony_ci 2362306a36Sopenharmony_ci return hyperv_fill_flush_guest_mapping_list(flush, range->start_gfn, 2462306a36Sopenharmony_ci range->pages); 2562306a36Sopenharmony_ci} 2662306a36Sopenharmony_ci 2762306a36Sopenharmony_cistatic inline int hv_remote_flush_root_tdp(hpa_t root_tdp, 2862306a36Sopenharmony_ci struct kvm_hv_tlb_range *range) 2962306a36Sopenharmony_ci{ 3062306a36Sopenharmony_ci if (range) 3162306a36Sopenharmony_ci return hyperv_flush_guest_mapping_range(root_tdp, 3262306a36Sopenharmony_ci kvm_fill_hv_flush_list_func, (void *)range); 3362306a36Sopenharmony_ci else 3462306a36Sopenharmony_ci return hyperv_flush_guest_mapping(root_tdp); 3562306a36Sopenharmony_ci} 3662306a36Sopenharmony_ci 3762306a36Sopenharmony_cistatic int __hv_flush_remote_tlbs_range(struct kvm *kvm, 3862306a36Sopenharmony_ci struct kvm_hv_tlb_range *range) 3962306a36Sopenharmony_ci{ 4062306a36Sopenharmony_ci struct kvm_arch *kvm_arch = &kvm->arch; 4162306a36Sopenharmony_ci struct kvm_vcpu *vcpu; 4262306a36Sopenharmony_ci int ret = 0, nr_unique_valid_roots; 4362306a36Sopenharmony_ci unsigned long i; 4462306a36Sopenharmony_ci hpa_t root; 4562306a36Sopenharmony_ci 4662306a36Sopenharmony_ci spin_lock(&kvm_arch->hv_root_tdp_lock); 4762306a36Sopenharmony_ci 4862306a36Sopenharmony_ci if (!VALID_PAGE(kvm_arch->hv_root_tdp)) { 4962306a36Sopenharmony_ci nr_unique_valid_roots = 0; 5062306a36Sopenharmony_ci 5162306a36Sopenharmony_ci /* 5262306a36Sopenharmony_ci * Flush all valid roots, and see if all vCPUs have converged 5362306a36Sopenharmony_ci * on a common root, in which case future flushes can skip the 5462306a36Sopenharmony_ci * loop and flush the common root. 5562306a36Sopenharmony_ci */ 5662306a36Sopenharmony_ci kvm_for_each_vcpu(i, vcpu, kvm) { 5762306a36Sopenharmony_ci root = vcpu->arch.hv_root_tdp; 5862306a36Sopenharmony_ci if (!VALID_PAGE(root) || root == kvm_arch->hv_root_tdp) 5962306a36Sopenharmony_ci continue; 6062306a36Sopenharmony_ci 6162306a36Sopenharmony_ci /* 6262306a36Sopenharmony_ci * Set the tracked root to the first valid root. Keep 6362306a36Sopenharmony_ci * this root for the entirety of the loop even if more 6462306a36Sopenharmony_ci * roots are encountered as a low effort optimization 6562306a36Sopenharmony_ci * to avoid flushing the same (first) root again. 6662306a36Sopenharmony_ci */ 6762306a36Sopenharmony_ci if (++nr_unique_valid_roots == 1) 6862306a36Sopenharmony_ci kvm_arch->hv_root_tdp = root; 6962306a36Sopenharmony_ci 7062306a36Sopenharmony_ci if (!ret) 7162306a36Sopenharmony_ci ret = hv_remote_flush_root_tdp(root, range); 7262306a36Sopenharmony_ci 7362306a36Sopenharmony_ci /* 7462306a36Sopenharmony_ci * Stop processing roots if a failure occurred and 7562306a36Sopenharmony_ci * multiple valid roots have already been detected. 7662306a36Sopenharmony_ci */ 7762306a36Sopenharmony_ci if (ret && nr_unique_valid_roots > 1) 7862306a36Sopenharmony_ci break; 7962306a36Sopenharmony_ci } 8062306a36Sopenharmony_ci 8162306a36Sopenharmony_ci /* 8262306a36Sopenharmony_ci * The optimized flush of a single root can't be used if there 8362306a36Sopenharmony_ci * are multiple valid roots (obviously). 8462306a36Sopenharmony_ci */ 8562306a36Sopenharmony_ci if (nr_unique_valid_roots > 1) 8662306a36Sopenharmony_ci kvm_arch->hv_root_tdp = INVALID_PAGE; 8762306a36Sopenharmony_ci } else { 8862306a36Sopenharmony_ci ret = hv_remote_flush_root_tdp(kvm_arch->hv_root_tdp, range); 8962306a36Sopenharmony_ci } 9062306a36Sopenharmony_ci 9162306a36Sopenharmony_ci spin_unlock(&kvm_arch->hv_root_tdp_lock); 9262306a36Sopenharmony_ci return ret; 9362306a36Sopenharmony_ci} 9462306a36Sopenharmony_ci 9562306a36Sopenharmony_ciint hv_flush_remote_tlbs_range(struct kvm *kvm, gfn_t start_gfn, gfn_t nr_pages) 9662306a36Sopenharmony_ci{ 9762306a36Sopenharmony_ci struct kvm_hv_tlb_range range = { 9862306a36Sopenharmony_ci .start_gfn = start_gfn, 9962306a36Sopenharmony_ci .pages = nr_pages, 10062306a36Sopenharmony_ci }; 10162306a36Sopenharmony_ci 10262306a36Sopenharmony_ci return __hv_flush_remote_tlbs_range(kvm, &range); 10362306a36Sopenharmony_ci} 10462306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(hv_flush_remote_tlbs_range); 10562306a36Sopenharmony_ci 10662306a36Sopenharmony_ciint hv_flush_remote_tlbs(struct kvm *kvm) 10762306a36Sopenharmony_ci{ 10862306a36Sopenharmony_ci return __hv_flush_remote_tlbs_range(kvm, NULL); 10962306a36Sopenharmony_ci} 11062306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(hv_flush_remote_tlbs); 11162306a36Sopenharmony_ci 11262306a36Sopenharmony_civoid hv_track_root_tdp(struct kvm_vcpu *vcpu, hpa_t root_tdp) 11362306a36Sopenharmony_ci{ 11462306a36Sopenharmony_ci struct kvm_arch *kvm_arch = &vcpu->kvm->arch; 11562306a36Sopenharmony_ci 11662306a36Sopenharmony_ci if (kvm_x86_ops.flush_remote_tlbs == hv_flush_remote_tlbs) { 11762306a36Sopenharmony_ci spin_lock(&kvm_arch->hv_root_tdp_lock); 11862306a36Sopenharmony_ci vcpu->arch.hv_root_tdp = root_tdp; 11962306a36Sopenharmony_ci if (root_tdp != kvm_arch->hv_root_tdp) 12062306a36Sopenharmony_ci kvm_arch->hv_root_tdp = INVALID_PAGE; 12162306a36Sopenharmony_ci spin_unlock(&kvm_arch->hv_root_tdp_lock); 12262306a36Sopenharmony_ci } 12362306a36Sopenharmony_ci} 12462306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(hv_track_root_tdp); 125