Linux内核设备驱动程序将DMA转换为内核空间

Ian*_*han 8 linux linux-device-driver linux-kernel dma

LDD3(p:453)演示dma_map_single使用作为参数传入的缓冲区.

bus_addr = dma_map_single(&dev->pci_dev->dev, buffer, count, dev->dma_dir);
Run Code Online (Sandbox Code Playgroud)

Q1:这个缓冲区来自哪里/哪里?

kmalloc

Q2:为什么DMA-API-HOWTO.txt状态我可以使用raw kmalloc进行DMA转换?

表格http://www.mjmwired.net/kernel/Documentation/DMA-API-HOWTO.txt

L:51如果您通过页面分配器kmalloc()获取了内存,那么您可以使用从这些例程返回的地址与该内存进行DMA.

L:74你不能接受kmap()调用和DMA的返回.

  1. 所以我可以将返回的地址传递kmalloc给我的硬件设备?
  2. 或者我应该virt_to_bus先运行它?
  3. 或者我应该把它传递给dma_map_single

问题3:DMA传输完成后,我可以通过kmalloc地址读取内核驱动程序中的数据吗?

addr = kmalloc(...);
...
printk("test result : 0x%08x\n", addr[0]);
Run Code Online (Sandbox Code Playgroud)

问题4:将这个用户空间送到用户空间的最佳方式是什么?

  1. copy_to_user
  2. mmap kmalloc内存?
  3. 其他?

gby*_*gby 16

  1. kmalloc确实是获取缓冲区的一个来源.另一个可以是带有GFP_DMA标志的alloc_page.

  2. 意思是kmalloc返回的内存保证在物理内存中连续,而不仅仅是虚拟内存,因此您可以将该指针的总线地址提供给您的硬件.你需要在返回的地址上使用dma_map_single(),这取决于确切的平台可能不再是virt_to_bus的包装,或者可以执行更多操作(设置IOMMU或GART表)

  3. 正确,只需确保遵循缓存一致性指南,如DMA指南所述.

  4. copy_to_user将正常工作,是最简单的答案.根据您的具体情况,它可能就足够了,或者您可能需要具有更好性能的东西.您不能正常地将kmalloced地址映射到用户空间,但您可以DMA到用户提供的地址(一些警告适用)或分配用户页面(alloc_page with GFP_USER)

祝好运!