DMA简介

DMA是一个数据转运小助手, 它主要是用来协助CPU,完成数据转运的工作

第一个程序: 在这个程序里,我们将使用DMA,进行存储器到存储器的数据转运, 也就是把一个数组里面的数据, 复制到另一个数组里

DMA简介

DMA外设, 是可以直接访问STM32内部的存诸器的, 包括运行内存SRAM, 程序存储器Flash和寄存器等等

DMA可以提供外设(一般是外设的数据寄存器DR, DataRegister, 比如ADC的数据寄存器, 串口的数据寄存器等等)和存储器(指的就是是运行内存SRAM和程序存储器Flash, 是我们存储变量数组和程序代码的地方)或者存储器和存储器之间的高速数据传输,无须CPU干预,节省了CPU的资源

DMA有12个独立可配置的通道: DMA1(7个通道), DMA2(5个通道), 这个通道就是数据转运的路径, 从一个地方移动到另一个地方, 就需要占用一个通道, 如果有多个通道进行转运, 那它们之间可以各转各的, 互不干扰

每个通道都支持软件触发和特定的(意思就是每个DMA的通道, 它的硬件触发源是不一样的, 你要使用某个外设的硬件触发源, 就得使用它连接的那个通道, 而不能任意选通道)硬件触发

这里如果DMA进行的是存储器到存储器的数据转运, 比如我们想把Flash里的一批数据, 转运到SRAM里去, 那就需要软件触发了,使用软件触发之后, DMA就会一股脑地,把这批数据, 以最快的速度 全部转运完成, 那如果DMA进行的是外设到存诸器的数据转运, 就不能一股脑地转运了, 因为外设的数据是有一定时机的, 所以这时我们就需要用硬件触发,比如转运ADC的数据, 那就得ADC每个通道AD转换完成后, 硬件触发一次DMA, 之后DMA再转运, 触发一次,转运一次

STM32F103C8T6 DMA资源:DMA1(7个通道)

STM32的存储器映像

起始地址 存储器 用途
0x0800 0000 程序存储器Flash(ROM) 存储C语言编译后的程序代码
0x1FFF F000 系统存储器(ROM) 存储BootLoader,用于串口下载(下一节讲串口)
0x1FFF F800 选项字节(ROM) 存储一些独立于程序代码的配置参数
0x2000 0000 运行内存SRAM(RAM) 存储运行过程中的临时变量(也就是我们在程序中定义变量、 数组、结构体的地方)
0x4000 0000 外设寄存器(RAM) 存储各个外设的配置参数(也就是我们初始化各个外设, 最终所读写的东西)
0xE000 0000 内核外设寄存器(RAM) 存储内核各个外设(内核外设就是NVIC和SysTick, 因为内核外设和其他外设不是一个厂家设计的, 所以它们的地址也是被分开了)的配置参数

从下面这个图总结出来的

在上面这个图里,有灰色填充的, 就是Reserved区域,也就是保留区域,没有使用到, 然后这个0地址,实际上也是没有存储器的,它这里写的是别名到Flash或者系统存储器,取决于BOOT引脚, 因为程序是从0地址开始运行的, 所以这里需要把我想要执行的程序, 映射到0地址来,如果映射在Flash区,就是从Flash执行, 如果映射在系统存储器区, 就是从系统存储器运行BootLoader, 如果映射到SRAM,就是从SRAM启动, 怎么选择就是由BOOT0和BOOT1两个引脚来决定, 这就是0地址里的别名区, 接着剩下的0800开始的Flash区, 用于存储程序代码

在这个表里, 无论是Flash, 还是SRAM, 还是外设寄存器, 它们都是存储器的一种, 包括外设寄存器, 实际上也是存储器 ,我们前面这里说的是外设到存储器, 存储器到存储器, 本质上其实都是存储器之间的数据转运,说成外设到存储器,只不过是STM32他特别指定了可以转运外设的存储器而已

0x0800开头, 那你就可以确定, 它是属于主闪存的数据

系统存储器和选项字节的存储介质一般也是Flash, 只不过是我们一般说Flash指的是主闪存Flash, 而不指这两块区域

选项字节里 存的主要是Flash的读保护, 写保护, 还有看门狗等等的配置

外设寄存器的存储介质其实也是SRAM, 只不过我们喜欢把运行内存叫SRAM, 外设寄存器就直接叫寄存器了

STM32的寻址能力是4GB, 而我们STM32的存储器都是KB级别的, 所以这个4GB的寻址空间, 会有大量的地址都是空的, 地址使用率还不到1%

DMA硬件电路图

总共就是CPU和存储器这两个东西

一方面, CPU可以对寄存器进行读写, 就像读写运行内存一样, 另一方面, 寄存器的每一位背后, 都连接了一根导线, 这些导线可以用于控制外设电路的状态, 比如置引脚的高低电平, 导通和断关, 切换数据选择器, 或者多位组合起来, 当做计数器、 数据寄存器, 等等等等, 所以寄存器是连接软件和硬件的桥梁, 软件读写寄存器,就相当于在控制硬件的执行

使用DMA进行数据转运, 都可以为一类问题, 就是从某个地址取内容, 再放到另一个地址去, 为了高效有条理地访问存储器, 这里设计了一个总线矩阵, 总线矩阵的左端, 是主动单元, 也就是拥有存储器的访问权, 右边这些,是被动单元, 它们的存储器只能被左边的主动单元读写, 主动单元这里, 内核有DCode和系统总线, 可以访问右边的存储器, 其中DCode总线是专门访问Flash的, 系统总线是访问其他东西的, 另外, 由于DMA要转运数据,所以DMA也必须要有访问的主动权, 那主动单元 除了内核CPU, 剩下的就是DMA总线了, 这里DMA1有一条DMA总线, DMA2也有一条DMA总线, 下面这还有一条DMA总线, 这是以太网外设自己私有的DMA, 这个可以不用管的, 在DMA1和DMA2里面, 可以看到 DMA1有7个通道, DMA2有5个通道, 各个通道可以分别设置它们转运数据的源地址和目的地址, 这样它们就可以各自独立地工作了, 接着下面这里有个仲裁器,这个是因为,虽然多个通道可以独立转运数据, 但是最终DMA总线只有一条, 所以所有的通道都只能分时复用这一条DMA总线, 如果产生了冲突, 那就会由仲裁器,根据通道的优先级, 来决定谁先用, 谁后用, 另外在总线矩阵这里,也会有个仲裁器, 如果DMA和CPU都要访问同一个目标, 那么DMA就会暂停CPU的访问,以防止冲突, 不过总线伸裁器, 仍然会保证CPU得到一半的总线带宽, 使CPU也能正常的工作, 下面这里是AHB从设备, 也就是DMA自身的寄存器, 因为DMA作为一个外设, 它自己也会有相应的配置寄存器, 这里连接在了总线右边的AHB总线上, 所以DMA, 即是总线矩阵的主动单元, 可以读写各种存储器, 也是AHB总线上的被动单元, CPU通过下面这一条线路,就可以对DMA进行配置了

接着继续看这里, 是DMA请求, 请求就是触发的意思, 这条线路右边的触发源,是各个外设

所以这个DMA请求就是DMA的硬件触发源, 比如ADC转换完成, 串口接收到数据, 需要触发DMA转运数据的时候, 就会通过这条线路, 向DMA发出硬件触发信号, 之后DMA就可以执行数据转运的工作了

再讲一个小问题, 就是这里的Flash, 它是ROM只读存诸器的一种, 如果通过总线直接访问的话, 无论是CPU, 还是DMA, 都是只读的,只能读取数据,而不能写入, 如果你DMA的自的地址, 填了Flash的区域, 那转运时, 就会出错, 当然Flash也不是绝对的不可写入, 我们可以配置这个Flash接口控制器, 对Flash进行写入, 这个流程就较麻烦了。 要先对Flash按页进行擦除,再写入数据(去年OS里面SSD这一章节), 然后SRAM是运行内存, 可以任意读写, 没有问题, 外设寄存器的话, 得看参考手册里面的描述, 有的寄存器是只读的, 有的寄存器是只写的, 不过我们主要用的是数据寄存器,数据寄存器都是可以正常读写的

DMA代码流程图

当M2M位给0, 那就是使用硬件触发了, 硬件触发源可以选择ADC, 串口, 定时器等等, 使用硬件触发的转运,一般都是与外设有关的转运, 这些转运需要一定的时机, 比如ADC转换完成, 串口收到数据、 定时时间到等等, 所以需要使用硬件触发, 在硬件达到这些时机时, 传一个信号过来, 来触发DMA进行转运

然后最后, 就是开关控制了, 也就是DMA_Cmd函数, 当给DMA使能后, DMA就准备就绪, 可以进行转运了, DMA进行转运,有几个条件, 第一, 就是开关控制, DMA_Cmd必须使能, 第二, 就是传输计数器必须大于0, 第三, 就是触发源, 必须有触发信号, 触发一次,转运一次, 传输计数器自减一次,当传输计数器等于0, 且没有自动重装时, 这时无论是否触发, DMA都不会再进行转运了, 此时就需要DMA_Cmd,给DISABLE, 关闭DMA, 再为传输计数器写入一个大于0的数, 再DMA_Cmd,给ENABLE, 开启DMA, DMA才能继续工作, 注意一下,写传输计数器时,必须要先关闭DMA,再进行(这是手册里面的规定)

DMA请求

就是上面DMA触发(软硬触发部分)的部分, 下面这张图就是DMA1的请求映像

下面是DMA的7个通道, 每个通道都有一个数据选择器, 可以选择硬件触发或软件触发

这里这个图不好理解

他把EN位画在了数据选择器的侧边, 一般数据选择器的侧边是输入选择控制位, 难道这里的意思是, EN给1, 选择硬件触发, EN给0, 选择软件触发吗?那显然不对, 而且它左边这里写的是软件触发(MEM2MEM位 == M2M), 难道M2M位是软件触发吗?

所以给这个图重新布了个局, 就上一章节的图, M2M位是数据选择器的控制位, 用于选择是硬件触发还是软件触发, EN位是开关控制, EN=0时不工作, EN=1时工作, 这样就好理解了

他那里这样画的意思是, EN并不是数据选择器的控制位而且决定这个数据选择器要不要工作, EN=0, 数据选择器不工作, EN=1, 数据选择器工作, 然后软件触发后面跟个M2M位的意思应该是, 当M2M位=1时选择软件触发

那然后继续看左边的硬件触发源, 这里是外设请求信号, 可以看到, 每个通道的硬件触发源都是不同的, 如果你需要用ADC1来触发的话那就必须选择通道1, 如果需要定时器2的更新事件(TIM2_UP)来触发的话,那就必须选择通道2

因为每个通道的硬件触发源都不同, 所以如果你想使用某个硬件触发源的话, 就必须使用它所在的通道, 而如果使用软件触发的话,那通道就可以任意选择了, 因为每个通道的软件触发都是一样的,所以在PPT的这前面,写的是, 每个通道都支持软件触发和特定的硬件触发,这就是特定的意思, 选择硬件触发是要看通道的

那到底是选哪个触发源呢?

这个是对应的外设是否开启了DMA输出来决定的, 比如你要使用ADC1, 那会有个库函数叫ADC_DMACmd, 必须使用这个库函数开启ADC1的这一路输出, 它才有效

所以, 这三个触发源,具体使用哪个, 取决于你把哪个外设的DMA输出开启了, 如果3个都开启了 那这边是一个或门, 理论上3个硬件都可以进行触发,不过一般情况下我们都是开启其中一个

之后, 这7个触发源, 进入到仲裁器, 进行优先级判断, 最终产生内部的DMA1请求, 这个优先级的判断类似于中断的优先级, 默认优先级是通道号越小优先级越高, 当然也可以在程序中配置优先级

数据宽度与对齐

如果数据宽度都一样, 那就是正常的一个个转运, 如果两个数据宽度不一样,那会怎么处理呢

下面这个表就是来说明这个问题的

相关推荐
whik11949 小时前
Keil-MDK开发环境编译后axf自动转换bin格式文件
stm32·arm·keil·axi·bin·mdk
stm32发烧友9 小时前
基于STM32的声控舞蹈机器人设计
stm32·嵌入式硬件·机器人
DS小龙哥11 小时前
基于STM32设计的粮食仓库(粮仓)环境监测系统
stm32·嵌入式硬件·信息可视化
Krysdon14 小时前
Freertos任务切换
stm32·freertos
云山工作室14 小时前
基于单片机的农田灌溉系统(论文+源码)
stm32·单片机·嵌入式硬件·毕业设计·毕设
LightningJie15 小时前
freertos是在哪里调用port.c函数的,如何调用的,也没有port.h头文件怎么调用的呢
stm32·嵌入式硬件·算法
python百炼成钢15 小时前
14-STM32 PWM脉冲宽度调制
stm32·单片机·嵌入式硬件
沐欣工作室_lvyiyi16 小时前
基于单片机的血氧心率检测与报警系统(论文+源码)
stm32·单片机·嵌入式硬件·物联网·毕业设计
蓝本生18 小时前
STM32标准库学习之寄存器方法点亮LED灯
stm32·嵌入式硬件·学习
番茄老夫子20 小时前
STM32 USB通信知识与应用详解
stm32·单片机·嵌入式硬件