dmaengine 框架-凯发app官方网站

凯发app官方网站-凯发k8官网下载客户端中心 | | 凯发app官方网站-凯发k8官网下载客户端中心
  • 博客访问: 263871
  • 博文数量: 90
  • 博客积分: 0
  • 博客等级: 民兵
  • 技术积分: 665
  • 用 户 组: 普通用户
  • 注册时间: 2018-10-15 14:13
个人简介

搭建一个和linux开发者知识共享和学习的平台

文章分类

全部博文(90)

文章存档

2024年(4)

2023年(24)

2022年(27)

2019年(8)

2018年(27)

相关博文
  • ·
  • ·
  • ·
  • ·
  • ·
  • ·
  • ·
  • ·
  • ·
  • ·

分类: linux

2023-05-31 11:12:49

dmaengine 框架
2.1.基本概述
dmaengine 是 linux 内核 dma 驱动框架,针对 dma 驱动的混乱局面内核社区提出了一个全新的框架驱动,目标在统一 dma api 让各个模块使用 dma 时不用关心硬件细节,同时代码复用提高。并且实现异步的数据传输,降低机器负载。


2.1.1.术语约定
? dma: direct memory access(直接内存存取)
? channel: dma 通道
? slave: 从通道,一般指设备通道
? master: 主通道,一般指内存


2.1.2.功能简介
dmaengine 向使用者提供统一的接口,不同的模式下使用不同的 dma 接口,省去使用过多的关注硬件接口。

2.2.基本结构

 

2.3.模式
2.3.1. 内存拷贝
纯粹地内存拷贝,即从指定的源地址拷贝到指定的目的地址。传输完毕会发生一个中断,并调用回函数。

2.3.2. 散列表
散列模式是把不连续的内存块直接传输到指定的目的地址。当传输完毕会发生一个中断,并调用回调函数

 

上述的散列拷贝操作是针对于 slave 设备而言的,它支持的是 slave 与 master 之间的拷贝,还有另一散列拷贝是专门对内存进行操作的,即 master 与 master 之间进行操作,具体形式图如下:

2.3.3. 循环缓存
循环模式就是把一块 ring buffer 切成若干片,周而复始的传输,每传完一个片会发生一个中断,同时调用回调函数

 

3. 接口介绍
3.1.通道相关

struct dma_chan *dma_request_channel(dma_cap_mask_t *mask,
dma_filter_fn fn, void *fn_param);

功能:申请一个可用通道
参数:mask 所有申请的传输类型的掩码。
fn dma 驱动私有的过滤函数。
fn_param 传入的私有参数。
返回:返回一个通道数据指针。

void dma_release_channel(struct dma_chan *chan)

功能:释放一个通道
参数:所需要释放的通道指针

3.2.配置相关
int dmaengine_slave_config(struct dma_chan *chan, struct dma_slave_config *config)
功能:配置一个从通道传输
参数:chan 通道结构指针
config 配置数据指针
返回:非零表示失败,零表示成功

struct dma_slave_config {
enum dma_transfer_direction direction;
dma_addr_t src_addr;
dma_addr_t dst_addr;
enum dma_slave_buswidth src_addr_width;
enum dma_slave_buswidth dst_addr_width;
u32 src_maxburst;
u32 dst_maxburst;
bool device_fc;
unsigned int slave_id;
};

direction: 传输方向,取值 mem_to_dev dev_to_mem mem_to_mem dev_to_dev
src_addr: 源地址,必须是物理地址;
dst_addr: 目的地址,必须是物理地址;
src_addr_width: 源数据宽度,byte 整数倍,取值 1,2,4,8;
dst_addr_width: 目的数据宽度,取值同上;
src_max_burst: 源突发长度,取值 1,4,8;
dst_max_burst: 目的突发长度,取值同上;

slave_id: 从通道 id 号,此处用作 drq 的设置,使用 sunxi_slave_id(d, s)宏设置,具体取值参照
include/linux/sunxi-dma.h 里使用;

3.3.传输相关
struct dma_async_tx_descriptor *dmaengine_prep_slave_single(struct dma_chan *chan,
dma_addr_t buf, size_t len, enum dma_transfer_direction dir,
unsigned long flags)
功能:准备一次单包传输
参数:chan 通道指针;
buf 需要传输地址;
dir 传输方向,此处为 dma_mem_to_dev,dma_dev_to_mem
flags 传输标志;
返回:返回一个传输描述符指针。

struct dma_async_tx_descriptor *dmaengine_prep_slave_sg(struct dma_chan *chan,
struct scatterlist *sgl, unsigned int sg_len,
enum dma_transfer_direction dir, unsigned long flags)

功能:准备一次多包传输,散列形式,(slave 模式)
参数:chan 通道指针;
sgl 散列表地址,此散列表传输之前需要建立;
sg_len 散列表内 buffer 的个数;
dir 传输方向,此处为 dma_mem_to_dev,dma_dev_to_mem
flags 传输标志;
返回:返回一个传输描述符指针

struct dma_async_tx_descriptor * (*device_prep_dma_sg)(
struct dma_chan *chan,
struct scatterlist *dst_sg, unsigned int dst_nents,
struct scatterlist *src_sg, unsigned int src_nents,
unsigned long flags);

功能:准备一次多包传输,散列形式,(master 模式)
参数:chan 通道指针;
dst_sg 目的散列表地址,此散列表传输之前需要建立;
dst_nents 散列表内 buffer 的个数;
src_sg 源散列表地址,此散列表传输之前需要建立;
src_nents 散列表内 buffer 的个数;
flags 传输标志;
返回:返回一个传输描述符指针

struct dma_async_tx_descriptor *dmaengine_prep_dma_cyclic(struct dma_chan *chan,
dma_addr_t buf_addr, size_t buf_len, size_t period_len,
enum dma_transfer_direction dir, unsigned long flags)

功能:准备一次环形 buffer 传输。
参数:chan 通道指针;
buf_addr 环形 buffer 起始地址,必须为物理地址;

buf_len 环形 buffer 的长度;
period_len 每一小片 buffer 的长度;
dir 传输方向,此处为 dma_mem_to_dev,dma_dev_to_mem
flags 传输标志;
返回:返回一个传输描述符指针

struct dma_async_tx_descriptor {
dma_cookie_t cookie;
enum dma_ctrl_flags flags; /* not a 'long' to pack with cookie */
dma_addr_t phys;
struct dma_chan *chan;
dma_cookie_t (*tx_submit)(struct dma_async_tx_descriptor *tx);
dma_async_tx_callback callback;
void *callback_param;
};

cookie 本次传输的 cookie,在此通道上唯一;
tx_submit 本次传输的提交执行函数;
callback 传输完成后的回调函数;
callback_param 回调函数的参数;

dma_cookie_t dmaengine_submit(struct dma_async_tx_descriptor *desc)

功能: 提交已经做好准备的传输。
参数:desc 传输描述符,由前面准备函数获得。
返回:返回一个 cookie,小于零为失败。

void dma_async_issue_pending(struct dma_chan *chan)

功能: 启动通道传输。
参数:chan 通道指针

3.4.其他

int dmaengine_terminate_all(struct dma_chan *chan)

功能: 停止通道上的传输。
参数:chan 通道指针
返回:非零失败,零成功
注:此功能会丢弃未开始的传输

int dmaengine_pause(struct dma_chan *chan)

功能: 暂停某通道的传输。
参数:chan 通道指针
返回:非零失败,零成功

int dmaengine_resume(struct dma_chan *chan)

功能: 恢复某通道的传输。
参数:chan 通道指针
返回:非零失败,零成功

enum dma_status dmaengine_tx_status(struct dma_chan *chan, dma_cookie_t cookie,
struct dma_tx_state *state)

功能: 查询某次提交的状态。
参数:chan 通道指针
cookie 提交返回的 id
state 用于获取状态的变量地址
返回:返回当前的状态,取值如下:
dma_success 传输成功完成
dma_in_progress 提交尚未处理或处理中
dma_pause 传输已经暂停
dma_error 此次传输失败

4. dmaengine 使用流程
本章节主要是讲解 dmaengine 的使用流程,以及注意事项

4.1.基本流程

4.2.注意事项
回调函数里不允许休眠,以及调度
回调函数时间不宜过长
pending 并不是立即传输而是等待软中断的到来,cyclic 模式除外
在 dma_slave_config 中的 slave_id 对于 deivces 必须要指定

5. 使用范例
5.1.范例

struct dma_chan *chan;
dma_cap_mask_t mask;
dma_cookie_t cookie;
struct dma_slave_config config;
struct dma_tx_state state;
struct dma_async_tx_descriptor *tx = null;
void *src_buf;
dma_addr_t src_dma;
dma_cap_zero(mask);
dma_cap_set(dma_slave, mask);
dma_cap_set(dma_cyclic, mask);
/* 申请一个可用通道 */
chan = dma_request_channel(dt->mask, null, null);
if (!chan){
return -einval;
}
src_buf = kmalloc(1024*4, gfp_kernel);
if (!src_buf) {
dma_release_channel(chan);
return -einval;
}
/* 映射地址用dma访问 */
src_dma = dma_map_single(null, src_buf, 1024*4, dma_to_device);
config.direction = dma_mem_to_dev;
config.src_addr = src_dma;
config.dst_addr = 0x01c;
config.src_addr_width = dma_slave_buswidth_2_bytes;
config.dst_addr_width = dma_slave_buswidth_2_bytes;
config.src_maxburst = 1;
config.dst_maxburst = 1;
config.slave_id = sunxi_slave_id(drqdst_audio_codec, drqsrc_sdram);
dmaengine_slave_config(chan, &config);
tx = dmaengine_pre_dma_cyclic(chan, scr_dma, 1024*4, 1024, dma_mem_to_dev,
dma_prep_interrupt | dma_ctrl_ack);
/* 设置回调函数 */
tx->callback = dma_callback;
tx->callback = null;
/* 提交及启动传输 */
cookie = dmaengine_submit(tx);
dma_async_issue_pending(chan);

阅读(711) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~
")); function link(t){ var href= $(t).attr('href'); href ="?url=" encodeuricomponent(location.href); $(t).attr('href',href); //setcookie("returnouturl", location.href, 60, "/"); }
网站地图