mirror of
https://github.com/torvalds/linux.git
synced 2025-12-07 20:06:24 +00:00
rust: auxiliary: unregister on parent device unbind
Guarantee that an auxiliary driver will be unbound before its parent is unbound; there is no point in operating an auxiliary device whose parent has been unbound. In practice, this guarantee allows us to assume that for a bound auxiliary device, also the parent device is bound. This is useful when an auxiliary driver calls into its parent, since it allows the parent to directly access device resources and its device private data due to the guaranteed bound device context. Reviewed-by: Alice Ryhl <aliceryhl@google.com> Reviewed-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org> Signed-off-by: Danilo Krummrich <dakr@kernel.org>
This commit is contained in:
@@ -3,6 +3,7 @@
|
|||||||
use kernel::{
|
use kernel::{
|
||||||
auxiliary, c_str,
|
auxiliary, c_str,
|
||||||
device::Core,
|
device::Core,
|
||||||
|
devres::Devres,
|
||||||
pci,
|
pci,
|
||||||
pci::{Class, ClassMask, Vendor},
|
pci::{Class, ClassMask, Vendor},
|
||||||
prelude::*,
|
prelude::*,
|
||||||
@@ -16,7 +17,8 @@ use crate::gpu::Gpu;
|
|||||||
pub(crate) struct NovaCore {
|
pub(crate) struct NovaCore {
|
||||||
#[pin]
|
#[pin]
|
||||||
pub(crate) gpu: Gpu,
|
pub(crate) gpu: Gpu,
|
||||||
_reg: auxiliary::Registration,
|
#[pin]
|
||||||
|
_reg: Devres<auxiliary::Registration>,
|
||||||
}
|
}
|
||||||
|
|
||||||
const BAR0_SIZE: usize = SZ_16M;
|
const BAR0_SIZE: usize = SZ_16M;
|
||||||
@@ -65,12 +67,12 @@ impl pci::Driver for NovaCore {
|
|||||||
|
|
||||||
Ok(try_pin_init!(Self {
|
Ok(try_pin_init!(Self {
|
||||||
gpu <- Gpu::new(pdev, bar.clone(), bar.access(pdev.as_ref())?),
|
gpu <- Gpu::new(pdev, bar.clone(), bar.access(pdev.as_ref())?),
|
||||||
_reg: auxiliary::Registration::new(
|
_reg <- auxiliary::Registration::new(
|
||||||
pdev.as_ref(),
|
pdev.as_ref(),
|
||||||
c_str!("nova-drm"),
|
c_str!("nova-drm"),
|
||||||
0, // TODO[XARR]: Once it lands, use XArray; for now we don't use the ID.
|
0, // TODO[XARR]: Once it lands, use XArray; for now we don't use the ID.
|
||||||
crate::MODULE_NAME
|
crate::MODULE_NAME
|
||||||
)?,
|
),
|
||||||
}))
|
}))
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -7,6 +7,7 @@
|
|||||||
use crate::{
|
use crate::{
|
||||||
bindings, container_of, device,
|
bindings, container_of, device,
|
||||||
device_id::{RawDeviceId, RawDeviceIdIndex},
|
device_id::{RawDeviceId, RawDeviceIdIndex},
|
||||||
|
devres::Devres,
|
||||||
driver,
|
driver,
|
||||||
error::{from_result, to_result, Result},
|
error::{from_result, to_result, Result},
|
||||||
prelude::*,
|
prelude::*,
|
||||||
@@ -279,8 +280,8 @@ unsafe impl Sync for Device {}
|
|||||||
|
|
||||||
/// The registration of an auxiliary device.
|
/// The registration of an auxiliary device.
|
||||||
///
|
///
|
||||||
/// This type represents the registration of a [`struct auxiliary_device`]. When an instance of this
|
/// This type represents the registration of a [`struct auxiliary_device`]. When its parent device
|
||||||
/// type is dropped, its respective auxiliary device will be unregistered from the system.
|
/// is unbound, the corresponding auxiliary device will be unregistered from the system.
|
||||||
///
|
///
|
||||||
/// # Invariants
|
/// # Invariants
|
||||||
///
|
///
|
||||||
@@ -290,44 +291,56 @@ pub struct Registration(NonNull<bindings::auxiliary_device>);
|
|||||||
|
|
||||||
impl Registration {
|
impl Registration {
|
||||||
/// Create and register a new auxiliary device.
|
/// Create and register a new auxiliary device.
|
||||||
pub fn new(parent: &device::Device, name: &CStr, id: u32, modname: &CStr) -> Result<Self> {
|
pub fn new<'a>(
|
||||||
let boxed = KBox::new(Opaque::<bindings::auxiliary_device>::zeroed(), GFP_KERNEL)?;
|
parent: &'a device::Device<device::Bound>,
|
||||||
let adev = boxed.get();
|
name: &'a CStr,
|
||||||
|
id: u32,
|
||||||
|
modname: &'a CStr,
|
||||||
|
) -> impl PinInit<Devres<Self>, Error> + 'a {
|
||||||
|
pin_init::pin_init_scope(move || {
|
||||||
|
let boxed = KBox::new(Opaque::<bindings::auxiliary_device>::zeroed(), GFP_KERNEL)?;
|
||||||
|
let adev = boxed.get();
|
||||||
|
|
||||||
// SAFETY: It's safe to set the fields of `struct auxiliary_device` on initialization.
|
// SAFETY: It's safe to set the fields of `struct auxiliary_device` on initialization.
|
||||||
unsafe {
|
unsafe {
|
||||||
(*adev).dev.parent = parent.as_raw();
|
(*adev).dev.parent = parent.as_raw();
|
||||||
(*adev).dev.release = Some(Device::release);
|
(*adev).dev.release = Some(Device::release);
|
||||||
(*adev).name = name.as_char_ptr();
|
(*adev).name = name.as_char_ptr();
|
||||||
(*adev).id = id;
|
(*adev).id = id;
|
||||||
}
|
}
|
||||||
|
|
||||||
// SAFETY: `adev` is guaranteed to be a valid pointer to a `struct auxiliary_device`,
|
|
||||||
// which has not been initialized yet.
|
|
||||||
unsafe { bindings::auxiliary_device_init(adev) };
|
|
||||||
|
|
||||||
// Now that `adev` is initialized, leak the `Box`; the corresponding memory will be freed
|
|
||||||
// by `Device::release` when the last reference to the `struct auxiliary_device` is dropped.
|
|
||||||
let _ = KBox::into_raw(boxed);
|
|
||||||
|
|
||||||
// SAFETY:
|
|
||||||
// - `adev` is guaranteed to be a valid pointer to a `struct auxiliary_device`, which has
|
|
||||||
// been initialialized,
|
|
||||||
// - `modname.as_char_ptr()` is a NULL terminated string.
|
|
||||||
let ret = unsafe { bindings::__auxiliary_device_add(adev, modname.as_char_ptr()) };
|
|
||||||
if ret != 0 {
|
|
||||||
// SAFETY: `adev` is guaranteed to be a valid pointer to a `struct auxiliary_device`,
|
// SAFETY: `adev` is guaranteed to be a valid pointer to a `struct auxiliary_device`,
|
||||||
// which has been initialialized.
|
// which has not been initialized yet.
|
||||||
unsafe { bindings::auxiliary_device_uninit(adev) };
|
unsafe { bindings::auxiliary_device_init(adev) };
|
||||||
|
|
||||||
return Err(Error::from_errno(ret));
|
// Now that `adev` is initialized, leak the `Box`; the corresponding memory will be
|
||||||
}
|
// freed by `Device::release` when the last reference to the `struct auxiliary_device`
|
||||||
|
// is dropped.
|
||||||
|
let _ = KBox::into_raw(boxed);
|
||||||
|
|
||||||
// SAFETY: `adev` is guaranteed to be non-null, since the `KBox` was allocated successfully.
|
// SAFETY:
|
||||||
//
|
// - `adev` is guaranteed to be a valid pointer to a `struct auxiliary_device`, which
|
||||||
// INVARIANT: The device will remain registered until `auxiliary_device_delete()` is called,
|
// has been initialized,
|
||||||
// which happens in `Self::drop()`.
|
// - `modname.as_char_ptr()` is a NULL terminated string.
|
||||||
Ok(Self(unsafe { NonNull::new_unchecked(adev) }))
|
let ret = unsafe { bindings::__auxiliary_device_add(adev, modname.as_char_ptr()) };
|
||||||
|
if ret != 0 {
|
||||||
|
// SAFETY: `adev` is guaranteed to be a valid pointer to a
|
||||||
|
// `struct auxiliary_device`, which has been initialized.
|
||||||
|
unsafe { bindings::auxiliary_device_uninit(adev) };
|
||||||
|
|
||||||
|
return Err(Error::from_errno(ret));
|
||||||
|
}
|
||||||
|
|
||||||
|
// SAFETY: `adev` is guaranteed to be non-null, since the `KBox` was allocated
|
||||||
|
// successfully.
|
||||||
|
//
|
||||||
|
// INVARIANT: The device will remain registered until `auxiliary_device_delete()` is
|
||||||
|
// called, which happens in `Self::drop()`.
|
||||||
|
Ok(Devres::new(
|
||||||
|
parent,
|
||||||
|
Self(unsafe { NonNull::new_unchecked(adev) }),
|
||||||
|
))
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -5,7 +5,8 @@
|
|||||||
//! To make this driver probe, QEMU must be run with `-device pci-testdev`.
|
//! To make this driver probe, QEMU must be run with `-device pci-testdev`.
|
||||||
|
|
||||||
use kernel::{
|
use kernel::{
|
||||||
auxiliary, c_str, device::Core, driver, error::Error, pci, prelude::*, InPlaceModule,
|
auxiliary, c_str, device::Core, devres::Devres, driver, error::Error, pci, prelude::*,
|
||||||
|
InPlaceModule,
|
||||||
};
|
};
|
||||||
|
|
||||||
use pin_init::PinInit;
|
use pin_init::PinInit;
|
||||||
@@ -40,8 +41,12 @@ impl auxiliary::Driver for AuxiliaryDriver {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[pin_data]
|
||||||
struct ParentDriver {
|
struct ParentDriver {
|
||||||
_reg: [auxiliary::Registration; 2],
|
#[pin]
|
||||||
|
_reg0: Devres<auxiliary::Registration>,
|
||||||
|
#[pin]
|
||||||
|
_reg1: Devres<auxiliary::Registration>,
|
||||||
}
|
}
|
||||||
|
|
||||||
kernel::pci_device_table!(
|
kernel::pci_device_table!(
|
||||||
@@ -57,11 +62,9 @@ impl pci::Driver for ParentDriver {
|
|||||||
const ID_TABLE: pci::IdTable<Self::IdInfo> = &PCI_TABLE;
|
const ID_TABLE: pci::IdTable<Self::IdInfo> = &PCI_TABLE;
|
||||||
|
|
||||||
fn probe(pdev: &pci::Device<Core>, _info: &Self::IdInfo) -> impl PinInit<Self, Error> {
|
fn probe(pdev: &pci::Device<Core>, _info: &Self::IdInfo) -> impl PinInit<Self, Error> {
|
||||||
Ok(Self {
|
try_pin_init!(Self {
|
||||||
_reg: [
|
_reg0 <- auxiliary::Registration::new(pdev.as_ref(), AUXILIARY_NAME, 0, MODULE_NAME),
|
||||||
auxiliary::Registration::new(pdev.as_ref(), AUXILIARY_NAME, 0, MODULE_NAME)?,
|
_reg1 <- auxiliary::Registration::new(pdev.as_ref(), AUXILIARY_NAME, 1, MODULE_NAME),
|
||||||
auxiliary::Registration::new(pdev.as_ref(), AUXILIARY_NAME, 1, MODULE_NAME)?,
|
|
||||||
],
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user