提交 cda73a10 编写于 作者: Z Zlatko Calusic 提交者: Linus Torvalds

mm: do not sleep in balance_pgdat if there's no i/o congestion

On a 4GB RAM machine, where Normal zone is much smaller than DMA32 zone,
the Normal zone gets fragmented in time.  This requires relatively more
pressure in balance_pgdat to get the zone above the required watermark.
Unfortunately, the congestion_wait() call in there slows it down for a
completely wrong reason, expecting that there's a lot of
writeback/swapout, even when there's none (much more common).  After a
few days, when fragmentation progresses, this flawed logic translates to
a very high CPU iowait times, even though there's no I/O congestion at
all.  If THP is enabled, the problem occurs sooner, but I was able to
see it even on !THP kernels, just by giving it a bit more time to occur.

The proper way to deal with this is to not wait, unless there's
congestion.  Thanks to Mel Gorman, we already have the function that
perfectly fits the job.  The patch was tested on a machine which nicely
revealed the problem after only 1 day of uptime, and it's been working
great.
Signed-off-by: NZlatko Calusic <zlatko.calusic@iskon.hr>
Acked-by: NMel Gorman <mgorman@suse.de>
Signed-off-by: NLinus Torvalds <torvalds@linux-foundation.org>
上级 f01af9f8
...@@ -2570,7 +2570,7 @@ static bool prepare_kswapd_sleep(pg_data_t *pgdat, int order, long remaining, ...@@ -2570,7 +2570,7 @@ static bool prepare_kswapd_sleep(pg_data_t *pgdat, int order, long remaining,
static unsigned long balance_pgdat(pg_data_t *pgdat, int order, static unsigned long balance_pgdat(pg_data_t *pgdat, int order,
int *classzone_idx) int *classzone_idx)
{ {
int all_zones_ok; struct zone *unbalanced_zone;
unsigned long balanced; unsigned long balanced;
int i; int i;
int end_zone = 0; /* Inclusive. 0 = ZONE_DMA */ int end_zone = 0; /* Inclusive. 0 = ZONE_DMA */
...@@ -2604,7 +2604,7 @@ static unsigned long balance_pgdat(pg_data_t *pgdat, int order, ...@@ -2604,7 +2604,7 @@ static unsigned long balance_pgdat(pg_data_t *pgdat, int order,
unsigned long lru_pages = 0; unsigned long lru_pages = 0;
int has_under_min_watermark_zone = 0; int has_under_min_watermark_zone = 0;
all_zones_ok = 1; unbalanced_zone = NULL;
balanced = 0; balanced = 0;
/* /*
...@@ -2743,7 +2743,7 @@ static unsigned long balance_pgdat(pg_data_t *pgdat, int order, ...@@ -2743,7 +2743,7 @@ static unsigned long balance_pgdat(pg_data_t *pgdat, int order,
} }
if (!zone_balanced(zone, testorder, 0, end_zone)) { if (!zone_balanced(zone, testorder, 0, end_zone)) {
all_zones_ok = 0; unbalanced_zone = zone;
/* /*
* We are still under min water mark. This * We are still under min water mark. This
* means that we have a GFP_ATOMIC allocation * means that we have a GFP_ATOMIC allocation
...@@ -2776,7 +2776,7 @@ static unsigned long balance_pgdat(pg_data_t *pgdat, int order, ...@@ -2776,7 +2776,7 @@ static unsigned long balance_pgdat(pg_data_t *pgdat, int order,
pfmemalloc_watermark_ok(pgdat)) pfmemalloc_watermark_ok(pgdat))
wake_up(&pgdat->pfmemalloc_wait); wake_up(&pgdat->pfmemalloc_wait);
if (all_zones_ok || (order && pgdat_balanced(pgdat, balanced, *classzone_idx))) if (!unbalanced_zone || (order && pgdat_balanced(pgdat, balanced, *classzone_idx)))
break; /* kswapd: all done */ break; /* kswapd: all done */
/* /*
* OK, kswapd is getting into trouble. Take a nap, then take * OK, kswapd is getting into trouble. Take a nap, then take
...@@ -2786,7 +2786,7 @@ static unsigned long balance_pgdat(pg_data_t *pgdat, int order, ...@@ -2786,7 +2786,7 @@ static unsigned long balance_pgdat(pg_data_t *pgdat, int order,
if (has_under_min_watermark_zone) if (has_under_min_watermark_zone)
count_vm_event(KSWAPD_SKIP_CONGESTION_WAIT); count_vm_event(KSWAPD_SKIP_CONGESTION_WAIT);
else else
congestion_wait(BLK_RW_ASYNC, HZ/10); wait_iff_congested(unbalanced_zone, BLK_RW_ASYNC, HZ/10);
} }
/* /*
...@@ -2805,7 +2805,7 @@ static unsigned long balance_pgdat(pg_data_t *pgdat, int order, ...@@ -2805,7 +2805,7 @@ static unsigned long balance_pgdat(pg_data_t *pgdat, int order,
* high-order: Balanced zones must make up at least 25% of the node * high-order: Balanced zones must make up at least 25% of the node
* for the node to be balanced * for the node to be balanced
*/ */
if (!(all_zones_ok || (order && pgdat_balanced(pgdat, balanced, *classzone_idx)))) { if (unbalanced_zone && (!order || !pgdat_balanced(pgdat, balanced, *classzone_idx))) {
cond_resched(); cond_resched();
try_to_freeze(); try_to_freeze();
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册