]> www.infradead.org Git - users/jedix/linux-maple.git/commitdiff
mm: specify separate file and vm_file params in vm_area_desc
authorLorenzo Stoakes <lorenzo.stoakes@oracle.com>
Wed, 3 Sep 2025 17:48:41 +0000 (18:48 +0100)
committerAndrew Morton <akpm@linux-foundation.org>
Fri, 12 Sep 2025 00:25:48 +0000 (17:25 -0700)
Patch series "mm: do not assume file == vma->vm_file in
compat_vma_mmap_prepare()", v2.

As part of the efforts to eliminate the problematic f_op->mmap callback, a
new callback - f_op->mmap_prepare was provided.

While we are converting these callbacks, we must deal with 'stacked'
filesystems and drivers - those which in their own f_op->mmap callback
invoke an inner f_op->mmap callback.

To accomodate for this, a compatibility layer is provided that, via
vfs_mmap(), detects if f_op->mmap_prepare is provided and if so, generates
a vm_area_desc containing the VMA's metadata and invokes the call.

So far, we have provided desc->file equal to vma->vm_file.  However this
is not necessarily valid, especially in the case of stacked drivers which
wish to assign a new file after the inner hook is invoked.

To account for this, we adjust vm_area_desc to have both file and vm_file
fields.  The .vm_file field is strictly set to vma->vm_file (or in the
case of a new mapping, what will become vma->vm_file).

However, .file is set to whichever file vfs_mmap() is invoked with when
using the compatibilty layer.

Therefore, if the VMA's file needs to be updated in .mmap_prepare,
desc->vm_file should be assigned, whilst desc->file should be read.

No current f_op->mmap_prepare users assign desc->file so this is safe to
do.

This makes the .mmap_prepare callback in the context of a stacked
filesystem or driver completely consistent with the existing .mmap
implementations.

While we're here, we do a few small cleanups, and ensure that we const-ify
things correctly in the vm_area_desc struct to avoid hooks accidentally
trying to assign fields they should not.

This patch (of 2):

Stacked filesystems and drivers may invoke mmap hooks with a struct file
pointer that differs from the overlying file.  We will make this
functionality possible in a subsequent patch.

In order to prepare for this, let's update vm_area_struct to separately
provide desc->file and desc->vm_file parameters.

The desc->file parameter is the file that the hook is expected to operate
upon, and is not assignable (though the hok may wish to e.g.  update the
file's accessed time for instance).

The desc->vm_file defaults to what will become vma->vm_file and is what
the hook must reassign should it wish to change the VMA"s vma->vm_file.

For now we keep desc->file, vm_file the same to remain consistent.

No f_op->mmap_prepare() callback sets a new vma->vm_file currently, so
this is safe to change.

While we're here, make the mm_struct desc->mm pointers at immutable as
well as the desc->mm field itself.

As part of this change, also update the single hook which this would
otherwise break - mlock_future_ok(), invoked by secretmem_mmap_prepare()).

We additionally update set_vma_from_desc() to compare fields in a more
logical fashion, checking the (possibly) user-modified fields as the first
operand against the existing value as the second one.

Additionally, update VMA tests to accommodate changes.

Link: https://lkml.kernel.org/r/cover.1756920635.git.lorenzo.stoakes@oracle.com
Link: https://lkml.kernel.org/r/3fa15a861bb7419f033d22970598aa61850ea267.1756920635.git.lorenzo.stoakes@oracle.com
Signed-off-by: Lorenzo Stoakes <lorenzo.stoakes@oracle.com>
Cc: Al Viro <viro@zeniv.linux.org.uk>
Cc: Christian Brauner <brauner@kernel.org>
Cc: David Hildenbrand <david@redhat.com>
Cc: Jan Kara <jack@suse.cz>
Cc: Jann Horn <jannh@google.com>
Cc: Liam Howlett <liam.howlett@oracle.com>
Cc: Michal Hocko <mhocko@suse.com>
Cc: Mike Rapoport <rppt@kernel.org>
Cc: Suren Baghdasaryan <surenb@google.com>
Cc: Vlastimil Babka <vbabka@suse.cz>
Cc: Pedro Falcato <pfalcato@suse.de>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
include/linux/mm_types.h
mm/internal.h
mm/mmap.c
mm/util.c
mm/vma.c
mm/vma.h
tools/testing/vma/vma_internal.h

index ff2b4e13215f4e05e39b67dfe30fc8e7d210c27d..cf759fe08bb3e1b61cd419be8afbc6b162296f8a 100644 (file)
@@ -779,13 +779,14 @@ struct pfnmap_track_ctx {
  */
 struct vm_area_desc {
        /* Immutable state. */
-       struct mm_struct *mm;
+       const struct mm_struct *const mm;
+       struct file *const file; /* May vary from vm_file in stacked callers. */
        unsigned long start;
        unsigned long end;
 
        /* Mutable fields. Populated with initial state. */
        pgoff_t pgoff;
-       struct file *file;
+       struct file *vm_file;
        vm_flags_t vm_flags;
        pgprot_t page_prot;
 
index 9b0129531d004ff0e9cec9cd84ca866fd8ef1585..456a41e8ed28d555bbecbc55a6a1e88b83c89f9e 100644 (file)
@@ -962,8 +962,8 @@ extern long populate_vma_page_range(struct vm_area_struct *vma,
                unsigned long start, unsigned long end, int *locked);
 extern long faultin_page_range(struct mm_struct *mm, unsigned long start,
                unsigned long end, bool write, int *locked);
-extern bool mlock_future_ok(struct mm_struct *mm, vm_flags_t vm_flags,
-                              unsigned long bytes);
+bool mlock_future_ok(const struct mm_struct *mm, vm_flags_t vm_flags,
+               unsigned long bytes);
 
 /*
  * NOTE: This function can't tell whether the folio is "fully mapped" in the
index 7a057e0e8da923bc45f3e9d02f35d2b1fc6ac200..5fd3b80fda1d5151c230321cb3228e935e4fa84b 100644 (file)
--- a/mm/mmap.c
+++ b/mm/mmap.c
@@ -225,7 +225,7 @@ static inline unsigned long round_hint_to_min(unsigned long hint)
        return hint;
 }
 
-bool mlock_future_ok(struct mm_struct *mm, vm_flags_t vm_flags,
+bool mlock_future_ok(const struct mm_struct *mm, vm_flags_t vm_flags,
                        unsigned long bytes)
 {
        unsigned long locked_pages, limit_pages;
index 732a2dfcaec7c54b97c0f21ba5d7111c5fb60d3c..215ecd0214b715f4a0ccbc8d04cf6fd3160d9f94 100644 (file)
--- a/mm/util.c
+++ b/mm/util.c
@@ -1161,10 +1161,20 @@ EXPORT_SYMBOL(flush_dcache_folio);
  */
 int compat_vma_mmap_prepare(struct file *file, struct vm_area_struct *vma)
 {
-       struct vm_area_desc desc;
+       struct vm_area_desc desc = {
+               .mm = vma->vm_mm,
+               .file = vma->vm_file,
+               .start = vma->vm_start,
+               .end = vma->vm_end,
+
+               .pgoff = vma->vm_pgoff,
+               .vm_file = vma->vm_file,
+               .vm_flags = vma->vm_flags,
+               .page_prot = vma->vm_page_prot,
+       };
        int err;
 
-       err = file->f_op->mmap_prepare(vma_to_desc(vma, &desc));
+       err = file->f_op->mmap_prepare(&desc);
        if (err)
                return err;
        set_vma_from_desc(vma, &desc);
index 3b12c7579831b3c75bf2e855784d2bf1ae0b30a7..abe0da33c8446157c6fa2fea9e2c22651136a873 100644 (file)
--- a/mm/vma.c
+++ b/mm/vma.c
@@ -2572,11 +2572,12 @@ static int call_mmap_prepare(struct mmap_state *map)
        int err;
        struct vm_area_desc desc = {
                .mm = map->mm,
+               .file = map->file,
                .start = map->addr,
                .end = map->end,
 
                .pgoff = map->pgoff,
-               .file = map->file,
+               .vm_file = map->file,
                .vm_flags = map->vm_flags,
                .page_prot = map->page_prot,
        };
@@ -2588,7 +2589,7 @@ static int call_mmap_prepare(struct mmap_state *map)
 
        /* Update fields permitted to be changed. */
        map->pgoff = desc.pgoff;
-       map->file = desc.file;
+       map->file = desc.vm_file;
        map->vm_flags = desc.vm_flags;
        map->page_prot = desc.page_prot;
        /* User-defined fields. */
index bcdc261c5b15c850f3f84170aaa9316e06ae68bb..9183fe549009076fad74863c49a00413fe33ef08 100644 (file)
--- a/mm/vma.h
+++ b/mm/vma.h
@@ -222,31 +222,11 @@ static inline int vma_iter_store_gfp(struct vma_iterator *vmi,
        return 0;
 }
 
-
 /*
- * Temporary helper functions for file systems which wrap an invocation of
+ * Temporary helper function for stacked mmap handlers which specify
  * f_op->mmap() but which might have an underlying file system which implements
  * f_op->mmap_prepare().
  */
-
-static inline struct vm_area_desc *vma_to_desc(struct vm_area_struct *vma,
-               struct vm_area_desc *desc)
-{
-       desc->mm = vma->vm_mm;
-       desc->start = vma->vm_start;
-       desc->end = vma->vm_end;
-
-       desc->pgoff = vma->vm_pgoff;
-       desc->file = vma->vm_file;
-       desc->vm_flags = vma->vm_flags;
-       desc->page_prot = vma->vm_page_prot;
-
-       desc->vm_ops = NULL;
-       desc->private_data = NULL;
-
-       return desc;
-}
-
 static inline void set_vma_from_desc(struct vm_area_struct *vma,
                struct vm_area_desc *desc)
 {
@@ -258,9 +238,9 @@ static inline void set_vma_from_desc(struct vm_area_struct *vma,
 
        /* Mutable fields. Populated with initial state. */
        vma->vm_pgoff = desc->pgoff;
-       if (vma->vm_file != desc->file)
-               vma_set_file(vma, desc->file);
-       if (vma->vm_flags != desc->vm_flags)
+       if (desc->vm_file != vma->vm_file)
+               vma_set_file(vma, desc->vm_file);
+       if (desc->vm_flags != vma->vm_flags)
                vm_flags_set(vma, desc->vm_flags);
        vma->vm_page_prot = desc->page_prot;
 
index 437d2a1013be418c6d3e77534fa19b0ee1518c9b..6dcbeaa9f9a0deeff384dc6072db877ff5327621 100644 (file)
@@ -283,13 +283,14 @@ struct vm_area_struct;
  */
 struct vm_area_desc {
        /* Immutable state. */
-       struct mm_struct *mm;
+       const struct mm_struct *const mm;
+       struct file *const file; /* May vary from vm_file in stacked callers. */
        unsigned long start;
        unsigned long end;
 
        /* Mutable fields. Populated with initial state. */
        pgoff_t pgoff;
-       struct file *file;
+       struct file *vm_file;
        vm_flags_t vm_flags;
        pgprot_t page_prot;
 
@@ -1299,8 +1300,8 @@ static inline bool capable(int cap)
        return true;
 }
 
-static inline bool mlock_future_ok(struct mm_struct *mm, vm_flags_t vm_flags,
-                       unsigned long bytes)
+static inline bool mlock_future_ok(const struct mm_struct *mm,
+               vm_flags_t vm_flags, unsigned long bytes)
 {
        unsigned long locked_pages, limit_pages;
 
@@ -1465,16 +1466,23 @@ static inline void free_anon_vma_name(struct vm_area_struct *vma)
 static inline void set_vma_from_desc(struct vm_area_struct *vma,
                struct vm_area_desc *desc);
 
-static inline struct vm_area_desc *vma_to_desc(struct vm_area_struct *vma,
-               struct vm_area_desc *desc);
-
-static int compat_vma_mmap_prepare(struct file *file,
+static inline int compat_vma_mmap_prepare(struct file *file,
                struct vm_area_struct *vma)
 {
-       struct vm_area_desc desc;
+       struct vm_area_desc desc = {
+               .mm = vma->vm_mm,
+               .file = vma->vm_file,
+               .start = vma->vm_start,
+               .end = vma->vm_end,
+
+               .pgoff = vma->vm_pgoff,
+               .vm_file = vma->vm_file,
+               .vm_flags = vma->vm_flags,
+               .page_prot = vma->vm_page_prot,
+       };
        int err;
 
-       err = file->f_op->mmap_prepare(vma_to_desc(vma, &desc));
+       err = file->f_op->mmap_prepare(&desc);
        if (err)
                return err;
        set_vma_from_desc(vma, &desc);