// SPDX-License-Identifier: GPL-2.0 //! Rust auxiliary driver sample (based on a PCI driver for QEMU's `pci-testdev`). //! //! To make this driver probe, QEMU must be run with `-device pci-testdev`. use kernel::{ auxiliary, device::{ Bound, Core, // }, driver, pci, prelude::*, types::ForLt, InPlaceModule, // }; const MODULE_NAME: &CStr = ::NAME; const AUXILIARY_NAME: &CStr = c"auxiliary"; struct AuxiliaryDriver; kernel::auxiliary_device_table!( AUX_TABLE, MODULE_AUX_TABLE, ::IdInfo, [(auxiliary::DeviceId::new(MODULE_NAME, AUXILIARY_NAME), ())] ); impl auxiliary::Driver for AuxiliaryDriver { type IdInfo = (); type Data<'bound> = Self; const ID_TABLE: auxiliary::IdTable = &AUX_TABLE; fn probe<'bound>( adev: &'bound auxiliary::Device>, _info: &'bound Self::IdInfo, ) -> impl PinInit + 'bound { dev_info!( adev, "Probing auxiliary driver for auxiliary device with id={}\n", adev.id() ); ParentDriver::connect(adev)?; Ok(Self) } } struct Data<'bound> { index: u32, parent: &'bound pci::Device, } struct ParentDriver; #[allow(clippy::type_complexity)] struct ParentData<'bound> { _reg0: auxiliary::Registration<'bound, ForLt!(Data<'_>)>, _reg1: auxiliary::Registration<'bound, ForLt!(Data<'_>)>, } kernel::pci_device_table!( PCI_TABLE, MODULE_PCI_TABLE, ::IdInfo, [(pci::DeviceId::from_id(pci::Vendor::REDHAT, 0x5), ())] ); impl pci::Driver for ParentDriver { type IdInfo = (); type Data<'bound> = ParentData<'bound>; const ID_TABLE: pci::IdTable = &PCI_TABLE; fn probe<'bound>( pdev: &'bound pci::Device>, _info: &'bound Self::IdInfo, ) -> impl PinInit, Error> + 'bound { Ok(ParentData { // SAFETY: `ParentData` is the driver's private data, which is dropped when the // device is unbound; i.e. `mem::forget()` is never called on it. _reg0: unsafe { auxiliary::Registration::new_with_lt( pdev.as_ref(), AUXILIARY_NAME, 0, MODULE_NAME, Data { index: 0, parent: pdev, }, )? }, // SAFETY: See `_reg0` above. _reg1: unsafe { auxiliary::Registration::new_with_lt( pdev.as_ref(), AUXILIARY_NAME, 1, MODULE_NAME, Data { index: 1, parent: pdev, }, )? }, }) } } impl ParentDriver { fn connect(adev: &auxiliary::Device) -> Result { let data = adev.registration_data::)>()?; let pdev = data.parent; dev_info!( pdev, "Connect auxiliary {} with parent: VendorID={}, DeviceID={:#x}\n", adev.id(), pdev.vendor_id(), pdev.device_id() ); dev_info!( pdev, "Connected to auxiliary device with index {}.\n", data.index ); Ok(()) } } #[pin_data] struct SampleModule { #[pin] _pci_driver: driver::Registration>, #[pin] _aux_driver: driver::Registration>, } impl InPlaceModule for SampleModule { fn init(module: &'static kernel::ThisModule) -> impl PinInit { try_pin_init!(Self { _pci_driver <- driver::Registration::new(MODULE_NAME, module), _aux_driver <- driver::Registration::new(MODULE_NAME, module), }) } } module! { type: SampleModule, name: "rust_driver_auxiliary", authors: ["Danilo Krummrich"], description: "Rust auxiliary driver", license: "GPL v2", }