Commit 38087d9b authored by Mel Gorman's avatar Mel Gorman Committed by Linus Torvalds

mm, vmscan: simplify the logic deciding whether kswapd sleeps

kswapd goes through some complex steps trying to figure out if it should
stay awake based on the classzone_idx and the requested order.  It is
unnecessarily complex and passes in an invalid classzone_idx to
balance_pgdat().  What matters most of all is whether a larger order has
been requsted and whether kswapd successfully reclaimed at the previous
order.  This patch irons out the logic to check just that and the end
result is less headache inducing.

Link: http://lkml.kernel.org/r/1467970510-21195-10-git-send-email-mgorman@techsingularity.netSigned-off-by: default avatarMel Gorman <mgorman@techsingularity.net>
Acked-by: default avatarJohannes Weiner <hannes@cmpxchg.org>
Acked-by: default avatarVlastimil Babka <vbabka@suse.cz>
Cc: Hillf Danton <hillf.zj@alibaba-inc.com>
Cc: Joonsoo Kim <iamjoonsoo.kim@lge.com>
Cc: Michal Hocko <mhocko@kernel.org>
Cc: Minchan Kim <minchan@kernel.org>
Cc: Rik van Riel <riel@surriel.com>
Signed-off-by: default avatarAndrew Morton <akpm@linux-foundation.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@linux-foundation.org>
parent 31483b6a
...@@ -668,8 +668,9 @@ typedef struct pglist_data { ...@@ -668,8 +668,9 @@ typedef struct pglist_data {
wait_queue_head_t pfmemalloc_wait; wait_queue_head_t pfmemalloc_wait;
struct task_struct *kswapd; /* Protected by struct task_struct *kswapd; /* Protected by
mem_hotplug_begin/end() */ mem_hotplug_begin/end() */
int kswapd_max_order; int kswapd_order;
enum zone_type classzone_idx; enum zone_type kswapd_classzone_idx;
#ifdef CONFIG_COMPACTION #ifdef CONFIG_COMPACTION
int kcompactd_max_order; int kcompactd_max_order;
enum zone_type kcompactd_classzone_idx; enum zone_type kcompactd_classzone_idx;
......
...@@ -1209,9 +1209,10 @@ static pg_data_t __ref *hotadd_new_pgdat(int nid, u64 start) ...@@ -1209,9 +1209,10 @@ static pg_data_t __ref *hotadd_new_pgdat(int nid, u64 start)
arch_refresh_nodedata(nid, pgdat); arch_refresh_nodedata(nid, pgdat);
} else { } else {
/* Reset the nr_zones and classzone_idx to 0 before reuse */ /* Reset the nr_zones, order and classzone_idx before reuse */
pgdat->nr_zones = 0; pgdat->nr_zones = 0;
pgdat->classzone_idx = 0; pgdat->kswapd_order = 0;
pgdat->kswapd_classzone_idx = 0;
} }
/* we can use NODE_DATA(nid) from here */ /* we can use NODE_DATA(nid) from here */
......
...@@ -6039,7 +6039,7 @@ void __paginginit free_area_init_node(int nid, unsigned long *zones_size, ...@@ -6039,7 +6039,7 @@ void __paginginit free_area_init_node(int nid, unsigned long *zones_size,
unsigned long end_pfn = 0; unsigned long end_pfn = 0;
/* pg_data_t should be reset to zero when it's allocated */ /* pg_data_t should be reset to zero when it's allocated */
WARN_ON(pgdat->nr_zones || pgdat->classzone_idx); WARN_ON(pgdat->nr_zones || pgdat->kswapd_classzone_idx);
reset_deferred_meminit(pgdat); reset_deferred_meminit(pgdat);
pgdat->node_id = nid; pgdat->node_id = nid;
......
...@@ -2762,7 +2762,7 @@ static bool pfmemalloc_watermark_ok(pg_data_t *pgdat) ...@@ -2762,7 +2762,7 @@ static bool pfmemalloc_watermark_ok(pg_data_t *pgdat)
/* kswapd must be awake if processes are being throttled */ /* kswapd must be awake if processes are being throttled */
if (!wmark_ok && waitqueue_active(&pgdat->kswapd_wait)) { if (!wmark_ok && waitqueue_active(&pgdat->kswapd_wait)) {
pgdat->classzone_idx = min(pgdat->classzone_idx, pgdat->kswapd_classzone_idx = min(pgdat->kswapd_classzone_idx,
(enum zone_type)ZONE_NORMAL); (enum zone_type)ZONE_NORMAL);
wake_up_interruptible(&pgdat->kswapd_wait); wake_up_interruptible(&pgdat->kswapd_wait);
} }
...@@ -3042,11 +3042,11 @@ static bool prepare_kswapd_sleep(pg_data_t *pgdat, int order, long remaining, ...@@ -3042,11 +3042,11 @@ static bool prepare_kswapd_sleep(pg_data_t *pgdat, int order, long remaining,
if (!populated_zone(zone)) if (!populated_zone(zone))
continue; continue;
if (zone_balanced(zone, order, classzone_idx)) if (!zone_balanced(zone, order, classzone_idx))
return true; return false;
} }
return false; return true;
} }
/* /*
...@@ -3238,8 +3238,8 @@ static int balance_pgdat(pg_data_t *pgdat, int order, int classzone_idx) ...@@ -3238,8 +3238,8 @@ static int balance_pgdat(pg_data_t *pgdat, int order, int classzone_idx)
return sc.order; return sc.order;
} }
static void kswapd_try_to_sleep(pg_data_t *pgdat, int order, static void kswapd_try_to_sleep(pg_data_t *pgdat, int alloc_order, int reclaim_order,
int classzone_idx, int balanced_classzone_idx) unsigned int classzone_idx)
{ {
long remaining = 0; long remaining = 0;
DEFINE_WAIT(wait); DEFINE_WAIT(wait);
...@@ -3250,8 +3250,7 @@ static void kswapd_try_to_sleep(pg_data_t *pgdat, int order, ...@@ -3250,8 +3250,7 @@ static void kswapd_try_to_sleep(pg_data_t *pgdat, int order,
prepare_to_wait(&pgdat->kswapd_wait, &wait, TASK_INTERRUPTIBLE); prepare_to_wait(&pgdat->kswapd_wait, &wait, TASK_INTERRUPTIBLE);
/* Try to sleep for a short interval */ /* Try to sleep for a short interval */
if (prepare_kswapd_sleep(pgdat, order, remaining, if (prepare_kswapd_sleep(pgdat, reclaim_order, remaining, classzone_idx)) {
balanced_classzone_idx)) {
/* /*
* Compaction records what page blocks it recently failed to * Compaction records what page blocks it recently failed to
* isolate pages from and skips them in the future scanning. * isolate pages from and skips them in the future scanning.
...@@ -3264,9 +3263,20 @@ static void kswapd_try_to_sleep(pg_data_t *pgdat, int order, ...@@ -3264,9 +3263,20 @@ static void kswapd_try_to_sleep(pg_data_t *pgdat, int order,
* We have freed the memory, now we should compact it to make * We have freed the memory, now we should compact it to make
* allocation of the requested order possible. * allocation of the requested order possible.
*/ */
wakeup_kcompactd(pgdat, order, classzone_idx); wakeup_kcompactd(pgdat, alloc_order, classzone_idx);
remaining = schedule_timeout(HZ/10); remaining = schedule_timeout(HZ/10);
/*
* If woken prematurely then reset kswapd_classzone_idx and
* order. The values will either be from a wakeup request or
* the previous request that slept prematurely.
*/
if (remaining) {
pgdat->kswapd_classzone_idx = max(pgdat->kswapd_classzone_idx, classzone_idx);
pgdat->kswapd_order = max(pgdat->kswapd_order, reclaim_order);
}
finish_wait(&pgdat->kswapd_wait, &wait); finish_wait(&pgdat->kswapd_wait, &wait);
prepare_to_wait(&pgdat->kswapd_wait, &wait, TASK_INTERRUPTIBLE); prepare_to_wait(&pgdat->kswapd_wait, &wait, TASK_INTERRUPTIBLE);
} }
...@@ -3275,8 +3285,7 @@ static void kswapd_try_to_sleep(pg_data_t *pgdat, int order, ...@@ -3275,8 +3285,7 @@ static void kswapd_try_to_sleep(pg_data_t *pgdat, int order,
* After a short sleep, check if it was a premature sleep. If not, then * After a short sleep, check if it was a premature sleep. If not, then
* go fully to sleep until explicitly woken up. * go fully to sleep until explicitly woken up.
*/ */
if (prepare_kswapd_sleep(pgdat, order, remaining, if (prepare_kswapd_sleep(pgdat, reclaim_order, remaining, classzone_idx)) {
balanced_classzone_idx)) {
trace_mm_vmscan_kswapd_sleep(pgdat->node_id); trace_mm_vmscan_kswapd_sleep(pgdat->node_id);
/* /*
...@@ -3317,9 +3326,7 @@ static void kswapd_try_to_sleep(pg_data_t *pgdat, int order, ...@@ -3317,9 +3326,7 @@ static void kswapd_try_to_sleep(pg_data_t *pgdat, int order,
*/ */
static int kswapd(void *p) static int kswapd(void *p)
{ {
unsigned long order, new_order; unsigned int alloc_order, reclaim_order, classzone_idx;
int classzone_idx, new_classzone_idx;
int balanced_classzone_idx;
pg_data_t *pgdat = (pg_data_t*)p; pg_data_t *pgdat = (pg_data_t*)p;
struct task_struct *tsk = current; struct task_struct *tsk = current;
...@@ -3349,38 +3356,20 @@ static int kswapd(void *p) ...@@ -3349,38 +3356,20 @@ static int kswapd(void *p)
tsk->flags |= PF_MEMALLOC | PF_SWAPWRITE | PF_KSWAPD; tsk->flags |= PF_MEMALLOC | PF_SWAPWRITE | PF_KSWAPD;
set_freezable(); set_freezable();
order = new_order = 0; pgdat->kswapd_order = alloc_order = reclaim_order = 0;
classzone_idx = new_classzone_idx = pgdat->nr_zones - 1; pgdat->kswapd_classzone_idx = classzone_idx = 0;
balanced_classzone_idx = classzone_idx;
for ( ; ; ) { for ( ; ; ) {
bool ret; bool ret;
/* kswapd_try_sleep:
* While we were reclaiming, there might have been another kswapd_try_to_sleep(pgdat, alloc_order, reclaim_order,
* wakeup, so check the values. classzone_idx);
*/
new_order = pgdat->kswapd_max_order;
new_classzone_idx = pgdat->classzone_idx;
pgdat->kswapd_max_order = 0;
pgdat->classzone_idx = pgdat->nr_zones - 1;
if (order < new_order || classzone_idx > new_classzone_idx) { /* Read the new order and classzone_idx */
/* alloc_order = reclaim_order = pgdat->kswapd_order;
* Don't sleep if someone wants a larger 'order' classzone_idx = pgdat->kswapd_classzone_idx;
* allocation or has tigher zone constraints pgdat->kswapd_order = 0;
*/ pgdat->kswapd_classzone_idx = 0;
order = new_order;
classzone_idx = new_classzone_idx;
} else {
kswapd_try_to_sleep(pgdat, order, classzone_idx,
balanced_classzone_idx);
order = pgdat->kswapd_max_order;
classzone_idx = pgdat->classzone_idx;
new_order = order;
new_classzone_idx = classzone_idx;
pgdat->kswapd_max_order = 0;
pgdat->classzone_idx = pgdat->nr_zones - 1;
}
ret = try_to_freeze(); ret = try_to_freeze();
if (kthread_should_stop()) if (kthread_should_stop())
...@@ -3390,12 +3379,24 @@ static int kswapd(void *p) ...@@ -3390,12 +3379,24 @@ static int kswapd(void *p)
* We can speed up thawing tasks if we don't call balance_pgdat * We can speed up thawing tasks if we don't call balance_pgdat
* after returning from the refrigerator * after returning from the refrigerator
*/ */
if (!ret) { if (ret)
trace_mm_vmscan_kswapd_wake(pgdat->node_id, order); continue;
/*
* Reclaim begins at the requested order but if a high-order
* reclaim fails then kswapd falls back to reclaiming for
* order-0. If that happens, kswapd will consider sleeping
* for the order it finished reclaiming at (reclaim_order)
* but kcompactd is woken to compact for the original
* request (alloc_order).
*/
trace_mm_vmscan_kswapd_wake(pgdat->node_id, alloc_order);
reclaim_order = balance_pgdat(pgdat, alloc_order, classzone_idx);
if (reclaim_order < alloc_order)
goto kswapd_try_sleep;
/* return value ignored until next patch */ alloc_order = reclaim_order = pgdat->kswapd_order;
balance_pgdat(pgdat, order, classzone_idx); classzone_idx = pgdat->kswapd_classzone_idx;
}
} }
tsk->flags &= ~(PF_MEMALLOC | PF_SWAPWRITE | PF_KSWAPD); tsk->flags &= ~(PF_MEMALLOC | PF_SWAPWRITE | PF_KSWAPD);
...@@ -3418,10 +3419,8 @@ void wakeup_kswapd(struct zone *zone, int order, enum zone_type classzone_idx) ...@@ -3418,10 +3419,8 @@ void wakeup_kswapd(struct zone *zone, int order, enum zone_type classzone_idx)
if (!cpuset_zone_allowed(zone, GFP_KERNEL | __GFP_HARDWALL)) if (!cpuset_zone_allowed(zone, GFP_KERNEL | __GFP_HARDWALL))
return; return;
pgdat = zone->zone_pgdat; pgdat = zone->zone_pgdat;
if (pgdat->kswapd_max_order < order) { pgdat->kswapd_classzone_idx = max(pgdat->kswapd_classzone_idx, classzone_idx);
pgdat->kswapd_max_order = order; pgdat->kswapd_order = max(pgdat->kswapd_order, order);
pgdat->classzone_idx = min(pgdat->classzone_idx, classzone_idx);
}
if (!waitqueue_active(&pgdat->kswapd_wait)) if (!waitqueue_active(&pgdat->kswapd_wait))
return; return;
if (zone_balanced(zone, order, 0)) if (zone_balanced(zone, order, 0))
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment