aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--fs/btrfs/ctree.h9
-rw-r--r--fs/btrfs/extent-tree.c35
2 files changed, 32 insertions, 12 deletions
diff --git a/fs/btrfs/ctree.h b/fs/btrfs/ctree.h
index 28188a786da..8b4b9d158a0 100644
--- a/fs/btrfs/ctree.h
+++ b/fs/btrfs/ctree.h
@@ -729,6 +729,15 @@ struct btrfs_space_info {
729 u64 disk_total; /* total bytes on disk, takes mirrors into 729 u64 disk_total; /* total bytes on disk, takes mirrors into
730 account */ 730 account */
731 731
732 /*
733 * we bump reservation progress every time we decrement
734 * bytes_reserved. This way people waiting for reservations
735 * know something good has happened and they can check
736 * for progress. The number here isn't to be trusted, it
737 * just shows reclaim activity
738 */
739 unsigned long reservation_progress;
740
732 int full; /* indicates that we cannot allocate any more 741 int full; /* indicates that we cannot allocate any more
733 chunks for this space */ 742 chunks for this space */
734 int force_alloc; /* set if we need to force a chunk alloc for 743 int force_alloc; /* set if we need to force a chunk alloc for
diff --git a/fs/btrfs/extent-tree.c b/fs/btrfs/extent-tree.c
index 100e409e905..f1db57d4a01 100644
--- a/fs/btrfs/extent-tree.c
+++ b/fs/btrfs/extent-tree.c
@@ -3343,15 +3343,16 @@ static int shrink_delalloc(struct btrfs_trans_handle *trans,
3343 u64 max_reclaim; 3343 u64 max_reclaim;
3344 u64 reclaimed = 0; 3344 u64 reclaimed = 0;
3345 long time_left; 3345 long time_left;
3346 int pause = 1;
3347 int nr_pages = (2 * 1024 * 1024) >> PAGE_CACHE_SHIFT; 3346 int nr_pages = (2 * 1024 * 1024) >> PAGE_CACHE_SHIFT;
3348 int loops = 0; 3347 int loops = 0;
3348 unsigned long progress;
3349 3349
3350 block_rsv = &root->fs_info->delalloc_block_rsv; 3350 block_rsv = &root->fs_info->delalloc_block_rsv;
3351 space_info = block_rsv->space_info; 3351 space_info = block_rsv->space_info;
3352 3352
3353 smp_mb(); 3353 smp_mb();
3354 reserved = space_info->bytes_reserved; 3354 reserved = space_info->bytes_reserved;
3355 progress = space_info->reservation_progress;
3355 3356
3356 if (reserved == 0) 3357 if (reserved == 0)
3357 return 0; 3358 return 0;
@@ -3366,31 +3367,36 @@ static int shrink_delalloc(struct btrfs_trans_handle *trans,
3366 writeback_inodes_sb_nr_if_idle(root->fs_info->sb, nr_pages); 3367 writeback_inodes_sb_nr_if_idle(root->fs_info->sb, nr_pages);
3367 3368
3368 spin_lock(&space_info->lock); 3369 spin_lock(&space_info->lock);
3369 if (reserved > space_info->bytes_reserved) { 3370 if (reserved > space_info->bytes_reserved)
3370 loops = 0;
3371 reclaimed += reserved - space_info->bytes_reserved; 3371 reclaimed += reserved - space_info->bytes_reserved;
3372 } else {
3373 loops++;
3374 }
3375 reserved = space_info->bytes_reserved; 3372 reserved = space_info->bytes_reserved;
3376 spin_unlock(&space_info->lock); 3373 spin_unlock(&space_info->lock);
3377 3374
3375 loops++;
3376
3378 if (reserved == 0 || reclaimed >= max_reclaim) 3377 if (reserved == 0 || reclaimed >= max_reclaim)
3379 break; 3378 break;
3380 3379
3381 if (trans && trans->transaction->blocked) 3380 if (trans && trans->transaction->blocked)
3382 return -EAGAIN; 3381 return -EAGAIN;
3383 3382
3384 __set_current_state(TASK_INTERRUPTIBLE); 3383 time_left = schedule_timeout_interruptible(1);
3385 time_left = schedule_timeout(pause);
3386 3384
3387 /* We were interrupted, exit */ 3385 /* We were interrupted, exit */
3388 if (time_left) 3386 if (time_left)
3389 break; 3387 break;
3390 3388
3391 pause <<= 1; 3389 /* we've kicked the IO a few times, if anything has been freed,
3392 if (pause > HZ / 10) 3390 * exit. There is no sense in looping here for a long time
3393 pause = HZ / 10; 3391 * when we really need to commit the transaction, or there are
3392 * just too many writers without enough free space
3393 */
3394
3395 if (loops > 3) {
3396 smp_mb();
3397 if (progress != space_info->reservation_progress)
3398 break;
3399 }
3394 3400
3395 } 3401 }
3396 return reclaimed >= to_reclaim; 3402 return reclaimed >= to_reclaim;
@@ -3613,6 +3619,7 @@ void block_rsv_release_bytes(struct btrfs_block_rsv *block_rsv,
3613 if (num_bytes) { 3619 if (num_bytes) {
3614 spin_lock(&space_info->lock); 3620 spin_lock(&space_info->lock);
3615 space_info->bytes_reserved -= num_bytes; 3621 space_info->bytes_reserved -= num_bytes;
3622 space_info->reservation_progress++;
3616 spin_unlock(&space_info->lock); 3623 spin_unlock(&space_info->lock);
3617 } 3624 }
3618 } 3625 }
@@ -3845,6 +3852,7 @@ static void update_global_block_rsv(struct btrfs_fs_info *fs_info)
3845 if (block_rsv->reserved >= block_rsv->size) { 3852 if (block_rsv->reserved >= block_rsv->size) {
3846 num_bytes = block_rsv->reserved - block_rsv->size; 3853 num_bytes = block_rsv->reserved - block_rsv->size;
3847 sinfo->bytes_reserved -= num_bytes; 3854 sinfo->bytes_reserved -= num_bytes;
3855 sinfo->reservation_progress++;
3848 block_rsv->reserved = block_rsv->size; 3856 block_rsv->reserved = block_rsv->size;
3849 block_rsv->full = 1; 3857 block_rsv->full = 1;
3850 } 3858 }
@@ -4006,7 +4014,6 @@ int btrfs_delalloc_reserve_metadata(struct inode *inode, u64 num_bytes)
4006 to_reserve = 0; 4014 to_reserve = 0;
4007 } 4015 }
4008 spin_unlock(&BTRFS_I(inode)->accounting_lock); 4016 spin_unlock(&BTRFS_I(inode)->accounting_lock);
4009
4010 to_reserve += calc_csum_metadata_size(inode, num_bytes); 4017 to_reserve += calc_csum_metadata_size(inode, num_bytes);
4011 ret = reserve_metadata_bytes(NULL, root, block_rsv, to_reserve, 1); 4018 ret = reserve_metadata_bytes(NULL, root, block_rsv, to_reserve, 1);
4012 if (ret) 4019 if (ret)
@@ -4134,6 +4141,7 @@ static int update_block_group(struct btrfs_trans_handle *trans,
4134 btrfs_set_block_group_used(&cache->item, old_val); 4141 btrfs_set_block_group_used(&cache->item, old_val);
4135 cache->reserved -= num_bytes; 4142 cache->reserved -= num_bytes;
4136 cache->space_info->bytes_reserved -= num_bytes; 4143 cache->space_info->bytes_reserved -= num_bytes;
4144 cache->space_info->reservation_progress++;
4137 cache->space_info->bytes_used += num_bytes; 4145 cache->space_info->bytes_used += num_bytes;
4138 cache->space_info->disk_used += num_bytes * factor; 4146 cache->space_info->disk_used += num_bytes * factor;
4139 spin_unlock(&cache->lock); 4147 spin_unlock(&cache->lock);
@@ -4185,6 +4193,7 @@ static int pin_down_extent(struct btrfs_root *root,
4185 if (reserved) { 4193 if (reserved) {
4186 cache->reserved -= num_bytes; 4194 cache->reserved -= num_bytes;
4187 cache->space_info->bytes_reserved -= num_bytes; 4195 cache->space_info->bytes_reserved -= num_bytes;
4196 cache->space_info->reservation_progress++;
4188 } 4197 }
4189 spin_unlock(&cache->lock); 4198 spin_unlock(&cache->lock);
4190 spin_unlock(&cache->space_info->lock); 4199 spin_unlock(&cache->space_info->lock);
@@ -4235,6 +4244,7 @@ static int update_reserved_bytes(struct btrfs_block_group_cache *cache,
4235 space_info->bytes_readonly += num_bytes; 4244 space_info->bytes_readonly += num_bytes;
4236 cache->reserved -= num_bytes; 4245 cache->reserved -= num_bytes;
4237 space_info->bytes_reserved -= num_bytes; 4246 space_info->bytes_reserved -= num_bytes;
4247 space_info->reservation_progress++;
4238 } 4248 }
4239 spin_unlock(&cache->lock); 4249 spin_unlock(&cache->lock);
4240 spin_unlock(&space_info->lock); 4250 spin_unlock(&space_info->lock);
@@ -4713,6 +4723,7 @@ void btrfs_free_tree_block(struct btrfs_trans_handle *trans,
4713 if (ret) { 4723 if (ret) {
4714 spin_lock(&cache->space_info->lock); 4724 spin_lock(&cache->space_info->lock);
4715 cache->space_info->bytes_reserved -= buf->len; 4725 cache->space_info->bytes_reserved -= buf->len;
4726 cache->space_info->reservation_progress++;
4716 spin_unlock(&cache->space_info->lock); 4727 spin_unlock(&cache->space_info->lock);
4717 } 4728 }
4718 goto out; 4729 goto out;