diff options
author | Filipe Manana <fdmanana@suse.com> | 2015-11-19 11:45:48 +0000 |
---|---|---|
committer | Chris Mason <clm@fb.com> | 2015-11-25 05:22:08 -0800 |
commit | 758f2dfcf8a249b1f1510aa32e625c2ec20642a3 (patch) | |
tree | a521154fb1e3384087e4a828197f69eca158d280 /fs/btrfs/extent-tree.c | |
parent | 020d5b7366fc03e4bf84142ae6f63031ac504e33 (diff) | |
download | op-kernel-dev-758f2dfcf8a249b1f1510aa32e625c2ec20642a3.zip op-kernel-dev-758f2dfcf8a249b1f1510aa32e625c2ec20642a3.tar.gz |
Btrfs: fix scrub preventing unused block groups from being deleted
Currently scrub can race with the cleaner kthread when the later attempts
to delete an unused block group, and the result is preventing the cleaner
kthread from ever deleting later the block group - unless the block group
becomes used and unused again. The following diagram illustrates that
race:
CPU 1 CPU 2
cleaner kthread
btrfs_delete_unused_bgs()
gets block group X from
fs_info->unused_bgs and
removes it from that list
scrub_enumerate_chunks()
searches device tree using
its commit root
finds device extent for
block group X
gets block group X from the tree
fs_info->block_group_cache_tree
(via btrfs_lookup_block_group())
sets bg X to RO
sees the block group is
already RO and therefore
doesn't delete it nor adds
it back to unused list
So fix this by making scrub add the block group again to the list of
unused block groups if the block group is still unused when it finished
scrubbing it and it hasn't been removed already.
Signed-off-by: Filipe Manana <fdmanana@suse.com>
Signed-off-by: Chris Mason <clm@fb.com>
Diffstat (limited to 'fs/btrfs/extent-tree.c')
-rw-r--r-- | fs/btrfs/extent-tree.c | 2 |
1 files changed, 1 insertions, 1 deletions
diff --git a/fs/btrfs/extent-tree.c b/fs/btrfs/extent-tree.c index e97d6d6..8fd14b6 100644 --- a/fs/btrfs/extent-tree.c +++ b/fs/btrfs/extent-tree.c @@ -124,7 +124,7 @@ static int block_group_bits(struct btrfs_block_group_cache *cache, u64 bits) return (cache->flags & bits) == bits; } -static void btrfs_get_block_group(struct btrfs_block_group_cache *cache) +void btrfs_get_block_group(struct btrfs_block_group_cache *cache) { atomic_inc(&cache->count); } |