STM32-DMA数据转运

devtools/2025/1/12 12:37:37/

注:DMA对应的库函数文件讲解


DMA_GetITStatus(uint32_t DMAy_IT) 是一个用于检查DMA(直接存储器访问)中断状态的库函数。它通常在使用STM32系列微控制器及其标准外设库时被调用。此函数的主要作用是返回指定DMA通道的特定中断标志的状态,以帮助开发者确定是否发生了特定类型的DMA事件。

参数 uint32_t DMAy_IT 用来指定你想要检查的DMA中断类型和对应的DMA流或通道。这个参数是一个组合值,通常由两个部分组成:

  1. DMAx:指明哪个DMA控制器(例如DMA1或DMA2),因为一些STM32芯片可能有多个DMA控制器。
  2. IT(Interrupt Type):指明具体的中断类型,比如传输完成(Transfer Complete, TC)、半传输完成(Half Transfer, HT)、传输错误(Transfer Error, TE)等。

函数会返回一个位标志,表明所选中断状态是设置(即事件发生)还是清除(即事件未发生)。这对于编写中断服务程序(ISR)非常重要,因为在ISR中你需要知道是什么类型的事件触发了中断,以便可以适当地处理它。

例如,如果你正在等待DMA传输完成中断,你可以使用 DMA_GetITStatus 来检查该中断是否已经发生。如果函数返回值表示中断已被设置,那么你可以安全地假设DMA传输已完成,并且可以继续执行后续的操作,如启动新的DMA传输、处理接收到的数据等。


DMA_ClearITPendingBit(uint32_t DMAy_IT) 是一个用于清除DMA(直接存储器访问)中断挂起位的库函数。在STM32系列微控制器中,当DMA传输过程中发生特定事件(如传输完成、半传输完成或传输错误),DMA硬件会设置相应的中断标志位,并可能触发中断请求。

然而,一旦这些事件被软件处理后,就需要清除相应的中断标志位,以确保相同的中断不会再次被误触发。这就是 DMA_ClearITPendingBit 函数的作用:它允许你手动清除指定DMA通道的特定中断挂起位,表明该事件已经被处理完毕。

参数 uint32_t DMAy_IT 用来指定要清除的DMA中断类型和对应的DMA流或通道。这个参数是一个组合值,通常由两个部分组成:

  1. DMAx:指明哪个DMA控制器(例如DMA1或DMA2)。
  2. IT(Interrupt Type):指明具体的中断类型,比如传输完成(Transfer Complete, TC)、半传输完成(Half Transfer, HT)、传输错误(Transfer Error, TE)等。

使用此函数是确保DMA中断系统正确工作的关键步骤之一。如果不清除这些标志位,可能会导致中断不断重复触发,或者新的相同类型的中断无法被正确识别。因此,在你的中断服务程序(ISR)中,你应该在检查并响应了某个DMA事件之后调用 DMA_ClearITPendingBit 来清除对应的中断挂起位。


定义:


DMA(Direct Memory Access)直接存储器存取 DMA可以提供外设和存储器或者存储器和存储器之间的高速数据传输,无须CPU干预,节省了CPU的资源 12个独立可配置的通道: DMA1(7个通道), DMA2(5个通道) 每个通道都支持软件触发和特定的硬件触发 STM32F103C8T6 DMA资源:DMA1(7个通道)


SRAM可以读也可以写 



 DMA基本结构




1.0 数据宽度与对齐


 在DMA中是如何解决数据宽度不一致的问题的,如果源端的数据大于目标端的数据,那么将读取出来的高位舍弃掉,然后只取低位。


 2.0 ADC与DMA数据转运

 连续扫描转运模式


3.0 手册解读



4.0 程序实现


注:本次程序主要实现的是DMA从存储器到存储器的数据转运,使用软件触发的方式

4.0.1 DMA初始化

初始化:主要包括RCC时钟初始化,GPIO初始化

 函数初始化实现

#include "stm32f10x.h" // Device headeruint16_t MyDMA_Size;/*** @brief  DMA初始化,包括时钟等* @param  ADDRA起始地址* @param  ADDRB结束地址* @param  SIZE大小* @return 无返回值*/
void MyDMA_Init(uint32_t AddrA, uint32_t AddrB, uint16_t Size)
{MyDMA_Size = Size;// RCC时钟初始化RCC_APB1PeriphClockCmd(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_PeripheralDataSize_Byte;// 地址是否自增DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;// 数据的传输方向,外设传输到存储器DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;// 重装计数器的大小DMA_InitStructure.DMA_BufferSize = Size;// DMA模式DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;// 触发方式DMA_InitStructure.DMA_M2M = DMA_M2M_Enable;// DMA优先级,通道的优先级设置为中等DMA_InitStructure.DMA_Priority = DMA_Priority_Medium;// 初始化DMADMA_Init(DMA1_Channel1, &DMA_InitStructure);// 使能DMA,初始化后会立即工作,等后续手动调用后再开始DMA_Cmd(DMA1_Channel1, DISABLE);
}

 4.0.2 数据转运函数

注:该函数的主要作用是,从新设置传输计数器的值

/*** @brief  DMA数据传输,重置* @param  MULL* @param  MULL* @param  SIZE大小* @return 无返回值*/
void MyDMA_Transfer(void)
{// DMA失能,在写入传输计数器之前,需要DMA暂停工作DMA_Cmd(DMA1_Channel1, DISABLE);// 写入传输计数器,指定将要转运的次数DMA_SetCurrDataCounter(DMA1_Channel1, MyDMA_Size);// 使能DMADMA_Cmd(DMA1_Channel1, ENABLE);// 等待DMA工作完成,工作完成标志位设置为1while (DMA_GetFlagStatus(DMA1_FLAG_TC1) == RESET);// 清除工作完成标志位DMA_ClearFlag(DMA1_FLAG_TC1);
}

4.0.3 全部程序

#include "stm32f10x.h" // Device headeruint16_t MyDMA_Size;/*** @brief  DMA初始化,包括时钟等* @param  ADDRA起始地址* @param  ADDRB结束地址* @param  SIZE大小* @return 无返回值*/
void MyDMA_Init(uint32_t AddrA, uint32_t AddrB, uint16_t Size)
{MyDMA_Size = Size;// RCC时钟初始化RCC_APB1PeriphClockCmd(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_PeripheralDataSize_Byte;// 地址是否自增DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;// 数据的传输方向,外设传输到存储器DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;// 重装计数器的大小DMA_InitStructure.DMA_BufferSize = Size;// DMA模式DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;// 触发方式DMA_InitStructure.DMA_M2M = DMA_M2M_Enable;// DMA优先级,通道的优先级设置为中等DMA_InitStructure.DMA_Priority = DMA_Priority_Medium;// 初始化DMADMA_Init(DMA1_Channel1, &DMA_InitStructure);// 使能DMA,初始化后会立即工作,等后续手动调用后再开始DMA_Cmd(DMA1_Channel1, DISABLE);
}/*** @brief  DMA数据传输,重置* @param  MULL* @param  MULL* @param  SIZE大小* @return 无返回值*/
void MyDMA_Transfer(void)
{// DMA失能,在写入传输计数器之前,需要DMA暂停工作DMA_Cmd(DMA1_Channel1, DISABLE);// 写入传输计数器,指定将要转运的次数DMA_SetCurrDataCounter(DMA1_Channel1, MyDMA_Size);// 使能DMADMA_Cmd(DMA1_Channel1, ENABLE);// 等待DMA工作完成,工作完成标志位设置为1while (DMA_GetFlagStatus(DMA1_FLAG_TC1) == RESET);// 清除工作完成标志位DMA_ClearFlag(DMA1_FLAG_TC1);
}

4.0.4 头文件


#ifndef __MYDMA_H
#define __MYDMA_Hvoid MyDMA_Init(uint32_t AddrA, uint32_t AddrB, uint16_t Size);
void MyDMA_Transfer(void);#endif

4.0.5 main函数文件


#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]++;Delay_ms(1000); // 延时1s,观察转运前的现象MyDMA_Transfer();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); // 使用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,观察转运后的现象}
}

......


5.0 DMA多通道


5.0.1 RCC时钟初始化

	// RCC 开启时钟RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);  // 开启ADC1时钟RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); // 开启GPIOA时钟RCC_APB2PeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);	  // 开启AHB的时钟RCC_ADCCLKConfig(RCC_PCLK2_Div6);

5.0.2 GPIO初始化

	// 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);

5.0.3 通道初始化

	// 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);

5.0.4 ADC结构体初始化

	ADC_InitTypeDef ADC_InitStructure;									// ADC 初始化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 = ENABLE;					// 连续转换,每转换完一次之后立即开始下一次的转换,使能ADC_InitStructure.ADC_ScanConvMode = ENABLE;						// 使能扫描模式,扫描规则组前面的4个通道ADC_InitStructure.ADC_NbrOfChannel = 4;								// 通道数为4个扫描规则组前面的4个通道ADC_Init(ADC1, &ADC_InitStructure);									// ADC结构体初始化

5.0.5 DMA结构体初始化

	// DMA初始化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_Circular;DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;DMA_InitStructure.DMA_Priority = DMA_Priority_Medium;DMA_Init(DMA1_Channel1, & DMA_InitStructure);

5.0.6 DMA使能与校准

	// DMA和ADC使能DMA_Cmd(DMA1_Channel1, ENABLE);			// DMA1通道使能ADC_DMACmd(ADC1, ENABLE);				// ADC1触发DMA1的信号使能ADC_Cmd(ADC1, ENABLE);					// ADC1使能// ADC校准ADC_ResetCalibration(ADC1);// 获取ADC转换标志位while (ADC_GetResetCalibrationStatus(ADC1) == SET);// 开启ADC转换ADC_StartCalibration(ADC1);// 获取ADC转换状态while (ADC_GetCalibrationStatus(ADC1) == SET);// ADC触发ADC_SoftwareStartConvCmd(ADC1, ENABLE); // 软件触发ADC开始工作,由于ADC处于连续转换模式,故触发一次后ADC就可以一直连续不断地工作

对应程序文件:

AD.C文件

#include "stm32f10x.h" // Device headeruint16_t AD_Value[4]; // 定义用于存放AD转换结果的全局数组void AD_Init(void)
{// RCC 开启时钟RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);  // 开启ADC1时钟RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); // 开启GPIOA时钟RCC_APB2PeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);	  // 开启AHB的时钟RCC_ADCCLKConfig(RCC_PCLK2_Div6);// 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);// 配置规则组通道ADC_RegularChannelConfig(ADC1, ADC_Channel_0, 1, ADC_SampleTime_55Cycles5); // 规则组序列1的位置ADC_RegularChannelConfig(ADC1, ADC_Channel_1, 2, ADC_SampleTime_55Cycles5); // 规则组序列2的位置ADC_RegularChannelConfig(ADC1, ADC_Channel_2, 3, ADC_SampleTime_55Cycles5); // 规则组序列3的位置ADC_RegularChannelConfig(ADC1, ADC_Channel_3, 4, ADC_SampleTime_55Cycles5); // 规则组序列4的位置ADC_InitTypeDef ADC_InitStructure;									// ADC 初始化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 = ENABLE;					// 连续转换,每转换完一次之后立即开始下一次的转换,使能ADC_InitStructure.ADC_ScanConvMode = ENABLE;						// 使能扫描模式,扫描规则组前面的4个通道ADC_InitStructure.ADC_NbrOfChannel = 4;								// 通道数为4个扫描规则组前面的4个通道ADC_Init(ADC1, &ADC_InitStructure);									// ADC结构体初始化// DMA初始化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_Circular;DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;DMA_InitStructure.DMA_Priority = DMA_Priority_Medium;DMA_Init(DMA1_Channel1, & DMA_InitStructure);// DMA和ADC使能DMA_Cmd(DMA1_Channel1, ENABLE);			// DMA1通道使能ADC_DMACmd(ADC1, ENABLE);				// ADC1触发DMA1的信号使能ADC_Cmd(ADC1, ENABLE);					// ADC1使能// ADC校准ADC_ResetCalibration(ADC1);// 获取ADC转换标志位while (ADC_GetResetCalibrationStatus(ADC1) == SET);// 开启ADC转换ADC_StartCalibration(ADC1);// 获取ADC转换状态while (ADC_GetCalibrationStatus(ADC1) == SET);// ADC触发ADC_SoftwareStartConvCmd(ADC1, ENABLE); // 软件触发ADC开始工作,由于ADC处于连续转换模式,故触发一次后ADC就可以一直连续不断地工作
}

AD.H文件

#ifndef __AD_H
#define __AD_Hextern uint16_t AD_Value[4];void AD_Init(void);#endif

MAIN.C文件

#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "OLED.h"
#include "AD.h"int main(void)
{/*模块初始化*/OLED_Init();				//OLED初始化AD_Init();					//AD初始化/*显示静态字符串*/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);		//显示转换结果第0个数据OLED_ShowNum(2, 5, AD_Value[1], 4);		//显示转换结果第1个数据OLED_ShowNum(3, 5, AD_Value[2], 4);		//显示转换结果第2个数据OLED_ShowNum(4, 5, AD_Value[3], 4);		//显示转换结果第3个数据Delay_ms(100);							//延时100ms,手动增加一些转换的间隔时间}
}

......


http://www.ppmy.cn/devtools/149873.html

相关文章

某漫画网站JS逆向反混淆流程分析

文章目录 1. 写在前面1. 接口分析2. 反混淆分析 【🏠作者主页】:吴秋霖 【💼作者介绍】:擅长爬虫与JS加密逆向分析!Python领域优质创作者、CSDN博客专家、阿里云博客专家、华为云享专家。一路走来长期坚守并致力于Pyth…

纯手工(不基于maven的pom.xml、Web容器)连接MySQL数据库的详细过程(Java Web学习笔记)

1 引言 最近读一些Java Web开发类的书籍时,发现书中的连接数据库的过程缺少了一些关键性的过程,这对初学者非常不友好。为此,本文将给出详细的连接MySQL数据库的过程,并且是纯手工,不依赖于pom.xml和Web容器&#xff…

一分钟学会文心一言API如何接入,文心一言API接入教程

一、前期准备 注册百度智能云账号: 前往百度智能云官网注册一个账号。这是接入文心一言API的基础。 了解API接口: 在百度智能云开放平台中,找到文心一言API的详情页,了解提供的API接口类型(如云端API、移动端API、离线…

自动化元素定位时,发现提示找不到元素,怎么处理?

你是否在自动化测试中遇到这样的问题:明明代码没有报错,定位方式也没错,但运行时总是提示“找不到元素”?别急,这种情况很常见,解决起来并不复杂。本文将为你拆解各种原因,并提供实用的解决方案…

Kafka 深度剖析

Kafka 深度剖析:从基础概念到集群实战 在当今大数据与分布式系统蓬勃发展的时代,Apache Kafka 作为一款极具影响力的分布式发布 - 订阅消息系统,宛如一颗璀璨的明星,照亮了数据流转与处理的诸多场景。它由 LinkedIn 公司于 2010 年…

大模型LLM-Prompt-CRISPE

1 CRISPE "CRISPE"是一个用于构建有效提示词(Prompt)的框架,特别适用于需要AI扮演特定角色或在特定背景下完成任务的场景。以下是"CRISPE"框架的组成部分: Capacity and Role(能力和角色&#xf…

记录一次Android Studio的下载、安装、配置

目录 一、下载和安装 Android Studio 1、搜索下载Android studio ​2、下载成功后点击安装包进行安装: 3、这里不用打勾,直接点击安装 : 4、完成安装: 5、这里点击Cancel就可以了 6、接下来 7、点击自定义安装&#xff1a…

Go oom分析(二)——导出dump离线分析

在 Go 程序中导出内存或 CPU 的 dump 文件(通常通过 pprof 工具生成)并进行分析,以下是详细步骤: 1. 在程序中开启 pprof 在你的 Go 程序中引入 net/http/pprof,开启 pprof 服务: import (_ "net/h…