mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git
synced 2026-03-03 18:28:01 +01:00
The previous Io<SIZE> type combined both the generic I/O access helpers and MMIO implementation details in a single struct. This coupling prevented reusing the I/O helpers for other backends, such as PCI configuration space. Establish a clean separation between the I/O interface and concrete backends by separating generic I/O helpers from MMIO implementation. Introduce a new trait hierarchy to handle different access capabilities: - IoCapable<T>: A marker trait indicating that a backend supports I/O operations of a certain type (u8, u16, u32, or u64). - Io trait: Defines fallible (try_read8, try_write8, etc.) and infallibile (read8, write8, etc.) I/O methods with runtime bounds checking and compile-time bounds checking. - IoKnownSize trait: The marker trait for types support infallible I/O methods. Move the MMIO-specific logic into a dedicated Mmio<SIZE> type that implements the Io traits. Rename IoRaw to MmioRaw and update consumers to use the new types. Cc: Alexandre Courbot <acourbot@nvidia.com> Cc: Alice Ryhl <aliceryhl@google.com> Cc: Bjorn Helgaas <helgaas@kernel.org> Cc: Gary Guo <gary@garyguo.net> Cc: Danilo Krummrich <dakr@kernel.org> Cc: John Hubbard <jhubbard@nvidia.com> Signed-off-by: Zhi Wang <zhiw@nvidia.com> Reviewed-by: Alice Ryhl <aliceryhl@google.com> Reviewed-by: Alexandre Courbot <acourbot@nvidia.com> Reviewed-by: Gary Guo <gary@garyguo.net> Link: https://patch.msgid.link/20260121202212.4438-3-zhiw@nvidia.com [ Add #[expect(unused)] to define_{read,write}!(). - Danilo ] Signed-off-by: Danilo Krummrich <dakr@kernel.org>
297 lines
8.7 KiB
Rust
297 lines
8.7 KiB
Rust
// SPDX-License-Identifier: GPL-2.0
|
|
|
|
//! Generic memory-mapped IO.
|
|
|
|
use core::ops::Deref;
|
|
|
|
use crate::{
|
|
device::{
|
|
Bound,
|
|
Device, //
|
|
},
|
|
devres::Devres,
|
|
io::{
|
|
self,
|
|
resource::{
|
|
Region,
|
|
Resource, //
|
|
},
|
|
Mmio,
|
|
MmioRaw, //
|
|
},
|
|
prelude::*,
|
|
};
|
|
|
|
/// An IO request for a specific device and resource.
|
|
pub struct IoRequest<'a> {
|
|
device: &'a Device<Bound>,
|
|
resource: &'a Resource,
|
|
}
|
|
|
|
impl<'a> IoRequest<'a> {
|
|
/// Creates a new [`IoRequest`] instance.
|
|
///
|
|
/// # Safety
|
|
///
|
|
/// Callers must ensure that `resource` is valid for `device` during the
|
|
/// lifetime `'a`.
|
|
pub(crate) unsafe fn new(device: &'a Device<Bound>, resource: &'a Resource) -> Self {
|
|
IoRequest { device, resource }
|
|
}
|
|
|
|
/// Maps an [`IoRequest`] where the size is known at compile time.
|
|
///
|
|
/// This uses the [`ioremap()`] C API.
|
|
///
|
|
/// [`ioremap()`]: https://docs.kernel.org/driver-api/device-io.html#getting-access-to-the-device
|
|
///
|
|
/// # Examples
|
|
///
|
|
/// The following example uses a [`kernel::platform::Device`] for
|
|
/// illustration purposes.
|
|
///
|
|
/// ```no_run
|
|
/// use kernel::{
|
|
/// bindings,
|
|
/// device::Core,
|
|
/// of,
|
|
/// platform,
|
|
/// };
|
|
/// struct SampleDriver;
|
|
///
|
|
/// impl platform::Driver for SampleDriver {
|
|
/// # type IdInfo = ();
|
|
///
|
|
/// fn probe(
|
|
/// pdev: &platform::Device<Core>,
|
|
/// info: Option<&Self::IdInfo>,
|
|
/// ) -> impl PinInit<Self, Error> {
|
|
/// let offset = 0; // Some offset.
|
|
///
|
|
/// // If the size is known at compile time, use [`Self::iomap_sized`].
|
|
/// //
|
|
/// // No runtime checks will apply when reading and writing.
|
|
/// let request = pdev.io_request_by_index(0).ok_or(ENODEV)?;
|
|
/// let iomem = request.iomap_sized::<42>();
|
|
/// let iomem = KBox::pin_init(iomem, GFP_KERNEL)?;
|
|
///
|
|
/// let io = iomem.access(pdev.as_ref())?;
|
|
///
|
|
/// // Read and write a 32-bit value at `offset`.
|
|
/// let data = io.read32_relaxed(offset);
|
|
///
|
|
/// io.write32_relaxed(data, offset);
|
|
///
|
|
/// # Ok(SampleDriver)
|
|
/// }
|
|
/// }
|
|
/// ```
|
|
pub fn iomap_sized<const SIZE: usize>(self) -> impl PinInit<Devres<IoMem<SIZE>>, Error> + 'a {
|
|
IoMem::new(self)
|
|
}
|
|
|
|
/// Same as [`Self::iomap_sized`] but with exclusive access to the
|
|
/// underlying region.
|
|
///
|
|
/// This uses the [`ioremap()`] C API.
|
|
///
|
|
/// [`ioremap()`]: https://docs.kernel.org/driver-api/device-io.html#getting-access-to-the-device
|
|
pub fn iomap_exclusive_sized<const SIZE: usize>(
|
|
self,
|
|
) -> impl PinInit<Devres<ExclusiveIoMem<SIZE>>, Error> + 'a {
|
|
ExclusiveIoMem::new(self)
|
|
}
|
|
|
|
/// Maps an [`IoRequest`] where the size is not known at compile time,
|
|
///
|
|
/// This uses the [`ioremap()`] C API.
|
|
///
|
|
/// [`ioremap()`]: https://docs.kernel.org/driver-api/device-io.html#getting-access-to-the-device
|
|
///
|
|
/// # Examples
|
|
///
|
|
/// The following example uses a [`kernel::platform::Device`] for
|
|
/// illustration purposes.
|
|
///
|
|
/// ```no_run
|
|
/// use kernel::{
|
|
/// bindings,
|
|
/// device::Core,
|
|
/// of,
|
|
/// platform,
|
|
/// };
|
|
/// struct SampleDriver;
|
|
///
|
|
/// impl platform::Driver for SampleDriver {
|
|
/// # type IdInfo = ();
|
|
///
|
|
/// fn probe(
|
|
/// pdev: &platform::Device<Core>,
|
|
/// info: Option<&Self::IdInfo>,
|
|
/// ) -> impl PinInit<Self, Error> {
|
|
/// let offset = 0; // Some offset.
|
|
///
|
|
/// // Unlike [`Self::iomap_sized`], here the size of the memory region
|
|
/// // is not known at compile time, so only the `try_read*` and `try_write*`
|
|
/// // family of functions should be used, leading to runtime checks on every
|
|
/// // access.
|
|
/// let request = pdev.io_request_by_index(0).ok_or(ENODEV)?;
|
|
/// let iomem = request.iomap();
|
|
/// let iomem = KBox::pin_init(iomem, GFP_KERNEL)?;
|
|
///
|
|
/// let io = iomem.access(pdev.as_ref())?;
|
|
///
|
|
/// let data = io.try_read32_relaxed(offset)?;
|
|
///
|
|
/// io.try_write32_relaxed(data, offset)?;
|
|
///
|
|
/// # Ok(SampleDriver)
|
|
/// }
|
|
/// }
|
|
/// ```
|
|
pub fn iomap(self) -> impl PinInit<Devres<IoMem<0>>, Error> + 'a {
|
|
Self::iomap_sized::<0>(self)
|
|
}
|
|
|
|
/// Same as [`Self::iomap`] but with exclusive access to the underlying
|
|
/// region.
|
|
pub fn iomap_exclusive(self) -> impl PinInit<Devres<ExclusiveIoMem<0>>, Error> + 'a {
|
|
Self::iomap_exclusive_sized::<0>(self)
|
|
}
|
|
}
|
|
|
|
/// An exclusive memory-mapped IO region.
|
|
///
|
|
/// # Invariants
|
|
///
|
|
/// - [`ExclusiveIoMem`] has exclusive access to the underlying [`IoMem`].
|
|
pub struct ExclusiveIoMem<const SIZE: usize> {
|
|
/// The underlying `IoMem` instance.
|
|
iomem: IoMem<SIZE>,
|
|
|
|
/// The region abstraction. This represents exclusive access to the
|
|
/// range represented by the underlying `iomem`.
|
|
///
|
|
/// This field is needed for ownership of the region.
|
|
_region: Region,
|
|
}
|
|
|
|
impl<const SIZE: usize> ExclusiveIoMem<SIZE> {
|
|
/// Creates a new `ExclusiveIoMem` instance.
|
|
fn ioremap(resource: &Resource) -> Result<Self> {
|
|
let start = resource.start();
|
|
let size = resource.size();
|
|
let name = resource.name().unwrap_or_default();
|
|
|
|
let region = resource
|
|
.request_region(
|
|
start,
|
|
size,
|
|
name.to_cstring()?,
|
|
io::resource::Flags::IORESOURCE_MEM,
|
|
)
|
|
.ok_or(EBUSY)?;
|
|
|
|
let iomem = IoMem::ioremap(resource)?;
|
|
|
|
let iomem = ExclusiveIoMem {
|
|
iomem,
|
|
_region: region,
|
|
};
|
|
|
|
Ok(iomem)
|
|
}
|
|
|
|
/// Creates a new `ExclusiveIoMem` instance from a previously acquired [`IoRequest`].
|
|
pub fn new<'a>(io_request: IoRequest<'a>) -> impl PinInit<Devres<Self>, Error> + 'a {
|
|
let dev = io_request.device;
|
|
let res = io_request.resource;
|
|
|
|
Devres::new(dev, Self::ioremap(res))
|
|
}
|
|
}
|
|
|
|
impl<const SIZE: usize> Deref for ExclusiveIoMem<SIZE> {
|
|
type Target = Mmio<SIZE>;
|
|
|
|
fn deref(&self) -> &Self::Target {
|
|
&self.iomem
|
|
}
|
|
}
|
|
|
|
/// A generic memory-mapped IO region.
|
|
///
|
|
/// Accesses to the underlying region is checked either at compile time, if the
|
|
/// region's size is known at that point, or at runtime otherwise.
|
|
///
|
|
/// # Invariants
|
|
///
|
|
/// [`IoMem`] always holds an [`MmioRaw`] instance that holds a valid pointer to the
|
|
/// start of the I/O memory mapped region.
|
|
pub struct IoMem<const SIZE: usize = 0> {
|
|
io: MmioRaw<SIZE>,
|
|
}
|
|
|
|
impl<const SIZE: usize> IoMem<SIZE> {
|
|
fn ioremap(resource: &Resource) -> Result<Self> {
|
|
// Note: Some ioremap() implementations use types that depend on the CPU
|
|
// word width rather than the bus address width.
|
|
//
|
|
// TODO: Properly address this in the C code to avoid this `try_into`.
|
|
let size = resource.size().try_into()?;
|
|
if size == 0 {
|
|
return Err(EINVAL);
|
|
}
|
|
|
|
let res_start = resource.start();
|
|
|
|
let addr = if resource
|
|
.flags()
|
|
.contains(io::resource::Flags::IORESOURCE_MEM_NONPOSTED)
|
|
{
|
|
// SAFETY:
|
|
// - `res_start` and `size` are read from a presumably valid `struct resource`.
|
|
// - `size` is known not to be zero at this point.
|
|
unsafe { bindings::ioremap_np(res_start, size) }
|
|
} else {
|
|
// SAFETY:
|
|
// - `res_start` and `size` are read from a presumably valid `struct resource`.
|
|
// - `size` is known not to be zero at this point.
|
|
unsafe { bindings::ioremap(res_start, size) }
|
|
};
|
|
|
|
if addr.is_null() {
|
|
return Err(ENOMEM);
|
|
}
|
|
|
|
let io = MmioRaw::new(addr as usize, size)?;
|
|
let io = IoMem { io };
|
|
|
|
Ok(io)
|
|
}
|
|
|
|
/// Creates a new `IoMem` instance from a previously acquired [`IoRequest`].
|
|
pub fn new<'a>(io_request: IoRequest<'a>) -> impl PinInit<Devres<Self>, Error> + 'a {
|
|
let dev = io_request.device;
|
|
let res = io_request.resource;
|
|
|
|
Devres::new(dev, Self::ioremap(res))
|
|
}
|
|
}
|
|
|
|
impl<const SIZE: usize> Drop for IoMem<SIZE> {
|
|
fn drop(&mut self) {
|
|
// SAFETY: Safe as by the invariant of `Io`.
|
|
unsafe { bindings::iounmap(self.io.addr() as *mut c_void) }
|
|
}
|
|
}
|
|
|
|
impl<const SIZE: usize> Deref for IoMem<SIZE> {
|
|
type Target = Mmio<SIZE>;
|
|
|
|
fn deref(&self) -> &Self::Target {
|
|
// SAFETY: Safe as by the invariant of `IoMem`.
|
|
unsafe { Mmio::from_raw(&self.io) }
|
|
}
|
|
}
|