diff options
author | Peter Zijlstra <a.p.zijlstra@chello.nl> | 2007-10-17 02:25:50 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@woody.linux-foundation.org> | 2007-10-17 11:42:45 -0400 |
commit | 04fbfdc14e5f48463820d6b9807daa5e9c92c51f (patch) | |
tree | c62905212c8c6373b2258c7f528398d3c831b075 /mm/backing-dev.c | |
parent | 145ca25eb2fbd20d4faf1bad4628c7650332058f (diff) |
mm: per device dirty threshold
Scale writeback cache per backing device, proportional to its writeout speed.
By decoupling the BDI dirty thresholds a number of problems we currently have
will go away, namely:
- mutual interference starvation (for any number of BDIs);
- deadlocks with stacked BDIs (loop, FUSE and local NFS mounts).
It might be that all dirty pages are for a single BDI while other BDIs are
idling. By giving each BDI a 'fair' share of the dirty limit, each one can have
dirty pages outstanding and make progress.
A global threshold also creates a deadlock for stacked BDIs; when A writes to
B, and A generates enough dirty pages to get throttled, B will never start
writeback until the dirty pages go away. Again, by giving each BDI its own
'independent' dirty limit, this problem is avoided.
So the problem is to determine how to distribute the total dirty limit across
the BDIs fairly and efficiently. A DBI that has a large dirty limit but does
not have any dirty pages outstanding is a waste.
What is done is to keep a floating proportion between the DBIs based on
writeback completions. This way faster/more active devices get a larger share
than slower/idle devices.
[akpm@linux-foundation.org: fix warnings]
[hugh@veritas.com: Fix occasional hang when a task couldn't get out of balance_dirty_pages]
Signed-off-by: Peter Zijlstra <a.p.zijlstra@chello.nl>
Signed-off-by: Hugh Dickins <hugh@veritas.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'mm/backing-dev.c')
-rw-r--r-- | mm/backing-dev.c | 19 |
1 files changed, 14 insertions, 5 deletions
diff --git a/mm/backing-dev.c b/mm/backing-dev.c index a47065e084a4..b0ceb29da4c7 100644 --- a/mm/backing-dev.c +++ b/mm/backing-dev.c | |||
@@ -12,11 +12,17 @@ int bdi_init(struct backing_dev_info *bdi) | |||
12 | 12 | ||
13 | for (i = 0; i < NR_BDI_STAT_ITEMS; i++) { | 13 | for (i = 0; i < NR_BDI_STAT_ITEMS; i++) { |
14 | err = percpu_counter_init_irq(&bdi->bdi_stat[i], 0); | 14 | err = percpu_counter_init_irq(&bdi->bdi_stat[i], 0); |
15 | if (err) { | 15 | if (err) |
16 | for (j = 0; j < i; j++) | 16 | goto err; |
17 | percpu_counter_destroy(&bdi->bdi_stat[i]); | 17 | } |
18 | break; | 18 | |
19 | } | 19 | bdi->dirty_exceeded = 0; |
20 | err = prop_local_init_percpu(&bdi->completions); | ||
21 | |||
22 | if (err) { | ||
23 | err: | ||
24 | for (j = 0; j < i; j++) | ||
25 | percpu_counter_destroy(&bdi->bdi_stat[i]); | ||
20 | } | 26 | } |
21 | 27 | ||
22 | return err; | 28 | return err; |
@@ -29,6 +35,8 @@ void bdi_destroy(struct backing_dev_info *bdi) | |||
29 | 35 | ||
30 | for (i = 0; i < NR_BDI_STAT_ITEMS; i++) | 36 | for (i = 0; i < NR_BDI_STAT_ITEMS; i++) |
31 | percpu_counter_destroy(&bdi->bdi_stat[i]); | 37 | percpu_counter_destroy(&bdi->bdi_stat[i]); |
38 | |||
39 | prop_local_destroy_percpu(&bdi->completions); | ||
32 | } | 40 | } |
33 | EXPORT_SYMBOL(bdi_destroy); | 41 | EXPORT_SYMBOL(bdi_destroy); |
34 | 42 | ||
@@ -81,3 +89,4 @@ long congestion_wait(int rw, long timeout) | |||
81 | return ret; | 89 | return ret; |
82 | } | 90 | } |
83 | EXPORT_SYMBOL(congestion_wait); | 91 | EXPORT_SYMBOL(congestion_wait); |
92 | |||