From: Ackerley Tng <[email protected]>

Extract a helper out of kvm_gmem_range_is_private() that checks that a
range has given attributes.

Optimize setting memory attributes by returning early if all pages in the
requested range already has the requested attributes.

Signed-off-by: Ackerley Tng <[email protected]>
---
 virt/kvm/guest_memfd.c | 33 ++++++++++++++++++++++++---------
 1 file changed, 24 insertions(+), 9 deletions(-)

diff --git a/virt/kvm/guest_memfd.c b/virt/kvm/guest_memfd.c
index d563d80d4accb..d8bdb51c50cf0 100644
--- a/virt/kvm/guest_memfd.c
+++ b/virt/kvm/guest_memfd.c
@@ -92,6 +92,23 @@ static bool kvm_gmem_is_shared_mem(struct inode *inode, 
pgoff_t index)
        return !kvm_gmem_is_private_mem(inode, index);
 }
 
+static bool kvm_gmem_range_has_attributes(struct maple_tree *mt,
+                                         pgoff_t index, size_t nr_pages,
+                                         u64 attributes)
+{
+       pgoff_t end = index + nr_pages - 1;
+       void *entry;
+
+       lockdep_assert(mt_lock_is_held(mt));
+
+       mt_for_each(mt, entry, index, end) {
+               if (xa_to_value(entry) != attributes)
+                       return false;
+       }
+
+       return true;
+}
+
 static int __kvm_gmem_prepare_folio(struct kvm *kvm, struct kvm_memory_slot 
*slot,
                                    pgoff_t index, struct folio *folio)
 {
@@ -667,6 +684,11 @@ static int __kvm_gmem_set_attributes(struct inode *inode, 
pgoff_t start,
 
        filemap_invalidate_lock(mapping);
 
+       if (kvm_gmem_range_has_attributes(mt, start, nr_pages, attrs)) {
+               r = 0;
+               goto out;
+       }
+
        mas_init(&mas, mt, start);
        r = kvm_gmem_mas_preallocate(&mas, attrs, start, nr_pages);
        if (r) {
@@ -1152,20 +1174,13 @@ EXPORT_SYMBOL_FOR_KVM_INTERNAL(kvm_gmem_get_pfn);
 static bool kvm_gmem_range_is_private(struct gmem_inode *gi, pgoff_t index,
                                      size_t nr_pages, struct kvm *kvm, gfn_t 
gfn)
 {
-       pgoff_t end = index + nr_pages - 1;
-       void *entry;
-
        if (vm_memory_attributes)
                return kvm_range_has_vm_memory_attributes(kvm, gfn, gfn + 
nr_pages,
                                                       
KVM_MEMORY_ATTRIBUTE_PRIVATE,
                                                       
KVM_MEMORY_ATTRIBUTE_PRIVATE);
 
-       mt_for_each(&gi->attributes, entry, index, end) {
-               if (xa_to_value(entry) != KVM_MEMORY_ATTRIBUTE_PRIVATE)
-                       return false;
-       }
-
-       return true;
+       return kvm_gmem_range_has_attributes(&gi->attributes, index, nr_pages,
+                                            KVM_MEMORY_ATTRIBUTE_PRIVATE);
 }
 
 static long __kvm_gmem_populate(struct kvm *kvm, struct kvm_memory_slot *slot,

-- 
2.54.0.545.g6539524ca2-goog



Reply via email to