]> git.ipfire.org Git - thirdparty/linux.git/commitdiff
fuse: reject fuse_notify() pagecache ops on directories
authorJann Horn <jannh@google.com>
Tue, 19 May 2026 14:29:38 +0000 (16:29 +0200)
committerChristian Brauner <brauner@kernel.org>
Fri, 22 May 2026 13:19:26 +0000 (15:19 +0200)
The operations FUSE_NOTIFY_STORE and FUSE_NOTIFY_RETRIEVE allow the
FUSE daemon to actively write/read pagecache contents.

For directories with FOPEN_CACHE_DIR, the pagecache is used as
kernel-internal cache storage, and userspace is not supposed to have
direct access to this cache - in particular, fuse_parse_cache() will hit
WARN_ON() if the cache contains bogus data.

Reject FUSE_NOTIFY_STORE and FUSE_NOTIFY_RETRIEVE on anything other than
regular files with -EINVAL.

Fixes: 5d7bc7e8680c ("fuse: allow using readdir cache")
Cc: stable@vger.kernel.org
Signed-off-by: Jann Horn <jannh@google.com>
Link: https://patch.msgid.link/20260519-fuse-dir-pagecache-v2-1-5428fa48e175@google.com
Acked-by: Miklos Szeredi <mszeredi@redhat.com>
Signed-off-by: Christian Brauner (Amutable) <brauner@kernel.org>
fs/fuse/dev.c

index 08d364ed7d6c16e72d66c8f1f0a0c4235aaab825..c105aaf9ff5d7dc900f8c6dc6b28426d55677639 100644 (file)
@@ -1793,6 +1793,10 @@ static int fuse_notify_store(struct fuse_conn *fc, unsigned int size,
        inode = fuse_ilookup(fc, nodeid,  NULL);
        if (!inode)
                goto out_up_killsb;
+       if (!S_ISREG(inode->i_mode)) {
+               err = -EINVAL;
+               goto out_iput;
+       }
 
        mapping = inode->i_mapping;
        file_size = i_size_read(inode);
@@ -1970,7 +1974,10 @@ static int fuse_notify_retrieve(struct fuse_conn *fc, unsigned int size,
 
        inode = fuse_ilookup(fc, nodeid, &fm);
        if (inode) {
-               err = fuse_retrieve(fm, inode, &outarg);
+               if (!S_ISREG(inode->i_mode))
+                       err = -EINVAL;
+               else
+                       err = fuse_retrieve(fm, inode, &outarg);
                iput(inode);
        }
        up_read(&fc->killsb);