2025-03-16
DMA(直接内存访问,Direct Memory Access)是一种允许硬件设备直接访问系统内存的技术,无需通过CPU的干预。它主要用于提高数据传输效率,减少CPU的负担。
初始化:CPU配置DMA控制器,指定数据传输的源地址、目标地址和数据量。
启动传输:DMA控制器接管,直接在设备和内存之间传输数据。
完成通知:传输完成后,DMA控制器通知CPU。
raspberry pi3b
Linux raspberrypi 6.1.21-v8+ #1642 SMP PREEMPT Mon Apr 3 17:24:16 BST 2023 aarch64 GNU/Linux
// simple.c
#include <linux/completion.h>
#include <linux/dma-mapping.h>
#include <linux/dmaengine.h>
#include <linux/init.h>
#include <linux/module.h>
#include <linux/slab.h>
static void _dma_transfer_completed(void *param) {
struct completion *cmp = (struct completion *)param;
complete(cmp);
}
static int __init simple_init(void) {
dma_cap_mask_t mask;
struct dma_chan *chan;
struct dma_async_tx_descriptor *chan_desc;
dma_cookie_t cookie;
dma_addr_t src_addr, dst_addr;
unsigned char *src_buf, *dst_buf;
struct completion cmp;
int status;
pr_info("simple_init\n");
dma_cap_zero(mask);
dma_cap_set(DMA_SLAVE | DMA_PRIVATE, mask);
chan = dma_request_channel(mask, NULL, NULL);
if (!chan) {
pr_err("dma_request_channel error\n");
return -ENODEV;
}
src_buf =
dma_alloc_coherent(chan->device->dev, 1024, &src_addr, GFP_KERNEL);
dst_buf =
dma_alloc_coherent(chan->device->dev, 1024, &dst_addr, GFP_KERNEL);
memset(src_buf, 0, 1024);
memset(dst_buf, 1, 1024);
pr_info("before src_buf[0] = %d, dst_buf[0] = %d\n", src_buf[0],
dst_buf[0]);
chan_desc = dmaengine_prep_dma_memcpy(chan, dst_addr, src_addr, 1024,
DMA_MEM_TO_MEM);
if (!chan_desc) {
pr_err("dmaengine_prep_dma_memcpy error\n");
status = -1;
goto free;
}
init_completion(&cmp);
chan_desc->callback = _dma_transfer_completed;
chan_desc->callback_param = &cmp;
cookie = dmaengine_submit(chan_desc);
dma_async_issue_pending(chan);
if (wait_for_completion_timeout(&cmp, msecs_to_jiffies(3000)) < 0) {
pr_err("completion timeout 3000ms\n");
status = -1;
goto rel;
}
status = dma_async_is_tx_complete(chan, cookie, NULL, NULL);
if (status == DMA_COMPLETE) {
status = 0;
pr_info("DMA transfer has completed\n");
pr_info("before src_buf[0] = %d, dst_buf[0] = %d\n", src_buf[0],
dst_buf[0]);
} else {
pr_err("dma_async_is_tx_complete error\n");
}
rel:
dmaengine_terminate_all(chan);
free:
dma_free_coherent(chan->device->dev, 1024, src_buf, src_addr);
dma_free_coherent(chan->device->dev, 1024, dst_buf, dst_addr);
dma_release_channel(chan);
return status;
}
static void __exit simple_exit(void) { pr_info("simple_exit\n"); }
module_init(simple_init);
module_exit(simple_exit);
MODULE_LICENSE("GPL");
MODULE_AUTHOR("heng30");
MODULE_VERSION("v0.0.1");
MODULE_DESCRIPTION("A simple kernel module");
#!/bin/sh
top-dir = $(shell pwd)
kernel-version = $(shell uname -r)
kernel-dir ?= /lib/modules/$(kernel-version)/build
obj-m += simple.o
all:
make -C $(kernel-dir) modules M=$(top-dir)
clean:
rm -f *.o *.ko *.mod *.mod.c *.order *.symvers *.dtbo
make -C $(kernel-dir) clean m=$(top-dir)
编译程序:make
安装驱动:insmod simple.ko
查看输出:dmesg
[194177.368278] simple_init
[194177.369413] before src_buf[0] = 0, dst_buf[0] = 1
[194177.396623] DMA transfer has completed
[194177.396853] before src_buf[0] = 0, dst_buf[0] = 0
[194197.014928] simple_exit
rmmod simple.ko