Skip to content
Snippets Groups Projects
  1. Sep 25, 2023
    • Alice Ryhl's avatar
      rust: workqueue: add helper for defining work_struct fields · 7324b889
      Alice Ryhl authored
      
      The main challenge with defining `work_struct` fields is making sure
      that the function pointer stored in the `work_struct` is appropriate for
      the work item type it is embedded in. It needs to know the offset of the
      `work_struct` field being used (even if there are several!) so that it
      can do a `container_of`, and it needs to know the type of the work item
      so that it can call into the right user-provided code. All of this needs
      to happen in a way that provides a safe API to the user, so that users
      of the workqueue cannot mix up the function pointers.
      
      There are three important pieces that are relevant when doing this:
      
       * The pointer type.
       * The work item struct. This is what the pointer points at.
       * The `work_struct` field. This is a field of the work item struct.
      
      This patch introduces a separate trait for each piece. The pointer type
      is given a `WorkItemPointer` trait, which pointer types need to
      implement to be usable with the workqueue. This trait will be
      implemented for `Arc` and `Box` in a later patch in this patchset.
      Implementing this trait is unsafe because this is where the
      `container_of` operation happens, but user-code will not need to
      implement it themselves.
      
      The work item struct should then implement the `WorkItem` trait. This
      trait is where user-code specifies what they want to happen when a work
      item is executed. It also specifies what the correct pointer type is.
      
      Finally, to make the work item struct know the offset of its
      `work_struct` field, we use a trait called `HasWork<T, ID>`. If a type
      implements this trait, then the type declares that, at the given offset,
      there is a field of type `Work<T, ID>`. The trait is marked unsafe
      because the OFFSET constant must be correct, but we provide an
      `impl_has_work!` macro that can safely implement `HasWork<T>` on a type.
      The macro expands to something that only compiles if the specified field
      really has the type `Work<T>`. It is used like this:
      
      ```
      struct MyWorkItem {
          work_field: Work<MyWorkItem, 1>,
      }
      
      impl_has_work! {
          impl HasWork<MyWorkItem, 1> for MyWorkItem { self.work_field }
      }
      ```
      
      Note that since the `Work` type is annotated with an id, you can have
      several `work_struct` fields by using a different id for each one.
      
      Co-developed-by: default avatarGary Guo <gary@garyguo.net>
      Signed-off-by: default avatarGary Guo <gary@garyguo.net>
      Signed-off-by: default avatarAlice Ryhl <aliceryhl@google.com>
      Reviewed-by: default avatarBenno Lossin <benno.lossin@proton.me>
      Reviewed-by: Martin Rodriguez Reboredo's avatarMartin Rodriguez Reboredo <yakoyoku@gmail.com>
      Reviewed-by: default avatarAndreas Hindborg <a.hindborg@samsung.com>
      Reviewed-by: default avatarBoqun Feng <boqun.feng@gmail.com>
      Signed-off-by: default avatarTejun Heo <tj@kernel.org>
      7324b889
  2. Aug 14, 2023
  3. Aug 07, 2023
  4. Jul 19, 2023
    • Miguel Ojeda's avatar
      rust: support running Rust documentation tests as KUnit ones · a66d733d
      Miguel Ojeda authored
      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
              # 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
      
      Therefore, add support for running Rust documentation tests
      in KUnit. Some other notes about the current implementation
      and support follow.
      
      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).
      
      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, containing 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
      
      This line follows the syntax for declaring test metadata in the
      proposed KTAP v2 spec [1], which may be used for the proposed
      KUnit test attributes API [2]. Thus hopefully this will make
      migration easier later on (suggested by David [3]).
      
      The original line in that test attribute is figured out by
      providing an anchor (suggested by Boqun [4]). The original file
      is found by walking the filesystem, checking directory prefixes
      to reduce the amount of combinations to check, and it is only
      done once per file. Ambiguities are detected and reported.
      
      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.
      
      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.
      
      Link: https://lore.kernel.org/lkml/20230420205734.1288498-1-rmoar@google.com/ [1]
      Link: https://lore.kernel.org/linux-kselftest/20230707210947.1208717-1-rmoar@google.com/ [2]
      Link: https://lore.kernel.org/rust-for-linux/CABVgOSkOLO-8v6kdAGpmYnZUb+LKOX0CtYCo-Bge7r_2YTuXDQ@mail.gmail.com/ [3]
      Link: https://lore.kernel.org/rust-for-linux/ZIps86MbJF%2FiGIzd@boqun-archlinux/
      
       [4]
      Signed-off-by: default avatarMiguel Ojeda <ojeda@kernel.org>
      Reviewed-by: default avatarDavid Gow <davidgow@google.com>
      Signed-off-by: default avatarShuah Khan <skhan@linuxfoundation.org>
      a66d733d
  5. Jun 12, 2023
  6. Apr 21, 2023
  7. Apr 12, 2023
  8. Jan 16, 2023
  9. Sep 28, 2022
Loading