【stm32】DMA的介绍与使用

DMA的介绍与使用

  • 1、DMA简介
  • 2、存储器映像
  • 3、DMA框图
  • 4、DMA基本结构
  • 5、DMA请求
  • 6、数据宽度与对齐
  • 7、数据转运+DMA(存储器到存储器的数据转运)
    • 程序编写:
  • 8、ADC连续扫描模式+DMA循环转运
    • DMA配置:
    • 程序编写:

1、DMA简介

  • DMA(Direct Memory Access)直接存储器存取
    • 可以直接访问STM32内部存储器,包括运行内存SRAM、程序存储器Flash和寄存器等
  • DMA可以提供外设和存储器或者存储器和存储器之间的高速数据传输,无须CPU干预,节省了CPU的资源
    • 外设:指的是外设寄存器(一般是外设数据寄存器DR,如ADC的数据寄存器、串口的数据寄存器)
    • 存储器:指运行内存SRAM和程序存储器Flash(存储变量数组和程序代码)
  • 12个独立可配置的通道: DMA1(7个通道), DMA2(5个通道)
  • 每个通道都支持软件触发和特定的硬件触发
    • 软件触发:一般使用存储器到存储器
    • 硬件触发:一般使用外设到存储器
  • STM32F103C8T6 DMA资源:DMA1(7个通道)

2、存储器映像

在这里插入图片描述
计算机系统的5大组成部分:运算器、控制器、存储器、输入设备和输出设备
运算器、控制器统称为CPU

3、DMA框图

在这里插入图片描述

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

4、DMA基本结构

在这里插入图片描述

  • 起始地址:外设端的起始地址和存储器端的起始地址(决定数据的方向)
  • 数据宽度:指定一次转运要按多大的数据宽度来进行(字节Byte[8位]、半字HalfWord[16位]和字Word[32位])
  • 地址是否自增:指定一次转运完成后,下一次转运,是不是要把地址移动到下一个位置去
  • 传输计数器:自减计数器,计数器值即DMA转运的次数 ,减到0后,自增的地址会恢复到起始地址的位置,以方便之后DMA开始新一轮的转运
  • 自动重装器:决定转运的模式,不重装为单次模式,重装为循环模式
    • 传输计数器减到0之后,是否要自动恢复到最初的值,如传输计数器值为5,如果不使用自动重装器,转运5次后,DMA转运结束,如果使用自动重装器,转运5次,计数器减到0后,会立即重装到初始值5,DMA再次开始转运数据
  • 触发控制:决定DMA需要在什么时机进行转运。
    • M2M(Memory to Memory)决定选择哪个触发源,M2M = 1,DMA选择软件触发(ps:软件触发和循环模式不能同时使用,因为软件触发就是把传输计数器清零,循环模式是清零后自动重装,如果同时使用,DMA无法停止);M2M = 0,DMA选择硬件触发(ADC,串口,定时器),硬件触发的转运,一般都与外设有关的转运,这些转运需要一定的时机,比如ADC转换完成、串口收到数据、定时时间到等,在硬件达到这些时机时,传一个信号过来,触发DMA进行转运。
  • 开关控制:调用DMA_Cmd函数,当给DMA使能后,DMA准备就绪,可以进行转运
    • DMA可以转运的条件:1、开关控制,DMA_Cmd必须使能;2、传输计数器必须大于0;3、触发源,必须有触发信号

注意:写传输计数器时,必须要先关闭DMA,再进行

5、DMA请求

在这里插入图片描述
DMA触发部分:
EN位:EN=0,数据选择器不工作,EN=1,数据选择器工作
软件触发后面跟个M2M位的意思:当M2M位=1时,选择软件触发

PS:想使用某个硬件触发源,必须使用它所在的通道;使用软件触发,通道可以任意选择

6、数据宽度与对齐

在这里插入图片描述
源端宽度比目标宽度小时:

  • 当源端和目标都是8位时,转运第一步,在源端的0位置,读数据B0,在自标的0位置,写数据B0,即把B0,从左边挪到右边,后续B1,B2,B3同理;
  • 当源端是8位,目标是16位,在源端读B0,在目标写00B0,之后,读B1,写00B1等,后续同理
    • 总结:若目标的数据宽度比源端的数据宽度大,在目标数据前面多出来的空位补0
  • 当源端是8位,目标是32位,同源端8位,目标16位处理方式相同

源端宽度比目标宽度大时:

  • 当源端是16位,目标是8位,读B1BO,只写入B0;读B3B2,只写入B2,
    • 总结:将高位舍弃
  • 其它方式依次类推

7、数据转运+DMA(存储器到存储器的数据转运)

在这里插入图片描述

将SRAM里的数组DataA,转运到另一个数组DataB中

  • 外设地址:DataA数组首地址 | 存储器地址:DataB数组的首地址
  • 数据宽度:两个数组的类型都是uint8_t,按8位的字节传输
  • 地址自增:数组DataA和DataB均自增
  • 转运方向:存储器向存储器转运
  • 传输计数器;转运7次,自动重装器不使用
  • 触发控制:软件触发

程序编写:

// myDMA.c
#include "stm32f10x.h"                  // Device header
/*
DMA 转运的3个条件
1、传输计数器大于0      Size > 0
2、触发源有触发信号 
3、DMA使能
*/uint16_t MyDMA_Size;void MyDMA_Init(uint32_t AddrA, uint32_t AddrB, uint16_t Size)
{MyDMA_Size = Size;// 开启DMA时钟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;                     // 传输模式,是否使用自动重装--不能和软件触发同时使用,选择不自动重装,计数减到0后停止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); // 初始化不立刻转运,使用MyDMA_Transfer函数进行转运
}
// 调用此函数,再次启动一次DMA转运
void MyDMA_Transfer(void)
{// 重新给传输计数器赋值,先把DMA失能DMA_Cmd(DMA1_Channel1, DISABLE);        // DMA失能DMA_SetCurrDataCounter(DMA1_Channel1, MyDMA_Size); // 传输计数器赋值DMA_Cmd(DMA1_Channel1, ENABLE);     // DMA 使能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();// DataA地址数据转运到DataB地址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);}
}

8、ADC连续扫描模式+DMA循环转运

在这里插入图片描述

左边为ADC扫描模式的执行流程:7个通道,触发一次后,7个通道依次进行AD转换,转换结果都放到ADC_DR数据寄存器中,需要在每个单独的通道转换完成后,进行一个DMA数据转运,并且目的地址进行自增,防止数据被覆盖。

DMA配置:

  • 外设地址写入ADC_DR寄存器的地址,存储器地址,可以在SRAM中定义一个数组ADValue,把ADValue的地址当做存储器的地址
  • 数据宽度:16位的半字传输
  • 地址是否自增;外设地址不自增,存储器地址自增
  • 传输方向:外设到存储器
  • 传输计数器:通道有7个,计数7次
  • 计数器是否自动重装,看ADC的配置,ADC如果是单次扫描,DMA的传输计数器可以不自动重装,如果ADC是连续扫描,DMA可以使用自动重装,在ADC启动下一轮转换时,DMA也启动下一轮转运,ADC和DMA同步工作
  • 触发选择:ADC_DR的值是在ADC单个通道转换完成后才会有效,所以DMA转运的时机,需要和ADC单个通道转换完成同步,DMA的触发要选择ADC的硬件触发

程序编写:

// AD.c
// 使用软件触发ADC采集数据--->ADC硬件触发DMA转运数据到存储器
#include "stm32f10x.h"                  // Device headeruint16_t AD_Value[4];// ADC连续扫描模式+DMA循环转运
void AD_Init(void)
{RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);RCC_ADCCLKConfig(RCC_PCLK2_Div6);GPIO_InitTypeDef GPIO_InitStructure;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_2 | GPIO_Pin_3;GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_Init(GPIOA, &GPIO_InitStructure);ADC_RegularChannelConfig(ADC1, ADC_Channel_0, 1, ADC_SampleTime_55Cycles5);ADC_RegularChannelConfig(ADC1, ADC_Channel_1, 2, ADC_SampleTime_55Cycles5);ADC_RegularChannelConfig(ADC1, ADC_Channel_2, 3, ADC_SampleTime_55Cycles5);ADC_RegularChannelConfig(ADC1, ADC_Channel_3, 4, ADC_SampleTime_55Cycles5);ADC_InitTypeDef ADC_InitStructure;ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;ADC_InitStructure.ADC_ContinuousConvMode = ENABLE; // 连续扫描ADC_InitStructure.ADC_ScanConvMode = ENABLE;  // 使用扫描模式ADC_InitStructure.ADC_NbrOfChannel = 4;ADC_Init(ADC1, &ADC_InitStructure);DMA_InitTypeDef DMA_InitStructure;DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)&ADC1->DR; // ADC DR的基地址,ADC采集到的数据存到此寄存器中DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord; // 16位DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;// 外设地址不需自增DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)AD_Value; // 转运数据到的目的地址DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord; // 16位DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;  // 存储器地址自增DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;DMA_InitStructure.DMA_BufferSize = 4;  //  传输次数-4个ADC通道DMA_InitStructure.DMA_Mode = DMA_Mode_Circular; // 循环模式DMA_InitStructure.DMA_M2M = DMA_M2M_Disable; // 硬件触发DMA_InitStructure.DMA_Priority = DMA_Priority_Medium;DMA_Init(DMA1_Channel1, &DMA_InitStructure); // ADC 硬件触发只在DMA1的通道1上DMA_Cmd(DMA1_Channel1, ENABLE);ADC_DMACmd(ADC1, ENABLE); // 开启ADC到DMA的输出ADC_Cmd(ADC1, ENABLE);ADC_ResetCalibration(ADC1);while (ADC_GetResetCalibrationStatus(ADC1) == SET);ADC_StartCalibration(ADC1);while (ADC_GetCalibrationStatus(ADC1) == SET);ADC_SoftwareStartConvCmd(ADC1, ENABLE);//ADC开始工作
}
//main.c
#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "OLED.h"
#include "AD.h"int main(void)
{OLED_Init();AD_Init();OLED_ShowString(1, 1, "AD0:");OLED_ShowString(2, 1, "AD1:");OLED_ShowString(3, 1, "AD2:");OLED_ShowString(4, 1, "AD3:");while (1){OLED_ShowNum(1, 5, AD_Value[0], 4);OLED_ShowNum(2, 5, AD_Value[1], 4);OLED_ShowNum(3, 5, AD_Value[2], 4);OLED_ShowNum(4, 5, AD_Value[3], 4);Delay_ms(100);}
}

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

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

相关文章

C++笔记之类三种的继承方式

C++笔记之类三种的继承方式 code review! 文章目录 C++笔记之类三种的继承方式1.《C++ Primer Plus》(第6版)中文版Page 5502.C++类继承方式与能否隐式向上转换的关系1.《C++ Primer Plus》(第6版)中文版Page 550 除基类私有成员变量外(基类公有成员变量和保护成员变量):…

Element-ui官方示例(Popover 弹出框)

Element-ui官方示例(Popover 弹出框),好用的弹出框。 使用 vue-cli3 我们为新版的 vue-cli 准备了相应的​Element 插件​,你可以用它们快速地搭建一个基于 Element 的项目。 使用 Starter Kit 我们提供了通用的项目模版&#…

gitLab配置ssh

1打开git命令行,创建秘钥 ssh-keygen -t rsa -b 4096 -C "用户名xxx.com" 2执行下面的命令查看公钥 cat ~/.ssh/id_rsa.pub 3#复制公钥到gitlab网址上ssh页面添加ssh的key(公钥) 4本地的git命令行中添加账户邮箱 git config -…

Windows7 X64 成功安装 .NET Framework 4.8 的两种方法

Windows7 X64 成功安装 .NET Framework 4.8 的两种方法 windows7系统SP1安装完成后,在安装某软件时,提示需要先安装4.6以上的版本net-framework包,正好电脑里有个net-framework4.8软件包,于是打算用上,可是在安装时&a…

JDK17下,使用SHA1算法报Certificates do not conform to algorithm constraints错误

JDK17从17.0.5开始,默认不再允许使用SHA1算法,如果引用的jar包或代码里使用了SHA1算法,会报以下错误。 Caused by: javax.net.ssl.SSLHandshakeException: Certificates do not conform to algorithm constraintsat java.base/sun.security.…

JS开发es8266板子,搞着玩-MAX7219模块 远程显示led字符串

JS开发es8266板子,搞着玩-MAX7219模块 板子为 esp8266 这里接了两个8x8 Led.Matrix espjs https://www.espruino.com/ 我是看了,这个文章 发现js可以开发esp板子的就尝试了下远程点灯,挺有意思就买了很多模块慢慢尝试 代码 这里我把wifi模块又包了一…

AI 视频工具合集

🐣个人主页 可惜已不在 🐤这篇在这个专栏AI_可惜已不在的博客-CSDN博客 🐥有用的话就留下一个三连吧😼 目录 前言: 正文: ​ 前言: AI 视频,科技与艺术的精彩融合。它借助先进的人工智能技术,为影像创…

力扣刷题-算法基础

hello各位小伙伴们,为了进行算法的学习,小编特意新开一个专题来讲解一些算法题 1.移除元素. - 力扣(LeetCode) 本题大概意思是给定一个数组和一个数val删除与val相同的元素,不要改变剩余元素的顺序,最后返回剩余元素的个数。 我们在这里使用双指针,这里的双指针并不是…

【OSCP Proving Grounds 靶场系列】Slort

作者:Eason_LYC 悲观者预言失败,十言九中。 乐观者创造奇迹,一次即可。 一个人的价值,在于他所拥有的。可以不学无术,但不能一无所有! 技术领域:WEB安全、网络攻防 关注WEB安全、网络攻防。我的…

【IPv6】IPv6 NAT66介绍

参考链接 IPv6-to-IPv6 Network Address Translation (NAT66) (ietf.org)https://datatracker.ietf.org/doc/id/draft-mrw-nat66-00.html IPv6 NAT66 NAT66,全称为Network Address Translation for IPv6 to IPv6,是一种用于IPv6网络的地址转换技术。在…

STM32-----I2C

1.基本原理: 上图是I2C的总线图和通讯协议图(就是I2C是怎么实现设备之间读写数据的) 下面主要介绍通讯协议的每一步: 1.发出开始信号: 一开始都为高电平为空闲状态。当SCL为高电平时,主机将SDA拉低即为发出开始信号&…

讲一讲Redis五大数据类型的底层实现

讲一讲Redis五大数据类型的底层实现 Redis五大数据类型的底层实现 Redis的五大数据类型分别是字符串(String)、列表(List)、哈希(Hash)、集合(Set)和有序集合(Zset&…

中药药材推荐系统

毕业设计还在发愁选题?又想实用又怕复杂?那这篇介绍你一定感兴趣! 今天为大家推荐一个基于Django框架开发的中药药材推荐系统,简洁易用,功能丰富,非常适合毕业设计。无论你是技术经验丰富的开发人员&#…

Jmeter监控服务器性能

目录 ServerAgent 安装 打开Jmeter ServerAgent 在Jmeter上监控服务器的性能比如CPU,内存等我们需要用到ServerAgent,这里可以下载我分享 ServerAgent-2.2.3.zip 链接: https://pan.baidu.com/s/1oZKsJGnrZx3iyt15DP1IYA?pwdedhs 提取码: edhs 安装…

【4.8】图搜索算法-BFS解单词接龙

一、题目 给 定 两 个 单 词 ( beginWord 和 endWord ) 和 一 个 字 典 , 找 到 从 beginWord 到 endWord 的最短转换序列的长度。转换需遵循如下规则: 1. 每次转换只能改变一个字母。 2. 转换过程中的中间单词必须是字…

JavaScript 网页设计案例:使用 Canvas 实现趣味打气球小游戏

JavaScript 网页设计案例:使用 Canvas 实现趣味打气球小游戏 在网页设计中,交互性和趣味性是吸引用户的重要因素。借助 JavaScript 和 HTML5 的 canvas 元素,我们可以轻松实现各种动画效果,今天将带你打造一个有趣的 打气球小游戏…

【银行科技岗】相关考试知识点总结及部分考题

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 一、**网络与安全**二、**软件开发与设计**三、**数据库与数据管理**四、**编程与系统**五、**计算机硬件与性能**六、**大数据与人工智能**七、**系统与应用**相关…

dfs +剪枝sudoku———poj2676

目录 前言 lowbit函数 数独 suduku 问题描述 输入 输出 问题分析 子网格位置 优化搜索顺序剪枝1 优化搜索顺序剪枝2 可行性剪枝 代码 前言 lowbit函数 这是一个利用二进制位运算取出二进制数最后一位’1‘的函数 数独 数独大家肯定都玩过,…

<<迷雾>> 第11章 全自动加法计算机(7)--部分自动化加法 示例电路

部分实现了自动化的连续加法电路. info::操作说明 增加了译码器模块, 把从内存中取数的步骤和装载/相加的步骤综合起来, 总共五步骤 存储器中已经提前预存了 5 个数. 如果地址计数器 AC 还没有清零, 则需要先清零. 闭合 K装载 开关, 断开 K相加 开关 将开关 K 连续按 5 次, 第…

SpringMVC后台控制端校验-表单验证深度分析与实战优化

前言 在实战开发中,数据校验也是十分重要的环节之一,数据校验大体分为三部分: 前端校验后端校验数据库校验 本文讲解如何在后端控制端进行表单校验的工作 案例实现 在进行项目开发的时候,前端(jquery-validate),后端,数据库都要进行相关的数据…