Received: by 2002:a05:6a10:8c0a:0:0:0:0 with SMTP id go10csp2643872pxb; Sun, 24 Jan 2021 15:06:42 -0800 (PST) X-Google-Smtp-Source: ABdhPJyWamddKpf5R89yrq21D7+wj102E8kkSwdeybFp9BZPNKF+vZaMD8IPjBJhnQrzGKc20KZH X-Received: by 2002:a17:906:76d6:: with SMTP id q22mr212200ejn.221.1611529602423; Sun, 24 Jan 2021 15:06:42 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1611529602; cv=none; d=google.com; s=arc-20160816; b=sUemEyfg9Y67hoQt+GK7jR6ymF/+VzLMp4jQUEFbW22IG+LO8175x7kIaoxZA7FoqL bIg9QXvUwcJiqHa1IyXJgsAiI5dIbqaL/H0JkeUhQSaL0+2GrDSlIMi7+8wnfFDDIzj7 miz6+qvdGV4nLqQGWV4E+EHmyhYM0IWhFpd7dQio3NV0SgFiyIeMuW/jbg6P0jRlKDzV 7hKz/r67SD+gP3OcunzCF8qxZ0Ho/zX+pa+mOwvmlGAr2x7hjYzq3MtBfpIQ2+wHe4wE ZUY5PFd2WCZPZbX46YWiRkHtTR7VBK7BxQQyFNTGlGbTrCpjIH7CZr7Unoziv7mKRL2E 1yHA== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:dkim-signature:content-transfer-encoding :content-language:in-reply-to:mime-version:user-agent:date :message-id:from:references:to:subject; bh=yFxaTbdHxCT1nIAuPVS+diJK7osfaBmEagwLoZD5A6w=; b=B75dqa0indcRLytfcId05iKaHXvRfk7Jl2vX35nRv+3TNlfSv93BziqJs736AUIwWi FiibdPntVZoZCPmo/R1+0nwLIi5/raOvPSiYpdbZHRowYWcgbjaUwgb25W/Jp0sTKJb0 XnHlw09zzVlNcaKTRLF2zXheu7D14Cdq8TZaDq1wkBgF/1gmfbkXz+eH1EIfh5Z04G4T f6PvFxaBmUZ60Q8PjixfJd2KCitIT2xVwCXRwRNdEILvjqxA3zCLl98om1w+a2CkgOvv qYimAmsLHnTkGMSthWHTVc0Q+5koqqzuK0Kg09mDWy05WtVbBlcga77LWUR/EQi7RTy8 gqdw== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@nvidia.com header.s=n1 header.b="W6G//EY6"; 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=nvidia.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [23.128.96.18]) by mx.google.com with ESMTP id f26si6576517edq.200.2021.01.24.15.06.13; Sun, 24 Jan 2021 15:06:42 -0800 (PST) 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=@nvidia.com header.s=n1 header.b="W6G//EY6"; 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=nvidia.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1726445AbhAXXEU (ORCPT + 99 others); Sun, 24 Jan 2021 18:04:20 -0500 Received: from hqnvemgate26.nvidia.com ([216.228.121.65]:13815 "EHLO hqnvemgate26.nvidia.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1726007AbhAXXEL (ORCPT ); Sun, 24 Jan 2021 18:04:11 -0500 Received: from hqmail.nvidia.com (Not Verified[216.228.121.13]) by hqnvemgate26.nvidia.com (using TLS: TLSv1.2, AES256-SHA) id ; Sun, 24 Jan 2021 15:03:29 -0800 Received: from [10.2.52.243] (172.20.145.6) by HQMAIL107.nvidia.com (172.20.187.13) with Microsoft SMTP Server (TLS) id 15.0.1473.3; Sun, 24 Jan 2021 23:03:21 +0000 Subject: Re: [PATCH v7 13/14] selftests/vm: test flag is broken To: Pavel Tatashin , , , , , , , , , , , , , , , , , , , , , , References: <20210122033748.924330-1-pasha.tatashin@soleen.com> <20210122033748.924330-14-pasha.tatashin@soleen.com> From: John Hubbard Message-ID: <6fe738b0-af13-333d-8b22-2d3b62a1ded7@nvidia.com> Date: Sun, 24 Jan 2021 15:03:20 -0800 User-Agent: Mozilla/5.0 (X11; Linux x86_64; rv:85.0) Gecko/20100101 Thunderbird/85.0 MIME-Version: 1.0 In-Reply-To: <20210122033748.924330-14-pasha.tatashin@soleen.com> Content-Type: text/plain; charset="UTF-8"; format=flowed Content-Language: en-US Content-Transfer-Encoding: 7bit X-Originating-IP: [172.20.145.6] X-ClientProxiedBy: HQMAIL105.nvidia.com (172.20.187.12) To HQMAIL107.nvidia.com (172.20.187.13) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=nvidia.com; s=n1; t=1611529409; bh=yFxaTbdHxCT1nIAuPVS+diJK7osfaBmEagwLoZD5A6w=; h=Subject:To:References:From:Message-ID:Date:User-Agent: MIME-Version:In-Reply-To:Content-Type:Content-Language: Content-Transfer-Encoding:X-Originating-IP:X-ClientProxiedBy; b=W6G//EY6qtt33EohnjNM/Y821r3itrZAkAvXL8++vljBhG/1VFuYmGZGDgGDd/bmJ wcNASytPFQYyE9E4ksOyTVB7LDz/arpB/ewwGMCREozP9YdoepmNEaRJoj9388PGk6 B2DIZHJRg7//TsXH6GIaT6QEov4Pr8NYhSg7TTBCf/zSjFBi0c7FstOJb8PhoO3PP5 0ACKQsgof7MPKFtf34Wx8eV4UesCFde3qb1zDcCVml9EPuPtyhJ9T+xmgvbASpm8Yb p+2MGP49TS/m+nxEVAJKSeSqz9/vpfX4c76RxaY5803vwz3GeWNMtUDbVl+1dOFDRj /BJc5Ei1qXisw== Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On 1/21/21 7:37 PM, Pavel Tatashin wrote: > In gup_test both gup_flags and test_flags use the same flags field. > This is broken. > > Farther, in the actual gup_test.c all the passed gup_flags are erased and > unconditionally replaced with FOLL_WRITE. > > Which means that test_flags are ignored, and code like this always > performs pin dump test: > > 155 if (gup->flags & GUP_TEST_FLAG_DUMP_PAGES_USE_PIN) > 156 nr = pin_user_pages(addr, nr, gup->flags, > 157 pages + i, NULL); > 158 else > 159 nr = get_user_pages(addr, nr, gup->flags, > 160 pages + i, NULL); > 161 break; > > Add a new test_flags field, to allow raw gup_flags to work. > Add a new subcommand for DUMP_USER_PAGES_TEST to specify that pin test > should be performed. > Remove unconditional overwriting of gup_flags via FOLL_WRITE. But, > preserve the previous behaviour where FOLL_WRITE was the default flag, > and add a new option "-W" to unset FOLL_WRITE. > > Rename flags with gup_flags. Hi Pavel, Thanks again for fixing this up! Looks good, with a tiny point about the subject line: 1) To follow convention, the subject line should say what you're doing, not what the previous condition was. Also, there are several tests in that directory, so we should say which one. So more like this: "selftests/vm: gup_test: fix test flag" That is just a minor documentation point, so either way, feel free to add: Reviewed-by: John Hubbard thanks, -- John Hubbard NVIDIA > > With the fix, dump works like this: > > root@virtme:/# gup_test -c > ---- page #0, starting from user virt addr: 0x7f8acb9e4000 > page:00000000d3d2ee27 refcount:2 mapcount:1 mapping:0000000000000000 > index:0x0 pfn:0x100bcf > anon flags: 0x300000000080016(referenced|uptodate|lru|swapbacked) > raw: 0300000000080016 ffffd0e204021608 ffffd0e208df2e88 ffff8ea04243ec61 > raw: 0000000000000000 0000000000000000 0000000200000000 0000000000000000 > page dumped because: gup_test: dump_pages() test > DUMP_USER_PAGES_TEST: done > > root@virtme:/# gup_test -c -p > ---- page #0, starting from user virt addr: 0x7fd19701b000 > page:00000000baed3c7d refcount:1025 mapcount:1 mapping:0000000000000000 > index:0x0 pfn:0x108008 > anon flags: 0x300000000080014(uptodate|lru|swapbacked) > raw: 0300000000080014 ffffd0e204200188 ffffd0e205e09088 ffff8ea04243ee71 > raw: 0000000000000000 0000000000000000 0000040100000000 0000000000000000 > page dumped because: gup_test: dump_pages() test > DUMP_USER_PAGES_TEST: done > > Refcount shows the difference between pin vs no-pin case. > Also change type of nr from int to long, as it counts number of pages. > > Signed-off-by: Pavel Tatashin > --- > mm/gup_test.c | 23 ++++++++++------------- > mm/gup_test.h | 3 ++- > tools/testing/selftests/vm/gup_test.c | 15 +++++++++++---- > 3 files changed, 23 insertions(+), 18 deletions(-) > > diff --git a/mm/gup_test.c b/mm/gup_test.c > index e3cf78e5873e..a6ed1c877679 100644 > --- a/mm/gup_test.c > +++ b/mm/gup_test.c > @@ -94,7 +94,7 @@ static int __gup_test_ioctl(unsigned int cmd, > { > ktime_t start_time, end_time; > unsigned long i, nr_pages, addr, next; > - int nr; > + long nr; > struct page **pages; > int ret = 0; > bool needs_mmap_lock = > @@ -126,37 +126,34 @@ static int __gup_test_ioctl(unsigned int cmd, > nr = (next - addr) / PAGE_SIZE; > } > > - /* Filter out most gup flags: only allow a tiny subset here: */ > - gup->flags &= FOLL_WRITE; > - > switch (cmd) { > case GUP_FAST_BENCHMARK: > - nr = get_user_pages_fast(addr, nr, gup->flags, > + nr = get_user_pages_fast(addr, nr, gup->gup_flags, > pages + i); > break; > case GUP_BASIC_TEST: > - nr = get_user_pages(addr, nr, gup->flags, pages + i, > + nr = get_user_pages(addr, nr, gup->gup_flags, pages + i, > NULL); > break; > case PIN_FAST_BENCHMARK: > - nr = pin_user_pages_fast(addr, nr, gup->flags, > + nr = pin_user_pages_fast(addr, nr, gup->gup_flags, > pages + i); > break; > case PIN_BASIC_TEST: > - nr = pin_user_pages(addr, nr, gup->flags, pages + i, > + nr = pin_user_pages(addr, nr, gup->gup_flags, pages + i, > NULL); > break; > case PIN_LONGTERM_BENCHMARK: > nr = pin_user_pages(addr, nr, > - gup->flags | FOLL_LONGTERM, > + gup->gup_flags | FOLL_LONGTERM, > pages + i, NULL); > break; > case DUMP_USER_PAGES_TEST: > - if (gup->flags & GUP_TEST_FLAG_DUMP_PAGES_USE_PIN) > - nr = pin_user_pages(addr, nr, gup->flags, > + if (gup->test_flags & GUP_TEST_FLAG_DUMP_PAGES_USE_PIN) > + nr = pin_user_pages(addr, nr, gup->gup_flags, > pages + i, NULL); > else > - nr = get_user_pages(addr, nr, gup->flags, > + nr = get_user_pages(addr, nr, gup->gup_flags, > pages + i, NULL); > break; > default: > @@ -187,7 +184,7 @@ static int __gup_test_ioctl(unsigned int cmd, > > start_time = ktime_get(); > > - put_back_pages(cmd, pages, nr_pages, gup->flags); > + put_back_pages(cmd, pages, nr_pages, gup->test_flags); > > end_time = ktime_get(); > gup->put_delta_usec = ktime_us_delta(end_time, start_time); > diff --git a/mm/gup_test.h b/mm/gup_test.h > index 90a6713d50eb..887ac1d5f5bc 100644 > --- a/mm/gup_test.h > +++ b/mm/gup_test.h > @@ -21,7 +21,8 @@ struct gup_test { > __u64 addr; > __u64 size; > __u32 nr_pages_per_call; > - __u32 flags; > + __u32 gup_flags; > + __u32 test_flags; > /* > * Each non-zero entry is the number of the page (1-based: first page is > * page 1, so that zero entries mean "do nothing") from the .addr base. > diff --git a/tools/testing/selftests/vm/gup_test.c b/tools/testing/selftests/vm/gup_test.c > index 6c6336dd3b7f..943cc2608dc2 100644 > --- a/tools/testing/selftests/vm/gup_test.c > +++ b/tools/testing/selftests/vm/gup_test.c > @@ -37,13 +37,13 @@ int main(int argc, char **argv) > { > struct gup_test gup = { 0 }; > unsigned long size = 128 * MB; > - int i, fd, filed, opt, nr_pages = 1, thp = -1, repeats = 1, write = 0; > + int i, fd, filed, opt, nr_pages = 1, thp = -1, repeats = 1, write = 1; > unsigned long cmd = GUP_FAST_BENCHMARK; > int flags = MAP_PRIVATE; > char *file = "/dev/zero"; > char *p; > > - while ((opt = getopt(argc, argv, "m:r:n:F:f:abctTLUuwSH")) != -1) { > + while ((opt = getopt(argc, argv, "m:r:n:F:f:abctTLUuwWSHp")) != -1) { > switch (opt) { > case 'a': > cmd = PIN_FAST_BENCHMARK; > @@ -65,9 +65,13 @@ int main(int argc, char **argv) > */ > gup.which_pages[0] = 1; > break; > + case 'p': > + /* works only with DUMP_USER_PAGES_TEST */ > + gup.test_flags |= GUP_TEST_FLAG_DUMP_PAGES_USE_PIN; > + break; > case 'F': > /* strtol, so you can pass flags in hex form */ > - gup.flags = strtol(optarg, 0, 0); > + gup.gup_flags = strtol(optarg, 0, 0); > break; > case 'm': > size = atoi(optarg) * MB; > @@ -93,6 +97,9 @@ int main(int argc, char **argv) > case 'w': > write = 1; > break; > + case 'W': > + write = 0; > + break; > case 'f': > file = optarg; > break; > @@ -140,7 +147,7 @@ int main(int argc, char **argv) > > gup.nr_pages_per_call = nr_pages; > if (write) > - gup.flags |= FOLL_WRITE; > + gup.gup_flags |= FOLL_WRITE; > > fd = open("/sys/kernel/debug/gup_test", O_RDWR); > if (fd == -1) { >