STM32 DMA

STM32暑假学习 DMA



前言

DMA是ADC多通道输出的好帮手,类似于饭店里的服务员,能帮助外设和存储器直接搬运数据。


一、DMA是什么?

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

存储器映像

在这里插入图片描述
STM32里的存储器和被安排的地址

二、DMA基本结构图

在这里插入图片描述
· 两大站点:外设寄存器站点和存储器站点(Flash和SRAM)
· 转运的方向有:外设到Flash和SRAM,Flash和SRAM到外设,Flash到SRAM
· 外设和存储器两个站点都有3个参数,起始地址、数据宽度、地址是否自增。
· 数据宽度:可以选择字节Byte(8位)、半字HalfWord(16位)和字Word(32位)
· 外设地址不用自增,存储器地址需要自增
· 如果外设起始地址写Flash地址,那他就会去Flash里去找数据。不用局限于一定要找外设寄存器。
· 传输计数器:记录需要转运的次数
· 自动重装器:例如传输计数器是5,需要执行5次转运,如果转运5次结束后,自动重装器开启了的话,那传输计数器自动恢复到5。类似于ADC中的连续模式
·M2M:Memory to Memory 当M2M等于1时,使用软件触发,软件触发一般适用于存储器到存储器的转运,是软件启动,不需要时机,并且向尽快完成任务。当M2M等于0时,使用硬件触发,硬件触发源可以选择ADC、串口、定时器等等,使用硬件触发的转运一般都是与外设有关的转运。这些转运需要一定的时机,比如ADC转换完成,串口收到数据,定时时间到等等,所以需要使用硬件触发,在硬件达到这些时机时,转一个信号过来触发控制

1.DMA进行转运,有以下条件:

1.开关控制,DMA_Cmd必须使能。
2.传输计数器必须大于0
3.触发源,必须有触发信号。触发一次,转运一次,传输计数器自减一次,当传输计数器等于0时,且没有自动重装时,这时无论是否触发,DMA都不会再进行转运了,此时需要DMA_Cmd给Disable,关闭DMA。再为传输计数器写入一个大于0的数,再开启DMA_Cmd,DMA才能继续工作。
·写传输计数器的时候,必须关闭DMA,再进行传输。不能在DMA 开启时写传输计数器。

2.数据宽度与对齐

如果目标的数据宽度比源端的数据宽度大,则高位补0。如果目标数据宽度比源端数据宽度小,则舍弃高位。

3.DMA是如何工作的

数据转运+DMA

在这里插入图片描述
· 将SRAM里的数组DataA,转运到另一个数组DataB中。
· 首先要填外设和存储器的起始地址,数据宽度,地址是否自增。
· 外设地址填DataA数组的首地址,存储器地址给DataB数组的首地址。
· 数据宽度,两组都是uint8_t
· 看需求,当DataA[0]转运到DataB[0]时,接下来是DataA[1]转到DataB[1],所以两个站点的地址都应该自增,都移动到下一个数据的位置
· 调用DMA_Cmd,给DMA使能
· 转运7次之后,传输计数器自减到0,DMA停止,转运完成

ADC扫描模式+DMA

在这里插入图片描述
左边ADC触发一次后,7个通道依次进行AD转换,然后转换结果都放到ADC_DR数据寄存器里面,在每个单独的通道转换完成后,进行一个DMA数据转运,并且目的地址进行自增,这样数据就不会被覆盖,外设地址,写入ADC_DR这个寄存器的地址,存储器的地址,可以在SRAM中定义一个数组ADValue,然后把ADValue的地址当做存储器的地址,之后数据宽度,因为ADC_DR和SRAM数组都是uint16_t,所以数据宽度都是16位的半字传输
· 从图可知,外设地址不自增,存储器地址自增,传输方向,是外设站点到存储器站点,传输计数器,这里通道有7个,所以计数7次
· 如果ADC是连续扫描,那DMA就可以使用自动重装,在ADC启动下一轮转换的时候,DMA也启动下一轮的转运,ADC和DMA同步工作
· 最后是触发选择,这里ADC_DR的值是在ADC单个通道转换完成后才会有效,所以DMA转运的时机,需要和ADC单个通道转换完成同步,所以DMA的触发要选择ADC的硬件触发
· 虽然单个通道转换完成后,不产生任何标志位和中断,但是它应该会产生DMA请求,去触发DMA转运
·一般来说,DMA最常见的用途就是配合ADC的扫描模式,因为ADC扫描模式有个数据覆盖的特征,或者可以说这个数据覆盖的问题是ADC固有的缺陷,ADC对DMA的需求是非常强烈

三、 数据转运+DMA示例代码及接线图

在这里插入图片描述

ADC1->DR
我们利用利用这种方式来访问到ADC的DR寄存器

DMA编程思路

· 第一步,RCC开启DMA的时钟
· 第二步,直接调用DMA_Init ,初始化各个参数,(外设和存储器的起始地址、数据宽度、地址是否自增、方向、传输计数器、是否需要自动重装、选择触发源、通道优先级
· 第三步DMA_Cmd,进行开关控制
· 如果选择硬件触发,记得在对应外设调用一下XXX_DMACmd,开启一下触发信号的输出
· 如果需要DMA的中断,那就调用DMA_ITConfig,开启中断输出,再在NVIC里,配置相应的中断通道,写中断函数就行了
· 最后,在运行过程中,如果转运完成,传输计数器清0了,这时再想给传输计数器赋值的话,需要将DMA失能,写传输计数器、DMA使能。

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);//清除中断挂起位

DMA初始化配置

#include "Device/Include/stm32f10x.h"   // Device header
uint16_t MyDMA_Size;

void MyDMA_Init(uint32_t AddrA,uint32_t AddrB,uint16_t Size)
{
	RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1,ENABLE);//开启RCC_AHB时钟
	
	DMA_InitTypeDef DMA_InitStruct;
	DMA_InitStruct.DMA_PeripheralBaseAddr = AddrA;//外设起始地址
	DMA_InitStruct.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;//外设数据宽度
	DMA_InitStruct.DMA_PeripheralInc = DMA_PeripheralInc_Enable;//外设地址是否自增
	DMA_InitStruct.DMA_MemoryBaseAddr = AddrB;//存储器起始地址
	DMA_InitStruct.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;//存储器数据宽度
	DMA_InitStruct.DMA_MemoryInc = DMA_MemoryInc_Enable;//存储器地址是否自增
	DMA_InitStruct.DMA_DIR = DMA_DIR_PeripheralSRC;//DMA的传输方向,(外设作为源头,传给存储器)
	DMA_InitStruct.DMA_BufferSize = Size;//传输计数器的次数
	DMA_InitStruct.DMA_Mode = DMA_Mode_Normal;//正常模式
	DMA_InitStruct.DMA_M2M = DMA_M2M_Enable;//启用软件触发
	DMA_InitStruct.DMA_Priority = DMA_Priority_Medium;//DMA优先级
	DMA_Init(DMA1_Channel1,&DMA_InitStruct);
	
	DMA_Cmd(DMA1_Channel1,DISABLE);//DMA失能
}

· 外设地址和存储器地址都设置为变量,根据调用函数的实际地址输入。
· DMA的传输方向可以选择外设作为目的地或者源头。
· 正常模式下不会开启自动重装,循环模式会开启自动重装
· M2M使能就是启用软件触发,失能就是启用硬件触发

DMA转运函数如下:

void MyDMA_Transfer(void)
{
	DMA_Cmd(DMA1_Channel1,DISABLE);//DMA失能
	DMA_SetCurrDataCounter(DMA1_Channel1,MyDMA_Size);//设置数据寄存器
	DMA_Cmd(DMA1_Channel1,ENABLE);//DMA使能

	while(DMA_GetFlagStatus(DMA1_FLAG_TC1) == RESET);//等待转运结束,标志位置1 
	DMA_ClearFlag(DMA1_FLAG_TC1);//清空标志位
	
}

main.c函数示例

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

uint8_t DataA[] = {0x01,0x02,0x03,0x04};
uint8_t DataB[] = {0,0,0,0};


int main(void)
{
	OLED_Init();
	
	MyDMA_Init((uint32_t)DataA,(uint32_t)DataB,4);

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

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

	while (1)
	{
		DataA[0] ++;
		DataA[1] ++;
		DataA[2] ++;
		DataA[3] ++;
		
	    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);
		
		Delay_ms(1000);
		
		MyDMA_Transfer();
		
		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);
		
		Delay_ms(1000);
	}
}

四、 ADC扫描模式+DMA示例代码及接线图

在这里插入图片描述

AD.c代码示例

#include "Device/Include/stm32f10x.h"   // Device header

uint16_t AD_Value[4];//端菜的目的地


void AD_Init(void)
{
	
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1,ENABLE);
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
	RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1,ENABLE);

	RCC_ADCCLKConfig(RCC_PCLK2_Div6);
	
	GPIO_InitTypeDef GPIO_InitStruct;
	GPIO_InitStruct.GPIO_Mode= GPIO_Mode_AIN;
	GPIO_InitStruct.GPIO_Pin= GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_3;
	GPIO_InitStruct.GPIO_Speed= GPIO_Speed_50MHz;
	GPIO_Init(GPIOA,&GPIO_InitStruct);
	
	//点4个菜(选择4个通带)
	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_InitTypeDef ADC_InitStruct;
	ADC_InitStruct.ADC_ContinuousConvMode= DISABLE;//单次扫描
	ADC_InitStruct.ADC_DataAlign= ADC_DataAlign_Right;
	ADC_InitStruct.ADC_ExternalTrigConv= ADC_ExternalTrigConv_None;
	ADC_InitStruct.ADC_Mode = ADC_Mode_Independent;
	ADC_InitStruct.ADC_NbrOfChannel = 4;//通道数量为4个
	ADC_InitStruct.ADC_ScanConvMode = ENABLE;//扫描模式开启
	ADC_Init(ADC1,&ADC_InitStruct);
	
	DMA_InitTypeDef DMA_InitStruct;
	DMA_InitStruct.DMA_PeripheralBaseAddr = (uint32_t)&ADC1->DR;//端菜源头,在ADC->DR地址
	DMA_InitStruct.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;//想要低16位的数据,所以选半字
	DMA_InitStruct.DMA_PeripheralInc = DMA_PeripheralInc_Disable;//外设地址不自增,始终转运同一个位置的数据
	DMA_InitStruct.DMA_MemoryBaseAddr = (uint32_t)AD_Value;//端菜的目的地
	DMA_InitStruct.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;//一样半字
	DMA_InitStruct.DMA_MemoryInc = DMA_MemoryInc_Enable;//存储器地址需要自增
	DMA_InitStruct.DMA_DIR = DMA_DIR_PeripheralSRC;
	DMA_InitStruct.DMA_BufferSize = 4;//传输数量
	DMA_InitStruct.DMA_Mode = DMA_Mode_Normal;
	DMA_InitStruct.DMA_M2M = DMA_M2M_Disable;//不使用软件触发
	DMA_InitStruct.DMA_Priority = DMA_Priority_Medium;
	DMA_Init(DMA1_Channel1,&DMA_InitStruct);
	
	DMA_Cmd(DMA1_Channel1,ENABLE);
	ADC_DMACmd(ADC1,ENABLE);//开启ADC到DMA的触发信号
	ADC_Cmd(ADC1,ENABLE);
	
	ADC_ResetCalibration(ADC1);
	while (ADC_GetResetCalibrationStatus(ADC1)==SET);
	ADC_StartCalibration(ADC1);
	while (ADC_GetCalibrationStatus(ADC1)==SET);
	
}


void AD_GetValue(void)
{
	//因为DMA是单次模式所以每次转运都要查询写入一下传输计数器
	DMA_Cmd(DMA1_Channel1,DISABLE);
	DMA_SetCurrDataCounter(DMA1_Channel1,4);
	DMA_Cmd(DMA1_Channel1,ENABLE);

	ADC_SoftwareStartConvCmd(ADC1,ENABLE);//因为是ADC是单次模式所以还需要软件触发一下
	
	while(DMA_GetFlagStatus(DMA1_FLAG_TC1) == RESET);
	DMA_ClearFlag(DMA1_FLAG_TC1);//等待DMA转换完成

}

main.c示例代码

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

float Voltage;

int main(void)
{
	OLED_Init();
	AD_Init();
	OLED_ShowString(1,1,"AD0:");
	OLED_ShowString(2,1,"AD1:");
	OLED_ShowString(3,1,"AD3:");
	
	while (1)
	{
		AD_GetValue();
		OLED_ShowNum(1,5,AD_Value[0],4);
		OLED_ShowNum(2,5,AD_Value[1],4);
		OLED_ShowNum(3,5,AD_Value[3],4);
		Delay_ms(100);
	}
}

ADC连续扫描+DMA循环转运模式

ad.c代码

#include "Device/Include/stm32f10x.h"   // Device header

uint16_t AD_Value[4];


void AD_Init(void)
{
	
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1,ENABLE);
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
	RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1,ENABLE);

	RCC_ADCCLKConfig(RCC_PCLK2_Div6);
	
	GPIO_InitTypeDef GPIO_InitStruct;
	GPIO_InitStruct.GPIO_Mode= GPIO_Mode_AIN;
	GPIO_InitStruct.GPIO_Pin= GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_3;
	GPIO_InitStruct.GPIO_Speed= GPIO_Speed_50MHz;
	GPIO_Init(GPIOA,&GPIO_InitStruct);
	
	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_InitTypeDef ADC_InitStruct;
	ADC_InitStruct.ADC_ContinuousConvMode= ENABLE;//****
	ADC_InitStruct.ADC_DataAlign= ADC_DataAlign_Right;
	ADC_InitStruct.ADC_ExternalTrigConv= ADC_ExternalTrigConv_None;
	ADC_InitStruct.ADC_Mode = ADC_Mode_Independent;
	ADC_InitStruct.ADC_NbrOfChannel = 4;
	ADC_InitStruct.ADC_ScanConvMode = ENABLE;//****
	ADC_Init(ADC1,&ADC_InitStruct);
	
	DMA_InitTypeDef DMA_InitStruct;
	DMA_InitStruct.DMA_PeripheralBaseAddr = (uint32_t)&ADC1->DR;
	DMA_InitStruct.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;
	DMA_InitStruct.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
	DMA_InitStruct.DMA_MemoryBaseAddr = (uint32_t)AD_Value;
	DMA_InitStruct.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;
	DMA_InitStruct.DMA_MemoryInc = DMA_MemoryInc_Enable;
	DMA_InitStruct.DMA_DIR = DMA_DIR_PeripheralSRC;
	DMA_InitStruct.DMA_BufferSize = 4;
	DMA_InitStruct.DMA_Mode = DMA_Mode_Circular;//******
	DMA_InitStruct.DMA_M2M = DMA_M2M_Disable;
	DMA_InitStruct.DMA_Priority = DMA_Priority_Medium;
	DMA_Init(DMA1_Channel1,&DMA_InitStruct);
	
	DMA_Cmd(DMA1_Channel1,ENABLE);
	ADC_DMACmd(ADC1,ENABLE);
	ADC_Cmd(ADC1,ENABLE);
	
	ADC_ResetCalibration(ADC1);
	while (ADC_GetResetCalibrationStatus(ADC1)==SET);
	ADC_StartCalibration(ADC1);
	while (ADC_GetCalibrationStatus(ADC1)==SET);
	
	ADC_SoftwareStartConvCmd(ADC1,ENABLE);//******
}

ADC连续模式+DMA更加简洁,且可以直接查询到转换的数值

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,"AD3:");
	while (1)
	{
		OLED_ShowNum(1,5,AD_Value[0],4);
		OLED_ShowNum(2,5,AD_Value[1],4);
		OLED_ShowNum(3,5,AD_Value[3],4);
		Delay_ms(100);
	}
}

总结

DMA多用于跟ADC多通道转换,可以让定时器输出通向ADC,DAC或其他定时器,ADC的触发源可以来自定时器或外部中断,DMA的触发源可以来自ADC、定时器、串口等等

  • 17
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 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、付费专栏及课程。

余额充值