diff options
-rw-r--r-- | fs/btrfs/ctree.h | 1 | ||||
-rw-r--r-- | fs/btrfs/disk-io.c | 25 | ||||
-rw-r--r-- | fs/btrfs/volumes.c | 6 |
3 files changed, 26 insertions, 6 deletions
diff --git a/fs/btrfs/ctree.h b/fs/btrfs/ctree.h index c88f1e16ce2d..bb4a8d2200d0 100644 --- a/fs/btrfs/ctree.h +++ b/fs/btrfs/ctree.h | |||
@@ -544,6 +544,7 @@ struct btrfs_fs_info { | |||
544 | struct list_head hashers; | 544 | struct list_head hashers; |
545 | struct list_head dead_roots; | 545 | struct list_head dead_roots; |
546 | atomic_t nr_async_submits; | 546 | atomic_t nr_async_submits; |
547 | atomic_t nr_async_bios; | ||
547 | 548 | ||
548 | /* | 549 | /* |
549 | * this is used by the balancing code to wait for all the pending | 550 | * this is used by the balancing code to wait for all the pending |
diff --git a/fs/btrfs/disk-io.c b/fs/btrfs/disk-io.c index 99bd9f9b9eed..9902d29abd06 100644 --- a/fs/btrfs/disk-io.c +++ b/fs/btrfs/disk-io.c | |||
@@ -429,6 +429,21 @@ int btrfs_bio_wq_end_io(struct btrfs_fs_info *info, struct bio *bio, | |||
429 | return 0; | 429 | return 0; |
430 | } | 430 | } |
431 | 431 | ||
432 | static int congested_async(struct btrfs_fs_info *info, int iodone) | ||
433 | { | ||
434 | int limit = 256 * info->fs_devices->open_devices; | ||
435 | |||
436 | if (iodone) | ||
437 | limit = (limit * 3) / 2; | ||
438 | if (atomic_read(&info->nr_async_submits) > limit) | ||
439 | return 1; | ||
440 | |||
441 | limit = 8192 * info->fs_devices->open_devices; | ||
442 | if (iodone) | ||
443 | limit = (limit * 3) / 2; | ||
444 | return atomic_read(&info->nr_async_bios) > limit; | ||
445 | } | ||
446 | |||
432 | static void run_one_async_submit(struct btrfs_work *work) | 447 | static void run_one_async_submit(struct btrfs_work *work) |
433 | { | 448 | { |
434 | struct btrfs_fs_info *fs_info; | 449 | struct btrfs_fs_info *fs_info; |
@@ -437,6 +452,11 @@ static void run_one_async_submit(struct btrfs_work *work) | |||
437 | async = container_of(work, struct async_submit_bio, work); | 452 | async = container_of(work, struct async_submit_bio, work); |
438 | fs_info = BTRFS_I(async->inode)->root->fs_info; | 453 | fs_info = BTRFS_I(async->inode)->root->fs_info; |
439 | atomic_dec(&fs_info->nr_async_submits); | 454 | atomic_dec(&fs_info->nr_async_submits); |
455 | |||
456 | if ((async->bio->bi_rw & (1 << BIO_RW)) && | ||
457 | !congested_async(fs_info, 1)) { | ||
458 | clear_bdi_congested(&fs_info->bdi, WRITE); | ||
459 | } | ||
440 | async->submit_bio_hook(async->inode, async->rw, async->bio, | 460 | async->submit_bio_hook(async->inode, async->rw, async->bio, |
441 | async->mirror_num); | 461 | async->mirror_num); |
442 | kfree(async); | 462 | kfree(async); |
@@ -938,15 +958,13 @@ static int btrfs_congested_fn(void *congested_data, int bdi_bits) | |||
938 | { | 958 | { |
939 | struct btrfs_fs_info *info = (struct btrfs_fs_info *)congested_data; | 959 | struct btrfs_fs_info *info = (struct btrfs_fs_info *)congested_data; |
940 | int ret = 0; | 960 | int ret = 0; |
941 | int limit = 256 * info->fs_devices->open_devices; | ||
942 | struct list_head *cur; | 961 | struct list_head *cur; |
943 | struct btrfs_device *device; | 962 | struct btrfs_device *device; |
944 | struct backing_dev_info *bdi; | 963 | struct backing_dev_info *bdi; |
945 | 964 | ||
946 | if ((bdi_bits & (1 << BDI_write_congested)) && | 965 | if ((bdi_bits & (1 << BDI_write_congested)) && |
947 | atomic_read(&info->nr_async_submits) > limit) { | 966 | congested_async(info, 0)) |
948 | return 1; | 967 | return 1; |
949 | } | ||
950 | 968 | ||
951 | list_for_each(cur, &info->fs_devices->devices) { | 969 | list_for_each(cur, &info->fs_devices->devices) { |
952 | device = list_entry(cur, struct btrfs_device, dev_list); | 970 | device = list_entry(cur, struct btrfs_device, dev_list); |
@@ -1250,6 +1268,7 @@ struct btrfs_root *open_ctree(struct super_block *sb, | |||
1250 | INIT_LIST_HEAD(&fs_info->space_info); | 1268 | INIT_LIST_HEAD(&fs_info->space_info); |
1251 | btrfs_mapping_init(&fs_info->mapping_tree); | 1269 | btrfs_mapping_init(&fs_info->mapping_tree); |
1252 | atomic_set(&fs_info->nr_async_submits, 0); | 1270 | atomic_set(&fs_info->nr_async_submits, 0); |
1271 | atomic_set(&fs_info->nr_async_bios, 0); | ||
1253 | atomic_set(&fs_info->throttles, 0); | 1272 | atomic_set(&fs_info->throttles, 0); |
1254 | atomic_set(&fs_info->throttle_gen, 0); | 1273 | atomic_set(&fs_info->throttle_gen, 0); |
1255 | fs_info->sb = sb; | 1274 | fs_info->sb = sb; |
diff --git a/fs/btrfs/volumes.c b/fs/btrfs/volumes.c index 09311b3066df..23a5b0aba00a 100644 --- a/fs/btrfs/volumes.c +++ b/fs/btrfs/volumes.c | |||
@@ -179,7 +179,7 @@ loop: | |||
179 | cur = pending; | 179 | cur = pending; |
180 | pending = pending->bi_next; | 180 | pending = pending->bi_next; |
181 | cur->bi_next = NULL; | 181 | cur->bi_next = NULL; |
182 | atomic_dec(&device->dev_root->fs_info->nr_async_submits); | 182 | atomic_dec(&device->dev_root->fs_info->nr_async_bios); |
183 | 183 | ||
184 | BUG_ON(atomic_read(&cur->bi_cnt) == 0); | 184 | BUG_ON(atomic_read(&cur->bi_cnt) == 0); |
185 | bio_get(cur); | 185 | bio_get(cur); |
@@ -2145,12 +2145,12 @@ int schedule_bio(struct btrfs_root *root, struct btrfs_device *device, | |||
2145 | } | 2145 | } |
2146 | 2146 | ||
2147 | /* | 2147 | /* |
2148 | * nr_async_sumbits allows us to reliably return congestion to the | 2148 | * nr_async_bios allows us to reliably return congestion to the |
2149 | * higher layers. Otherwise, the async bio makes it appear we have | 2149 | * higher layers. Otherwise, the async bio makes it appear we have |
2150 | * made progress against dirty pages when we've really just put it | 2150 | * made progress against dirty pages when we've really just put it |
2151 | * on a queue for later | 2151 | * on a queue for later |
2152 | */ | 2152 | */ |
2153 | atomic_inc(&root->fs_info->nr_async_submits); | 2153 | atomic_inc(&root->fs_info->nr_async_bios); |
2154 | WARN_ON(bio->bi_next); | 2154 | WARN_ON(bio->bi_next); |
2155 | bio->bi_next = NULL; | 2155 | bio->bi_next = NULL; |
2156 | bio->bi_rw |= rw; | 2156 | bio->bi_rw |= rw; |