news 2026/2/12 9:34:30

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/2/5 15:41:53

QMCDecode音频格式转换终极指南:Mac音乐解密完整教程

QMCDecode音频格式转换终极指南:Mac音乐解密完整教程 【免费下载链接】QMCDecode QQ音乐QMC格式转换为普通格式(qmcflac转flac,qmc0,qmc3转mp3, mflac,mflac0等转flac),仅支持macOS,可自动识别到QQ音乐下载目录,默认转…

作者头像 李华
网站建设 2026/2/5 21:37:05

基于Python+Vue开发的家具商城管理系统源码+运行步骤+计算机专业

项目简介 该项目是基于PythonVue开发的家具商城管理系统(前后端分离),这是一项为大学生课程设计作业而开发的项目。该系统旨在帮助大学生学习并掌握Python编程技能,同时锻炼他们的项目设计与开发能力。通过学习基于Python的家具商…

作者头像 李华
网站建设 2026/2/7 11:09:55

从零玩转RT-Thread(23):你必须知道的坑——定时器使用注意事项

本小节介绍定时器使用时的常见注意事项,从而避免常见错误和不当用法,实现学会在实际项目中更稳定、安全地使用定时器回调函数运行在中断/任务上下文对于HARD_TIMER模式的定时器,由系统时钟节拍中断处理程序扫描定时器列表并执行回调函数&…

作者头像 李华
网站建设 2026/2/7 22:07:56

Proxy Audio Device:macOS虚拟音频驱动完全指南

Proxy Audio Device:macOS虚拟音频驱动完全指南 【免费下载链接】proxy-audio-device A virtual audio driver for macOS to sends all audio to another output 项目地址: https://gitcode.com/gh_mirrors/pr/proxy-audio-device 项目简介 Proxy Audio Dev…

作者头像 李华
网站建设 2026/2/10 9:36:28

企业IT管理必备:虚拟光驱在软件部署中的妙用

快速体验 打开 InsCode(快马)平台 https://www.inscode.net输入框内输入如下内容: 创建一个企业级虚拟光驱管理工具,功能包括:1.批量挂载多个镜像文件 2.支持自动执行脚本部署 3.提供权限管理系统 4.记录操作日志 5.支持远程管理。要求使用C…

作者头像 李华
网站建设 2026/2/12 2:08:14

夜莺监控设计思考(二)边缘机房架构思考

一篇我们遗留了一个话题,就是如果贵司有多个数据中心,而且数据中心之间网络链路较差,此时应该怎么办?夜莺边缘架构模式举个例子,假设有北京、上海、美东三个数据中心,北京和上海之间有良好的专线打通&#…

作者头像 李华