memory: move exclusivity detection in do_wp_page() into wp_can_reuse_anon_folio()
authorDavid Hildenbrand <david@redhat.com>
Mon, 2 Oct 2023 14:29:49 +0000 (16:29 +0200)
committerAndrew Morton <akpm@linux-foundation.org>
Wed, 18 Oct 2023 21:34:14 +0000 (14:34 -0700)
Let's clean up do_wp_page() a bit, removing two labels and making it a
easier to read.

wp_can_reuse_anon_folio() now only operates on the whole folio.  Move the
SetPageAnonExclusive() out into do_wp_page().  No need to do this under
page lock -- the page table lock is sufficient.

Link: https://lkml.kernel.org/r/20231002142949.235104-4-david@redhat.com
Signed-off-by: David Hildenbrand <david@redhat.com>
Cc: Mike Kravetz <mike.kravetz@oracle.com>
Cc: Muchun Song <muchun.song@linux.dev>
Cc: Suren Baghdasaryan <surenb@google.com>
Cc: Matthew Wilcox <willy@infradead.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
mm/memory.c

index 6c67828f934c571c0727c3121182fe5846eebdcc..22784d3b886d01b39d4c0d9e4d4a36b8140f7fa6 100644 (file)
@@ -3355,6 +3355,44 @@ static vm_fault_t wp_page_shared(struct vm_fault *vmf, struct folio *folio)
        return ret;
 }
 
+static bool wp_can_reuse_anon_folio(struct folio *folio,
+                                   struct vm_area_struct *vma)
+{
+       /*
+        * We have to verify under folio lock: these early checks are
+        * just an optimization to avoid locking the folio and freeing
+        * the swapcache if there is little hope that we can reuse.
+        *
+        * KSM doesn't necessarily raise the folio refcount.
+        */
+       if (folio_test_ksm(folio) || folio_ref_count(folio) > 3)
+               return false;
+       if (!folio_test_lru(folio))
+               /*
+                * We cannot easily detect+handle references from
+                * remote LRU caches or references to LRU folios.
+                */
+               lru_add_drain();
+       if (folio_ref_count(folio) > 1 + folio_test_swapcache(folio))
+               return false;
+       if (!folio_trylock(folio))
+               return false;
+       if (folio_test_swapcache(folio))
+               folio_free_swap(folio);
+       if (folio_test_ksm(folio) || folio_ref_count(folio) != 1) {
+               folio_unlock(folio);
+               return false;
+       }
+       /*
+        * Ok, we've got the only folio reference from our mapping
+        * and the folio is locked, it's dark out, and we're wearing
+        * sunglasses. Hit it.
+        */
+       folio_move_anon_rmap(folio, vma);
+       folio_unlock(folio);
+       return true;
+}
+
 /*
  * This routine handles present pages, when
  * * users try to write to a shared page (FAULT_FLAG_WRITE)
@@ -3441,49 +3479,14 @@ static vm_fault_t do_wp_page(struct vm_fault *vmf)
        /*
         * Private mapping: create an exclusive anonymous page copy if reuse
         * is impossible. We might miss VM_WRITE for FOLL_FORCE handling.
+        *
+        * If we encounter a page that is marked exclusive, we must reuse
+        * the page without further checks.
         */
-       if (folio && folio_test_anon(folio)) {
-               /*
-                * If the page is exclusive to this process we must reuse the
-                * page without further checks.
-                */
-               if (PageAnonExclusive(vmf->page))
-                       goto reuse;
-
-               /*
-                * We have to verify under folio lock: these early checks are
-                * just an optimization to avoid locking the folio and freeing
-                * the swapcache if there is little hope that we can reuse.
-                *
-                * KSM doesn't necessarily raise the folio refcount.
-                */
-               if (folio_test_ksm(folio) || folio_ref_count(folio) > 3)
-                       goto copy;
-               if (!folio_test_lru(folio))
-                       /*
-                        * We cannot easily detect+handle references from
-                        * remote LRU caches or references to LRU folios.
-                        */
-                       lru_add_drain();
-               if (folio_ref_count(folio) > 1 + folio_test_swapcache(folio))
-                       goto copy;
-               if (!folio_trylock(folio))
-                       goto copy;
-               if (folio_test_swapcache(folio))
-                       folio_free_swap(folio);
-               if (folio_test_ksm(folio) || folio_ref_count(folio) != 1) {
-                       folio_unlock(folio);
-                       goto copy;
-               }
-               /*
-                * Ok, we've got the only folio reference from our mapping
-                * and the folio is locked, it's dark out, and we're wearing
-                * sunglasses. Hit it.
-                */
-               folio_move_anon_rmap(folio, vma);
-               SetPageAnonExclusive(vmf->page);
-               folio_unlock(folio);
-reuse:
+       if (folio && folio_test_anon(folio) &&
+           (PageAnonExclusive(vmf->page) || wp_can_reuse_anon_folio(folio, vma))) {
+               if (!PageAnonExclusive(vmf->page))
+                       SetPageAnonExclusive(vmf->page);
                if (unlikely(unshare)) {
                        pte_unmap_unlock(vmf->pte, vmf->ptl);
                        return 0;
@@ -3491,7 +3494,6 @@ reuse:
                wp_page_reuse(vmf);
                return 0;
        }
-copy:
        /*
         * Ok, we need to copy. Oh, well..
         */