Commit 4ef51e8b authored by Al Viro's avatar Al Viro

Merge branch 'for-linus' into for-next

parents b853a161 e4f95517
...@@ -1239,13 +1239,13 @@ static void d_walk(struct dentry *parent, void *data, ...@@ -1239,13 +1239,13 @@ static void d_walk(struct dentry *parent, void *data,
/* might go back up the wrong parent if we have had a rename. */ /* might go back up the wrong parent if we have had a rename. */
if (need_seqretry(&rename_lock, seq)) if (need_seqretry(&rename_lock, seq))
goto rename_retry; goto rename_retry;
next = child->d_child.next; /* go into the first sibling still alive */
while (unlikely(child->d_flags & DCACHE_DENTRY_KILLED)) { do {
next = child->d_child.next;
if (next == &this_parent->d_subdirs) if (next == &this_parent->d_subdirs)
goto ascend; goto ascend;
child = list_entry(next, struct dentry, d_child); child = list_entry(next, struct dentry, d_child);
next = next->next; } while (unlikely(child->d_flags & DCACHE_DENTRY_KILLED));
}
rcu_read_unlock(); rcu_read_unlock();
goto resume; goto resume;
} }
......
...@@ -1145,6 +1145,8 @@ static int ncp_rename(struct inode *old_dir, struct dentry *old_dentry, ...@@ -1145,6 +1145,8 @@ static int ncp_rename(struct inode *old_dir, struct dentry *old_dentry,
case 0x00: case 0x00:
ncp_dbg(1, "renamed %pd -> %pd\n", ncp_dbg(1, "renamed %pd -> %pd\n",
old_dentry, new_dentry); old_dentry, new_dentry);
ncp_d_prune(old_dentry);
ncp_d_prune(new_dentry);
break; break;
case 0x9E: case 0x9E:
error = -ENAMETOOLONG; error = -ENAMETOOLONG;
......
...@@ -51,8 +51,8 @@ void ufs_free_fragments(struct inode *inode, u64 fragment, unsigned count) ...@@ -51,8 +51,8 @@ void ufs_free_fragments(struct inode *inode, u64 fragment, unsigned count)
if (ufs_fragnum(fragment) + count > uspi->s_fpg) if (ufs_fragnum(fragment) + count > uspi->s_fpg)
ufs_error (sb, "ufs_free_fragments", "internal error"); ufs_error (sb, "ufs_free_fragments", "internal error");
lock_ufs(sb); mutex_lock(&UFS_SB(sb)->s_lock);
cgno = ufs_dtog(uspi, fragment); cgno = ufs_dtog(uspi, fragment);
bit = ufs_dtogd(uspi, fragment); bit = ufs_dtogd(uspi, fragment);
...@@ -115,13 +115,13 @@ void ufs_free_fragments(struct inode *inode, u64 fragment, unsigned count) ...@@ -115,13 +115,13 @@ void ufs_free_fragments(struct inode *inode, u64 fragment, unsigned count)
if (sb->s_flags & MS_SYNCHRONOUS) if (sb->s_flags & MS_SYNCHRONOUS)
ubh_sync_block(UCPI_UBH(ucpi)); ubh_sync_block(UCPI_UBH(ucpi));
ufs_mark_sb_dirty(sb); ufs_mark_sb_dirty(sb);
unlock_ufs(sb); mutex_unlock(&UFS_SB(sb)->s_lock);
UFSD("EXIT\n"); UFSD("EXIT\n");
return; return;
failed: failed:
unlock_ufs(sb); mutex_unlock(&UFS_SB(sb)->s_lock);
UFSD("EXIT (FAILED)\n"); UFSD("EXIT (FAILED)\n");
return; return;
} }
...@@ -151,7 +151,7 @@ void ufs_free_blocks(struct inode *inode, u64 fragment, unsigned count) ...@@ -151,7 +151,7 @@ void ufs_free_blocks(struct inode *inode, u64 fragment, unsigned count)
goto failed; goto failed;
} }
lock_ufs(sb); mutex_lock(&UFS_SB(sb)->s_lock);
do_more: do_more:
overflow = 0; overflow = 0;
...@@ -211,12 +211,12 @@ void ufs_free_blocks(struct inode *inode, u64 fragment, unsigned count) ...@@ -211,12 +211,12 @@ void ufs_free_blocks(struct inode *inode, u64 fragment, unsigned count)
} }
ufs_mark_sb_dirty(sb); ufs_mark_sb_dirty(sb);
unlock_ufs(sb); mutex_unlock(&UFS_SB(sb)->s_lock);
UFSD("EXIT\n"); UFSD("EXIT\n");
return; return;
failed_unlock: failed_unlock:
unlock_ufs(sb); mutex_unlock(&UFS_SB(sb)->s_lock);
failed: failed:
UFSD("EXIT (FAILED)\n"); UFSD("EXIT (FAILED)\n");
return; return;
...@@ -357,7 +357,7 @@ u64 ufs_new_fragments(struct inode *inode, void *p, u64 fragment, ...@@ -357,7 +357,7 @@ u64 ufs_new_fragments(struct inode *inode, void *p, u64 fragment,
usb1 = ubh_get_usb_first(uspi); usb1 = ubh_get_usb_first(uspi);
*err = -ENOSPC; *err = -ENOSPC;
lock_ufs(sb); mutex_lock(&UFS_SB(sb)->s_lock);
tmp = ufs_data_ptr_to_cpu(sb, p); tmp = ufs_data_ptr_to_cpu(sb, p);
if (count + ufs_fragnum(fragment) > uspi->s_fpb) { if (count + ufs_fragnum(fragment) > uspi->s_fpb) {
...@@ -378,19 +378,19 @@ u64 ufs_new_fragments(struct inode *inode, void *p, u64 fragment, ...@@ -378,19 +378,19 @@ u64 ufs_new_fragments(struct inode *inode, void *p, u64 fragment,
"fragment %llu, tmp %llu\n", "fragment %llu, tmp %llu\n",
(unsigned long long)fragment, (unsigned long long)fragment,
(unsigned long long)tmp); (unsigned long long)tmp);
unlock_ufs(sb); mutex_unlock(&UFS_SB(sb)->s_lock);
return INVBLOCK; return INVBLOCK;
} }
if (fragment < UFS_I(inode)->i_lastfrag) { if (fragment < UFS_I(inode)->i_lastfrag) {
UFSD("EXIT (ALREADY ALLOCATED)\n"); UFSD("EXIT (ALREADY ALLOCATED)\n");
unlock_ufs(sb); mutex_unlock(&UFS_SB(sb)->s_lock);
return 0; return 0;
} }
} }
else { else {
if (tmp) { if (tmp) {
UFSD("EXIT (ALREADY ALLOCATED)\n"); UFSD("EXIT (ALREADY ALLOCATED)\n");
unlock_ufs(sb); mutex_unlock(&UFS_SB(sb)->s_lock);
return 0; return 0;
} }
} }
...@@ -399,7 +399,7 @@ u64 ufs_new_fragments(struct inode *inode, void *p, u64 fragment, ...@@ -399,7 +399,7 @@ u64 ufs_new_fragments(struct inode *inode, void *p, u64 fragment,
* There is not enough space for user on the device * There is not enough space for user on the device
*/ */
if (!capable(CAP_SYS_RESOURCE) && ufs_freespace(uspi, UFS_MINFREE) <= 0) { if (!capable(CAP_SYS_RESOURCE) && ufs_freespace(uspi, UFS_MINFREE) <= 0) {
unlock_ufs(sb); mutex_unlock(&UFS_SB(sb)->s_lock);
UFSD("EXIT (FAILED)\n"); UFSD("EXIT (FAILED)\n");
return 0; return 0;
} }
...@@ -424,7 +424,7 @@ u64 ufs_new_fragments(struct inode *inode, void *p, u64 fragment, ...@@ -424,7 +424,7 @@ u64 ufs_new_fragments(struct inode *inode, void *p, u64 fragment,
ufs_clear_frags(inode, result + oldcount, ufs_clear_frags(inode, result + oldcount,
newcount - oldcount, locked_page != NULL); newcount - oldcount, locked_page != NULL);
} }
unlock_ufs(sb); mutex_unlock(&UFS_SB(sb)->s_lock);
UFSD("EXIT, result %llu\n", (unsigned long long)result); UFSD("EXIT, result %llu\n", (unsigned long long)result);
return result; return result;
} }
...@@ -439,7 +439,7 @@ u64 ufs_new_fragments(struct inode *inode, void *p, u64 fragment, ...@@ -439,7 +439,7 @@ u64 ufs_new_fragments(struct inode *inode, void *p, u64 fragment,
fragment + count); fragment + count);
ufs_clear_frags(inode, result + oldcount, newcount - oldcount, ufs_clear_frags(inode, result + oldcount, newcount - oldcount,
locked_page != NULL); locked_page != NULL);
unlock_ufs(sb); mutex_unlock(&UFS_SB(sb)->s_lock);
UFSD("EXIT, result %llu\n", (unsigned long long)result); UFSD("EXIT, result %llu\n", (unsigned long long)result);
return result; return result;
} }
...@@ -477,7 +477,7 @@ u64 ufs_new_fragments(struct inode *inode, void *p, u64 fragment, ...@@ -477,7 +477,7 @@ u64 ufs_new_fragments(struct inode *inode, void *p, u64 fragment,
*err = 0; *err = 0;
UFS_I(inode)->i_lastfrag = max(UFS_I(inode)->i_lastfrag, UFS_I(inode)->i_lastfrag = max(UFS_I(inode)->i_lastfrag,
fragment + count); fragment + count);
unlock_ufs(sb); mutex_unlock(&UFS_SB(sb)->s_lock);
if (newcount < request) if (newcount < request)
ufs_free_fragments (inode, result + newcount, request - newcount); ufs_free_fragments (inode, result + newcount, request - newcount);
ufs_free_fragments (inode, tmp, oldcount); ufs_free_fragments (inode, tmp, oldcount);
...@@ -485,7 +485,7 @@ u64 ufs_new_fragments(struct inode *inode, void *p, u64 fragment, ...@@ -485,7 +485,7 @@ u64 ufs_new_fragments(struct inode *inode, void *p, u64 fragment,
return result; return result;
} }
unlock_ufs(sb); mutex_unlock(&UFS_SB(sb)->s_lock);
UFSD("EXIT (FAILED)\n"); UFSD("EXIT (FAILED)\n");
return 0; return 0;
} }
......
...@@ -87,7 +87,8 @@ ino_t ufs_inode_by_name(struct inode *dir, const struct qstr *qstr) ...@@ -87,7 +87,8 @@ ino_t ufs_inode_by_name(struct inode *dir, const struct qstr *qstr)
/* Releases the page */ /* Releases the page */
void ufs_set_link(struct inode *dir, struct ufs_dir_entry *de, void ufs_set_link(struct inode *dir, struct ufs_dir_entry *de,
struct page *page, struct inode *inode) struct page *page, struct inode *inode,
bool update_times)
{ {
loff_t pos = page_offset(page) + loff_t pos = page_offset(page) +
(char *) de - (char *) page_address(page); (char *) de - (char *) page_address(page);
...@@ -103,7 +104,8 @@ void ufs_set_link(struct inode *dir, struct ufs_dir_entry *de, ...@@ -103,7 +104,8 @@ void ufs_set_link(struct inode *dir, struct ufs_dir_entry *de,
err = ufs_commit_chunk(page, pos, len); err = ufs_commit_chunk(page, pos, len);
ufs_put_page(page); ufs_put_page(page);
dir->i_mtime = dir->i_ctime = CURRENT_TIME_SEC; if (update_times)
dir->i_mtime = dir->i_ctime = CURRENT_TIME_SEC;
mark_inode_dirty(dir); mark_inode_dirty(dir);
} }
......
...@@ -69,11 +69,11 @@ void ufs_free_inode (struct inode * inode) ...@@ -69,11 +69,11 @@ void ufs_free_inode (struct inode * inode)
ino = inode->i_ino; ino = inode->i_ino;
lock_ufs(sb); mutex_lock(&UFS_SB(sb)->s_lock);
if (!((ino > 1) && (ino < (uspi->s_ncg * uspi->s_ipg )))) { if (!((ino > 1) && (ino < (uspi->s_ncg * uspi->s_ipg )))) {
ufs_warning(sb, "ufs_free_inode", "reserved inode or nonexistent inode %u\n", ino); ufs_warning(sb, "ufs_free_inode", "reserved inode or nonexistent inode %u\n", ino);
unlock_ufs(sb); mutex_unlock(&UFS_SB(sb)->s_lock);
return; return;
} }
...@@ -81,7 +81,7 @@ void ufs_free_inode (struct inode * inode) ...@@ -81,7 +81,7 @@ void ufs_free_inode (struct inode * inode)
bit = ufs_inotocgoff (ino); bit = ufs_inotocgoff (ino);
ucpi = ufs_load_cylinder (sb, cg); ucpi = ufs_load_cylinder (sb, cg);
if (!ucpi) { if (!ucpi) {
unlock_ufs(sb); mutex_unlock(&UFS_SB(sb)->s_lock);
return; return;
} }
ucg = ubh_get_ucg(UCPI_UBH(ucpi)); ucg = ubh_get_ucg(UCPI_UBH(ucpi));
...@@ -115,7 +115,7 @@ void ufs_free_inode (struct inode * inode) ...@@ -115,7 +115,7 @@ void ufs_free_inode (struct inode * inode)
ubh_sync_block(UCPI_UBH(ucpi)); ubh_sync_block(UCPI_UBH(ucpi));
ufs_mark_sb_dirty(sb); ufs_mark_sb_dirty(sb);
unlock_ufs(sb); mutex_unlock(&UFS_SB(sb)->s_lock);
UFSD("EXIT\n"); UFSD("EXIT\n");
} }
...@@ -193,7 +193,7 @@ struct inode *ufs_new_inode(struct inode *dir, umode_t mode) ...@@ -193,7 +193,7 @@ struct inode *ufs_new_inode(struct inode *dir, umode_t mode)
sbi = UFS_SB(sb); sbi = UFS_SB(sb);
uspi = sbi->s_uspi; uspi = sbi->s_uspi;
lock_ufs(sb); mutex_lock(&sbi->s_lock);
/* /*
* Try to place the inode in its parent directory * Try to place the inode in its parent directory
...@@ -331,21 +331,21 @@ struct inode *ufs_new_inode(struct inode *dir, umode_t mode) ...@@ -331,21 +331,21 @@ struct inode *ufs_new_inode(struct inode *dir, umode_t mode)
sync_dirty_buffer(bh); sync_dirty_buffer(bh);
brelse(bh); brelse(bh);
} }
unlock_ufs(sb); mutex_unlock(&sbi->s_lock);
UFSD("allocating inode %lu\n", inode->i_ino); UFSD("allocating inode %lu\n", inode->i_ino);
UFSD("EXIT\n"); UFSD("EXIT\n");
return inode; return inode;
fail_remove_inode: fail_remove_inode:
unlock_ufs(sb); mutex_unlock(&sbi->s_lock);
clear_nlink(inode); clear_nlink(inode);
unlock_new_inode(inode); unlock_new_inode(inode);
iput(inode); iput(inode);
UFSD("EXIT (FAILED): err %d\n", err); UFSD("EXIT (FAILED): err %d\n", err);
return ERR_PTR(err); return ERR_PTR(err);
failed: failed:
unlock_ufs(sb); mutex_unlock(&sbi->s_lock);
make_bad_inode(inode); make_bad_inode(inode);
iput (inode); iput (inode);
UFSD("EXIT (FAILED): err %d\n", err); UFSD("EXIT (FAILED): err %d\n", err);
......
...@@ -903,6 +903,9 @@ void ufs_evict_inode(struct inode * inode) ...@@ -903,6 +903,9 @@ void ufs_evict_inode(struct inode * inode)
invalidate_inode_buffers(inode); invalidate_inode_buffers(inode);
clear_inode(inode); clear_inode(inode);
if (want_delete) if (want_delete) {
lock_ufs(inode->i_sb);
ufs_free_inode(inode); ufs_free_inode(inode);
unlock_ufs(inode->i_sb);
}
} }
...@@ -56,11 +56,9 @@ static struct dentry *ufs_lookup(struct inode * dir, struct dentry *dentry, unsi ...@@ -56,11 +56,9 @@ static struct dentry *ufs_lookup(struct inode * dir, struct dentry *dentry, unsi
if (dentry->d_name.len > UFS_MAXNAMLEN) if (dentry->d_name.len > UFS_MAXNAMLEN)
return ERR_PTR(-ENAMETOOLONG); return ERR_PTR(-ENAMETOOLONG);
lock_ufs(dir->i_sb);
ino = ufs_inode_by_name(dir, &dentry->d_name); ino = ufs_inode_by_name(dir, &dentry->d_name);
if (ino) if (ino)
inode = ufs_iget(dir->i_sb, ino); inode = ufs_iget(dir->i_sb, ino);
unlock_ufs(dir->i_sb);
return d_splice_alias(inode, dentry); return d_splice_alias(inode, dentry);
} }
...@@ -76,24 +74,16 @@ static int ufs_create (struct inode * dir, struct dentry * dentry, umode_t mode, ...@@ -76,24 +74,16 @@ static int ufs_create (struct inode * dir, struct dentry * dentry, umode_t mode,
bool excl) bool excl)
{ {
struct inode *inode; struct inode *inode;
int err;
UFSD("BEGIN\n");
inode = ufs_new_inode(dir, mode); inode = ufs_new_inode(dir, mode);
err = PTR_ERR(inode); if (IS_ERR(inode))
return PTR_ERR(inode);
if (!IS_ERR(inode)) { inode->i_op = &ufs_file_inode_operations;
inode->i_op = &ufs_file_inode_operations; inode->i_fop = &ufs_file_operations;
inode->i_fop = &ufs_file_operations; inode->i_mapping->a_ops = &ufs_aops;
inode->i_mapping->a_ops = &ufs_aops; mark_inode_dirty(inode);
mark_inode_dirty(inode); return ufs_add_nondir(dentry, inode);
lock_ufs(dir->i_sb);
err = ufs_add_nondir(dentry, inode);
unlock_ufs(dir->i_sb);
}
UFSD("END: err=%d\n", err);
return err;
} }
static int ufs_mknod(struct inode *dir, struct dentry *dentry, umode_t mode, dev_t rdev) static int ufs_mknod(struct inode *dir, struct dentry *dentry, umode_t mode, dev_t rdev)
...@@ -110,9 +100,7 @@ static int ufs_mknod(struct inode *dir, struct dentry *dentry, umode_t mode, dev ...@@ -110,9 +100,7 @@ static int ufs_mknod(struct inode *dir, struct dentry *dentry, umode_t mode, dev
init_special_inode(inode, mode, rdev); init_special_inode(inode, mode, rdev);
ufs_set_inode_dev(inode->i_sb, UFS_I(inode), rdev); ufs_set_inode_dev(inode->i_sb, UFS_I(inode), rdev);
mark_inode_dirty(inode); mark_inode_dirty(inode);
lock_ufs(dir->i_sb);
err = ufs_add_nondir(dentry, inode); err = ufs_add_nondir(dentry, inode);
unlock_ufs(dir->i_sb);
} }
return err; return err;
} }
...@@ -121,19 +109,18 @@ static int ufs_symlink (struct inode * dir, struct dentry * dentry, ...@@ -121,19 +109,18 @@ static int ufs_symlink (struct inode * dir, struct dentry * dentry,
const char * symname) const char * symname)
{ {
struct super_block * sb = dir->i_sb; struct super_block * sb = dir->i_sb;
int err = -ENAMETOOLONG; int err;
unsigned l = strlen(symname)+1; unsigned l = strlen(symname)+1;
struct inode * inode; struct inode * inode;
if (l > sb->s_blocksize) if (l > sb->s_blocksize)
goto out_notlocked; return -ENAMETOOLONG;
inode = ufs_new_inode(dir, S_IFLNK | S_IRWXUGO); inode = ufs_new_inode(dir, S_IFLNK | S_IRWXUGO);
err = PTR_ERR(inode); err = PTR_ERR(inode);
if (IS_ERR(inode)) if (IS_ERR(inode))
goto out_notlocked; return err;
lock_ufs(dir->i_sb);
if (l > UFS_SB(sb)->s_uspi->s_maxsymlinklen) { if (l > UFS_SB(sb)->s_uspi->s_maxsymlinklen) {
/* slow symlink */ /* slow symlink */
inode->i_op = &ufs_symlink_inode_operations; inode->i_op = &ufs_symlink_inode_operations;
...@@ -150,17 +137,13 @@ static int ufs_symlink (struct inode * dir, struct dentry * dentry, ...@@ -150,17 +137,13 @@ static int ufs_symlink (struct inode * dir, struct dentry * dentry,
} }
mark_inode_dirty(inode); mark_inode_dirty(inode);
err = ufs_add_nondir(dentry, inode); return ufs_add_nondir(dentry, inode);
out:
unlock_ufs(dir->i_sb);
out_notlocked:
return err;
out_fail: out_fail:
inode_dec_link_count(inode); inode_dec_link_count(inode);
unlock_new_inode(inode); unlock_new_inode(inode);
iput(inode); iput(inode);
goto out; return err;
} }
static int ufs_link (struct dentry * old_dentry, struct inode * dir, static int ufs_link (struct dentry * old_dentry, struct inode * dir,
...@@ -169,14 +152,16 @@ static int ufs_link (struct dentry * old_dentry, struct inode * dir, ...@@ -169,14 +152,16 @@ static int ufs_link (struct dentry * old_dentry, struct inode * dir,
struct inode *inode = d_inode(old_dentry); struct inode *inode = d_inode(old_dentry);
int error; int error;
lock_ufs(dir->i_sb);
inode->i_ctime = CURRENT_TIME_SEC; inode->i_ctime = CURRENT_TIME_SEC;
inode_inc_link_count(inode); inode_inc_link_count(inode);
ihold(inode); ihold(inode);
error = ufs_add_nondir(dentry, inode); error = ufs_add_link(dentry, inode);
unlock_ufs(dir->i_sb); if (error) {
inode_dec_link_count(inode);
iput(inode);
} else
d_instantiate(dentry, inode);
return error; return error;
} }
...@@ -185,9 +170,12 @@ static int ufs_mkdir(struct inode * dir, struct dentry * dentry, umode_t mode) ...@@ -185,9 +170,12 @@ static int ufs_mkdir(struct inode * dir, struct dentry * dentry, umode_t mode)
struct inode * inode; struct inode * inode;
int err; int err;
inode_inc_link_count(dir);
inode = ufs_new_inode(dir, S_IFDIR|mode); inode = ufs_new_inode(dir, S_IFDIR|mode);
err = PTR_ERR(inode);
if (IS_ERR(inode)) if (IS_ERR(inode))
return PTR_ERR(inode); goto out_dir;
inode->i_op = &ufs_dir_inode_operations; inode->i_op = &ufs_dir_inode_operations;
inode->i_fop = &ufs_dir_operations; inode->i_fop = &ufs_dir_operations;
...@@ -195,9 +183,6 @@ static int ufs_mkdir(struct inode * dir, struct dentry * dentry, umode_t mode) ...@@ -195,9 +183,6 @@ static int ufs_mkdir(struct inode * dir, struct dentry * dentry, umode_t mode)
inode_inc_link_count(inode); inode_inc_link_count(inode);
lock_ufs(dir->i_sb);
inode_inc_link_count(dir);
err = ufs_make_empty(inode, dir); err = ufs_make_empty(inode, dir);
if (err) if (err)
goto out_fail; goto out_fail;
...@@ -205,20 +190,19 @@ static int ufs_mkdir(struct inode * dir, struct dentry * dentry, umode_t mode) ...@@ -205,20 +190,19 @@ static int ufs_mkdir(struct inode * dir, struct dentry * dentry, umode_t mode)
err = ufs_add_link(dentry, inode); err = ufs_add_link(dentry, inode);
if (err) if (err)
goto out_fail; goto out_fail;
unlock_ufs(dir->i_sb);
unlock_new_inode(inode);
d_instantiate(dentry, inode); d_instantiate(dentry, inode);
out: return 0;
return err;
out_fail: out_fail:
inode_dec_link_count(inode); inode_dec_link_count(inode);
inode_dec_link_count(inode); inode_dec_link_count(inode);
unlock_new_inode(inode); unlock_new_inode(inode);
iput (inode); iput (inode);
out_dir:
inode_dec_link_count(dir); inode_dec_link_count(dir);
unlock_ufs(dir->i_sb); return err;
goto out;
} }
static int ufs_unlink(struct inode *dir, struct dentry *dentry) static int ufs_unlink(struct inode *dir, struct dentry *dentry)
...@@ -248,7 +232,6 @@ static int ufs_rmdir (struct inode * dir, struct dentry *dentry) ...@@ -248,7 +232,6 @@ static int ufs_rmdir (struct inode * dir, struct dentry *dentry)
struct inode * inode = d_inode(dentry); struct inode * inode = d_inode(dentry);
int err= -ENOTEMPTY; int err= -ENOTEMPTY;
lock_ufs(dir->i_sb);
if (ufs_empty_dir (inode)) { if (ufs_empty_dir (inode)) {
err = ufs_unlink(dir, dentry); err = ufs_unlink(dir, dentry);
if (!err) { if (!err) {
...@@ -257,7 +240,6 @@ static int ufs_rmdir (struct inode * dir, struct dentry *dentry) ...@@ -257,7 +240,6 @@ static int ufs_rmdir (struct inode * dir, struct dentry *dentry)
inode_dec_link_count(dir); inode_dec_link_count(dir);
} }
} }
unlock_ufs(dir->i_sb);
return err; return err;
} }
...@@ -295,7 +277,7 @@ static int ufs_rename(struct inode *old_dir, struct dentry *old_dentry, ...@@ -295,7 +277,7 @@ static int ufs_rename(struct inode *old_dir, struct dentry *old_dentry,
new_de = ufs_find_entry(new_dir, &new_dentry->d_name, &new_page); new_de = ufs_find_entry(new_dir, &new_dentry->d_name, &new_page);
if (!new_de) if (!new_de)
goto out_dir; goto out_dir;
ufs_set_link(new_dir, new_de, new_page, old_inode); ufs_set_link(new_dir, new_de, new_page, old_inode, 1);
new_inode->i_ctime = CURRENT_TIME_SEC; new_inode->i_ctime = CURRENT_TIME_SEC;
if (dir_de) if (dir_de)
drop_nlink(new_inode); drop_nlink(new_inode);
...@@ -318,7 +300,12 @@ static int ufs_rename(struct inode *old_dir, struct dentry *old_dentry, ...@@ -318,7 +300,12 @@ static int ufs_rename(struct inode *old_dir, struct dentry *old_dentry,
mark_inode_dirty(old_inode); mark_inode_dirty(old_inode);
if (dir_de) { if (dir_de) {
ufs_set_link(old_inode, dir_de, dir_page, new_dir); if (old_dir != new_dir)
ufs_set_link(old_inode, dir_de, dir_page, new_dir, 0);
else {
kunmap(dir_page);
page_cache_release(dir_page);
}
inode_dec_link_count(old_dir); inode_dec_link_count(old_dir);
} }
return 0; return 0;
......
...@@ -694,6 +694,7 @@ static int ufs_sync_fs(struct super_block *sb, int wait) ...@@ -694,6 +694,7 @@ static int ufs_sync_fs(struct super_block *sb, int wait)
unsigned flags; unsigned flags;
lock_ufs(sb); lock_ufs(sb);
mutex_lock(&UFS_SB(sb)->s_lock);
UFSD("ENTER\n"); UFSD("ENTER\n");
...@@ -711,6 +712,7 @@ static int ufs_sync_fs(struct super_block *sb, int wait) ...@@ -711,6 +712,7 @@ static int ufs_sync_fs(struct super_block *sb, int wait)
ufs_put_cstotal(sb); ufs_put_cstotal(sb);
UFSD("EXIT\n"); UFSD("EXIT\n");
mutex_unlock(&UFS_SB(sb)->s_lock);
unlock_ufs(sb); unlock_ufs(sb);
return 0; return 0;
...@@ -799,6 +801,7 @@ static int ufs_fill_super(struct super_block *sb, void *data, int silent) ...@@ -799,6 +801,7 @@ static int ufs_fill_super(struct super_block *sb, void *data, int silent)
UFSD("flag %u\n", (int)(sb->s_flags & MS_RDONLY)); UFSD("flag %u\n", (int)(sb->s_flags & MS_RDONLY));
mutex_init(&sbi->mutex); mutex_init(&sbi->mutex);
mutex_init(&sbi->s_lock);
spin_lock_init(&sbi->work_lock); spin_lock_init(&sbi->work_lock);
INIT_DELAYED_WORK(&sbi->sync_work, delayed_sync_fs); INIT_DELAYED_WORK(&sbi->sync_work, delayed_sync_fs);
/* /*
...@@ -1277,6 +1280,7 @@ static int ufs_remount (struct super_block *sb, int *mount_flags, char *data) ...@@ -1277,6 +1280,7 @@ static int ufs_remount (struct super_block *sb, int *mount_flags, char *data)
sync_filesystem(sb); sync_filesystem(sb);
lock_ufs(sb); lock_ufs(sb);
mutex_lock(&UFS_SB(sb)->s_lock);
uspi = UFS_SB(sb)->s_uspi; uspi = UFS_SB(sb)->s_uspi;
flags = UFS_SB(sb)->s_flags; flags = UFS_SB(sb)->s_flags;
usb1 = ubh_get_usb_first(uspi); usb1 = ubh_get_usb_first(uspi);
...@@ -1290,6 +1294,7 @@ static int ufs_remount (struct super_block *sb, int *mount_flags, char *data) ...@@ -1290,6 +1294,7 @@ static int ufs_remount (struct super_block *sb, int *mount_flags, char *data)
new_mount_opt = 0; new_mount_opt = 0;
ufs_set_opt (new_mount_opt, ONERROR_LOCK); ufs_set_opt (new_mount_opt, ONERROR_LOCK);
if (!ufs_parse_options (data, &new_mount_opt)) { if (!ufs_parse_options (data, &new_mount_opt)) {
mutex_unlock(&UFS_SB(sb)->s_lock);
unlock_ufs(sb); unlock_ufs(sb);
return -EINVAL; return -EINVAL;
} }
...@@ -1297,12 +1302,14 @@ static int ufs_remount (struct super_block *sb, int *mount_flags, char *data) ...@@ -1297,12 +1302,14 @@ static int ufs_remount (struct super_block *sb, int *mount_flags, char *data)
new_mount_opt |= ufstype; new_mount_opt |= ufstype;
} else if ((new_mount_opt & UFS_MOUNT_UFSTYPE) != ufstype) { } else if ((new_mount_opt & UFS_MOUNT_UFSTYPE) != ufstype) {
pr_err("ufstype can't be changed during remount\n"); pr_err("ufstype can't be changed during remount\n");
mutex_unlock(&UFS_SB(sb)->s_lock);
unlock_ufs(sb); unlock_ufs(sb);
return -EINVAL; return -EINVAL;
} }
if ((*mount_flags & MS_RDONLY) == (sb->s_flags & MS_RDONLY)) { if ((*mount_flags & MS_RDONLY) == (sb->s_flags & MS_RDONLY)) {
UFS_SB(sb)->s_mount_opt = new_mount_opt; UFS_SB(sb)->s_mount_opt = new_mount_opt;
mutex_unlock(&UFS_SB(sb)->s_lock);
unlock_ufs(sb); unlock_ufs(sb);
return 0; return 0;
} }
...@@ -1326,6 +1333,7 @@ static int ufs_remount (struct super_block *sb, int *mount_flags, char *data) ...@@ -1326,6 +1333,7 @@ static int ufs_remount (struct super_block *sb, int *mount_flags, char *data)
*/ */
#ifndef CONFIG_UFS_FS_WRITE #ifndef CONFIG_UFS_FS_WRITE
pr_err("ufs was compiled with read-only support, can't be mounted as read-write\n"); pr_err("ufs was compiled with read-only support, can't be mounted as read-write\n");
mutex_unlock(&UFS_SB(sb)->s_lock);
unlock_ufs(sb); unlock_ufs(sb);
return -EINVAL; return -EINVAL;
#else #else
...@@ -1335,11 +1343,13 @@ static int ufs_remount (struct super_block *sb, int *mount_flags, char *data) ...@@ -1335,11 +1343,13 @@ static int ufs_remount (struct super_block *sb, int *mount_flags, char *data)
ufstype != UFS_MOUNT_UFSTYPE_SUNx86 && ufstype != UFS_MOUNT_UFSTYPE_SUNx86 &&
ufstype != UFS_MOUNT_UFSTYPE_UFS2) { ufstype != UFS_MOUNT_UFSTYPE_UFS2) {
pr_err("this ufstype is read-only supported\n"); pr_err("this ufstype is read-only supported\n");
mutex_unlock(&UFS_SB(sb)->s_lock);
unlock_ufs(sb); unlock_ufs(sb);
return -EINVAL; return -EINVAL;
} }
if (!ufs_read_cylinder_structures(sb)) { if (!ufs_read_cylinder_structures(sb)) {
pr_err("failed during remounting\n"); pr_err("failed during remounting\n");
mutex_unlock(&UFS_SB(sb)->s_lock);
unlock_ufs(sb); unlock_ufs(sb);
return -EPERM; return -EPERM;
} }
...@@ -1347,6 +1357,7 @@ static int ufs_remount (struct super_block *sb, int *mount_flags, char *data) ...@@ -1347,6 +1357,7 @@ static int ufs_remount (struct super_block *sb, int *mount_flags, char *data)
#endif #endif
} }
UFS_SB(sb)->s_mount_opt = new_mount_opt; UFS_SB(sb)->s_mount_opt = new_mount_opt;
mutex_unlock(&UFS_SB(sb)->s_lock);
unlock_ufs(sb); unlock_ufs(sb);
return 0; return 0;
} }
......
...@@ -30,6 +30,7 @@ struct ufs_sb_info { ...@@ -30,6 +30,7 @@ struct ufs_sb_info {
int work_queued; /* non-zero if the delayed work is queued */ int work_queued; /* non-zero if the delayed work is queued */
struct delayed_work sync_work; /* FS sync delayed work */ struct delayed_work sync_work; /* FS sync delayed work */
spinlock_t work_lock; /* protects sync_work and work_queued */ spinlock_t work_lock; /* protects sync_work and work_queued */
struct mutex s_lock;
}; };
struct ufs_inode_info { struct ufs_inode_info {
...@@ -105,7 +106,7 @@ extern int ufs_delete_entry(struct inode *, struct ufs_dir_entry *, struct page ...@@ -105,7 +106,7 @@ extern int ufs_delete_entry(struct inode *, struct ufs_dir_entry *, struct page
extern int ufs_empty_dir (struct inode *); extern int ufs_empty_dir (struct inode *);
extern struct ufs_dir_entry *ufs_dotdot(struct inode *, struct page **); extern struct ufs_dir_entry *ufs_dotdot(struct inode *, struct page **);
extern void ufs_set_link(struct inode *dir, struct ufs_dir_entry *de, extern void ufs_set_link(struct inode *dir, struct ufs_dir_entry *de,
struct page *page, struct inode *inode); struct page *page, struct inode *inode, bool update_times);
/* file.c */ /* file.c */
extern const struct inode_operations ufs_file_inode_operations; extern const struct inode_operations ufs_file_inode_operations;
......
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