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