From: Andreas Hindborg <a.hindborg@kernel.org>
To: "Boqun Feng" <boqun.feng@gmail.com>,
"Jens Axboe" <axboe@kernel.dk>, "Miguel Ojeda" <ojeda@kernel.org>,
"Gary Guo" <gary@garyguo.net>,
"Björn Roy Baron" <bjorn3_gh@protonmail.com>,
"Benno Lossin" <lossin@kernel.org>,
"Alice Ryhl" <aliceryhl@google.com>,
"Trevor Gross" <tmgross@umich.edu>,
"Danilo Krummrich" <dakr@kernel.org>,
"FUJITA Tomonori" <fujita.tomonori@gmail.com>,
"Frederic Weisbecker" <frederic@kernel.org>,
"Lyude Paul" <lyude@redhat.com>,
"Thomas Gleixner" <tglx@kernel.org>,
"Anna-Maria Behnsen" <anna-maria@linutronix.de>,
"John Stultz" <jstultz@google.com>,
"Stephen Boyd" <sboyd@kernel.org>,
"Lorenzo Stoakes" <lorenzo.stoakes@oracle.com>,
"Liam R. Howlett" <Liam.Howlett@oracle.com>
Cc: linux-block@vger.kernel.org, rust-for-linux@vger.kernel.org,
linux-kernel@vger.kernel.org, linux-mm@kvack.org,
Andreas Hindborg <a.hindborg@kernel.org>
Subject: [PATCH 05/79] block: rust: change `queue_rq` request type to `Owned`
Date: Mon, 16 Feb 2026 00:34:52 +0100 [thread overview]
Message-ID: <20260216-rnull-v6-19-rc5-send-v1-5-de9a7af4b469@kernel.org> (raw)
In-Reply-To: <20260216-rnull-v6-19-rc5-send-v1-0-de9a7af4b469@kernel.org>
Simplify the reference counting scheme for `Request` from 4 states to 3
states. This is achieved by coalescing the zero state between block layer
owned and uniquely owned by driver.
Implement `Ownable` for `Request` and deliver `Request` to drivers as
`Owned<Request>`. In this process:
- Move uniqueness assertions out of `rnull` as these are now guaranteed by
the `Owned` type.
- Move `start_unchecked`, `try_set_end` and `end_ok` from `Request` to
`Owned<Request>`, relying on type invariant for uniqueness.
Signed-off-by: Andreas Hindborg <a.hindborg@kernel.org>
---
drivers/block/rnull/rnull.rs | 26 ++---
rust/kernel/block/mq.rs | 10 +-
rust/kernel/block/mq/operations.rs | 27 +++--
rust/kernel/block/mq/request.rs | 218 +++++++++++++++++++++----------------
4 files changed, 158 insertions(+), 123 deletions(-)
diff --git a/drivers/block/rnull/rnull.rs b/drivers/block/rnull/rnull.rs
index bd883a5061cab..6a7f660d31998 100644
--- a/drivers/block/rnull/rnull.rs
+++ b/drivers/block/rnull/rnull.rs
@@ -19,7 +19,8 @@
},
},
error::Result,
- new_mutex, pr_info,
+ new_mutex,
+ pr_info,
prelude::*,
str::CString,
sync::{
@@ -27,6 +28,10 @@
Arc,
Mutex, //
},
+ types::{
+ OwnableRefCounted,
+ Owned, //
+ }, //
};
use pin_init::PinInit;
@@ -131,15 +136,10 @@ impl Operations for NullBlkDevice {
type QueueData = KBox<QueueData>;
#[inline(always)]
- fn queue_rq(queue_data: &QueueData, rq: ARef<mq::Request<Self>>, _is_last: bool) -> Result {
+ fn queue_rq(queue_data: &QueueData, rq: Owned<mq::Request<Self>>, _is_last: bool) -> Result {
match queue_data.irq_mode {
- IRQMode::None => mq::Request::end_ok(rq)
- .map_err(|_e| kernel::error::code::EIO)
- // We take no refcounts on the request, so we expect to be able to
- // end the request. The request reference must be unique at this
- // point, and so `end_ok` cannot fail.
- .expect("Fatal error - expected to be able to end request"),
- IRQMode::Soft => mq::Request::complete(rq),
+ IRQMode::None => rq.end_ok(),
+ IRQMode::Soft => mq::Request::complete(rq.into()),
}
Ok(())
}
@@ -147,11 +147,9 @@ fn queue_rq(queue_data: &QueueData, rq: ARef<mq::Request<Self>>, _is_last: bool)
fn commit_rqs(_queue_data: &QueueData) {}
fn complete(rq: ARef<mq::Request<Self>>) {
- mq::Request::end_ok(rq)
+ OwnableRefCounted::try_from_shared(rq)
.map_err(|_e| kernel::error::code::EIO)
- // We take no refcounts on the request, so we expect to be able to
- // end the request. The request reference must be unique at this
- // point, and so `end_ok` cannot fail.
- .expect("Fatal error - expected to be able to end request");
+ .expect("Failed to complete request")
+ .end_ok();
}
}
diff --git a/rust/kernel/block/mq.rs b/rust/kernel/block/mq.rs
index 1fd0d54dd5493..b8ecd69abe980 100644
--- a/rust/kernel/block/mq.rs
+++ b/rust/kernel/block/mq.rs
@@ -62,6 +62,7 @@
//! new_mutex,
//! prelude::*,
//! sync::{aref::ARef, Arc, Mutex},
+//! types::{ForeignOwnable, OwnableRefCounted, Owned},
//! };
//!
//! struct MyBlkDevice;
@@ -70,17 +71,18 @@
//! impl Operations for MyBlkDevice {
//! type QueueData = ();
//!
-//! fn queue_rq(_queue_data: (), rq: ARef<Request<Self>>, _is_last: bool) -> Result {
-//! Request::end_ok(rq);
+//! fn queue_rq(_queue_data: (), rq: Owned<Request<Self>>, _is_last: bool) -> Result {
+//! rq.end_ok();
//! Ok(())
//! }
//!
//! fn commit_rqs(_queue_data: ()) {}
//!
//! fn complete(rq: ARef<Request<Self>>) {
-//! Request::end_ok(rq)
+//! OwnableRefCounted::try_from_shared(rq)
//! .map_err(|_e| kernel::error::code::EIO)
-//! .expect("Fatal error - expected to be able to end request");
+//! .expect("Fatal error - expected to be able to end request")
+//! .end_ok();
//! }
//! }
//!
diff --git a/rust/kernel/block/mq/operations.rs b/rust/kernel/block/mq/operations.rs
index b68c0208efc66..3dea79d647ff7 100644
--- a/rust/kernel/block/mq/operations.rs
+++ b/rust/kernel/block/mq/operations.rs
@@ -9,10 +9,10 @@
block::mq::{request::RequestDataWrapper, Request},
error::{from_result, Result},
prelude::*,
- sync::{aref::ARef, Refcount},
- types::ForeignOwnable,
+ sync::{aref::ARef, atomic::ordering, Refcount},
+ types::{ForeignOwnable, Owned},
};
-use core::marker::PhantomData;
+use core::{marker::PhantomData, ptr::NonNull};
type ForeignBorrowed<'a, T> = <T as ForeignOwnable>::Borrowed<'a>;
@@ -36,7 +36,7 @@ pub trait Operations: Sized {
/// `false`, the driver is allowed to defer committing the request.
fn queue_rq(
queue_data: ForeignBorrowed<'_, Self::QueueData>,
- rq: ARef<Request<Self>>,
+ rq: Owned<Request<Self>>,
is_last: bool,
) -> Result;
@@ -90,16 +90,23 @@ impl<T: Operations> OperationsVTable<T> {
// this function.
let request = unsafe { &*(*bd).rq.cast::<Request<T>>() };
- // One refcount for the ARef, one for being in flight
- request.wrapper_ref().refcount().set(2);
+ debug_assert!(
+ request
+ .wrapper_ref()
+ .refcount()
+ .as_atomic()
+ .load(ordering::Acquire)
+ == 0
+ );
// SAFETY:
- // - We own a refcount that we took above. We pass that to `ARef`.
+ // - By API contract, we own the request.
// - By the safety requirements of this function, `request` is a valid
// `struct request` and the private data is properly initialized.
// - `rq` will be alive until `blk_mq_end_request` is called and is
- // reference counted by `ARef` until then.
- let rq = unsafe { Request::aref_from_raw((*bd).rq) };
+ // reference counted by until then.
+ let mut rq =
+ unsafe { Owned::from_raw(NonNull::<Request<T>>::new_unchecked((*bd).rq.cast())) };
// SAFETY: `hctx` is valid as required by this function.
let queue_data = unsafe { (*(*hctx).queue).queuedata };
@@ -111,7 +118,7 @@ impl<T: Operations> OperationsVTable<T> {
let queue_data = unsafe { T::QueueData::borrow(queue_data) };
// SAFETY: We have exclusive access and we just set the refcount above.
- unsafe { Request::start_unchecked(&rq) };
+ unsafe { rq.start_unchecked() };
let ret = T::queue_rq(
queue_data,
diff --git a/rust/kernel/block/mq/request.rs b/rust/kernel/block/mq/request.rs
index cf013b9e2cacf..148348b4ef245 100644
--- a/rust/kernel/block/mq/request.rs
+++ b/rust/kernel/block/mq/request.rs
@@ -7,13 +7,12 @@
use crate::{
bindings,
block::mq::Operations,
- error::Result,
sync::{
- aref::{ARef, AlwaysRefCounted, RefCounted},
- atomic::Relaxed,
+ aref::{ARef, RefCounted},
+ atomic::ordering,
Refcount,
},
- types::Opaque,
+ types::{Opaque, Ownable, OwnableRefCounted, Owned},
};
use core::{marker::PhantomData, ptr::NonNull};
@@ -21,25 +20,21 @@
///
/// # Implementation details
///
-/// There are four states for a request that the Rust bindings care about:
+/// There are three states for a request that the Rust bindings care about:
///
-/// 1. Request is owned by block layer (refcount 0).
-/// 2. Request is owned by driver but with zero [`ARef`]s in existence
-/// (refcount 1).
-/// 3. Request is owned by driver with exactly one [`ARef`] in existence
-/// (refcount 2).
-/// 4. Request is owned by driver with more than one [`ARef`] in existence
-/// (refcount > 2).
+/// - 0: The request is owned by C block layer or is uniquely referenced (by [`Owned<_>`]).
+/// - 1: The request is owned by Rust abstractions but is not referenced.
+/// - 2+: There is one or more [`ARef`] instances referencing the request.
///
+/// We need to track 1 and 2 to make sure that `tag_to_rq` does not issue any
+/// [`ARef`] to requests not owned by the driver, or to requests that have a
+/// [`Owned`] referencing it.
///
-/// We need to track 1 and 2 to ensure we fail tag to request conversions for
-/// requests that are not owned by the driver.
-///
-/// We need to track 3 and 4 to ensure that it is safe to end the request and hand
-/// back ownership to the block layer.
+/// We need to track 3 to know when it is safe to convert an [`ARef`] to a
+/// [`Owned`].
///
/// Note that the driver can still obtain new `ARef` even if there is no `ARef`s in existence by
-/// using `tag_to_rq`, hence the need to distinguish B and C.
+/// using `tag_to_rq`, hence the need to distinct 1 and 2.
///
/// The states are tracked through the private `refcount` field of
/// `RequestDataWrapper`. This structure lives in the private data area of the C
@@ -66,6 +61,7 @@ impl<T: Operations> Request<T> {
///
/// * The caller must own a refcount on `ptr` that is transferred to the
/// returned [`ARef`].
+ /// * The refcount must be >= 2.
/// * The type invariants for [`Request`] must hold for the pointee of `ptr`.
///
/// [`struct request`]: srctree/include/linux/blk-mq.h
@@ -76,72 +72,6 @@ pub(crate) unsafe fn aref_from_raw(ptr: *mut bindings::request) -> ARef<Self> {
unsafe { ARef::from_raw(NonNull::new_unchecked(ptr.cast())) }
}
- /// Notify the block layer that a request is going to be processed now.
- ///
- /// The block layer uses this hook to do proper initializations such as
- /// starting the timeout timer. It is a requirement that block device
- /// drivers call this function when starting to process a request.
- ///
- /// # Safety
- ///
- /// The caller must have exclusive ownership of `self`, that is
- /// `self.wrapper_ref().refcount() == 2`.
- pub(crate) unsafe fn start_unchecked(this: &ARef<Self>) {
- // SAFETY: By type invariant, `self.0` is a valid `struct request` and
- // we have exclusive access.
- unsafe { bindings::blk_mq_start_request(this.0.get()) };
- }
-
- /// Try to take exclusive ownership of `this` by dropping the refcount to 0.
- /// This fails if `this` is not the only [`ARef`] pointing to the underlying
- /// [`Request`].
- ///
- /// If the operation is successful, [`Ok`] is returned with a pointer to the
- /// C [`struct request`]. If the operation fails, `this` is returned in the
- /// [`Err`] variant.
- ///
- /// [`struct request`]: srctree/include/linux/blk-mq.h
- fn try_set_end(this: ARef<Self>) -> Result<*mut bindings::request, ARef<Self>> {
- // To hand back the ownership, we need the current refcount to be 2.
- // Since we can race with `TagSet::tag_to_rq`, this needs to atomically reduce
- // refcount to 0. `Refcount` does not provide a way to do this, so use the underlying
- // atomics directly.
- if let Err(_old) = this
- .wrapper_ref()
- .refcount()
- .as_atomic()
- .cmpxchg(2, 0, Relaxed)
- {
- return Err(this);
- }
-
- let request_ptr = this.0.get();
- core::mem::forget(this);
-
- Ok(request_ptr)
- }
-
- /// Notify the block layer that the request has been completed without errors.
- ///
- /// This function will return [`Err`] if `this` is not the only [`ARef`]
- /// referencing the request.
- pub fn end_ok(this: ARef<Self>) -> Result<(), ARef<Self>> {
- let request_ptr = Self::try_set_end(this)?;
-
- // SAFETY: By type invariant, `this.0` was a valid `struct request`. The
- // success of the call to `try_set_end` guarantees that there are no
- // `ARef`s pointing to this request. Therefore it is safe to hand it
- // back to the block layer.
- unsafe {
- bindings::blk_mq_end_request(
- request_ptr,
- bindings::BLK_STS_OK as bindings::blk_status_t,
- )
- };
-
- Ok(())
- }
-
/// Complete the request by scheduling `Operations::complete` for
/// execution.
///
@@ -234,27 +164,125 @@ unsafe impl<T: Operations> Sync for Request<T> {}
// matching reference count decrement is executed.
unsafe impl<T: Operations> RefCounted for Request<T> {
fn inc_ref(&self) {
- self.wrapper_ref().refcount().inc();
+ let refcount = &self.wrapper_ref().refcount().as_atomic();
+
+ // Load acquire, store relaxed. We sync with store release of
+ // `OwnableRefCounted::into_shared`. After that all unique references are dead and we have
+ // shared access. We can use relaxed ordering for the store.
+ #[cfg_attr(not(debug_assertions), allow(unused_variables))]
+ let old = refcount.fetch_add(1, ordering::Acquire);
+
+ debug_assert!(old >= 1, "Request refcount zero clone");
}
unsafe fn dec_ref(obj: core::ptr::NonNull<Self>) {
- // SAFETY: The type invariants of `ARef` guarantee that `obj` is valid
+ // SAFETY: The type invariants of `RefCounted` guarantee that `obj` is valid
// for read.
let wrapper_ptr = unsafe { Self::wrapper_ptr(obj.as_ptr()).as_ptr() };
// SAFETY: The type invariant of `Request` guarantees that the private
// data area is initialized and valid.
let refcount = unsafe { &*RequestDataWrapper::refcount_ptr(wrapper_ptr) };
- #[cfg_attr(not(CONFIG_DEBUG_MISC), allow(unused_variables))]
- let is_zero = refcount.dec_and_test();
+ // Store release to sync with load acquire in
+ // `OwnableRefCounted::try_from_shared`.
+ #[cfg_attr(not(debug_assertions), allow(unused_variables))]
+ let old = refcount.as_atomic().fetch_sub(1, ordering::Release);
- #[cfg(CONFIG_DEBUG_MISC)]
- if is_zero {
- panic!("Request reached refcount zero in Rust abstractions");
- }
+ debug_assert!(
+ old > 1,
+ "Request reached refcount zero in Rust abstractions"
+ );
+ }
+}
+
+impl<T: Operations> Owned<Request<T>> {
+ /// Notify the block layer that a request is going to be processed now.
+ ///
+ /// The block layer uses this hook to do proper initializations such as
+ /// starting the timeout timer. It is a requirement that block device
+ /// drivers call this function when starting to process a request.
+ ///
+ /// # Safety
+ ///
+ /// The caller must have exclusive ownership of `self`, that is
+ /// `self.wrapper_ref().refcount() == 0`.
+ ///
+ /// This can only be called once in the request life cycle.
+ pub(crate) unsafe fn start_unchecked(&mut self) {
+ // SAFETY: By type invariant, `self.0` is a valid `struct request` and
+ // we have exclusive access.
+ unsafe { bindings::blk_mq_start_request(self.0.get()) };
+ }
+
+ /// Notify the block layer that the request has been completed without errors.
+ pub fn end_ok(self) {
+ let request_ptr = self.0.get().cast();
+ core::mem::forget(self);
+ // SAFETY: By type invariant, `this.0` was a valid `struct request`. The
+ // existence of `self` guarantees that there are no `ARef`s pointing to
+ // this request. Therefore it is safe to hand it back to the block
+ // layer.
+ unsafe { bindings::blk_mq_end_request(request_ptr, bindings::BLK_STS_OK) };
+ }
+}
+
+// SAFETY: The `release` implementation frees the underlying request according to the reference
+// counting scheme for `Request`.
+unsafe impl<T: Operations> Ownable for Request<T> {
+ unsafe fn release(this: NonNull<Self>) {
+ // SAFETY: The safety requirements of this function guarantee that `this`
+ // is valid for read.
+ let wrapper_ptr = unsafe { Self::wrapper_ptr(this.as_ptr()).as_ptr() };
+ // SAFETY: The type invariant of `Request` guarantees that the private
+ // data area is initialized and valid.
+ let refcount = unsafe { &*RequestDataWrapper::refcount_ptr(wrapper_ptr) };
+
+ // Store release to sync with load acquire when converting back to owned.
+ #[cfg_attr(not(debug_assertions), allow(unused_variables))]
+ let old = refcount.as_atomic().fetch_add(1, ordering::Release);
+
+ debug_assert!(
+ old == 0,
+ "Invalid refcount when releasing `Owned<Request<T>>`"
+ );
}
}
-// SAFETY: We currently do not implement `Ownable`, thus it is okay to obtain an `ARef<Request>`
-// from a `&Request` (but this will change in the future).
-unsafe impl<T: Operations> AlwaysRefCounted for Request<T> {}
+impl<T: Operations> OwnableRefCounted for Request<T> {
+ fn try_from_shared(this: ARef<Self>) -> core::result::Result<Owned<Self>, ARef<Self>> {
+ // Load acquire to sync with decrement store release to make sure all
+ // shared access has ended.
+ let updated = this
+ .wrapper_ref()
+ .refcount()
+ .as_atomic()
+ .cmpxchg(2, 0, ordering::Acquire);
+
+ match updated {
+ Ok(_) => Ok(
+ // SAFETY: We achieved unique ownership above.
+ unsafe { Owned::from_raw(ARef::into_raw(this)) },
+ ),
+ Err(_) => Err(this),
+ }
+ }
+
+ fn into_shared(this: Owned<Self>) -> ARef<Self> {
+ // Store release to sync with future increments using load acquire to
+ // make sure exclusive access has ended before shared access start.
+ #[cfg_attr(not(debug_assertions), allow(unused_variables))]
+ let old = this
+ .wrapper_ref()
+ .refcount()
+ .as_atomic()
+ .fetch_add(2, ordering::Release);
+
+ debug_assert!(
+ old == 0,
+ "Invalid refcount when upgrading `Owned<Request<T>>`"
+ );
+
+ // SAFETY: We incremented the refcount above.
+ unsafe { ARef::from_raw(Owned::into_raw(this)) }
+ }
+}
--
2.51.2
next prev parent reply other threads:[~2026-02-16 4:17 UTC|newest]
Thread overview: 82+ messages / expand[flat|nested] mbox.gz Atom feed top
2026-02-15 23:34 [PATCH 00/79] block: rnull: complete the rust null block driver Andreas Hindborg
2026-02-15 23:34 ` [PATCH 01/79] block: rnull: adopt new formatting guidelines Andreas Hindborg
2026-02-15 23:34 ` [PATCH 02/79] block: rnull: add module parameters Andreas Hindborg
2026-02-15 23:34 ` [PATCH 03/79] block: rnull: add macros to define configfs attributes Andreas Hindborg
2026-02-15 23:34 ` [PATCH 04/79] block: rust: fix generation of bindings to `BLK_STS_.*` Andreas Hindborg
2026-02-15 23:34 ` Andreas Hindborg [this message]
2026-02-15 23:34 ` [PATCH 06/79] block: rust: add `Request` private data support Andreas Hindborg
2026-02-15 23:34 ` [PATCH 07/79] block: rust: allow `hrtimer::Timer` in `RequestData` Andreas Hindborg
2026-02-15 23:34 ` [PATCH 08/79] block: rnull: add timer completion mode Andreas Hindborg
2026-02-15 23:34 ` [PATCH 09/79] block: rust: introduce `kernel::block::bio` module Andreas Hindborg
2026-02-15 23:34 ` [PATCH 10/79] block: rust: add `command` getter to `Request` Andreas Hindborg
2026-02-15 23:34 ` [PATCH 11/79] block: rust: mq: use GFP_KERNEL from prelude Andreas Hindborg
2026-02-15 23:34 ` [PATCH 12/79] block: rust: add `TagSet` flags Andreas Hindborg
2026-02-15 23:35 ` [PATCH 13/79] block: rnull: add memory backing Andreas Hindborg
2026-02-15 23:35 ` [PATCH 14/79] block: rnull: add submit queue count config option Andreas Hindborg
2026-02-15 23:35 ` [PATCH 15/79] block: rnull: add `use_per_node_hctx` " Andreas Hindborg
2026-02-15 23:35 ` [PATCH 16/79] block: rust: allow specifying home node when constructing `TagSet` Andreas Hindborg
2026-02-15 23:35 ` [PATCH 17/79] block: rnull: allow specifying the home numa node Andreas Hindborg
2026-02-15 23:35 ` [PATCH 18/79] block: rust: add Request::sectors() method Andreas Hindborg
2026-02-15 23:35 ` [PATCH 19/79] block: rust: mq: add max_hw_discard_sectors support to GenDiskBuilder Andreas Hindborg
2026-02-15 23:35 ` [PATCH 20/79] block: rnull: add discard support Andreas Hindborg
2026-02-15 23:35 ` [PATCH 21/79] block: rust: add `NoDefaultScheduler` flag for `TagSet` Andreas Hindborg
2026-02-15 23:35 ` [PATCH 22/79] block: rnull: add no_sched module parameter and configfs attribute Andreas Hindborg
2026-02-15 23:35 ` [PATCH 23/79] block: rust: change sector type from usize to u64 Andreas Hindborg
2026-02-15 23:35 ` [PATCH 24/79] block: rust: add `BadBlocks` for bad block tracking Andreas Hindborg
2026-02-15 23:35 ` [PATCH 25/79] block: rust: mq: add Request::end() method for custom status codes Andreas Hindborg
2026-02-15 23:35 ` [PATCH 26/79] block: rnull: add badblocks support Andreas Hindborg
2026-02-15 23:35 ` [PATCH 27/79] block: rnull: add badblocks_once support Andreas Hindborg
2026-02-15 23:35 ` [PATCH 28/79] block: rnull: add partial I/O support for bad blocks Andreas Hindborg
2026-02-15 23:35 ` [PATCH 29/79] block: rust: add `TagSet` private data support Andreas Hindborg
2026-02-15 23:35 ` [PATCH 30/79] block: rust: add `hctx` " Andreas Hindborg
2026-02-15 23:35 ` [PATCH 31/79] block: rnull: add volatile cache emulation Andreas Hindborg
2026-02-15 23:35 ` [PATCH 32/79] block: rust: implement `Sync` for `GenDisk` Andreas Hindborg
2026-02-15 23:35 ` [PATCH 33/79] block: rust: add a back reference feature to `GenDisk` Andreas Hindborg
2026-02-15 23:35 ` [PATCH 34/79] block: rust: introduce an idle type state for `Request` Andreas Hindborg
2026-02-15 23:35 ` [PATCH 35/79] block: rust: add a request queue abstraction Andreas Hindborg
2026-02-15 23:35 ` [PATCH 36/79] block: rust: add a method to get the request queue for a request Andreas Hindborg
2026-02-15 23:35 ` [PATCH 37/79] block: rust: introduce `kernel::block::error` Andreas Hindborg
2026-02-15 23:35 ` [PATCH 38/79] block: rust: require `queue_rq` to return a `BlkResult` Andreas Hindborg
2026-02-15 23:35 ` [PATCH 39/79] block: rust: add `GenDisk::queue_data` Andreas Hindborg
2026-02-15 23:35 ` [PATCH 40/79] block: rnull: add bandwidth limiting Andreas Hindborg
2026-02-15 23:35 ` [PATCH 41/79] block: rnull: add blocking queue mode Andreas Hindborg
2026-02-15 23:35 ` [PATCH 42/79] block: rnull: add shared tags Andreas Hindborg
2026-02-15 23:35 ` [PATCH 43/79] block: rnull: add queue depth config option Andreas Hindborg
2026-02-15 23:35 ` [PATCH 44/79] block: rust: add an abstraction for `bindings::req_op` Andreas Hindborg
2026-02-15 23:35 ` [PATCH 45/79] block: rust: add a method to set the target sector of a request Andreas Hindborg
2026-02-15 23:35 ` [PATCH 46/79] block: rust: move gendisk vtable construction to separate function Andreas Hindborg
2026-02-15 23:35 ` [PATCH 47/79] block: rust: add zoned block device support Andreas Hindborg
2026-02-15 23:35 ` [PATCH 48/79] block: rnull: add zoned storage support Andreas Hindborg
2026-02-15 23:35 ` [PATCH 49/79] block: rust: add `map_queues` support Andreas Hindborg
2026-02-15 23:35 ` [PATCH 50/79] block: rust: add an abstraction for `struct blk_mq_queue_map` Andreas Hindborg
2026-02-15 23:35 ` [PATCH 51/79] block: rust: add polled completion support Andreas Hindborg
2026-02-15 23:35 ` [PATCH 52/79] block: rust: add accessors to `TagSet` Andreas Hindborg
2026-02-15 23:35 ` [PATCH 53/79] block: rnull: add polled completion support Andreas Hindborg
2026-02-15 23:35 ` [PATCH 54/79] block: rnull: add REQ_OP_FLUSH support Andreas Hindborg
2026-02-15 23:35 ` [PATCH 55/79] block: rust: add request flags abstraction Andreas Hindborg
2026-02-15 23:35 ` [PATCH 56/79] block: rust: add abstraction for block queue feature flags Andreas Hindborg
2026-02-15 23:35 ` [PATCH 57/79] block: rust: allow setting write cache and FUA flags for `GenDisk` Andreas Hindborg
2026-02-15 23:35 ` [PATCH 58/79] block: rust: add `Segment::copy_to_page_limit` Andreas Hindborg
2026-02-15 23:35 ` [PATCH 59/79] block: rnull: add fua support Andreas Hindborg
2026-02-15 23:35 ` [PATCH 60/79] block: fix arg type in `blk_mq_update_nr_hw_queues` Andreas Hindborg
2026-02-15 23:35 ` [PATCH 61/79] block: rust: add `GenDisk::tag_set` Andreas Hindborg
2026-02-15 23:35 ` [PATCH 62/79] block: rust: add `TagSet::update_hw_queue_count` Andreas Hindborg
2026-02-16 23:59 ` Ken Kurematsu
2026-02-17 9:54 ` Andreas Hindborg
2026-02-15 23:35 ` [PATCH 63/79] block: rnull: add an option to change the number of hardware queues Andreas Hindborg
2026-02-15 23:35 ` [PATCH 64/79] block: rust: add an abstraction for `struct rq_list` Andreas Hindborg
2026-02-15 23:35 ` [PATCH 65/79] block: rust: add `queue_rqs` vtable hook Andreas Hindborg
2026-02-15 23:35 ` [PATCH 66/79] block: rnull: support queue_rqs Andreas Hindborg
2026-02-15 23:35 ` [PATCH 67/79] block: rust: remove the `is_poll` parameter from `queue_rq` Andreas Hindborg
2026-02-15 23:35 ` [PATCH 68/79] block: rust: add a debug assert for refcounts Andreas Hindborg
2026-02-15 23:35 ` [PATCH 69/79] block: rust: add `TagSet::tag_to_rq` Andreas Hindborg
2026-02-15 23:35 ` [PATCH 70/79] block: rust: add `Request::queue_index` Andreas Hindborg
2026-02-15 23:35 ` [PATCH 71/79] block: rust: add `Request::requeue` Andreas Hindborg
2026-02-15 23:35 ` [PATCH 72/79] block: rust: add `request_timeout` hook Andreas Hindborg
2026-02-15 23:36 ` [PATCH 73/79] block: rnull: add fault injection support Andreas Hindborg
2026-02-15 23:36 ` [PATCH 74/79] block: rust: add max_sectors option to `GenDiskBuilder` Andreas Hindborg
2026-02-15 23:36 ` [PATCH 75/79] block: rnull: allow configuration of the maximum IO size Andreas Hindborg
2026-02-15 23:36 ` [PATCH 76/79] block: rust: add `virt_boundary_mask` option to `GenDiskBuilder` Andreas Hindborg
2026-02-15 23:36 ` [PATCH 77/79] block: rnull: add `virt_boundary` option Andreas Hindborg
2026-02-15 23:36 ` [PATCH 78/79] block: rnull: add `shared_tag_bitmap` config option Andreas Hindborg
2026-02-15 23:36 ` [PATCH 79/79] block: rnull: add zone offline and readonly configfs files Andreas Hindborg
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=20260216-rnull-v6-19-rc5-send-v1-5-de9a7af4b469@kernel.org \
--to=a.hindborg@kernel.org \
--cc=Liam.Howlett@oracle.com \
--cc=aliceryhl@google.com \
--cc=anna-maria@linutronix.de \
--cc=axboe@kernel.dk \
--cc=bjorn3_gh@protonmail.com \
--cc=boqun.feng@gmail.com \
--cc=dakr@kernel.org \
--cc=frederic@kernel.org \
--cc=fujita.tomonori@gmail.com \
--cc=gary@garyguo.net \
--cc=jstultz@google.com \
--cc=linux-block@vger.kernel.org \
--cc=linux-kernel@vger.kernel.org \
--cc=linux-mm@kvack.org \
--cc=lorenzo.stoakes@oracle.com \
--cc=lossin@kernel.org \
--cc=lyude@redhat.com \
--cc=ojeda@kernel.org \
--cc=rust-for-linux@vger.kernel.org \
--cc=sboyd@kernel.org \
--cc=tglx@kernel.org \
--cc=tmgross@umich.edu \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox