btrfs: handle space_info::total_bytes_pinned inside the delayed ref itself
authorJosef Bacik <josef@toxicpanda.com>
Fri, 15 Jan 2021 21:48:55 +0000 (16:48 -0500)
committerDavid Sterba <dsterba@suse.com>
Mon, 8 Feb 2021 21:58:55 +0000 (22:58 +0100)
Currently we pass things around to figure out if we maybe freeing data
based on the state of the delayed refs head.  This makes the accounting
sort of confusing and hard to follow, as it's distinctly separate from
the delayed ref heads stuff, but also depends on it entirely.

Fix this by explicitly adjusting the space_info->total_bytes_pinned in
the delayed refs code.  We now have two places where we modify this
counter, once where we create the delayed and destroy the delayed refs,
and once when we pin and unpin the extents.  This means there is a
slight overlap between delayed refs and the pin/unpin mechanisms, but
this is simply used by the ENOSPC infrastructure to determine if we need
to commit the transaction, so there's no adverse affect from this, we
might simply commit thinking it will give us enough space when it might
not.

CC: stable@vger.kernel.org # 5.10
Reviewed-by: Nikolay Borisov <nborisov@suse.com>
Signed-off-by: Josef Bacik <josef@toxicpanda.com>
Signed-off-by: David Sterba <dsterba@suse.com>
fs/btrfs/block-group.c
fs/btrfs/delayed-ref.c
fs/btrfs/delayed-ref.h
fs/btrfs/extent-tree.c
fs/btrfs/space-info.h

index 2d7294d816160d3c02f3f14a8c991ce05acc46e0..763a3671b7afcad8c9cf3bf4b545954d7ed844f4 100644 (file)
@@ -1371,9 +1371,7 @@ void btrfs_delete_unused_bgs(struct btrfs_fs_info *fs_info)
                btrfs_space_info_update_bytes_pinned(fs_info, space_info,
                                                     -block_group->pinned);
                space_info->bytes_readonly += block_group->pinned;
                btrfs_space_info_update_bytes_pinned(fs_info, space_info,
                                                     -block_group->pinned);
                space_info->bytes_readonly += block_group->pinned;
-               percpu_counter_add_batch(&space_info->total_bytes_pinned,
-                                  -block_group->pinned,
-                                  BTRFS_TOTAL_BYTES_PINNED_BATCH);
+               __btrfs_mod_total_bytes_pinned(space_info, -block_group->pinned);
                block_group->pinned = 0;
 
                spin_unlock(&block_group->lock);
                block_group->pinned = 0;
 
                spin_unlock(&block_group->lock);
@@ -2898,10 +2896,8 @@ int btrfs_update_block_group(struct btrfs_trans_handle *trans,
                        spin_unlock(&cache->lock);
                        spin_unlock(&cache->space_info->lock);
 
                        spin_unlock(&cache->lock);
                        spin_unlock(&cache->space_info->lock);
 
-                       percpu_counter_add_batch(
-                                       &cache->space_info->total_bytes_pinned,
-                                       num_bytes,
-                                       BTRFS_TOTAL_BYTES_PINNED_BATCH);
+                       __btrfs_mod_total_bytes_pinned(cache->space_info,
+                                                      num_bytes);
                        set_extent_dirty(&trans->transaction->pinned_extents,
                                         bytenr, bytenr + num_bytes - 1,
                                         GFP_NOFS | __GFP_NOFAIL);
                        set_extent_dirty(&trans->transaction->pinned_extents,
                                         bytenr, bytenr + num_bytes - 1,
                                         GFP_NOFS | __GFP_NOFAIL);
index 88a1e27d2fc286067c5fb8613c8e53d974e4fafc..a540ace3e03a0451883d19e5e4f03fa0ae7fe55e 100644 (file)
@@ -651,12 +651,12 @@ inserted:
  */
 static noinline void update_existing_head_ref(struct btrfs_trans_handle *trans,
                         struct btrfs_delayed_ref_head *existing,
  */
 static noinline void update_existing_head_ref(struct btrfs_trans_handle *trans,
                         struct btrfs_delayed_ref_head *existing,
-                        struct btrfs_delayed_ref_head *update,
-                        int *old_ref_mod_ret)
+                        struct btrfs_delayed_ref_head *update)
 {
        struct btrfs_delayed_ref_root *delayed_refs =
                &trans->transaction->delayed_refs;
        struct btrfs_fs_info *fs_info = trans->fs_info;
 {
        struct btrfs_delayed_ref_root *delayed_refs =
                &trans->transaction->delayed_refs;
        struct btrfs_fs_info *fs_info = trans->fs_info;
+       u64 flags = btrfs_ref_head_to_space_flags(existing);
        int old_ref_mod;
 
        BUG_ON(existing->is_data != update->is_data);
        int old_ref_mod;
 
        BUG_ON(existing->is_data != update->is_data);
@@ -704,8 +704,6 @@ static noinline void update_existing_head_ref(struct btrfs_trans_handle *trans,
         * currently, for refs we just added we know we're a-ok.
         */
        old_ref_mod = existing->total_ref_mod;
         * currently, for refs we just added we know we're a-ok.
         */
        old_ref_mod = existing->total_ref_mod;
-       if (old_ref_mod_ret)
-               *old_ref_mod_ret = old_ref_mod;
        existing->ref_mod += update->ref_mod;
        existing->total_ref_mod += update->ref_mod;
 
        existing->ref_mod += update->ref_mod;
        existing->total_ref_mod += update->ref_mod;
 
@@ -727,6 +725,22 @@ static noinline void update_existing_head_ref(struct btrfs_trans_handle *trans,
                        trans->delayed_ref_updates += csum_leaves;
                }
        }
                        trans->delayed_ref_updates += csum_leaves;
                }
        }
+
+       /*
+        * This handles the following conditions:
+        *
+        * 1. We had a ref mod of 0 or more and went negative, indicating that
+        *    we may be freeing space, so add our space to the
+        *    total_bytes_pinned counter.
+        * 2. We were negative and went to 0 or positive, so no longer can say
+        *    that the space would be pinned, decrement our counter from the
+        *    total_bytes_pinned counter.
+        */
+       if (existing->total_ref_mod < 0 && old_ref_mod >= 0)
+               btrfs_mod_total_bytes_pinned(fs_info, flags, existing->num_bytes);
+       else if (existing->total_ref_mod >= 0 && old_ref_mod < 0)
+               btrfs_mod_total_bytes_pinned(fs_info, flags, -existing->num_bytes);
+
        spin_unlock(&existing->lock);
 }
 
        spin_unlock(&existing->lock);
 }
 
@@ -801,8 +815,7 @@ static noinline struct btrfs_delayed_ref_head *
 add_delayed_ref_head(struct btrfs_trans_handle *trans,
                     struct btrfs_delayed_ref_head *head_ref,
                     struct btrfs_qgroup_extent_record *qrecord,
 add_delayed_ref_head(struct btrfs_trans_handle *trans,
                     struct btrfs_delayed_ref_head *head_ref,
                     struct btrfs_qgroup_extent_record *qrecord,
-                    int action, int *qrecord_inserted_ret,
-                    int *old_ref_mod, int *new_ref_mod)
+                    int action, int *qrecord_inserted_ret)
 {
        struct btrfs_delayed_ref_head *existing;
        struct btrfs_delayed_ref_root *delayed_refs;
 {
        struct btrfs_delayed_ref_head *existing;
        struct btrfs_delayed_ref_root *delayed_refs;
@@ -824,8 +837,7 @@ add_delayed_ref_head(struct btrfs_trans_handle *trans,
        existing = htree_insert(&delayed_refs->href_root,
                                &head_ref->href_node);
        if (existing) {
        existing = htree_insert(&delayed_refs->href_root,
                                &head_ref->href_node);
        if (existing) {
-               update_existing_head_ref(trans, existing, head_ref,
-                                        old_ref_mod);
+               update_existing_head_ref(trans, existing, head_ref);
                /*
                 * we've updated the existing ref, free the newly
                 * allocated ref
                /*
                 * we've updated the existing ref, free the newly
                 * allocated ref
@@ -833,14 +845,17 @@ add_delayed_ref_head(struct btrfs_trans_handle *trans,
                kmem_cache_free(btrfs_delayed_ref_head_cachep, head_ref);
                head_ref = existing;
        } else {
                kmem_cache_free(btrfs_delayed_ref_head_cachep, head_ref);
                head_ref = existing;
        } else {
-               if (old_ref_mod)
-                       *old_ref_mod = 0;
+               u64 flags = btrfs_ref_head_to_space_flags(head_ref);
+
                if (head_ref->is_data && head_ref->ref_mod < 0) {
                        delayed_refs->pending_csums += head_ref->num_bytes;
                        trans->delayed_ref_updates +=
                                btrfs_csum_bytes_to_leaves(trans->fs_info,
                                                           head_ref->num_bytes);
                }
                if (head_ref->is_data && head_ref->ref_mod < 0) {
                        delayed_refs->pending_csums += head_ref->num_bytes;
                        trans->delayed_ref_updates +=
                                btrfs_csum_bytes_to_leaves(trans->fs_info,
                                                           head_ref->num_bytes);
                }
+               if (head_ref->ref_mod < 0)
+                       btrfs_mod_total_bytes_pinned(trans->fs_info, flags,
+                                                    head_ref->num_bytes);
                delayed_refs->num_heads++;
                delayed_refs->num_heads_ready++;
                atomic_inc(&delayed_refs->num_entries);
                delayed_refs->num_heads++;
                delayed_refs->num_heads_ready++;
                atomic_inc(&delayed_refs->num_entries);
@@ -848,8 +863,6 @@ add_delayed_ref_head(struct btrfs_trans_handle *trans,
        }
        if (qrecord_inserted_ret)
                *qrecord_inserted_ret = qrecord_inserted;
        }
        if (qrecord_inserted_ret)
                *qrecord_inserted_ret = qrecord_inserted;
-       if (new_ref_mod)
-               *new_ref_mod = head_ref->total_ref_mod;
 
        return head_ref;
 }
 
        return head_ref;
 }
@@ -912,8 +925,7 @@ static void init_delayed_ref_common(struct btrfs_fs_info *fs_info,
  */
 int btrfs_add_delayed_tree_ref(struct btrfs_trans_handle *trans,
                               struct btrfs_ref *generic_ref,
  */
 int btrfs_add_delayed_tree_ref(struct btrfs_trans_handle *trans,
                               struct btrfs_ref *generic_ref,
-                              struct btrfs_delayed_extent_op *extent_op,
-                              int *old_ref_mod, int *new_ref_mod)
+                              struct btrfs_delayed_extent_op *extent_op)
 {
        struct btrfs_fs_info *fs_info = trans->fs_info;
        struct btrfs_delayed_tree_ref *ref;
 {
        struct btrfs_fs_info *fs_info = trans->fs_info;
        struct btrfs_delayed_tree_ref *ref;
@@ -980,8 +992,7 @@ int btrfs_add_delayed_tree_ref(struct btrfs_trans_handle *trans,
         * the spin lock
         */
        head_ref = add_delayed_ref_head(trans, head_ref, record,
         * the spin lock
         */
        head_ref = add_delayed_ref_head(trans, head_ref, record,
-                                       action, &qrecord_inserted,
-                                       old_ref_mod, new_ref_mod);
+                                       action, &qrecord_inserted);
 
        ret = insert_delayed_ref(trans, delayed_refs, head_ref, &ref->node);
        spin_unlock(&delayed_refs->lock);
 
        ret = insert_delayed_ref(trans, delayed_refs, head_ref, &ref->node);
        spin_unlock(&delayed_refs->lock);
@@ -1009,8 +1020,7 @@ int btrfs_add_delayed_tree_ref(struct btrfs_trans_handle *trans,
  */
 int btrfs_add_delayed_data_ref(struct btrfs_trans_handle *trans,
                               struct btrfs_ref *generic_ref,
  */
 int btrfs_add_delayed_data_ref(struct btrfs_trans_handle *trans,
                               struct btrfs_ref *generic_ref,
-                              u64 reserved, int *old_ref_mod,
-                              int *new_ref_mod)
+                              u64 reserved)
 {
        struct btrfs_fs_info *fs_info = trans->fs_info;
        struct btrfs_delayed_data_ref *ref;
 {
        struct btrfs_fs_info *fs_info = trans->fs_info;
        struct btrfs_delayed_data_ref *ref;
@@ -1076,8 +1086,7 @@ int btrfs_add_delayed_data_ref(struct btrfs_trans_handle *trans,
         * the spin lock
         */
        head_ref = add_delayed_ref_head(trans, head_ref, record,
         * the spin lock
         */
        head_ref = add_delayed_ref_head(trans, head_ref, record,
-                                       action, &qrecord_inserted,
-                                       old_ref_mod, new_ref_mod);
+                                       action, &qrecord_inserted);
 
        ret = insert_delayed_ref(trans, delayed_refs, head_ref, &ref->node);
        spin_unlock(&delayed_refs->lock);
 
        ret = insert_delayed_ref(trans, delayed_refs, head_ref, &ref->node);
        spin_unlock(&delayed_refs->lock);
@@ -1120,7 +1129,7 @@ int btrfs_add_delayed_extent_op(struct btrfs_trans_handle *trans,
        spin_lock(&delayed_refs->lock);
 
        add_delayed_ref_head(trans, head_ref, NULL, BTRFS_UPDATE_DELAYED_HEAD,
        spin_lock(&delayed_refs->lock);
 
        add_delayed_ref_head(trans, head_ref, NULL, BTRFS_UPDATE_DELAYED_HEAD,
-                            NULL, NULL, NULL);
+                            NULL);
 
        spin_unlock(&delayed_refs->lock);
 
 
        spin_unlock(&delayed_refs->lock);
 
index 1c977e6d45dc3157e6f1424d47b5daa5d76cd8c8..3ba140468f12635b096a92a5fa22a26306e5eaf7 100644 (file)
@@ -326,6 +326,16 @@ static inline void btrfs_put_delayed_ref(struct btrfs_delayed_ref_node *ref)
        }
 }
 
        }
 }
 
+static inline u64 btrfs_ref_head_to_space_flags(
+                               struct btrfs_delayed_ref_head *head_ref)
+{
+       if (head_ref->is_data)
+               return BTRFS_BLOCK_GROUP_DATA;
+       else if (head_ref->is_system)
+               return BTRFS_BLOCK_GROUP_SYSTEM;
+       return BTRFS_BLOCK_GROUP_METADATA;
+}
+
 static inline void btrfs_put_delayed_ref_head(struct btrfs_delayed_ref_head *head)
 {
        if (refcount_dec_and_test(&head->refs))
 static inline void btrfs_put_delayed_ref_head(struct btrfs_delayed_ref_head *head)
 {
        if (refcount_dec_and_test(&head->refs))
@@ -334,12 +344,10 @@ static inline void btrfs_put_delayed_ref_head(struct btrfs_delayed_ref_head *hea
 
 int btrfs_add_delayed_tree_ref(struct btrfs_trans_handle *trans,
                               struct btrfs_ref *generic_ref,
 
 int btrfs_add_delayed_tree_ref(struct btrfs_trans_handle *trans,
                               struct btrfs_ref *generic_ref,
-                              struct btrfs_delayed_extent_op *extent_op,
-                              int *old_ref_mod, int *new_ref_mod);
+                              struct btrfs_delayed_extent_op *extent_op);
 int btrfs_add_delayed_data_ref(struct btrfs_trans_handle *trans,
                               struct btrfs_ref *generic_ref,
 int btrfs_add_delayed_data_ref(struct btrfs_trans_handle *trans,
                               struct btrfs_ref *generic_ref,
-                              u64 reserved, int *old_ref_mod,
-                              int *new_ref_mod);
+                              u64 reserved);
 int btrfs_add_delayed_extent_op(struct btrfs_trans_handle *trans,
                                u64 bytenr, u64 num_bytes,
                                struct btrfs_delayed_extent_op *extent_op);
 int btrfs_add_delayed_extent_op(struct btrfs_trans_handle *trans,
                                u64 bytenr, u64 num_bytes,
                                struct btrfs_delayed_extent_op *extent_op);
index 0c335dae5af7adca7987ec8c7fcb5343c9a6f6f9..2f591036ffc1b58c972d603ea1a8a0dd9e257141 100644 (file)
@@ -82,41 +82,6 @@ void btrfs_free_excluded_extents(struct btrfs_block_group *cache)
                          EXTENT_UPTODATE);
 }
 
                          EXTENT_UPTODATE);
 }
 
-static u64 generic_ref_to_space_flags(struct btrfs_ref *ref)
-{
-       if (ref->type == BTRFS_REF_METADATA) {
-               if (ref->tree_ref.root == BTRFS_CHUNK_TREE_OBJECTID)
-                       return BTRFS_BLOCK_GROUP_SYSTEM;
-               else
-                       return BTRFS_BLOCK_GROUP_METADATA;
-       }
-       return BTRFS_BLOCK_GROUP_DATA;
-}
-
-static void add_pinned_bytes(struct btrfs_fs_info *fs_info,
-                            struct btrfs_ref *ref)
-{
-       struct btrfs_space_info *space_info;
-       u64 flags = generic_ref_to_space_flags(ref);
-
-       space_info = btrfs_find_space_info(fs_info, flags);
-       ASSERT(space_info);
-       percpu_counter_add_batch(&space_info->total_bytes_pinned, ref->len,
-                   BTRFS_TOTAL_BYTES_PINNED_BATCH);
-}
-
-static void sub_pinned_bytes(struct btrfs_fs_info *fs_info,
-                            struct btrfs_ref *ref)
-{
-       struct btrfs_space_info *space_info;
-       u64 flags = generic_ref_to_space_flags(ref);
-
-       space_info = btrfs_find_space_info(fs_info, flags);
-       ASSERT(space_info);
-       percpu_counter_add_batch(&space_info->total_bytes_pinned, -ref->len,
-                   BTRFS_TOTAL_BYTES_PINNED_BATCH);
-}
-
 /* simple helper to search for an existing data extent at a given offset */
 int btrfs_lookup_data_extent(struct btrfs_fs_info *fs_info, u64 start, u64 len)
 {
 /* simple helper to search for an existing data extent at a given offset */
 int btrfs_lookup_data_extent(struct btrfs_fs_info *fs_info, u64 start, u64 len)
 {
@@ -1388,7 +1353,6 @@ int btrfs_inc_extent_ref(struct btrfs_trans_handle *trans,
                         struct btrfs_ref *generic_ref)
 {
        struct btrfs_fs_info *fs_info = trans->fs_info;
                         struct btrfs_ref *generic_ref)
 {
        struct btrfs_fs_info *fs_info = trans->fs_info;
-       int old_ref_mod, new_ref_mod;
        int ret;
 
        ASSERT(generic_ref->type != BTRFS_REF_NOT_SET &&
        int ret;
 
        ASSERT(generic_ref->type != BTRFS_REF_NOT_SET &&
@@ -1397,17 +1361,12 @@ int btrfs_inc_extent_ref(struct btrfs_trans_handle *trans,
               generic_ref->tree_ref.root == BTRFS_TREE_LOG_OBJECTID);
 
        if (generic_ref->type == BTRFS_REF_METADATA)
               generic_ref->tree_ref.root == BTRFS_TREE_LOG_OBJECTID);
 
        if (generic_ref->type == BTRFS_REF_METADATA)
-               ret = btrfs_add_delayed_tree_ref(trans, generic_ref,
-                               NULL, &old_ref_mod, &new_ref_mod);
+               ret = btrfs_add_delayed_tree_ref(trans, generic_ref, NULL);
        else
        else
-               ret = btrfs_add_delayed_data_ref(trans, generic_ref, 0,
-                                                &old_ref_mod, &new_ref_mod);
+               ret = btrfs_add_delayed_data_ref(trans, generic_ref, 0);
 
        btrfs_ref_tree_mod(fs_info, generic_ref);
 
 
        btrfs_ref_tree_mod(fs_info, generic_ref);
 
-       if (ret == 0 && old_ref_mod < 0 && new_ref_mod >= 0)
-               sub_pinned_bytes(fs_info, generic_ref);
-
        return ret;
 }
 
        return ret;
 }
 
@@ -1796,20 +1755,9 @@ void btrfs_cleanup_ref_head_accounting(struct btrfs_fs_info *fs_info,
        int nr_items = 1;       /* Dropping this ref head update. */
 
        if (head->total_ref_mod < 0) {
        int nr_items = 1;       /* Dropping this ref head update. */
 
        if (head->total_ref_mod < 0) {
-               struct btrfs_space_info *space_info;
-               u64 flags;
+               u64 flags = btrfs_ref_head_to_space_flags(head);
 
 
-               if (head->is_data)
-                       flags = BTRFS_BLOCK_GROUP_DATA;
-               else if (head->is_system)
-                       flags = BTRFS_BLOCK_GROUP_SYSTEM;
-               else
-                       flags = BTRFS_BLOCK_GROUP_METADATA;
-               space_info = btrfs_find_space_info(fs_info, flags);
-               ASSERT(space_info);
-               percpu_counter_add_batch(&space_info->total_bytes_pinned,
-                                  -head->num_bytes,
-                                  BTRFS_TOTAL_BYTES_PINNED_BATCH);
+               btrfs_mod_total_bytes_pinned(fs_info, flags, -head->num_bytes);
 
                /*
                 * We had csum deletions accounted for in our delayed refs rsv,
 
                /*
                 * We had csum deletions accounted for in our delayed refs rsv,
@@ -2572,8 +2520,7 @@ static int pin_down_extent(struct btrfs_trans_handle *trans,
        spin_unlock(&cache->lock);
        spin_unlock(&cache->space_info->lock);
 
        spin_unlock(&cache->lock);
        spin_unlock(&cache->space_info->lock);
 
-       percpu_counter_add_batch(&cache->space_info->total_bytes_pinned,
-                   num_bytes, BTRFS_TOTAL_BYTES_PINNED_BATCH);
+       __btrfs_mod_total_bytes_pinned(cache->space_info, num_bytes);
        set_extent_dirty(&trans->transaction->pinned_extents, bytenr,
                         bytenr + num_bytes - 1, GFP_NOFS | __GFP_NOFAIL);
        return 0;
        set_extent_dirty(&trans->transaction->pinned_extents, bytenr,
                         bytenr + num_bytes - 1, GFP_NOFS | __GFP_NOFAIL);
        return 0;
@@ -2784,8 +2731,7 @@ static int unpin_extent_range(struct btrfs_fs_info *fs_info,
                cache->pinned -= len;
                btrfs_space_info_update_bytes_pinned(fs_info, space_info, -len);
                space_info->max_extent_size = 0;
                cache->pinned -= len;
                btrfs_space_info_update_bytes_pinned(fs_info, space_info, -len);
                space_info->max_extent_size = 0;
-               percpu_counter_add_batch(&space_info->total_bytes_pinned,
-                           -len, BTRFS_TOTAL_BYTES_PINNED_BATCH);
+               __btrfs_mod_total_bytes_pinned(space_info, -len);
                if (cache->ro) {
                        space_info->bytes_readonly += len;
                        readonly = true;
                if (cache->ro) {
                        space_info->bytes_readonly += len;
                        readonly = true;
@@ -3318,7 +3264,6 @@ void btrfs_free_tree_block(struct btrfs_trans_handle *trans,
 {
        struct btrfs_fs_info *fs_info = root->fs_info;
        struct btrfs_ref generic_ref = { 0 };
 {
        struct btrfs_fs_info *fs_info = root->fs_info;
        struct btrfs_ref generic_ref = { 0 };
-       int pin = 1;
        int ret;
 
        btrfs_init_generic_ref(&generic_ref, BTRFS_DROP_DELAYED_REF,
        int ret;
 
        btrfs_init_generic_ref(&generic_ref, BTRFS_DROP_DELAYED_REF,
@@ -3327,13 +3272,9 @@ void btrfs_free_tree_block(struct btrfs_trans_handle *trans,
                            root->root_key.objectid);
 
        if (root->root_key.objectid != BTRFS_TREE_LOG_OBJECTID) {
                            root->root_key.objectid);
 
        if (root->root_key.objectid != BTRFS_TREE_LOG_OBJECTID) {
-               int old_ref_mod, new_ref_mod;
-
                btrfs_ref_tree_mod(fs_info, &generic_ref);
                btrfs_ref_tree_mod(fs_info, &generic_ref);
-               ret = btrfs_add_delayed_tree_ref(trans, &generic_ref, NULL,
-                                                &old_ref_mod, &new_ref_mod);
+               ret = btrfs_add_delayed_tree_ref(trans, &generic_ref, NULL);
                BUG_ON(ret); /* -ENOMEM */
                BUG_ON(ret); /* -ENOMEM */
-               pin = old_ref_mod >= 0 && new_ref_mod < 0;
        }
 
        if (last_ref && btrfs_header_generation(buf) == trans->transid) {
        }
 
        if (last_ref && btrfs_header_generation(buf) == trans->transid) {
@@ -3345,7 +3286,6 @@ void btrfs_free_tree_block(struct btrfs_trans_handle *trans,
                                goto out;
                }
 
                                goto out;
                }
 
-               pin = 0;
                cache = btrfs_lookup_block_group(fs_info, buf->start);
 
                if (btrfs_header_flag(buf, BTRFS_HEADER_FLAG_WRITTEN)) {
                cache = btrfs_lookup_block_group(fs_info, buf->start);
 
                if (btrfs_header_flag(buf, BTRFS_HEADER_FLAG_WRITTEN)) {
@@ -3362,9 +3302,6 @@ void btrfs_free_tree_block(struct btrfs_trans_handle *trans,
                trace_btrfs_reserved_extent_free(fs_info, buf->start, buf->len);
        }
 out:
                trace_btrfs_reserved_extent_free(fs_info, buf->start, buf->len);
        }
 out:
-       if (pin)
-               add_pinned_bytes(fs_info, &generic_ref);
-
        if (last_ref) {
                /*
                 * Deleting the buffer, clear the corrupt flag since it doesn't
        if (last_ref) {
                /*
                 * Deleting the buffer, clear the corrupt flag since it doesn't
@@ -3378,7 +3315,6 @@ out:
 int btrfs_free_extent(struct btrfs_trans_handle *trans, struct btrfs_ref *ref)
 {
        struct btrfs_fs_info *fs_info = trans->fs_info;
 int btrfs_free_extent(struct btrfs_trans_handle *trans, struct btrfs_ref *ref)
 {
        struct btrfs_fs_info *fs_info = trans->fs_info;
-       int old_ref_mod, new_ref_mod;
        int ret;
 
        if (btrfs_is_testing(fs_info))
        int ret;
 
        if (btrfs_is_testing(fs_info))
@@ -3394,14 +3330,11 @@ int btrfs_free_extent(struct btrfs_trans_handle *trans, struct btrfs_ref *ref)
             ref->data_ref.ref_root == BTRFS_TREE_LOG_OBJECTID)) {
                /* unlocks the pinned mutex */
                btrfs_pin_extent(trans, ref->bytenr, ref->len, 1);
             ref->data_ref.ref_root == BTRFS_TREE_LOG_OBJECTID)) {
                /* unlocks the pinned mutex */
                btrfs_pin_extent(trans, ref->bytenr, ref->len, 1);
-               old_ref_mod = new_ref_mod = 0;
                ret = 0;
        } else if (ref->type == BTRFS_REF_METADATA) {
                ret = 0;
        } else if (ref->type == BTRFS_REF_METADATA) {
-               ret = btrfs_add_delayed_tree_ref(trans, ref, NULL,
-                                                &old_ref_mod, &new_ref_mod);
+               ret = btrfs_add_delayed_tree_ref(trans, ref, NULL);
        } else {
        } else {
-               ret = btrfs_add_delayed_data_ref(trans, ref, 0,
-                                                &old_ref_mod, &new_ref_mod);
+               ret = btrfs_add_delayed_data_ref(trans, ref, 0);
        }
 
        if (!((ref->type == BTRFS_REF_METADATA &&
        }
 
        if (!((ref->type == BTRFS_REF_METADATA &&
@@ -3410,9 +3343,6 @@ int btrfs_free_extent(struct btrfs_trans_handle *trans, struct btrfs_ref *ref)
               ref->data_ref.ref_root == BTRFS_TREE_LOG_OBJECTID)))
                btrfs_ref_tree_mod(fs_info, ref);
 
               ref->data_ref.ref_root == BTRFS_TREE_LOG_OBJECTID)))
                btrfs_ref_tree_mod(fs_info, ref);
 
-       if (ret == 0 && old_ref_mod >= 0 && new_ref_mod < 0)
-               add_pinned_bytes(fs_info, ref);
-
        return ret;
 }
 
        return ret;
 }
 
@@ -4528,7 +4458,6 @@ int btrfs_alloc_reserved_file_extent(struct btrfs_trans_handle *trans,
                                     struct btrfs_key *ins)
 {
        struct btrfs_ref generic_ref = { 0 };
                                     struct btrfs_key *ins)
 {
        struct btrfs_ref generic_ref = { 0 };
-       int ret;
 
        BUG_ON(root->root_key.objectid == BTRFS_TREE_LOG_OBJECTID);
 
 
        BUG_ON(root->root_key.objectid == BTRFS_TREE_LOG_OBJECTID);
 
@@ -4536,9 +4465,8 @@ int btrfs_alloc_reserved_file_extent(struct btrfs_trans_handle *trans,
                               ins->objectid, ins->offset, 0);
        btrfs_init_data_ref(&generic_ref, root->root_key.objectid, owner, offset);
        btrfs_ref_tree_mod(root->fs_info, &generic_ref);
                               ins->objectid, ins->offset, 0);
        btrfs_init_data_ref(&generic_ref, root->root_key.objectid, owner, offset);
        btrfs_ref_tree_mod(root->fs_info, &generic_ref);
-       ret = btrfs_add_delayed_data_ref(trans, &generic_ref,
-                                        ram_bytes, NULL, NULL);
-       return ret;
+
+       return btrfs_add_delayed_data_ref(trans, &generic_ref, ram_bytes);
 }
 
 /*
 }
 
 /*
@@ -4730,8 +4658,7 @@ struct extent_buffer *btrfs_alloc_tree_block(struct btrfs_trans_handle *trans,
                generic_ref.real_root = root->root_key.objectid;
                btrfs_init_tree_ref(&generic_ref, level, root_objectid);
                btrfs_ref_tree_mod(fs_info, &generic_ref);
                generic_ref.real_root = root->root_key.objectid;
                btrfs_init_tree_ref(&generic_ref, level, root_objectid);
                btrfs_ref_tree_mod(fs_info, &generic_ref);
-               ret = btrfs_add_delayed_tree_ref(trans, &generic_ref,
-                                                extent_op, NULL, NULL);
+               ret = btrfs_add_delayed_tree_ref(trans, &generic_ref, extent_op);
                if (ret)
                        goto out_free_delayed;
        }
                if (ret)
                        goto out_free_delayed;
        }
index 5646393b928c9c8123dd5965158ec6782f5f572f..74706f604bce157ff5ec22c00187ed490fc21dae 100644 (file)
@@ -152,4 +152,21 @@ static inline void btrfs_space_info_free_bytes_may_use(
 int btrfs_reserve_data_bytes(struct btrfs_fs_info *fs_info, u64 bytes,
                             enum btrfs_reserve_flush_enum flush);
 
 int btrfs_reserve_data_bytes(struct btrfs_fs_info *fs_info, u64 bytes,
                             enum btrfs_reserve_flush_enum flush);
 
+static inline void __btrfs_mod_total_bytes_pinned(
+                                       struct btrfs_space_info *space_info,
+                                       s64 mod)
+{
+       percpu_counter_add_batch(&space_info->total_bytes_pinned, mod,
+                                BTRFS_TOTAL_BYTES_PINNED_BATCH);
+}
+
+static inline void btrfs_mod_total_bytes_pinned(struct btrfs_fs_info *fs_info,
+                                               u64 flags, s64 mod)
+{
+       struct btrfs_space_info *space_info = btrfs_find_space_info(fs_info, flags);
+
+       ASSERT(space_info);
+       __btrfs_mod_total_bytes_pinned(space_info, mod);
+}
+
 #endif /* BTRFS_SPACE_INFO_H */
 #endif /* BTRFS_SPACE_INFO_H */