news 2026/6/10 17:16:56

DMA直接存储器存取

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
DMA直接存储器存取

1.DMA简介

注意:存储器到存储器的数据转运一般使用软件触发;外设到存储器的数据转运一般使用硬件触发

2.STM32的存储器映像

内核外设是NVIC和SysTick

3.DWA大致的内部结构

整体结构就是CPU+存储器组成的,寄存器是连接软件和硬件的桥梁

细节:

1.DCode专门访问Flash,系统总线访问其他东西

2.仲裁器:虽然多个通道可以独立转运数据,但是DMA总线只有一条,所有通道只能分时复用这一条总线,此时仲裁器就会根据优先级来分出使用顺序。

3.AHB从设备:DMA自身的寄存器(CPU通过其可以配置DMA)

4.DMA请求(用于硬件触发DMA数据转运):就是DMA的硬件触发源(触发DMA转运数据)

5.这里的Flash是只读的,不能通过总线访问;SRAM是运行内存,可以正常读写。

4.DMA基本结构(具体)

1.传输计数器(自减计数器):指定转运次数(注意:写传输计数器时必须先关闭DMA

2.自动重装器:决定是模式单次模式(不重装,即不会回到原来的次数)还是循环模式(重装,即会回到原来的次数)

3.触发控制:由M2M参数决定,为1时,DMA选择软件触发(连续触发DMA尽早将计数器清零,不能与循环模式同时使用,一般用于存储器到存储器的转运);为0时,DMA选择硬件触发(触发源可以为ADC、串口、定时器等,一般都与外设有关)

5.细节

1.DMA请求

M2M:数据选择器的控制位

EN:开关控制(为0时不工作,为1时工作)

注意:使用某个硬件触发源的话,就必须使用它所在的通道,而软件触发可以任意选择

选择哪个硬件触发源取决于哪个外设的DMA输出开启了

2.数据宽度与对齐

存在原因:数据转运的两个站点的数据宽度可能会不一样

1.如果目标的数据宽度比源端的数据宽度大,那就在目标数据前面补0

2.如果目标的数据宽度比源端的数据宽度小,就把多出来的高位舍弃掉

3.例子

1.数据转运+DMA(数组间的数据转运,相当于复制)

2.ADC扫描模式+DMA

6.实战代码

1.部分函数功能

//DMA初始化 void DMA_DeInit(DMA_Channel_TypeDef* DMAy_Channelx); void DMA_Init(DMA_Channel_TypeDef* DMAy_Channelx, DMA_InitTypeDef* DMA_InitStruct); void DMA_StructInit(DMA_InitTypeDef* DMA_InitStruct); void DMA_Cmd(DMA_Channel_TypeDef* DMAy_Channelx, FunctionalState NewState);//输出使能 void DMA_ITConfig(DMA_Channel_TypeDef* DMAy_Channelx, uint32_t DMA_IT, FunctionalState NewState); //中断输出使能 //设置数据寄存器(给传输数据寄存器写入转运次数) void DMA_SetCurrDataCounter(DMA_Channel_TypeDef* DMAy_Channelx, uint16_t DataNumber); //获取当前数据寄存器的值(剩余的转运次数) uint16_t DMA_GetCurrDataCounter(DMA_Channel_TypeDef* DMAy_Channelx); FlagStatus DMA_GetFlagStatus(uint32_t DMAy_FLAG);//获取标志位状态 void DMA_ClearFlag(uint32_t DMAy_FLAG);//清除标志位 ITStatus DMA_GetITStatus(uint32_t DMAy_IT);//获取中断状态 void DMA_ClearITPendingBit(uint32_t DMAy_IT);//清除中断挂起位

2.配置思路

1.RCC开启时钟(把DMA的时钟打开)

2.DMA初始化(外设和存储器站点的起始地址、数据宽度、地址是否自增、传输计数器等)

3.打开DMA(用DMA_Cmd使能)

注意:如果使用的是硬件触发,要开启对应的外设的触发信号的输出;如果需要DMA中断就调用DMA_ITConfig来开启中断输出

3.基本配置格式

//DMA数据转运 void MyDMA_Init(uint32_t AddrA,uint32_t AddrB,uint16_t Size) { MyDMA_Size=Size; //开启DMA的时钟(DMA是AHB总线上的设备) RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); //初始化DMA 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,ENABLE); } //DMA传输函数(调用一次就在进行一次DMA转运) //手动实现多次转运(也可以通过重装实现,但是软件触发和重装不能同时使用) 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);//手动清除标志位 } //DMA+AD多通道(AD扫描模式下需要DMA转运数据) //此处配置为ADC单次扫描模式+DMA单次转运 //注意:当模式配置成ADC连续扫描+DMA循环转运模式时,可以去掉AD_GetValue函数,然后在ADC校准完成后软件触发一次ADC即可实现连续转换,循环转运 uint16_t AD_Value[4]; void AD_Init(void) { //ADC都是APB2上的设备 RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1,ENABLE);//开启ADC1的时钟 //需要用到PA0口将可调的电压输出 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE); //开启DMA的时钟(DMA是AHB总线上的设备) RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); //配置ADCCLK RCC_ADCCLKConfig(RCC_PCLK2_Div6); //配置GPIO时钟 GPIO_InitTypeDef GPIO_InitStructure;//结构体定义 GPIO_InitStructure.GPIO_Mode=GPIO_Mode_AIN; //模拟输入(GPIO无效,即为ADC专属模式) GPIO_InitStructure.GPIO_Pin=GPIO_Pin_0|GPIO_Pin_1|GPIO_Pin_2|GPIO_Pin_3;//IO口 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 ADC_InitTypeDef ADC_InitStructure; ADC_InitStructure.ADC_Mode=ADC_Mode_Independent;//ADC模式(独立还是双模式) ADC_InitStructure.ADC_DataAlign=ADC_DataAlign_Right;//数据对齐 ADC_InitStructure.ADC_ExternalTrigConv=ADC_ExternalTrigConv_None; //外部触发转换选择(触发源)(此处为软件触发) ADC_InitStructure.ADC_ContinuousConvMode=DISABLE;//连续转换还是单次转换模式 ADC_InitStructure.ADC_ScanConvMode=ENABLE;//扫描还是非扫描模式(此处为扫描模式) ADC_InitStructure.ADC_NbrOfChannel=1; //扫描模式下总工会用到的通道数 ADC_Init(ADC1,&ADC_InitStructure); //初始化DMA(ADC扫描模式和DNA组合下的配置) DMA_InitTypeDef DMA_InitStructure; DMA_InitStructure.DMA_PeripheralBaseAddr=(uint32_t)&ADC1->DR;//外设站点的起始地址 DMA_InitStructure.DMA_PeripheralDataSize=DMA_PeripheralDataSize_HalfWord;//数据宽度 DMA_InitStructure.DMA_PeripheralInc=DMA_PeripheralInc_Disable;//是否自增(此处不自增) DMA_InitStructure.DMA_MemoryBaseAddr=(uint32_t)AD_Value;//存储器站点起始地址 DMA_InitStructure.DMA_MemoryDataSize=DMA_MemoryDataSize_HalfWord;//数据宽度 DMA_InitStructure.DMA_MemoryInc=DMA_MemoryInc_Enable;//是否自增(此处自增) DMA_InitStructure.DMA_DIR=DMA_DIR_PeripheralSRC;//传输方向(外设站点作为数据源) DMA_InitStructure.DMA_BufferSize=4;//传输计数器 DMA_InitStructure.DMA_Mode=DMA_Mode_Normal;//传输模式(是否重装)(此处不重装) DMA_InitStructure.DMA_M2M=DMA_M2M_Disable;//(软件触发还是硬件触发)(此处为软件触发) DMA_InitStructure.DMA_Priority=DMA_Priority_Medium;//优先级(此处中等优先级) DMA_Init(DMA1_Channel1,&DMA_InitStructure); DMA_Cmd(DMA1_Channel1,ENABLE); //开启ADC到DMA的输出 ADC_DMACmd(ADC1,ENABLE); ADC_Cmd(ADC1,ENABLE);//开启ADC //校准 ADC_ResetCalibration(ADC1);//复位校准 while(ADC_GetResetCalibrationStatus(ADC1)==SET);//获取复位校准状态 ADC_StartCalibration(ADC1);//开始校准 while(ADC_GetCalibrationStatus(ADC1)==SET);//获取开始校准状态 } //转换过程 void AD_GetValue(void) { DMA_Cmd(DMA1_Channel1,DISABLE);//DMA失能 DMA_SetCurrDataCounter(DMA1_Channel1,4); DMA_Cmd(DMA1_Channel1,ENABLE);//重新使能 ADC_SoftwareStartConvCmd(ADC1,ENABLE);//开启软件触发(单次模式下每次都要触发一下) //因为转运总在转换之后,所以直接等待转运完成即可 while(DMA_GetFlagStatus(DMA1_FLAG_TC1)==RESET);//等待标志位 DMA_ClearFlag(DMA1_FLAG_TC1);//手动清除标志位 }
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/10 1:33:47

32、使用FluidSynth实现卡拉OK播放器

使用FluidSynth实现卡拉OK播放器 1. 引言 在多媒体应用开发中,实现一个功能丰富的卡拉OK播放器是一个有趣且具有挑战性的任务。本文将介绍如何利用FluidSynth、Gtk、FFmpeg等库来实现一个卡拉OK播放器,该播放器可以播放背景视频,并在视频上动态显示歌词。 2. 整体思路 要…

作者头像 李华
网站建设 2026/6/10 14:11:27

Ivanti提醒注意 EPM 中严重的代码执行漏洞

聚焦源代码安全,网罗国内外最新资讯!编译:代码卫士美国IT软件公司 Ivanti 提醒用户修复位于端点管理器 (EPM) 中的一个新的严重漏洞CVE-2025-10573,它可导致攻击者远程执行代码。Ivanti 公司通过遍布全球7000多家组织机构向超过4万…

作者头像 李华
网站建设 2026/6/10 14:06:57

谷歌Gemini Enterprise存在漏洞,可导致企业数据遭暴露

聚焦源代码安全,网罗国内外最新资讯!编译:代码卫士最近,谷歌修复了 Gemini Enterprise中的一个漏洞。AI 安全公司 Noma Security 提到,该漏洞可用于获取企业敏感数据。该攻击被命名为 GeminiJack,无需任何用…

作者头像 李华
网站建设 2026/6/10 14:04:18

什么是天猫代运营?

天猫代运营,是品牌将整个天猫旗舰店的运营权,委托给专业第三方公司(通常称为TP,即TaoBao Partner)的商业合作模式。其本质是品牌在电商领域的 “专业分工”与“增长投资”。简单来说,它解决了品牌“不会做、…

作者头像 李华
网站建设 2026/6/10 14:07:23

Proxmark3实战:Mifare Ultralight C 3DES加密通信完全指南

还在为Mifare Ultralight C标签的3DES加密通信而困惑?本指南将带你从基础概念到实战操作,彻底掌握这一安全NFC标签的加密机制。通过Proxmark3的强大功能,你将能够轻松应对各种3DES认证场景。 【免费下载链接】proxmark3 Iceman Fork - Proxma…

作者头像 李华
网站建设 2026/6/10 14:03:16

从零开始:如何用 C# 开发一款媲美 “AnyTxt” 的文件内容搜索工具

起文件内容搜索工具,那么不得不提到“AnyTxt”,号称本地知识库检索的终极答案。唯一的不足可能就是索引更新机制,不能实时监视文件更改从而更新索引,最小定期更新间隔为半小时,容易导致cpu占用率高,毕竟是全…

作者头像 李华