Donate to e Foundation | Murena handsets with /e/OS | Own a part of Murena! Learn more

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

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
Loading
Loading
Loading
Loading
+54 −45
Original line number Original line Diff line number Diff line
@@ -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,
/*
/*
 * 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,
 */
 */
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,
/* 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,
	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,
			     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,
			       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,
			      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,
	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,
	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,
		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);
+17 −13
Original line number Original line Diff line number Diff line
@@ -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 ocfs2_dir_entry		*dl_entry;
};
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 inode *dir,
				     struct ocfs2_dir_entry **res_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,
				 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,
+77 −73
Original line number Original line Diff line number Diff line
@@ -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,
	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,
		.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,
	/* 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,


	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,
		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,
	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,


	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,


	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,
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,
	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,
	}
	}


	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,


	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,


	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,
	}
	}


	/* 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,
	}
	}


	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,
			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,
	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,
	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,
	 * 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,
		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,


	/* 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,


		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,
			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,
		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,
		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,
		    (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,
		}
		}


		/* 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,
		/* 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,
	 * 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,
		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,


	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,
	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,
		.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,


	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,


	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,


	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,
				    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,


	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,
			    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,
	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,
	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,
	     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,
	}
	}


leave:
leave:
	brelse(target_de_bh);
	ocfs2_free_dir_lookup_result(&lookup);


	mlog_exit(status);
	mlog_exit(status);
	return status;
	return status;