diff options
author | Andrew Morton <akpm@linux-foundation.org> | 2007-02-28 23:13:21 -0500 |
---|---|---|
committer | Linus Torvalds <torvalds@woody.linux-foundation.org> | 2007-03-01 17:53:38 -0500 |
commit | 232ea4d69d81169453344b7d05203425c88d973b (patch) | |
tree | 00799a50022f97a93c0e7524752b817399955851 | |
parent | b1a316f6f9c54d668df4304ddf935595501ccb25 (diff) |
[PATCH] throttle_vm_writeout(): don't loop on GFP_NOFS and GFP_NOIO allocations
throttle_vm_writeout() is designed to wait for the dirty levels to subside.
But if the caller holds IO or FS locks, we might be holding up that writeout.
So change it to take a single nap to give other devices a chance to clean some
memory, then return.
Cc: Nick Piggin <nickpiggin@yahoo.com.au>
Cc: OGAWA Hirofumi <hirofumi@mail.parknet.co.jp>
Cc: Kumar Gala <galak@kernel.crashing.org>
Cc: Pete Zaitcev <zaitcev@redhat.com>
Cc: <stable@kernel.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
-rw-r--r-- | include/linux/writeback.h | 2 | ||||
-rw-r--r-- | mm/page-writeback.c | 13 | ||||
-rw-r--r-- | mm/vmscan.c | 2 |
3 files changed, 13 insertions, 4 deletions
diff --git a/include/linux/writeback.h b/include/linux/writeback.h index fc35e6bdfb93..0c78f7f4a976 100644 --- a/include/linux/writeback.h +++ b/include/linux/writeback.h | |||
@@ -84,7 +84,7 @@ static inline void wait_on_inode(struct inode *inode) | |||
84 | int wakeup_pdflush(long nr_pages); | 84 | int wakeup_pdflush(long nr_pages); |
85 | void laptop_io_completion(void); | 85 | void laptop_io_completion(void); |
86 | void laptop_sync_completion(void); | 86 | void laptop_sync_completion(void); |
87 | void throttle_vm_writeout(void); | 87 | void throttle_vm_writeout(gfp_t gfp_mask); |
88 | 88 | ||
89 | /* These are exported to sysctl. */ | 89 | /* These are exported to sysctl. */ |
90 | extern int dirty_background_ratio; | 90 | extern int dirty_background_ratio; |
diff --git a/mm/page-writeback.c b/mm/page-writeback.c index f7e088f5a309..f469e3cd08e8 100644 --- a/mm/page-writeback.c +++ b/mm/page-writeback.c | |||
@@ -296,11 +296,21 @@ void balance_dirty_pages_ratelimited_nr(struct address_space *mapping, | |||
296 | } | 296 | } |
297 | EXPORT_SYMBOL(balance_dirty_pages_ratelimited_nr); | 297 | EXPORT_SYMBOL(balance_dirty_pages_ratelimited_nr); |
298 | 298 | ||
299 | void throttle_vm_writeout(void) | 299 | void throttle_vm_writeout(gfp_t gfp_mask) |
300 | { | 300 | { |
301 | long background_thresh; | 301 | long background_thresh; |
302 | long dirty_thresh; | 302 | long dirty_thresh; |
303 | 303 | ||
304 | if ((gfp_mask & (__GFP_FS|__GFP_IO)) != (__GFP_FS|__GFP_IO)) { | ||
305 | /* | ||
306 | * The caller might hold locks which can prevent IO completion | ||
307 | * or progress in the filesystem. So we cannot just sit here | ||
308 | * waiting for IO to complete. | ||
309 | */ | ||
310 | congestion_wait(WRITE, HZ/10); | ||
311 | return; | ||
312 | } | ||
313 | |||
304 | for ( ; ; ) { | 314 | for ( ; ; ) { |
305 | get_dirty_limits(&background_thresh, &dirty_thresh, NULL); | 315 | get_dirty_limits(&background_thresh, &dirty_thresh, NULL); |
306 | 316 | ||
@@ -317,7 +327,6 @@ void throttle_vm_writeout(void) | |||
317 | } | 327 | } |
318 | } | 328 | } |
319 | 329 | ||
320 | |||
321 | /* | 330 | /* |
322 | * writeback at least _min_pages, and keep writing until the amount of dirty | 331 | * writeback at least _min_pages, and keep writing until the amount of dirty |
323 | * memory is less than the background threshold, or until we're all clean. | 332 | * memory is less than the background threshold, or until we're all clean. |
diff --git a/mm/vmscan.c b/mm/vmscan.c index 0655d5fe73e8..db023e2ff385 100644 --- a/mm/vmscan.c +++ b/mm/vmscan.c | |||
@@ -952,7 +952,7 @@ static unsigned long shrink_zone(int priority, struct zone *zone, | |||
952 | } | 952 | } |
953 | } | 953 | } |
954 | 954 | ||
955 | throttle_vm_writeout(); | 955 | throttle_vm_writeout(sc->gfp_mask); |
956 | 956 | ||
957 | atomic_dec(&zone->reclaim_in_progress); | 957 | atomic_dec(&zone->reclaim_in_progress); |
958 | return nr_reclaimed; | 958 | return nr_reclaimed; |