From mboxrd@z Thu Jan 1 00:00:00 1970 Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org X-Spam-Level: X-Spam-Status: No, score=-18.9 required=3.0 tests=BAYES_00,DKIMWL_WL_MED, DKIM_SIGNED,DKIM_VALID,DKIM_VALID_AU,HEADER_FROM_DIFFERENT_DOMAINS, INCLUDES_PATCH,MAILING_LIST_MULTI,SIGNED_OFF_BY,SPF_HELO_NONE,SPF_PASS, URIBL_BLOCKED,USER_AGENT_SANE_1,USER_IN_DEF_DKIM_WL autolearn=ham autolearn_force=no version=3.4.0 Received: from mail.kernel.org (mail.kernel.org [198.145.29.99]) by smtp.lore.kernel.org (Postfix) with ESMTP id ED807C5519F for ; Mon, 16 Nov 2020 15:47:36 +0000 (UTC) Received: from kanga.kvack.org (kanga.kvack.org [205.233.56.17]) by mail.kernel.org (Postfix) with ESMTP id 4DEC5207BC for ; Mon, 16 Nov 2020 15:47:36 +0000 (UTC) Authentication-Results: mail.kernel.org; dkim=pass (2048-bit key) header.d=google.com header.i=@google.com header.b="ShfON6UV" DMARC-Filter: OpenDMARC Filter v1.3.2 mail.kernel.org 4DEC5207BC Authentication-Results: mail.kernel.org; dmarc=fail (p=reject dis=none) header.from=google.com Authentication-Results: mail.kernel.org; spf=pass smtp.mailfrom=owner-linux-mm@kvack.org Received: by kanga.kvack.org (Postfix) id B618C6B006E; Mon, 16 Nov 2020 10:47:35 -0500 (EST) Received: by kanga.kvack.org (Postfix, from userid 40) id AA2376B0070; Mon, 16 Nov 2020 10:47:35 -0500 (EST) X-Delivered-To: int-list-linux-mm@kvack.org Received: by kanga.kvack.org (Postfix, from userid 63042) id 832A26B0071; Mon, 16 Nov 2020 10:47:35 -0500 (EST) X-Delivered-To: linux-mm@kvack.org Received: from forelay.hostedemail.com (smtprelay0103.hostedemail.com [216.40.44.103]) by kanga.kvack.org (Postfix) with ESMTP id 3BB8E6B006E for ; Mon, 16 Nov 2020 10:47:35 -0500 (EST) Received: from smtpin19.hostedemail.com (10.5.19.251.rfc1918.com [10.5.19.251]) by forelay02.hostedemail.com (Postfix) with ESMTP id BA7E13626 for ; Mon, 16 Nov 2020 15:47:34 +0000 (UTC) X-FDA: 77490711228.19.car39_3316aa32732a Received: from filter.hostedemail.com (10.5.16.251.rfc1918.com [10.5.16.251]) by smtpin19.hostedemail.com (Postfix) with ESMTP id 963C91AD1B3 for ; Mon, 16 Nov 2020 15:47:34 +0000 (UTC) X-HE-Tag: car39_3316aa32732a X-Filterd-Recvd-Size: 17757 Received: from mail-wr1-f67.google.com (mail-wr1-f67.google.com [209.85.221.67]) by imf14.hostedemail.com (Postfix) with ESMTP for ; Mon, 16 Nov 2020 15:47:34 +0000 (UTC) Received: by mail-wr1-f67.google.com with SMTP id c17so19159550wrc.11 for ; Mon, 16 Nov 2020 07:47:33 -0800 (PST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=20161025; h=date:from:to:cc:subject:message-id:references:mime-version :content-disposition:in-reply-to:user-agent; bh=/B72MSgPAcGFOtHsFbzRv4hXre435lWI6Ni4s+kykJk=; b=ShfON6UVsNZgOQHD1xyp9+4W7XLzRCXFpaDvkBd7K1EcKue1dX1hxN9pn+/YTF0FPS Kv7PXx1anWqCuPjasBlWxkhrzBXTkCZG8IERV/F7zX/yfH0qM1tf1E6P7g4bXa8ePjoi aaXoTh15bC7RsVnsrmAvJm7qjR1XxRvLKZujXNKuW11ZyKEgQPdOPnuExdmSnTc0emg7 Xe7PHIBqSgAkFn3ctGGtDHH71ohI0txAU43oHqz4Zhc+TKnsecbHAImYXbipGhLxfVUz VBrNTheosmQRZ6BMQnxuo/AeuZmUE/BLHsHMmbs9i/IMXGmWhzDocYVHQj721qSsSmgq ks6w== 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:message-id:references :mime-version:content-disposition:in-reply-to:user-agent; bh=/B72MSgPAcGFOtHsFbzRv4hXre435lWI6Ni4s+kykJk=; b=fuqC8eMQT0U1tJPgVMG/yrkMZ5r+9+R7CSejHKDx4VVyRLtPd/4WpPxl+rw5IsDAkd ZMTp4QR/YqIOBIW7o9lxVqDgBThoIYiWKGr6KGJCZmekvrdrmbGtiuQdN5Mbh099oTXo GimzLg7Rr40q2hx+ZqTnAFEEBkajl1X6/cHlZwpI1FSgewK+65/mtUYZhrzHBjAWp6z2 X3LGhzDhH/o6QfBPxwqxFEPGXNI26GEKICohwiMH9e3RrhPwwzrcnlJsKQLgMY32A2kd z7r7bBhaO6mPtYjyfacxekgHLV7OwCK10IKZvcSKhA1ZF38aRrUpIfG3v+6jZduXXhy6 9mlQ== X-Gm-Message-State: AOAM5322n/TTW4u8Au2cTv7g4L0ZN+d1uzakTKRpAPT0MXRgIvhYc+P1 uQVdNil+dF4/+PCk+vEaXTUZyw== X-Google-Smtp-Source: ABdhPJxTqdmXeH9efOZlnxKr7/PzB/GbyOWkjsiAU70Q1F4r19caIrV7CqxKDubn5UD22NTzr8jDAg== X-Received: by 2002:a05:6000:10c9:: with SMTP id b9mr21240865wrx.251.1605541652752; Mon, 16 Nov 2020 07:47:32 -0800 (PST) Received: from elver.google.com ([2a00:79e0:15:13:f693:9fff:fef4:2449]) by smtp.gmail.com with ESMTPSA id t15sm19609192wmn.19.2020.11.16.07.47.31 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Mon, 16 Nov 2020 07:47:31 -0800 (PST) Date: Mon, 16 Nov 2020 16:47:26 +0100 From: Marco Elver To: Andrey Konovalov Cc: Andrew Morton , Catalin Marinas , Will Deacon , Vincenzo Frascino , Dmitry Vyukov , Andrey Ryabinin , Alexander Potapenko , Evgenii Stepanov , Branislav Rankov , Kevin Brodsky , kasan-dev@googlegroups.com, linux-arm-kernel@lists.infradead.org, linux-mm@kvack.org, linux-kernel@vger.kernel.org Subject: Re: [PATCH mm v3 19/19] kasan: update documentation Message-ID: <20201116154726.GI1357314@elver.google.com> References: <1668fc1d020f9c1f2b8bf57467f0433bfd74d0a3.1605305978.git.andreyknvl@google.com> MIME-Version: 1.0 Content-Type: text/plain; charset=us-ascii Content-Disposition: inline In-Reply-To: <1668fc1d020f9c1f2b8bf57467f0433bfd74d0a3.1605305978.git.andreyknvl@google.com> User-Agent: Mutt/1.14.6 (2020-07-11) X-Bogosity: Ham, tests=bogofilter, spamicity=0.000000, version=1.2.4 Sender: owner-linux-mm@kvack.org Precedence: bulk X-Loop: owner-majordomo@kvack.org List-ID: On Fri, Nov 13, 2020 at 11:20PM +0100, Andrey Konovalov wrote: > This change updates KASAN documentation to reflect the addition of boot > parameters and also reworks and clarifies some of the existing sections, > in particular: defines what a memory granule is, mentions quarantine, > makes Kunit section more readable. > > Signed-off-by: Andrey Konovalov > Link: https://linux-review.googlesource.com/id/Ib1f83e91be273264b25f42b04448ac96b858849f Reviewed-by: Marco Elver > --- > Documentation/dev-tools/kasan.rst | 186 +++++++++++++++++++----------- > 1 file changed, 116 insertions(+), 70 deletions(-) > > diff --git a/Documentation/dev-tools/kasan.rst b/Documentation/dev-tools/kasan.rst > index ffbae8ce5748..0d5d77919b1a 100644 > --- a/Documentation/dev-tools/kasan.rst > +++ b/Documentation/dev-tools/kasan.rst > @@ -4,8 +4,9 @@ The Kernel Address Sanitizer (KASAN) > Overview > -------- > > -KernelAddressSANitizer (KASAN) is a dynamic memory error detector designed to > -find out-of-bound and use-after-free bugs. KASAN has three modes: > +KernelAddressSANitizer (KASAN) is a dynamic memory safety error detector > +designed to find out-of-bound and use-after-free bugs. KASAN has three modes: > + > 1. generic KASAN (similar to userspace ASan), > 2. software tag-based KASAN (similar to userspace HWASan), > 3. hardware tag-based KASAN (based on hardware memory tagging). > @@ -39,23 +40,13 @@ CONFIG_KASAN_INLINE. Outline and inline are compiler instrumentation types. > The former produces smaller binary while the latter is 1.1 - 2 times faster. > > Both software KASAN modes work with both SLUB and SLAB memory allocators, > -hardware tag-based KASAN currently only support SLUB. > -For better bug detection and nicer reporting, enable CONFIG_STACKTRACE. > +while the hardware tag-based KASAN currently only support SLUB. > + > +For better error reports that include stack traces, enable CONFIG_STACKTRACE. > > To augment reports with last allocation and freeing stack of the physical page, > it is recommended to enable also CONFIG_PAGE_OWNER and boot with page_owner=on. > > -To disable instrumentation for specific files or directories, add a line > -similar to the following to the respective kernel Makefile: > - > -- For a single file (e.g. main.o):: > - > - KASAN_SANITIZE_main.o := n > - > -- For all files in one directory:: > - > - KASAN_SANITIZE := n > - > Error reports > ~~~~~~~~~~~~~ > > @@ -140,22 +131,75 @@ freed (in case of a use-after-free bug report). Next comes a description of > the accessed slab object and information about the accessed memory page. > > In the last section the report shows memory state around the accessed address. > -Reading this part requires some understanding of how KASAN works. > - > -The state of each 8 aligned bytes of memory is encoded in one shadow byte. > -Those 8 bytes can be accessible, partially accessible, freed or be a redzone. > -We use the following encoding for each shadow byte: 0 means that all 8 bytes > -of the corresponding memory region are accessible; number N (1 <= N <= 7) means > -that the first N bytes are accessible, and other (8 - N) bytes are not; > -any negative value indicates that the entire 8-byte word is inaccessible. > -We use different negative values to distinguish between different kinds of > -inaccessible memory like redzones or freed memory (see mm/kasan/kasan.h). > +Internally KASAN tracks memory state separately for each memory granule, which > +is either 8 or 16 aligned bytes depending on KASAN mode. Each number in the > +memory state section of the report shows the state of one of the memory > +granules that surround the accessed address. > + > +For generic KASAN the size of each memory granule is 8. The state of each > +granule is encoded in one shadow byte. Those 8 bytes can be accessible, > +partially accessible, freed or be a part of a redzone. KASAN uses the following > +encoding for each shadow byte: 0 means that all 8 bytes of the corresponding > +memory region are accessible; number N (1 <= N <= 7) means that the first N > +bytes are accessible, and other (8 - N) bytes are not; any negative value > +indicates that the entire 8-byte word is inaccessible. KASAN uses different > +negative values to distinguish between different kinds of inaccessible memory > +like redzones or freed memory (see mm/kasan/kasan.h). > > In the report above the arrows point to the shadow byte 03, which means that > the accessed address is partially accessible. > > For tag-based KASAN this last report section shows the memory tags around the > -accessed address (see Implementation details section). > +accessed address (see `Implementation details`_ section). > + > +Boot parameters > +~~~~~~~~~~~~~~~ > + > +Hardware tag-based KASAN mode (see the section about different mode below) is > +intended for use in production as a security mitigation. Therefore it supports > +boot parameters that allow to disable KASAN competely or otherwise control > +particular KASAN features. > + > +The things that can be controlled are: > + > +1. Whether KASAN is enabled at all. > +2. Whether KASAN collects and saves alloc/free stacks. > +3. Whether KASAN panics on a detected bug or not. > + > +The ``kasan.mode`` boot parameter allows to choose one of three main modes: > + > +- ``kasan.mode=off`` - KASAN is disabled, no tag checks are performed > +- ``kasan.mode=prod`` - only essential production features are enabled > +- ``kasan.mode=full`` - all KASAN features are enabled > + > +The chosen mode provides default control values for the features mentioned > +above. However it's also possible to override the default values by providing: > + > +- ``kasan.stacktrace=off`` or ``=on`` - enable alloc/free stack collection > + (default: ``on`` for ``mode=full``, > + otherwise ``off``) > +- ``kasan.fault=report`` or ``=panic`` - only print KASAN report or also panic > + (default: ``report``) > + > +If ``kasan.mode`` parameter is not provided, it defaults to ``full`` when > +``CONFIG_DEBUG_KERNEL`` is enabled, and to ``prod`` otherwise. > + > +For developers > +~~~~~~~~~~~~~~ > + > +Software KASAN modes use compiler instrumentation to insert validity checks. > +Such instrumentation might be incompatible with some part of the kernel, and > +therefore needs to be disabled. To disable instrumentation for specific files > +or directories, add a line similar to the following to the respective kernel > +Makefile: > + > +- For a single file (e.g. main.o):: > + > + KASAN_SANITIZE_main.o := n > + > +- For all files in one directory:: > + > + KASAN_SANITIZE := n > > > Implementation details > @@ -164,10 +208,10 @@ Implementation details > Generic KASAN > ~~~~~~~~~~~~~ > > -From a high level, our approach to memory error detection is similar to that > -of kmemcheck: use shadow memory to record whether each byte of memory is safe > -to access, and use compile-time instrumentation to insert checks of shadow > -memory on each memory access. > +From a high level perspective, KASAN's approach to memory error detection is > +similar to that of kmemcheck: use shadow memory to record whether each byte of > +memory is safe to access, and use compile-time instrumentation to insert checks > +of shadow memory on each memory access. > > Generic KASAN dedicates 1/8th of kernel memory to its shadow memory (e.g. 16TB > to cover 128TB on x86_64) and uses direct mapping with a scale and offset to > @@ -194,7 +238,10 @@ function calls GCC directly inserts the code to check the shadow memory. > This option significantly enlarges kernel but it gives x1.1-x2 performance > boost over outline instrumented kernel. > > -Generic KASAN prints up to 2 call_rcu() call stacks in reports, the last one > +Generic KASAN is the only mode that delays the reuse of freed object via > +quarantine (see mm/kasan/quarantine.c for implementation). > + > +Generic KASAN prints up to two call_rcu() call stacks in reports, the last one > and the second to last. > > Software tag-based KASAN > @@ -304,15 +351,15 @@ therefore be wasteful. Furthermore, to ensure that different mappings > use different shadow pages, mappings would have to be aligned to > ``KASAN_GRANULE_SIZE * PAGE_SIZE``. > > -Instead, we share backing space across multiple mappings. We allocate > +Instead, KASAN shares backing space across multiple mappings. It allocates > a backing page when a mapping in vmalloc space uses a particular page > of the shadow region. This page can be shared by other vmalloc > mappings later on. > > -We hook in to the vmap infrastructure to lazily clean up unused shadow > +KASAN hooks into the vmap infrastructure to lazily clean up unused shadow > memory. > > -To avoid the difficulties around swapping mappings around, we expect > +To avoid the difficulties around swapping mappings around, KASAN expects > that the part of the shadow region that covers the vmalloc space will > not be covered by the early shadow page, but will be left > unmapped. This will require changes in arch-specific code. > @@ -323,24 +370,31 @@ architectures that do not have a fixed module region. > CONFIG_KASAN_KUNIT_TEST & CONFIG_TEST_KASAN_MODULE > -------------------------------------------------- > > -``CONFIG_KASAN_KUNIT_TEST`` utilizes the KUnit Test Framework for testing. > -This means each test focuses on a small unit of functionality and > -there are a few ways these tests can be run. > +KASAN tests consist on two parts: > + > +1. Tests that are integrated with the KUnit Test Framework. Enabled with > +``CONFIG_KASAN_KUNIT_TEST``. These tests can be run and partially verified > +automatically in a few different ways, see the instructions below. > > -Each test will print the KASAN report if an error is detected and then > -print the number of the test and the status of the test: > +2. Tests that are currently incompatible with KUnit. Enabled with > +``CONFIG_TEST_KASAN_MODULE`` and can only be run as a module. These tests can > +only be verified manually, by loading the kernel module and inspecting the > +kernel log for KASAN reports. > > -pass:: > +Each KUnit-compatible KASAN test prints a KASAN report if an error is detected. > +Then the test prints its number and status. > + > +When a test passes:: > > ok 28 - kmalloc_double_kzfree > > -or, if kmalloc failed:: > +When a test fails due to a failed ``kmalloc``:: > > # kmalloc_large_oob_right: ASSERTION FAILED at lib/test_kasan.c:163 > Expected ptr is not null, but is > not ok 4 - kmalloc_large_oob_right > > -or, if a KASAN report was expected, but not found:: > +When a test fails due to a missing KASAN report:: > > # kmalloc_double_kzfree: EXPECTATION FAILED at lib/test_kasan.c:629 > Expected kasan_data->report_expected == kasan_data->report_found, but > @@ -348,46 +402,38 @@ or, if a KASAN report was expected, but not found:: > kasan_data->report_found == 0 > not ok 28 - kmalloc_double_kzfree > > -All test statuses are tracked as they run and an overall status will > -be printed at the end:: > +At the end the cumulative status of all KASAN tests is printed. On success:: > > ok 1 - kasan > > -or:: > +Or, if one of the tests failed:: > > not ok 1 - kasan > > -(1) Loadable Module > -~~~~~~~~~~~~~~~~~~~~ > + > +There are a few ways to run KUnit-compatible KASAN tests. > + > +1. Loadable module > +~~~~~~~~~~~~~~~~~~ > > With ``CONFIG_KUNIT`` enabled, ``CONFIG_KASAN_KUNIT_TEST`` can be built as > -a loadable module and run on any architecture that supports KASAN > -using something like insmod or modprobe. The module is called ``test_kasan``. > +a loadable module and run on any architecture that supports KASAN by loading > +the module with insmod or modprobe. The module is called ``test_kasan``. > > -(2) Built-In > -~~~~~~~~~~~~~ > +2. Built-In > +~~~~~~~~~~~ > > With ``CONFIG_KUNIT`` built-in, ``CONFIG_KASAN_KUNIT_TEST`` can be built-in > -on any architecure that supports KASAN. These and any other KUnit > -tests enabled will run and print the results at boot as a late-init > -call. > +on any architecure that supports KASAN. These and any other KUnit tests enabled > +will run and print the results at boot as a late-init call. > > -(3) Using kunit_tool > -~~~~~~~~~~~~~~~~~~~~~ > +3. Using kunit_tool > +~~~~~~~~~~~~~~~~~~~ > > -With ``CONFIG_KUNIT`` and ``CONFIG_KASAN_KUNIT_TEST`` built-in, we can also > -use kunit_tool to see the results of these along with other KUnit > -tests in a more readable way. This will not print the KASAN reports > -of tests that passed. Use `KUnit documentation `_ for more up-to-date > -information on kunit_tool. > +With ``CONFIG_KUNIT`` and ``CONFIG_KASAN_KUNIT_TEST`` built-in, it's also > +possible use ``kunit_tool`` to see the results of these and other KUnit tests > +in a more readable way. This will not print the KASAN reports of the tests that > +passed. Use `KUnit documentation `_ > +for more up-to-date information on ``kunit_tool``. > > .. _KUnit: https://www.kernel.org/doc/html/latest/dev-tools/kunit/index.html > - > -``CONFIG_TEST_KASAN_MODULE`` is a set of KASAN tests that could not be > -converted to KUnit. These tests can be run only as a module with > -``CONFIG_TEST_KASAN_MODULE`` built as a loadable module and > -``CONFIG_KASAN`` built-in. The type of error expected and the > -function being run is printed before the expression expected to give > -an error. Then the error is printed, if found, and that test > -should be interpretted to pass only if the error was the one expected > -by the test. > -- > 2.29.2.299.gdc1121823c-goog >