Received: by 2002:a05:6358:3188:b0:123:57c1:9b43 with SMTP id q8csp2116242rwd; Thu, 15 Jun 2023 22:30:53 -0700 (PDT) X-Google-Smtp-Source: ACHHUZ7egi6pKPPWigCZB3xxGbC30TUZAsckwZd4GtuGY2BxnLfNXcMFcKiOCiYIG9ZB9qfBEWPB X-Received: by 2002:a05:6a20:2583:b0:106:5dff:5dc6 with SMTP id k3-20020a056a20258300b001065dff5dc6mr1735757pzd.16.1686893453134; Thu, 15 Jun 2023 22:30:53 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1686893453; cv=none; d=google.com; s=arc-20160816; b=fEtADpPklNpx6dK/T3MTBeHR9KQTUqayTnuowAzdLXoBHqvmf/hSq4trtSKBweR4qd My6+6PZWqiQUj7HtCQjJtKZPs6LyL+f6rwdEYYeyR5bPprRFjfmZbuaL0LAB7Oq9KJ2e 7dd647GRRvTPqslN2jdzOw2Wwu0c6Qb5nyFdH3KaSJF9bBtWwK9j0Clbk6si/2roedZW FYK32lHgfNNeV1eM58wKQNcaBp/tZVyUXglBcaFhZvSEwi8YqVcDBlLyJGPF948qW6Se +tqBwGf9zFKgikSIa+9kGNEGKfJMmFBwHZ1uomoJxMs7YvLtRz8LM48ESbCzviWS8KXO ZRhg== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:cc:to:subject:message-id:date:from:in-reply-to :references:mime-version:dkim-signature; bh=YJ3Fs6L7Cs7vHzQSbDH7lYJEv08a6Q+QOnaauah+Wug=; b=iHLT8t6XdEUUYjjSqAO90j76MgP04un9q9vitGeHfPfj/A6n8zPa63g8Wn9GWjlHc+ VO34/iSSjeC7QsZP2IcbT2fu06lGFdOfcJqX8T1hYDlmgOyiQiT5wEyPsT47ydgaKkCv zSI6ENBPs2TFY+gRZvbge2LieGL7XtQPoSFZrCgNCOWUkCWW/kKNnZVOXiJc07rel0GF rlueE33EJocTnD3ucdTaAkPVNxva58qXAAvdg1re71MiirpE+dugBE0c7j/5za6Qxbuj psdnEB8wwP+U2hlZKsQx+S1gK5urOq6/m0+T1NG1Z+p/8JrnIwnKTyZG0NiYtgyJdamJ hrSQ== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@google.com header.s=20221208 header.b=i3HCCE+j; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 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 out1.vger.email (out1.vger.email. [2620:137:e000::1:20]) by mx.google.com with ESMTP id v71-20020a63894a000000b0055339136a78si1686357pgd.517.2023.06.15.22.30.39; Thu, 15 Jun 2023 22:30:52 -0700 (PDT) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) client-ip=2620:137:e000::1:20; Authentication-Results: mx.google.com; dkim=pass header.i=@google.com header.s=20221208 header.b=i3HCCE+j; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 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 S239240AbjFPExN (ORCPT + 99 others); Fri, 16 Jun 2023 00:53:13 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:56954 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S238246AbjFPEwf (ORCPT ); Fri, 16 Jun 2023 00:52:35 -0400 Received: from mail-ed1-x531.google.com (mail-ed1-x531.google.com [IPv6:2a00:1450:4864:20::531]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id D4A052D68 for ; Thu, 15 Jun 2023 21:52:25 -0700 (PDT) Received: by mail-ed1-x531.google.com with SMTP id 4fb4d7f45d1cf-514ad92d1e3so3518a12.1 for ; Thu, 15 Jun 2023 21:52:25 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=20221208; t=1686891144; x=1689483144; h=cc:to:subject:message-id:date:from:in-reply-to:references :mime-version:from:to:cc:subject:date:message-id:reply-to; bh=YJ3Fs6L7Cs7vHzQSbDH7lYJEv08a6Q+QOnaauah+Wug=; b=i3HCCE+jk6OxRgotqlh7miRYkjRN1oQ2CWKTuPGb/tPnaLCRU+Gm50w0MlUzoUbPyc EzJ4J5lzSjyxkYNtyvyaeVUSLxk6aorW7KrwFcSRGOM6eXe0JEdiD4P57NRroXihGBA8 SRuzJCDR5XT3wHpfLTEpIFyfFIp2nyTB43i881tviVzuj+Dyy7Jw6uzmfbwt9RHYU6fA uqrUN/LGiDfwozIqAopq/mG5/8XPIVH+vubM0m2swpHxKb+HYt0ohNb6RiDdOOIozLvW agpKX1CHTaYyRtPjutwDRl4jq6FUfQ6bp66wS/UFiFaKmX9qJ6/vv7jDDw+5EnrWHQfb x0BQ== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20221208; t=1686891144; x=1689483144; h=cc:to:subject:message-id:date:from:in-reply-to:references :mime-version:x-gm-message-state:from:to:cc:subject:date:message-id :reply-to; bh=YJ3Fs6L7Cs7vHzQSbDH7lYJEv08a6Q+QOnaauah+Wug=; b=MoyFy5/IS6Km0RvVOR8D0u0nkPC3hKjsFCa7GgnQpSgRpHJ7SuwNYEjCzDCLXl5Yx8 7rFesmGXMvF76REmGiEibP/+AP3vPGlyPjl3H44LkHuUdR9o8bWVRgMe8ilQerrMlYpE n1qlXhvjxaUDdd4506L/IJgQDRF/72RsgA7JX7JMEwB0wzYa6ybxr3IcEwVh9QXJCC6E 2Mxp80Rnd0TRlidTGREHhWW33L/OA39Xgngnp9qyD0/m+A86MFLTBf47DRAHLU3i7GcE jtHs+KmgblNKWs04UNf+UcbPGqkwp5W0pYw8kQbeyFggRqMDiQhLuwdv+ETZOeMqdSp0 5yhw== X-Gm-Message-State: AC+VfDw/MOLp60MCPk+FsiuuWj8Mi7knx0Tvpf7E62xV6IBHM65eXHI6 KWC3uLpGGSAFMDVsTRIG1oukpPC/WxFzLP2zzLCblA== X-Received: by 2002:a50:9e07:0:b0:50b:f6ce:2f3d with SMTP id z7-20020a509e07000000b0050bf6ce2f3dmr199435ede.0.1686891144027; Thu, 15 Jun 2023 21:52:24 -0700 (PDT) MIME-Version: 1.0 References: <20230614180837.630180-1-ojeda@kernel.org> <20230614180837.630180-6-ojeda@kernel.org> In-Reply-To: <20230614180837.630180-6-ojeda@kernel.org> From: David Gow Date: Fri, 16 Jun 2023 12:52:11 +0800 Message-ID: Subject: Re: [PATCH 5/6] rust: support running Rust documentation tests as KUnit ones To: Miguel Ojeda Cc: Brendan Higgins , Wedson Almeida Filho , Alex Gaynor , Boqun Feng , Gary Guo , =?UTF-8?Q?Bj=C3=B6rn_Roy_Baron?= , Benno Lossin , Alice Ryhl , Andreas Hindborg , Philip Li , kunit-dev@googlegroups.com, linux-kselftest@vger.kernel.org, rust-for-linux@vger.kernel.org, linux-kernel@vger.kernel.org, patches@lists.linux.dev Content-Type: multipart/signed; protocol="application/pkcs7-signature"; micalg=sha-256; boundary="00000000000073202105fe37f3fb" X-Spam-Status: No, score=-17.6 required=5.0 tests=BAYES_00,DKIMWL_WL_MED, DKIM_SIGNED,DKIM_VALID,DKIM_VALID_AU,DKIM_VALID_EF, ENV_AND_HDR_SPF_MATCH,RCVD_IN_DNSWL_NONE,SPF_HELO_NONE,SPF_PASS, T_SCC_BODY_TEXT_LINE,URIBL_BLOCKED,USER_IN_DEF_DKIM_WL, USER_IN_DEF_SPF_WL autolearn=ham autolearn_force=no version=3.4.6 X-Spam-Checker-Version: SpamAssassin 3.4.6 (2021-04-09) on lindbergh.monkeyblade.net Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org --00000000000073202105fe37f3fb Content-Type: text/plain; charset="UTF-8" On Thu, 15 Jun 2023 at 02:09, Miguel Ojeda wrote: > > Rust has documentation tests: these are typically examples of > usage of any item (e.g. function, struct, module...). > > They are very convenient because they are just written > alongside the documentation. For instance: > > /// Sums two numbers. > /// > /// ``` > /// assert_eq!(mymod::f(10, 20), 30); > /// ``` > pub fn f(a: i32, b: i32) -> i32 { > a + b > } > > In userspace, the tests are collected and run via `rustdoc`. > Using the tool as-is would be useful already, since it allows > to compile-test most tests (thus enforcing they are kept > in sync with the code they document) and run those that do not > depend on in-kernel APIs. > > However, by transforming the tests into a KUnit test suite, > they can also be run inside the kernel. Moreover, the tests > get to be compiled as other Rust kernel objects instead of > targeting userspace. > > On top of that, the integration with KUnit means the Rust > support gets to reuse the existing testing facilities. For > instance, the kernel log would look like: > > KTAP version 1 > 1..1 > KTAP version 1 > # Subtest: rust_doctests_kernel > 1..59 > # Doctest from line 13 > ok 1 rust_doctest_kernel_build_assert_rs_0 > # Doctest from line 56 > ok 2 rust_doctest_kernel_build_assert_rs_1 > # Doctest from line 122 > ok 3 rust_doctest_kernel_init_rs_0 > ... > # Doctest from line 150 > ok 59 rust_doctest_kernel_types_rs_2 > # rust_doctests_kernel: pass:59 fail:0 skip:0 total:59 > # Totals: pass:59 fail:0 skip:0 total:59 > ok 1 rust_doctests_kernel > > Therefore, add support for running Rust documentation tests > in KUnit. Some other notes about the current implementation > and support follow. This is very neat, and works very well here -- thanks! I can reproduce your above results with: ./tools/testing/kunit/kunit.py run --kconfig_add CONFIG_RUST=y --make_options LLVM=1 'rust_doctests_kernel' (And also with --arch x86_64 to run under Qemu, rather than UML) > > The transformation is performed by a couple scripts written > as Rust hostprogs. > > Tests using the `?` operator are also supported as usual, e.g.: > > /// ``` > /// # use kernel::{spawn_work_item, workqueue}; > /// spawn_work_item!(workqueue::system(), || pr_info!("x"))?; > /// # Ok::<(), Error>(()) > /// ``` > > The tests are also compiled with Clippy under `CLIPPY=1`, just like > normal code, thus also benefitting from extra linting. > > The names of the tests are currently automatically generated. > This allows to reduce the burden for documentation writers, > while keeping them fairly stable for bisection. This is an > improvement over the `rustdoc`-generated names, which include > the line number; but ideally we would like to get `rustdoc` to > provide the Rust item path and a number (for multiple examples > in a single documented Rust item). Yeah, it'd be great to have rustdoc generate a better name. I agree that _ is the best balance of stability and descriptivity in the meantime. It should only change if new tests are added or reordered, not just with any change to the actual code. The other idea we discussed of doing, e.g., hashes of the test source would be more stable, but the ugliness of it probably isn't worth it. > In order for developers to easily see from which original line > a failed doctests came from, a KTAP diagnostic line is printed > to the log. In the future, we may be able to use a proper KUnit > facility to append this sort of information instead. Yeah, we're looking at a general "test attributes" feature, which would provide a standard way of providing extra data about test cases, which would be included in the output, and could even be filtered upon. There's a link to it below. > A notable difference from KUnit C tests is that the Rust tests > appear to assert using the usual `assert!` and `assert_eq!` > macros from the Rust standard library (`core`). We provide > a custom version that forwards the call to KUnit instead. > Importantly, these macros do not require passing context, > unlike the KUnit C ones (i.e. `struct kunit *`). This makes > them easier to use, and readers of the documentation do not need > to care about which testing framework is used. In addition, it > may allow us to test third-party code more easily in the future. While I have some reservations about this in general, I think it makes sense for the doctests at least. It may be worth also supplying "more direct" bindings for features which map less well to existing Rust assertions and features, such as expectations, resources, mocks, etc. However, I think we can add these (a) as we develop tests which would benefit for them, and (b) once we expand the scope of KUnit testing beyond just doctests. The context becoming implicit is also a tradeoff (as mentioned a bit below), but I think it's worth doing for those stated reasons. If it proves really successful, we can look into changing the C API to implicitly use the current task's test context as well, though we'll accept the divergence for now. > However, a current limitation is that KUnit does not support > assertions in other tasks. Thus we presently simply print an > error to the kernel log if an assertion actually failed. This > should be revisited to properly fail the test, perhaps saving > the context somewhere else, or letting KUnit handle it. Yes: this is a restriction on the KUnit execution environment at the moment. Tests are expected to be single-threaded, or at least to manage any extra threads themselves. This means that: - assertions don't work in non-main threads (though the KUnit 'expectation' feature does, which we can expose in Rust if it is needed) - it's the user's responsibility to make sure all threads exit when the test's main thread ends. These obviously aren't ideal (particularly compared to the level of safety Rust usually provides), but we'll look at better ways of handling tests which spawn threads in KUnit going forward. > Signed-off-by: Miguel Ojeda > --- This looks good to me. I've left a few comments below, but none of them are deal breakers if you feel strongly about keeping things the way they are. Otherwise, Reviewed-by: David Gow Thanks a bunch, -- David > lib/Kconfig.debug | 13 +++ > rust/.gitignore | 2 + > rust/Makefile | 29 ++++++ > rust/bindings/bindings_helper.h | 1 + > rust/helpers.c | 7 ++ > rust/kernel/kunit.rs | 156 ++++++++++++++++++++++++++++++ > rust/kernel/lib.rs | 2 + > scripts/.gitignore | 2 + > scripts/Makefile | 4 + > scripts/rustdoc_test_builder.rs | 73 ++++++++++++++ > scripts/rustdoc_test_gen.rs | 162 ++++++++++++++++++++++++++++++++ > 11 files changed, 451 insertions(+) > create mode 100644 rust/kernel/kunit.rs > create mode 100644 scripts/rustdoc_test_builder.rs > create mode 100644 scripts/rustdoc_test_gen.rs > > diff --git a/lib/Kconfig.debug b/lib/Kconfig.debug > index ce51d4dc6803..49f5e9c42200 100644 > --- a/lib/Kconfig.debug > +++ b/lib/Kconfig.debug > @@ -2911,6 +2911,19 @@ config RUST_BUILD_ASSERT_ALLOW > > If unsure, say N. > > +config RUST_KERNEL_KUNIT_TEST > + bool "KUnit test for the `kernel` crate" if !KUNIT_ALL_TESTS I wonder whether it'd make sense to have the name / description here call out doctests specifically. If we want to add Rust support for more explicit KUnit tests, as in https://github.com/Rust-for-Linux/linux/pull/950 then those might have a better claim to the _KUNIT_TEST suffix. I'd be happy if this became RUST_KERNEL_DOCTESTS or similar. > + depends on RUST && KUNIT=y > + default KUNIT_ALL_TESTS > + help > + This builds the documentation tests of the `kernel` crate > + as KUnit tests. > + > + For more information on KUnit and unit tests in general, > + please refer to the KUnit documentation in Documentation/dev-tools/kunit/. > + > + If unsure, say N. > + > endmenu # "Rust" > > endmenu # Kernel hacking > diff --git a/rust/.gitignore b/rust/.gitignore > index 21552992b401..d3829ffab80b 100644 > --- a/rust/.gitignore > +++ b/rust/.gitignore > @@ -2,6 +2,8 @@ > > bindings_generated.rs > bindings_helpers_generated.rs > +doctests_kernel_generated.rs > +doctests_kernel_generated_kunit.c > uapi_generated.rs > exports_*_generated.h > doc/ > diff --git a/rust/Makefile b/rust/Makefile > index 7c9d9f11aec5..e4e8b83752e2 100644 > --- a/rust/Makefile > +++ b/rust/Makefile > @@ -27,6 +27,12 @@ endif > > obj-$(CONFIG_RUST) += exports.o > > +always-$(CONFIG_RUST_KERNEL_KUNIT_TEST) += doctests_kernel_generated.rs > +always-$(CONFIG_RUST_KERNEL_KUNIT_TEST) += doctests_kernel_generated_kunit.c > + > +obj-$(CONFIG_RUST_KERNEL_KUNIT_TEST) += doctests_kernel_generated.o > +obj-$(CONFIG_RUST_KERNEL_KUNIT_TEST) += doctests_kernel_generated_kunit.o > + > # Avoids running `$(RUSTC)` for the sysroot when it may not be available. > ifdef CONFIG_RUST > > @@ -39,9 +45,11 @@ ifeq ($(quiet),silent_) > cargo_quiet=-q > rust_test_quiet=-q > rustdoc_test_quiet=--test-args -q > +rustdoc_test_kernel_quiet=>/dev/null > else ifeq ($(quiet),quiet_) > rust_test_quiet=-q > rustdoc_test_quiet=--test-args -q > +rustdoc_test_kernel_quiet=>/dev/null > else > cargo_quiet=--verbose > endif > @@ -157,6 +165,27 @@ quiet_cmd_rustdoc_test = RUSTDOC T $< > -L$(objtree)/$(obj)/test --output $(objtree)/$(obj)/doc \ > --crate-name $(subst rusttest-,,$@) $< > > +quiet_cmd_rustdoc_test_kernel = RUSTDOC TK $< > + cmd_rustdoc_test_kernel = \ > + rm -rf $(objtree)/$(obj)/test/doctests/kernel; \ > + mkdir -p $(objtree)/$(obj)/test/doctests/kernel; \ > + OBJTREE=$(abspath $(objtree)) \ > + $(RUSTDOC) --test $(rust_flags) \ > + @$(objtree)/include/generated/rustc_cfg \ > + -L$(objtree)/$(obj) --extern alloc --extern kernel \ > + --extern build_error --extern macros \ > + --extern bindings --extern uapi \ > + --no-run --crate-name kernel -Zunstable-options \ > + --test-builder $(objtree)/scripts/rustdoc_test_builder \ > + $< $(rustdoc_test_kernel_quiet); \ > + $(objtree)/scripts/rustdoc_test_gen > + > +%/doctests_kernel_generated.rs %/doctests_kernel_generated_kunit.c: \ > + $(src)/kernel/lib.rs $(obj)/kernel.o \ > + $(objtree)/scripts/rustdoc_test_builder \ > + $(objtree)/scripts/rustdoc_test_gen FORCE > + $(call if_changed,rustdoc_test_kernel) > + > # We cannot use `-Zpanic-abort-tests` because some tests are dynamic, > # so for the moment we skip `-Cpanic=abort`. > quiet_cmd_rustc_test = RUSTC T $< > diff --git a/rust/bindings/bindings_helper.h b/rust/bindings/bindings_helper.h > index 3e601ce2548d..0f8d37c31ac2 100644 > --- a/rust/bindings/bindings_helper.h > +++ b/rust/bindings/bindings_helper.h > @@ -6,6 +6,7 @@ > * Sorted alphabetically. > */ > > +#include > #include > #include > #include > diff --git a/rust/helpers.c b/rust/helpers.c > index bb594da56137..49a5e1a4f0ae 100644 > --- a/rust/helpers.c > +++ b/rust/helpers.c > @@ -18,6 +18,7 @@ > * accidentally exposed. > */ > > +#include > #include > #include > #include > @@ -135,6 +136,12 @@ void rust_helper_put_task_struct(struct task_struct *t) > } > EXPORT_SYMBOL_GPL(rust_helper_put_task_struct); > > +struct kunit *rust_helper_kunit_get_current_test(void) > +{ > + return kunit_get_current_test(); > +} > +EXPORT_SYMBOL_GPL(rust_helper_kunit_get_current_test); > + > /* > * We use `bindgen`'s `--size_t-is-usize` option to bind the C `size_t` type > * as the Rust `usize` type, so we can use it in contexts where Rust > diff --git a/rust/kernel/kunit.rs b/rust/kernel/kunit.rs > new file mode 100644 > index 000000000000..3c94efcd7f76 > --- /dev/null > +++ b/rust/kernel/kunit.rs > @@ -0,0 +1,156 @@ > +// SPDX-License-Identifier: GPL-2.0 > + > +//! KUnit-based macros for Rust unit tests. > +//! > +//! C header: [`include/kunit/test.h`](../../../../../include/kunit/test.h) > +//! > +//! Reference: > + > +use core::{ffi::c_void, fmt}; > + > +/// Prints a KUnit error. > +/// > +/// Public but hidden since it should only be used from KUnit generated code. > +#[doc(hidden)] > +pub fn err(args: fmt::Arguments<'_>) { > + // SAFETY: The format string is null-terminated and the `%pA` specifier matches the argument we > + // are passing. > + #[cfg(CONFIG_PRINTK)] > + unsafe { > + bindings::_printk( > + b"\x013%pA\0".as_ptr() as _, > + &args as *const _ as *const c_void, > + ); > + } > +} > + > +/// Prints a KUnit error. > +/// > +/// Public but hidden since it should only be used from KUnit generated code. > +#[doc(hidden)] > +pub fn info(args: fmt::Arguments<'_>) { > + // SAFETY: The format string is null-terminated and the `%pA` specifier matches the argument we > + // are passing. > + #[cfg(CONFIG_PRINTK)] > + unsafe { > + bindings::_printk( > + b"\x016%pA\0".as_ptr() as _, > + &args as *const _ as *const c_void, > + ); > + } > +} > + > +/// Asserts that a boolean expression is `true` at runtime. > +/// > +/// Public but hidden since it should only be used from generated tests. > +/// > +/// Unlike the one in `core`, this one does not panic; instead, it is mapped to the KUnit > +/// facilities. See [`assert!`] for more details. > +#[doc(hidden)] > +#[macro_export] > +macro_rules! kunit_assert { > + ($name:literal, $condition:expr $(,)?) => { > + 'out: { > + // Do nothing if the condition is `true`. > + if $condition { > + break 'out; > + } > + > + static LINE: i32 = core::line!() as i32; > + static FILE: &'static $crate::str::CStr = $crate::c_str!(core::file!()); > + static CONDITION: &'static $crate::str::CStr = $crate::c_str!(stringify!($condition)); > + > + // SAFETY: FFI call without safety requirements. > + let kunit_test = unsafe { $crate::bindings::kunit_get_current_test() }; > + if kunit_test.is_null() { > + // The assertion failed but this task is not running a KUnit test, so we cannot call > + // KUnit, but at least print an error to the kernel log. This may happen if this > + // macro is called from an spawned thread in a test (see > + // `scripts/rustdoc_test_gen.rs`) or if some non-test code calls this macro by > + // mistake (it is hidden to prevent that). > + // > + // This mimics KUnit's failed assertion format. > + $crate::kunit::err(format_args!( > + " # {}: ASSERTION FAILED at {FILE}:{LINE}\n", > + $name > + )); > + $crate::kunit::err(format_args!( > + " Expected {CONDITION} to be true, but is false\n" > + )); > + $crate::kunit::err(format_args!( > + " Failure not reported to KUnit since this is a non-KUnit task\n" > + )); > + break 'out; > + } > + > + #[repr(transparent)] > + struct Location($crate::bindings::kunit_loc); > + > + #[repr(transparent)] > + struct UnaryAssert($crate::bindings::kunit_unary_assert); > + > + // SAFETY: There is only a static instance and in that one the pointer field points to > + // an immutable C string. > + unsafe impl Sync for Location {} > + > + // SAFETY: There is only a static instance and in that one the pointer field points to > + // an immutable C string. > + unsafe impl Sync for UnaryAssert {} > + > + static LOCATION: Location = Location($crate::bindings::kunit_loc { > + file: FILE.as_char_ptr(), > + line: LINE, > + }); > + static ASSERTION: UnaryAssert = UnaryAssert($crate::bindings::kunit_unary_assert { > + assert: $crate::bindings::kunit_assert {}, > + condition: CONDITION.as_char_ptr(), > + expected_true: true, > + }); > + > + // SAFETY: > + // - FFI call. > + // - The `kunit_test` pointer is valid because we got it from > + // `kunit_get_current_test()` and it was not null. This means we are in a KUnit > + // test, and that the pointer can be passed to KUnit functions and assertions. > + // - The string pointers (`file` and `condition` above) point to null-terminated > + // strings since they are `CStr`s. > + // - The function pointer (`format`) points to the proper function. > + // - The pointers passed will remain valid since they point to `static`s. > + // - The format string is allowed to be null. > + // - There are, however, problems with this: first of all, this will end up stopping > + // the thread, without running destructors. While that is problematic in itself, > + // it is considered UB to have what is effectively a forced foreign unwind > + // with `extern "C"` ABI. One could observe the stack that is now gone from > + // another thread. We should avoid pinning stack variables to prevent library UB, > + // too. For the moment, given that test failures are reported immediately before the > + // next test runs, that test failures should be fixed and that KUnit is explicitly > + // documented as not suitable for production environments, we feel it is reasonable. > + unsafe { > + $crate::bindings::kunit_do_failed_assertion( > + kunit_test, > + core::ptr::addr_of!(LOCATION.0), > + $crate::bindings::kunit_assert_type_KUNIT_ASSERTION, > + core::ptr::addr_of!(ASSERTION.0.assert), > + Some($crate::bindings::kunit_unary_assert_format), > + core::ptr::null(), > + ); This will conflict with: https://git.kernel.org/pub/scm/linux/kernel/git/shuah/linux-kselftest.git/commit/?h=kunit&id=260755184cbdb267a046e7ffd397c1d2ba09bb5e Basically, kunit_do_failed_assertion() has gained a couple of underscores to become __kunit_do_failed_assertion(), and it no longer aborts the test itself, requiring an extra call to __kunit_abort(). With those two changes, this works well here: I assume they'll be picked up on a rebase if we want to push this via the kselftest/kunit tree for 6.6. > + } > + } > + }; > +} > + > +/// Asserts that two expressions are equal to each other (using [`PartialEq`]). > +/// > +/// Public but hidden since it should only be used from generated tests. > +/// > +/// Unlike the one in `core`, this one does not panic; instead, it is mapped to the KUnit > +/// facilities. See [`assert!`] for more details. > +#[doc(hidden)] > +#[macro_export] > +macro_rules! kunit_assert_eq { > + ($name:literal, $left:expr, $right:expr $(,)?) => {{ > + // For the moment, we just forward to the expression assert because, for binary asserts, > + // KUnit supports only a few types (e.g. integers). > + $crate::kunit_assert!($name, $left == $right); > + }}; > +} > diff --git a/rust/kernel/lib.rs b/rust/kernel/lib.rs > index 85b261209977..3642cadc34b1 100644 > --- a/rust/kernel/lib.rs > +++ b/rust/kernel/lib.rs > @@ -34,6 +34,8 @@ > pub mod error; > pub mod init; > pub mod ioctl; > +#[cfg(CONFIG_KUNIT)] > +pub mod kunit; > pub mod prelude; > pub mod print; > mod static_assert; > diff --git a/scripts/.gitignore b/scripts/.gitignore > index 6e9ce6720a05..3dbb8bb2457b 100644 > --- a/scripts/.gitignore > +++ b/scripts/.gitignore > @@ -5,6 +5,8 @@ > /kallsyms > /module.lds > /recordmcount > +/rustdoc_test_builder > +/rustdoc_test_gen > /sign-file > /sorttable > /target.json > diff --git a/scripts/Makefile b/scripts/Makefile > index 32b6ba722728..d5a5382e753c 100644 > --- a/scripts/Makefile > +++ b/scripts/Makefile > @@ -9,6 +9,8 @@ hostprogs-always-$(CONFIG_BUILDTIME_TABLE_SORT) += sorttable > hostprogs-always-$(CONFIG_ASN1) += asn1_compiler > hostprogs-always-$(CONFIG_MODULE_SIG_FORMAT) += sign-file > hostprogs-always-$(CONFIG_SYSTEM_EXTRA_CERTIFICATE) += insert-sys-cert > +hostprogs-always-$(CONFIG_RUST_KERNEL_KUNIT_TEST) += rustdoc_test_builder > +hostprogs-always-$(CONFIG_RUST_KERNEL_KUNIT_TEST) += rustdoc_test_gen > always-$(CONFIG_RUST) += target.json > > filechk_rust_target = $< < include/config/auto.conf > @@ -18,6 +20,8 @@ $(obj)/target.json: scripts/generate_rust_target include/config/auto.conf FORCE > > hostprogs += generate_rust_target > generate_rust_target-rust := y > +rustdoc_test_builder-rust := y > +rustdoc_test_gen-rust := y > > HOSTCFLAGS_sorttable.o = -I$(srctree)/tools/include > HOSTLDLIBS_sorttable = -lpthread > diff --git a/scripts/rustdoc_test_builder.rs b/scripts/rustdoc_test_builder.rs > new file mode 100644 > index 000000000000..e3b7138fb4f9 > --- /dev/null > +++ b/scripts/rustdoc_test_builder.rs > @@ -0,0 +1,73 @@ > +// SPDX-License-Identifier: GPL-2.0 > + > +//! Test builder for `rustdoc`-generated tests. > +//! > +//! This script is a hack to extract the test from `rustdoc`'s output. Ideally, `rustdoc` would > +//! have an option to generate this information instead, e.g. as JSON output. > +//! > +//! The `rustdoc`-generated test names look like `{file}_{line}_{number}`, e.g. > +//! `...path_rust_kernel_sync_arc_rs_42_0`. `number` is the "test number", needed in cases like > +//! a macro that expands into items with doctests is invoked several times within the same line. > +//! > +//! However, since these names are used for bisection in CI, the line number makes it not stable > +//! at all. In the future, we would like `rustdoc` to give us the Rust item path associated with > +//! the test, plus a "test number" (for cases with several examples per item) and generate a name > +//! from that. For the moment, we generate ourselves a new name, `{file}_{number}` instead, in > +//! the `gen` script (done there since we need to be aware of all the tests in a given file). > + > +use std::fs::File; > +use std::io::{BufWriter, Read, Write}; > + > +fn main() { > + let mut stdin = std::io::stdin().lock(); > + let mut body = String::new(); > + stdin.read_to_string(&mut body).unwrap(); > + > + // Find the generated function name looking for the inner function inside `main()`. > + // > + // The line we are looking for looks like one of the following: > + // > + // ``` > + // fn main() { #[allow(non_snake_case)] fn _doctest_main_rust_kernel_file_rs_28_0() { > + // fn main() { #[allow(non_snake_case)] fn _doctest_main_rust_kernel_file_rs_37_0() -> Result<(), impl core::fmt::Debug> { > + // ``` > + // > + // It should be unlikely that doctest code matches such lines (when code is formatted properly). > + let rustdoc_function_name = body > + .lines() > + .find_map(|line| { > + Some( > + line.split_once("fn main() {")? > + .1 > + .split_once("fn ")? > + .1 > + .split_once("()")? > + .0, > + ) > + .filter(|x| x.chars().all(|c| c.is_alphanumeric() || c == '_')) > + }) > + .expect("No test function found in `rustdoc`'s output."); > + > + // Qualify `Result` to avoid the collision with our own `Result` coming from the prelude. > + let body = body.replace( > + &format!("{rustdoc_function_name}() -> Result<(), impl core::fmt::Debug> {{"), > + &format!("{rustdoc_function_name}() -> core::result::Result<(), impl core::fmt::Debug> {{"), > + ); > + > + // For tests that get generated with `Result`, like above, `rustdoc` generates an `unwrap()` on > + // the return value to check there were no returned errors. Instead, we use our assert macro > + // since we want to just fail the test, not panic the kernel. > + // > + // We save the result in a variable so that the failed assertion message looks nicer. > + let body = body.replace( > + &format!("}} {rustdoc_function_name}().unwrap() }}"), > + &format!("}} let test_return_value = {rustdoc_function_name}(); assert!(test_return_value.is_ok()); }}"), > + ); > + > + // Figure out a smaller test name based on the generated function name. > + let name = rustdoc_function_name.split_once("_rust_kernel_").unwrap().1; > + > + let path = format!("rust/test/doctests/kernel/{name}"); > + > + write!(BufWriter::new(File::create(path).unwrap()), "{body}").unwrap(); > +} > diff --git a/scripts/rustdoc_test_gen.rs b/scripts/rustdoc_test_gen.rs > new file mode 100644 > index 000000000000..793885c32c0d > --- /dev/null > +++ b/scripts/rustdoc_test_gen.rs > @@ -0,0 +1,162 @@ > +// SPDX-License-Identifier: GPL-2.0 > + > +//! Generates KUnit tests from saved `rustdoc`-generated tests. > +//! > +//! KUnit passes a context (`struct kunit *`) to each test, which should be forwarded to the other > +//! KUnit functions and macros. > +//! > +//! However, we want to keep this as an implementation detail because: > +//! > +//! - Test code should not care about the implementation. > +//! > +//! - Documentation looks worse if it needs to carry extra details unrelated to the piece > +//! being described. > +//! > +//! - Test code should be able to define functions and call them, without having to carry > +//! the context. > +//! > +//! - Later on, we may want to be able to test non-kernel code (e.g. `core`, `alloc` or > +//! third-party crates) which likely use the standard library `assert*!` macros. > +//! > +//! For this reason, instead of the passed context, `kunit_get_current_test()` is used instead > +//! (i.e. `current->kunit_test`). While I think there are some advantages to passing the `struct kunit *` context around manually (in particular, I suspect a number of the lifecycle issues we have with threads would actually be nicely modelled by the Rust ownership/borrowing model on the context), I agree that being more consistent with the way doctests work outside the kernel is the better path here. We may revisit this with future KUnit bindings, though, or we may update the C version to use kunit_get_current_test() more. (Particularly once we have a better solution for the threading issues below.) > +//! > +//! Note that this means other threads/tasks potentially spawned by a given test, if failing, will > +//! report the failure in the kernel log but will not fail the actual test. Saving the pointer in > +//! e.g. a `static` per test does not fully solve the issue either, because currently KUnit does > +//! not support assertions (only expectations) from other tasks. Thus leave that feature for > +//! the future, which simplifies the code here too. We could also simply not allow `assert`s in > +//! other tasks, but that seems overly constraining, and we do want to support them, eventually. > + > +use std::io::{BufWriter, Read, Write}; > +use std::{fs, fs::File}; > + > +fn main() { > + let mut paths = fs::read_dir("rust/test/doctests/kernel") > + .unwrap() > + .map(|entry| entry.unwrap().path()) > + .collect::>(); > + > + // Sort paths for clarity. > + paths.sort(); > + > + let mut rust_tests = String::new(); > + let mut c_test_declarations = String::new(); > + let mut c_test_cases = String::new(); > + let mut body = String::new(); > + let mut last_file = String::new(); > + let mut number = 0; > + for path in paths { > + // The `name` follows the `{file}_{line}_{number}` pattern (see description in > + // `scripts/rustdoc_test_builder.rs`). Discard the `number`. > + let name = path.file_name().unwrap().to_str().unwrap().to_string(); > + > + // Extract the `file` and the `line`, discarding the `number`. > + let (file, line) = name.rsplit_once('_').unwrap().0.rsplit_once('_').unwrap(); > + > + // Generate an ID sequence ("test number") for each one in the file. > + if file == last_file { > + number += 1; > + } else { > + number = 0; > + last_file = file.to_string(); > + } > + > + // Generate a KUnit name (i.e. test name and C symbol) for this test. > + // > + // We avoid the line number, like `rustdoc` does, to make things slightly more stable for > + // bisection purposes. However, to aid developers in mapping back what test failed, we will > + // print a diagnostics line in the KTAP report. > + let kunit_name = format!("rust_doctest_kernel_{file}_{number}"); > + > + // Read the test's text contents to dump it below. > + body.clear(); > + File::open(path).unwrap().read_to_string(&mut body).unwrap(); > + > + let line = line.parse::().unwrap(); > + > + use std::fmt::Write; > + write!( > + rust_tests, > + r#"/// Generated `{name}` KUnit test case from a Rust documentation test. > +#[no_mangle] > +pub extern "C" fn {kunit_name}(__kunit_test: *mut kernel::bindings::kunit) {{ > + /// Overrides the usual [`assert!`] macro with one that calls KUnit instead. > + #[allow(unused)] > + macro_rules! assert {{ > + ($cond:expr $(,)?) => {{{{ > + kernel::kunit_assert!("{kunit_name}", $cond); > + }}}} > + }} > + > + /// Overrides the usual [`assert_eq!`] macro with one that calls KUnit instead. > + #[allow(unused)] > + macro_rules! assert_eq {{ > + ($left:expr, $right:expr $(,)?) => {{{{ > + kernel::kunit_assert_eq!("{kunit_name}", $left, $right); > + }}}} > + }} > + > + // Many tests need the prelude, so provide it by default. > + #[allow(unused)] > + use kernel::prelude::*; > + > + // Display line number so that developers can map the test easily to the source code. > + kernel::kunit::info(format_args!(" # Doctest from line {line}\n")); Ideally, I think this should try to follow the (still in progress) KTAP attribute format: https://lore.kernel.org/lkml/20230420205734.1288498-1-rmoar@google.com/T/ Ultimately, that should probably be implemented as a KUnit attribute, but just printing something like "# line: {}" or "# .line: {}" ought to work until that's landed: https://lore.kernel.org/linux-kselftest/20230610005149.1145665-1-rmoar@google.com/ I think Boqun's suggestion from the cover letter to map assertion file/lines is better still, but that obviously only triggers when an assertion actually fails. And while kunit.py does hide diagnostic lines when a test passes, something like this could still be useful for people running KUnit manually, or using the --raw_output option. > + > + {{ > + {body} > + main(); > + }} > +}} > + > +"# > + ) > + .unwrap(); > + > + write!(c_test_declarations, "void {kunit_name}(struct kunit *);\n").unwrap(); > + write!(c_test_cases, " KUNIT_CASE({kunit_name}),\n").unwrap(); > + } > + > + let rust_tests = rust_tests.trim(); > + let c_test_declarations = c_test_declarations.trim(); > + let c_test_cases = c_test_cases.trim(); > + > + write!( > + BufWriter::new(File::create("rust/doctests_kernel_generated.rs").unwrap()), > + r#"//! `kernel` crate documentation tests. > + > +const __LOG_PREFIX: &[u8] = b"rust_doctests_kernel\0"; > + > +{rust_tests} > +"# > + ) > + .unwrap(); > + > + write!( > + BufWriter::new(File::create("rust/doctests_kernel_generated_kunit.c").unwrap()), We may want to eventually do all of this in Rust, rather than generating a C file, but that obviously should wait until more complete KUnit bindings exist. (And, even then, it may still make sense to piggy-back on the C macro implementations if they still make sense.) > + r#"/* > + * `kernel` crate documentation tests. > + */ > + > +#include > + > +{c_test_declarations} > + > +static struct kunit_case test_cases[] = {{ > + {c_test_cases} > + {{ }} > +}}; > + > +static struct kunit_suite test_suite = {{ > + .name = "rust_doctests_kernel", > + .test_cases = test_cases, > +}}; > + > +kunit_test_suite(test_suite); > + > +MODULE_LICENSE("GPL"); > +"# > + ) > + .unwrap(); > +} > -- > 2.41.0 > --00000000000073202105fe37f3fb Content-Type: application/pkcs7-signature; name="smime.p7s" Content-Transfer-Encoding: base64 Content-Disposition: attachment; filename="smime.p7s" Content-Description: S/MIME Cryptographic Signature MIIPnwYJKoZIhvcNAQcCoIIPkDCCD4wCAQExDzANBglghkgBZQMEAgEFADALBgkqhkiG9w0BBwGg ggz5MIIEtjCCA56gAwIBAgIQeAMYYHb81ngUVR0WyMTzqzANBgkqhkiG9w0BAQsFADBMMSAwHgYD VQQLExdHbG9iYWxTaWduIFJvb3QgQ0EgLSBSMzETMBEGA1UEChMKR2xvYmFsU2lnbjETMBEGA1UE AxMKR2xvYmFsU2lnbjAeFw0yMDA3MjgwMDAwMDBaFw0yOTAzMTgwMDAwMDBaMFQxCzAJBgNVBAYT AkJFMRkwFwYDVQQKExBHbG9iYWxTaWduIG52LXNhMSowKAYDVQQDEyFHbG9iYWxTaWduIEF0bGFz IFIzIFNNSU1FIENBIDIwMjAwggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIBAQCvLe9xPU9W dpiHLAvX7kFnaFZPuJLey7LYaMO8P/xSngB9IN73mVc7YiLov12Fekdtn5kL8PjmDBEvTYmWsuQS 6VBo3vdlqqXZ0M9eMkjcKqijrmDRleudEoPDzTumwQ18VB/3I+vbN039HIaRQ5x+NHGiPHVfk6Rx c6KAbYceyeqqfuJEcq23vhTdium/Bf5hHqYUhuJwnBQ+dAUcFndUKMJrth6lHeoifkbw2bv81zxJ I9cvIy516+oUekqiSFGfzAqByv41OrgLV4fLGCDH3yRh1tj7EtV3l2TngqtrDLUs5R+sWIItPa/4 AJXB1Q3nGNl2tNjVpcSn0uJ7aFPbAgMBAAGjggGKMIIBhjAOBgNVHQ8BAf8EBAMCAYYwHQYDVR0l BBYwFAYIKwYBBQUHAwIGCCsGAQUFBwMEMBIGA1UdEwEB/wQIMAYBAf8CAQAwHQYDVR0OBBYEFHzM CmjXouseLHIb0c1dlW+N+/JjMB8GA1UdIwQYMBaAFI/wS3+oLkUkrk1Q+mOai97i3Ru8MHsGCCsG AQUFBwEBBG8wbTAuBggrBgEFBQcwAYYiaHR0cDovL29jc3AyLmdsb2JhbHNpZ24uY29tL3Jvb3Ry MzA7BggrBgEFBQcwAoYvaHR0cDovL3NlY3VyZS5nbG9iYWxzaWduLmNvbS9jYWNlcnQvcm9vdC1y My5jcnQwNgYDVR0fBC8wLTAroCmgJ4YlaHR0cDovL2NybC5nbG9iYWxzaWduLmNvbS9yb290LXIz LmNybDBMBgNVHSAERTBDMEEGCSsGAQQBoDIBKDA0MDIGCCsGAQUFBwIBFiZodHRwczovL3d3dy5n bG9iYWxzaWduLmNvbS9yZXBvc2l0b3J5LzANBgkqhkiG9w0BAQsFAAOCAQEANyYcO+9JZYyqQt41 TMwvFWAw3vLoLOQIfIn48/yea/ekOcParTb0mbhsvVSZ6sGn+txYAZb33wIb1f4wK4xQ7+RUYBfI TuTPL7olF9hDpojC2F6Eu8nuEf1XD9qNI8zFd4kfjg4rb+AME0L81WaCL/WhP2kDCnRU4jm6TryB CHhZqtxkIvXGPGHjwJJazJBnX5NayIce4fGuUEJ7HkuCthVZ3Rws0UyHSAXesT/0tXATND4mNr1X El6adiSQy619ybVERnRi5aDe1PTwE+qNiotEEaeujz1a/+yYaaTY+k+qJcVxi7tbyQ0hi0UB3myM A/z2HmGEwO8hx7hDjKmKbDCCA18wggJHoAMCAQICCwQAAAAAASFYUwiiMA0GCSqGSIb3DQEBCwUA MEwxIDAeBgNVBAsTF0dsb2JhbFNpZ24gUm9vdCBDQSAtIFIzMRMwEQYDVQQKEwpHbG9iYWxTaWdu MRMwEQYDVQQDEwpHbG9iYWxTaWduMB4XDTA5MDMxODEwMDAwMFoXDTI5MDMxODEwMDAwMFowTDEg MB4GA1UECxMXR2xvYmFsU2lnbiBSb290IENBIC0gUjMxEzARBgNVBAoTCkdsb2JhbFNpZ24xEzAR BgNVBAMTCkdsb2JhbFNpZ24wggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIBAQDMJXaQeQZ4 Ihb1wIO2hMoonv0FdhHFrYhy/EYCQ8eyip0EXyTLLkvhYIJG4VKrDIFHcGzdZNHr9SyjD4I9DCuu l9e2FIYQebs7E4B3jAjhSdJqYi8fXvqWaN+JJ5U4nwbXPsnLJlkNc96wyOkmDoMVxu9bi9IEYMpJ pij2aTv2y8gokeWdimFXN6x0FNx04Druci8unPvQu7/1PQDhBjPogiuuU6Y6FnOM3UEOIDrAtKeh 6bJPkC4yYOlXy7kEkmho5TgmYHWyn3f/kRTvriBJ/K1AFUjRAjFhGV64l++td7dkmnq/X8ET75ti +w1s4FRpFqkD2m7pg5NxdsZphYIXAgMBAAGjQjBAMA4GA1UdDwEB/wQEAwIBBjAPBgNVHRMBAf8E BTADAQH/MB0GA1UdDgQWBBSP8Et/qC5FJK5NUPpjmove4t0bvDANBgkqhkiG9w0BAQsFAAOCAQEA S0DbwFCq/sgM7/eWVEVJu5YACUGssxOGhigHM8pr5nS5ugAtrqQK0/Xx8Q+Kv3NnSoPHRHt44K9u bG8DKY4zOUXDjuS5V2yq/BKW7FPGLeQkbLmUY/vcU2hnVj6DuM81IcPJaP7O2sJTqsyQiunwXUaM ld16WCgaLx3ezQA3QY/tRG3XUyiXfvNnBB4V14qWtNPeTCekTBtzc3b0F5nCH3oO4y0IrQocLP88 q1UOD5F+NuvDV0m+4S4tfGCLw0FREyOdzvcya5QBqJnnLDMfOjsl0oZAzjsshnjJYS8Uuu7bVW/f hO4FCU29KNhyztNiUGUe65KXgzHZs7XKR1g/XzCCBNgwggPAoAMCAQICEAEDPnEOWzT2vYIrJhGq c1swDQYJKoZIhvcNAQELBQAwVDELMAkGA1UEBhMCQkUxGTAXBgNVBAoTEEdsb2JhbFNpZ24gbnYt c2ExKjAoBgNVBAMTIUdsb2JhbFNpZ24gQXRsYXMgUjMgU01JTUUgQ0EgMjAyMDAeFw0yMzA1MTIx NjMzMjlaFw0yMzExMDgxNjMzMjlaMCQxIjAgBgkqhkiG9w0BCQEWE2RhdmlkZ293QGdvb2dsZS5j b20wggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIBAQCfIQuFV9ECjSKrnHc+/gEoEHeMu29G hkC9x5KA7Tgm7ZISSdxxP+b9Q23vqKKYcaXlXzxDUweAEa7KrhRdZMpcF1p14/qI6AG7rBn8otbO t6QSE9nwXQRL5ITEHtPRcQzLU5H9Yyq4b9MmEZAq+ByKX1t6FrXw461kqV8I/oCueKmD0p6mU/4k xzQWik4ZqST0MXkJiZenSKDDN+U1qGgHKC3HAzsIlWpNh/WsWcD4RRcEtwfW1h9DwRfGFp78OFQg 65qXbeub4G7ELSIdjGygCzVG+g1jo6we5uqPep3iRCzn92KROEVxP5lG9FlwQ2YWMt+dNiGrJdKy Kw4TK7CrAgMBAAGjggHUMIIB0DAeBgNVHREEFzAVgRNkYXZpZGdvd0Bnb29nbGUuY29tMA4GA1Ud DwEB/wQEAwIFoDAdBgNVHSUEFjAUBggrBgEFBQcDBAYIKwYBBQUHAwIwHQYDVR0OBBYEFG/UTu3x 9IGQSBx2i4m+hGXJpET+MEwGA1UdIARFMEMwQQYJKwYBBAGgMgEoMDQwMgYIKwYBBQUHAgEWJmh0 dHBzOi8vd3d3Lmdsb2JhbHNpZ24uY29tL3JlcG9zaXRvcnkvMAwGA1UdEwEB/wQCMAAwgZoGCCsG AQUFBwEBBIGNMIGKMD4GCCsGAQUFBzABhjJodHRwOi8vb2NzcC5nbG9iYWxzaWduLmNvbS9jYS9n c2F0bGFzcjNzbWltZWNhMjAyMDBIBggrBgEFBQcwAoY8aHR0cDovL3NlY3VyZS5nbG9iYWxzaWdu LmNvbS9jYWNlcnQvZ3NhdGxhc3Izc21pbWVjYTIwMjAuY3J0MB8GA1UdIwQYMBaAFHzMCmjXouse LHIb0c1dlW+N+/JjMEYGA1UdHwQ/MD0wO6A5oDeGNWh0dHA6Ly9jcmwuZ2xvYmFsc2lnbi5jb20v Y2EvZ3NhdGxhc3Izc21pbWVjYTIwMjAuY3JsMA0GCSqGSIb3DQEBCwUAA4IBAQCRI3Z4cAidgFcv Usqdz765x6KMZSfg/WtFrYg8ewsP2NpCxVM2+EhPyyEQ0k0DhtzdtGoI/Ug+jdFDyCKB9P2+EPLh iMjMnFILp7Zs4r18ECHlvZuDZfH9m0BchXIxu5jLIuQyKUWrCRDZZEDNr510ZhhVfYSFPA8ms1nk jyzYFOHYQyv5IfML/3IBFKlON5OZa+V8EZYULYcNkp03DdWglafj7SXZ1/XgAbVYrC381UvrsYN8 jndVvoa1GWwe+NVlIIK7Q3uAjV3qLEDQpaNPg1rr0oAn6YmvTccjVMqj2YNwN+RHhKNzgRGxY5ct FaN+8fXZhRhpv3bVbAWuPZXoMYICajCCAmYCAQEwaDBUMQswCQYDVQQGEwJCRTEZMBcGA1UEChMQ R2xvYmFsU2lnbiBudi1zYTEqMCgGA1UEAxMhR2xvYmFsU2lnbiBBdGxhcyBSMyBTTUlNRSBDQSAy MDIwAhABAz5xDls09r2CKyYRqnNbMA0GCWCGSAFlAwQCAQUAoIHUMC8GCSqGSIb3DQEJBDEiBCAC EDboXjw6YU/Qw77KBv7YQOBJvEjlwMUR6rnj9ZtnnzAYBgkqhkiG9w0BCQMxCwYJKoZIhvcNAQcB MBwGCSqGSIb3DQEJBTEPFw0yMzA2MTYwNDUyMjRaMGkGCSqGSIb3DQEJDzFcMFowCwYJYIZIAWUD BAEqMAsGCWCGSAFlAwQBFjALBglghkgBZQMEAQIwCgYIKoZIhvcNAwcwCwYJKoZIhvcNAQEKMAsG CSqGSIb3DQEBBzALBglghkgBZQMEAgEwDQYJKoZIhvcNAQEBBQAEggEAieQTFKXaibtDNihLsd15 O/Uf1nmfHugk9Jr1JKg2RsX7wHKISqDyMrFr6Vca20I0vrPKigiwGpKeUOODXGm4WnaratlxQpCD NwOIvO9MW5Mp2QeWAAoh7ssx3i0Eh1a1TO/SB/9hyCDt+YX8Oiv7g/RicTnl/ezstvwTcJN4+MxW MwxNRoiagubV7yd1sNUyM6cZsBWBJks/zPIDwHivZFWoFhWqskHAfEr2YXElWOYkwwBnrzQQg+TZ jkU9aWHrxQLc64FMznIWd5gpfkQIhh1cH0ecohJbuKmM1LGtZgxWTj5+SaarELMMMBj02oBKCb/D Rk4dyUizVyRyWrY0Gg== --00000000000073202105fe37f3fb--