ocfs2: Move buckets up into ocfs2_add_new_xattr_bucket().
authorJoel Becker <joel.becker@oracle.com>
Wed, 26 Nov 2008 23:08:44 +0000 (15:08 -0800)
committerMark Fasheh <mfasheh@suse.com>
Mon, 5 Jan 2009 16:40:30 +0000 (08:40 -0800)
Lift the buckets from ocfs2_add_new_xattr_cluster() up into
ocfs2_add_new_xattr_bucket().  Now ocfs2_add_new_xattr_cluster()
doesn't deal with buffer_heads.  In fact, we no longer have to play
get_bh() tricks at all.

Signed-off-by: Joel Becker <joel.becker@oracle.com>
Signed-off-by: Mark Fasheh <mfasheh@suse.com>
fs/ocfs2/xattr.c

index 4b247047b7aa78a381f871c09aa2f06f3ead186c..5a5a1bd7eedee79932fba9e051a9b16fa7fea0cc 100644 (file)
@@ -4148,11 +4148,10 @@ static int ocfs2_adjust_xattr_cross_cluster(struct inode *inode,
  */
 static int ocfs2_add_new_xattr_cluster(struct inode *inode,
                                       struct buffer_head *root_bh,
-                                      struct buffer_head **first_bh,
-                                      struct buffer_head **header_bh,
+                                      struct ocfs2_xattr_bucket *first,
+                                      struct ocfs2_xattr_bucket *target,
                                       u32 *num_clusters,
                                       u32 prev_cpos,
-                                      u64 prev_blkno,
                                       int *extend,
                                       struct ocfs2_xattr_set_ctxt *ctxt)
 {
@@ -4164,38 +4163,14 @@ static int ocfs2_add_new_xattr_cluster(struct inode *inode,
        handle_t *handle = ctxt->handle;
        struct ocfs2_super *osb = OCFS2_SB(inode->i_sb);
        struct ocfs2_extent_tree et;
-       struct ocfs2_xattr_bucket *first, *target;
 
        mlog(0, "Add new xattr cluster for %llu, previous xattr hash = %u, "
             "previous xattr blkno = %llu\n",
             (unsigned long long)OCFS2_I(inode)->ip_blkno,
-            prev_cpos, (unsigned long long)prev_blkno);
+            prev_cpos, (unsigned long long)bucket_blkno(first));
 
        ocfs2_init_xattr_tree_extent_tree(&et, inode, root_bh);
 
-       /* The first bucket of the original extent */
-       first = ocfs2_xattr_bucket_new(inode);
-       /* The target bucket for insert */
-       target = ocfs2_xattr_bucket_new(inode);
-       if (!first || !target) {
-               ret = -ENOMEM;
-               mlog_errno(ret);
-               goto leave;
-       }
-
-       BUG_ON(prev_blkno != (*first_bh)->b_blocknr);
-       ret = ocfs2_read_xattr_bucket(first, prev_blkno);
-       if (ret) {
-               mlog_errno(ret);
-               goto leave;
-       }
-
-       ret = ocfs2_read_xattr_bucket(target, (*header_bh)->b_blocknr);
-       if (ret) {
-               mlog_errno(ret);
-               goto leave;
-       }
-
        ret = ocfs2_journal_access(handle, inode, root_bh,
                                   OCFS2_JOURNAL_ACCESS_WRITE);
        if (ret < 0) {
@@ -4217,7 +4192,7 @@ static int ocfs2_add_new_xattr_cluster(struct inode *inode,
        mlog(0, "Allocating %u clusters at block %u for xattr in inode %llu\n",
             num_bits, bit_off, (unsigned long long)OCFS2_I(inode)->ip_blkno);
 
-       if (prev_blkno + prev_clusters * bpc == block &&
+       if (bucket_blkno(first) + (prev_clusters * bpc) == block &&
            (prev_clusters + num_bits) << osb->s_clustersize_bits <=
             OCFS2_MAX_XATTR_TREE_LEAF_SIZE) {
                /*
@@ -4246,17 +4221,6 @@ static int ocfs2_add_new_xattr_cluster(struct inode *inode,
                        mlog_errno(ret);
                        goto leave;
                }
-
-               /* Did first+target get moved? */
-               if (prev_blkno != bucket_blkno(first)) {
-                       brelse(*first_bh);
-                       *first_bh = first->bu_bhs[0];
-                       get_bh(*first_bh);
-
-                       brelse(*header_bh);
-                       *header_bh = target->bu_bhs[0];
-                       get_bh(*header_bh);
-               }
        }
 
        mlog(0, "Insert %u clusters at block %llu for xattr at %u\n",
@@ -4273,8 +4237,6 @@ static int ocfs2_add_new_xattr_cluster(struct inode *inode,
                mlog_errno(ret);
 
 leave:
-       ocfs2_xattr_bucket_free(first);
-       ocfs2_xattr_bucket_free(target);
        return ret;
 }
 
@@ -4357,16 +4319,16 @@ out:
  * We will move all the buckets starting from header_bh to the next place. As
  * for this one, half num of its xattrs will be moved to the next one.
  *
- * We will allocate a new cluster if current cluster is full and adjust
- * header_bh and first_bh if the insert place is moved to the new cluster.
+ * We will allocate a new cluster if current cluster is full.  The
+ * underlying calls will make sure that there is space at the target
+ * bucket, shifting buckets around if necessary.  'target' may be updated
+ * by those calls.
  */
 static int ocfs2_add_new_xattr_bucket(struct inode *inode,
                                      struct buffer_head *xb_bh,
                                      struct buffer_head *header_bh,
                                      struct ocfs2_xattr_set_ctxt *ctxt)
 {
-       struct ocfs2_xattr_header *first_xh = NULL;
-       struct buffer_head *first_bh = NULL;
        struct ocfs2_xattr_block *xb =
                        (struct ocfs2_xattr_block *)xb_bh->b_data;
        struct ocfs2_xattr_tree_root *xb_root = &xb->xb_attrs.xb_root;
@@ -4374,31 +4336,26 @@ static int ocfs2_add_new_xattr_bucket(struct inode *inode,
        struct ocfs2_xattr_header *xh =
                        (struct ocfs2_xattr_header *)header_bh->b_data;
        u32 name_hash = le32_to_cpu(xh->xh_entries[0].xe_name_hash);
-       struct super_block *sb = inode->i_sb;
-       struct ocfs2_super *osb = OCFS2_SB(sb);
+       struct ocfs2_super *osb = OCFS2_SB(inode->i_sb);
        int ret, num_buckets, extend = 1;
        u64 p_blkno;
        u32 e_cpos, num_clusters;
        /* The bucket at the front of the extent */
-       struct ocfs2_xattr_bucket *first;
+       struct ocfs2_xattr_bucket *first, *target;
 
        mlog(0, "Add new xattr bucket starting form %llu\n",
             (unsigned long long)header_bh->b_blocknr);
 
+       /* The first bucket of the original extent */
        first = ocfs2_xattr_bucket_new(inode);
-       if (!first) {
+       /* The target bucket for insert */
+       target = ocfs2_xattr_bucket_new(inode);
+       if (!first || !target) {
                ret = -ENOMEM;
                mlog_errno(ret);
                goto out;
        }
 
-       /*
-        * Add refrence for header_bh here because it may be
-        * changed in ocfs2_add_new_xattr_cluster and we need
-        * to free it in the end.
-        */
-       get_bh(header_bh);
-
        ret = ocfs2_xattr_get_rec(inode, name_hash, &p_blkno, &e_cpos,
                                  &num_clusters, el);
        if (ret) {
@@ -4406,23 +4363,30 @@ static int ocfs2_add_new_xattr_bucket(struct inode *inode,
                goto out;
        }
 
-       ret = ocfs2_read_block(inode, p_blkno, &first_bh, NULL);
+       ret = ocfs2_read_xattr_bucket(first, p_blkno);
        if (ret) {
                mlog_errno(ret);
                goto out;
        }
 
-       num_buckets = ocfs2_xattr_buckets_per_cluster(osb) * num_clusters;
-       first_xh = (struct ocfs2_xattr_header *)first_bh->b_data;
+       ret = ocfs2_read_xattr_bucket(target, header_bh->b_blocknr);
+       if (ret) {
+               mlog_errno(ret);
+               goto out;
+       }
 
-       if (num_buckets == le16_to_cpu(first_xh->xh_num_buckets)) {
+       num_buckets = ocfs2_xattr_buckets_per_cluster(osb) * num_clusters;
+       if (num_buckets == le16_to_cpu(bucket_xh(first)->xh_num_buckets)) {
+               /*
+                * This can move first+target if the target bucket moves
+                * to the new extent.
+                */
                ret = ocfs2_add_new_xattr_cluster(inode,
                                                  xb_bh,
-                                                 &first_bh,
-                                                 &header_bh,
+                                                 first,
+                                                 target,
                                                  &num_clusters,
                                                  e_cpos,
-                                                 p_blkno,
                                                  &extend,
                                                  ctxt);
                if (ret) {
@@ -4432,24 +4396,19 @@ static int ocfs2_add_new_xattr_bucket(struct inode *inode,
        }
 
        if (extend) {
-               /* These bucket reads should be cached */
-               ret = ocfs2_read_xattr_bucket(first, first_bh->b_blocknr);
-               if (ret) {
-                       mlog_errno(ret);
-                       goto out;
-               }
                ret = ocfs2_extend_xattr_bucket(inode,
                                                ctxt->handle,
-                                               first, header_bh->b_blocknr,
+                                               first,
+                                               bucket_blkno(target),
                                                num_clusters);
                if (ret)
                        mlog_errno(ret);
        }
 
 out:
-       brelse(first_bh);
-       brelse(header_bh);
        ocfs2_xattr_bucket_free(first);
+       ocfs2_xattr_bucket_free(target);
+
        return ret;
 }