当前位置: 首页 > news >正文

DMA数据转运

1、DMA简介

  • DMA(Direct Memory Access)直接存储器存取

  • DMA可以通过外设和存储器或者存储器和存储器之间的告诉数据传输,无须CPU干预,节省了CPU的资源

  • 12个独立可配置的通道,DMA1(7个通道),DMA2(5个通道)

  • 每个通道都支持软件触发和特定的硬件触发

  • STM32F103C8T6 DMA资源通道:DMA1(7个通道)

2、存储器映像

5f4b6417-8343-4c5b-99b6-58aa5626792e

3、DMA框图

e4ece7c9-f499-4974-b0f2-07e7254f7528

4、DMA基本结构

1ebc4961-4adb-4dc9-b30e-a70570218eb7

5、DMA请求

adb0d6b5-9760-46cb-bc04-f816e90de400

6、数据宽度与对齐

564a2d9b-8527-4749-9d13-acd602f3133b

7、数据转运+DMA

38050c41-185e-4753-a4ea-a5d9f5e8a93e

8、ADC扫描模式+DMA

9ef37320-791c-4c94-bab3-89660a01914b

9、代码

#include "stm32f10x.h"                  // Device headeruint16_t MyDMA_Size;                    //定义全局变量,用于记住Init函数的Size,供Transfer函数使用/*** 函    数:DMA初始化* 参    数:AddrA 原数组的首地址* 参    数:AddrB 目的数组的首地址* 参    数:Size 转运的数据大小(转运次数)* 返 回 值:无*/
void MyDMA_Init(uint32_t AddrA, uint32_t AddrB, uint16_t Size)
{MyDMA_Size = Size;                    //将Size写入到全局变量,记住参数Size/*开启时钟*/RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);                        //开启DMA的时钟/*DMA初始化*/DMA_InitTypeDef DMA_InitStructure;                                        //定义结构体变量DMA_InitStructure.DMA_PeripheralBaseAddr = AddrA;                        //外设基地址,给定形参AddrADMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;    //外设数据宽度,选择字节DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Enable;            //外设地址自增,选择使能DMA_InitStructure.DMA_MemoryBaseAddr = AddrB;                            //存储器基地址,给定形参AddrBDMA_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_Init,配置DMA1的通道1/*DMA使能*/DMA_Cmd(DMA1_Channel1, DISABLE);    //这里先不给使能,初始化后不会立刻工作,等后续调用Transfer后,再开始
}/*** 函    数:启动DMA数据转运* 参    数:无* 返 回 值:无*/
void MyDMA_Transfer(void)
{DMA_Cmd(DMA1_Channel1, DISABLE);                    //DMA失能,在写入传输计数器之前,需要DMA暂停工作DMA_SetCurrDataCounter(DMA1_Channel1, MyDMA_Size);    //写入传输计数器,指定将要转运的次数DMA_Cmd(DMA1_Channel1, ENABLE);                        //DMA使能,开始工作while (DMA_GetFlagStatus(DMA1_FLAG_TC1) == RESET);    //等待DMA工作完成DMA_ClearFlag(DMA1_FLAG_TC1);                        //清除工作完成标志位
}
#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "OLED.h"
#include "MyDMA.h"uint8_t DataA[] = {0x01, 0x02, 0x03, 0x04};                //定义测试数组DataA,为数据源
uint8_t DataB[] = {0, 0, 0, 0};                            //定义测试数组DataB,为数据目的地int main(void)
{/*模块初始化*/OLED_Init();                //OLED初始化MyDMA_Init((uint32_t)DataA, (uint32_t)DataB, 4);    //DMA初始化,把源数组和目的数组的地址传入/*显示静态字符串*/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);        //显示数组DataAOLED_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);        //显示数组DataBOLED_ShowHexNum(4, 4, DataB[1], 2);OLED_ShowHexNum(4, 7, DataB[2], 2);OLED_ShowHexNum(4, 10, DataB[3], 2);Delay_ms(1000);        //延时1s,观察转运前的现象MyDMA_Transfer();    //使用DMA转运数组,从DataA转运到DataBOLED_ShowHexNum(2, 1, DataA[0], 2);        //显示数组DataAOLED_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);        //显示数组DataBOLED_ShowHexNum(4, 4, DataB[1], 2);OLED_ShowHexNum(4, 7, DataB[2], 2);OLED_ShowHexNum(4, 10, DataB[3], 2);Delay_ms(1000);        //延时1s,观察转运后的现象}
}

ADC+ADM代码

#include "stm32f10x.h" // Device header 
uint16_t AD_Value[4]; //定义用于存放AD转换结果的全局数组 
/*** 函 数:AD初始化* 参 数:无* 返 回 值:无*/
void AD_Init(void)
{/*开启时钟*/RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE); //开启ADC1的时钟RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); //开启GPIOA的时钟RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); //开启DMA1的时钟/*设置ADC时钟*/RCC_ADCCLKConfig(RCC_PCLK2_Div6); //选择时钟6分频,ADCCLK = 72MHz / 6 = 12MHz/*GPIO初始化*/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); //将PA0、PA1、PA2和PA3引脚初始化为模拟输入/*规则组通道配置*/ADC_RegularChannelConfig(ADC1, ADC_Channel_0, 1, ADC_SampleTime_55Cycles5); //规则组序列1的位置,配置为通道0ADC_RegularChannelConfig(ADC1, ADC_Channel_1, 2, ADC_SampleTime_55Cycles5); //规则组序列2的位置,配置为通道1ADC_RegularChannelConfig(ADC1, ADC_Channel_2, 3, ADC_SampleTime_55Cycles5); //规则组序列3的位置,配置为通道2ADC_RegularChannelConfig(ADC1, ADC_Channel_3, 4, ADC_SampleTime_55Cycles5); //规则组序列4的位置,配置为通道3/*ADC初始化*/ADC_InitTypeDef ADC_InitStructure; //定义结构体变量ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; //模式,选择独立模式,即单独使用ADC1ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; //数据对齐,选择右对齐ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None; //外部触发,使用软件触发,不需要外部触发ADC_InitStructure.ADC_ContinuousConvMode = ENABLE; //连续转换,使能,每转换一次规则组序列后立刻开始下一次转换ADC_InitStructure.ADC_ScanConvMode = ENABLE; //扫描模式,使能,扫描规则组的序列,扫描数量由ADC_NbrOfChannel确定ADC_InitStructure.ADC_NbrOfChannel = 4; //通道数,为4,扫描规则组的前4个通道ADC_Init(ADC1, &ADC_InitStructure); //将结构体变量交给ADC_Init,配置ADC1/*DMA初始化*/DMA_InitTypeDef DMA_InitStructure; //定义结构体变量DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)&ADC1->DR; //外设基地址,给定形参AddrADMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord; //外设数据宽度,选择半字,对应16为的ADC数据寄存器DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable; //外设地址自增,选择失能,始终以ADC数据寄存器为源DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)AD_Value; //存储器基地址,给定存放AD转换结果的全局数组AD_ValueDMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord; //存储器数据宽度,选择半字,与源数据宽度对应DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; //存储器地址自增,选择使能,每次转运后,数组移到下一个位置DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC; //数据传输方向,选择由外设到存储器,ADC数据寄存器转到数组DMA_InitStructure.DMA_BufferSize = 4; //转运的数据大小(转运次数),与ADC通道数一致DMA_InitStructure.DMA_Mode = DMA_Mode_Circular; //模式,选择循环模式,与ADC的连续转换一致DMA_InitStructure.DMA_M2M = DMA_M2M_Disable; //存储器到存储器,选择失能,数据由ADC外设触发转运到存储器DMA_InitStructure.DMA_Priority = DMA_Priority_Medium; //优先级,选择中等DMA_Init(DMA1_Channel1, &DMA_InitStructure); //将结构体变量交给DMA_Init,配置DMA1的通道1/*DMA和ADC使能*/DMA_Cmd(DMA1_Channel1, ENABLE); //DMA1的通道1使能ADC_DMACmd(ADC1, ENABLE); //ADC1触发DMA1的信号使能ADC_Cmd(ADC1, ENABLE); //ADC1使能/*ADC校准*/ADC_ResetCalibration(ADC1); //固定流程,内部有电路会自动执行校准while (ADC_GetResetCalibrationStatus(ADC1) == SET);ADC_StartCalibration(ADC1);while (ADC_GetCalibrationStatus(ADC1) == SET);/*ADC触发*/ADC_SoftwareStartConvCmd(ADC1, ENABLE); //软件触发ADC开始工作,由于ADC处于连续转换模式,故触发一次后ADC就可以一直连续不断地工作
}
http://www.jsqmd.com/news/100561/

相关文章:

  • Dify 1.7.0音频时长无法延长?资深架构师教你4招完美绕过
  • 【混合检索优化实战】:Dify响应时间降低50%的三大核心技术揭秘
  • 人呼吸道合胞病毒(hRSV)重组蛋白技术详解:F- G 蛋白结构与试剂级表征指南
  • 面试数据库八股文五问五答第四期
  • 揭秘Dify与Spring AI协同部署难点:5步实现生产环境稳定上线
  • 你不知道的单细胞数据降维黑科技:UMAP与t-SNE的R语言优化实战
  • 探索含DG的33节点配电网谐波潮流计算
  • Qwen3-8B实战测评:小模型为何超越大模型
  • RSA加密
  • Git下载速度慢?切换清华镜像提升效率300%
  • LFT2730 硬质平膜型压力变送器
  • 揭秘Dify测试瓶颈:如何用Agent工具构建高覆盖率用例?
  • FINCON:融合概念性语言强化的 LLM 多智能体金融决策框架
  • 揭秘私有化Dify的SSL配置难题:5步完成高强度加密部署
  • 深入解析:SQL Server 大数据量分表
  • 【Dify 权限架构升级必读】:基于混合检索的3层权限模型设计与落地
  • Dify与Spring AI性能对比(从吞吐量到内存占用的全面剖析)
  • 【Agent工具注册元数据全解析】:Dify平台高效集成的5大核心要素
  • 第八章作业
  • Dify 1.7.0降噪效果为何碾压前代?:基于频谱掩码技术的深度剖析
  • 【Agent工具高效开发秘籍】:Dify文档生成全栈实战指南
  • 提示工程架构师视角:Agentic AI的未来展望
  • 如何用Docker Buildx在10分钟内完成ARM64和AMD64双架构镜像构建?真相令人震惊
  • Dify依赖检查没人讲清楚?这篇万字长文彻底说透了
  • P14344 [JOISC 2019] 两道料理 / Two Dishes
  • CVE-2025-68080:Saad Iqbal用户头像插件中的存储型跨站脚本漏洞深度解析
  • LobeChat Docker镜像下载地址与验证方法全记录
  • LobeChat能否实现AI绘画集成?图文生成联动尝试
  • linux 进程内存占用查看 - Sanny.Liu
  • 如何用Dify调度Tesseract实现全自动批量文本提取?一线工程师深度分享