Merge branch 'for-linus' into for-next
This commit is contained in:
@@ -1239,13 +1239,13 @@ ascend:
|
|||||||
/* 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;
|
||||||
|
/* go into the first sibling still alive */
|
||||||
|
do {
|
||||||
next = child->d_child.next;
|
next = child->d_child.next;
|
||||||
while (unlikely(child->d_flags & DCACHE_DENTRY_KILLED)) {
|
|
||||||
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,
|
|||||||
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;
|
||||||
|
@@ -52,7 +52,7 @@ 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);
|
||||||
@@ -116,12 +116,12 @@ void ufs_free_fragments(struct inode *inode, u64 fragment, unsigned count)
|
|||||||
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)
|
|||||||
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 @@ do_more:
|
|||||||
}
|
}
|
||||||
|
|
||||||
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,
|
|||||||
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,
|
|||||||
"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,
|
|||||||
* 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,
|
|||||||
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,
|
|||||||
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,
|
|||||||
*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,
|
|||||||
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)
|
|||||||
|
|
||||||
/* 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,6 +104,7 @@ 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);
|
||||||
|
if (update_times)
|
||||||
dir->i_mtime = dir->i_ctime = CURRENT_TIME_SEC;
|
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)
|
|||||||
|
|
||||||
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)
|
|||||||
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)
|
|||||||
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)
|
|||||||
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 @@ cg_found:
|
|||||||
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)
|
|||||||
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
|
|||||||
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,
|
|||||||
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);
|
||||||
lock_ufs(dir->i_sb);
|
return ufs_add_nondir(dentry, inode);
|
||||||
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
|
|||||||
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,
|
|||||||
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,
|
|||||||
}
|
}
|
||||||
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,
|
|||||||
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)
|
|||||||
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)
|
|||||||
|
|
||||||
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)
|
|||||||
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)
|
|||||||
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)
|
|||||||
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,
|
|||||||
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,
|
|||||||
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)
|
|||||||
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)
|
|||||||
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)
|
|||||||
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)
|
|||||||
|
|
||||||
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)
|
|||||||
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)
|
|||||||
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)
|
|||||||
*/
|
*/
|
||||||
#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)
|
|||||||
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)
|
|||||||
#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 {
|
|||||||
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
|
|||||||
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;
|
||||||
|
Reference in New Issue
Block a user