]> git.ipfire.org Git - thirdparty/kernel/stable.git/commitdiff
shmem: Fix shmem_rename2()
authorChuck Lever <chuck.lever@oracle.com>
Mon, 15 Apr 2024 15:20:56 +0000 (11:20 -0400)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Thu, 30 May 2024 07:49:00 +0000 (09:49 +0200)
[ Upstream commit ad191eb6d6942bb835a0b20b647f7c53c1d99ca4 ]

When renaming onto an existing directory entry, user space expects
the replacement entry to have the same directory offset as the
original one.

Link: https://gitlab.alpinelinux.org/alpine/aports/-/issues/15966
Fixes: a2e459555c5f ("shmem: stable directory offsets")
Signed-off-by: Chuck Lever <chuck.lever@oracle.com>
Link: https://lore.kernel.org/r/20240415152057.4605-4-cel@kernel.org
Signed-off-by: Christian Brauner <brauner@kernel.org>
Signed-off-by: Sasha Levin <sashal@kernel.org>
fs/libfs.c

index bb6731fa63059d10c50c1fb4de8f9aba9ec38334..ef700d39f0f3591ed745cdc01b19ab6679fdd082 100644 (file)
@@ -365,6 +365,9 @@ int simple_offset_empty(struct dentry *dentry)
  *
  * Caller provides appropriate serialization.
  *
+ * User space expects the directory offset value of the replaced
+ * (new) directory entry to be unchanged after a rename.
+ *
  * Returns zero on success, a negative errno value on failure.
  */
 int simple_offset_rename(struct inode *old_dir, struct dentry *old_dentry,
@@ -372,8 +375,14 @@ int simple_offset_rename(struct inode *old_dir, struct dentry *old_dentry,
 {
        struct offset_ctx *old_ctx = old_dir->i_op->get_offset_ctx(old_dir);
        struct offset_ctx *new_ctx = new_dir->i_op->get_offset_ctx(new_dir);
+       long new_offset = dentry2offset(new_dentry);
 
        simple_offset_remove(old_ctx, old_dentry);
+
+       if (new_offset) {
+               offset_set(new_dentry, 0);
+               return simple_offset_replace(new_ctx, old_dentry, new_offset);
+       }
        return simple_offset_add(new_ctx, old_dentry);
 }