STM32 DMA详解

news2025/1/9 15:12:28

1.DMA简介

DMA (Direct Memory Access) 直接存储器存取

DMA 可以提供外设和存储器或者存储器和存储器之间的高速数据传输,无须CPU干预,节省了CPU的资源( 比如想把Flash里的一批数据转运到SRAM里,需要软件触发,使用软件触发之后,DMA会以最快都速度把这批数据转运)如果进行外设到存储器的转运,就不能一股脑地转运了,因为外设的数据是有一定的时机的,这时就需要用硬件触发   (比如转运ADC的数据,就需要每个ADC通道转换完成后,硬件触发一次DMA再转运)触发一次转运一次,这样数据才是正确的,才是我们想要的效果,所以存储器到存储器的数据转运一般使用软件触发,外设到存储器要硬件触发

每个DMA通道硬件触发源是不一样的,要使用某个外设的硬件触发源,就必须使用它连接的那个通道

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

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

C8T6 DMA资源:DMA1 

2、存储器映像

const常量,存储在flash,则地址开头0x0800 0000,节省SRAM,若有很大的查找表或者字库,最好加const

定义一个临时变量,则地址开头0x2000 0000

外设寄存器的地址是固定的,例如ADC1->DR(4001 244C),

在数据手册中,起始地址+偏移地址

ADC1的地址0x4001 2400,DR的地址偏移为4C

 计算机系统5大组成部分:运算器,控制器,存储器,输入设备和输出设备。

3、DMA数据转运

 运算器和控制器一般会合在一起,叫做CPU,所以计算机的核心关键部分就是CPU和存储器

程序代码以及const常量存放在Flash区域

变量存放在RAM区域

存储器的重要知识点:存储的内容和存储器的地址。外设也是存储器

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

RAM:随机存储器,易失,掉电丢失

ROM 分为3块,分别为程序存储器Flash(主闪存),系统存储器和选项字节

RAM:运行内存SRAM,外设寄存器,内核外设寄存器。

4、DMA框图 

 在上图当中,数据从Flash或者SRAM寄存器当中传到总线矩阵当中,传到总线矩阵之后又开始分为通道1、2、3再进行传输

A.寄存器是一种特殊的存储器,一方面,CPU可以对寄存器进行读写,就像读写运行内存一样。另一方面,寄存器的每一位背后,都连接了一根导线,可以用于控制外设电路的状态

B.比如:置引脚的高低电平,导通和断开开关,切换数据选择器,或者多位结合起来,当作计数器,数据寄存器等。所以寄存器是连接软件和硬件的桥梁,软件读写寄存器,就相当于再控制硬件的执行。


C.所以使用DMA进行数据转运就相当于:从某个地址取内容,再放到另一个地址里。为了高效有条理地访问存储器,STM32设计了总线矩阵,左端是主动单元,也就是存储器的访问权。右边是被动单元,只能被左边的主动单元读取。主动单元包括内核的DCode和系统总线,可以访问右边的存储器,DCode专门访问Flash,系统总线访问其他东西。

D.由于DMA要访问数据,所以DMA也必须要有访问的主动权主动单元除了内核CPU,剩下的就是DMA了,每个DMA通道可以分别设置他们转运数据的源地址和目的地址,这样他们就可以各自独立的进行工作了。

D.仲裁器的作用:虽然每个通道可以独立进行设置,但是DMA总线只有一条,所以所有的通道都只能分时复用这一条DMA总线。如果产生了冲突,就会由仲裁器,根据通道的优先级决定谁先用谁后用。

E.在总线矩阵里,也会有仲裁器,如果DMA和CPU都要访问同一个目标,那么DMA就会暂停CPU的访问,防止冲突。不过。此时总线仲裁器仍然会保证CPU得到一般的总线带宽,使CPU能够正常的工作。


F.

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

DMA请求就是触发的意思,右边的触发源就是各个外设,DMA请求就是DMA的硬件触发源,比如ADC转换完成,串口就受到数据等。

需要触发DMA转运数据的时候就通过DMA请求线路向DMA发出硬件触发信号,之后DMA就可以执行数据转运的动作了。

DMA的各个部分和作用:

DMA总线:用于访问各个存储器,内部的多个通道可以进行独立的数据转运

仲裁器:用于调度各个通道,防止产生冲突

AHB从设备:用于配置DMA参数

DMA请求:用于硬件触发DMA的数据转运

STM32的Flash是RAM的一种,如果通过总线直接访问的话,无论是CPU还是DMA,都是只读的,只能读取数据而不能写入,如果DMA的目的地址填写了Flash区域,转运时就会出错

也可以配置Flash接口控制器对Flash进行写入。

5、DMA基本结构 

画圈的是数据转换的站点,左边是外设寄存器站点,右边是存储器站点包括Flash和SRAM。

STM32存储器一般特指Flash和SRAM,不包含外设寄存器,外设寄存器一般直接称为外设,所以就是外设到存储器,存储器到存储器这样来描述。

外设和存储器参数的作用:

外设和存储器起始地址作用:决定了数据从哪里来,到哪里去。

数据宽度:作用是指定一次转运要按多大的数据宽度来进行,可以选择字节Byte,半字HalhWord和字Word。字节是8位,半字是16位

地址是否自增作用:指定一次转运完成后,下一次转运要把地址移动到下一个位置去,相当于指针P++ 。比如ADC扫描模式,用DMA进行数据转运,外设地址是ADC_DR寄存器,寄存器的地址不用自增,不然就会移位到其他寄存器里面了。存储器地址需要自增,每转运一次,就往后移动一位,不然会覆盖上一个数据。

进行存储器到存储器的转运,就需要把其中一个存储器的地址放在外设的这个站点。只要在存储器里写Flash或SRAM的地址,他就会去这两个地方找数据。

站点虽然叫外设存储器,并不是说只能写寄存器的地址只是一个名字而已。

 传输计数器:

用来指定总共需要转运几次,是一个自减计数器,比如写入5,那DMA就只能进行5次数据转运,转运过程中,每转运一次,计数器就会减1,当传输计数器减到0之后,DMA就不会进行数据转运了,转运地址也会恢复为起始地址,方便下一次转运。

自动重装器:

作用:传输计数器减到0之后,是否要恢复到最初的值,比如最初的值是5,如果不使用自动重装器,那么转运5次之后就DMA结束。使用自动重装器,转运5次,计数器减到0之后,计数器就会立即重装到初始值5. 不重装,就是单次模式,重装就是循环模式

比如说要转运一个数组,一般是单次模式,转运一轮就结束了。如果是ADC扫描模式+连续转换,为了配合ADC。DMA也需要使用循环模式。和ADC差不多,都是指定一轮工作完成后,是不是立即开始下一轮工作

DMA触发控制:

决定DMA在什么时机进行转运,触发源有硬件触发和软件触发,具体选择哪个有M2M参数决定。

M2M就是(Memory to Memory)即存储器到存储器,选择1DMA就会选择软件触发

软件触发执行逻辑:以最快的速度,连续不断地触发DMA,争取快速完成传输计数器清零,完成这一轮的转换工作,给0就是硬件触发。

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

6、开关控制:

使用 DMA_Cmd函数,给DMA使能后,DMA准备就绪,可以进行转运

 DMA转运的条件:

1.开关控制,DMA_Cmd必须使能

2.传输计数器必须大于0

3.触发源必须有触发信号,触发一次,转运一次,传输计数器自减一次。当传输计数器等于0,且没有自动重装时,无论是否触发,DMA都不会进行转运,此时需要DMA_Cmd给DISABLE关闭DMA再为传输计数器写入一个大于0的数,再DMA_Cmd,给ENABLE,开启DMA才能继续工作。在写传输计数器时,必须要先关闭DMA,在进行写入,不能开启时写入EN=0时不工作

硬件触发注意事项:使用相应的硬件触发需要选择对应的通道,不然就触发不了,选择哪个触发源由对应的外设是否开启DMA输出决定的。

比如要使用ADC1,就需要使用ADC_DMACmd函数进行使能,必须使用这个库函数开启ADC1这一路输出才有效。使用定时器3,会有TIM_DMACmd函数用来进行DMA输出控制,使用哪一个外设触发源,取决于把哪个通道开启了,全开启理论上都可以使用

开关控制开启之后,这七个触发源进入到仲裁器进行优先级判断,最终产生内部DMA1请求,优先级判断与中断类似,序号越小优先级越高,也可以在程序中配置优先级。

数据宽度与对齐:

如果转运的数据宽度一样,就是正常的一个个转运,如果数据宽度不一样,

比如源宽度是8位,目标宽度是16位,传输B0时,需要在前面高位补上0,传输结果就是00B0.

比如源宽度是16位,目标宽度是8位,传输B1B0时,需要把高位舍去,传输结果就是B0.

7、源码

MyDMA.c

#include "stm32f10x.h"                  // Device header

uint16_t MyDMA_Size;

void MyDMA_Init(uint32_t AddrA, uint32_t AddrB, uint16_t Size)
{
	MyDMA_Size = Size;
	
	RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);
	
	DMA_InitTypeDef DMA_InitStructure;
  //前面三个是在调节外设站点
	DMA_InitStructure.DMA_PeripheralBaseAddr = AddrA;    //发送基地址
	DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;  //发送长度单位
	DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Enable;   //基地址是否是自增
  //后面是在设置被传送的基地址
	DMA_InitStructure.DMA_MemoryBaseAddr = AddrB;    //设置目的地址
	DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;   //设置数据宽度,记住在这里是存储器数据宽度
	DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;  //地址自动增加
	DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;    //设置传输方向,可以设置成外设到存储器,存储器到存储器等
	DMA_InitStructure.DMA_BufferSize = Size;  //待定传输的数据量
	DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;  //设置选择的传输模式,可以选择一次传输还是循环传输
	DMA_InitStructure.DMA_M2M = DMA_M2M_Enable;  //设置数据传输从存储器到存储器
	DMA_InitStructure.DMA_Priority = DMA_Priority_Medium;  //设置仲裁器的优先级
	DMA_Init(DMA1_Channel1, &DMA_InitStructure);
	
	DMA_Cmd(DMA1_Channel1, DISABLE);  //先中断传输,到后面再进行
}



void MyDMA_Transfer(void)
{
	DMA_Cmd(DMA1_Channel1, DISABLE);
  //函数再重置自动重装器,然后让DMA不断进行
	DMA_SetCurrDataCounter(DMA1_Channel1, MyDMA_Size);
	DMA_Cmd(DMA1_Channel1, ENABLE);
	
  //等待转完成
	while (DMA_GetFlagStatus(DMA1_FLAG_TC1) == RESET);
	DMA_ClearFlag(DMA1_FLAG_TC1);   //清除标志位,标志位需要手动清除
}

main.c代码

#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "OLED.h"
#include "MyDMA.h"

uint8_t DataA[] = {0x01, 0x02, 0x03, 0x04};
uint8_t DataB[] = {0, 0, 0, 0};

int main(void)
{
	OLED_Init();
	
	MyDMA_Init((uint32_t)DataA, (uint32_t)DataB, 4);
	
	OLED_ShowString(1, 1, "DataA");
	OLED_ShowString(3, 1, "DataB");
	OLED_ShowHexNum(1, 8, (uint32_t)DataA, 8);
	OLED_ShowHexNum(3, 8, (uint32_t)DataB, 8);
		
	while (1)
	{
		DataA[0] ++;
		DataA[1] ++;
		DataA[2] ++;
		DataA[3] ++;
		
		OLED_ShowHexNum(2, 1, DataA[0], 2);
		OLED_ShowHexNum(2, 4, DataA[1], 2);
		OLED_ShowHexNum(2, 7, DataA[2], 2);
		OLED_ShowHexNum(2, 10, DataA[3], 2);
		OLED_ShowHexNum(4, 1, DataB[0], 2);
		OLED_ShowHexNum(4, 4, DataB[1], 2);
		OLED_ShowHexNum(4, 7, DataB[2], 2);
		OLED_ShowHexNum(4, 10, DataB[3], 2);
		
		Delay_ms(1000);
		
		MyDMA_Transfer();
		
		OLED_ShowHexNum(2, 1, DataA[0], 2);
		OLED_ShowHexNum(2, 4, DataA[1], 2);
		OLED_ShowHexNum(2, 7, DataA[2], 2);
		OLED_ShowHexNum(2, 10, DataA[3], 2);
		OLED_ShowHexNum(4, 1, DataB[0], 2);
		OLED_ShowHexNum(4, 4, DataB[1], 2);
		OLED_ShowHexNum(4, 7, DataB[2], 2);
		OLED_ShowHexNum(4, 10, DataB[3], 2);

		Delay_ms(1000);
	}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/457974.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

NAT网络地址转换技术入门到详解

本文目录 1、NAT简介1.1、SNAT 和IP伪装(Masquerade)1.2、DNAT1.3、Full NAT (也称为Full Cone NAT)1.4、PAT (也称为NAPT) 2、如何通过iptables将一台多网卡的主机配置成NAT路由器3、汇总 本文会从NAT的简介入手,详解NAT技术本身,通过本文,你…

巧用千寻位置GNSS软件| 电力线勘测如何实现?

正如大家所知,电力线勘测是在做电力线路设计之前对设计线路沿途自然环境进行勘察测量,最后把手簿测量数据在电脑端经过转换输出为电力软件专用格式数据的专用功能。 那么在千寻位置GNSS软件中该如何操作完成电力线的勘察测量呢? 点击【测量】…

市场岗位都在通缩,Framework开发就业环境怎么样?

随着 Android 设备的普及和应用领域的不断扩大,Android Framework 开发需求量将会持续增长,并且会越来越多地向行业、企业级应用和系统优化等方向发展。以下是一些 Android Framework 开发相关的应用场景: 特定垂直领域的智能设备&#xff1…

写最好的Nacos Server稳定版(nacos-server-2.1.1)在Centos、Docker和Windows上安装部署(单机、集群)教程

写最好的Nacos Server稳定版(nacos-server-2.1.1)在Centos、Docker和Windows上安装部署(单机、集群)教程 一、前言二、Nacos Server在 Centos7 安装部署(单机模式)2.1 下载 nacos-server-2.1.1 安装包2.1.1…

Matplotlib绘图库的高级使用

Matplotlib绘图库的高级使用 Matplotlib的三层结构容器层辅助显示层图像层 Matplotlib的绘图配置设置画布属性绘图保存自定义x与y刻度解决中文显示异常网格显示多次plot绘图标记显示图例多个坐标系显示 Matplotlib的三层结构 Matplotlib从层次结构上分,可以分为三层…

在线安装QT5.15.2+VS2019-16.11.26

在线安装QT5.15.2VS2019-16.11.26 一、安装QT5 官方下载: https://download.qt.io/archive/online_installers/4.5/ 选择【qt-unified-windows-x64-4.5.2-online.exe】 登录账户 需要提前注册,过程省略。 安装位置(自定义) …

GitHub Repo

GitHub Repo 之前笔记写了 git 和 gitup(pullpush),这里记一下 giehub repo 二三事。 权限 我不是很确定 github 的企业版是什么样的,不过我们用的是 gitlab 的企业版,这个是需要通过 vpn 才能连接的,如…

【C语言】基础语法6:字符串和字符处理

上一篇:数组和指针 下一篇:文件操作 ❤️‍🔥前情提要❤️‍🔥   欢迎来到C语言基本语法教程   在本专栏结束后会将所有内容整理成思维导图(结束换链接)并免费提供给大家学习,希望大家纠错…

redis7 安装 与 启动

文章目录 1. redis 的 概述2. redis 的 安装3. redis 的启动4. redis 的卸载 1. redis 的 概述 redis : 是 远程 词典服务器 ,是 一个基于内存的 键值型 Nosql 数据库. 官方解释 : Remote Dictionary Server(远程字典服务)是完全开源的,使用ANSIC语言编写…

QMS-云质说质量 - 6 中小企业常用的结构化问题解决方法有哪些?

云质QMS原创 转载请注明来源 作者:王洪石 引言 爱因斯坦如何解决问题 面对问题时,有的人可能很盲目地开始行动,干到一定程度,却突然发现自己所做的是无用功。 有人问过科学巨匠爱因斯坦,如果给他一个关系到他生命的问…

R基础函数概览(一)

rep 函数形式:rep(x, time , length , each ,) 参数说明: x:代表的是你要进行复制的对象,可以是一个向量或者是一个因子。 times:代表的是复制的次数,只能为正数。负数以及NA值都会为错误值。复制是指的…

[oeasy]python0139_尝试捕获异常_ try_except_traceback

尝试捕获异常 回忆上次内容 变量相加 整型数字变量可以相加字符串变量也可以拼接 但是 字符串 和 整型数字整型数字 和 字符串不能相加 怎么办? 转格式int(“1”)str(2) 可是 如果输入的苹果数量是 字符串"abc" int(“abc”)会发生什么?&…

通信算法之149:EVM测量

1.星座图 h scatterplot(sqrt(sps)*txSig(sps*span1:end-sps*span),sps,offset); hold on scatterplot(rxSigFilt(span1:end-span),n,offset,bx,h) scatterplot(dataMod,n,offset,r,h) legend(Transmit Signal,Received Signal,Ideal,location,best) 2. 眼图 Eye Diagram D…

华为OD机试真题(Java),简单密码(100%通过+复盘思路)

一、题目描述 现在有一种密码变换算法。 九键手机键盘上的数字与字母的对应: 1--1, abc--2, def--3, ghi--4, jkl--5, mno--6, pqrs--7, tuv--8 wxyz--9, 0--0,把密码中出现的小写字母都变成九键键盘对应的数字,如:a …

C++ -4- 类和对象(下)

文章目录 1.初始化列表什么是初始化列表?初始化列表的 意义及使用 2.explicit关键字单参数构造函数(C98)多参数的构造函数(C11)(了解) 3.static静态成员静态成员变量与静态成员函数静态成员变量…

Java并发(三)----创建线程的三种方式及查看进程线程

一、直接使用 Thread // 创建线程对象 Thread t new Thread() {public void run() {// 要执行的任务} }; // 启动线程 t.start(); 例如: // 构造方法的参数是给线程指定名字,推荐 Thread t1 new Thread("t1") {Override// run 方法内实现…

Codeforces Round 864 (Div. 2)(A~D)

A. Li Hua and Maze 给出两个不相邻的点,最少需要堵上几个方格,才能使得两个方格之间不能互相到达。 思路:显然,对于不邻任何边界的方格来说,最少需要的是4,即上下左右都堵上;邻一个边界就-1&a…

Python樱花树

文章目录 前言一、Turtle基础1.1 Turtle画板1.2 Turtle画笔1.3 Turtle画图1.4 Turtle填色1.5 Turtle写字 二、Python樱花树2.1 樱花类2.2 樱花树2.3 主函数2.4 程序分析2.5 樱花林 尾声 前言 粉色系最爱!Python樱花树等你获取~ 哈喽小伙伴们好久不见啦,…

几何感知Transformer用于3D原子系统建模

基于机器学习的方法在预测分子能量和性质方面表现出很强的能力。分子能量至少与原子、键、键角、扭转角和非键原子对有关。以前的Transformer模型只使用原子作为输入,缺乏对上述因素的显式建模。为了减轻这种限制,作者提出了Moleformer,这是一…

ChatGPT课程送账号啦,让你成为新生代AI程序员

ChatGPT能帮助程序员 解决哪些具体问题? 程序员在日常工作中可能会遇到各种各样的问题,如语法错误、逻辑问题、性能问题等等。 不同业务场景的问题,都可以利用ChatGPT获取各自场景下的知识,并使用ChatGPT提供的代码示例和问题解…