我用的是正点的
STM32F103
来进行学习,板子和教程是野火的指南者
。之后的这个系列笔记开头未标明的话,用的也是这个板子和教程。
DMA的基础知识与用法
CPU在板子里是个大忙人,各个外设都需要CPU操心,经常出现找不到人的情况。
所以为了给CPU减负,DMA应运而生。
DMA,全称Direct Memory Access,即直接存储器访问。
DMA传输将数据从一个地址空间复制到另一个地址空间,提供在外设和存储器之间或者存储器和存储器之间的高速数据传输。
换句话说,让两个外设自己打电话。
DMA 数据 外设2 外设1 CPU
如果有了DMA,那么传输是这样的:
参考文章如下:
【STM32】 DMA原理,步骤超细详解,一文看懂DMA
一、DMA功能框图
笔者使用的板子是野火的指南者,MCU为STM32F103。
在开发手册中可以看到
- DMA请求
- 通道
- 仲裁器
1、DMA请求
如果外设要想通过 DMA 来传输数据,必须先给 DMA 控制器发送 DMA 请求,DMA 收到请求信号之后,控制器会给外设一个应答信号,当外设应答后且 DMA 控制器收到应答信号之后,就会启动 DMA 的传输,直到传输完毕。
- 外设发送DMA请求 2. DMA发送应答信号 3. DMA收到外设应答 外设 DMA 启动DMA传输
DMA 有 DMA1 和 DMA2 两个控制器,DMA1 有 7 个通道,DMA2 有 5 个通道,不同的 DMA 控制器的通道对应着不同的外设请求,这决定了我们在软件编程上该怎么设置。
2、通道
DMA 具有 12 个独立可编程的通道,其中 DMA1 有 7 个通道,DMA2 有 5 个通道,每个通道对应不同的外设的 DMA 请求。虽然每个通道可以接收多个外设的请求,但是同一时间只能接收一个,不能同时接收多个。
3、仲裁器
当发生多个 DMA 通道请求时,就意味着有先后响应处理的顺序问题,这个就由仲裁器也管理。
仲裁器管理 DMA 通道请求分为两个阶段。
-
第一阶段属于软件阶段,可以在 DMA_CCRx 寄存器中设置,有 4 个等级:
- 非常高
- 高
- 中
- 低
-
第二阶段属于硬件阶段,如果两个或以上的 DMA 通道请求设置的优先级一样,则他们优先级取决于通道编号,编号越低优先权越高,比如通道 0 高于通道 1。
在大容量产品和互联型产品中,DMA1 控制器拥有高于 DMA2 控制器的优先级。
二、DMA传输设置
通常来说,DMA传输最重要的三个内容为:
- 数据来源与数据去向
- 每次传输大小
- 传输结束标志
1、数据来源与数据去向
我们知道 DMA 传输数据的方向有三个:
- 从外设到存储器 P->M
- 从存储器到外设 M->P
- 从存储器到存储器 M->M
具体的方向 DMA_CCR 位 4 DIR 配置:0 表示从外设到存储器,1 表示从存储器到外设。
这里面涉及到的外设地址由 DMA_CPAR 配置,存储器地址由 DMA_CMAR 配置。
外设到存储器
当我们使用从外设到存储器传输时,以 ADC 采集为例。
DMA 外设寄存器的地址对应的就是 ADC数据寄存器的地址,DMA 存储器的地址就是我们自定义的变量(用来接收存储 AD 采集的数据)的地址。
外设到存储器 自定义的变量地址 ADC数据寄存器地址
存储器到外设
当我们使用从存储器到外设传输时,以串口向电脑端发送数据为例。
DMA 外设寄存器的地址对应的就是串口数据寄存器的地址,DMA 存储器的地址就是我们自定义的变量(相当于一个缓冲区,用来存储通过串口发送到电脑的数据)的地址。
存储器到外设 串口数据寄存器的地址 自定义的变量地址
存储器到存储器
当我们使用从存储器到存储器传输时,以内部 FLASH 向内部 SRAM 复制数据为例。
DMA 外设寄存器的地址对应的就是内部 FLASH(我们这里把内部 FALSH 当作一个外设来看)的地址,DMA存储器的地址就是我们自定义的变量(相当于一个缓冲区,用来存储来自内部 FLASH 的数据)的地址。
存储器到存储器 自定义变量的地址 内部FLASH地址
(当做外设看待)
跟上面两个不一样的是,这里需要把DMA_CCR 位 14:MEM2MEM:存储器到存储器模式配置为 1,启动 M2M 模式。
2、每次传输大小
当我们配置好数据要从哪里来到哪里去之后,我们还需要知道我们要传输的数据是多少,数据的单位是什么。
传输数据设置 如果打开了自动重加载模式
数据量传输至0后复位 设置源地址和目标地址的增量模式
(设置数据存放的方向) 设置源地址和目标地址的宽度
(设置传输单位) 设置传输数据量大小
(设置传输总量)
以串口向电脑发送数据为例,我们可以一次性给电脑发送很多数据,具体多少由 DMA_CNDTR配置,这是一个 32 位的寄存器,一次最多只能传输 65535 个数据。
要想数据传输正确,源和目标地址存储的数据宽度还必须一致,串口数据寄存器是 8 位的,所以我们定义的要发送的数据也必须是 8 位。
外设的数据宽度由 DMA_CCRx 的 PSIZE[1:0] 配置,可以是 8/16/32 位,存储器的数据宽度由 DMA_CCRx 的 MSIZE[1:0] 配置,可以是 8/16/32 位。
在 DMA 控制器的控制下,数据要想有条不紊的从一个地方搬到另外一个地方,还必须正确设置两边数据指针的增量模式。
外设的地址指针由 DMA_CCRx 的 PINC 配置,存储器的地址指针由MINC 配置。
以串口向电脑发送数据为例,要发送的数据很多,每发送完一个,那么存储器的地址指针就应该加 1,而串口数据寄存器只有一个,那么外设的地址指针就固定不变。具体的数据指针的增量模式由实际情况决定。
3、传输结束标志
数据什么时候传输完成,我们可以通过查询标志位或者通过中断的方式来鉴别。
每个 DMA 通道在 DMA 传输过半、传输完成和传输错误时都会有相应的标志位,如果使能了该类型的中断后,则会产生中断。
有关各个标志位的详细描述请参考 DMA 中断状态寄存器 DMA_ISR 的详细描述。
传输完成还分两种模式,是一次传输还是循环传输,一次传输很好理解,即是传输一次之后就停止,要想再传输的话,必须关断 DMA 使能后再重新配置后才能继续传输。循环传输则是一次传输完成之后又恢复第一次传输时的配置循环传输,不断的重复。
具体的由 DMA_CCRx 寄存器的CIRC 循环模式位控制。
未完成,先占一个位置