On Wed, Jun 19, 2024 at 01:39:48AM +0200, Danilo Krummrich wrote: > Implement the generic `Registration` type and the `DriverOps` trait. I don't think this is needed, more below... > The `Registration` structure is the common type that represents a driver > registration and is typically bound to the lifetime of a module. However, > it doesn't implement actual calls to the kernel's driver core to register > drivers itself. But that's not what normally happens, more below... > Instead the `DriverOps` trait is provided to subsystems, which have to > implement `DriverOps::register` and `DrvierOps::unregister`. Subsystems > have to provide an implementation for both of those methods where the > subsystem specific variants to register / unregister a driver have to > implemented. So you are saying this should be something that a "bus" would do? Please be explicit as to what you mean by "subsystem" here. > For instance, the PCI subsystem would call __pci_register_driver() from > `DriverOps::register` and pci_unregister_driver() from > `DrvierOps::unregister`. So this is a BusOps, or more in general, a "subsystem" if it's not a bus (i.e. it's a class). Note, we used to use the term "subsystem" a very long time ago but got rid of them in the driver core, let's not bring it back unless we REALLY know we want it this time. So why isn't this just a BusOps? > This patch is based on previous work from Wedson Almeida Filho. > > Co-developed-by: Wedson Almeida Filho <wedsonaf@xxxxxxxxx> > Signed-off-by: Wedson Almeida Filho <wedsonaf@xxxxxxxxx> > Signed-off-by: Danilo Krummrich <dakr@xxxxxxxxxx> > --- > rust/kernel/driver.rs | 128 ++++++++++++++++++++++++++++++++++++++++++ > rust/kernel/lib.rs | 1 + > 2 files changed, 129 insertions(+) > create mode 100644 rust/kernel/driver.rs > > diff --git a/rust/kernel/driver.rs b/rust/kernel/driver.rs > new file mode 100644 > index 000000000000..e04406b93b56 > --- /dev/null > +++ b/rust/kernel/driver.rs > @@ -0,0 +1,128 @@ > +// SPDX-License-Identifier: GPL-2.0 > + > +//! Generic support for drivers of different buses (e.g., PCI, Platform, Amba, etc.). See, you think it's a bus, let's call it a bus! :) > +//! > +//! Each bus / subsystem is expected to implement [`DriverOps`], which allows drivers to register > +//! using the [`Registration`] class. > + > +use crate::error::{Error, Result}; > +use crate::{init::PinInit, str::CStr, try_pin_init, types::Opaque, ThisModule}; > +use core::pin::Pin; > +use macros::{pin_data, pinned_drop}; > + > +/// The [`DriverOps`] trait serves as generic interface for subsystems (e.g., PCI, Platform, Amba, > +/// etc.) to privide the corresponding subsystem specific implementation to register / unregister a > +/// driver of the particular type (`RegType`). > +/// > +/// For instance, the PCI subsystem would set `RegType` to `bindings::pci_driver` and call > +/// `bindings::__pci_register_driver` from `DriverOps::register` and > +/// `bindings::pci_unregister_driver` from `DriverOps::unregister`. > +pub trait DriverOps { > + /// The type that holds information about the registration. This is typically a struct defined > + /// by the C portion of the kernel. > + type RegType: Default; > + > + /// Registers a driver. > + /// > + /// # Safety > + /// > + /// `reg` must point to valid, initialised, and writable memory. It may be modified by this > + /// function to hold registration state. > + /// > + /// On success, `reg` must remain pinned and valid until the matching call to > + /// [`DriverOps::unregister`]. > + fn register( > + reg: &mut Self::RegType, > + name: &'static CStr, > + module: &'static ThisModule, > + ) -> Result; > + > + /// Unregisters a driver previously registered with [`DriverOps::register`]. > + /// > + /// # Safety > + /// > + /// `reg` must point to valid writable memory, initialised by a previous successful call to > + /// [`DriverOps::register`]. > + fn unregister(reg: &mut Self::RegType); > +} So you are getting into what a bus wants/needs to support here, why is register/unregister the "big" things to be implemented first? Why not just map the current register/unregister bus functions to a bus-specific trait for now? And then, if you think it really should be generic, we can make it that way then. I don't see why this needs to be generic now as you aren't implementing a bus in rust at this point in time, right? > + > +/// A [`Registration`] is a generic type that represents the registration of some driver type (e.g. > +/// `bindings::pci_driver`). Therefore a [`Registration`] is initialized with some type that > +/// implements the [`DriverOps`] trait, such that the generic `T::register` and `T::unregister` > +/// calls result in the subsystem specific registration calls. > +/// > +///Once the `Registration` structure is dropped, the driver is unregistered. > +#[pin_data(PinnedDrop)] > +pub struct Registration<T: DriverOps> { > + #[pin] > + reg: Opaque<T::RegType>, > +} > + > +// SAFETY: `Registration` has no fields or methods accessible via `&Registration`, so it is safe to > +// share references to it with multiple threads as nothing can be done. > +unsafe impl<T: DriverOps> Sync for Registration<T> {} > + > +// SAFETY: Both registration and unregistration are implemented in C and safe to be performed from > +// any thread, so `Registration` is `Send`. > +unsafe impl<T: DriverOps> Send for Registration<T> {} > + > +impl<T: DriverOps> Registration<T> { > + /// Creates a new instance of the registration object. > + pub fn new(name: &'static CStr, module: &'static ThisModule) -> impl PinInit<Self, Error> { Drivers have modules, not busses. So you are registering a driver with a bus here, it's not something that a driver itself implements as you have named here. > + try_pin_init!(Self { > + reg <- Opaque::try_ffi_init(|ptr: *mut T::RegType| { > + // SAFETY: `try_ffi_init` guarantees that `ptr` is valid for write. > + unsafe { ptr.write(T::RegType::default()) }; > + > + // SAFETY: `try_ffi_init` guarantees that `ptr` is valid for write, and it has > + // just been initialised above, so it's also valid for read. > + let drv = unsafe { &mut *ptr }; > + > + T::register(drv, name, module) > + }), > + }) > + } > +} > + > +#[pinned_drop] > +impl<T: DriverOps> PinnedDrop for Registration<T> { > + fn drop(self: Pin<&mut Self>) { > + let drv = unsafe { &mut *self.reg.get() }; > + > + T::unregister(drv); > + } > +} > + > +/// A kernel module that only registers the given driver on init. > +/// > +/// This is a helper struct to make it easier to define single-functionality modules, in this case, > +/// modules that offer a single driver. > +#[pin_data] > +pub struct Module<T: DriverOps> { > + #[pin] > + _driver: Registration<T>, > +} While these are nice, let's not add them just yet, let's keep it simple for now until we work out the proper model and see what is, and is not, common for drivers to do. That way we keep the review simpler as well, and saves you time rewriting things as we rename / modify stuff. > --- a/rust/kernel/lib.rs > +++ b/rust/kernel/lib.rs > @@ -29,6 +29,7 @@ > pub mod alloc; > mod build_assert; > pub mod device; > +pub mod driver; Nope, this is a bus :) thanks, greg k-h