aboutsummaryrefslogtreecommitdiffstats
path: root/fs/namei.c
diff options
context:
space:
mode:
authorJ. Bruce Fields <bfields@redhat.com>2012-03-05 11:40:41 -0500
committerAl Viro <viro@zeniv.linux.org.uk>2013-11-09 00:16:40 -0500
commit6cedba8962f440c72447f811d0d530a8a9dc637a (patch)
tree740e74df27113d34d06da011b3c5561d19eb3349 /fs/namei.c
parent40bd22c9f8617ddd5da06044c81f72a2cf700791 (diff)
vfs: take i_mutex on renamed file
A read delegation is used by NFSv4 as a guarantee that a client can perform local read opens without informing the server. The open operation takes the last component of the pathname as an argument, thus is also a lookup operation, and giving the client the above guarantee means informing the client before we allow anything that would change the set of names pointing to the inode. Therefore, we need to break delegations on rename, link, and unlink. We also need to prevent new delegations from being acquired while one of these operations is in progress. We could add some completely new locking for that purpose, but it's simpler to use the i_mutex, since that's already taken by all the operations we care about. The single exception is rename. So, modify rename to take the i_mutex on the file that is being renamed. Also fix up lockdep and Documentation/filesystems/directory-locking to reflect the change. Acked-by: Jeff Layton <jlayton@redhat.com> Signed-off-by: J. Bruce Fields <bfields@redhat.com> Signed-off-by: Al Viro <viro@zeniv.linux.org.uk>
Diffstat (limited to 'fs/namei.c')
-rw-r--r--fs/namei.c10
1 files changed, 5 insertions, 5 deletions
diff --git a/fs/namei.c b/fs/namei.c
index 2a5a7aa9f43f..88cec0330bf7 100644
--- a/fs/namei.c
+++ b/fs/namei.c
@@ -3918,7 +3918,8 @@ SYSCALL_DEFINE2(link, const char __user *, oldname, const char __user *, newname
3918 * That's where 4.4 screws up. Current fix: serialization on 3918 * That's where 4.4 screws up. Current fix: serialization on
3919 * sb->s_vfs_rename_mutex. We might be more accurate, but that's another 3919 * sb->s_vfs_rename_mutex. We might be more accurate, but that's another
3920 * story. 3920 * story.
3921 * c) we have to lock _three_ objects - parents and victim (if it exists). 3921 * c) we have to lock _four_ objects - parents and victim (if it exists),
3922 * and source (if it is not a directory).
3922 * And that - after we got ->i_mutex on parents (until then we don't know 3923 * And that - after we got ->i_mutex on parents (until then we don't know
3923 * whether the target exists). Solution: try to be smart with locking 3924 * whether the target exists). Solution: try to be smart with locking
3924 * order for inodes. We rely on the fact that tree topology may change 3925 * order for inodes. We rely on the fact that tree topology may change
@@ -3994,6 +3995,7 @@ static int vfs_rename_other(struct inode *old_dir, struct dentry *old_dentry,
3994 struct inode *new_dir, struct dentry *new_dentry) 3995 struct inode *new_dir, struct dentry *new_dentry)
3995{ 3996{
3996 struct inode *target = new_dentry->d_inode; 3997 struct inode *target = new_dentry->d_inode;
3998 struct inode *source = old_dentry->d_inode;
3997 int error; 3999 int error;
3998 4000
3999 error = security_inode_rename(old_dir, old_dentry, new_dir, new_dentry); 4001 error = security_inode_rename(old_dir, old_dentry, new_dir, new_dentry);
@@ -4001,8 +4003,7 @@ static int vfs_rename_other(struct inode *old_dir, struct dentry *old_dentry,
4001 return error; 4003 return error;
4002 4004
4003 dget(new_dentry); 4005 dget(new_dentry);
4004 if (target) 4006 lock_two_nondirectories(source, target);
4005 mutex_lock(&target->i_mutex);
4006 4007
4007 error = -EBUSY; 4008 error = -EBUSY;
4008 if (d_mountpoint(old_dentry)||d_mountpoint(new_dentry)) 4009 if (d_mountpoint(old_dentry)||d_mountpoint(new_dentry))
@@ -4017,8 +4018,7 @@ static int vfs_rename_other(struct inode *old_dir, struct dentry *old_dentry,
4017 if (!(old_dir->i_sb->s_type->fs_flags & FS_RENAME_DOES_D_MOVE)) 4018 if (!(old_dir->i_sb->s_type->fs_flags & FS_RENAME_DOES_D_MOVE))
4018 d_move(old_dentry, new_dentry); 4019 d_move(old_dentry, new_dentry);
4019out: 4020out:
4020 if (target) 4021 unlock_two_nondirectories(source, target);
4021 mutex_unlock(&target->i_mutex);
4022 dput(new_dentry); 4022 dput(new_dentry);
4023 return error; 4023 return error;
4024} 4024}