mm: memcontrol: fold mem_cgroup_do_charge()
These patches rework memcg charge lifetime to integrate more naturally with the lifetime of user pages. This drastically simplifies the code and reduces charging and uncharging overhead. The most expensive part of charging and uncharging is the page_cgroup bit spinlock, which is removed entirely after this series. Here are the top-10 profile entries of a stress test that reads a 128G sparse file on a freshly booted box, without even a dedicated cgroup (i.e. executing in the root memcg). Before: 15.36% cat [kernel.kallsyms] [k] copy_user_generic_string 13.31% cat [kernel.kallsyms] [k] memset 11.48% cat [kernel.kallsyms] [k] do_mpage_readpage 4.23% cat [kernel.kallsyms] [k] get_page_from_freelist 2.38% cat [kernel.kallsyms] [k] put_page 2.32% cat [kernel.kallsyms] [k] __mem_cgroup_commit_charge 2.18% kswapd0 [kernel.kallsyms] [k] __mem_cgroup_uncharge_common 1.92% kswapd0 [kernel.kallsyms] [k] shrink_page_list 1.86% cat [kernel.kallsyms] [k] __radix_tree_lookup 1.62% cat [kernel.kallsyms] [k] __pagevec_lru_add_fn After: 15.67% cat [kernel.kallsyms] [k] copy_user_generic_string 13.48% cat [kernel.kallsyms] [k] memset 11.42% cat [kernel.kallsyms] [k] do_mpage_readpage 3.98% cat [kernel.kallsyms] [k] get_page_from_freelist 2.46% cat [kernel.kallsyms] [k] put_page 2.13% kswapd0 [kernel.kallsyms] [k] shrink_page_list 1.88% cat [kernel.kallsyms] [k] __radix_tree_lookup 1.67% cat [kernel.kallsyms] [k] __pagevec_lru_add_fn 1.39% kswapd0 [kernel.kallsyms] [k] free_pcppages_bulk 1.30% cat [kernel.kallsyms] [k] kfree As you can see, the memcg footprint has shrunk quite a bit. text data bss dec hex filename 37970 9892 400 48262 bc86 mm/memcontrol.o.old 35239 9892 400 45531 b1db mm/memcontrol.o This patch (of 13): This function was split out because mem_cgroup_try_charge() got too big. But having essentially one sequence of operations arbitrarily split in half is not good for reworking the code. Fold it back in. Signed-off-by: Johannes Weiner <hannes@cmpxchg.org> Acked-by: Michal Hocko <mhocko@suse.cz> Cc: Hugh Dickins <hughd@google.com> Cc: Tejun Heo <tj@kernel.org> Cc: Vladimir Davydov <vdavydov@parallels.com> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
This commit is contained in:
parent
2f3e442ccc
commit
6539cc0538
1 changed files with 64 additions and 102 deletions
160
mm/memcontrol.c
160
mm/memcontrol.c
|
@ -2551,80 +2551,6 @@ static int memcg_cpu_hotplug_callback(struct notifier_block *nb,
|
||||||
return NOTIFY_OK;
|
return NOTIFY_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/* See mem_cgroup_try_charge() for details */
|
|
||||||
enum {
|
|
||||||
CHARGE_OK, /* success */
|
|
||||||
CHARGE_RETRY, /* need to retry but retry is not bad */
|
|
||||||
CHARGE_NOMEM, /* we can't do more. return -ENOMEM */
|
|
||||||
CHARGE_WOULDBLOCK, /* GFP_WAIT wasn't set and no enough res. */
|
|
||||||
};
|
|
||||||
|
|
||||||
static int mem_cgroup_do_charge(struct mem_cgroup *memcg, gfp_t gfp_mask,
|
|
||||||
unsigned int nr_pages, unsigned int min_pages,
|
|
||||||
bool invoke_oom)
|
|
||||||
{
|
|
||||||
unsigned long csize = nr_pages * PAGE_SIZE;
|
|
||||||
struct mem_cgroup *mem_over_limit;
|
|
||||||
struct res_counter *fail_res;
|
|
||||||
unsigned long flags = 0;
|
|
||||||
int ret;
|
|
||||||
|
|
||||||
ret = res_counter_charge(&memcg->res, csize, &fail_res);
|
|
||||||
|
|
||||||
if (likely(!ret)) {
|
|
||||||
if (!do_swap_account)
|
|
||||||
return CHARGE_OK;
|
|
||||||
ret = res_counter_charge(&memcg->memsw, csize, &fail_res);
|
|
||||||
if (likely(!ret))
|
|
||||||
return CHARGE_OK;
|
|
||||||
|
|
||||||
res_counter_uncharge(&memcg->res, csize);
|
|
||||||
mem_over_limit = mem_cgroup_from_res_counter(fail_res, memsw);
|
|
||||||
flags |= MEM_CGROUP_RECLAIM_NOSWAP;
|
|
||||||
} else
|
|
||||||
mem_over_limit = mem_cgroup_from_res_counter(fail_res, res);
|
|
||||||
/*
|
|
||||||
* Never reclaim on behalf of optional batching, retry with a
|
|
||||||
* single page instead.
|
|
||||||
*/
|
|
||||||
if (nr_pages > min_pages)
|
|
||||||
return CHARGE_RETRY;
|
|
||||||
|
|
||||||
if (!(gfp_mask & __GFP_WAIT))
|
|
||||||
return CHARGE_WOULDBLOCK;
|
|
||||||
|
|
||||||
if (gfp_mask & __GFP_NORETRY)
|
|
||||||
return CHARGE_NOMEM;
|
|
||||||
|
|
||||||
ret = mem_cgroup_reclaim(mem_over_limit, gfp_mask, flags);
|
|
||||||
if (mem_cgroup_margin(mem_over_limit) >= nr_pages)
|
|
||||||
return CHARGE_RETRY;
|
|
||||||
/*
|
|
||||||
* Even though the limit is exceeded at this point, reclaim
|
|
||||||
* may have been able to free some pages. Retry the charge
|
|
||||||
* before killing the task.
|
|
||||||
*
|
|
||||||
* Only for regular pages, though: huge pages are rather
|
|
||||||
* unlikely to succeed so close to the limit, and we fall back
|
|
||||||
* to regular pages anyway in case of failure.
|
|
||||||
*/
|
|
||||||
if (nr_pages <= (1 << PAGE_ALLOC_COSTLY_ORDER) && ret)
|
|
||||||
return CHARGE_RETRY;
|
|
||||||
|
|
||||||
/*
|
|
||||||
* At task move, charge accounts can be doubly counted. So, it's
|
|
||||||
* better to wait until the end of task_move if something is going on.
|
|
||||||
*/
|
|
||||||
if (mem_cgroup_wait_acct_move(mem_over_limit))
|
|
||||||
return CHARGE_RETRY;
|
|
||||||
|
|
||||||
if (invoke_oom)
|
|
||||||
mem_cgroup_oom(mem_over_limit, gfp_mask, get_order(csize));
|
|
||||||
|
|
||||||
return CHARGE_NOMEM;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* mem_cgroup_try_charge - try charging a memcg
|
* mem_cgroup_try_charge - try charging a memcg
|
||||||
* @memcg: memcg to charge
|
* @memcg: memcg to charge
|
||||||
|
@ -2641,7 +2567,11 @@ static int mem_cgroup_try_charge(struct mem_cgroup *memcg,
|
||||||
{
|
{
|
||||||
unsigned int batch = max(CHARGE_BATCH, nr_pages);
|
unsigned int batch = max(CHARGE_BATCH, nr_pages);
|
||||||
int nr_oom_retries = MEM_CGROUP_RECLAIM_RETRIES;
|
int nr_oom_retries = MEM_CGROUP_RECLAIM_RETRIES;
|
||||||
int ret;
|
struct mem_cgroup *mem_over_limit;
|
||||||
|
struct res_counter *fail_res;
|
||||||
|
unsigned long nr_reclaimed;
|
||||||
|
unsigned long flags = 0;
|
||||||
|
unsigned long long size;
|
||||||
|
|
||||||
if (mem_cgroup_is_root(memcg))
|
if (mem_cgroup_is_root(memcg))
|
||||||
goto done;
|
goto done;
|
||||||
|
@ -2661,44 +2591,76 @@ static int mem_cgroup_try_charge(struct mem_cgroup *memcg,
|
||||||
|
|
||||||
if (gfp_mask & __GFP_NOFAIL)
|
if (gfp_mask & __GFP_NOFAIL)
|
||||||
oom = false;
|
oom = false;
|
||||||
again:
|
retry:
|
||||||
if (consume_stock(memcg, nr_pages))
|
if (consume_stock(memcg, nr_pages))
|
||||||
goto done;
|
goto done;
|
||||||
|
|
||||||
do {
|
size = batch * PAGE_SIZE;
|
||||||
bool invoke_oom = oom && !nr_oom_retries;
|
if (!res_counter_charge(&memcg->res, size, &fail_res)) {
|
||||||
|
if (!do_swap_account)
|
||||||
|
goto done_restock;
|
||||||
|
if (!res_counter_charge(&memcg->memsw, size, &fail_res))
|
||||||
|
goto done_restock;
|
||||||
|
res_counter_uncharge(&memcg->res, size);
|
||||||
|
mem_over_limit = mem_cgroup_from_res_counter(fail_res, memsw);
|
||||||
|
flags |= MEM_CGROUP_RECLAIM_NOSWAP;
|
||||||
|
} else
|
||||||
|
mem_over_limit = mem_cgroup_from_res_counter(fail_res, res);
|
||||||
|
|
||||||
|
if (batch > nr_pages) {
|
||||||
|
batch = nr_pages;
|
||||||
|
goto retry;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!(gfp_mask & __GFP_WAIT))
|
||||||
|
goto nomem;
|
||||||
|
|
||||||
|
if (gfp_mask & __GFP_NORETRY)
|
||||||
|
goto nomem;
|
||||||
|
|
||||||
|
nr_reclaimed = mem_cgroup_reclaim(mem_over_limit, gfp_mask, flags);
|
||||||
|
|
||||||
|
if (mem_cgroup_margin(mem_over_limit) >= batch)
|
||||||
|
goto retry;
|
||||||
|
/*
|
||||||
|
* Even though the limit is exceeded at this point, reclaim
|
||||||
|
* may have been able to free some pages. Retry the charge
|
||||||
|
* before killing the task.
|
||||||
|
*
|
||||||
|
* Only for regular pages, though: huge pages are rather
|
||||||
|
* unlikely to succeed so close to the limit, and we fall back
|
||||||
|
* to regular pages anyway in case of failure.
|
||||||
|
*/
|
||||||
|
if (nr_reclaimed && batch <= (1 << PAGE_ALLOC_COSTLY_ORDER))
|
||||||
|
goto retry;
|
||||||
|
/*
|
||||||
|
* At task move, charge accounts can be doubly counted. So, it's
|
||||||
|
* better to wait until the end of task_move if something is going on.
|
||||||
|
*/
|
||||||
|
if (mem_cgroup_wait_acct_move(mem_over_limit))
|
||||||
|
goto retry;
|
||||||
|
|
||||||
/* If killed, bypass charge */
|
|
||||||
if (fatal_signal_pending(current))
|
if (fatal_signal_pending(current))
|
||||||
goto bypass;
|
goto bypass;
|
||||||
|
|
||||||
ret = mem_cgroup_do_charge(memcg, gfp_mask, batch,
|
if (!oom)
|
||||||
nr_pages, invoke_oom);
|
|
||||||
switch (ret) {
|
|
||||||
case CHARGE_OK:
|
|
||||||
break;
|
|
||||||
case CHARGE_RETRY: /* not in OOM situation but retry */
|
|
||||||
batch = nr_pages;
|
|
||||||
goto again;
|
|
||||||
case CHARGE_WOULDBLOCK: /* !__GFP_WAIT */
|
|
||||||
goto nomem;
|
goto nomem;
|
||||||
case CHARGE_NOMEM: /* OOM routine works */
|
|
||||||
if (!oom || invoke_oom)
|
|
||||||
goto nomem;
|
|
||||||
nr_oom_retries--;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
} while (ret != CHARGE_OK);
|
|
||||||
|
|
||||||
if (batch > nr_pages)
|
if (nr_oom_retries--)
|
||||||
refill_stock(memcg, batch - nr_pages);
|
goto retry;
|
||||||
done:
|
|
||||||
return 0;
|
mem_cgroup_oom(mem_over_limit, gfp_mask, get_order(batch));
|
||||||
nomem:
|
nomem:
|
||||||
if (!(gfp_mask & __GFP_NOFAIL))
|
if (!(gfp_mask & __GFP_NOFAIL))
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
bypass:
|
bypass:
|
||||||
return -EINTR;
|
return -EINTR;
|
||||||
|
|
||||||
|
done_restock:
|
||||||
|
if (batch > nr_pages)
|
||||||
|
refill_stock(memcg, batch - nr_pages);
|
||||||
|
done:
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
Loading…
Reference in a new issue