Received: by 2002:a25:b323:0:0:0:0:0 with SMTP id l35csp1343099ybj; Fri, 20 Sep 2019 08:58:49 -0700 (PDT) X-Google-Smtp-Source: APXvYqznQh9RDmH0GDsq+gCx59K1s0jbO81PEbrunj2rscQNovhLs1GVF1LmXtaThPtdUpUqTc4h X-Received: by 2002:a17:906:82d3:: with SMTP id a19mr9396092ejy.151.1568995129610; Fri, 20 Sep 2019 08:58:49 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1568995129; cv=none; d=google.com; s=arc-20160816; b=uakbsrS6G4uESzzS8BaRmqbDBBIgCY8TuUKvhngHmNSCRgstXAHk0/FWOkPTBIKw28 YuEuLUYbdRdGzh093L4gvkUZzgDmNlaxSdFKCJZAbSi8AXm2oJidBO3un8HkXhL8zMTZ jMrbZfXk3SyhiIUwxNwHiTyXP7fz+lOLEQ1cllKEe6E84VRDj1V3gChHRp2uarTZ3C2A bnnDuR7K6xamCXbzeVEAkk4e94RYs2H8P6B7WkhJBZBF02/+7w+IoIwiajfD9aMkm65r UEzTcUd5Bkz+qAXexr8gEaTDbl3w7xfHt+tD4QWRNZCJknObXntkmOMkuB6HiwZRrhVZ 8KMg== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:cc:to:from:subject:references :mime-version:message-id:in-reply-to:date:dkim-signature; bh=TLk+W3TtdJSuJIMoUgUWtiVZOG1pWX4LnjAI9xModHw=; b=Qg2UZhJwARS0yFUnhnOi+/3tmatR5Rv4ziVXOXLHsJ+EK2zfQU+VTOdGu3plpNUvBs o2+6U6bqjd78K8FYCd4Q4+VcTqP0ZN+uH2WrKMpi1r2SeYLK/uw1fIuFPRFhvDPxuJBt hvwCxtGLTcZnNYq6fSaWI/+I9UqrJlMxELQEjHzCtsqQxy7WbKYmIWtUQrsgoEMShr8m IhVw/fzqPgTH7hMgDW8jQtUOTTF2FdEMvXejLiz1lhh+aU9rohe7xJJzrPDCIK6AZqJo JN1+yCRM2XQl5SubSoP6fRm56DP+1AshPNdnlwqFLb5NeAP6Hy3i9LgpzgJqg1IyRBwQ 6rqg== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@google.com header.s=20161025 header.b=EriOvxCX; 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 y11si1311412ejj.363.2019.09.20.08.58.24; Fri, 20 Sep 2019 08:58:49 -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=EriOvxCX; 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 S2436498AbfISWYm (ORCPT + 99 others); Thu, 19 Sep 2019 18:24:42 -0400 Received: from mail-pl1-f201.google.com ([209.85.214.201]:56600 "EHLO mail-pl1-f201.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S2407205AbfISWYe (ORCPT ); Thu, 19 Sep 2019 18:24:34 -0400 Received: by mail-pl1-f201.google.com with SMTP id v4so3057559plp.23 for ; Thu, 19 Sep 2019 15:24:33 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=20161025; h=date:in-reply-to:message-id:mime-version:references:subject:from:to :cc; bh=TLk+W3TtdJSuJIMoUgUWtiVZOG1pWX4LnjAI9xModHw=; b=EriOvxCXW2Dyp+0lmXbX3M6Cfl9G3gxeiJ94XhY9pqSKmh5hJQZW90HGAaFORsIP+6 GD7FcKPE3YLxPXG7I1cOqlwNJM8z5a/sYkY21FfC6tDkfMPCpK693LHzEk3BoJshaB6r 3KpQ812jdsbg1IG++B5lAwM5k99H4inEsoaU4TrvIY8Lmp8m5DYkhlr0+obO6AXtB2i/ dhZpSCfftkZaZKVDREoqDnk57WjHto/17AyWsbvGRAhnwOEngqRhEDEpEvYrqlvnd0Re YStAy8vh9xkjBuAAhwcQEIYlYxRpzJPEYbx73GdFjdWlmpnM9ks0z9U7+l5YILrI1QOP Yavg== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:date:in-reply-to:message-id:mime-version :references:subject:from:to:cc; bh=TLk+W3TtdJSuJIMoUgUWtiVZOG1pWX4LnjAI9xModHw=; b=lba9rG86pdjV1zdUmAypMg960Vw5aIIF+50YR6Q68i8zshCdnEf9VMQcdAr8AsrMKR c2vM7s9IzYxM/wd5lao14s0RsaOqPk2yaxIHk0ShAWlnQaBZSmIZbQFaVtKqaDPtdjke 9d6mVlOOkO7QLH6NReWImPdMdr+G6oXVsL97dLu01ouWO4WYDyBqCLTsNhLAMLyzKhme cuwWbKjg+ict4BCFLmFl5MWAiONOAp/0fOBYp6RzJifgVo2XW/nUGHoImVwCet3eYAR2 VViZJBBgUA49HGl5Nh5ZH0ZgiZ7HooMYwP8Ekp6G2/YsDRaI8jSIyNPDfUl6xy+uAKGZ 4NIQ== X-Gm-Message-State: APjAAAXT0NYBhSZESf9YLhfuc5OVTs6UBAOaellh6NKOCgs2dvdpoXW5 RQh/8oTOzuT7z2fxOEByztsQDeo/uy4voJOKnA== X-Received: by 2002:a63:ff4a:: with SMTP id s10mr11385929pgk.166.1568931873111; Thu, 19 Sep 2019 15:24:33 -0700 (PDT) Date: Thu, 19 Sep 2019 15:24:17 -0700 In-Reply-To: <20190919222421.27408-1-almasrymina@google.com> Message-Id: <20190919222421.27408-4-almasrymina@google.com> Mime-Version: 1.0 References: <20190919222421.27408-1-almasrymina@google.com> X-Mailer: git-send-email 2.23.0.351.gc4317032e6-goog Subject: [PATCH v5 3/7] hugetlb_cgroup: add reservation accounting for private mappings From: Mina Almasry To: mike.kravetz@oracle.com Cc: shuah@kernel.org, almasrymina@google.com, rientjes@google.com, shakeelb@google.com, gthelen@google.com, akpm@linux-foundation.org, khalid.aziz@oracle.com, linux-kernel@vger.kernel.org, linux-mm@kvack.org, linux-kselftest@vger.kernel.org, cgroups@vger.kernel.org, aneesh.kumar@linux.vnet.ibm.com, mkoutny@suse.com, Hillf Danton Content-Type: text/plain; charset="UTF-8" Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Normally the pointer to the cgroup to uncharge hangs off the struct page, and gets queried when it's time to free the page. With hugetlb_cgroup reservations, this is not possible. Because it's possible for a page to be reserved by one task and actually faulted in by another task. The best place to put the hugetlb_cgroup pointer to uncharge for reservations is in the resv_map. But, because the resv_map has different semantics for private and shared mappings, the code patch to charge/uncharge shared and private mappings is different. This patch implements charging and uncharging for private mappings. For private mappings, the counter to uncharge is in resv_map->reservation_counter. On initializing the resv_map this is set to NULL. On reservation of a region in private mapping, the tasks hugetlb_cgroup is charged and the hugetlb_cgroup is placed is resv_map->reservation_counter. On hugetlb_vm_op_close, we uncharge resv_map->reservation_counter. Signed-off-by: Mina Almasry Acked-by: Hillf Danton --- include/linux/hugetlb.h | 8 +++++++ include/linux/hugetlb_cgroup.h | 11 +++++++++ mm/hugetlb.c | 44 +++++++++++++++++++++++++++++++++- mm/hugetlb_cgroup.c | 12 ---------- 4 files changed, 62 insertions(+), 13 deletions(-) diff --git a/include/linux/hugetlb.h b/include/linux/hugetlb.h index 3d70a17cc0c3..230f44f730fa 100644 --- a/include/linux/hugetlb.h +++ b/include/linux/hugetlb.h @@ -46,6 +46,14 @@ struct resv_map { long adds_in_progress; struct list_head region_cache; long region_cache_count; +#ifdef CONFIG_CGROUP_HUGETLB + /* + * On private mappings, the counter to uncharge reservations is stored + * here. If these fields are 0, then the mapping is shared. + */ + struct page_counter *reservation_counter; + unsigned long pages_per_hpage; +#endif }; extern struct resv_map *resv_map_alloc(void); void resv_map_release(struct kref *ref); diff --git a/include/linux/hugetlb_cgroup.h b/include/linux/hugetlb_cgroup.h index de35997bb5f9..31c4a9e1cf91 100644 --- a/include/linux/hugetlb_cgroup.h +++ b/include/linux/hugetlb_cgroup.h @@ -25,6 +25,17 @@ struct hugetlb_cgroup; #define HUGETLB_CGROUP_MIN_ORDER 2 #ifdef CONFIG_CGROUP_HUGETLB +struct hugetlb_cgroup { + struct cgroup_subsys_state css; + /* + * the counter to account for hugepages from hugetlb. + */ + struct page_counter hugepage[HUGE_MAX_HSTATE]; + /* + * the counter to account for hugepage reservations from hugetlb. + */ + struct page_counter reserved_hugepage[HUGE_MAX_HSTATE]; +}; static inline struct hugetlb_cgroup *hugetlb_cgroup_from_page(struct page *page) { diff --git a/mm/hugetlb.c b/mm/hugetlb.c index a52efcb70d04..bac1cbdd027c 100644 --- a/mm/hugetlb.c +++ b/mm/hugetlb.c @@ -665,6 +665,16 @@ struct resv_map *resv_map_alloc(void) INIT_LIST_HEAD(&resv_map->regions); resv_map->adds_in_progress = 0; +#ifdef CONFIG_CGROUP_HUGETLB + /* + * Initialize these to 0. On shared mappings, 0's here indicate these + * fields don't do cgroup accounting. On private mappings, these will be + * re-initialized to the proper values, to indicate that hugetlb cgroup + * reservations are to be un-charged from here. + */ + resv_map->reservation_counter = NULL; + resv_map->pages_per_hpage = 0; +#endif INIT_LIST_HEAD(&resv_map->region_cache); list_add(&rg->link, &resv_map->region_cache); @@ -3147,7 +3157,18 @@ static void hugetlb_vm_op_close(struct vm_area_struct *vma) reserve = (end - start) - region_count(resv, start, end); - kref_put(&resv->refs, resv_map_release); +#ifdef CONFIG_CGROUP_HUGETLB + /* + * Since we check for HPAGE_RESV_OWNER above, this must a private + * mapping, and these values should be none-zero, and should point to + * the hugetlb_cgroup counter to uncharge for this reservation. + */ + WARN_ON(!resv->reservation_counter); + WARN_ON(!resv->pages_per_hpage); + + hugetlb_cgroup_uncharge_counter(resv->reservation_counter, + (end - start) * resv->pages_per_hpage); +#endif if (reserve) { /* @@ -3157,6 +3178,8 @@ static void hugetlb_vm_op_close(struct vm_area_struct *vma) gbl_reserve = hugepage_subpool_put_pages(spool, reserve); hugetlb_acct_memory(h, -gbl_reserve); } + + kref_put(&resv->refs, resv_map_release); } static int hugetlb_vm_op_split(struct vm_area_struct *vma, unsigned long addr) @@ -4490,6 +4513,7 @@ int hugetlb_reserve_pages(struct inode *inode, struct hstate *h = hstate_inode(inode); struct hugepage_subpool *spool = subpool_inode(inode); struct resv_map *resv_map; + struct hugetlb_cgroup *h_cg; long gbl_reserve; /* This should never happen */ @@ -4523,12 +4547,30 @@ int hugetlb_reserve_pages(struct inode *inode, chg = region_chg(resv_map, from, to); } else { + /* Private mapping. */ resv_map = resv_map_alloc(); if (!resv_map) return -ENOMEM; chg = to - from; + if (hugetlb_cgroup_charge_cgroup(hstate_index(h), + chg * pages_per_huge_page(h), + &h_cg, true)) { + kref_put(&resv_map->refs, resv_map_release); + return -ENOMEM; + } + +#ifdef CONFIG_CGROUP_HUGETLB + /* + * Since this branch handles private mappings, we attach the + * counter to uncharge for this reservation off resv_map. + */ + resv_map->reservation_counter = + &h_cg->reserved_hugepage[hstate_index(h)]; + resv_map->pages_per_hpage = pages_per_huge_page(h); +#endif + set_vma_resv_map(vma, resv_map); set_vma_resv_flags(vma, HPAGE_RESV_OWNER); } diff --git a/mm/hugetlb_cgroup.c b/mm/hugetlb_cgroup.c index dc1ddc9b09c4..ae359ae61cf2 100644 --- a/mm/hugetlb_cgroup.c +++ b/mm/hugetlb_cgroup.c @@ -19,18 +19,6 @@ #include #include -struct hugetlb_cgroup { - struct cgroup_subsys_state css; - /* - * the counter to account for hugepages from hugetlb. - */ - struct page_counter hugepage[HUGE_MAX_HSTATE]; - /* - * the counter to account for hugepage reservations from hugetlb. - */ - struct page_counter reserved_hugepage[HUGE_MAX_HSTATE]; -}; - #define MEMFILE_PRIVATE(x, val) (((x) << 16) | (val)) #define MEMFILE_IDX(val) (((val) >> 16) & 0xffff) #define MEMFILE_ATTR(val) ((val) & 0xffff) -- 2.23.0.351.gc4317032e6-goog