mirror of
git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-08-05 16:54:27 +00:00
rust: pci: fix unrestricted &mut pci::Device
As by now, pci::Device is implemented as:
#[derive(Clone)]
pub struct Device(ARef<device::Device>);
This may be convenient, but has the implication that drivers can call
device methods that require a mutable reference concurrently at any
point of time.
Instead define pci::Device as
pub struct Device<Ctx: DeviceContext = Normal>(
Opaque<bindings::pci_dev>,
PhantomData<Ctx>,
);
and manually implement the AlwaysRefCounted trait.
With this we can implement methods that should only be called from
bus callbacks (such as probe()) for pci::Device<Core>. Consequently, we
make this type accessible in bus callbacks only.
Arbitrary references taken by the driver are still of type
ARef<pci::Device> and hence don't provide access to methods that are
reserved for bus callbacks.
Fixes: 1bd8b6b2c5
("rust: pci: add basic PCI device / driver abstractions")
Reviewed-by: Benno Lossin <benno.lossin@proton.me>
Signed-off-by: Danilo Krummrich <dakr@kernel.org>
Acked-by: Boqun Feng <boqun.feng@gmail.com>
Link: https://lore.kernel.org/r/20250314160932.100165-4-dakr@kernel.org
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
This commit is contained in:
parent
4d032779ab
commit
7b948a2af6
2 changed files with 91 additions and 53 deletions
|
@ -6,7 +6,7 @@
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
alloc::flags::*,
|
alloc::flags::*,
|
||||||
bindings, container_of, device,
|
bindings, device,
|
||||||
device_id::RawDeviceId,
|
device_id::RawDeviceId,
|
||||||
devres::Devres,
|
devres::Devres,
|
||||||
driver,
|
driver,
|
||||||
|
@ -17,7 +17,11 @@ use crate::{
|
||||||
types::{ARef, ForeignOwnable, Opaque},
|
types::{ARef, ForeignOwnable, Opaque},
|
||||||
ThisModule,
|
ThisModule,
|
||||||
};
|
};
|
||||||
use core::{ops::Deref, ptr::addr_of_mut};
|
use core::{
|
||||||
|
marker::PhantomData,
|
||||||
|
ops::Deref,
|
||||||
|
ptr::{addr_of_mut, NonNull},
|
||||||
|
};
|
||||||
use kernel::prelude::*;
|
use kernel::prelude::*;
|
||||||
|
|
||||||
/// An adapter for the registration of PCI drivers.
|
/// An adapter for the registration of PCI drivers.
|
||||||
|
@ -60,17 +64,16 @@ impl<T: Driver + 'static> Adapter<T> {
|
||||||
) -> kernel::ffi::c_int {
|
) -> kernel::ffi::c_int {
|
||||||
// SAFETY: The PCI bus only ever calls the probe callback with a valid pointer to a
|
// SAFETY: The PCI bus only ever calls the probe callback with a valid pointer to a
|
||||||
// `struct pci_dev`.
|
// `struct pci_dev`.
|
||||||
let dev = unsafe { device::Device::get_device(addr_of_mut!((*pdev).dev)) };
|
//
|
||||||
// SAFETY: `dev` is guaranteed to be embedded in a valid `struct pci_dev` by the call
|
// INVARIANT: `pdev` is valid for the duration of `probe_callback()`.
|
||||||
// above.
|
let pdev = unsafe { &*pdev.cast::<Device<device::Core>>() };
|
||||||
let mut pdev = unsafe { Device::from_dev(dev) };
|
|
||||||
|
|
||||||
// SAFETY: `DeviceId` is a `#[repr(transparent)` wrapper of `struct pci_device_id` and
|
// SAFETY: `DeviceId` is a `#[repr(transparent)` wrapper of `struct pci_device_id` and
|
||||||
// does not add additional invariants, so it's safe to transmute.
|
// does not add additional invariants, so it's safe to transmute.
|
||||||
let id = unsafe { &*id.cast::<DeviceId>() };
|
let id = unsafe { &*id.cast::<DeviceId>() };
|
||||||
let info = T::ID_TABLE.info(id.index());
|
let info = T::ID_TABLE.info(id.index());
|
||||||
|
|
||||||
match T::probe(&mut pdev, info) {
|
match T::probe(pdev, info) {
|
||||||
Ok(data) => {
|
Ok(data) => {
|
||||||
// Let the `struct pci_dev` own a reference of the driver's private data.
|
// Let the `struct pci_dev` own a reference of the driver's private data.
|
||||||
// SAFETY: By the type invariant `pdev.as_raw` returns a valid pointer to a
|
// SAFETY: By the type invariant `pdev.as_raw` returns a valid pointer to a
|
||||||
|
@ -192,7 +195,7 @@ macro_rules! pci_device_table {
|
||||||
/// # Example
|
/// # Example
|
||||||
///
|
///
|
||||||
///```
|
///```
|
||||||
/// # use kernel::{bindings, pci};
|
/// # use kernel::{bindings, device::Core, pci};
|
||||||
///
|
///
|
||||||
/// struct MyDriver;
|
/// struct MyDriver;
|
||||||
///
|
///
|
||||||
|
@ -210,7 +213,7 @@ macro_rules! pci_device_table {
|
||||||
/// const ID_TABLE: pci::IdTable<Self::IdInfo> = &PCI_TABLE;
|
/// const ID_TABLE: pci::IdTable<Self::IdInfo> = &PCI_TABLE;
|
||||||
///
|
///
|
||||||
/// fn probe(
|
/// fn probe(
|
||||||
/// _pdev: &mut pci::Device,
|
/// _pdev: &pci::Device<Core>,
|
||||||
/// _id_info: &Self::IdInfo,
|
/// _id_info: &Self::IdInfo,
|
||||||
/// ) -> Result<Pin<KBox<Self>>> {
|
/// ) -> Result<Pin<KBox<Self>>> {
|
||||||
/// Err(ENODEV)
|
/// Err(ENODEV)
|
||||||
|
@ -234,20 +237,23 @@ pub trait Driver {
|
||||||
///
|
///
|
||||||
/// Called when a new platform device is added or discovered.
|
/// Called when a new platform device is added or discovered.
|
||||||
/// Implementers should attempt to initialize the device here.
|
/// Implementers should attempt to initialize the device here.
|
||||||
fn probe(dev: &mut Device, id_info: &Self::IdInfo) -> Result<Pin<KBox<Self>>>;
|
fn probe(dev: &Device<device::Core>, id_info: &Self::IdInfo) -> Result<Pin<KBox<Self>>>;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// The PCI device representation.
|
/// The PCI device representation.
|
||||||
///
|
///
|
||||||
/// A PCI device is based on an always reference counted `device:Device` instance. Cloning a PCI
|
/// This structure represents the Rust abstraction for a C `struct pci_dev`. The implementation
|
||||||
/// device, hence, also increments the base device' reference count.
|
/// abstracts the usage of an already existing C `struct pci_dev` within Rust code that we get
|
||||||
|
/// passed from the C side.
|
||||||
///
|
///
|
||||||
/// # Invariants
|
/// # Invariants
|
||||||
///
|
///
|
||||||
/// `Device` hold a valid reference of `ARef<device::Device>` whose underlying `struct device` is a
|
/// A [`Device`] instance represents a valid `struct device` created by the C portion of the kernel.
|
||||||
/// member of a `struct pci_dev`.
|
#[repr(transparent)]
|
||||||
#[derive(Clone)]
|
pub struct Device<Ctx: device::DeviceContext = device::Normal>(
|
||||||
pub struct Device(ARef<device::Device>);
|
Opaque<bindings::pci_dev>,
|
||||||
|
PhantomData<Ctx>,
|
||||||
|
);
|
||||||
|
|
||||||
/// A PCI BAR to perform I/O-Operations on.
|
/// A PCI BAR to perform I/O-Operations on.
|
||||||
///
|
///
|
||||||
|
@ -256,13 +262,13 @@ pub struct Device(ARef<device::Device>);
|
||||||
/// `Bar` always holds an `IoRaw` inststance that holds a valid pointer to the start of the I/O
|
/// `Bar` always holds an `IoRaw` inststance that holds a valid pointer to the start of the I/O
|
||||||
/// memory mapped PCI bar and its size.
|
/// memory mapped PCI bar and its size.
|
||||||
pub struct Bar<const SIZE: usize = 0> {
|
pub struct Bar<const SIZE: usize = 0> {
|
||||||
pdev: Device,
|
pdev: ARef<Device>,
|
||||||
io: IoRaw<SIZE>,
|
io: IoRaw<SIZE>,
|
||||||
num: i32,
|
num: i32,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<const SIZE: usize> Bar<SIZE> {
|
impl<const SIZE: usize> Bar<SIZE> {
|
||||||
fn new(pdev: Device, num: u32, name: &CStr) -> Result<Self> {
|
fn new(pdev: &Device, num: u32, name: &CStr) -> Result<Self> {
|
||||||
let len = pdev.resource_len(num)?;
|
let len = pdev.resource_len(num)?;
|
||||||
if len == 0 {
|
if len == 0 {
|
||||||
return Err(ENOMEM);
|
return Err(ENOMEM);
|
||||||
|
@ -300,12 +306,16 @@ impl<const SIZE: usize> Bar<SIZE> {
|
||||||
// `pdev` is valid by the invariants of `Device`.
|
// `pdev` is valid by the invariants of `Device`.
|
||||||
// `ioptr` is guaranteed to be the start of a valid I/O mapped memory region.
|
// `ioptr` is guaranteed to be the start of a valid I/O mapped memory region.
|
||||||
// `num` is checked for validity by a previous call to `Device::resource_len`.
|
// `num` is checked for validity by a previous call to `Device::resource_len`.
|
||||||
unsafe { Self::do_release(&pdev, ioptr, num) };
|
unsafe { Self::do_release(pdev, ioptr, num) };
|
||||||
return Err(err);
|
return Err(err);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
Ok(Bar { pdev, io, num })
|
Ok(Bar {
|
||||||
|
pdev: pdev.into(),
|
||||||
|
io,
|
||||||
|
num,
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
/// # Safety
|
/// # Safety
|
||||||
|
@ -351,20 +361,8 @@ impl<const SIZE: usize> Deref for Bar<SIZE> {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Device {
|
impl Device {
|
||||||
/// Create a PCI Device instance from an existing `device::Device`.
|
|
||||||
///
|
|
||||||
/// # Safety
|
|
||||||
///
|
|
||||||
/// `dev` must be an `ARef<device::Device>` whose underlying `bindings::device` is a member of
|
|
||||||
/// a `bindings::pci_dev`.
|
|
||||||
pub unsafe fn from_dev(dev: ARef<device::Device>) -> Self {
|
|
||||||
Self(dev)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn as_raw(&self) -> *mut bindings::pci_dev {
|
fn as_raw(&self) -> *mut bindings::pci_dev {
|
||||||
// SAFETY: By the type invariant `self.0.as_raw` is a pointer to the `struct device`
|
self.0.get()
|
||||||
// embedded in `struct pci_dev`.
|
|
||||||
unsafe { container_of!(self.0.as_raw(), bindings::pci_dev, dev) as _ }
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Returns the PCI vendor ID.
|
/// Returns the PCI vendor ID.
|
||||||
|
@ -379,18 +377,6 @@ impl Device {
|
||||||
unsafe { (*self.as_raw()).device }
|
unsafe { (*self.as_raw()).device }
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Enable memory resources for this device.
|
|
||||||
pub fn enable_device_mem(&self) -> Result {
|
|
||||||
// SAFETY: `self.as_raw` is guaranteed to be a pointer to a valid `struct pci_dev`.
|
|
||||||
to_result(unsafe { bindings::pci_enable_device_mem(self.as_raw()) })
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Enable bus-mastering for this device.
|
|
||||||
pub fn set_master(&self) {
|
|
||||||
// SAFETY: `self.as_raw` is guaranteed to be a pointer to a valid `struct pci_dev`.
|
|
||||||
unsafe { bindings::pci_set_master(self.as_raw()) };
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Returns the size of the given PCI bar resource.
|
/// Returns the size of the given PCI bar resource.
|
||||||
pub fn resource_len(&self, bar: u32) -> Result<bindings::resource_size_t> {
|
pub fn resource_len(&self, bar: u32) -> Result<bindings::resource_size_t> {
|
||||||
if !Bar::index_is_valid(bar) {
|
if !Bar::index_is_valid(bar) {
|
||||||
|
@ -410,7 +396,7 @@ impl Device {
|
||||||
bar: u32,
|
bar: u32,
|
||||||
name: &CStr,
|
name: &CStr,
|
||||||
) -> Result<Devres<Bar<SIZE>>> {
|
) -> Result<Devres<Bar<SIZE>>> {
|
||||||
let bar = Bar::<SIZE>::new(self.clone(), bar, name)?;
|
let bar = Bar::<SIZE>::new(self, bar, name)?;
|
||||||
let devres = Devres::new(self.as_ref(), bar, GFP_KERNEL)?;
|
let devres = Devres::new(self.as_ref(), bar, GFP_KERNEL)?;
|
||||||
|
|
||||||
Ok(devres)
|
Ok(devres)
|
||||||
|
@ -422,8 +408,60 @@ impl Device {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl AsRef<device::Device> for Device {
|
impl Device<device::Core> {
|
||||||
fn as_ref(&self) -> &device::Device {
|
/// Enable memory resources for this device.
|
||||||
&self.0
|
pub fn enable_device_mem(&self) -> Result {
|
||||||
|
// SAFETY: `self.as_raw` is guaranteed to be a pointer to a valid `struct pci_dev`.
|
||||||
|
to_result(unsafe { bindings::pci_enable_device_mem(self.as_raw()) })
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Enable bus-mastering for this device.
|
||||||
|
pub fn set_master(&self) {
|
||||||
|
// SAFETY: `self.as_raw` is guaranteed to be a pointer to a valid `struct pci_dev`.
|
||||||
|
unsafe { bindings::pci_set_master(self.as_raw()) };
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Deref for Device<device::Core> {
|
||||||
|
type Target = Device;
|
||||||
|
|
||||||
|
fn deref(&self) -> &Self::Target {
|
||||||
|
let ptr: *const Self = self;
|
||||||
|
|
||||||
|
// CAST: `Device<Ctx>` is a transparent wrapper of `Opaque<bindings::pci_dev>`.
|
||||||
|
let ptr = ptr.cast::<Device>();
|
||||||
|
|
||||||
|
// SAFETY: `ptr` was derived from `&self`.
|
||||||
|
unsafe { &*ptr }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<&Device<device::Core>> for ARef<Device> {
|
||||||
|
fn from(dev: &Device<device::Core>) -> Self {
|
||||||
|
(&**dev).into()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// SAFETY: Instances of `Device` are always reference-counted.
|
||||||
|
unsafe impl crate::types::AlwaysRefCounted for Device {
|
||||||
|
fn inc_ref(&self) {
|
||||||
|
// SAFETY: The existence of a shared reference guarantees that the refcount is non-zero.
|
||||||
|
unsafe { bindings::pci_dev_get(self.as_raw()) };
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe fn dec_ref(obj: NonNull<Self>) {
|
||||||
|
// SAFETY: The safety requirements guarantee that the refcount is non-zero.
|
||||||
|
unsafe { bindings::pci_dev_put(obj.cast().as_ptr()) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl AsRef<device::Device> for Device {
|
||||||
|
fn as_ref(&self) -> &device::Device {
|
||||||
|
// SAFETY: By the type invariant of `Self`, `self.as_raw()` is a pointer to a valid
|
||||||
|
// `struct pci_dev`.
|
||||||
|
let dev = unsafe { addr_of_mut!((*self.as_raw()).dev) };
|
||||||
|
|
||||||
|
// SAFETY: `dev` points to a valid `struct device`.
|
||||||
|
unsafe { device::Device::as_ref(dev) }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,7 +4,7 @@
|
||||||
//!
|
//!
|
||||||
//! 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::{bindings, c_str, devres::Devres, pci, prelude::*};
|
use kernel::{bindings, c_str, device::Core, devres::Devres, pci, prelude::*, types::ARef};
|
||||||
|
|
||||||
struct Regs;
|
struct Regs;
|
||||||
|
|
||||||
|
@ -26,7 +26,7 @@ impl TestIndex {
|
||||||
}
|
}
|
||||||
|
|
||||||
struct SampleDriver {
|
struct SampleDriver {
|
||||||
pdev: pci::Device,
|
pdev: ARef<pci::Device>,
|
||||||
bar: Devres<Bar0>,
|
bar: Devres<Bar0>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -62,7 +62,7 @@ impl pci::Driver for SampleDriver {
|
||||||
|
|
||||||
const ID_TABLE: pci::IdTable<Self::IdInfo> = &PCI_TABLE;
|
const ID_TABLE: pci::IdTable<Self::IdInfo> = &PCI_TABLE;
|
||||||
|
|
||||||
fn probe(pdev: &mut pci::Device, info: &Self::IdInfo) -> Result<Pin<KBox<Self>>> {
|
fn probe(pdev: &pci::Device<Core>, info: &Self::IdInfo) -> Result<Pin<KBox<Self>>> {
|
||||||
dev_dbg!(
|
dev_dbg!(
|
||||||
pdev.as_ref(),
|
pdev.as_ref(),
|
||||||
"Probe Rust PCI driver sample (PCI ID: 0x{:x}, 0x{:x}).\n",
|
"Probe Rust PCI driver sample (PCI ID: 0x{:x}, 0x{:x}).\n",
|
||||||
|
@ -77,7 +77,7 @@ impl pci::Driver for SampleDriver {
|
||||||
|
|
||||||
let drvdata = KBox::new(
|
let drvdata = KBox::new(
|
||||||
Self {
|
Self {
|
||||||
pdev: pdev.clone(),
|
pdev: pdev.into(),
|
||||||
bar,
|
bar,
|
||||||
},
|
},
|
||||||
GFP_KERNEL,
|
GFP_KERNEL,
|
||||||
|
|
Loading…
Add table
Reference in a new issue