Commit 49db9b1b authored by Jan Kara's avatar Jan Kara Committed by Al Viro

reiserfs: Avoid touching renamed directory if parent does not change

The VFS will not be locking moved directory if its parent does not
change. Change reiserfs rename code to avoid touching renamed directory
if its parent does not change as without locking that can corrupt the
filesystem.
Signed-off-by: default avatarJan Kara <jack@suse.cz>
Signed-off-by: default avatarAl Viro <viro@zeniv.linux.org.uk>
parent 4a0b33f7
...@@ -1324,8 +1324,8 @@ static int reiserfs_rename(struct mnt_idmap *idmap, ...@@ -1324,8 +1324,8 @@ static int reiserfs_rename(struct mnt_idmap *idmap,
struct inode *old_inode, *new_dentry_inode; struct inode *old_inode, *new_dentry_inode;
struct reiserfs_transaction_handle th; struct reiserfs_transaction_handle th;
int jbegin_count; int jbegin_count;
umode_t old_inode_mode;
unsigned long savelink = 1; unsigned long savelink = 1;
bool update_dir_parent = false;
if (flags & ~RENAME_NOREPLACE) if (flags & ~RENAME_NOREPLACE)
return -EINVAL; return -EINVAL;
...@@ -1375,8 +1375,7 @@ static int reiserfs_rename(struct mnt_idmap *idmap, ...@@ -1375,8 +1375,7 @@ static int reiserfs_rename(struct mnt_idmap *idmap,
return -ENOENT; return -ENOENT;
} }
old_inode_mode = old_inode->i_mode; if (S_ISDIR(old_inode->i_mode)) {
if (S_ISDIR(old_inode_mode)) {
/* /*
* make sure that directory being renamed has correct ".." * make sure that directory being renamed has correct ".."
* and that its new parent directory has not too many links * and that its new parent directory has not too many links
...@@ -1389,13 +1388,15 @@ static int reiserfs_rename(struct mnt_idmap *idmap, ...@@ -1389,13 +1388,15 @@ static int reiserfs_rename(struct mnt_idmap *idmap,
} }
} }
if (old_dir != new_dir) {
/* /*
* directory is renamed, its parent directory will be changed, * directory is renamed, its parent directory will be
* so find ".." entry * changed, so find ".." entry
*/ */
dot_dot_de.de_gen_number_bit_string = NULL; dot_dot_de.de_gen_number_bit_string = NULL;
retval = retval =
reiserfs_find_entry(old_inode, "..", 2, &dot_dot_entry_path, reiserfs_find_entry(old_inode, "..", 2,
&dot_dot_entry_path,
&dot_dot_de); &dot_dot_de);
pathrelse(&dot_dot_entry_path); pathrelse(&dot_dot_entry_path);
if (retval != NAME_FOUND) { if (retval != NAME_FOUND) {
...@@ -1408,6 +1409,8 @@ static int reiserfs_rename(struct mnt_idmap *idmap, ...@@ -1408,6 +1409,8 @@ static int reiserfs_rename(struct mnt_idmap *idmap,
reiserfs_write_unlock(old_dir->i_sb); reiserfs_write_unlock(old_dir->i_sb);
return -EIO; return -EIO;
} }
update_dir_parent = true;
}
} }
retval = journal_begin(&th, old_dir->i_sb, jbegin_count); retval = journal_begin(&th, old_dir->i_sb, jbegin_count);
...@@ -1486,7 +1489,7 @@ static int reiserfs_rename(struct mnt_idmap *idmap, ...@@ -1486,7 +1489,7 @@ static int reiserfs_rename(struct mnt_idmap *idmap,
reiserfs_prepare_for_journal(old_inode->i_sb, new_de.de_bh, 1); reiserfs_prepare_for_journal(old_inode->i_sb, new_de.de_bh, 1);
if (S_ISDIR(old_inode->i_mode)) { if (update_dir_parent) {
if ((retval = if ((retval =
search_by_entry_key(new_dir->i_sb, search_by_entry_key(new_dir->i_sb,
&dot_dot_de.de_entry_key, &dot_dot_de.de_entry_key,
...@@ -1534,14 +1537,14 @@ static int reiserfs_rename(struct mnt_idmap *idmap, ...@@ -1534,14 +1537,14 @@ static int reiserfs_rename(struct mnt_idmap *idmap,
new_de.de_bh); new_de.de_bh);
reiserfs_restore_prepared_buffer(old_inode->i_sb, reiserfs_restore_prepared_buffer(old_inode->i_sb,
old_de.de_bh); old_de.de_bh);
if (S_ISDIR(old_inode_mode)) if (update_dir_parent)
reiserfs_restore_prepared_buffer(old_inode-> reiserfs_restore_prepared_buffer(old_inode->
i_sb, i_sb,
dot_dot_de. dot_dot_de.
de_bh); de_bh);
continue; continue;
} }
if (S_ISDIR(old_inode_mode)) { if (update_dir_parent) {
if (item_moved(&dot_dot_ih, &dot_dot_entry_path) || if (item_moved(&dot_dot_ih, &dot_dot_entry_path) ||
!entry_points_to_object("..", 2, &dot_dot_de, !entry_points_to_object("..", 2, &dot_dot_de,
old_dir)) { old_dir)) {
...@@ -1559,7 +1562,7 @@ static int reiserfs_rename(struct mnt_idmap *idmap, ...@@ -1559,7 +1562,7 @@ static int reiserfs_rename(struct mnt_idmap *idmap,
} }
} }
RFALSE(S_ISDIR(old_inode_mode) && RFALSE(update_dir_parent &&
!buffer_journal_prepared(dot_dot_de.de_bh), ""); !buffer_journal_prepared(dot_dot_de.de_bh), "");
break; break;
...@@ -1592,11 +1595,12 @@ static int reiserfs_rename(struct mnt_idmap *idmap, ...@@ -1592,11 +1595,12 @@ static int reiserfs_rename(struct mnt_idmap *idmap,
savelink = new_dentry_inode->i_nlink; savelink = new_dentry_inode->i_nlink;
} }
if (S_ISDIR(old_inode_mode)) { if (update_dir_parent) {
/* adjust ".." of renamed directory */ /* adjust ".." of renamed directory */
set_ino_in_dir_entry(&dot_dot_de, INODE_PKEY(new_dir)); set_ino_in_dir_entry(&dot_dot_de, INODE_PKEY(new_dir));
journal_mark_dirty(&th, dot_dot_de.de_bh); journal_mark_dirty(&th, dot_dot_de.de_bh);
}
if (S_ISDIR(old_inode->i_mode)) {
/* /*
* there (in new_dir) was no directory, so it got new link * there (in new_dir) was no directory, so it got new link
* (".." of renamed directory) * (".." of renamed directory)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment