news 2026/4/3 5:53:40

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/4/2 16:31:59

选 AI 智能体开发公司?合肥玄微子科技有限公司的思路可参考

随着 AI 技术落地加速,企业对 AI 智能体的需求持续增长,但市场上 AI 智能体开发公司资质参差不齐,如何筛选可靠合作方成为企业关注的核心问题。可靠的 AI 智能体开发公司需具备技术落地能力、行业深耕经验与完整服务体系,而玄微子…

作者头像 李华
网站建设 2026/3/31 20:52:15

Vue.js:现代前端开发的革新与实践

引言:Vue的崛起与核心价值在当今前端开发领域,Vue.js凭借其渐进式框架特性和优雅的设计哲学,已成为构建高效Web应用的标杆工具。自2014年诞生以来,Vue通过响应式系统、组件化架构和丰富的生态系统,赋能开发者从简单页面…

作者头像 李华
网站建设 2026/4/2 18:20:52

神经网络可视化终极指南:5步掌握交互式深度学习实验

神经网络可视化终极指南:5步掌握交互式深度学习实验 【免费下载链接】playground Play with neural networks! 项目地址: https://gitcode.com/gh_mirrors/pl/playground 想要在浏览器中轻松调试神经网络参数吗?gh_mirrors/pl/playground项目为你…

作者头像 李华
网站建设 2026/4/2 6:39:09

终极指南:如何用Lago开源计费系统快速搭建企业级计费平台

终极指南:如何用Lago开源计费系统快速搭建企业级计费平台 【免费下载链接】lago Open Source Metering and Usage Based Billing 项目地址: https://gitcode.com/GitHub_Trending/la/lago Lago开源计费系统为企业提供了强大的使用计量和基于事件的计费解决方…

作者头像 李华
网站建设 2026/3/31 7:04:46

590S-0350-5-0-0-00欧陆Eurotherm直流调速器故障代码

590S-0350-5-0-0-00 欧陆直流调速器(属派克 590 系列)的常见故障代码及其含义和处理方法如下:常见故障代码及含义OVERSPEED(超速报警)含义:速度反馈超过额定速度的 125%。可能原因:速度反馈信号…

作者头像 李华