STM32DMA


前言

介绍STM32DMA的原理以及功能,使用DMA配合ADC实现自动化AD转换。


一、介绍部分

DMA简介

在这里插入图片描述

存储器映像

在这里插入图片描述

DMA框图

这里APB1、APB2位置互换。

在这里插入图片描述

仲裁器

在这里插入图片描述

DMA基本结构

自动重装不能与存储器到存储器模式同时设置,会使DMA 永远执行

在这里插入图片描述

请求映像

每个硬件触发对应特定的DMA通道。

在这里插入图片描述

数据对齐方式

发送的数据于接收数据的大小设置不对等时,大转小:高位补0,小转大:高位舍弃。

在这里插入图片描述

数据转运

在这里插入图片描述

ADC扫描模式配合DMA

在这里插入图片描述

二、代码部分

DMA数据转运

连接电路

连接一个OLED即可
在这里插入图片描述

代码实现

封装ThisDMA.c

#include "stm32f10x.h"                  // Device header

uint16_t DMA_Size;

void ThisDMA_Init(uint32_t ArrA,uint32_t ArrB,uint16_t BufSize){
	RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1,ENABLE);
	
	DMA_Size = BufSize;
	DMA_InitTypeDef DMA_InitStructure;
	DMA_InitStructure.DMA_BufferSize = BufSize;	// 传输计数器的值
	DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;	// 外设作为发送方
	DMA_InitStructure.DMA_M2M = DMA_M2M_Enable;		// 软件触发还是硬件触发
	DMA_InitStructure.DMA_MemoryBaseAddr = ArrA;	// 存储器初始地址
	DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;	// 存储器数据大小
	DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;				// 转运时是否自增
	DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;	// 传输计数器是否重装
	DMA_InitStructure.DMA_PeripheralBaseAddr = ArrB;	// 外设初始地址
	DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;	// 外设数据大小
	DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Enable;				// 转运时是否自增
	DMA_InitStructure.DMA_Priority = DMA_Priority_Medium;		// 优先级
	DMA_Init(DMA1_Channel1,&DMA_InitStructure);
	
	// 初始化后先不开始转运
	DMA_Cmd(DMA1_Channel1,DISABLE);
}

void ThisDMA_Transfer(void){
	// DMA失能,先关闭,再修改数据
	DMA_Cmd(DMA1_Channel1,DISABLE);
	// 重新赋值传输计数器,使其再次工作
	DMA_SetCurrDataCounter(DMA1_Channel1,DMA_Size);
	// DMA使能
	DMA_Cmd(DMA1_Channel1,ENABLE);
	// 等待转运完成
	while(DMA_GetFlagStatus(DMA1_FLAG_TC1) == RESET);
	// 清除标志位
	DMA_ClearFlag(DMA1_FLAG_TC1);
}


主函数main.c

#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "OLED.h"
#include "ThisDMA.h"


// 需要转运的数据
uint8_t DataA[] = {0x01,0x02,0x03,0x04};
// 用于接收数据
uint8_t DataB[] = {0,0,0,0};
uint8_t i;
// 显示数据函数
void ShowData(void){
	OLED_ShowHexNum(1,4,(uint32_t)&DataA,8);
	OLED_ShowHexNum(3,4,(uint32_t)&DataB,8);
	OLED_ShowHexNum(2,1,DataA[0],2);
	OLED_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);
	OLED_ShowHexNum(4,4,DataB[1],2);
	OLED_ShowHexNum(4,7,DataB[2],2);
	OLED_ShowHexNum(4,10,DataB[3],2);
}

int main(void)
{
	OLED_Init();
	ThisDMA_Init((uint32_t)DataB,(uint32_t)DataA,4);
	OLED_ShowString(1,1,"A");
	OLED_ShowString(3,1,"B");
	ShowData();
	while (1)
	{
		for(i=0;i<4;i++)
			DataA[i]++;
		
		ShowData();
		Delay_ms(1000);
		ThisDMA_Transfer();
		ShowData();
		Delay_ms(1000);
	}
}

ADC扫描+DMA

实现思路

使用AD 的连续转换扫描模式,通过DMA帮助在数据寄存器及时的取走数据来实现自动化的AD转换

在这里插入图片描述

连接电路

在这里插入图片描述

代码实现

改写AD.c

#include "stm32f10x.h"                  // Device header

uint16_t ADBufArr[4];	// 在头文件声明为外部变量

void AD_Init(void){
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1,ENABLE);
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
	RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1,ENABLE);
	
	// 设置ADC时钟 72/6=12MHz
	RCC_ADCCLKConfig(RCC_PCLK2_Div6);
	
	// 初始化GPIO
	GPIO_InitTypeDef GPIO_Structure;
	GPIO_Structure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_2 | GPIO_Pin_3;
	GPIO_Structure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Structure.GPIO_Mode = GPIO_Mode_AIN;		// 模拟输入模式(ADC专用)
	GPIO_Init(GPIOA,&GPIO_Structure);
	
	// 使用规则组
	// 配置规则组通道(ADC、此ADC的通道、序号、采样时间)
	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_ContinuousConvMode = ENABLE;										// 连续模式还是单次模式
	ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;								// 数据对齐,右对齐
	ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;		// 触发转换的触发源,不使用外部触发使用软件触发
	ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;										// 独立模式or双ADC模式
	ADC_InitStructure.ADC_NbrOfChannel = 4;																// 指定要使用多少个通道
	ADC_InitStructure.ADC_ScanConvMode = ENABLE;													// 扫描模式还是非扫描模式
	ADC_Init(ADC1,&ADC_InitStructure);
	
	// 初始化DMA
	DMA_InitTypeDef DMA_InitStructure;
	DMA_InitStructure.DMA_BufferSize = 4;	// 传输计数器的值
	DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;	// 外设作为发送方
	DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;		// 软件触发还是硬件触发
	DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)ADBufArr;	// 存储器初始地址
	DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;	// 存储器数据大小
	DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;				// 转运时是否自增
	DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;	// 传输计数器是否重装
	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_Priority = DMA_Priority_Medium;		// 优先级
	DMA_Init(DMA1_Channel1,&DMA_InitStructure);

	// 开启DMA
	DMA_Cmd(DMA1_Channel1,ENABLE);
	
	// 开启ADC1触发DMA的通道
	ADC_DMACmd(ADC1,ENABLE);
	
	// 开启ADC
	ADC_Cmd(ADC1,ENABLE);
	
	// 校准
	ADC_ResetCalibration(ADC1);													// 复位校准
	while(ADC_GetResetCalibrationStatus(ADC1) == SET);	// 获取复位校准的状态,等待复位完成
	ADC_StartCalibration(ADC1);													// 开始校准
	while(ADC_GetCalibrationStatus(ADC1) == SET);				// 获取校准状态,等待校准完成

	// ADC开始运转
	ADC_SoftwareStartConvCmd(ADC1,ENABLE);
}


主函数main.c

#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "OLED.h"
#include "AD.h"


int main(void)
{
	OLED_Init();
	AD_Init();
	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,ADBufArr[0],4);
		OLED_ShowNum(2,5,ADBufArr[1],4);
		OLED_ShowNum(3,5,ADBufArr[2],4);
		OLED_ShowNum(4,5,ADBufArr[3],4);
		Delay_ms(100);
	}
}


总结

函数相关

// 打开此ADC的DMA触发通道
void ADC_DMACmd(ADC_TypeDef* ADCx, FunctionalState NewState);
// DMA初始化
void DMA_Init(DMA_Channel_TypeDef* DMAy_Channelx, DMA_InitTypeDef* DMA_InitStruct);
// DMA结构体负初值
void DMA_StructInit(DMA_InitTypeDef* DMA_InitStruct);
// DMA使能
void DMA_Cmd(DMA_Channel_TypeDef* DMAy_Channelx, FunctionalState NewState);
// DMA中断使能
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);
  • 14
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: STM32中的DMA(直接内存访问)是一种高效的数据传输机制,它可以以较少的CPU干预完成大量的数据传输。 DMA FIFO是DMA传输中的一个缓冲区,它可以储存数据以便DMA直接从中读取和写入数据。DMA FIFO可以大大提升DMA传输的效率,因为它可以避免DMA频繁访问内存,从而减少CPU的干预,提升数据传输的速率。 在STM32中,不同的DMA通道都有自己独立的DMA FIFO。DMA FIFO可以通过编程控制其大小和数据传输的方向。此外,DMA FIFO还支持半总线传输和内存突发传输等高级功能,以进一步提高数据传输的效率。 需要注意的是,DMA FIFO的大小不能太小,否则会影响传输速度。同时,在使用DMA FIFO时,需要注意内存的地址对齐,否则会影响DMA传输的效率。 总之,STM32 DMA FIFO是一种高效的DMA缓冲区,它能够大大提高数据传输的效率,减少CPU的干预,是STM32高效数据传输的重要手段之一。 ### 回答2: STM32 DMA FIFO是一种在STM32微控制器内部使用的先进的数据传输机制,旨在提高数据传输的效率和可靠性。DMA(FIFO)是指直接内存存储器访问(FIFO)的简称,这种传输方式可以通过DMA控制器自动处理,而无需CPU的干预。 STM32 DMA FIFO可以通过提供缓存缓冲区来处理和管理数据传输,这些缓存区是可以在存储器或外设之间共享数据的独立存储器区域。缓存区的大小和数量可以根据应用程序的数据传输速率和带宽要求进行配置。通过使用FIFO机制,可以缓解传输速率不匹配的问题,并且可以在数据传输时提供额外的保密性。 STM32 DMA FIFO的优点包括高效的数据传输速率,使得数据传输更加快速和可靠;同时,它还可以减少CPU的负担,提高程序执行效率。STM32 DMA FIFO对于大数据传输、高速传输以及多模块数据传输等应用非常有用。 因此,STM32 DMA FIFO已经成为了STM32微控制器的一个标准组件,许多工业自动化、智能制造和物联网应用程序中,都广泛采用了这种数据传输机制。 ### 回答3: STM32 DMA FIFO是STM32微控制器中的一种DMA传输方式,它使用了FIFO缓冲区来提高DMA传输效率。FIFO(First-In-First-Out)的缓冲区可以在一端输入数据,在另一端输出数据,所有数据按照入队的顺序依次出队。在DMA传输中,FIFO缓冲区可以减少DMA传输对CPU的干扰,并且可以缓存大量的数据,以增强数据传输的连续性与稳定性。通过使用DMA FIFO,可以在DMA传输期间减少数据丢失和重复读取以及提高数据传输的吞吐量。 STM32 DMA FIFO的主要特点包括: 1. 多通道支持:STM32 DMA FIFO可以支持多个通道同时进行DMA传输,并且可以进行通道间的数据拷贝。 2. 高效传输:使用DMA FIFO可以减少CPU的干扰,并且可以增强数据传输的连续性,从而提高数据传输的效率。 3. 灵活配置:STM32 DMA FIFO可以通过编程的方式配置传输方式、传输数据长度、传输地址等参数,以适应不同的应用场景。 总之,STM32 DMA FIFO是一种高效的DMA传输方式,适用于需要高效、稳定、连续的数据传输场景,例如音频、视频、存储器等大量数据传输场景。它可以提高数据传输效率,减少CPU的负担,提高系统的稳定性和可靠性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值