]> git.ipfire.org Git - thirdparty/linux.git/commitdiff
fuse: make sure dentry is evicted if stale
authorMiklos Szeredi <mszeredi@redhat.com>
Wed, 14 Jan 2026 14:53:39 +0000 (15:53 +0100)
committerChristian Brauner <brauner@kernel.org>
Fri, 16 Jan 2026 18:15:14 +0000 (19:15 +0100)
d_dispose_if_unused() may find the dentry with a positive refcount, in
which case it won't be put on the dispose list even though it has already
timed out.

"Reinstall" the d_delete() callback, which was optimized out in
fuse_dentry_settime().  This will result in the dentry being evicted as
soon as the refcount hits zero.

Fixes: ab84ad597386 ("fuse: new work queue to periodically invalidate expired dentries")
Signed-off-by: Miklos Szeredi <mszeredi@redhat.com>
Link: https://patch.msgid.link/20260114145344.468856-3-mszeredi@redhat.com
Signed-off-by: Christian Brauner <brauner@kernel.org>
fs/fuse/dir.c

index ea90dd682bc369934f103a9712385f5d8dfa76c2..c9922af79dfac374774b90cadf9f71db6ebf6f97 100644 (file)
@@ -172,6 +172,10 @@ static void fuse_dentry_tree_work(struct work_struct *work)
                        if (time_after64(get_jiffies_64(), fd->time)) {
                                rb_erase(&fd->node, &dentry_hash[i].tree);
                                RB_CLEAR_NODE(&fd->node);
+                               spin_lock(&fd->dentry->d_lock);
+                               /* If dentry is still referenced, let next dput release it */
+                               fd->dentry->d_flags |= DCACHE_OP_DELETE;
+                               spin_unlock(&fd->dentry->d_lock);
                                d_dispose_if_unused(fd->dentry, &dispose);
                                spin_unlock(&dentry_hash[i].lock);
                                cond_resched();