Received: by 2002:ac0:a5a7:0:0:0:0:0 with SMTP id m36-v6csp19679imm; Fri, 13 Jul 2018 16:09:02 -0700 (PDT) X-Google-Smtp-Source: AAOMgpdjaqQ5b4N02RAtBLsj/CZyf5JdNGPgkivIlYMhy4r1Bb1ToKBbFEZQS7XQhkaETpzjUi5g X-Received: by 2002:a63:1c13:: with SMTP id c19-v6mr7761999pgc.332.1531523342679; Fri, 13 Jul 2018 16:09:02 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1531523342; cv=none; d=google.com; s=arc-20160816; b=ERY+hf5m9rbwZrelQ/kam1sgpfzmC+XU892lSJlP77tRiJMPaJsXsyWSKy7fOOMLhP H0ZfMi4Il2XNpn/Mm7lXJnBFihY7RNlnavksbmbiMjikKBtlzzpt90m84b8nKgJcOnfT n72IpQwjSquYaAPB1IOgV4GNcydJoNMk7ki9FC1A66thscpxgvcf+joPDAhcQaHu2UXQ O3n8eUqHJoIdB/xbXg8nFGahDLUSBQRQhxEYJAHTthZ/0aV0q2PC69D9z/GUJTM/cZkv z2r5lqAiB1aknvb7jAk8QO1HkN2kgk6/QODBet4i/HOlsofZOi87nlHeB36Xr17u7ZdB SCSA== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:mime-version:user-agent:references :message-id:in-reply-to:subject:cc:to:from:date:dkim-signature :arc-authentication-results; bh=2dli7BAjWUBwBEPdTYiXJD6n06w69Pv49ZrHQ9c7Wp4=; b=JERqhh0MlRzGi4hKcK5UTnozzeIcASoX8BOLGsZvKxe8SVbEY+FgBZU7VuMr/97f2C BraFpL/fm4ZI1/b62en2QQfSFOrZlHHYed4VGrDIVqmMZhNqRUmFTUF0kmF6SUGAAnJD /R+3BWfab7sWGCWl8pk08SD3QYLw7Vf6seFzXmDcZKXkrAEYQEcBvNxSHQ4IqJPs5sAU dqXJ44AcujlMn/CUjcVB2p2R0XRccEMwWK20ImuQyrYm3Imc5Gr1dTnYXiN3u0/ZOb89 fBMdUNgjrQ3qu34osYatKg8KKR13jU44h4Sqyq9zvzPTnibeOk5pCxJh3PDhF579fFfD 88dg== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@google.com header.s=20161025 header.b=cico51aM; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=REJECT sp=REJECT dis=NONE) header.from=google.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id v16-v6si23987072pgb.96.2018.07.13.16.08.30; Fri, 13 Jul 2018 16:09:02 -0700 (PDT) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; dkim=pass header.i=@google.com header.s=20161025 header.b=cico51aM; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=REJECT sp=REJECT dis=NONE) header.from=google.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1731950AbeGMXYS (ORCPT + 99 others); Fri, 13 Jul 2018 19:24:18 -0400 Received: from mail-pl0-f67.google.com ([209.85.160.67]:33796 "EHLO mail-pl0-f67.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1731592AbeGMXYP (ORCPT ); Fri, 13 Jul 2018 19:24:15 -0400 Received: by mail-pl0-f67.google.com with SMTP id z9-v6so12782852plo.1 for ; Fri, 13 Jul 2018 16:07:32 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=20161025; h=date:from:to:cc:subject:in-reply-to:message-id:references :user-agent:mime-version; bh=2dli7BAjWUBwBEPdTYiXJD6n06w69Pv49ZrHQ9c7Wp4=; b=cico51aM2aJTt+AVf6cr1/rYUHm8+U/fzKV7c4waMuo3C6nEiSI3LQ52hbzU9O8fKN 7MJn5oX2AhJ0uJTkTkiSwuzUsBlm1LyKQvrW/V7ARw1vDwXBW+g9eAog3DzzrT09sDQT YUExhKjSBlHmEEs889h6H+qfLnQgwYCm7SjKiUMECko7GYPZR4KQ1dHyFClwdP6WOZ8G +XhO0Wz0D1D1THYof0ROB65IvZK1hL8p2zuArm68V2K3WuBoUVYZmctMBPI7BF7fiUA5 buAy5k/H3h15/3G7jFMym2/4KhY6w5k/12DnbFHvgxnxXp60yj3ppIX4Gc+dahEWqcMA gylg== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:date:from:to:cc:subject:in-reply-to:message-id :references:user-agent:mime-version; bh=2dli7BAjWUBwBEPdTYiXJD6n06w69Pv49ZrHQ9c7Wp4=; b=KCadGNMzvWBpCvNDlDc170Ote/9aAwvz6TkqYPJTBsjUY9uoQ1SoBbim/3u+F9W/vX Rv9rQa3vemVXkib3y7mew2K5FJ88/ZR3es5Tqh0ugBCsep8/SLafyRJk1bnipMHADEZ2 H2EK7Q02/d/ZlPhq2G4ICMU9bbR7SW+myxb2jA+ruJiO5PDyL4t/ATnLX+MbsH1+KUxr g5zvc1nVZr4hOotzKwEmtOEXVLPAY403twQLSG/hiZa1w+Jpe4Tint2T2lLUsCeVKQ+a ZOSPtN6SizL14aoArV1P1yxpnvafPt4EdShEMxjYPZOKeb8F6Dwc447HiU7EbfTCCmM1 fhNA== X-Gm-Message-State: AOUpUlEgZsKINqQjSbbe1Xuf0exDGtzJqEPoLSavWF5Z618AaVyeD/DA 0VRD9M0qGbaNDMsw1FSufBsOhg== X-Received: by 2002:a17:902:ab90:: with SMTP id f16-v6mr8359292plr.182.1531523252154; Fri, 13 Jul 2018 16:07:32 -0700 (PDT) Received: from [2620:15c:17:3:3a5:23a7:5e32:4598] ([2620:15c:17:3:3a5:23a7:5e32:4598]) by smtp.gmail.com with ESMTPSA id 65-v6sm37172129pfq.81.2018.07.13.16.07.31 (version=TLS1_2 cipher=ECDHE-RSA-CHACHA20-POLY1305 bits=256/256); Fri, 13 Jul 2018 16:07:31 -0700 (PDT) Date: Fri, 13 Jul 2018 16:07:31 -0700 (PDT) From: David Rientjes X-X-Sender: rientjes@chino.kir.corp.google.com To: Andrew Morton , Roman Gushchin cc: Michal Hocko , Vladimir Davydov , Johannes Weiner , Tejun Heo , cgroups@vger.kernel.org, linux-kernel@vger.kernel.org, linux-mm@kvack.org Subject: [patch v3 -mm 4/6] mm, memcg: evaluate root and leaf memcgs fairly on oom In-Reply-To: Message-ID: References: User-Agent: Alpine 2.21 (DEB 202 2017-01-01) MIME-Version: 1.0 Content-Type: text/plain; charset=US-ASCII Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org There are several downsides to the current implementation that compares the root mem cgroup with leaf mem cgroups for the cgroup-aware oom killer. For example, /proc/pid/oom_score_adj is accounted for processes attached to the root mem cgroup but not leaves. This leads to wild inconsistencies that unfairly bias for or against the root mem cgroup. Assume a 728KB bash shell is attached to the root mem cgroup without any other processes having a non-default /proc/pid/oom_score_adj. At the time of system oom, the root mem cgroup evaluated to 43,474 pages after boot. If the bash shell adjusts its /proc/self/oom_score_adj to 1000, however, the root mem cgroup evaluates to 24,765,482 pages lol. It would take a cgroup 95GB of memory to outweigh the root mem cgroup's evaluation. The reverse is even more confusing: if the bash shell adjusts its /proc/self/oom_score_adj to -999, the root mem cgroup evaluates to 42,268 pages, a basically meaningless transformation. /proc/pid/oom_score_adj is discounted, however, for processes attached to leaf mem cgroups. If a sole process using 250MB of memory is attached to a mem cgroup, it evaluates to 250MB >> PAGE_SHIFT. If its /proc/pid/oom_score_adj is changed to -999, or even 1000, the evaluation remains the same for the mem cgroup. The heuristic that is used for the root mem cgroup also differs from leaf mem cgroups. For the root mem cgroup, the evaluation is the sum of all process's /proc/pid/oom_score. Besides factoring in oom_score_adj, it is based on the sum of rss + swap + page tables for all processes attached to it. For leaf mem cgroups, it is based on the amount of anonymous or unevictable memory + unreclaimable slab + kernel stack + sock + swap. There's also an exemption for root mem cgroup processes that do not intersect the allocating process's mems_allowed. Because the current heuristic is based on oom_badness(), the evaluation of the root mem cgroup disregards all processes attached to it that have disjoint mems_allowed making oom selection specifically dependant on the allocating process for system oom conditions! This patch introduces completely fair comparison between the root mem cgroup and leaf mem cgroups. It compares them with the same heuristic and does not prefer one over the other. It disregards oom_score_adj as the cgroup-aware oom killer should, if enabled by memory.oom_policy. The goal is to target the most memory consuming cgroup on the system, not consider per-process adjustment. The fact that the evaluation of all mem cgroups depends on the mempolicy of the allocating process, which is completely undocumented for the cgroup-aware oom killer, will be addressed in a subsequent patch. Signed-off-by: David Rientjes --- Documentation/admin-guide/cgroup-v2.rst | 7 +- mm/memcontrol.c | 149 ++++++++++++------------ 2 files changed, 76 insertions(+), 80 deletions(-) diff --git a/Documentation/admin-guide/cgroup-v2.rst b/Documentation/admin-guide/cgroup-v2.rst --- a/Documentation/admin-guide/cgroup-v2.rst +++ b/Documentation/admin-guide/cgroup-v2.rst @@ -1382,12 +1382,7 @@ OOM killer to kill all processes attached to the cgroup, except processes with /proc/pid/oom_score_adj set to -1000 (oom disabled). The root cgroup is treated as a leaf memory cgroup as well, so it is -compared with other leaf memory cgroups. Due to internal implementation -restrictions the size of the root cgroup is the cumulative sum of -oom_badness of all its tasks (in other words oom_score_adj of each task -is obeyed). Relying on oom_score_adj (apart from OOM_SCORE_ADJ_MIN) can -lead to over- or underestimation of the root cgroup consumption and it is -therefore discouraged. This might change in the future, however. +compared with other leaf memory cgroups. Please, note that memory charges are not migrating if tasks are moved between different memory cgroups. Moving tasks with diff --git a/mm/memcontrol.c b/mm/memcontrol.c --- a/mm/memcontrol.c +++ b/mm/memcontrol.c @@ -94,6 +94,8 @@ int do_swap_account __read_mostly; #define do_swap_account 0 #endif +static atomic_long_t total_sock_pages; + /* Whether legacy memory+swap accounting is active */ static bool do_memsw_account(void) { @@ -2818,9 +2820,9 @@ static inline bool memcg_has_children(struct mem_cgroup *memcg) } static long memcg_oom_badness(struct mem_cgroup *memcg, - const nodemask_t *nodemask, - unsigned long totalpages) + const nodemask_t *nodemask) { + const bool is_root_memcg = memcg == root_mem_cgroup; long points = 0; int nid; pg_data_t *pgdat; @@ -2829,92 +2831,65 @@ static long memcg_oom_badness(struct mem_cgroup *memcg, if (nodemask && !node_isset(nid, *nodemask)) continue; - points += mem_cgroup_node_nr_lru_pages(memcg, nid, - LRU_ALL_ANON | BIT(LRU_UNEVICTABLE)); - pgdat = NODE_DATA(nid); - points += lruvec_page_state(mem_cgroup_lruvec(pgdat, memcg), - NR_SLAB_UNRECLAIMABLE); + if (is_root_memcg) { + points += node_page_state(pgdat, NR_ACTIVE_ANON) + + node_page_state(pgdat, NR_INACTIVE_ANON); + points += node_page_state(pgdat, NR_SLAB_UNRECLAIMABLE); + } else { + points += mem_cgroup_node_nr_lru_pages(memcg, nid, + LRU_ALL_ANON); + points += lruvec_page_state(mem_cgroup_lruvec(pgdat, memcg), + NR_SLAB_UNRECLAIMABLE); + } } - points += memcg_page_state(memcg, MEMCG_KERNEL_STACK_KB) / - (PAGE_SIZE / 1024); - points += memcg_page_state(memcg, MEMCG_SOCK); - points += memcg_page_state(memcg, MEMCG_SWAP); - + if (is_root_memcg) { + points += global_zone_page_state(NR_KERNEL_STACK_KB) / + (PAGE_SIZE / 1024); + points += atomic_long_read(&total_sock_pages); + points += total_swap_pages - get_nr_swap_pages(); + } else { + points += memcg_page_state(memcg, MEMCG_KERNEL_STACK_KB) / + (PAGE_SIZE / 1024); + points += memcg_page_state(memcg, MEMCG_SOCK); + points += memcg_page_state(memcg, MEMCG_SWAP); + } return points; } /* - * Checks if the given memcg is a valid OOM victim and returns a number, - * which means the folowing: - * -1: there are inflight OOM victim tasks, belonging to the memcg - * 0: memcg is not eligible, e.g. all belonging tasks are protected - * by oom_score_adj set to OOM_SCORE_ADJ_MIN + * Checks if the given non-root memcg has a valid OOM victim and returns a + * number, which means the following: + * -1: there is an inflight OOM victim process attached to the memcg + * 0: memcg is not eligible because all tasks attached are unkillable + * (kthreads or oom_score_adj set to OOM_SCORE_ADJ_MIN) * >0: memcg is eligible, and the returned value is an estimation * of the memory footprint */ static long oom_evaluate_memcg(struct mem_cgroup *memcg, - const nodemask_t *nodemask, - unsigned long totalpages) + const nodemask_t *nodemask) { struct css_task_iter it; struct task_struct *task; int eligible = 0; /* - * Root memory cgroup is a special case: - * we don't have necessary stats to evaluate it exactly as - * leaf memory cgroups, so we approximate it's oom_score - * by summing oom_score of all belonging tasks, which are - * owners of their mm structs. - * - * If there are inflight OOM victim tasks inside - * the root memcg, we return -1. - */ - if (memcg == root_mem_cgroup) { - struct css_task_iter it; - struct task_struct *task; - long score = 0; - - css_task_iter_start(&memcg->css, 0, &it); - while ((task = css_task_iter_next(&it))) { - if (tsk_is_oom_victim(task) && - !test_bit(MMF_OOM_SKIP, - &task->signal->oom_mm->flags)) { - score = -1; - break; - } - - task_lock(task); - if (!task->mm) { - task_unlock(task); - continue; - } - task_unlock(task); - - score += oom_badness(task, memcg, nodemask, - totalpages); - } - css_task_iter_end(&it); - - return score; - } - - /* - * Memcg is OOM eligible if there are OOM killable tasks inside. - * - * We treat tasks with oom_score_adj set to OOM_SCORE_ADJ_MIN - * as unkillable. - * - * If there are inflight OOM victim tasks inside the memcg, - * we return -1. + * Memcg is eligible for oom kill if at least one process is eligible + * to be killed. Processes with oom_score_adj of OOM_SCORE_ADJ_MIN + * are unkillable. */ css_task_iter_start(&memcg->css, 0, &it); while ((task = css_task_iter_next(&it))) { + task_lock(task); + if (!task->mm) { + task_unlock(task); + continue; + } if (!eligible && task->signal->oom_score_adj != OOM_SCORE_ADJ_MIN) eligible = 1; + task_unlock(task); if (tsk_is_oom_victim(task) && !test_bit(MMF_OOM_SKIP, &task->signal->oom_mm->flags)) { @@ -2927,13 +2902,14 @@ static long oom_evaluate_memcg(struct mem_cgroup *memcg, if (eligible <= 0) return eligible; - return memcg_oom_badness(memcg, nodemask, totalpages); + return memcg_oom_badness(memcg, nodemask); } static void select_victim_memcg(struct mem_cgroup *root, struct oom_control *oc) { struct mem_cgroup *iter, *group = NULL; long group_score = 0; + long leaf_score = 0; oc->chosen_memcg = NULL; oc->chosen_points = 0; @@ -2959,12 +2935,18 @@ static void select_victim_memcg(struct mem_cgroup *root, struct oom_control *oc) for_each_mem_cgroup_tree(iter, root) { long score; + /* + * Root memory cgroup will be considered after iteration, + * if eligible. + */ + if (iter == root_mem_cgroup) + continue; + /* * We don't consider non-leaf non-oom_group memory cgroups * without the oom policy of "tree" as OOM victims. */ - if (memcg_has_children(iter) && iter != root_mem_cgroup && - !mem_cgroup_oom_group(iter) && + if (memcg_has_children(iter) && !mem_cgroup_oom_group(iter) && iter->oom_policy != MEMCG_OOM_POLICY_TREE) continue; @@ -2972,16 +2954,15 @@ static void select_victim_memcg(struct mem_cgroup *root, struct oom_control *oc) * If group is not set or we've ran out of the group's sub-tree, * we should set group and reset group_score. */ - if (!group || group == root_mem_cgroup || - !mem_cgroup_is_descendant(iter, group)) { + if (!group || !mem_cgroup_is_descendant(iter, group)) { group = iter; group_score = 0; } - if (memcg_has_children(iter) && iter != root_mem_cgroup) + if (memcg_has_children(iter)) continue; - score = oom_evaluate_memcg(iter, oc->nodemask, oc->totalpages); + score = oom_evaluate_memcg(iter, oc->nodemask); /* * Ignore empty and non-eligible memory cgroups. @@ -3000,6 +2981,7 @@ static void select_victim_memcg(struct mem_cgroup *root, struct oom_control *oc) } group_score += score; + leaf_score += score; if (group_score > oc->chosen_points) { oc->chosen_points = group_score; @@ -3007,8 +2989,25 @@ static void select_victim_memcg(struct mem_cgroup *root, struct oom_control *oc) } } - if (oc->chosen_memcg && oc->chosen_memcg != INFLIGHT_VICTIM) - css_get(&oc->chosen_memcg->css); + if (oc->chosen_memcg != INFLIGHT_VICTIM) { + if (root == root_mem_cgroup) { + group_score = oom_evaluate_memcg(root_mem_cgroup, + oc->nodemask); + if (group_score > leaf_score) { + /* + * Discount the sum of all leaf scores to find + * root score. + */ + group_score -= leaf_score; + if (group_score > oc->chosen_points) { + oc->chosen_points = group_score; + oc->chosen_memcg = root_mem_cgroup; + } + } + } + if (oc->chosen_memcg) + css_get(&oc->chosen_memcg->css); + } rcu_read_unlock(); } @@ -6491,6 +6490,7 @@ bool mem_cgroup_charge_skmem(struct mem_cgroup *memcg, unsigned int nr_pages) gfp_mask = GFP_NOWAIT; mod_memcg_state(memcg, MEMCG_SOCK, nr_pages); + atomic_long_add(nr_pages, &total_sock_pages); if (try_charge(memcg, gfp_mask, nr_pages) == 0) return true; @@ -6512,6 +6512,7 @@ void mem_cgroup_uncharge_skmem(struct mem_cgroup *memcg, unsigned int nr_pages) } mod_memcg_state(memcg, MEMCG_SOCK, -nr_pages); + atomic_long_add(-nr_pages, &total_sock_pages); refill_stock(memcg, nr_pages); }