From: Harry Yoo <harry.yoo@oracle.com>
To: Andrew Morton <akpm@linux-foundation.org>,
Suren Baghdasaryan <surenb@google.com>,
"Liam R . Howlett" <Liam.Howlett@oracle.com>,
Lorenzo Stoakes <lorenzo.stoakes@oracle.com>,
David Hildenbrand <david@redhat.com>, Kees Cook <kees@kernel.org>
Cc: Vlastimil Babka <vbabka@suse.cz>,
Shakeel Butt <shakeel.butt@linux.dev>,
Mike Rapoport <rppt@kernel.org>, Michal Hocko <mhocko@suse.com>,
Jonathan Corbet <corbet@lwn.net>, Jann Horn <jannh@google.com>,
Pedro Falcato <pfalcato@suse.de>, Rik van Riel <riel@surriel.com>,
linux-mm@kvack.org, linux-doc@vger.kernel.org,
linux-kernel@vger.kernel.org, Harry Yoo <harry.yoo@oracle.com>
Subject: [PATCH V1 1/2] docs/mm: explain when and why rmap locks need to be taken during mremap()
Date: Tue, 26 Aug 2025 15:58:47 +0900 [thread overview]
Message-ID: <20250826065848.346066-1-harry.yoo@oracle.com> (raw)
While move_ptes() has a comment explaining why rmap locks are needed,
Documentation/mm/process_addrs.rst does not. Without being aware of that
comment, I spent hours figuring out how things could go wrong and why,
in some cases, rmap locks can be safely skipped.
Add a more comprehensive explanation to the documentation to save time
for others.
Signed-off-by: Harry Yoo <harry.yoo@oracle.com>
---
Documentation/mm/process_addrs.rst | 32 ++++++++++++++++++++++++++++++
1 file changed, 32 insertions(+)
diff --git a/Documentation/mm/process_addrs.rst b/Documentation/mm/process_addrs.rst
index be49e2a269e4..ee7c0dba339e 100644
--- a/Documentation/mm/process_addrs.rst
+++ b/Documentation/mm/process_addrs.rst
@@ -744,6 +744,38 @@ You can observe this in the :c:func:`!mremap` implementation in the functions
:c:func:`!take_rmap_locks` and :c:func:`!drop_rmap_locks` which perform the rmap
side of lock acquisition, invoked ultimately by :c:func:`!move_page_tables`.
+.. note:: If :c:func:`!mremap()` -> :c:func:`!move_ptes()` does not take rmap
+ locks, :c:func:`!rmap_walk()` may miss a pte for the folio.
+
+ The problematic sequence is as follows:
+
+ 1. :c:func:`!rmap_walk()` checks the destination VMA, finds no pte,
+ and releases the page table lock.
+ 2. :c:func:`!move_ptes()` moves the page tables from the source to the
+ destination.
+ 3. :c:func:`!rmap_walk()` checks the source VMA, finds no pte, and
+ thus rmap walk misses it.
+
+ Taking rmap locks in :c:func:`!move_ptes()` ensures that
+ :c:func:`!rmap_walk()` sees the pte in either the source or
+ destination VMA.
+
+ There are two cases where rmap locks can be skipped:
+
+ 1. If the source VMA is guaranteed to be visited before the
+ destination VMA during rmap walk, :c:func:`!rmap_walk()` will
+ encounter the pte in one of the two VMAs. VMAs associated with
+ an anon_vma are organized in an interval tree, so the src->dst
+ order is guaranteed when the source VMA's vm_pgoff precedes
+ the destination VMA's vm_pgoff.
+
+ 2. When :c:func:`!exec()` relocates a temporary stack VMA via
+ :c:func:`!relocate_vma_down()`, there is no separate destination
+ VMA. Instead, the source VMA is marked as a temporary stack and
+ relocated. In this case, the folios belonging to the VMA cannot be
+ migrated until the relocation is complete, avoiding the need to
+ acquire rmap locks for performance reasons.
+
VMA lock internals
------------------
--
2.43.0
next reply other threads:[~2025-08-26 6:59 UTC|newest]
Thread overview: 12+ messages / expand[flat|nested] mbox.gz Atom feed top
2025-08-26 6:58 Harry Yoo [this message]
2025-08-26 6:58 ` [PATCH V1 2/2] mm: document when rmap locks can be skipped when setting need_rmap_locks Harry Yoo
2025-08-26 9:46 ` Lorenzo Stoakes
2025-08-27 6:52 ` Harry Yoo
2025-08-27 11:16 ` Lorenzo Stoakes
2025-08-26 7:22 ` [PATCH V1 1/2] docs/mm: explain when and why rmap locks need to be taken during mremap() Jonathan Corbet
2025-08-26 8:37 ` Lorenzo Stoakes
2025-08-26 9:48 ` Harry Yoo
2025-08-26 9:58 ` Lorenzo Stoakes
2025-08-27 7:18 ` Harry Yoo
2025-08-27 9:25 ` Lorenzo Stoakes
2025-08-26 9:55 ` Lorenzo Stoakes
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=20250826065848.346066-1-harry.yoo@oracle.com \
--to=harry.yoo@oracle.com \
--cc=Liam.Howlett@oracle.com \
--cc=akpm@linux-foundation.org \
--cc=corbet@lwn.net \
--cc=david@redhat.com \
--cc=jannh@google.com \
--cc=kees@kernel.org \
--cc=linux-doc@vger.kernel.org \
--cc=linux-kernel@vger.kernel.org \
--cc=linux-mm@kvack.org \
--cc=lorenzo.stoakes@oracle.com \
--cc=mhocko@suse.com \
--cc=pfalcato@suse.de \
--cc=riel@surriel.com \
--cc=rppt@kernel.org \
--cc=shakeel.butt@linux.dev \
--cc=surenb@google.com \
--cc=vbabka@suse.cz \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox