From: "Gary Guo" <gary@garyguo.net>
To: "Danilo Krummrich" <dakr@kernel.org>, <aliceryhl@google.com>,
<daniel.almeida@collabora.com>, <acourbot@nvidia.com>,
<ecourtney@nvidia.com>, <ojeda@kernel.org>, <boqun@kernel.org>,
<gary@garyguo.net>, <bjorn3_gh@protonmail.com>,
<lossin@kernel.org>, <a.hindborg@kernel.org>, <tmgross@umich.edu>,
<deborah.brouwer@collabora.com>, <boris.brezillon@collabora.com>
Cc: <driver-core@lists.linux.dev>, <linux-kernel@vger.kernel.org>,
<nova-gpu@lists.linux.dev>, <dri-devel@lists.freedesktop.org>,
<rust-for-linux@vger.kernel.org>
Subject: Re: [PATCH v2 3/7] rust: drm: Add RegistrationData to drm::Driver
Date: Wed, 03 Jun 2026 12:51:37 +0100 [thread overview]
Message-ID: <DIZE54GCJ9MS.DW75EP5OM604@garyguo.net> (raw)
In-Reply-To: <20260603011711.2077361-4-dakr@kernel.org>
On Wed Jun 3, 2026 at 2:15 AM BST, Danilo Krummrich wrote:
> Add a RegistrationData associated type to drm::Driver. This is a ForLt
> type whose lifetime is tied to the parent bus device binding scope.
>
> Registration<'a, T> takes ownership of the data via Pin<KBox<_>>,
> storing it with its real lifetime. The pointer is written to drm::Device
> before drm_dev_register() to ensure it is already in place when ioctls
> arrive.
>
> UnbindGuard::registration_data_with() provides access with the lifetime
> shortened from 'static via ForLt::cast_ref_unchecked. Since
> Registration::drop() calls drm_dev_unplug(), which performs an SRCU
> barrier waiting for all drm_dev_enter() critical sections to complete,
> the data is guaranteed to remain valid for the duration of any
> UnbindGuard.
>
> Signed-off-by: Danilo Krummrich <dakr@kernel.org>
> ---
> drivers/gpu/drm/nova/driver.rs | 16 +++--
> drivers/gpu/drm/tyr/driver.rs | 16 +++--
> rust/kernel/drm/device.rs | 49 +++++++++++++
> rust/kernel/drm/driver.rs | 123 ++++++++++++++++++++++++---------
> rust/kernel/drm/mod.rs | 1 +
> 5 files changed, 162 insertions(+), 43 deletions(-)
>
> diff --git a/drivers/gpu/drm/nova/driver.rs b/drivers/gpu/drm/nova/driver.rs
> index c5b0313006bd..4267e6e6dbb4 100644
> --- a/drivers/gpu/drm/nova/driver.rs
> +++ b/drivers/gpu/drm/nova/driver.rs
> @@ -12,7 +12,8 @@
> ioctl, //
> },
> prelude::*,
> - sync::aref::ARef, //
> + sync::aref::ARef,
> + types::ForLt, //
> };
>
> use crate::file::File;
> @@ -20,9 +21,10 @@
>
> pub(crate) struct NovaDriver;
>
> -pub(crate) struct Nova {
> +pub(crate) struct Nova<'bound> {
> #[expect(unused)]
> drm: ARef<drm::Device<NovaDriver>>,
> + _reg: drm::Registration<'bound, NovaDriver>,
> }
>
> /// Convienence type alias for the DRM device type for this driver
> @@ -56,7 +58,7 @@ pub(crate) struct NovaData {
>
> impl auxiliary::Driver for NovaDriver {
> type IdInfo = ();
> - type Data<'bound> = Nova;
> + type Data<'bound> = Nova<'bound>;
> const ID_TABLE: auxiliary::IdTable<Self::IdInfo> = &AUX_TABLE;
>
> fn probe<'bound>(
> @@ -66,15 +68,19 @@ fn probe<'bound>(
> let data = try_pin_init!(NovaData { adev: adev.into() });
>
> let drm = drm::UnregisteredDevice::<Self>::new(adev, data)?;
> - let drm = drm::Registration::new_foreign_owned(drm, adev.as_ref(), 0)?;
> + let reg = drm::Registration::new(adev.as_ref(), drm, (), 0)?;
>
> - Ok(Nova { drm: drm.into() })
> + Ok(Nova {
> + drm: reg.device().into(),
> + _reg: reg,
> + })
> }
> }
>
> #[vtable]
> impl drm::Driver for NovaDriver {
> type Data = NovaData;
> + type RegistrationData = ForLt!(());
> type File = File;
> type Object<Ctx: drm::DeviceContext> = gem::Object<NovaObject, Ctx>;
> type ParentDevice<Ctx: DeviceContext> = auxiliary::Device<Ctx>;
> diff --git a/drivers/gpu/drm/tyr/driver.rs b/drivers/gpu/drm/tyr/driver.rs
> index 338c25ccc151..819f64a7573d 100644
> --- a/drivers/gpu/drm/tyr/driver.rs
> +++ b/drivers/gpu/drm/tyr/driver.rs
> @@ -31,7 +31,8 @@
> aref::ARef,
> Mutex, //
> },
> - time, //
> + time,
> + types::ForLt, //
> };
>
> use crate::{
> @@ -52,8 +53,9 @@
> pub(crate) struct TyrPlatformDriver;
>
> #[pin_data(PinnedDrop)]
> -pub(crate) struct TyrPlatformDriverData {
> +pub(crate) struct TyrPlatformDriverData<'bound> {
> _device: ARef<TyrDrmDevice>,
> + _reg: drm::Registration<'bound, TyrDrmDriver>,
> }
>
> #[pin_data]
> @@ -98,7 +100,7 @@ fn issue_soft_reset(dev: &Device, iomem: &IoMem<'_>) -> Result {
>
> impl platform::Driver for TyrPlatformDriver {
> type IdInfo = ();
> - type Data<'bound> = TyrPlatformDriverData;
> + type Data<'bound> = TyrPlatformDriverData<'bound>;
> const OF_ID_TABLE: Option<of::IdTable<Self::IdInfo>> = Some(&OF_TABLE);
>
> fn probe<'bound>(
> @@ -150,10 +152,11 @@ fn probe<'bound>(
> });
>
> let tdev = drm::UnregisteredDevice::<TyrDrmDriver>::new(pdev, data)?;
> - let tdev = drm::driver::Registration::new_foreign_owned(tdev, pdev.as_ref(), 0)?;
> + let reg = drm::Registration::new(pdev.as_ref(), tdev, (), 0)?;
>
> let driver = TyrPlatformDriverData {
> - _device: tdev.into(),
> + _device: reg.device().into(),
> + _reg: reg,
> };
>
> // We need this to be dev_info!() because dev_dbg!() does not work at
> @@ -164,7 +167,7 @@ fn probe<'bound>(
> }
>
> #[pinned_drop]
> -impl PinnedDrop for TyrPlatformDriverData {
> +impl PinnedDrop for TyrPlatformDriverData<'_> {
> fn drop(self: Pin<&mut Self>) {}
> }
>
> @@ -181,6 +184,7 @@ fn drop(self: Pin<&mut Self>) {}
> #[vtable]
> impl drm::Driver for TyrDrmDriver {
> type Data = TyrDrmDeviceData;
> + type RegistrationData = ForLt!(());
> type File = TyrDrmFileData;
> type Object<R: drm::DeviceContext> = drm::gem::shmem::Object<BoData, R>;
> type ParentDevice<Ctx: DeviceContext> = platform::Device<Ctx>;
> diff --git a/rust/kernel/drm/device.rs b/rust/kernel/drm/device.rs
> index 828618ae19af..9e5e069b5135 100644
> --- a/rust/kernel/drm/device.rs
> +++ b/rust/kernel/drm/device.rs
> @@ -23,6 +23,7 @@
> AlwaysRefCounted, //
> },
> types::{
> + ForLt,
> NotThreadSafe,
> Opaque, //
> },
> @@ -35,6 +36,7 @@
> };
> use core::{
> alloc::Layout,
> + cell::UnsafeCell,
> marker::PhantomData,
> mem,
> ops::Deref,
> @@ -259,6 +261,9 @@ pub fn new(
> // SAFETY: `drm_dev` is still private to this function.
> unsafe { (*drm_dev).driver = const { &Self::VTABLE } };
>
> + // SAFETY: `raw_drm` is valid; no concurrent access before registration.
> + unsafe { (*raw_drm.as_ptr()).registration_data = UnsafeCell::new(NonNull::dangling()) };
> +
> // SAFETY: The reference count is one, and now we take ownership of that reference as a
> // `drm::Device`.
> // INVARIANT: We just created the device above, but have yet to call `drm_dev_register`.
> @@ -290,6 +295,7 @@ pub fn new(
> pub struct Device<T: drm::Driver, C: DeviceContext = Registered> {
> dev: Opaque<bindings::drm_device>,
> data: T::Data,
> + pub(super) registration_data: UnsafeCell<NonNull<<T::RegistrationData as ForLt>::Of<'static>>>,
> _ctx: PhantomData<C>,
> }
>
> @@ -298,6 +304,27 @@ pub(crate) fn as_raw(&self) -> *mut bindings::drm_device {
> self.dev.get()
> }
>
> + /// Returns a reference to the registration data with lifetime shortened from `'static`.
> + ///
> + /// # Safety
> + ///
> + /// The caller must ensure that:
> + ///
> + /// - The parent bus device is bound (e.g. by holding an active `drm_dev_enter()` critical
> + /// section via [`UnbindGuard`]).
> + /// - The returned reference is not exposed to code that can choose a concrete lifetime for
> + /// it, as that would be unsound for types that are invariant over their lifetime parameter
> + /// (e.g. it must be passed through an HRTB-bounded closure).
> + unsafe fn registration_data_unchecked(&self) -> &<T::RegistrationData as ForLt>::Of<'_> {
> + // SAFETY: Caller guarantees the parent bus device is bound, hence
> + // the pointer is valid.
> + let static_ref = unsafe { (*self.registration_data.get()).as_ref() };
> +
> + // SAFETY: Caller guarantees the reference is only used behind an HRTB, making the
> + // round-trip from `'static` sound regardless of variance.
> + unsafe { T::RegistrationData::cast_ref_unchecked(static_ref) }
> + }
> +
> /// # Safety
> ///
> /// `ptr` must be a valid pointer to a `struct device` embedded in `Self`.
> @@ -412,6 +439,28 @@ pub struct UnbindGuard<'a, T: drm::Driver> {
> idx: i32,
> }
>
> +impl<T: drm::Driver> UnbindGuard<'_, T> {
> + /// Access the parent device and registration data through a closure, with both lifetimes
> + /// tied to the closure scope.
> + ///
> + /// The data is owned by [`Registration`](drm::Registration) and is guaranteed to remain valid
> + /// for the duration of this guard, since [`Registration`](drm::Registration)'s `drop` calls
> + /// `drm_dev_unplug()` which waits for all `drm_dev_enter()` critical sections to complete.
> + pub fn registration_data_with<R, F>(&self, f: F) -> R
> + where
> + F: for<'a> FnOnce(
> + &'a T::ParentDevice<device::Bound>,
> + &'a <T::RegistrationData as ForLt>::Of<'a>,
> + ) -> R,
> + {
> + // SAFETY: We hold an active `drm_dev_enter()` critical section, so the parent bus
> + // device is guaranteed to be bound. The closure's HRTB `for<'a>` prevents the caller
> + // from smuggling in references with a concrete short lifetime, satisfying the lifetime
> + // requirement of `registration_data_unchecked`.
> + f(&**self, unsafe { self.dev.registration_data_unchecked() })
> + }
> +}
> +
> impl<T: drm::Driver> Deref for UnbindGuard<'_, T> {
> type Target = T::ParentDevice<device::Bound>;
>
> diff --git a/rust/kernel/drm/driver.rs b/rust/kernel/drm/driver.rs
> index f68a17d8939d..6aa1cb99cc7f 100644
> --- a/rust/kernel/drm/driver.rs
> +++ b/rust/kernel/drm/driver.rs
> @@ -7,11 +7,11 @@
> use crate::{
> bindings,
> device,
> - devres,
> drm,
> error::to_result,
> prelude::*,
> - sync::aref::ARef, //
> + sync::aref::ARef,
> + types::ForLt, //
> };
> use core::{
> mem,
> @@ -110,6 +110,16 @@ pub trait Driver {
> /// Context data associated with the DRM driver
> type Data: Sync + Send;
>
> + /// Data owned by the [`Registration`] and accessible through [`drm::device::UnbindGuard`].
> + ///
> + /// This is a [`ForLt`](trait@ForLt) type whose lifetime is tied to the parent bus
> + /// device binding scope.
> + /// The data is only accessible while the parent bus device is bound (i.e. within a
> + /// `drm_dev_enter/exit` critical section), and references handed out by
> + /// [`UnbindGuard::registration_data_with()`](drm::device::UnbindGuard::registration_data_with)
> + /// ties the lifetime to the closure scope.
> + type RegistrationData: ForLt;
> +
> /// The type used to manage memory for this driver.
> type Object<Ctx: drm::DeviceContext>: AllocImpl;
>
> @@ -139,12 +149,57 @@ pub trait Driver {
> /// The registration type of a `drm::Device`.
> ///
> /// Once the `Registration` structure is dropped, the device is unregistered.
> -pub struct Registration<T: Driver>(ARef<drm::Device<T>>);
> +pub struct Registration<'a, T: Driver> {
> + drm: ARef<drm::Device<T>>,
> + _reg_data: Pin<KBox<<T::RegistrationData as ForLt>::Of<'a>>>,
> +}
> +
> +impl<'a, T: Driver> Registration<'a, T>
> +where
> + for<'b> <T::RegistrationData as ForLt>::Of<'b>: Send,
> +{
> + /// Register a new [`UnregisteredDevice`](drm::UnregisteredDevice) with userspace.
> + ///
> + /// # Safety
> + ///
> + /// The caller must not `mem::forget()` the returned [`Registration`] or otherwise prevent its
> + /// [`Drop`] implementation from running, since the registration data may contain borrowed
> + /// references that become invalid after `'a` ends.
> + ///
> + /// If the registration data is `'static`, use the safe [`Registration::new()`] instead.
> + pub unsafe fn new_with_lt<E>(
> + dev: &'a device::Device<device::Bound>,
> + drm: drm::UnregisteredDevice<T>,
> + reg_data: impl PinInit<<T::RegistrationData as ForLt>::Of<'a>, E>,
> + flags: usize,
> + ) -> Result<Self>
> + where
> + Error: From<E>,
> + {
> + if drm.as_ref().as_raw() != dev.as_raw() {
> + return Err(EINVAL);
> + }
>
> -impl<T: Driver> Registration<T> {
> - fn new(drm: drm::UnregisteredDevice<T>, flags: usize) -> Result<Self> {
> - // SAFETY: `drm.as_raw()` is valid by the invariants of `drm::Device`.
> - to_result(unsafe { bindings::drm_dev_register(drm.as_raw(), flags) })?;
> + let reg_data: Pin<KBox<<T::RegistrationData as ForLt>::Of<'a>>> =
> + KBox::pin_init(reg_data, GFP_KERNEL)?;
> +
> + // Store the registration data pointer in the device before registration, so that it is
> + // visible once ioctls can be called.
> + //
> + // SAFETY: Lifetimes do not affect layout, so the pointer cast is sound.
> + let ptr: NonNull<<T::RegistrationData as ForLt>::Of<'static>> =
> + unsafe { mem::transmute(NonNull::from(Pin::get_ref(reg_data.as_ref()))) };
> +
> + // SAFETY: No concurrent access; the device is not yet registered.
> + unsafe { *drm.registration_data.get() = ptr };
> +
> + // SAFETY: `drm` is a valid, initialized but not yet registered DRM device.
> + let ret = unsafe { bindings::drm_dev_register(drm.as_raw(), flags) };
> + if let Err(e) = to_result(ret) {
> + // SAFETY: No concurrent access; registration failed.
> + unsafe { *drm.registration_data.get() = NonNull::dangling() };
> + return Err(e);
> + }
>
> // SAFETY: We just called `drm_dev_register` above
> let new = NonNull::from(unsafe { drm.assume_ctx() });
> @@ -156,48 +211,49 @@ fn new(drm: drm::UnregisteredDevice<T>, flags: usize) -> Result<Self> {
> // one reference to the device - which we take ownership over here.
> let new = unsafe { ARef::from_raw(new) };
>
> - Ok(Self(new))
> + Ok(Self {
> + drm: new,
> + _reg_data: reg_data,
> + })
> }
>
> - /// Registers a new [`UnregisteredDevice`](drm::UnregisteredDevice) with userspace.
> - ///
> - /// Ownership of the [`Registration`] object is passed to [`devres::register`].
> - pub fn new_foreign_owned<'a>(
> - drm: drm::UnregisteredDevice<T>,
> + /// Safe variant of [`Registration::new_with_lt()`] for registration data that does not contain
> + /// borrowed references.
> + pub fn new<E>(
This is currently unsound, as leaking the unbind guard also gives out
`&Device<Bound>` in addition to the registration data.
I think we should just remove the not pass `&Device<Bound>` to ioctl callbacks.
Giving back registration data is sufficient; if a device driver needs
`&Device<Bound>` it can just store a reference in its registration data; more
commonly I suspect it will just store whatever device resource is needed and
doesn't need `&Device<Bound>` (with the introduction of lifetime, we have much
fewer cases that we actually need `&Device<Bound>` and cannot be replaced with a
direct reference to the device resource).
Not passing this bound device allows us to make this safe, and also remove the
need of patch 1 and patch 5. Actually, your patch 7 is a good demonstration of
the pattern that I described :)
Best,
Gary
> dev: &'a device::Device<device::Bound>,
> + drm: drm::UnregisteredDevice<T>,
> + reg_data: impl PinInit<<T::RegistrationData as ForLt>::Of<'a>, E>,
> flags: usize,
> - ) -> Result<&'a drm::Device<T>>
> + ) -> Result<Self>
> where
> - T: 'static,
> + <T::RegistrationData as ForLt>::Of<'a>: 'static,
> + Error: From<E>,
> {
> - if drm.as_ref().as_raw() != dev.as_raw() {
> - return Err(EINVAL);
> - }
> -
> - let reg = Registration::<T>::new(drm, flags)?;
> - let drm = NonNull::from(reg.device());
> -
> - devres::register(dev, reg, GFP_KERNEL)?;
> -
> - // SAFETY: Since `reg` was passed to devres::register(), the device now owns the lifetime
> - // of the DRM registration - ensuring that this references lives for at least as long as 'a.
> - Ok(unsafe { drm.as_ref() })
> + // SAFETY: `Of<'a>: 'static` guarantees the data contains no borrowed references,
> + // so forgetting the `Registration` cannot cause use-after-free.
> + unsafe { Self::new_with_lt(dev, drm, reg_data, flags) }
> }
>
> /// Returns a reference to the `Device` instance for this registration.
> pub fn device(&self) -> &drm::Device<T> {
> - &self.0
> + &self.drm
> }
> }
>
> // SAFETY: `Registration` doesn't offer any methods or access to fields when shared between
> // threads, hence it's safe to share it.
> -unsafe impl<T: Driver> Sync for Registration<T> {}
> +unsafe impl<T: Driver> Sync for Registration<'_, T> where
> + for<'a> <T::RegistrationData as ForLt>::Of<'a>: Send
> +{
> +}
>
> // SAFETY: Registration with and unregistration from the DRM subsystem can happen from any thread.
> -unsafe impl<T: Driver> Send for Registration<T> {}
> +unsafe impl<T: Driver> Send for Registration<'_, T> where
> + for<'a> <T::RegistrationData as ForLt>::Of<'a>: Send
> +{
> +}
>
> -impl<T: Driver> Drop for Registration<T> {
> +impl<T: Driver> Drop for Registration<'_, T> {
> fn drop(&mut self) {
> // Use `drm_dev_unplug` rather than `drm_dev_unregister` to ensure that existing
> // `drm_dev_enter()` critical sections complete before unregistration proceeds. This
> @@ -207,6 +263,9 @@ fn drop(&mut self) {
> //
> // SAFETY: Safe by the invariant of `ARef<drm::Device<T>>`. The existence of this
> // `Registration` also guarantees that this `drm::Device` is actually registered.
> - unsafe { bindings::drm_dev_unplug(self.0.as_raw()) };
> + unsafe { bindings::drm_dev_unplug(self.drm.as_raw()) };
> + // After drm_dev_unplug(), the SRCU barrier guarantees that all UnbindGuard critical
> + // sections have completed, so no one holds a reference to reg_data anymore.
> + // reg_data is dropped here automatically.
> }
> }
> diff --git a/rust/kernel/drm/mod.rs b/rust/kernel/drm/mod.rs
> index a66e7166f66b..59870bb09de2 100644
> --- a/rust/kernel/drm/mod.rs
> +++ b/rust/kernel/drm/mod.rs
> @@ -12,6 +12,7 @@
> pub use self::device::Device;
> pub use self::device::DeviceContext;
> pub use self::device::Registered;
> +pub use self::device::UnbindGuard;
> pub use self::device::Uninit;
> pub use self::device::UnregisteredDevice;
> pub use self::driver::Driver;
next prev parent reply other threads:[~2026-06-03 11:51 UTC|newest]
Thread overview: 13+ messages / expand[flat|nested] mbox.gz Atom feed top
2026-06-03 1:15 [PATCH v2 0/7] rust: drm: Higher-Ranked Lifetime private data Danilo Krummrich
2026-06-03 1:15 ` [PATCH v2 1/7] rust: drm: Add Driver::ParentDevice associated type Danilo Krummrich
2026-06-03 1:15 ` [PATCH v2 2/7] rust: drm: Add UnbindGuard for drm_dev_enter/exit critical sections Danilo Krummrich
2026-06-03 11:47 ` Gary Guo
2026-06-03 1:15 ` [PATCH v2 3/7] rust: drm: Add RegistrationData to drm::Driver Danilo Krummrich
2026-06-03 11:51 ` Gary Guo [this message]
2026-06-03 22:24 ` Danilo Krummrich
2026-06-03 22:36 ` Gary Guo
2026-06-03 23:29 ` Deborah Brouwer
2026-06-03 1:15 ` [PATCH v2 4/7] rust: drm: Wrap ioctl dispatch in UnbindGuard Danilo Krummrich
2026-06-03 1:15 ` [PATCH v2 5/7] rust: drm: Pass bound parent device to ioctl handlers Danilo Krummrich
2026-06-03 1:15 ` [PATCH v2 6/7] rust: drm: Pass registration data " Danilo Krummrich
2026-06-03 1:15 ` [PATCH v2 7/7] drm: nova: convert to use DRM registration data Danilo Krummrich
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=DIZE54GCJ9MS.DW75EP5OM604@garyguo.net \
--to=gary@garyguo.net \
--cc=a.hindborg@kernel.org \
--cc=acourbot@nvidia.com \
--cc=aliceryhl@google.com \
--cc=bjorn3_gh@protonmail.com \
--cc=boqun@kernel.org \
--cc=boris.brezillon@collabora.com \
--cc=dakr@kernel.org \
--cc=daniel.almeida@collabora.com \
--cc=deborah.brouwer@collabora.com \
--cc=dri-devel@lists.freedesktop.org \
--cc=driver-core@lists.linux.dev \
--cc=ecourtney@nvidia.com \
--cc=linux-kernel@vger.kernel.org \
--cc=lossin@kernel.org \
--cc=nova-gpu@lists.linux.dev \
--cc=ojeda@kernel.org \
--cc=rust-for-linux@vger.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