diff options
Diffstat (limited to 'rust/kernel/pci.rs')
| -rw-r--r-- | rust/kernel/pci.rs | 180 | 
1 files changed, 163 insertions, 17 deletions
| diff --git a/rust/kernel/pci.rs b/rust/kernel/pci.rs index 887ee611b553..7fcc5f6022c1 100644 --- a/rust/kernel/pci.rs +++ b/rust/kernel/pci.rs @@ -10,10 +10,11 @@ use crate::{      devres::Devres,      driver,      error::{from_result, to_result, Result}, -    io::Io, -    io::IoRaw, +    io::{Io, IoRaw}, +    irq::{self, IrqRequest},      str::CStr, -    types::{ARef, Opaque}, +    sync::aref::ARef, +    types::Opaque,      ThisModule,  };  use core::{ @@ -23,6 +24,10 @@ use core::{  };  use kernel::prelude::*; +mod id; + +pub use self::id::{Class, ClassMask, Vendor}; +  /// An adapter for the registration of PCI drivers.  pub struct Adapter<T: Driver>(T); @@ -60,7 +65,7 @@ impl<T: Driver + 'static> Adapter<T> {      extern "C" fn probe_callback(          pdev: *mut bindings::pci_dev,          id: *const bindings::pci_device_id, -    ) -> kernel::ffi::c_int { +    ) -> c_int {          // SAFETY: The PCI bus only ever calls the probe callback with a valid pointer to a          // `struct pci_dev`.          // @@ -128,10 +133,11 @@ impl DeviceId {      /// Equivalent to C's `PCI_DEVICE` macro.      /// -    /// Create a new `pci::DeviceId` from a vendor and device ID number. -    pub const fn from_id(vendor: u32, device: u32) -> Self { +    /// Create a new `pci::DeviceId` from a vendor and device ID. +    #[inline] +    pub const fn from_id(vendor: Vendor, device: u32) -> Self {          Self(bindings::pci_device_id { -            vendor, +            vendor: vendor.as_raw() as u32,              device,              subvendor: DeviceId::PCI_ANY_ID,              subdevice: DeviceId::PCI_ANY_ID, @@ -145,6 +151,7 @@ impl DeviceId {      /// Equivalent to C's `PCI_DEVICE_CLASS` macro.      ///      /// Create a new `pci::DeviceId` from a class number and mask. +    #[inline]      pub const fn from_class(class: u32, class_mask: u32) -> Self {          Self(bindings::pci_device_id {              vendor: DeviceId::PCI_ANY_ID, @@ -157,6 +164,29 @@ impl DeviceId {              override_only: 0,          })      } + +    /// Create a new [`DeviceId`] from a class number, mask, and specific vendor. +    /// +    /// This is more targeted than [`DeviceId::from_class`]: in addition to matching by [`Vendor`], +    /// it also matches the PCI [`Class`] (up to the entire 24 bits, depending on the +    /// [`ClassMask`]). +    #[inline] +    pub const fn from_class_and_vendor( +        class: Class, +        class_mask: ClassMask, +        vendor: Vendor, +    ) -> Self { +        Self(bindings::pci_device_id { +            vendor: vendor.as_raw() as u32, +            device: DeviceId::PCI_ANY_ID, +            subvendor: DeviceId::PCI_ANY_ID, +            subdevice: DeviceId::PCI_ANY_ID, +            class: class.as_raw(), +            class_mask: class_mask.as_raw(), +            driver_data: 0, +            override_only: 0, +        }) +    }  }  // SAFETY: `DeviceId` is a `#[repr(transparent)]` wrapper of `pci_device_id` and does not add @@ -206,7 +236,7 @@ macro_rules! pci_device_table {  ///     <MyDriver as pci::Driver>::IdInfo,  ///     [  ///         ( -///             pci::DeviceId::from_id(bindings::PCI_VENDOR_ID_REDHAT, bindings::PCI_ANY_ID as u32), +///             pci::DeviceId::from_id(pci::Vendor::REDHAT, bindings::PCI_ANY_ID as u32),  ///             (),  ///         )  ///     ] @@ -240,11 +270,11 @@ pub trait Driver: Send {      /// PCI driver probe.      /// -    /// Called when a new platform device is added or discovered. -    /// Implementers should attempt to initialize the device here. +    /// Called when a new pci device is added or discovered. Implementers should +    /// attempt to initialize the device here.      fn probe(dev: &Device<device::Core>, id_info: &Self::IdInfo) -> Result<Pin<KBox<Self>>>; -    /// Platform driver unbind. +    /// PCI driver unbind.      ///      /// Called when a [`Device`] is unbound from its bound [`Driver`]. Implementing this callback      /// is optional. @@ -347,7 +377,7 @@ impl<const SIZE: usize> Bar<SIZE> {          // `ioptr` is valid by the safety requirements.          // `num` is valid by the safety requirements.          unsafe { -            bindings::pci_iounmap(pdev.as_raw(), ioptr as *mut kernel::ffi::c_void); +            bindings::pci_iounmap(pdev.as_raw(), ioptr as *mut c_void);              bindings::pci_release_region(pdev.as_raw(), num);          }      } @@ -359,6 +389,7 @@ impl<const SIZE: usize> Bar<SIZE> {  }  impl Bar { +    #[inline]      fn index_is_valid(index: u32) -> bool {          // A `struct pci_dev` owns an array of resources with at most `PCI_NUM_RESOURCES` entries.          index < bindings::PCI_NUM_RESOURCES @@ -381,24 +412,90 @@ impl<const SIZE: usize> Deref for Bar<SIZE> {  }  impl<Ctx: device::DeviceContext> Device<Ctx> { +    #[inline]      fn as_raw(&self) -> *mut bindings::pci_dev {          self.0.get()      }  }  impl Device { -    /// Returns the PCI vendor ID. -    pub fn vendor_id(&self) -> u16 { +    /// Returns the PCI vendor ID as [`Vendor`]. +    /// +    /// # Examples +    /// +    /// ``` +    /// # use kernel::{device::Core, pci::{self, Vendor}, prelude::*}; +    /// fn log_device_info(pdev: &pci::Device<Core>) -> Result { +    ///     // Get an instance of `Vendor`. +    ///     let vendor = pdev.vendor_id(); +    ///     dev_info!( +    ///         pdev.as_ref(), +    ///         "Device: Vendor={}, Device=0x{:x}\n", +    ///         vendor, +    ///         pdev.device_id() +    ///     ); +    ///     Ok(()) +    /// } +    /// ``` +    #[inline] +    pub fn vendor_id(&self) -> Vendor {          // SAFETY: `self.as_raw` is a valid pointer to a `struct pci_dev`. -        unsafe { (*self.as_raw()).vendor } +        let vendor_id = unsafe { (*self.as_raw()).vendor }; +        Vendor::from_raw(vendor_id)      }      /// Returns the PCI device ID. +    #[inline]      pub fn device_id(&self) -> u16 { -        // SAFETY: `self.as_raw` is a valid pointer to a `struct pci_dev`. +        // SAFETY: By its type invariant `self.as_raw` is always a valid pointer to a +        // `struct pci_dev`.          unsafe { (*self.as_raw()).device }      } +    /// Returns the PCI revision ID. +    #[inline] +    pub fn revision_id(&self) -> u8 { +        // SAFETY: By its type invariant `self.as_raw` is always a valid pointer to a +        // `struct pci_dev`. +        unsafe { (*self.as_raw()).revision } +    } + +    /// Returns the PCI bus device/function. +    #[inline] +    pub fn dev_id(&self) -> u16 { +        // SAFETY: By its type invariant `self.as_raw` is always a valid pointer to a +        // `struct pci_dev`. +        unsafe { bindings::pci_dev_id(self.as_raw()) } +    } + +    /// Returns the PCI subsystem vendor ID. +    #[inline] +    pub fn subsystem_vendor_id(&self) -> u16 { +        // SAFETY: By its type invariant `self.as_raw` is always a valid pointer to a +        // `struct pci_dev`. +        unsafe { (*self.as_raw()).subsystem_vendor } +    } + +    /// Returns the PCI subsystem device ID. +    #[inline] +    pub fn subsystem_device_id(&self) -> u16 { +        // SAFETY: By its type invariant `self.as_raw` is always a valid pointer to a +        // `struct pci_dev`. +        unsafe { (*self.as_raw()).subsystem_device } +    } + +    /// Returns the start of the given PCI bar resource. +    pub fn resource_start(&self, bar: u32) -> Result<bindings::resource_size_t> { +        if !Bar::index_is_valid(bar) { +            return Err(EINVAL); +        } + +        // SAFETY: +        // - `bar` is a valid bar number, as guaranteed by the above call to `Bar::index_is_valid`, +        // - by its type invariant `self.as_raw` is always a valid pointer to a `struct pci_dev`. +        Ok(unsafe { bindings::pci_resource_start(self.as_raw(), bar.try_into()?) }) +    } +      /// Returns the size of the given PCI bar resource.      pub fn resource_len(&self, bar: u32) -> Result<bindings::resource_size_t> {          if !Bar::index_is_valid(bar) { @@ -410,6 +507,13 @@ impl Device {          // - by its type invariant `self.as_raw` is always a valid pointer to a `struct pci_dev`.          Ok(unsafe { bindings::pci_resource_len(self.as_raw(), bar.try_into()?) })      } + +    /// Returns the PCI class as a `Class` struct. +    #[inline] +    pub fn pci_class(&self) -> Class { +        // SAFETY: `self.as_raw` is a valid pointer to a `struct pci_dev`. +        Class::from_raw(unsafe { (*self.as_raw()).class }) +    }  }  impl Device<device::Bound> { @@ -431,6 +535,47 @@ impl Device<device::Bound> {      ) -> impl PinInit<Devres<Bar>, Error> + 'a {          self.iomap_region_sized::<0>(bar, name)      } + +    /// Returns an [`IrqRequest`] for the IRQ vector at the given index, if any. +    pub fn irq_vector(&self, index: u32) -> Result<IrqRequest<'_>> { +        // SAFETY: `self.as_raw` returns a valid pointer to a `struct pci_dev`. +        let irq = unsafe { crate::bindings::pci_irq_vector(self.as_raw(), index) }; +        if irq < 0 { +            return Err(crate::error::Error::from_errno(irq)); +        } +        // SAFETY: `irq` is guaranteed to be a valid IRQ number for `&self`. +        Ok(unsafe { IrqRequest::new(self.as_ref(), irq as u32) }) +    } + +    /// Returns a [`kernel::irq::Registration`] for the IRQ vector at the given +    /// index. +    pub fn request_irq<'a, T: crate::irq::Handler + 'static>( +        &'a self, +        index: u32, +        flags: irq::Flags, +        name: &'static CStr, +        handler: impl PinInit<T, Error> + 'a, +    ) -> Result<impl PinInit<irq::Registration<T>, Error> + 'a> { +        let request = self.irq_vector(index)?; + +        Ok(irq::Registration::<T>::new(request, flags, name, handler)) +    } + +    /// Returns a [`kernel::irq::ThreadedRegistration`] for the IRQ vector at +    /// the given index. +    pub fn request_threaded_irq<'a, T: crate::irq::ThreadedHandler + 'static>( +        &'a self, +        index: u32, +        flags: irq::Flags, +        name: &'static CStr, +        handler: impl PinInit<T, Error> + 'a, +    ) -> Result<impl PinInit<irq::ThreadedRegistration<T>, Error> + 'a> { +        let request = self.irq_vector(index)?; + +        Ok(irq::ThreadedRegistration::<T>::new( +            request, flags, name, handler, +        )) +    }  }  impl Device<device::Core> { @@ -441,6 +586,7 @@ impl Device<device::Core> {      }      /// Enable bus-mastering for this device. +    #[inline]      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()) }; @@ -455,7 +601,7 @@ kernel::impl_device_context_into_aref!(Device);  impl crate::dma::Device for Device<device::Core> {}  // SAFETY: Instances of `Device` are always reference-counted. -unsafe impl crate::types::AlwaysRefCounted for Device { +unsafe impl crate::sync::aref::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()) }; | 
