STM32学习笔记(七) —— DMA传输(MTM)

DMA,全称是Direct Memory Access(直接内存访问)。可以在存储器和存储器之间或者外设和存储器之间传输数据,而不需要CPU的干预,这样可以节省CPU的资源,提高工作效率。

1.功能框图

在这里插入图片描述

STM32F103RCT6有两个DMA控制器,需要DMA传输的时候,外设会向DMA发送请求,DMA也会进行应答,随后会进行DMA传输。

2.DMA通道

在这里插入图片描述

在这里插入图片描述

两个DMA控制器一共有12个通道(DMA1有7个,DMA2有5个),上面图中列出了每个外设的请求与DMA通道的对应关系。当有多个DMA通道请求时,由于每次只能响应一个DMA通道请求,所以DMA仲裁器会根据软件优先级以及硬件优先级来确定先响应哪个通道。

3.优先级
1° 软件优先级可以在DMA_CCRx寄存器中设置,一共有最高优先级、高优先级、中等优先级、低优先级这4个等级。
2° 硬件优先级由通道号决定,通道号小的优先级高于通道号大的,比如通道2优先级高于通道4.
DMA仲裁器会优先对比软件优先级,软件优先级相同时再对比硬件优先级。

4.DMA传输模式
DMA传输的数据量可以通过DMA_CNDTRx寄存器最大编程为65535,每次传输后会递减,在传输了一半的数据和数据传输完成后都会有相应的标志位标识,并且可以使能相应的中断。
1° 当通道配置为非循环模式时,完成数据传输后将不再产生DMA操作,如果要开始新的DMA传输,需要在关闭DMA通道的情况下,重新配置要传输的数据个数(重新配置DMA_CNDTRx寄存器);
2° 在循环模式下,最后一次传输结束时,DMA_CNDTRx寄存器会被自动重载为初始数值。
3° 存储器到存储器模式。DMA的传输方向可以是存储器到外设、外设到存储器,也可以是存储器到存储器。这种情况下需要借用外设端口,当使用外设通道代表存储器时,通道可以随意选择,注意不能与循环模式同时使用。

5.软件编程
1° 在内存中开辟两个数组空间 mData_Buffer and pData_Buffer
2° 向 mData_Buffer 中存入数据
3° 使用 DMA 将 mData_Buffer 中数据移动到 pData_Buffer 中

/* 全局变量 */
char mData_Buffer[7] = "eckard";
char pData_Buffer[7] = "before";/* 在while循环之前添加 *///开启 DMA1 时钟, 使用外设之前一定要开启时钟RCC->AHBENR |= (uint16_t)0x01 << 0;//清除 CCR 寄存器DMA1_Channel1->CCR &= (uint16_t)0xffff8000;//数据从存储器读, 因为我们这里是要验证存储器到存储器模式DMA1_Channel1->CCR |= (uint16_t)0x01 << 4;//不执行循环操作, 注意存储器到存储器模式不能与循环模式同时使用DMA1_Channel1->CCR |= (uint16_t)0x00 << 5;//外设地址增量操作,地址递增的意思是传输完当前地址中的数据后,地址变为下一个要传输的数据的地址DMA1_Channel1->CCR |= (uint16_t)0x01 << 6;//存储器地址增量操作DMA1_Channel1->CCR |= (uint16_t)0x01 << 7;//外设数据宽度8位DMA1_Channel1->CCR |= (uint16_t)0x00 << 8;//存储器数据宽度8位DMA1_Channel1->CCR |= (uint16_t)0x00 << 10;//通道优先级低,这里的优先级是软件设置的优先级,硬件优先级由通道号决定的DMA1_Channel1->CCR |= (uint16_t)0x00 << 12;//启动存储器到存储器模式DMA1_Channel1->CCR |= (uint16_t)0x01 << 14;//数据传输数量,每次传输后会递减,如果在循环模式下,最后一次传输结束时,此寄存器会被自动重载为初始数值,比如7DMA1_Channel1->CNDTR = (uint16_t)7;//外设地址,我们上述设置的是从存储器读,所以数据传输的方向是mData_Buffer到pData_Buffer, 此地址为目标地址DMA1_Channel1->CPAR  = (uint32_t)pData_Buffer;//存储器地址, 此地址是源地址DMA1_Channel1->CMAR  = (uint32_t)mData_Buffer;printf("transfer start.\r\n");printf("mData_Buffer = %s\r\n", mData_Buffer);printf("pData_Buffer = %s\r\n", pData_Buffer);printf("\r\n");//开启通道DMA1_Channel1->CCR |= (uint16_t)0x01 << 0;//等待传输完成while(DMA1->ISR & ((uint32_t)0x01 << 1));/* 清除传输完成标志 */DMA1->IFCR |= (uint32_t)0x01 << 1;printf("transfer completed.\r\n");printf("mData_Buffer = %s\r\n", mData_Buffer);printf("pData_Buffer = %s\r\n", pData_Buffer);

将程序下载后,打开串口调试助手,可以查看打印的信息。

在这里插入图片描述

上述使用的是寄存器编程,下面使用STM32CubeMX来初始化DMA

在这里插入图片描述

在这里插入图片描述

生成代码后,发现新增了dma.c文件,已经完成了DMA的初始化

在这里插入图片描述

在main函数中也有调用

在这里插入图片描述

STM32CubeMX只能帮助我们初始化外设的配置,应用程序需要我们自己编写,我们这里需要调用HAL_DMA_Start函数来开启DMA传输

/* 全局变量 */
char mData_Buffer[7] = "eckard";
char pData_Buffer[7] = "before";/* 在while循环之前添加 */printf("transfer start.\r\n");printf("mData_Buffer = %s\r\n", mData_Buffer);printf("pData_Buffer = %s\r\n", pData_Buffer);printf("\r\n");/* 开启DMA传输 */HAL_DMA_Start (&hdma_memtomem_dma1_channel1, (uint32_t)mData_Buffer, (uint32_t)pData_Buffer, 7);/* 等待传输完成 */while(__HAL_DMA_GET_FLAG(&hdma_memtomem_dma1_channel1, __HAL_DMA_GET_TC_FLAG_INDEX(&hdma_memtomem_dma1_channel1)) == RESET);/* 清除传输完成标志 */__HAL_DMA_CLEAR_FLAG(&hdma_memtomem_dma1_channel1, __HAL_DMA_GET_TC_FLAG_INDEX(&hdma_memtomem_dma1_channel1));printf("transfer completed.\r\n");printf("mData_Buffer = %s\r\n", mData_Buffer);printf("pData_Buffer = %s\r\n", pData_Buffer);

将程序下载后,打开串口调试助手,可以查看到打印的信息与上述采用寄存器编程时是一样的。

本例程代码可以在HAL库工程模板这一章节的最后,百度网盘链接分享处获取

以上是通过开发板进行实际验证的,下面使用软件仿真,

我们进入调试界面,打开串口窗口,然后点击运行( 前面章节有提到,所以本篇以及后续章节都不再重复提及 )

运行结果如下,与在开发板上验证的结果一致。

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

力扣773. 滑动谜题(BFS)

Problem: 773. 滑动谜题 文章目录 题目描述思路解题方法复杂度Code 题目描述 思路 由于题目提到最小步数&#xff0c;则可以使用BFS来穷举出最小的结果 1.转换为BFS问题&#xff1a;由于0代表空着的可以移动的位置&#xff0c;所以我们只需要从当前位置和0的相邻位置移动从而转…

代码随想录刷题笔记-Day20

1. 二叉树的最近公共祖先 236. 二叉树的最近公共祖先https://leetcode.cn/problems/lowest-common-ancestor-of-a-binary-tree/ 给定一个二叉树, 找到该树中两个指定节点的最近公共祖先。 百度百科中最近公共祖先的定义为&#xff1a;“对于有根树 T 的两个节点 p、q&#x…

ARM体系在linux中的中断抢占

上一篇说到系统调用等异常通过向量el1_sync做处理&#xff0c;中断通过向量el1_irq做处理&#xff0c;然后gic的工作都是为中断处理服务&#xff0c;在rtos中&#xff0c;我们一般都会有中断嵌套和优先级反转的概念&#xff0c;但是在linux中&#xff0c;中断是否会被其他中断抢…

Cannot invoke “java.sql.Connection.prepareStatement(String)“ because “conn“

下载sqlite-jdbc&#xff0c;放在目录下&#xff0c;然后IDEA右键jar文件选择“加入库”即可解决 Central Repository: org/xerial/sqlite-jdbc/3.36.0.1

JavaScript 的location 对象API 介绍

JavaScript 中&#xff0c;location 对象提供了访问当前页面 URL 相关信息的属性和方法。通过 location 对象&#xff0c;我们可以获取当前页面的 URL、查询参数、锚点等信息&#xff0c;并且可以使用 JavaScript 来修改当前页面的 URL。 以下是一些 location 对象的常用属性和…

设计模式二:代理模式

1、什么是动态代理 可能很多小伙伴首次接触动态代理这个名词的时候&#xff0c;或者是在面试过程中被问到动态代理的时候&#xff0c;不能很好的描述出来&#xff0c;动态代理到底是个什么高大上的技术。不方&#xff0c;其实动态代理的使用非常广泛&#xff0c;例如我们平常使…

学习求余

题目描述 白浅妹妹今天学习了求余运算&#xff0c;她很好奇求余运算和乘法运算结合起来会是什么样子&#xff0c;于是她设计了这样一道题目。 给定数字 n&#xff0c;你可以任选一个数字 k(1 ≤ k ≤ n)&#xff0c;然后计算出 n%k 的值(其中% 为求余运算)&#xff0c;记为 q…

php伪协议之phar

一.phar协议 用于将多个 PHP 文件、类、库、资源&#xff08;如图像、样式表&#xff09;等打包成一个单独的文件。这个归档文件可以像其他 PHP 文件一样被包含&#xff08;include&#xff09;或执行。PHAR 归档提供了一种方便的方式来分发和安装 PHP 应用程序和库&#xff0c…

【RT-DETR有效改进】可变形大核注意力 | Deformable-LKA适用于复杂背景或不同光照场景

👑欢迎大家订阅本专栏,一起学习RT-DETR👑 一、本文介绍 本文给大家带来的改进内容是Deformable-LKA(可变形大核注意力)。Deformable-LKA结合了大卷积核的广阔感受野和可变形卷积的灵活性,有效地处理复杂的视觉信息。这一机制通过动态调整卷积核的形状和大小来适…

Sui主网升级至V1.18.1版本

Sui主网现已升级至V1.18.1版本&#xff0c;同时Sui协议升级至36版本。其他升级要点如下所示&#xff1a; #15794 解析错误不再停止编译&#xff0c;并且后续编译阶段的诊断信息也可能包含在编译结果中&#xff0c;所以开发者可能会看到比以前更多的编译器诊断信息。 #12337 …

蓝桥杯嵌入式STM32G431RBT6知识点(主观题部分)

目录 1 前置准备 1.1 Keil 1.1.1 编译器版本及微库 1.1.2 添加官方提供的LCD及I2C文件 1.2 CubeMX 1.2.1 时钟树 1.2.2 其他 1.2.3 明确CubeMX路径&#xff0c;放置芯片包 2 GPIO 2.1 实验1&#xff1a;LED1-LED8循环亮灭 ​编辑 2.2 实验2&#xff1a…

MySQL篇之MVCC

一、什么是MVCC 全称 Multi-Version Concurrency Control&#xff0c;多版本并发控制。指维护一个数据的多个版本&#xff0c;使得读写操作没有冲突。 事务5查询的记录是哪个事务版本的记录呢&#xff1f;MVCC的具体实现&#xff0c;主要依赖于数据库记录中的隐式字段、undo lo…