]> git.ipfire.org Git - thirdparty/linux.git/commitdiff
ovl: change ovl_cleanup_and_whiteout() to take rename lock as needed
authorNeilBrown <neil@brown.name>
Wed, 16 Jul 2025 00:44:28 +0000 (10:44 +1000)
committerChristian Brauner <brauner@kernel.org>
Fri, 18 Jul 2025 09:10:42 +0000 (11:10 +0200)
Rather than locking the directory(s) before calling
ovl_cleanup_and_whiteout(), change it (and ovl_whiteout()) to do the
locking, so the locking can be fine grained as will be needed for
proposed locking changes.

Sometimes this is called to whiteout something in the index dir, in
which case only that dir must be locked.  In one case it is called on
something in an upperdir, so two directories must be locked.  We use
ovl_lock_rename_workdir() for this and remove the restriction that
upperdir cannot be indexdir - because now sometimes it is.

Reviewed-by: Amir Goldstein <amir73il@gmail.com>
Signed-off-by: NeilBrown <neil@brown.name>
Link: https://lore.kernel.org/20250716004725.1206467-18-neil@brown.name
Signed-off-by: Christian Brauner <brauner@kernel.org>
fs/overlayfs/dir.c
fs/overlayfs/readdir.c
fs/overlayfs/util.c

index 340f8679b6e7fe5179b7f9cc69f3a38b358b365f..6a70faeee6fad2b2be05563b85ce5b87b891fef3 100644 (file)
@@ -77,7 +77,6 @@ struct dentry *ovl_lookup_temp(struct ovl_fs *ofs, struct dentry *workdir)
        return temp;
 }
 
-/* caller holds i_mutex on workdir */
 static struct dentry *ovl_whiteout(struct ovl_fs *ofs)
 {
        int err;
@@ -85,6 +84,7 @@ static struct dentry *ovl_whiteout(struct ovl_fs *ofs)
        struct dentry *workdir = ofs->workdir;
        struct inode *wdir = workdir->d_inode;
 
+       inode_lock_nested(wdir, I_MUTEX_PARENT);
        if (!ofs->whiteout) {
                whiteout = ovl_lookup_temp(ofs, workdir);
                if (IS_ERR(whiteout))
@@ -118,14 +118,13 @@ static struct dentry *ovl_whiteout(struct ovl_fs *ofs)
        whiteout = ofs->whiteout;
        ofs->whiteout = NULL;
 out:
+       inode_unlock(wdir);
        return whiteout;
 }
 
-/* Caller must hold i_mutex on both workdir and dir */
 int ovl_cleanup_and_whiteout(struct ovl_fs *ofs, struct dentry *dir,
                             struct dentry *dentry)
 {
-       struct inode *wdir = ofs->workdir->d_inode;
        struct dentry *whiteout;
        int err;
        int flags = 0;
@@ -138,18 +137,22 @@ int ovl_cleanup_and_whiteout(struct ovl_fs *ofs, struct dentry *dir,
        if (d_is_dir(dentry))
                flags = RENAME_EXCHANGE;
 
-       err = ovl_do_rename(ofs, ofs->workdir, whiteout, dir, dentry, flags);
+       err = ovl_lock_rename_workdir(ofs->workdir, whiteout, dir, dentry);
+       if (!err) {
+               err = ovl_do_rename(ofs, ofs->workdir, whiteout, dir, dentry, flags);
+               unlock_rename(ofs->workdir, dir);
+       }
        if (err)
                goto kill_whiteout;
        if (flags)
-               ovl_cleanup(ofs, wdir, dentry);
+               ovl_cleanup_unlocked(ofs, ofs->workdir, dentry);
 
 out:
        dput(whiteout);
        return err;
 
 kill_whiteout:
-       ovl_cleanup(ofs, wdir, whiteout);
+       ovl_cleanup_unlocked(ofs, ofs->workdir, whiteout);
        goto out;
 }
 
@@ -783,16 +786,11 @@ static int ovl_remove_and_whiteout(struct dentry *dentry,
                goto out_dput_upper;
        }
 
-       err = ovl_lock_rename_workdir(workdir, NULL, upperdir, upper);
-       if (err)
-               goto out_dput_upper;
-
        err = ovl_cleanup_and_whiteout(ofs, upperdir, upper);
        if (!err)
                ovl_dir_modified(dentry->d_parent, true);
 
        d_drop(dentry);
-       unlock_rename(workdir, upperdir);
 out_dput_upper:
        dput(upper);
 out_dput:
index e2d0c314df6cf63c3e2f36b3715beba0d70d6d6c..ecbf39a49d572624185950677283a8ff604e1143 100644 (file)
@@ -1230,11 +1230,7 @@ int ovl_indexdir_cleanup(struct ovl_fs *ofs)
                         * Whiteout orphan index to block future open by
                         * handle after overlay nlink dropped to zero.
                         */
-                       err = ovl_parent_lock(indexdir, index);
-                       if (!err) {
-                               err = ovl_cleanup_and_whiteout(ofs, indexdir, index);
-                               ovl_parent_unlock(indexdir);
-                       }
+                       err = ovl_cleanup_and_whiteout(ofs, indexdir, index);
                } else {
                        /* Cleanup orphan index entries */
                        err = ovl_cleanup_unlocked(ofs, indexdir, index);
index 7a1ca9380601e78d7901531b3640b79e430928a4..a63fda9ef6fd8c6ba7bdd004107e789588ed57f5 100644 (file)
@@ -1119,12 +1119,8 @@ static void ovl_cleanup_index(struct dentry *dentry)
                index = NULL;
        } else if (ovl_index_all(dentry->d_sb)) {
                /* Whiteout orphan index to block future open by handle */
-               err = ovl_parent_lock(indexdir, index);
-               if (!err) {
-                       err = ovl_cleanup_and_whiteout(OVL_FS(dentry->d_sb),
-                                                      indexdir, index);
-                       ovl_parent_unlock(indexdir);
-               }
+               err = ovl_cleanup_and_whiteout(OVL_FS(dentry->d_sb),
+                                              indexdir, index);
        } else {
                /* Cleanup orphan index entries */
                err = ovl_cleanup_unlocked(ofs, indexdir, index);
@@ -1232,10 +1228,6 @@ int ovl_lock_rename_workdir(struct dentry *workdir, struct dentry *work,
 {
        struct dentry *trap;
 
-       /* Workdir should not be the same as upperdir */
-       if (workdir == upperdir)
-               goto err;
-
        /* Workdir should not be subdir of upperdir and vice versa */
        trap = lock_rename(workdir, upperdir);
        if (IS_ERR(trap))