| // SPDX-License-Identifier: GPL-2.0 |
| |
| //! Rust DMA api test (based on QEMU's `pci-testdev`). |
| //! |
| //! To make this driver probe, QEMU must be run with `-device pci-testdev`. |
| |
| use kernel::{ |
| bindings, |
| device::Core, |
| dma::{CoherentAllocation, Device, DmaMask}, |
| pci, |
| prelude::*, |
| types::ARef, |
| }; |
| |
| struct DmaSampleDriver { |
| pdev: ARef<pci::Device>, |
| ca: CoherentAllocation<MyStruct>, |
| } |
| |
| const TEST_VALUES: [(u32, u32); 5] = [ |
| (0xa, 0xb), |
| (0xc, 0xd), |
| (0xe, 0xf), |
| (0xab, 0xba), |
| (0xcd, 0xef), |
| ]; |
| |
| struct MyStruct { |
| h: u32, |
| b: u32, |
| } |
| |
| impl MyStruct { |
| fn new(h: u32, b: u32) -> Self { |
| Self { h, b } |
| } |
| } |
| // SAFETY: All bit patterns are acceptable values for `MyStruct`. |
| unsafe impl kernel::transmute::AsBytes for MyStruct {} |
| // SAFETY: Instances of `MyStruct` have no uninitialized portions. |
| unsafe impl kernel::transmute::FromBytes for MyStruct {} |
| |
| kernel::pci_device_table!( |
| PCI_TABLE, |
| MODULE_PCI_TABLE, |
| <DmaSampleDriver as pci::Driver>::IdInfo, |
| [( |
| pci::DeviceId::from_id(bindings::PCI_VENDOR_ID_REDHAT, 0x5), |
| () |
| )] |
| ); |
| |
| impl pci::Driver for DmaSampleDriver { |
| type IdInfo = (); |
| const ID_TABLE: pci::IdTable<Self::IdInfo> = &PCI_TABLE; |
| |
| fn probe(pdev: &pci::Device<Core>, _info: &Self::IdInfo) -> Result<Pin<KBox<Self>>> { |
| dev_info!(pdev.as_ref(), "Probe DMA test driver.\n"); |
| |
| let mask = DmaMask::new::<64>(); |
| |
| // SAFETY: There are no concurrent calls to DMA allocation and mapping primitives. |
| unsafe { pdev.dma_set_mask_and_coherent(mask)? }; |
| |
| let ca: CoherentAllocation<MyStruct> = |
| CoherentAllocation::alloc_coherent(pdev.as_ref(), TEST_VALUES.len(), GFP_KERNEL)?; |
| |
| for (i, value) in TEST_VALUES.into_iter().enumerate() { |
| kernel::dma_write!(ca[i] = MyStruct::new(value.0, value.1))?; |
| } |
| |
| let drvdata = KBox::new( |
| Self { |
| pdev: pdev.into(), |
| ca, |
| }, |
| GFP_KERNEL, |
| )?; |
| |
| Ok(drvdata.into()) |
| } |
| } |
| |
| impl Drop for DmaSampleDriver { |
| fn drop(&mut self) { |
| dev_info!(self.pdev.as_ref(), "Unload DMA test driver.\n"); |
| |
| for (i, value) in TEST_VALUES.into_iter().enumerate() { |
| let val0 = kernel::dma_read!(self.ca[i].h); |
| let val1 = kernel::dma_read!(self.ca[i].b); |
| assert!(val0.is_ok()); |
| assert!(val1.is_ok()); |
| |
| if let Ok(val0) = val0 { |
| assert_eq!(val0, value.0); |
| } |
| if let Ok(val1) = val1 { |
| assert_eq!(val1, value.1); |
| } |
| } |
| } |
| } |
| |
| kernel::module_pci_driver! { |
| type: DmaSampleDriver, |
| name: "rust_dma", |
| authors: ["Abdiel Janulgue"], |
| description: "Rust DMA test", |
| license: "GPL v2", |
| } |