diff options
-rw-r--r-- | fs/btrfs/ctree.h | 4 | ||||
-rw-r--r-- | fs/btrfs/extent-tree.c | 30 |
2 files changed, 28 insertions, 6 deletions
diff --git a/fs/btrfs/ctree.h b/fs/btrfs/ctree.h index 0d00a07b5b29..2e61fe1b6b8c 100644 --- a/fs/btrfs/ctree.h +++ b/fs/btrfs/ctree.h | |||
@@ -740,8 +740,10 @@ struct btrfs_space_info { | |||
740 | */ | 740 | */ |
741 | unsigned long reservation_progress; | 741 | unsigned long reservation_progress; |
742 | 742 | ||
743 | int full; /* indicates that we cannot allocate any more | 743 | int full:1; /* indicates that we cannot allocate any more |
744 | chunks for this space */ | 744 | chunks for this space */ |
745 | int chunk_alloc:1; /* set if we are allocating a chunk */ | ||
746 | |||
745 | int force_alloc; /* set if we need to force a chunk alloc for | 747 | int force_alloc; /* set if we need to force a chunk alloc for |
746 | this space */ | 748 | this space */ |
747 | 749 | ||
diff --git a/fs/btrfs/extent-tree.c b/fs/btrfs/extent-tree.c index 26479484180d..31f33ba56fe8 100644 --- a/fs/btrfs/extent-tree.c +++ b/fs/btrfs/extent-tree.c | |||
@@ -3039,6 +3039,7 @@ static int update_space_info(struct btrfs_fs_info *info, u64 flags, | |||
3039 | found->bytes_may_use = 0; | 3039 | found->bytes_may_use = 0; |
3040 | found->full = 0; | 3040 | found->full = 0; |
3041 | found->force_alloc = CHUNK_ALLOC_NO_FORCE; | 3041 | found->force_alloc = CHUNK_ALLOC_NO_FORCE; |
3042 | found->chunk_alloc = 0; | ||
3042 | *space_info = found; | 3043 | *space_info = found; |
3043 | list_add_rcu(&found->list, &info->space_info); | 3044 | list_add_rcu(&found->list, &info->space_info); |
3044 | atomic_set(&found->caching_threads, 0); | 3045 | atomic_set(&found->caching_threads, 0); |
@@ -3318,10 +3319,9 @@ static int do_chunk_alloc(struct btrfs_trans_handle *trans, | |||
3318 | { | 3319 | { |
3319 | struct btrfs_space_info *space_info; | 3320 | struct btrfs_space_info *space_info; |
3320 | struct btrfs_fs_info *fs_info = extent_root->fs_info; | 3321 | struct btrfs_fs_info *fs_info = extent_root->fs_info; |
3322 | int wait_for_alloc = 0; | ||
3321 | int ret = 0; | 3323 | int ret = 0; |
3322 | 3324 | ||
3323 | mutex_lock(&fs_info->chunk_mutex); | ||
3324 | |||
3325 | flags = btrfs_reduce_alloc_profile(extent_root, flags); | 3325 | flags = btrfs_reduce_alloc_profile(extent_root, flags); |
3326 | 3326 | ||
3327 | space_info = __find_space_info(extent_root->fs_info, flags); | 3327 | space_info = __find_space_info(extent_root->fs_info, flags); |
@@ -3332,21 +3332,40 @@ static int do_chunk_alloc(struct btrfs_trans_handle *trans, | |||
3332 | } | 3332 | } |
3333 | BUG_ON(!space_info); | 3333 | BUG_ON(!space_info); |
3334 | 3334 | ||
3335 | again: | ||
3335 | spin_lock(&space_info->lock); | 3336 | spin_lock(&space_info->lock); |
3336 | if (space_info->force_alloc) | 3337 | if (space_info->force_alloc) |
3337 | force = space_info->force_alloc; | 3338 | force = space_info->force_alloc; |
3338 | if (space_info->full) { | 3339 | if (space_info->full) { |
3339 | spin_unlock(&space_info->lock); | 3340 | spin_unlock(&space_info->lock); |
3340 | goto out; | 3341 | return 0; |
3341 | } | 3342 | } |
3342 | 3343 | ||
3343 | if (!should_alloc_chunk(extent_root, space_info, alloc_bytes, force)) { | 3344 | if (!should_alloc_chunk(extent_root, space_info, alloc_bytes, force)) { |
3344 | spin_unlock(&space_info->lock); | 3345 | spin_unlock(&space_info->lock); |
3345 | goto out; | 3346 | return 0; |
3347 | } else if (space_info->chunk_alloc) { | ||
3348 | wait_for_alloc = 1; | ||
3349 | } else { | ||
3350 | space_info->chunk_alloc = 1; | ||
3346 | } | 3351 | } |
3347 | 3352 | ||
3348 | spin_unlock(&space_info->lock); | 3353 | spin_unlock(&space_info->lock); |
3349 | 3354 | ||
3355 | mutex_lock(&fs_info->chunk_mutex); | ||
3356 | |||
3357 | /* | ||
3358 | * The chunk_mutex is held throughout the entirety of a chunk | ||
3359 | * allocation, so once we've acquired the chunk_mutex we know that the | ||
3360 | * other guy is done and we need to recheck and see if we should | ||
3361 | * allocate. | ||
3362 | */ | ||
3363 | if (wait_for_alloc) { | ||
3364 | mutex_unlock(&fs_info->chunk_mutex); | ||
3365 | wait_for_alloc = 0; | ||
3366 | goto again; | ||
3367 | } | ||
3368 | |||
3350 | /* | 3369 | /* |
3351 | * If we have mixed data/metadata chunks we want to make sure we keep | 3370 | * If we have mixed data/metadata chunks we want to make sure we keep |
3352 | * allocating mixed chunks instead of individual chunks. | 3371 | * allocating mixed chunks instead of individual chunks. |
@@ -3372,9 +3391,10 @@ static int do_chunk_alloc(struct btrfs_trans_handle *trans, | |||
3372 | space_info->full = 1; | 3391 | space_info->full = 1; |
3373 | else | 3392 | else |
3374 | ret = 1; | 3393 | ret = 1; |
3394 | |||
3375 | space_info->force_alloc = CHUNK_ALLOC_NO_FORCE; | 3395 | space_info->force_alloc = CHUNK_ALLOC_NO_FORCE; |
3396 | space_info->chunk_alloc = 0; | ||
3376 | spin_unlock(&space_info->lock); | 3397 | spin_unlock(&space_info->lock); |
3377 | out: | ||
3378 | mutex_unlock(&extent_root->fs_info->chunk_mutex); | 3398 | mutex_unlock(&extent_root->fs_info->chunk_mutex); |
3379 | return ret; | 3399 | return ret; |
3380 | } | 3400 | } |