Received: by 2002:a05:6a10:9848:0:0:0:0 with SMTP id x8csp3715990pxf; Mon, 22 Mar 2021 13:10:16 -0700 (PDT) X-Google-Smtp-Source: ABdhPJws/UlEgFjnkRGcv6PNoaV+H7/iu9iUSnYZcKJjbwtHvDlgj2aD2TLV0nVP4tD3ba6tu5NE X-Received: by 2002:a17:906:a155:: with SMTP id bu21mr1476896ejb.400.1616443815983; Mon, 22 Mar 2021 13:10:15 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1616443815; cv=none; d=google.com; s=arc-20160816; b=bjUnbZFikh+4pamcPZIzD7jJMmnzhjxOSn9qQ0qWNK1xYse4naojxe87RUfW1wKl99 0G+skKdQbH7KH94tfg+XN18Aj06ksn1BcDpBlN3I90SE75QkRmGuwiaDSQcof1eEsgry jqiZc67lKN6F8qn6eF1StA97MBotr0SbubDBkf2fYVLXLfbRFjL2AIQpJ16TW0Tab0Zc rlaEWAJJT+BlrDlLcRQ8FJwDBwrdmOYZWfuE2bP7M5R81vuqUt4X73vryP6MSxJY/bMa 4PSo2gnSC9QN98npbrX21q+Jvn3vMwIewZ9SESVJaiPVtNtgajkpbYpX0ELKKAqOOXhg YIhA== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:content-transfer-encoding:mime-version:reply-to :references:in-reply-to:message-id:date:subject:cc:to:from :dkim-signature:dkim-signature; bh=ajlXTX+VE3P76p/4QePv4HubnvF0lqkwv8RBMLkCfPo=; b=aNJawBT92kRcJ9nIS+5HDaMeoFquL0LhqLKC6txFU5RHlLYQ58xPRV5vmZVRtUebq5 sIuCnHZwtKuXGSL73dUmXZB1nLBFLyC0I2yWL0JDcfn7w55gOi+vENrL6sWc53mpjx/D 8dlN92FBIIy/5o0irRTfSqGrdoRq077/WphR93nkdcKssGddbV0h/u8fjSnpgEQBI0sh djKbHjOJwsNkxnyDxXDMDkRnYlq7nVaAEeMDCqGWR/jA7A8T6SFC4tMTX8WfqzczqjaZ ++Pi85BynG9tBK/bq8DHGt4H67GHw3sQ7mC+Bck10zqi4RFw7cpiHIWjP9BbN5/REsvs fyPA== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@sent.com header.s=fm3 header.b=hboJOUx8; dkim=pass header.i=@messagingengine.com header.s=fm2 header.b=M8ebOeEh; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=sent.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [23.128.96.18]) by mx.google.com with ESMTP id h18si12372216eds.345.2021.03.22.13.09.54; Mon, 22 Mar 2021 13:10:15 -0700 (PDT) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) client-ip=23.128.96.18; Authentication-Results: mx.google.com; dkim=pass header.i=@sent.com header.s=fm3 header.b=hboJOUx8; dkim=pass header.i=@messagingengine.com header.s=fm2 header.b=M8ebOeEh; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=sent.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S232316AbhCVUHW (ORCPT + 99 others); Mon, 22 Mar 2021 16:07:22 -0400 Received: from new4-smtp.messagingengine.com ([66.111.4.230]:45689 "EHLO new4-smtp.messagingengine.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S230358AbhCVUGg (ORCPT ); Mon, 22 Mar 2021 16:06:36 -0400 Received: from compute2.internal (compute2.nyi.internal [10.202.2.42]) by mailnew.nyi.internal (Postfix) with ESMTP id 209AA5803A5; Mon, 22 Mar 2021 16:06:30 -0400 (EDT) Received: from mailfrontend2 ([10.202.2.163]) by compute2.internal (MEProxy); Mon, 22 Mar 2021 16:06:30 -0400 DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=sent.com; h=from :to:cc:subject:date:message-id:in-reply-to:references:reply-to :mime-version:content-transfer-encoding; s=fm3; bh=ajlXTX+VE3P76 p/4QePv4HubnvF0lqkwv8RBMLkCfPo=; b=hboJOUx8lU1hJbQC1Q9cdibozTvhF l5nm9BV7BK5rhC4KrtO3APm+HauI42sONlVg5in1GaSyMOknMAskcYqhx37dScri Wn8SfE+hLApkjYsfkIN/XUikhvvQE8B3jE8Txt3367yjBOeNqKIU8L0lPa1WvvS/ F2HOpHH0bYiDDnugFzfmCAZ961GU8gWeYOFwbBhj3bvUGfKY6vN1hR6uO91ar/fs uQDnaeAFuF2ffuO7G3bW7NksIhT/1lm+F2KhRfCHEvUueBDuA/GrIYIyF/ROngeC la0TREUVWb7g2JZjQTpWXsgRgiDS0kmDfrYuj5/HNLHl9eWyNrSERIaMA== DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d= messagingengine.com; h=cc:content-transfer-encoding:date:from :in-reply-to:message-id:mime-version:references:reply-to:subject :to:x-me-proxy:x-me-proxy:x-me-sender:x-me-sender:x-sasl-enc; s= fm2; bh=ajlXTX+VE3P76p/4QePv4HubnvF0lqkwv8RBMLkCfPo=; b=M8ebOeEh 5gaFIYWVVCnxv5JOOpU3FAxzPVlyqIrwjTaoQYinTEH2LXSREYHThfXUH95fOqFW jLT4kA7vpsSSnGq3F2HolsWSr033du3JO2+J2z/9mER/7q6Gj6jwxSvz6JcVpG0l MPxfORsugD/MnytE9NNDvyODHPBYudLfQmjoK2osGM4qQvp/RERskRvoyDt/pPCG vU/wVdvk2T3UGx7rsfMdxIxc1II2XPfzpdc/cm9MwyYFl1NIIXJPwpMCBlsSlxNT 78WH9H6rAm50ryA/T4fj9BSX9ozPQCIO6Dof98szeyaoCJpN/9V6+/+B2eQnmwpG kDH7WFujWJyjKA== X-ME-Sender: X-ME-Proxy-Cause: gggruggvucftvghtrhhoucdtuddrgeduledrudeggedgudeffecutefuodetggdotefrod ftvfcurfhrohhfihhlvgemucfhrghsthforghilhdpqfgfvfdpuffrtefokffrpgfnqfgh necuuegrihhlohhuthemuceftddtnecusecvtfgvtghiphhivghnthhsucdlqddutddtmd enucfjughrpefhvffufffkofgjfhhrggfgsedtqhertdertddtnecuhfhrohhmpegkihcu jggrnhcuoeiiihdrhigrnhesshgvnhhtrdgtohhmqeenucggtffrrghtthgvrhhnpeeije euvdeuudeuhfeghfehieeuvdetvdeugfeigeevteeuieeuhedtgeduheefleenucfkphep vddtledriedrjedurddvudefnecuvehluhhsthgvrhfuihiivgeptdenucfrrghrrghmpe hmrghilhhfrhhomhepiihirdihrghnsehsvghnthdrtghomh X-ME-Proxy: Received: from Threadripper.local (209-6-71-213.s2259.c3-0.abr-cbr1.sbo-abr.ma.cable.rcncustomer.com [209.6.71.213]) by mail.messagingengine.com (Postfix) with ESMTPA id 0BF2C1080066; Mon, 22 Mar 2021 16:06:29 -0400 (EDT) From: Zi Yan To: linux-mm@kvack.org Cc: linux-kernel@vger.kernel.org, linux-kselftest@vger.kernel.org, "Kirill A . Shutemov" , Andrew Morton , Shuah Khan , John Hubbard , Sandipan Das , David Hildenbrand , Yang Shi , Mika Penttila , Zi Yan Subject: [PATCH v6 2/2] mm: huge_memory: debugfs for file-backed THP split. Date: Mon, 22 Mar 2021 16:05:47 -0400 Message-Id: <20210322200547.68587-2-zi.yan@sent.com> X-Mailer: git-send-email 2.30.2 In-Reply-To: <20210322200547.68587-1-zi.yan@sent.com> References: <20210322200547.68587-1-zi.yan@sent.com> Reply-To: Zi Yan MIME-Version: 1.0 Content-Transfer-Encoding: quoted-printable Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org From: Zi Yan Further extend /split_huge_pages to accept ",," for file-backed THP split tests since tmpfs may have file backed by THP that mapped nowhere. Update selftest program to test file-backed THP split too. Suggested-by: Kirill A. Shutemov Signed-off-by: Zi Yan Reviewed-by: Yang Shi --- mm/huge_memory.c | 91 ++++++++++++++++++- .../selftests/vm/split_huge_page_test.c | 79 +++++++++++++++- 2 files changed, 164 insertions(+), 6 deletions(-) diff --git a/mm/huge_memory.c b/mm/huge_memory.c index b653255a548e..d3b20e101df2 100644 --- a/mm/huge_memory.c +++ b/mm/huge_memory.c @@ -3055,6 +3055,65 @@ static int split_huge_pages_pid(int pid, unsigned lo= ng vaddr_start, return ret; } =20 +static int split_huge_pages_in_file(const char *file_path, pgoff_t off_sta= rt, + pgoff_t off_end) +{ + struct filename *file; + struct file *candidate; + struct address_space *mapping; + int ret =3D -EINVAL; + pgoff_t off_cur; + unsigned long total =3D 0, split =3D 0; + + file =3D getname_kernel(file_path); + if (IS_ERR(file)) + return ret; + + candidate =3D file_open_name(file, O_RDONLY, 0); + if (IS_ERR(candidate)) + goto out; + + pr_info("split file-backed THPs in file: %s, offset: [0x%lx - 0x%lx]\n", + file_path, off_start, off_end); + + mapping =3D candidate->f_mapping; + + for (off_cur =3D off_start; off_cur < off_end;) { + struct page *fpage =3D pagecache_get_page(mapping, off_cur, + FGP_ENTRY | FGP_HEAD, 0); + + if (xa_is_value(fpage) || !fpage) { + off_cur +=3D PAGE_SIZE; + continue; + } + + if (!is_transparent_hugepage(fpage)) { + off_cur +=3D PAGE_SIZE; + goto next; + } + total++; + off_cur =3D fpage->index + thp_size(fpage); + + if (!trylock_page(fpage)) + goto next; + + if (!split_huge_page(fpage)) + split++; + + unlock_page(fpage); +next: + put_page(fpage); + } + + filp_close(candidate, NULL); + ret =3D 0; + + pr_info("%lu of %lu file-backed THP split\n", split, total); +out: + putname(file); + return ret; +} + #define MAX_INPUT_BUF_SZ 255 =20 static ssize_t split_huge_pages_write(struct file *file, const char __user= *buf, @@ -3062,7 +3121,8 @@ static ssize_t split_huge_pages_write(struct file *fi= le, const char __user *buf, { static DEFINE_MUTEX(split_debug_mutex); ssize_t ret; - char input_buf[MAX_INPUT_BUF_SZ]; /* hold pid, start_vaddr, end_vaddr */ + /* hold pid, start_vaddr, end_vaddr or file_path, off_start, off_end */ + char input_buf[MAX_INPUT_BUF_SZ]; int pid; unsigned long vaddr_start, vaddr_end; =20 @@ -3077,6 +3137,35 @@ static ssize_t split_huge_pages_write(struct file *f= ile, const char __user *buf, goto out; =20 input_buf[MAX_INPUT_BUF_SZ - 1] =3D '\0'; + + if (input_buf[0] =3D=3D '/') { + char *tok; + char *buf =3D input_buf; + char file_path[MAX_INPUT_BUF_SZ]; + pgoff_t off_start =3D 0, off_end =3D 0; + size_t input_len =3D strlen(input_buf); + + tok =3D strsep(&buf, ","); + if (tok) { + strncpy(file_path, tok, MAX_INPUT_BUF_SZ); + } else { + ret =3D -EINVAL; + goto out; + } + + ret =3D sscanf(buf, "0x%lx,0x%lx", &off_start, &off_end); + if (ret !=3D 2) { + pr_info("ret: %ld\n", ret); + ret =3D -EINVAL; + goto out; + } + ret =3D split_huge_pages_in_file(file_path, off_start, off_end); + if (!ret) + ret =3D input_len; + + goto out; + } + ret =3D sscanf(input_buf, "%d,0x%lx,0x%lx", &pid, &vaddr_start, &vaddr_en= d); if (ret =3D=3D 1 && pid =3D=3D 1) { split_huge_pages_all(); diff --git a/tools/testing/selftests/vm/split_huge_page_test.c b/tools/test= ing/selftests/vm/split_huge_page_test.c index 2c0c18e60c57..ebdf2d738978 100644 --- a/tools/testing/selftests/vm/split_huge_page_test.c +++ b/tools/testing/selftests/vm/split_huge_page_test.c @@ -7,11 +7,13 @@ #define _GNU_SOURCE #include #include +#include #include #include #include #include #include +#include #include #include =20 @@ -24,6 +26,9 @@ uint64_t pmd_pagesize; #define SMAP_PATH "/proc/self/smaps" #define INPUT_MAX 80 =20 +#define PID_FMT "%d,0x%lx,0x%lx" +#define PATH_FMT "%s,0x%lx,0x%lx" + #define PFN_MASK ((1UL<<55)-1) #define KPF_THP (1UL<<22) =20 @@ -87,13 +92,16 @@ static int write_file(const char *path, const char *buf= , size_t buflen) return (unsigned int) numwritten; } =20 -static void write_debugfs(int pid, uint64_t vaddr_start, uint64_t vaddr_en= d) +static void write_debugfs(const char *fmt, ...) { char input[INPUT_MAX]; int ret; + va_list argp; + + va_start(argp, fmt); + ret =3D vsnprintf(input, INPUT_MAX, fmt, argp); + va_end(argp); =20 - ret =3D snprintf(input, INPUT_MAX, "%d,0x%lx,0x%lx", pid, vaddr_start, - vaddr_end); if (ret >=3D INPUT_MAX) { printf("%s: Debugfs input is too long\n", __func__); exit(EXIT_FAILURE); @@ -183,7 +191,8 @@ void split_pmd_thp(void) } =20 /* split all THPs */ - write_debugfs(getpid(), (uint64_t)one_page, (uint64_t)one_page + len); + write_debugfs(PID_FMT, getpid(), (uint64_t)one_page, + (uint64_t)one_page + len); =20 for (i =3D 0; i < len; i++) if (one_page[i] !=3D (char)i) { @@ -274,7 +283,7 @@ void split_pte_mapped_thp(void) } =20 /* split all remapped THPs */ - write_debugfs(getpid(), (uint64_t)pte_mapped, + write_debugfs(PID_FMT, getpid(), (uint64_t)pte_mapped, (uint64_t)pte_mapped + pagesize * 4); =20 /* smap does not show THPs after mremap, use kpageflags instead */ @@ -300,6 +309,65 @@ void split_pte_mapped_thp(void) close(kpageflags_fd); } =20 +void split_file_backed_thp(void) +{ + int status; + int fd; + ssize_t num_written; + char tmpfs_template[] =3D "/tmp/thp_split_XXXXXX"; + const char *tmpfs_loc =3D mkdtemp(tmpfs_template); + char testfile[INPUT_MAX]; + + status =3D mount("tmpfs", tmpfs_loc, "tmpfs", 0, "huge=3Dalways,size=3D4m= "); + + if (status) { + printf("Unable to create a tmpfs for testing\n"); + exit(EXIT_FAILURE); + } + + status =3D snprintf(testfile, INPUT_MAX, "%s/thp_file", tmpfs_loc); + if (status >=3D INPUT_MAX) { + printf("Fail to create file-backed THP split testing file\n"); + goto cleanup; + } + + fd =3D open(testfile, O_CREAT|O_WRONLY); + if (fd =3D=3D -1) { + perror("Cannot open testing file\n"); + goto cleanup; + } + + /* write something to the file, so a file-backed THP can be allocated */ + num_written =3D write(fd, tmpfs_loc, sizeof(tmpfs_loc)); + close(fd); + + if (num_written < 1) { + printf("Fail to write data to testing file\n"); + goto cleanup; + } + + /* split the file-backed THP */ + write_debugfs(PATH_FMT, testfile, 0, 1024); + + status =3D unlink(testfile); + if (status) + perror("Cannot remove testing file\n"); + +cleanup: + status =3D umount(tmpfs_loc); + if (status) { + printf("Unable to umount %s\n", tmpfs_loc); + exit(EXIT_FAILURE); + } + status =3D rmdir(tmpfs_loc); + if (status) { + perror("cannot remove tmp dir"); + exit(EXIT_FAILURE); + } + + printf("file-backed THP split test done, please check dmesg for more info= rmation\n"); +} + int main(int argc, char **argv) { if (geteuid() !=3D 0) { @@ -313,6 +381,7 @@ int main(int argc, char **argv) =20 split_pmd_thp(); split_pte_mapped_thp(); + split_file_backed_thp(); =20 return 0; } --=20 2.30.2