| .. SPDX-License-Identifier: GPL-2.0 | 
 |  | 
 | Testing | 
 | ======= | 
 |  | 
 | This document contains useful information how to test the Rust code in the | 
 | kernel. | 
 |  | 
 | There are three sorts of tests: | 
 |  | 
 | - The KUnit tests. | 
 | - The ``#[test]`` tests. | 
 | - The Kselftests. | 
 |  | 
 | The KUnit tests | 
 | --------------- | 
 |  | 
 | These are the tests that come from the examples in the Rust documentation. They | 
 | get transformed into KUnit tests. | 
 |  | 
 | Usage | 
 | ***** | 
 |  | 
 | These tests can be run via KUnit. For example via ``kunit_tool`` (``kunit.py``) | 
 | on the command line:: | 
 |  | 
 | 	./tools/testing/kunit/kunit.py run --make_options LLVM=1 --arch x86_64 --kconfig_add CONFIG_RUST=y | 
 |  | 
 | Alternatively, KUnit can run them as kernel built-in at boot. Refer to | 
 | Documentation/dev-tools/kunit/index.rst for the general KUnit documentation | 
 | and Documentation/dev-tools/kunit/architecture.rst for the details of kernel | 
 | built-in vs. command line testing. | 
 |  | 
 | To use these KUnit doctests, the following must be enabled:: | 
 |  | 
 | 	CONFIG_KUNIT | 
 | 	   Kernel hacking -> Kernel Testing and Coverage -> KUnit - Enable support for unit tests | 
 | 	CONFIG_RUST_KERNEL_DOCTESTS | 
 | 	   Kernel hacking -> Rust hacking -> Doctests for the `kernel` crate | 
 |  | 
 | in the kernel config system. | 
 |  | 
 | KUnit tests are documentation tests | 
 | *********************************** | 
 |  | 
 | These documentation tests 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: | 
 |  | 
 | .. code-block:: rust | 
 |  | 
 | 	/// 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 verifying that examples compile | 
 | (thus enforcing they are kept in sync with the code they document) and as well | 
 | as running those that do not depend on in-kernel APIs. | 
 |  | 
 | For the kernel, however, these tests get transformed into KUnit test suites. | 
 | This means that doctests get compiled as Rust kernel objects, allowing them to | 
 | run against a built kernel. | 
 |  | 
 | A benefit of this KUnit integration is that Rust doctests get to reuse 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 | 
 | 	    # rust_doctest_kernel_build_assert_rs_0.location: rust/kernel/build_assert.rs:13 | 
 | 	    ok 1 rust_doctest_kernel_build_assert_rs_0 | 
 | 	    # rust_doctest_kernel_build_assert_rs_1.location: rust/kernel/build_assert.rs:56 | 
 | 	    ok 2 rust_doctest_kernel_build_assert_rs_1 | 
 | 	    # rust_doctest_kernel_init_rs_0.location: rust/kernel/init.rs:122 | 
 | 	    ok 3 rust_doctest_kernel_init_rs_0 | 
 | 	    ... | 
 | 	    # rust_doctest_kernel_types_rs_2.location: rust/kernel/types.rs: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 | 
 |  | 
 | Tests using the `? <https://doc.rust-lang.org/reference/expressions/operator-expr.html#the-question-mark-operator>`_ | 
 | operator are also supported as usual, e.g.: | 
 |  | 
 | .. code-block:: rust | 
 |  | 
 | 	/// ``` | 
 | 	/// # use kernel::{spawn_work_item, workqueue}; | 
 | 	/// spawn_work_item!(workqueue::system(), || pr_info!("x\n"))?; | 
 | 	/// # Ok::<(), Error>(()) | 
 | 	/// ``` | 
 |  | 
 | The tests are also compiled with Clippy under ``CLIPPY=1``, just like normal | 
 | code, thus also benefitting from extra linting. | 
 |  | 
 | In order for developers to easily see which line of doctest code caused a | 
 | failure, a KTAP diagnostic line is printed to the log. This contains the | 
 | location (file and line) of the original test (i.e. instead of the location in | 
 | the generated Rust file):: | 
 |  | 
 | 	# rust_doctest_kernel_types_rs_2.location: rust/kernel/types.rs:150 | 
 |  | 
 | 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 those for KUnit testing (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. | 
 |  | 
 | 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. Additionally, doctests are not run for nonpublic functions. | 
 |  | 
 | Since these tests are examples, i.e. they are part of the documentation, they | 
 | should generally be written like "real code". Thus, for example, instead of | 
 | using ``unwrap()`` or ``expect()``, use the ``?`` operator. For more background, | 
 | please see: | 
 |  | 
 | 	https://rust.docs.kernel.org/kernel/error/type.Result.html#error-codes-in-c-and-rust | 
 |  | 
 | The ``#[test]`` tests | 
 | --------------------- | 
 |  | 
 | Additionally, there are the ``#[test]`` tests. Like for documentation tests, | 
 | these are also fairly similar to what you would expect from userspace, and they | 
 | are also mapped to KUnit. | 
 |  | 
 | These tests are introduced by the ``kunit_tests`` procedural macro, which takes | 
 | the name of the test suite as an argument. | 
 |  | 
 | For instance, assume we want to test the function ``f`` from the documentation | 
 | tests section. We could write, in the same file where we have our function: | 
 |  | 
 | .. code-block:: rust | 
 |  | 
 | 	#[kunit_tests(rust_kernel_mymod)] | 
 | 	mod tests { | 
 | 	    use super::*; | 
 |  | 
 | 	    #[test] | 
 | 	    fn test_f() { | 
 | 	        assert_eq!(f(10, 20), 30); | 
 | 	    } | 
 | 	} | 
 |  | 
 | And if we run it, the kernel log would look like:: | 
 |  | 
 | 	    KTAP version 1 | 
 | 	    # Subtest: rust_kernel_mymod | 
 | 	    # speed: normal | 
 | 	    1..1 | 
 | 	    # test_f.speed: normal | 
 | 	    ok 1 test_f | 
 | 	ok 1 rust_kernel_mymod | 
 |  | 
 | Like documentation tests, the ``assert!`` and ``assert_eq!`` macros are mapped | 
 | back to KUnit and do not panic. Similarly, the | 
 | `? <https://doc.rust-lang.org/reference/expressions/operator-expr.html#the-question-mark-operator>`_ | 
 | operator is supported, i.e. the test functions may return either nothing (i.e. | 
 | the unit type ``()``) or ``Result`` (i.e. any ``Result<T, E>``). For instance: | 
 |  | 
 | .. code-block:: rust | 
 |  | 
 | 	#[kunit_tests(rust_kernel_mymod)] | 
 | 	mod tests { | 
 | 	    use super::*; | 
 |  | 
 | 	    #[test] | 
 | 	    fn test_g() -> Result { | 
 | 	        let x = g()?; | 
 | 	        assert_eq!(x, 30); | 
 | 	        Ok(()) | 
 | 	    } | 
 | 	} | 
 |  | 
 | If we run the test and the call to ``g`` fails, then the kernel log would show:: | 
 |  | 
 | 	    KTAP version 1 | 
 | 	    # Subtest: rust_kernel_mymod | 
 | 	    # speed: normal | 
 | 	    1..1 | 
 | 	    # test_g: ASSERTION FAILED at rust/kernel/lib.rs:335 | 
 | 	    Expected is_test_result_ok(test_g()) to be true, but is false | 
 | 	    # test_g.speed: normal | 
 | 	    not ok 1 test_g | 
 | 	not ok 1 rust_kernel_mymod | 
 |  | 
 | If a ``#[test]`` test could be useful as an example for the user, then please | 
 | use a documentation test instead. Even edge cases of an API, e.g. error or | 
 | boundary cases, can be interesting to show in examples. | 
 |  | 
 | The ``rusttest`` host tests | 
 | --------------------------- | 
 |  | 
 | These are userspace tests that can be built and run in the host (i.e. the one | 
 | that performs the kernel build) using the ``rusttest`` Make target:: | 
 |  | 
 | 	make LLVM=1 rusttest | 
 |  | 
 | This requires the kernel ``.config``. | 
 |  | 
 | Currently, they are mostly used for testing the ``macros`` crate's examples. | 
 |  | 
 | The Kselftests | 
 | -------------- | 
 |  | 
 | Kselftests are also available in the ``tools/testing/selftests/rust`` folder. | 
 |  | 
 | The kernel config options required for the tests are listed in the | 
 | ``tools/testing/selftests/rust/config`` file and can be included with the aid | 
 | of the ``merge_config.sh`` script:: | 
 |  | 
 | 	./scripts/kconfig/merge_config.sh .config tools/testing/selftests/rust/config | 
 |  | 
 | The kselftests are built within the kernel source tree and are intended to | 
 | be executed on a system that is running the same kernel. | 
 |  | 
 | Once a kernel matching the source tree has been installed and booted, the | 
 | tests can be compiled and executed using the following command:: | 
 |  | 
 | 	make TARGETS="rust" kselftest | 
 |  | 
 | Refer to Documentation/dev-tools/kselftest.rst for the general Kselftest | 
 | documentation. |