Commit 4a12ca3a authored by Mark Fasheh's avatar Mark Fasheh

ocfs2: Introduce dir lookup helper struct

Many directory manipulation calls pass around a tuple of dirent, and it's
containing buffer_head. Dir indexing has a bit more state, but instead of
adding yet more arguments to functions, we introduce 'struct
ocfs2_dir_lookup_result'. In this patch, it simply holds the same tuple, but
future patches will add more state.
Signed-off-by: default avatarMark Fasheh <mfasheh@suse.com>
Acked-by: default avatarJoel Becker <joel.becker@oracle.com>
parent 59b526a3
...@@ -152,6 +152,11 @@ static void ocfs2_init_dir_trailer(struct inode *inode, ...@@ -152,6 +152,11 @@ static void ocfs2_init_dir_trailer(struct inode *inode,
trailer->db_blkno = cpu_to_le64(bh->b_blocknr); trailer->db_blkno = cpu_to_le64(bh->b_blocknr);
} }
void ocfs2_free_dir_lookup_result(struct ocfs2_dir_lookup_result *res)
{
brelse(res->dl_leaf_bh);
}
/* /*
* bh passed here can be an inode block or a dir data block, depending * bh passed here can be an inode block or a dir data block, depending
* on the inode inline data flag. * on the inode inline data flag.
...@@ -483,36 +488,46 @@ static struct buffer_head *ocfs2_find_entry_el(const char *name, int namelen, ...@@ -483,36 +488,46 @@ static struct buffer_head *ocfs2_find_entry_el(const char *name, int namelen,
/* /*
* Try to find an entry of the provided name within 'dir'. * Try to find an entry of the provided name within 'dir'.
* *
* If nothing was found, NULL is returned. Otherwise, a buffer_head * If nothing was found, -ENOENT is returned. Otherwise, zero is
* and pointer to the dir entry are passed back. * returned and the struct 'res' will contain information useful to
* other directory manipulation functions.
* *
* Caller can NOT assume anything about the contents of the * Caller can NOT assume anything about the contents of the
* buffer_head - it is passed back only so that it can be passed into * buffer_heads - they are passed back only so that it can be passed into
* any one of the manipulation functions (add entry, delete entry, * any one of the manipulation functions (add entry, delete entry,
* etc). As an example, bh in the extent directory case is a data * etc). As an example, bh in the extent directory case is a data
* block, in the inline-data case it actually points to an inode. * block, in the inline-data case it actually points to an inode.
*/ */
struct buffer_head *ocfs2_find_entry(const char *name, int namelen, int ocfs2_find_entry(const char *name, int namelen,
struct inode *dir, struct inode *dir, struct ocfs2_dir_lookup_result *lookup)
struct ocfs2_dir_entry **res_dir)
{ {
*res_dir = NULL; struct buffer_head *bh;
struct ocfs2_dir_entry *res_dir = NULL;
if (OCFS2_I(dir)->ip_dyn_features & OCFS2_INLINE_DATA_FL) if (OCFS2_I(dir)->ip_dyn_features & OCFS2_INLINE_DATA_FL)
return ocfs2_find_entry_id(name, namelen, dir, res_dir); bh = ocfs2_find_entry_id(name, namelen, dir, &res_dir);
else
bh = ocfs2_find_entry_el(name, namelen, dir, &res_dir);
return ocfs2_find_entry_el(name, namelen, dir, res_dir); if (bh == NULL)
return -ENOENT;
lookup->dl_leaf_bh = bh;
lookup->dl_entry = res_dir;
return 0;
} }
/* /*
* Update inode number and type of a previously found directory entry. * Update inode number and type of a previously found directory entry.
*/ */
int ocfs2_update_entry(struct inode *dir, handle_t *handle, int ocfs2_update_entry(struct inode *dir, handle_t *handle,
struct buffer_head *de_bh, struct ocfs2_dir_entry *de, struct ocfs2_dir_lookup_result *res,
struct inode *new_entry_inode) struct inode *new_entry_inode)
{ {
int ret; int ret;
ocfs2_journal_access_func access = ocfs2_journal_access_db; ocfs2_journal_access_func access = ocfs2_journal_access_db;
struct ocfs2_dir_entry *de = res->dl_entry;
struct buffer_head *de_bh = res->dl_leaf_bh;
/* /*
* The same code works fine for both inline-data and extent * The same code works fine for both inline-data and extent
...@@ -629,13 +644,14 @@ static inline int ocfs2_delete_entry_el(handle_t *handle, ...@@ -629,13 +644,14 @@ static inline int ocfs2_delete_entry_el(handle_t *handle,
*/ */
int ocfs2_delete_entry(handle_t *handle, int ocfs2_delete_entry(handle_t *handle,
struct inode *dir, struct inode *dir,
struct ocfs2_dir_entry *de_del, struct ocfs2_dir_lookup_result *res)
struct buffer_head *bh)
{ {
if (OCFS2_I(dir)->ip_dyn_features & OCFS2_INLINE_DATA_FL) if (OCFS2_I(dir)->ip_dyn_features & OCFS2_INLINE_DATA_FL)
return ocfs2_delete_entry_id(handle, dir, de_del, bh); return ocfs2_delete_entry_id(handle, dir, res->dl_entry,
res->dl_leaf_bh);
return ocfs2_delete_entry_el(handle, dir, de_del, bh); return ocfs2_delete_entry_el(handle, dir, res->dl_entry,
res->dl_leaf_bh);
} }
/* /*
...@@ -666,15 +682,15 @@ static inline int ocfs2_dirent_would_fit(struct ocfs2_dir_entry *de, ...@@ -666,15 +682,15 @@ static inline int ocfs2_dirent_would_fit(struct ocfs2_dir_entry *de,
/* we don't always have a dentry for what we want to add, so people /* we don't always have a dentry for what we want to add, so people
* like orphan dir can call this instead. * like orphan dir can call this instead.
* *
* If you pass me insert_bh, I'll skip the search of the other dir * The lookup context must have been filled from
* blocks and put the record in there. * ocfs2_prepare_dir_for_insert.
*/ */
int __ocfs2_add_entry(handle_t *handle, int __ocfs2_add_entry(handle_t *handle,
struct inode *dir, struct inode *dir,
const char *name, int namelen, const char *name, int namelen,
struct inode *inode, u64 blkno, struct inode *inode, u64 blkno,
struct buffer_head *parent_fe_bh, struct buffer_head *parent_fe_bh,
struct buffer_head *insert_bh) struct ocfs2_dir_lookup_result *lookup)
{ {
unsigned long offset; unsigned long offset;
unsigned short rec_len; unsigned short rec_len;
...@@ -683,6 +699,7 @@ int __ocfs2_add_entry(handle_t *handle, ...@@ -683,6 +699,7 @@ int __ocfs2_add_entry(handle_t *handle,
struct super_block *sb = dir->i_sb; struct super_block *sb = dir->i_sb;
int retval, status; int retval, status;
unsigned int size = sb->s_blocksize; unsigned int size = sb->s_blocksize;
struct buffer_head *insert_bh = lookup->dl_leaf_bh;
char *data_start = insert_bh->b_data; char *data_start = insert_bh->b_data;
mlog_entry_void(); mlog_entry_void();
...@@ -1071,31 +1088,22 @@ int ocfs2_find_files_on_disk(const char *name, ...@@ -1071,31 +1088,22 @@ int ocfs2_find_files_on_disk(const char *name,
int namelen, int namelen,
u64 *blkno, u64 *blkno,
struct inode *inode, struct inode *inode,
struct buffer_head **dirent_bh, struct ocfs2_dir_lookup_result *lookup)
struct ocfs2_dir_entry **dirent)
{ {
int status = -ENOENT; int status = -ENOENT;
mlog_entry("(name=%.*s, blkno=%p, inode=%p, dirent_bh=%p, dirent=%p)\n", mlog(0, "name=%.*s, blkno=%p, inode=%llu\n", namelen, name, blkno,
namelen, name, blkno, inode, dirent_bh, dirent); (unsigned long long)OCFS2_I(inode)->ip_blkno);
*dirent_bh = ocfs2_find_entry(name, namelen, inode, dirent); status = ocfs2_find_entry(name, namelen, inode, lookup);
if (!*dirent_bh || !*dirent) { if (status)
status = -ENOENT;
goto leave; goto leave;
}
*blkno = le64_to_cpu((*dirent)->inode); *blkno = le64_to_cpu(lookup->dl_entry->inode);
status = 0; status = 0;
leave: leave:
if (status < 0) {
*dirent = NULL;
brelse(*dirent_bh);
*dirent_bh = NULL;
}
mlog_exit(status);
return status; return status;
} }
...@@ -1107,11 +1115,10 @@ int ocfs2_lookup_ino_from_name(struct inode *dir, const char *name, ...@@ -1107,11 +1115,10 @@ int ocfs2_lookup_ino_from_name(struct inode *dir, const char *name,
int namelen, u64 *blkno) int namelen, u64 *blkno)
{ {
int ret; int ret;
struct buffer_head *bh = NULL; struct ocfs2_dir_lookup_result lookup = { NULL, };
struct ocfs2_dir_entry *dirent = NULL;
ret = ocfs2_find_files_on_disk(name, namelen, blkno, dir, &bh, &dirent); ret = ocfs2_find_files_on_disk(name, namelen, blkno, dir, &lookup);
brelse(bh); ocfs2_free_dir_lookup_result(&lookup);
return ret; return ret;
} }
...@@ -1128,20 +1135,18 @@ int ocfs2_check_dir_for_entry(struct inode *dir, ...@@ -1128,20 +1135,18 @@ int ocfs2_check_dir_for_entry(struct inode *dir,
int namelen) int namelen)
{ {
int ret; int ret;
struct buffer_head *dirent_bh = NULL; struct ocfs2_dir_lookup_result lookup = { NULL, };
struct ocfs2_dir_entry *dirent = NULL;
mlog_entry("dir %llu, name '%.*s'\n", mlog_entry("dir %llu, name '%.*s'\n",
(unsigned long long)OCFS2_I(dir)->ip_blkno, namelen, name); (unsigned long long)OCFS2_I(dir)->ip_blkno, namelen, name);
ret = -EEXIST; ret = -EEXIST;
dirent_bh = ocfs2_find_entry(name, namelen, dir, &dirent); if (ocfs2_find_entry(name, namelen, dir, &lookup) == 0)
if (dirent_bh)
goto bail; goto bail;
ret = 0; ret = 0;
bail: bail:
brelse(dirent_bh); ocfs2_free_dir_lookup_result(&lookup);
mlog_exit(ret); mlog_exit(ret);
return ret; return ret;
...@@ -1970,12 +1975,18 @@ static int ocfs2_find_dir_space_el(struct inode *dir, const char *name, ...@@ -1970,12 +1975,18 @@ static int ocfs2_find_dir_space_el(struct inode *dir, const char *name,
return status; return status;
} }
/*
* Get a directory ready for insert. Any directory allocation required
* happens here. Success returns zero, and enough context in the dir
* lookup result that ocfs2_add_entry() will be able complete the task
* with minimal performance impact.
*/
int ocfs2_prepare_dir_for_insert(struct ocfs2_super *osb, int ocfs2_prepare_dir_for_insert(struct ocfs2_super *osb,
struct inode *dir, struct inode *dir,
struct buffer_head *parent_fe_bh, struct buffer_head *parent_fe_bh,
const char *name, const char *name,
int namelen, int namelen,
struct buffer_head **ret_de_bh) struct ocfs2_dir_lookup_result *lookup)
{ {
int ret; int ret;
unsigned int blocks_wanted = 1; unsigned int blocks_wanted = 1;
...@@ -1984,8 +1995,6 @@ int ocfs2_prepare_dir_for_insert(struct ocfs2_super *osb, ...@@ -1984,8 +1995,6 @@ int ocfs2_prepare_dir_for_insert(struct ocfs2_super *osb,
mlog(0, "getting ready to insert namelen %d into dir %llu\n", mlog(0, "getting ready to insert namelen %d into dir %llu\n",
namelen, (unsigned long long)OCFS2_I(dir)->ip_blkno); namelen, (unsigned long long)OCFS2_I(dir)->ip_blkno);
*ret_de_bh = NULL;
if (!namelen) { if (!namelen) {
ret = -EINVAL; ret = -EINVAL;
mlog_errno(ret); mlog_errno(ret);
...@@ -2020,7 +2029,7 @@ int ocfs2_prepare_dir_for_insert(struct ocfs2_super *osb, ...@@ -2020,7 +2029,7 @@ int ocfs2_prepare_dir_for_insert(struct ocfs2_super *osb,
BUG_ON(!bh); BUG_ON(!bh);
} }
*ret_de_bh = bh; lookup->dl_leaf_bh = bh;
bh = NULL; bh = NULL;
out: out:
brelse(bh); brelse(bh);
......
...@@ -26,44 +26,48 @@ ...@@ -26,44 +26,48 @@
#ifndef OCFS2_DIR_H #ifndef OCFS2_DIR_H
#define OCFS2_DIR_H #define OCFS2_DIR_H
struct buffer_head *ocfs2_find_entry(const char *name, struct ocfs2_dir_lookup_result {
int namelen, struct buffer_head *dl_leaf_bh;
struct inode *dir, struct ocfs2_dir_entry *dl_entry;
struct ocfs2_dir_entry **res_dir); };
void ocfs2_free_dir_lookup_result(struct ocfs2_dir_lookup_result *res);
int ocfs2_find_entry(const char *name, int namelen,
struct inode *dir,
struct ocfs2_dir_lookup_result *lookup);
int ocfs2_delete_entry(handle_t *handle, int ocfs2_delete_entry(handle_t *handle,
struct inode *dir, struct inode *dir,
struct ocfs2_dir_entry *de_del, struct ocfs2_dir_lookup_result *res);
struct buffer_head *bh);
int __ocfs2_add_entry(handle_t *handle, int __ocfs2_add_entry(handle_t *handle,
struct inode *dir, struct inode *dir,
const char *name, int namelen, const char *name, int namelen,
struct inode *inode, u64 blkno, struct inode *inode, u64 blkno,
struct buffer_head *parent_fe_bh, struct buffer_head *parent_fe_bh,
struct buffer_head *insert_bh); struct ocfs2_dir_lookup_result *lookup);
static inline int ocfs2_add_entry(handle_t *handle, static inline int ocfs2_add_entry(handle_t *handle,
struct dentry *dentry, struct dentry *dentry,
struct inode *inode, u64 blkno, struct inode *inode, u64 blkno,
struct buffer_head *parent_fe_bh, struct buffer_head *parent_fe_bh,
struct buffer_head *insert_bh) struct ocfs2_dir_lookup_result *lookup)
{ {
return __ocfs2_add_entry(handle, dentry->d_parent->d_inode, return __ocfs2_add_entry(handle, dentry->d_parent->d_inode,
dentry->d_name.name, dentry->d_name.len, dentry->d_name.name, dentry->d_name.len,
inode, blkno, parent_fe_bh, insert_bh); inode, blkno, parent_fe_bh, lookup);
} }
int ocfs2_update_entry(struct inode *dir, handle_t *handle, int ocfs2_update_entry(struct inode *dir, handle_t *handle,
struct buffer_head *de_bh, struct ocfs2_dir_entry *de, struct ocfs2_dir_lookup_result *res,
struct inode *new_entry_inode); struct inode *new_entry_inode);
int ocfs2_check_dir_for_entry(struct inode *dir, int ocfs2_check_dir_for_entry(struct inode *dir,
const char *name, const char *name,
int namelen); int namelen);
int ocfs2_empty_dir(struct inode *inode); int ocfs2_empty_dir(struct inode *inode);
int ocfs2_find_files_on_disk(const char *name, int ocfs2_find_files_on_disk(const char *name,
int namelen, int namelen,
u64 *blkno, u64 *blkno,
struct inode *inode, struct inode *inode,
struct buffer_head **dirent_bh, struct ocfs2_dir_lookup_result *res);
struct ocfs2_dir_entry **dirent);
int ocfs2_lookup_ino_from_name(struct inode *dir, const char *name, int ocfs2_lookup_ino_from_name(struct inode *dir, const char *name,
int namelen, u64 *blkno); int namelen, u64 *blkno);
int ocfs2_readdir(struct file *filp, void *dirent, filldir_t filldir); int ocfs2_readdir(struct file *filp, void *dirent, filldir_t filldir);
...@@ -74,7 +78,7 @@ int ocfs2_prepare_dir_for_insert(struct ocfs2_super *osb, ...@@ -74,7 +78,7 @@ int ocfs2_prepare_dir_for_insert(struct ocfs2_super *osb,
struct buffer_head *parent_fe_bh, struct buffer_head *parent_fe_bh,
const char *name, const char *name,
int namelen, int namelen,
struct buffer_head **ret_de_bh); struct ocfs2_dir_lookup_result *lookup);
struct ocfs2_alloc_context; struct ocfs2_alloc_context;
int ocfs2_fill_new_dir(struct ocfs2_super *osb, int ocfs2_fill_new_dir(struct ocfs2_super *osb,
handle_t *handle, handle_t *handle,
......
...@@ -80,14 +80,14 @@ static int ocfs2_prepare_orphan_dir(struct ocfs2_super *osb, ...@@ -80,14 +80,14 @@ static int ocfs2_prepare_orphan_dir(struct ocfs2_super *osb,
struct inode **ret_orphan_dir, struct inode **ret_orphan_dir,
struct inode *inode, struct inode *inode,
char *name, char *name,
struct buffer_head **de_bh); struct ocfs2_dir_lookup_result *lookup);
static int ocfs2_orphan_add(struct ocfs2_super *osb, static int ocfs2_orphan_add(struct ocfs2_super *osb,
handle_t *handle, handle_t *handle,
struct inode *inode, struct inode *inode,
struct ocfs2_dinode *fe, struct ocfs2_dinode *fe,
char *name, char *name,
struct buffer_head *de_bh, struct ocfs2_dir_lookup_result *lookup,
struct inode *orphan_dir_inode); struct inode *orphan_dir_inode);
static int ocfs2_create_symlink_data(struct ocfs2_super *osb, static int ocfs2_create_symlink_data(struct ocfs2_super *osb,
...@@ -228,7 +228,6 @@ static int ocfs2_mknod(struct inode *dir, ...@@ -228,7 +228,6 @@ static int ocfs2_mknod(struct inode *dir,
struct ocfs2_super *osb; struct ocfs2_super *osb;
struct ocfs2_dinode *dirfe; struct ocfs2_dinode *dirfe;
struct buffer_head *new_fe_bh = NULL; struct buffer_head *new_fe_bh = NULL;
struct buffer_head *de_bh = NULL;
struct inode *inode = NULL; struct inode *inode = NULL;
struct ocfs2_alloc_context *inode_ac = NULL; struct ocfs2_alloc_context *inode_ac = NULL;
struct ocfs2_alloc_context *data_ac = NULL; struct ocfs2_alloc_context *data_ac = NULL;
...@@ -239,6 +238,7 @@ static int ocfs2_mknod(struct inode *dir, ...@@ -239,6 +238,7 @@ static int ocfs2_mknod(struct inode *dir,
.enable = 1, .enable = 1,
}; };
int did_quota_inode = 0; int did_quota_inode = 0;
struct ocfs2_dir_lookup_result lookup = { NULL, };
mlog_entry("(0x%p, 0x%p, %d, %lu, '%.*s')\n", dir, dentry, mode, mlog_entry("(0x%p, 0x%p, %d, %lu, '%.*s')\n", dir, dentry, mode,
(unsigned long)dev, dentry->d_name.len, (unsigned long)dev, dentry->d_name.len,
...@@ -274,7 +274,7 @@ static int ocfs2_mknod(struct inode *dir, ...@@ -274,7 +274,7 @@ static int ocfs2_mknod(struct inode *dir,
/* get a spot inside the dir. */ /* get a spot inside the dir. */
status = ocfs2_prepare_dir_for_insert(osb, dir, parent_fe_bh, status = ocfs2_prepare_dir_for_insert(osb, dir, parent_fe_bh,
dentry->d_name.name, dentry->d_name.name,
dentry->d_name.len, &de_bh); dentry->d_name.len, &lookup);
if (status < 0) { if (status < 0) {
mlog_errno(status); mlog_errno(status);
goto leave; goto leave;
...@@ -394,7 +394,7 @@ static int ocfs2_mknod(struct inode *dir, ...@@ -394,7 +394,7 @@ static int ocfs2_mknod(struct inode *dir,
status = ocfs2_add_entry(handle, dentry, inode, status = ocfs2_add_entry(handle, dentry, inode,
OCFS2_I(inode)->ip_blkno, parent_fe_bh, OCFS2_I(inode)->ip_blkno, parent_fe_bh,
de_bh); &lookup);
if (status < 0) { if (status < 0) {
mlog_errno(status); mlog_errno(status);
goto leave; goto leave;
...@@ -423,11 +423,12 @@ static int ocfs2_mknod(struct inode *dir, ...@@ -423,11 +423,12 @@ static int ocfs2_mknod(struct inode *dir,
mlog(0, "Disk is full\n"); mlog(0, "Disk is full\n");
brelse(new_fe_bh); brelse(new_fe_bh);
brelse(de_bh);
brelse(parent_fe_bh); brelse(parent_fe_bh);
kfree(si.name); kfree(si.name);
kfree(si.value); kfree(si.value);
ocfs2_free_dir_lookup_result(&lookup);
if ((status < 0) && inode) { if ((status < 0) && inode) {
clear_nlink(inode); clear_nlink(inode);
iput(inode); iput(inode);
...@@ -608,9 +609,9 @@ static int ocfs2_link(struct dentry *old_dentry, ...@@ -608,9 +609,9 @@ static int ocfs2_link(struct dentry *old_dentry,
int err; int err;
struct buffer_head *fe_bh = NULL; struct buffer_head *fe_bh = NULL;
struct buffer_head *parent_fe_bh = NULL; struct buffer_head *parent_fe_bh = NULL;
struct buffer_head *de_bh = NULL;
struct ocfs2_dinode *fe = NULL; struct ocfs2_dinode *fe = NULL;
struct ocfs2_super *osb = OCFS2_SB(dir->i_sb); struct ocfs2_super *osb = OCFS2_SB(dir->i_sb);
struct ocfs2_dir_lookup_result lookup = { NULL, };
mlog_entry("(inode=%lu, old='%.*s' new='%.*s')\n", inode->i_ino, mlog_entry("(inode=%lu, old='%.*s' new='%.*s')\n", inode->i_ino,
old_dentry->d_name.len, old_dentry->d_name.name, old_dentry->d_name.len, old_dentry->d_name.name,
...@@ -638,7 +639,7 @@ static int ocfs2_link(struct dentry *old_dentry, ...@@ -638,7 +639,7 @@ static int ocfs2_link(struct dentry *old_dentry,
err = ocfs2_prepare_dir_for_insert(osb, dir, parent_fe_bh, err = ocfs2_prepare_dir_for_insert(osb, dir, parent_fe_bh,
dentry->d_name.name, dentry->d_name.name,
dentry->d_name.len, &de_bh); dentry->d_name.len, &lookup);
if (err < 0) { if (err < 0) {
mlog_errno(err); mlog_errno(err);
goto out; goto out;
...@@ -688,7 +689,7 @@ static int ocfs2_link(struct dentry *old_dentry, ...@@ -688,7 +689,7 @@ static int ocfs2_link(struct dentry *old_dentry,
err = ocfs2_add_entry(handle, dentry, inode, err = ocfs2_add_entry(handle, dentry, inode,
OCFS2_I(inode)->ip_blkno, OCFS2_I(inode)->ip_blkno,
parent_fe_bh, de_bh); parent_fe_bh, &lookup);
if (err) { if (err) {
le16_add_cpu(&fe->i_links_count, -1); le16_add_cpu(&fe->i_links_count, -1);
drop_nlink(inode); drop_nlink(inode);
...@@ -714,10 +715,11 @@ static int ocfs2_link(struct dentry *old_dentry, ...@@ -714,10 +715,11 @@ static int ocfs2_link(struct dentry *old_dentry,
out: out:
ocfs2_inode_unlock(dir, 1); ocfs2_inode_unlock(dir, 1);
brelse(de_bh);
brelse(fe_bh); brelse(fe_bh);
brelse(parent_fe_bh); brelse(parent_fe_bh);
ocfs2_free_dir_lookup_result(&lookup);
mlog_exit(err); mlog_exit(err);
return err; return err;
...@@ -766,10 +768,9 @@ static int ocfs2_unlink(struct inode *dir, ...@@ -766,10 +768,9 @@ static int ocfs2_unlink(struct inode *dir,
struct buffer_head *fe_bh = NULL; struct buffer_head *fe_bh = NULL;
struct buffer_head *parent_node_bh = NULL; struct buffer_head *parent_node_bh = NULL;
handle_t *handle = NULL; handle_t *handle = NULL;
struct ocfs2_dir_entry *dirent = NULL;
struct buffer_head *dirent_bh = NULL;
char orphan_name[OCFS2_ORPHAN_NAMELEN + 1]; char orphan_name[OCFS2_ORPHAN_NAMELEN + 1];
struct buffer_head *orphan_entry_bh = NULL; struct ocfs2_dir_lookup_result lookup = { NULL, };
struct ocfs2_dir_lookup_result orphan_insert = { NULL, };
mlog_entry("(0x%p, 0x%p, '%.*s')\n", dir, dentry, mlog_entry("(0x%p, 0x%p, '%.*s')\n", dir, dentry,
dentry->d_name.len, dentry->d_name.name); dentry->d_name.len, dentry->d_name.name);
...@@ -791,8 +792,8 @@ static int ocfs2_unlink(struct inode *dir, ...@@ -791,8 +792,8 @@ static int ocfs2_unlink(struct inode *dir,
} }
status = ocfs2_find_files_on_disk(dentry->d_name.name, status = ocfs2_find_files_on_disk(dentry->d_name.name,
dentry->d_name.len, &blkno, dentry->d_name.len, &blkno, dir,
dir, &dirent_bh, &dirent); &lookup);
if (status < 0) { if (status < 0) {
if (status != -ENOENT) if (status != -ENOENT)
mlog_errno(status); mlog_errno(status);
...@@ -836,8 +837,7 @@ static int ocfs2_unlink(struct inode *dir, ...@@ -836,8 +837,7 @@ static int ocfs2_unlink(struct inode *dir,
if (inode_is_unlinkable(inode)) { if (inode_is_unlinkable(inode)) {
status = ocfs2_prepare_orphan_dir(osb, &orphan_dir, inode, status = ocfs2_prepare_orphan_dir(osb, &orphan_dir, inode,
orphan_name, orphan_name, &orphan_insert);
&orphan_entry_bh);
if (status < 0) { if (status < 0) {
mlog_errno(status); mlog_errno(status);
goto leave; goto leave;
...@@ -863,7 +863,7 @@ static int ocfs2_unlink(struct inode *dir, ...@@ -863,7 +863,7 @@ static int ocfs2_unlink(struct inode *dir,
if (inode_is_unlinkable(inode)) { if (inode_is_unlinkable(inode)) {
status = ocfs2_orphan_add(osb, handle, inode, fe, orphan_name, status = ocfs2_orphan_add(osb, handle, inode, fe, orphan_name,
orphan_entry_bh, orphan_dir); &orphan_insert, orphan_dir);
if (status < 0) { if (status < 0) {
mlog_errno(status); mlog_errno(status);
goto leave; goto leave;
...@@ -871,7 +871,7 @@ static int ocfs2_unlink(struct inode *dir, ...@@ -871,7 +871,7 @@ static int ocfs2_unlink(struct inode *dir,
} }
/* delete the name from the parent dir */ /* delete the name from the parent dir */
status = ocfs2_delete_entry(handle, dir, dirent, dirent_bh); status = ocfs2_delete_entry(handle, dir, &lookup);
if (status < 0) { if (status < 0) {
mlog_errno(status); mlog_errno(status);
goto leave; goto leave;
...@@ -916,9 +916,10 @@ static int ocfs2_unlink(struct inode *dir, ...@@ -916,9 +916,10 @@ static int ocfs2_unlink(struct inode *dir,
} }
brelse(fe_bh); brelse(fe_bh);
brelse(dirent_bh);
brelse(parent_node_bh); brelse(parent_node_bh);
brelse(orphan_entry_bh);
ocfs2_free_dir_lookup_result(&orphan_insert);
ocfs2_free_dir_lookup_result(&lookup);
mlog_exit(status); mlog_exit(status);
...@@ -1004,8 +1005,8 @@ static int ocfs2_rename(struct inode *old_dir, ...@@ -1004,8 +1005,8 @@ static int ocfs2_rename(struct inode *old_dir,
struct inode *new_dir, struct inode *new_dir,
struct dentry *new_dentry) struct dentry *new_dentry)
{ {
int status = 0, rename_lock = 0, parents_locked = 0; int status = 0, rename_lock = 0, parents_locked = 0, target_exists = 0;
int old_child_locked = 0, new_child_locked = 0; int old_child_locked = 0, new_child_locked = 0, update_dot_dot = 0;
struct inode *old_inode = old_dentry->d_inode; struct inode *old_inode = old_dentry->d_inode;
struct inode *new_inode = new_dentry->d_inode; struct inode *new_inode = new_dentry->d_inode;
struct inode *orphan_dir = NULL; struct inode *orphan_dir = NULL;
...@@ -1020,13 +1021,13 @@ static int ocfs2_rename(struct inode *old_dir, ...@@ -1020,13 +1021,13 @@ static int ocfs2_rename(struct inode *old_dir,
handle_t *handle = NULL; handle_t *handle = NULL;
struct buffer_head *old_dir_bh = NULL; struct buffer_head *old_dir_bh = NULL;
struct buffer_head *new_dir_bh = NULL; struct buffer_head *new_dir_bh = NULL;
struct ocfs2_dir_entry *old_inode_dot_dot_de = NULL, *old_de = NULL,
*new_de = NULL;
struct buffer_head *new_de_bh = NULL, *old_de_bh = NULL; // bhs for above
struct buffer_head *old_inode_de_bh = NULL; // if old_dentry is a dir,
// this is the 1st dirent bh
nlink_t old_dir_nlink = old_dir->i_nlink; nlink_t old_dir_nlink = old_dir->i_nlink;
struct ocfs2_dinode *old_di; struct ocfs2_dinode *old_di;
struct ocfs2_dir_lookup_result old_inode_dot_dot_res = { NULL, };
struct ocfs2_dir_lookup_result target_lookup_res = { NULL, };
struct ocfs2_dir_lookup_result old_entry_lookup = { NULL, };
struct ocfs2_dir_lookup_result orphan_insert = { NULL, };
struct ocfs2_dir_lookup_result target_insert = { NULL, };
/* At some point it might be nice to break this function up a /* At some point it might be nice to break this function up a
* bit. */ * bit. */
...@@ -1108,9 +1109,10 @@ static int ocfs2_rename(struct inode *old_dir, ...@@ -1108,9 +1109,10 @@ static int ocfs2_rename(struct inode *old_dir,
if (S_ISDIR(old_inode->i_mode)) { if (S_ISDIR(old_inode->i_mode)) {
u64 old_inode_parent; u64 old_inode_parent;
update_dot_dot = 1;
status = ocfs2_find_files_on_disk("..", 2, &old_inode_parent, status = ocfs2_find_files_on_disk("..", 2, &old_inode_parent,
old_inode, &old_inode_de_bh, old_inode,
&old_inode_dot_dot_de); &old_inode_dot_dot_res);
if (status) { if (status) {
status = -EIO; status = -EIO;
goto bail; goto bail;
...@@ -1151,8 +1153,8 @@ static int ocfs2_rename(struct inode *old_dir, ...@@ -1151,8 +1153,8 @@ static int ocfs2_rename(struct inode *old_dir,
* to delete it */ * to delete it */
status = ocfs2_find_files_on_disk(new_dentry->d_name.name, status = ocfs2_find_files_on_disk(new_dentry->d_name.name,
new_dentry->d_name.len, new_dentry->d_name.len,
&newfe_blkno, new_dir, &new_de_bh, &newfe_blkno, new_dir,
&new_de); &target_lookup_res);
/* The only error we allow here is -ENOENT because the new /* The only error we allow here is -ENOENT because the new
* file not existing is perfectly valid. */ * file not existing is perfectly valid. */
if ((status < 0) && (status != -ENOENT)) { if ((status < 0) && (status != -ENOENT)) {
...@@ -1161,8 +1163,10 @@ static int ocfs2_rename(struct inode *old_dir, ...@@ -1161,8 +1163,10 @@ static int ocfs2_rename(struct inode *old_dir,
mlog_errno(status); mlog_errno(status);
goto bail; goto bail;
} }
if (status == 0)
target_exists = 1;
if (!new_de && new_inode) { if (!target_exists && new_inode) {
/* /*
* Target was unlinked by another node while we were * Target was unlinked by another node while we were
* waiting to get to ocfs2_rename(). There isn't * waiting to get to ocfs2_rename(). There isn't
...@@ -1175,7 +1179,7 @@ static int ocfs2_rename(struct inode *old_dir, ...@@ -1175,7 +1179,7 @@ static int ocfs2_rename(struct inode *old_dir,
/* In case we need to overwrite an existing file, we blow it /* In case we need to overwrite an existing file, we blow it
* away first */ * away first */
if (new_de) { if (target_exists) {
/* VFS didn't think there existed an inode here, but /* VFS didn't think there existed an inode here, but
* someone else in the cluster must have raced our * someone else in the cluster must have raced our
* rename to create one. Today we error cleanly, in * rename to create one. Today we error cleanly, in
...@@ -1216,8 +1220,8 @@ static int ocfs2_rename(struct inode *old_dir, ...@@ -1216,8 +1220,8 @@ static int ocfs2_rename(struct inode *old_dir,
newfe = (struct ocfs2_dinode *) newfe_bh->b_data; newfe = (struct ocfs2_dinode *) newfe_bh->b_data;
mlog(0, "aha rename over existing... new_de=%p new_blkno=%llu " mlog(0, "aha rename over existing... new_blkno=%llu "
"newfebh=%p bhblocknr=%llu\n", new_de, "newfebh=%p bhblocknr=%llu\n",
(unsigned long long)newfe_blkno, newfe_bh, newfe_bh ? (unsigned long long)newfe_blkno, newfe_bh, newfe_bh ?
(unsigned long long)newfe_bh->b_blocknr : 0ULL); (unsigned long long)newfe_bh->b_blocknr : 0ULL);
...@@ -1225,7 +1229,7 @@ static int ocfs2_rename(struct inode *old_dir, ...@@ -1225,7 +1229,7 @@ static int ocfs2_rename(struct inode *old_dir,
status = ocfs2_prepare_orphan_dir(osb, &orphan_dir, status = ocfs2_prepare_orphan_dir(osb, &orphan_dir,
new_inode, new_inode,
orphan_name, orphan_name,
&orphan_entry_bh); &orphan_insert);
if (status < 0) { if (status < 0) {
mlog_errno(status); mlog_errno(status);
goto bail; goto bail;
...@@ -1243,7 +1247,7 @@ static int ocfs2_rename(struct inode *old_dir, ...@@ -1243,7 +1247,7 @@ static int ocfs2_rename(struct inode *old_dir,
status = ocfs2_prepare_dir_for_insert(osb, new_dir, new_dir_bh, status = ocfs2_prepare_dir_for_insert(osb, new_dir, new_dir_bh,
new_dentry->d_name.name, new_dentry->d_name.name,
new_dentry->d_name.len, new_dentry->d_name.len,
&insert_entry_bh); &target_insert);
if (status < 0) { if (status < 0) {
mlog_errno(status); mlog_errno(status);
goto bail; goto bail;
...@@ -1258,7 +1262,7 @@ static int ocfs2_rename(struct inode *old_dir, ...@@ -1258,7 +1262,7 @@ static int ocfs2_rename(struct inode *old_dir,
goto bail; goto bail;
} }
if (new_de) { if (target_exists) {
if (S_ISDIR(new_inode->i_mode)) { if (S_ISDIR(new_inode->i_mode)) {
if (!ocfs2_empty_dir(new_inode) || if (!ocfs2_empty_dir(new_inode) ||
new_inode->i_nlink != 2) { new_inode->i_nlink != 2) {
...@@ -1277,7 +1281,7 @@ static int ocfs2_rename(struct inode *old_dir, ...@@ -1277,7 +1281,7 @@ static int ocfs2_rename(struct inode *old_dir,
(newfe->i_links_count == cpu_to_le16(1))){ (newfe->i_links_count == cpu_to_le16(1))){
status = ocfs2_orphan_add(osb, handle, new_inode, status = ocfs2_orphan_add(osb, handle, new_inode,
newfe, orphan_name, newfe, orphan_name,
orphan_entry_bh, orphan_dir); &orphan_insert, orphan_dir);
if (status < 0) { if (status < 0) {
mlog_errno(status); mlog_errno(status);
goto bail; goto bail;
...@@ -1285,8 +1289,8 @@ static int ocfs2_rename(struct inode *old_dir, ...@@ -1285,8 +1289,8 @@ static int ocfs2_rename(struct inode *old_dir,
} }
/* change the dirent to point to the correct inode */ /* change the dirent to point to the correct inode */
status = ocfs2_update_entry(new_dir, handle, new_de_bh, status = ocfs2_update_entry(new_dir, handle, &target_lookup_res,
new_de, old_inode); old_inode);
if (status < 0) { if (status < 0) {
mlog_errno(status); mlog_errno(status);
goto bail; goto bail;
...@@ -1307,7 +1311,7 @@ static int ocfs2_rename(struct inode *old_dir, ...@@ -1307,7 +1311,7 @@ static int ocfs2_rename(struct inode *old_dir,
/* if the name was not found in new_dir, add it now */ /* if the name was not found in new_dir, add it now */
status = ocfs2_add_entry(handle, new_dentry, old_inode, status = ocfs2_add_entry(handle, new_dentry, old_inode,
OCFS2_I(old_inode)->ip_blkno, OCFS2_I(old_inode)->ip_blkno,
new_dir_bh, insert_entry_bh); new_dir_bh, &target_insert);
} }
old_inode->i_ctime = CURRENT_TIME; old_inode->i_ctime = CURRENT_TIME;
...@@ -1334,15 +1338,13 @@ static int ocfs2_rename(struct inode *old_dir, ...@@ -1334,15 +1338,13 @@ static int ocfs2_rename(struct inode *old_dir,
* because the insert might have changed the type of directory * because the insert might have changed the type of directory
* we're dealing with. * we're dealing with.
*/ */
old_de_bh = ocfs2_find_entry(old_dentry->d_name.name, status = ocfs2_find_entry(old_dentry->d_name.name,
old_dentry->d_name.len, old_dentry->d_name.len, old_dir,
old_dir, &old_de); &old_entry_lookup);
if (!old_de_bh) { if (status)
status = -EIO;
goto bail; goto bail;
}
status = ocfs2_delete_entry(handle, old_dir, old_de, old_de_bh); status = ocfs2_delete_entry(handle, old_dir, &old_entry_lookup);
if (status < 0) { if (status < 0) {
mlog_errno(status); mlog_errno(status);
goto bail; goto bail;
...@@ -1353,9 +1355,10 @@ static int ocfs2_rename(struct inode *old_dir, ...@@ -1353,9 +1355,10 @@ static int ocfs2_rename(struct inode *old_dir,
new_inode->i_ctime = CURRENT_TIME; new_inode->i_ctime = CURRENT_TIME;
} }
old_dir->i_ctime = old_dir->i_mtime = CURRENT_TIME; old_dir->i_ctime = old_dir->i_mtime = CURRENT_TIME;
if (old_inode_de_bh) {
status = ocfs2_update_entry(old_inode, handle, old_inode_de_bh, if (update_dot_dot) {
old_inode_dot_dot_de, new_dir); status = ocfs2_update_entry(old_inode, handle,
&old_inode_dot_dot_res, new_dir);
old_dir->i_nlink--; old_dir->i_nlink--;
if (new_inode) { if (new_inode) {
new_inode->i_nlink--; new_inode->i_nlink--;
...@@ -1429,13 +1432,17 @@ static int ocfs2_rename(struct inode *old_dir, ...@@ -1429,13 +1432,17 @@ static int ocfs2_rename(struct inode *old_dir,
if (new_inode) if (new_inode)
iput(new_inode); iput(new_inode);
ocfs2_free_dir_lookup_result(&target_lookup_res);
ocfs2_free_dir_lookup_result(&old_entry_lookup);
ocfs2_free_dir_lookup_result(&old_inode_dot_dot_res);
ocfs2_free_dir_lookup_result(&orphan_insert);
ocfs2_free_dir_lookup_result(&target_insert);
brelse(newfe_bh); brelse(newfe_bh);
brelse(old_inode_bh); brelse(old_inode_bh);
brelse(old_dir_bh); brelse(old_dir_bh);
brelse(new_dir_bh); brelse(new_dir_bh);
brelse(new_de_bh);
brelse(old_de_bh);
brelse(old_inode_de_bh);
brelse(orphan_entry_bh); brelse(orphan_entry_bh);
brelse(insert_entry_bh); brelse(insert_entry_bh);
...@@ -1558,7 +1565,6 @@ static int ocfs2_symlink(struct inode *dir, ...@@ -1558,7 +1565,6 @@ static int ocfs2_symlink(struct inode *dir,
struct inode *inode = NULL; struct inode *inode = NULL;
struct super_block *sb; struct super_block *sb;
struct buffer_head *new_fe_bh = NULL; struct buffer_head *new_fe_bh = NULL;
struct buffer_head *de_bh = NULL;
struct buffer_head *parent_fe_bh = NULL; struct buffer_head *parent_fe_bh = NULL;
struct ocfs2_dinode *fe = NULL; struct ocfs2_dinode *fe = NULL;
struct ocfs2_dinode *dirfe; struct ocfs2_dinode *dirfe;
...@@ -1572,6 +1578,7 @@ static int ocfs2_symlink(struct inode *dir, ...@@ -1572,6 +1578,7 @@ static int ocfs2_symlink(struct inode *dir,
.enable = 1, .enable = 1,
}; };
int did_quota = 0, did_quota_inode = 0; int did_quota = 0, did_quota_inode = 0;
struct ocfs2_dir_lookup_result lookup = { NULL, };
mlog_entry("(0x%p, 0x%p, symname='%s' actual='%.*s')\n", dir, mlog_entry("(0x%p, 0x%p, symname='%s' actual='%.*s')\n", dir,
dentry, symname, dentry->d_name.len, dentry->d_name.name); dentry, symname, dentry->d_name.len, dentry->d_name.name);
...@@ -1605,7 +1612,7 @@ static int ocfs2_symlink(struct inode *dir, ...@@ -1605,7 +1612,7 @@ static int ocfs2_symlink(struct inode *dir,
status = ocfs2_prepare_dir_for_insert(osb, dir, parent_fe_bh, status = ocfs2_prepare_dir_for_insert(osb, dir, parent_fe_bh,
dentry->d_name.name, dentry->d_name.name,
dentry->d_name.len, &de_bh); dentry->d_name.len, &lookup);
if (status < 0) { if (status < 0) {
mlog_errno(status); mlog_errno(status);
goto bail; goto bail;
...@@ -1744,7 +1751,7 @@ static int ocfs2_symlink(struct inode *dir, ...@@ -1744,7 +1751,7 @@ static int ocfs2_symlink(struct inode *dir,
status = ocfs2_add_entry(handle, dentry, inode, status = ocfs2_add_entry(handle, dentry, inode,
le64_to_cpu(fe->i_blkno), parent_fe_bh, le64_to_cpu(fe->i_blkno), parent_fe_bh,
de_bh); &lookup);
if (status < 0) { if (status < 0) {
mlog_errno(status); mlog_errno(status);
goto bail; goto bail;
...@@ -1772,9 +1779,9 @@ static int ocfs2_symlink(struct inode *dir, ...@@ -1772,9 +1779,9 @@ static int ocfs2_symlink(struct inode *dir,
brelse(new_fe_bh); brelse(new_fe_bh);
brelse(parent_fe_bh); brelse(parent_fe_bh);
brelse(de_bh);
kfree(si.name); kfree(si.name);
kfree(si.value); kfree(si.value);
ocfs2_free_dir_lookup_result(&lookup);
if (inode_ac) if (inode_ac)
ocfs2_free_alloc_context(inode_ac); ocfs2_free_alloc_context(inode_ac);
if (data_ac) if (data_ac)
...@@ -1826,7 +1833,7 @@ static int ocfs2_prepare_orphan_dir(struct ocfs2_super *osb, ...@@ -1826,7 +1833,7 @@ static int ocfs2_prepare_orphan_dir(struct ocfs2_super *osb,
struct inode **ret_orphan_dir, struct inode **ret_orphan_dir,
struct inode *inode, struct inode *inode,
char *name, char *name,
struct buffer_head **de_bh) struct ocfs2_dir_lookup_result *lookup)
{ {
struct inode *orphan_dir_inode; struct inode *orphan_dir_inode;
struct buffer_head *orphan_dir_bh = NULL; struct buffer_head *orphan_dir_bh = NULL;
...@@ -1857,7 +1864,7 @@ static int ocfs2_prepare_orphan_dir(struct ocfs2_super *osb, ...@@ -1857,7 +1864,7 @@ static int ocfs2_prepare_orphan_dir(struct ocfs2_super *osb,
status = ocfs2_prepare_dir_for_insert(osb, orphan_dir_inode, status = ocfs2_prepare_dir_for_insert(osb, orphan_dir_inode,
orphan_dir_bh, name, orphan_dir_bh, name,
OCFS2_ORPHAN_NAMELEN, de_bh); OCFS2_ORPHAN_NAMELEN, lookup);
if (status < 0) { if (status < 0) {
ocfs2_inode_unlock(orphan_dir_inode, 1); ocfs2_inode_unlock(orphan_dir_inode, 1);
...@@ -1884,7 +1891,7 @@ static int ocfs2_orphan_add(struct ocfs2_super *osb, ...@@ -1884,7 +1891,7 @@ static int ocfs2_orphan_add(struct ocfs2_super *osb,
struct inode *inode, struct inode *inode,
struct ocfs2_dinode *fe, struct ocfs2_dinode *fe,
char *name, char *name,
struct buffer_head *de_bh, struct ocfs2_dir_lookup_result *lookup,
struct inode *orphan_dir_inode) struct inode *orphan_dir_inode)
{ {
struct buffer_head *orphan_dir_bh = NULL; struct buffer_head *orphan_dir_bh = NULL;
...@@ -1922,7 +1929,7 @@ static int ocfs2_orphan_add(struct ocfs2_super *osb, ...@@ -1922,7 +1929,7 @@ static int ocfs2_orphan_add(struct ocfs2_super *osb,
status = __ocfs2_add_entry(handle, orphan_dir_inode, name, status = __ocfs2_add_entry(handle, orphan_dir_inode, name,
OCFS2_ORPHAN_NAMELEN, inode, OCFS2_ORPHAN_NAMELEN, inode,
OCFS2_I(inode)->ip_blkno, OCFS2_I(inode)->ip_blkno,
orphan_dir_bh, de_bh); orphan_dir_bh, lookup);
if (status < 0) { if (status < 0) {
mlog_errno(status); mlog_errno(status);
goto leave; goto leave;
...@@ -1955,8 +1962,7 @@ int ocfs2_orphan_del(struct ocfs2_super *osb, ...@@ -1955,8 +1962,7 @@ int ocfs2_orphan_del(struct ocfs2_super *osb,
char name[OCFS2_ORPHAN_NAMELEN + 1]; char name[OCFS2_ORPHAN_NAMELEN + 1];
struct ocfs2_dinode *orphan_fe; struct ocfs2_dinode *orphan_fe;
int status = 0; int status = 0;
struct buffer_head *target_de_bh = NULL; struct ocfs2_dir_lookup_result lookup = { NULL, };
struct ocfs2_dir_entry *target_de = NULL;
mlog_entry_void(); mlog_entry_void();
...@@ -1971,17 +1977,15 @@ int ocfs2_orphan_del(struct ocfs2_super *osb, ...@@ -1971,17 +1977,15 @@ int ocfs2_orphan_del(struct ocfs2_super *osb,
OCFS2_ORPHAN_NAMELEN); OCFS2_ORPHAN_NAMELEN);
/* find it's spot in the orphan directory */ /* find it's spot in the orphan directory */
target_de_bh = ocfs2_find_entry(name, OCFS2_ORPHAN_NAMELEN, status = ocfs2_find_entry(name, OCFS2_ORPHAN_NAMELEN, orphan_dir_inode,
orphan_dir_inode, &target_de); &lookup);
if (!target_de_bh) { if (status) {
status = -ENOENT;
mlog_errno(status); mlog_errno(status);
goto leave; goto leave;
} }
/* remove it from the orphan directory */ /* remove it from the orphan directory */
status = ocfs2_delete_entry(handle, orphan_dir_inode, target_de, status = ocfs2_delete_entry(handle, orphan_dir_inode, &lookup);
target_de_bh);
if (status < 0) { if (status < 0) {
mlog_errno(status); mlog_errno(status);
goto leave; goto leave;
...@@ -2007,7 +2011,7 @@ int ocfs2_orphan_del(struct ocfs2_super *osb, ...@@ -2007,7 +2011,7 @@ int ocfs2_orphan_del(struct ocfs2_super *osb,
} }
leave: leave:
brelse(target_de_bh); ocfs2_free_dir_lookup_result(&lookup);
mlog_exit(status); mlog_exit(status);
return status; return status;
......
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