江协科技STM32学习- P23 DMA 直接存储器存取

🚀write in front🚀

🔎大家好,我是黄桃罐头,希望你看完之后,能对你有所帮助,不足请指正!共同学习交流

🎁欢迎各位→点赞👍 + 收藏⭐️ + 留言📝​

💬本系列哔哩哔哩江科大STM32的视频为主以及自己的总结梳理📚

🚀Projeet source code🚀

💾工程代码放在了本人的Gitee仓库:iPickCan (iPickCan) - Gitee.com

引用:

STM32入门教程-2023版 细致讲解 中文字幕_哔哩哔哩_bilibili

Keil5 MDK版 下载与安装教程(STM32单片机编程软件)_mdk528-CSDN博客

STM32之Keil5 MDK的安装与下载_keil5下载程序到单片机stm32-CSDN博客

0. 江协科技/江科大-STM32入门教程-各章节详细笔记-查阅传送门-STM32标准库开发_江协科技stm32笔记-CSDN博客

【STM32】江科大STM32学习笔记汇总(已完结)_stm32江科大笔记-CSDN博客

江科大STM32学习笔记(上)_stm32博客-CSDN博客

STM32学习笔记一(基于标准库学习)_电平输出推免-CSDN博客

STM32 MCU学习资源-CSDN博客

stm32学习笔记-作者: Vera工程师养成记

stem32江科大自学笔记-CSDN博客

术语:

|--------------------------------------|---------------------------------------------|
| 英文缩写 | 描述 |
| GPIO:General Purpose Input Onuput | 通用输入输出 |
| AFIO:Alternate Function Input Output | 复用输入输出 |
| AO:Analog Output | 模拟输出 |
| DO:Digital Output | 数字输出 |
| 内部时钟源 CK_INT:Clock Internal | 内部时钟源 |
| 外部时钟源 ETR:External clock | 时钟源 External clock |
| 外部时钟源 ETR:External clock mode 1 | 外部时钟源 Extern Input pin 时钟模式1 |
| 外部时钟源 ETR:External clock mode 2 | 外部时钟源 Extern Trigger 时钟模式2 |
| 外部时钟源 ITRx:Internal trigger inputs | 外部时钟源,ITRx (Internal trigger inputs)内部触发输入 |
| 外部时钟源 TIx:external input pin | 外部时钟源 TIx (external input pin)外部输入引脚 |
| CCR:Capture/Comapre Register | 捕获/比较寄存器 |
| OC:Output Compare | 输出比较 |
| IC:Input Capture | 输入捕获 |
| TI1FP1:TI1 Filter Polarity 1 | Extern Input 1 Filter Polarity 1,外部输入1滤波极性1 |
| TI1FP2:TI1 Filter Polarity 2 | Extern Input 1 Filter Polarity 2,外部输入1滤波极性2 |
| DMA:Direct Memory Access | 直接存储器存取 |

正文:

0. 概述

从 2024/06/12 定下计划开始学习下江协科技STM32课程,接下来将会按照哔站上江协科技STM32的教学视频来学习入门STM32 开发,本文是视频教程 P2 STM32简介一讲的笔记。


定时器共四个部分,分为八个小节笔记。本小节为第一部分第一节。

🌳在第一部分,是定时器的基本定时的功能:定时中断功能、内外时钟源选择

🌳在第二部分,是定时器的输出比较功能,最常见的用途是产生PWM波形,用于驱动电机等设备

🌳在第三部分,是定时器的输入捕获功能和主从触发模式,来实现测量方波频率

🌳在第四部分,是定时器的编码器接口,能够更加方便读取正交编码器的输出波形,编码电机测速


1.🚢DMA

本节我们来学习DMA,直接存储器存取。

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

2.🚢DMA的简介

DMA(Direct Memory Access)直接存储器存取,或者叫直接存储器访问;

🦄从这个DMA名字的意思来看,DMA这个外设是可以直接访问STM32内部的存储器的,包括运行内存SRAM、程序存储器flash和寄存器等等,DMA都有权限访问它们,所以DMA才能完成数据转运的工作。

🦄DMA可以提供外设和存储器或者存储器和存储器之间的高速数据传输,无须CPU干预,节省了CPU的资源

🦄这里外设指的就是外设寄存器,一般是外设的数据寄存器DR、Data、 Register。比如ADC的数据寄存器、串口的数据寄存器等等。

🦄这里存储器指的就是运行内存SRAM和程序存储器flash,是我们存储变量数组和程序代码的地方,在外设和存储器或者存储器和存储器之间进行数据转运,就可以使用DMA来完成。并且在转运的过程中,无需CPU的参与,节省了CPU的资源,CPU省下时间就可以干一些其它的更加专业的事情,搬运数据这种杂活交给DMA就行了。

🦄12个独立可配置的通道: DMA1(7个通道), DMA2(5个通道)

🦄这个通道就是数据转运的路径,从一个地方移动到另一个地方,就需要占用一个通道。如果有多个通道进行转运,那它们之间可以各转各的互不干扰,这就是DMA的通道。

🦄每个通道都支持软件触发和特定的硬件触发

🦄如果DMA进行的是存储器到存储器的数据转运。比如我们想把flash里的一批数据转运到SRAM里去,那就需要软件触发了。

🦄使用软件触发之后,DMA就会一股脑地把这批数据以最快的速度全部转运完成。这也是我们想要的效果。

🦄那如果DMA进行的是外设到存储器的数据转运,就不能一股脑地转运了。因为外设的数据是有一定时机的 。所以这时我们就需要用硬件触发 ,比如转运ADC的数据。那就得ADC每个通道AD转换完成后,硬件触发一次DMA,之后DMA再转运,触发一次,转运一次,这样数据才是正确的,才是我们想要的效果。

🦄所以存储器到存储器的数据转运, 我们一般使用 软件触发, 外设到存储器的数据转运 我们一般使用 硬件触发

🦄特定的硬件触发意思就是每个DMA的通道,它的硬件触发源是不一样的。你要使用某个外设的硬件触发源,就得使用它连接的那个通道,而不能任意选择通道

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

🦄我们这个芯片只有DMA1的七个通道,没有DMA2。

3. 🚢存储器映像

既然DMA是在存储器之间进行数据转运的,那我们就应该要了解一下STM32中都有哪些存储器,这些存储器又是被安排到了哪些地址上,这就是存储器映像的内容。

计算机系统的五大组成部分是运算器、控制器、存储器、输入设备和输出设备。其中运算器和控制器一般会合在一起,叫做CPU。

所以计算机的核心关键部分就是CPU和存储器 。存储器又有两个重要知识点,一个是存储器的内容,另一个就是存储器的地址。那STM32也不例外,这个表就是STM32中所有类型的存储器和它们所被安排的地址。

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

在上表里,存储器总共分成两大类ROM和RAM。

ROM 就是只读存储器 是一种非 易失 掉电不丢失的存储器

RAM 就是随机存储器 是一种 易失 掉电丢失的存储器

其中ROM分为了三块:

  • 第一块是程序存储器flash,也就是主闪存。它的用途就是存储C语言编译后的程序代码,也就是我们下载程序的位置,运行程序一般也是从主闪存里面开始运行的。这一块存储器STM32给它分配的起始地址是0x0800 0000。然后剩余字节的地址依次增长,每个字节都分配一个独一无二的地址。终止地址取决于它的容量编到哪里,哪里就是终止地址,这就是主闪存的地址范围。之后如果在软件里看到某个数据的地址是0x0800开头的,那就可以确定它是属于主闪存的数据。
  • **接着下面两块系统存储器和选项字节,这两块存储器也是ROM的一种,掉电不丢失。**实际上它们的存储介质也是flash,只不过是我们一般说flash指的是主闪存flash,而不是指系统存储器和选项字节。它们的地址都是0x1FFF开头的,紧跟着0x2000开头的就是RAM区。

所以可以看出系统存储器 和选项字节这两块存储器的位置是在ROM区的最后面。

系统存储器的用途是存储 BootLoader 用于串口下载 程序存储的位置就被分配到BootLoaderBootLoader程序是芯片出厂自动写入的,一般也不允许我们修改。

选项字节是用于存储一些独立于程序代码的配置参数。 它的位置是在ROM区的最后面。下载程序可以不刷新选项字节的内容,这样选项字节的配置就可以保持不变。选项字节里存的主要是flash的读保护 写保护 还有看门狗等等的配置。

然后我们看一下RAM区。

  • 首先是运行内存 SRAM 分配的地址是0x2000 0000,用途是存储运行过程中的临时变量 也就是我们在程序中定义变量 数组 结构体的地方。 你可以试一下定义一个变量,再取它的地址显示出来。那这个地址肯定就是0x2000开头的。类比于电脑的话 运行内存就是内存条
  • 然后RAM区剩下的还有外设寄存器,它的地址是0x4000 0000这块区域,用途是存储各个外设的配置参数 ,也就是我们初始化各个外设最终所读写的东西。
    刚才我们说了
    外设寄存器也是存储器的一种。它的存储介质其实也是 SRAM,只不过我们一般习惯把运行内存叫SRAM,外设寄存器就直接叫寄存器。
  • 内核外设寄存器 地址是 0xE000 0000 这片区域 用途是存储内核各个外设的配置 ****,****参数内核外设就是NVIC和SysTick。因为内核外设和其它外设不是一个厂家设计的,所以它们的地址也是被分开了。内核外设是0xE000,其它外设是0x4000 。

以上这些就是STM32里的存储器和它们被安排的地址。

接下来我们看一下数据手册中的这张图(对应上面的表格)

在STM32中所有的存储器都被安排到了0~0xFFFF FFFF这个地址范围内

因为CPU是32位的,所以寻址范围就是32位的范围。32位的寻址范围是非常大的,最大可以支持4GB容量的存储器。而我们STM32的存储器都是KB级别的。所以这个4GB的寻址空间会有大量的地址都是空的,算一下地址的使用率还不到百分之一。

在这个图里,灰色填充的就是reserve的区域,也就是保留区域,没有使用到。

0 地址 实际上也是没有存储器的,它这里写的是别名到flash或者系统存储器取决于boot引脚。

因为程序是从0地址开始运行的,所以这里需要把我们想要执行的程序映射到零地址来。如果映射在flash区,就是从flash执行。

如果映射在系统存储器区,就是从系统存储器运行BootLoader。

如果映射到SRAM,就是从SRAM启动。

怎么选择由BOOT0和BOOT1两个引脚来决定,这就是0地址里的别名区。

剩下的0x0800开始的flash区,用于存储程序代码。

0x1FFF开始的系统存储器和选项字节是在ROM区的最后面。

0x2000开始的是SRAM区,0x4000开始的是外设寄存器

这里面可以展开,就是右边这些东西

每个外设又有它们自己的起始地址,比如TIM2的地址是0x4000 0000。

然后外设地址里面又可以具体细分到每个寄存器的地址、寄存器里每个字节的地址,最终所有字节的地址就都可以算出来了。

最后 上面这里 0xE000 开始的区域存放在就是内核里面的外设 寄存器

接下来我们来看一下DMA的框图。

4.🚢DMA框图

这个框图我们在第一节STM32的系统结构里也见过一个类似的。

左上角这里是Cortex-M3内核里面包含了CPU和内核外设等等。

剩下的所有东西都可以把它看成是存储器。

  • 🤠所以这个框图总共就是CPU和存储器两个东西。
  • 🤠flash是主闪存,SRAM 是运行内存 各个外设都可以看成是寄存器 也是一种 SRAM 存储器 因为 寄存器是一种特殊的存储器

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

🤠所以寄存器是连接软件和硬件的桥梁。 软件读写寄存器 就相当于在控制硬件的执行 所以我们可以把外设抽象成一个个寄存器,CPU控制外设本质上就是读写寄存器

**既然外设就是寄存器,寄存器就是存储器,**那使用DMA进行数据转运,就都可以归为一类问题了,就是从某个地址取内容再放到另一个地址去。

我们看向上面的框图,为了高效有条理的访问存储器,这里设计了一个总线矩阵 。总线矩阵的左端是主动单元 也就是拥有存储器的访问权。 右边这些是被动单元,它们的存储器只能被左边的主动单元读写。

主动单元 这块 内核有 DCode 和系统总线 可以访问右边的存储器 ****,****其中DCode的总线是专门访问flash的,系统总线是访问其它东西的。

另外,由于DMA要转运数据,所以DMA也必须要有访问的主动权。那主动单元除了内核 CPU, 剩下的就是 DMA ****线了。****这里DMA1有一条DMA总线,DMA2也有一条DMA总线。下面还有一条是以太网外设自己私有的DMA总线,这个可以不用管。

在DMA1和DMA2里面可以看到DMA1有七个通道,,DMA2有五个通道,各个通道可以分别设置它们转移数据的原地址和目的地址。这样 它们 就可以各自独立的工作了。

接着下面这里有个仲裁器。这个是因为虽然多个通道可以独立转运数据,但是最终DMA总线只有一条。所以所有的通道都只能分时复用这一条DMA总线,如果产生了冲突,那就会由仲裁器根据通道的优先级决定谁先用谁后用。

另外在总线矩阵里也会有个仲裁器 如果 DMA CPU 都要访问同一个目标 那么 DMA 就会暂停 CPU 的访问 以防止冲突。 不过总线仲裁器仍然会保证CPU得到一半的总线带宽,使CPU也能正常的工作。这就是仲裁器的作用。

后继续看下面这里是AHB 从设备, 也就是 DMA 自身的寄存器。因为 DMA 作为一个外设 它自己也会有相应的配置 寄存器 ****。****这里连接在了总线右边的AHB总线上。所以,DMA既是总线矩阵的主动单元,可以读写各种存储器,也是AHB总线上的被动单元。

CPU通过这一条线路就可以对DMA进行配置了。

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

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

到这里,有关DMA的结构就讲的差不多了。

总结DMA的结构其中包括:

  • 🤠用于访问各个存储器的DMA 总线
  • 🤠内部的多个通道可以进行独立的数据转运;
  • 🤠仲裁器用于调度各个通道,防止产冲突;
  • 🤠AHB从设备用于配置DMA参数;
  • 🤠DMA请求用于硬件触发DMA的数据转移。

这就是这个DMA的各个部分和作用。

最后再额外说一个问题,就是这里的flash它是 ROM 只读存储器 的一种,如果通过总线直接访问的话,无论是CPU还是DMA,都是只读的,只能读取数据 而不能写入如果你 DMA 的目的地址填写在flash的区域 那转运时就会出错。这个注意一下。

当然flash也不是绝对的不可写入,我们可以配置这个flash接口控制器对flash进行写入,这个流程就比较麻烦了,要先对flash按页进行擦除,再写入数据。不过这是另一个课题了,这里就不再讨论。

总之就是 CPU 或者 DMA 直接访问flash的话 是只可以读而不可以写的。

SRAM 是运行内存 可以任意读写。

外设寄存器得看参考手册里面的描述,有的寄存器是只读的,有的寄存器是只写的。不过我们主要用的是数据寄存器,数据寄存器都是可以正常读写的。

5. 🚢DMA基本结构图

刚才的框图只是一个笼统的结构图,对于DMA内部的执行细节,它还是没体现出来。如果想编写代码,实际去控制DMA的话,就按下图来。

图中这两部分就是数据转运的两大站点,左边是外设寄存器站点,右边是存储器站点包括flash和SRAM。

STM32 手册里所说的存储器 一般是特指flash和 SRAM, 不包含外设 寄存 器。 外设寄存 器一般直接 称作外设 所以就是外设到存储器 ****存储器到存储器这样来描述。****虽然我们刚才说了,寄存器也是存储器的一种,但是STM32还是使用了外设和存储器来作为区分,注意一下描述方法的不同。

看图中的箭头方向就知道DMA的数据转运可以是从外设到存储器,也是可以从存储器到外设,具体是向左还是向右,有一个方向的参数可以进行控制。

另外还有一种转运方式,就是存储器到存储器,比如flash到SRAM或者SRAM到SRAM这两种方式。

由于flash是只读的 所以 DMA 不可以进行 SRAM 到flash或者flash到flash的转移操作。

然后我们继续看这两边的参数,既然要进行数据转运,那肯定就要指定从哪里转到哪里,具体怎么转,所以外设和存储器两个站点就都有三个参数。

  • 🤠第一个是起始地址,有外设端的起始地址和存储器端的起始地址,这两个参数决定了数据是从哪里来到哪里去的。

  • 🤠第二个参数是数据宽度,这个参数的作用是指定一次转运要按多大的数据宽度来进行。它可以选择字节Byte、半字HalfWord和字word。字节就是八位,也就是一次转用一个uint8_t这么大的数据;半字是十六位,就是一次转用一个uint16_t;字是32位,就是一次转用uint32_t。比如转运ADC的数据,ADC的结果是uint16_t这么大,所以这个参数就要选择半字一次转运。

  • 第三个参数是地址是否自增 ,这个参数的作用是指定一次转运完成后,下一次转运是不是要把地址移动到下一个位置去,这就相当于是指针p++这个意思,比如ADC扫描模式,用DMA进行数据转运。外设地址是ADC_DR寄存器,寄存器这边显然地址是不用自增的,如果自增,那下一次转运就跑到别的寄存器那里去了。存储器这边地址就需要指针,每转运一个数据后,就往后挪个坑,要不然下次再转,就把上次的覆盖掉了,这就是地址是否自增的作用,就是指定是不是叫转运一次挪个坑这个意思。

如果要进行存储器到存储器的数据转运 那我们就需要把其中一个存储器的地址放在外设的这个 这样就能进行存储器到存储器的转运了。

只要在外设起始地址里写flash或者SRAM的地址,那它就会去flash或SRAM找数据。

这个站点虽然叫外设存储器,但是它就只是个名字而已。并不是说这个地址只能写寄存器的地址。如果写flash的地址,那它就会去flash里,找写SRAM,它就会去SRAM里找,这个没有限制。甚至你可以在外设站点写存储器的地址,存储器站点写外设的地址,然后方向参数给反过来,这样也是可以的,只是ST公司给它起的这样的名字而已。你也可以把它叫做站点A,站点B,不必拘泥于它写的外设站点和存储器站点这个名字。

接着往下面看,这里有个东西叫做传输 计数 这个东西就是用来指定我总共需要转运几次的。这个传输计 器是一个自 减计数 ****,****比如给它写个5,那DMA就只能进行5次数据转运。转运过程中,每转运一次,计数器的数就会减1。当传输计数器减到零之后,DMA就不会再进行数据转运了。

另外 0 之后 之前自增的地址也会恢复到起始地址的位置 ****,****以方便之后DMA开始新一轮的转换。

在传输计数器的右边有一个自动重装器,这个自动重装器的作用就是传输计数器减到零之后 是否要自动恢复到最初的值 ****,****比如最初传输计数器给5,如果不使用自动重装器,那转用5次后DMA就结束了。如果使用自动重装器,那转运5次计数器减到0后,就会立即重装到初始值5。

自动重装器决定了转运的模式 ,如果不重装,就是正常的单次模式,如果重装就是循环模式,比如如果想转运一个数组,那一般就是单次模式,转运一轮就结束了。如果是ADC扫描模式加连续转换,那为了配合ADC,DMA也需要使用循环模式。所以这个循环模式和ADC的连续模式差不多,都是指定一轮工作完成后,是不是立即开始下一轮工作

然后继续往下看,这一块就是DMA的触发控制。

触发就是决定 DMA 需要在什么时机进行转运的。触发源 硬件触发和软件触发 具体选择哪个由 M2M 这个参数决定M2M 就是memory to memory 存储器到存储器的意思。

  • 当我们给M2M位1时,DMA就会选择软件触发,应用在存储器到存储器转运的情况。
  • 这个软件触发并不是调用某个函数一次触发一次,这个软件触发的执行逻辑是以最快的速度连续不断的触发DMA争取早日把传输计数器清零,完成这一轮的转换。所以这里的软件触发和我们之前外部中断和ADC的软件触发可能不太一样,可以把它理解成连续触发。

💯💯注意:这个软件触发和循环模式不能同时用,因为软件触发就是想把传输计数器清零,循环模式是清零后自动重装。如果同时用的话,那DMA就停不下来了。

软件触发 ****一般适用于存储器到存储器的转运。****因为存储器到存储器的转运是软件启动,不需要时机,并且想尽快完成的任务。

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

最后就是开关控制,也就是DMA_Cmd函数,当DMA使能后,DMA就准备就绪可以进行转运了。

DMA进行转运有几个条件:

  • 😎第一就是开关控制,DMA_Cmd必须使能;
  • 😎第二就是传输计数器必须大于零;
  • 😎第三就是触发源必须有触发信号,触发一次转运一次,传输计数器自减一次,当传输计数器等于0且没有自动重装时,这时无论是否触发,DMA都不会再进行转运,此时就需要DMA_Cmd给disable关闭DMA。当传输计数器写一个大于零的时候,DMA_Cmd给ENable开启DMA,DMA才能继续工作。

⚠️⚠️⚠️注意:写传输计数器时, 必须要先关闭 DMA 再进行 不能在 DMA 开启时写传输 计数 这是手册里的规定。

接下来我们再看几个细节的问题:

6. 🚢DMA请求

这张图是DMA1的请求映象

这张图表示的就是结构图中这部分结构,DMA触发的部分:

请求映像图中可以看到DMA的七个通道,每个通道都有一个数据选择器,可以选择硬件触发或软件触发。EN决定这个数据选择器要不要工作,EN等于0数据选择器不工作,EN等于1数据选择器工作,然后软件触发后面跟个M2M的意思应该是当M2M等于1时选择软件触发。

图中左边的硬件触发源,是外设请求信号 ,可以看到每个通道的硬件触发源都是不同的。如果需要用ADC1来触发的话,那就必须选择通道1。如果需要定时器二的更新事件来触发的话,那就必须选择通道二,剩下的也是同理。因为每个通道的硬件触发源都不同 ,所以如果想使用某个硬件触发源的话,就必须使用它所在的通道,这就是硬件触发的注意事项。

而如果使用 软件触发 的话 通道就可以任意选择。 因为每个通道的软件触发都是一样的。这就是最开始DMA的简介那部分所讲的每个通道都支持软件触发和特定的硬件触发,这就是特定的意思,即选择硬件触发是要看通道的。

再看,比如 通道 1 硬件触发 ADC 1、 定时器 2是 通道 3, 定时器 4是 通道 1 那到底是选择哪个触发

这个是对应的外设是否开启了 DMA ****输出来决定的。****比如要使用ADC1,会有个库函数ADC_DMACmd,必须使用这个库函数开启ADC1的这一路输出,它才有效。如果想选择定时器二的通道三,那也会有个TIM_DMACmd函数用来进行DMA输出控制。

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

七个触发 进入到仲裁器 经优先级判断 最终产生内部的 DMA 1 请求。这个优先级的判断 类似于中断的优先级 默认优先级是通道号越小 ****优先级越高。****当然也可以在程序中配置优先级,这个其实影响并不是很大,大家了解一下就行了。

7. 🚢数据宽度与对齐

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

这个表就是来说明这个问题的:

第一列是源端宽度,第二列是目标宽度,第三列是传输数目。

当源端和目标宽度都是八位时,转运第一步是在源端的0位置读数据B0,在目标的零位置写数据B0,就是把这个B0从左边挪到右边。之后的步骤就是把B1从左边挪到右边,接着B2,B3,这是源端和目标都是八位的情况。

当源端是8位,目标是16位,那它的操作就是在源端读B0,在目标写00B0。之后读B1写00B1。这个意思就是如果你目标的数据宽度比源端的数据宽度大,那就在目标数据前面多出来的空位补零。

八位转运到32位也是一样的处理,前面空出来的都补零。

当目标数据宽度比 源端 数据宽度小时 比如由十六位转到八位 就是读 B1、B0 只写入 B0 。读 B3、B2 只写入 B2, 也就是把多出来的高位舍弃掉。

之后那些也都是类似的操作。

总之这个表的意思就是如果把小的数据转到大的里面去,高位就会补零。如果把大的数据转到小的里面去,高位就会舍弃掉。如果数据宽度一样,那就没事。就是跟uint8_t uint16_t和uint32_t变量之间相互赋值一样。不够就补零,超了就舍弃高位。

最后我们再来看两个例子,看看在这些实际的任务下,DMA是如何工作的。

8.🚢实例

第一个例子就是数据转运+DMA,第二个例子是ADC扫描模式+DMA。这两个例子将和我们演示的两个程序是对应的。

8.1数据转运 + DMA

先看第一个例子,这个例子的任务是将SRAM里的数组DataA转运到另一个数组DataB中。我们看一下这种情况下,这个基本结构里的各个参数,该如何配置。

首先是外设站点和存储器站点的起始地址、数据宽度、地址是否自增这三个参数。

在这个任务里,外设地址显然应该填DataA数组的首地址,存储器地址给DataB数组的首地址,然后数据宽度两个数组的类型都是uint8_t,所以数据宽度都是按八位的字节传输。关于地址是否自增,我们想要的效果是DataA[0]转到DataB[0],DataA[1]转到DataB[1]等等,两个数组的位置一一对应。所以转运完DataA[0]和DataB[0]之后,两个站点的地址都应该自增,都移动到下一个数据的位置,继续转运DataA[1]和DataB[1]这样来进行。如果左边不自增,右边自增,效果就是这样的,转运完成后,DataB的所有数据都会等于DataA[0]

如果左边自增,右边不自增,那效果就是这样的,转运完成后DataB[0]等于DataA的最后一个数DataB,其它的数不变

如果左右都不自增,那就一直是DataA[0]转到DataB[0],其它的数据不变,这就是地址是否自增的效果。

方向参数显然就是外设站点转运到存储机站点。如果想把DataB的数据转运到DataA,那可以把方向参数换过来,这样就是反向转运。

然后是传输计数器和是否要自动重装,在这里显然要转运7次,所以传输计数器给7,自动重装暂时不需要。

触发选择部分这里我们要使用软件触发,因为这是存储器到存储器的数据转运,是不需要等待硬件时机的,尽快转运完成就行了。

最后调用DMA_Cmd给DMA使能,这样数据就会从DataA转运到DataB了。

转运7次之后,传输计数器自减到0,DMA停止,转运完成。

这里的数据转运是一种复制转运。转运完成后,DataA的数据并不会消失。这个过程相当于是把DataA的数据复制到了DataB的位置。这就是第一个任务,存储器到存储器的数据转移。

8.2ADC 扫描模式 + DMA

接着看第二个任务ADC扫描模式+DMA

左边是ADC扫描模式的执行流程,在这里有七个通道,触发一次后七个通道依次进行AD转换,然后转换结果都放到ADC_DR数据寄存器里面。

那我们要做的就是在每个单独的通道转换完成后,进行一次DMA数据转运,并且目的地址进行自增,这样数据就不会被覆盖了。所以在这里,DMA的配置就是外设地址写入ADC_DR这个寄存器的地址,存储器的地址可以在SRAM中定一个数组ADvalue,然后把ADvalue的地址当做存储器的地址。

之后数据宽度,因为ADC_DR和SRAM数组我们要的都是uint16_t的数据,所以数据宽度都是十六位的半字传输。

地址是否自增,从这个图里显然是外设地址不自增,存储器地址自增。

传输方向是外设站点到存储器站点。

传输计数器,这里通道有七个,所以计数七次。

计数器是否自动重装,ADC如果是单次扫描,那DMA的传输计数器可以不自动重装,转换一轮就停止。如果ADC是连续扫描,那DMA就可以使用自动重装,在ADC启动下一轮转换的时候,DMA也启动下一轮的转运,ADC和DMA同步工作。

触发选择,这里ADC_DR的值是在ADC单个通道转换完成后才会有效,所以DMA转运的时机需要和ADC单个通道转换完成同步,因此DMA的触发要选择ADC的硬件触发。

⚠️⚠️注意:ADC扫描模式在每个单独的通道转换完成后,没有任何标志位也不会触发中断 。所以我们程序不太好判断某一个通道转换完成的时机是什么时候。但是,虽然单个通道转换完成后,不产生任何标志位和中断,但是它应该会产生DMA请求去触发DMA转运。这部分内容手册里并没有详细描述,根据实际实验,单个通道的DMA请求肯定是有的。

一般来说,DMA最常见的用途就是配合ADC的扫描模式 。因为ADC扫描模式有个数据覆盖的特征,或者可以说这个数据覆盖的问题是ADC固有的缺陷。这个缺陷使ADC和DMA成了最常见的伙伴。ADC对DMA的需求是非常强烈的 ,而其它的一些外设使用DMAA可以提高效率是锦上添花的操作,但是不使用也是可以的,顶多是损失一些性能。但是这个ADC的扫描模式如果不使用DMA功能都会受到很大的限制。所以ADC和DMA的结合最为常见。

最后补充一下手册中的一个知识点,详情可以去手册上看看。

存储器包含两个位段区域 ,这两个位段区映射的外设寄存器和SRAM中全部的位 。这个位段区就相当于是位寻址 ,它把外设寄存器和SRAM中所有的位都分配了地址,你操作这个新的地址就相当于操作其中某一个位。因为32位的地址有99%都是空的,所以地址空间很充足,即使把每一位都单独编码,那也毫无压力。所以就存在了这样一个位段,用于单独操作寄存器或SRAM的某一位,位段区是另早那一个地方开辟了一段地址区域。其中SRAM位段区是2200开头的区域,外设寄存器的位段区是4200开头的区域。

9. 🚢嵌入式闪存

闪存被分为了很多页,它们的地址都是0800开头的。在闪存区的最后就是系统存储器和选项字节,这两个区域统称为信息块。下面这是闪存接口寄存器,这是外设的一部分,这个外设可以对闪存进行读写。

10. 🚢启动配置

配置BOOT0和BOOT1两个引脚来来选择程序从哪里启动

本节的内容到这里就结束了,下节来开始写代码。

相关推荐
HZU_Puzzle12 分钟前
【6】STM32·FreeRTOS·列表和列表项
stm32·单片机·嵌入式硬件
我真的太难了啊24 分钟前
学习QT第二天
开发语言·qt·学习
澜舟孟子开源社区32 分钟前
“AI玩手机”原理揭秘:大模型驱动的移动端GUI智能体
人工智能·科技·agi
城电科技1 小时前
城电科技|太阳能智能座椅对零碳园区建设有什么意义?
科技
醉陌离1 小时前
渗透测试学习笔记——shodan(3)
笔记·学习
流着口水看上帝2 小时前
JavaScript学习路线
学习
Ting丶丶2 小时前
安卓应用安装过程学习
android·学习·安全·web安全·网络安全
被猫枕的咸鱼2 小时前
项目学习:仿b站的视频网站项目03-注册功能
学习
@小博的博客2 小时前
C++初阶学习第十三弹——容器适配器和优先级队列的概念
开发语言·数据结构·c++·学习
Whappy0012 小时前
5.STM32之通信接口《精讲》之USART通信---实验串口接收程序
stm32·单片机·嵌入式硬件