Skip to content

Commit 6df8cdf

Browse files
ereshetovakdave
authored andcommitted
btrfs: convert btrfs_delayed_ref_node.refs from atomic_t to refcount_t
refcount_t type and corresponding API should be used instead of atomic_t when the variable is used as a reference counter. This allows to avoid accidental refcounter overflows that might lead to use-after-free situations. Signed-off-by: Elena Reshetova <elena.reshetova@intel.com> Signed-off-by: Hans Liljestrand <ishkamiel@gmail.com> Signed-off-by: Kees Cook <keescook@chromium.org> Signed-off-by: David Windsor <dwindsor@gmail.com> Signed-off-by: David Sterba <dsterba@suse.com>
1 parent 1e4f471 commit 6df8cdf

File tree

5 files changed

+14
-12
lines changed

5 files changed

+14
-12
lines changed

fs/btrfs/backref.c

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -1286,7 +1286,7 @@ static int find_parent_nodes(struct btrfs_trans_handle *trans,
12861286
head = btrfs_find_delayed_ref_head(delayed_refs, bytenr);
12871287
if (head) {
12881288
if (!mutex_trylock(&head->mutex)) {
1289-
atomic_inc(&head->node.refs);
1289+
refcount_inc(&head->node.refs);
12901290
spin_unlock(&delayed_refs->lock);
12911291

12921292
btrfs_release_path(path);

fs/btrfs/delayed-ref.c

Lines changed: 4 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -164,7 +164,7 @@ int btrfs_delayed_ref_lock(struct btrfs_trans_handle *trans,
164164
if (mutex_trylock(&head->mutex))
165165
return 0;
166166

167-
atomic_inc(&head->node.refs);
167+
refcount_inc(&head->node.refs);
168168
spin_unlock(&delayed_refs->lock);
169169

170170
mutex_lock(&head->mutex);
@@ -590,7 +590,7 @@ add_delayed_ref_head(struct btrfs_fs_info *fs_info,
590590
delayed_refs = &trans->transaction->delayed_refs;
591591

592592
/* first set the basic ref node struct up */
593-
atomic_set(&ref->refs, 1);
593+
refcount_set(&ref->refs, 1);
594594
ref->bytenr = bytenr;
595595
ref->num_bytes = num_bytes;
596596
ref->ref_mod = count_mod;
@@ -682,7 +682,7 @@ add_delayed_tree_ref(struct btrfs_fs_info *fs_info,
682682
delayed_refs = &trans->transaction->delayed_refs;
683683

684684
/* first set the basic ref node struct up */
685-
atomic_set(&ref->refs, 1);
685+
refcount_set(&ref->refs, 1);
686686
ref->bytenr = bytenr;
687687
ref->num_bytes = num_bytes;
688688
ref->ref_mod = 1;
@@ -739,7 +739,7 @@ add_delayed_data_ref(struct btrfs_fs_info *fs_info,
739739
seq = atomic64_read(&fs_info->tree_mod_seq);
740740

741741
/* first set the basic ref node struct up */
742-
atomic_set(&ref->refs, 1);
742+
refcount_set(&ref->refs, 1);
743743
ref->bytenr = bytenr;
744744
ref->num_bytes = num_bytes;
745745
ref->ref_mod = 1;

fs/btrfs/delayed-ref.h

Lines changed: 5 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,8 @@
1818
#ifndef __DELAYED_REF__
1919
#define __DELAYED_REF__
2020

21+
#include <linux/refcount.h>
22+
2123
/* these are the possible values of struct btrfs_delayed_ref_node->action */
2224
#define BTRFS_ADD_DELAYED_REF 1 /* add one backref to the tree */
2325
#define BTRFS_DROP_DELAYED_REF 2 /* delete one backref from the tree */
@@ -53,7 +55,7 @@ struct btrfs_delayed_ref_node {
5355
u64 seq;
5456

5557
/* ref count on this data structure */
56-
atomic_t refs;
58+
refcount_t refs;
5759

5860
/*
5961
* how many refs is this entry adding or deleting. For
@@ -220,8 +222,8 @@ btrfs_free_delayed_extent_op(struct btrfs_delayed_extent_op *op)
220222

221223
static inline void btrfs_put_delayed_ref(struct btrfs_delayed_ref_node *ref)
222224
{
223-
WARN_ON(atomic_read(&ref->refs) == 0);
224-
if (atomic_dec_and_test(&ref->refs)) {
225+
WARN_ON(refcount_read(&ref->refs) == 0);
226+
if (refcount_dec_and_test(&ref->refs)) {
225227
WARN_ON(ref->in_tree);
226228
switch (ref->type) {
227229
case BTRFS_TREE_BLOCK_REF_KEY:

fs/btrfs/disk-io.c

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -4343,7 +4343,7 @@ static int btrfs_destroy_delayed_refs(struct btrfs_transaction *trans,
43434343
head = rb_entry(node, struct btrfs_delayed_ref_head,
43444344
href_node);
43454345
if (!mutex_trylock(&head->mutex)) {
4346-
atomic_inc(&head->node.refs);
4346+
refcount_inc(&head->node.refs);
43474347
spin_unlock(&delayed_refs->lock);
43484348

43494349
mutex_lock(&head->mutex);

fs/btrfs/extent-tree.c

Lines changed: 3 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -892,7 +892,7 @@ int btrfs_lookup_extent_info(struct btrfs_trans_handle *trans,
892892
head = btrfs_find_delayed_ref_head(delayed_refs, bytenr);
893893
if (head) {
894894
if (!mutex_trylock(&head->mutex)) {
895-
atomic_inc(&head->node.refs);
895+
refcount_inc(&head->node.refs);
896896
spin_unlock(&delayed_refs->lock);
897897

898898
btrfs_release_path(path);
@@ -2980,7 +2980,7 @@ int btrfs_run_delayed_refs(struct btrfs_trans_handle *trans,
29802980
struct btrfs_delayed_ref_node *ref;
29812981

29822982
ref = &head->node;
2983-
atomic_inc(&ref->refs);
2983+
refcount_inc(&ref->refs);
29842984

29852985
spin_unlock(&delayed_refs->lock);
29862986
/*
@@ -3057,7 +3057,7 @@ static noinline int check_delayed_ref(struct btrfs_root *root,
30573057
}
30583058

30593059
if (!mutex_trylock(&head->mutex)) {
3060-
atomic_inc(&head->node.refs);
3060+
refcount_inc(&head->node.refs);
30613061
spin_unlock(&delayed_refs->lock);
30623062

30633063
btrfs_release_path(path);

0 commit comments

Comments
 (0)