diff options
-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; |