STM32基础-------DMA
写在前面
这两天学习了DMA,在这里写一下关于自己对DMA的一些理解。
第一次写博客,不足地方肯定很多,错误地方希望大家指正。
一、DMA介绍
1.DMA是什么
DMA—Direct Memory Access, 翻译过来就是数据直接存储访问。官方一点的描述是:
DMA传输将数据从一个地址空间复制到另一个地址空间,提供在在外设和存储器之间或者存储器和存储器之间的高速数据传输。
所以DMA其实就相当于一个搬运工,把数据从一个地方搬到另一个地方。当大量数据需要传输的时候,我们就给DMA发送命令,让DMA去处理这些数据,从而解放CPU,让CPU去处理更复杂的事件。
2.DMA框图
从上图可见,STM32有2个DMA,它们都挂载在AHB总线上,而且和Cortex-M3核心共享数据总线,可以直接进行存储器数据传输。DMA1有7个通道,DMA2有5个通道,每个通道分别对应着不同的外设寄存器。
2.1DMA请求和仲裁
当外设产生DMA请求时,这个请求信号发送给DMA,首先由DMA的仲裁器仲裁,根据通道优先级决定哪个通道先进行数据传输。DMA通道优先级由软件和硬件共同管理。 软件:有四个等级,可以通过DMA_CCRx寄存器配置。 硬件:如果两个通道的优先级相同,则根据通道编号来决定哪个通道优先级更高,编号小的优先级高。
2.2DMA通道
当外设和存储器之间进行数据传输时,每个通道都可以在有固定地址的外设寄存器和存储器之间进行DMA传输。DMA传输数据量是可以通过软件来配置的,最大为65535。 从上图我们可以看到每个通道都有对应的外设,但是DMA不是也可以进行存储器到存储器之间的传输吗?为什么没有看到对应的通道呢?其实,DMA的任何一个通道都可以进行存储器到存储器之间的数据传输。只不过在存储器到存储器之间进行数据传输前,需要将DMA_CCRx的MEM2MEM位置位。 下图更加清晰地表现出DMA通道优先级和数据传输模式的关系 从图中可见,每个通道都有MEM2MEM位,因此均可通过配置该位来实现存储器到存储器之间数据传输。
2.3DMA数据传输格式及对齐
DMA进行数据传输时,源地址和目标地址的数据宽度可以设定1byte、2byte、4byte。 下图为源地址数据宽度1byte,目标地址数据宽度1byte数据传输及对齐方式。 下图为源地址数据宽度为1byte,目标地址数据宽度为4byte数据传输及对齐方式。 下图为源地址为4byte,目标地址为1byte数据传输及对齐方式
二、DMA代码详解
Memory to Memory
程序目标:使用DMA实现存储器到存储器的数据传输。将flash ROM中的数据复制到SRAM中
DMA初始化
dam_mtm.c文件
#include"dma_mtm.h"
const u32 RSC_CONST_BUFFER[BUFFERSIZE] =
{
0x11111111,0x22222222,0x33333333,
0x44444444,0x55555555,0x66666666,
0x77777777,0x88888888,0x99999999
};
u32 DST_BUFFER[BUFFERSIZE];
void DMA_MTM_Config(void)
{
DMA_InitTypeDef DMA_InitStruct;
RCC_AHBPeriphClockCmd (MTM_DMA_CLK ,ENABLE);
DMA_InitStruct.DMA_PeripheralBaseAddr = (u32)RSC_CONST_BUFFER;
DMA_InitStruct.DMA_MemoryBaseAddr = (u32)DST_BUFFER;
DMA_InitStruct.DMA_DIR = DMA_DIR_PeripheralSRC ;
DMA_InitStruct.DMA_BufferSize = BUFFERSIZE;
DMA_InitStruct.DMA_MemoryDataSize = DMA_MemoryDataSize_Word;
DMA_InitStruct.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Word;
DMA_InitStruct.DMA_MemoryInc = DMA_MemoryInc_Enable;
DMA_InitStruct.DMA_PeripheralInc = DMA_PeripheralInc_Enable;
DMA_InitStruct.DMA_Mode = DMA_Mode_Normal;
DMA_InitStruct.DMA_M2M = DMA_M2M_Enable;
DMA_InitStruct.DMA_Priority = DMA_Priority_High;
DMA_Init (MTM_DMA,&DMA_InitStruct);
DMA_Cmd (MTM_DMA,ENABLE);
}
u8 Buffer_Compare(const u32* pBuffer1,u32* pBuffer2,u16 len)
{
while(len--)
{
if(*pBuffer1 != *pBuffer2)
return 0;
pBuffer1++;
pBuffer2++;
}
return 1;
}
dam_mtm.h文件
#ifndef DMA_MTM_H
#define DMA_MTM_H
#include "stm32f10x.h"
#define BUFFERSIZE 9
#define MTM_DMA DMA1_Channel1
#define MTM_DMA_CLK RCC_AHBPeriph_DMA1
void DMA_MTM_Config(void);
u8 Buffer_Compare(const u32* pBuffer1,u32* pBuffer2,u16 len);
#endif
main.c文件
#include "delay.h"
#include "sys.h"
#include "./dma/dma_mtm.h"
#include "./dma/dma_mtp.h"
#include "./dma/dma_ptm.h"
#include "./led/led.h"
extern u32 DST_BUFFER[BUFFERSIZE];
extern const u32 RSC_CONST_BUFFER[BUFFERSIZE];
int main(void)
{
int i;
SystemInit();
delay_init();
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
LED_GPIO_Config();
uart_init(115200);
DMA_MTM_Config();
printf("RSC_CONST_BUFFER:\n");
for(i=0; i<BUFFERSIZE; i++)
{
printf("0x%x ",RSC_CONST_BUFFER[i]);
}
printf("\nDST_BUFFER:\n");
for(i=0; i<BUFFERSIZE; i++)
{
printf("0x%x ",DST_BUFFER[i]);
}
if (Buffer_Compare(RSC_CONST_BUFFER,DST_BUFFER,BUFFERSIZE) == 0)
{
printf("\nData transmission failure\n");
}
else
{
printf("\nData transmission completed\n");
}
while(1)
{
delay_ms(133);
LED_G_TOGGLE;
}
}
Memory to Peripheral
程序目标:将内部SRAM中的数据通过DMA传输到USART DR寄存器中,通过串口打印数据
dma_mtp.c文件
#include"dma_mtp.h"
u8 SRC_BUFFER[USART_BUFFERSIZE];
void DMA_MTP_Config(void)
{
DMA_InitTypeDef DMA_InitStruct;
RCC_AHBPeriphClockCmd (MTP_DMA_CLK ,ENABLE);
DMA_InitStruct.DMA_PeripheralBaseAddr = (u32)DST_ADDR;
DMA_InitStruct.DMA_MemoryBaseAddr = (u32)SRC_BUFFER;
DMA_InitStruct.DMA_DIR = DMA_DIR_PeripheralDST ;
DMA_InitStruct.DMA_BufferSize = USART_BUFFERSIZE;
DMA_InitStruct.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;
DMA_InitStruct.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;
DMA_InitStruct.DMA_MemoryInc = DMA_MemoryInc_Enable;
DMA_InitStruct.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
DMA_InitStruct.DMA_Mode = DMA_Mode_Normal;
DMA_InitStruct.DMA_Priority = DMA_Priority_High;
DMA_InitStruct.DMA_M2M = DMA_M2M_Disable;
DMA_ClearFlag(DMA1_FLAG_TC4);
DMA_Init (MTP_DMA,&DMA_InitStruct);
DMA_Cmd (MTP_DMA,ENABLE);
}
dma_mpt.h文件
#ifndef DMA_MTP_H
#define DMA_MTP_H
#include "stm32f10x.h"
#define USART_BUFFERSIZE 10000
#define MTP_DMA DMA1_Channel4
#define MTP_DMA_CLK RCC_AHBPeriph_DMA1
#define DST_ADDR (USART1_BASE + 0x04)
void DMA_MTP_Config(void);
#endif
main.c文件
#include "delay.h"
#include "sys.h"
#include "./dma/dma_mtm.h"
#include "./dma/dma_mtp.h"
#include "./dma/dma_ptm.h"
#include "./led/led.h"
extern u8 SRC_BUFFER[USART_BUFFERSIZE];
int main(void)
{
int i;
SystemInit();
delay_init();
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
LED_GPIO_Config();
uart_init(115200);
for (i=0; i<USART_BUFFERSIZE;i++)
{
SRC_BUFFER[i] = 'c';
}
DMA_MTP_Config();
USART_DMACmd(USART1,USART_DMAReq_Tx,ENABLE);
while(1)
{
delay_ms(133);
LED_G_TOGGLE;
}
}
Peripheral to Memory
程序目标:将串口接收到的数据通过DMA传送到SRAM中
串口初始化
void uart_init(u32 bound)
{
GPIO_InitTypeDef GPIO_InitStructure;
USART_InitTypeDef USART_InitStructure;
NVIC_InitTypeDef NVIC_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1|RCC_APB2Periph_GPIOA, ENABLE);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
GPIO_Init(GPIOA, &GPIO_InitStructure);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
GPIO_Init(GPIOA, &GPIO_InitStructure);
NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3 ;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);
USART_InitStructure.USART_BaudRate = bound;
USART_InitStructure.USART_WordLength = USART_WordLength_8b;
USART_InitStructure.USART_StopBits = USART_StopBits_1;
USART_InitStructure.USART_Parity = USART_Parity_No;
USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;
USART_Init(USART1, &USART_InitStructure);
USART_ITConfig(USART1, USART_IT_IDLE, ENABLE);
USART_Cmd(USART1, ENABLE);
}
串口中断
void USART1_IRQHandler(void)
{
uint16_t t;
if(USART_GetITStatus(USART1,USART_IT_IDLE) == SET)
{
DMA_Cmd(DMA1_Channel5,DISABLE);
t = DMA_GetCurrDataCounter(DMA1_Channel5);
Usart_SendArray(USART1,DST_Data,DMA_DATASIZE-t);
DMA_SetCurrDataCounter(DMA1_Channel5,DMA_DATASIZE);
DMA_Cmd(DMA1_Channel5,ENABLE);
USART_ReceiveData(USART1);
USART_ClearFlag(USART1,USART_FLAG_IDLE);
}
}
串口中断完成的任务是,当串口接收到一包数据后,数据通过DMA传送到SRAM中,并产生中断。这时将DMA关闭,防止串口再接收到数据从而扰乱发送,接着重置DMA接收数据数量,再次开启DMA接收数据。 也就是说当产生中断时,DMA已经将数据搬运完成了。
dma_ptm.c文件
#include"dma_ptm.h"
u8 DST_Data[DMA_DATASIZE];
void DMA_PTM_Config(void)
{
DMA_InitTypeDef DMA_InitStruct;
RCC_AHBPeriphClockCmd (RCC_AHBPeriph_DMA1 ,ENABLE);
DMA_InitStruct.DMA_MemoryBaseAddr = (u32)DST_Data;
DMA_InitStruct.DMA_PeripheralBaseAddr = SOR_ADDR;
DMA_InitStruct.DMA_DIR = DMA_DIR_PeripheralSRC;
DMA_InitStruct.DMA_BufferSize = DMA_DATASIZE;
DMA_InitStruct.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;
DMA_InitStruct.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte ;
DMA_InitStruct.DMA_MemoryInc = DMA_MemoryInc_Enable ;
DMA_InitStruct.DMA_PeripheralInc = DMA_PeripheralInc_Disable ;
DMA_InitStruct.DMA_Mode = DMA_Mode_Normal;
DMA_InitStruct.DMA_Priority = DMA_Priority_High ;
DMA_InitStruct.DMA_M2M = DMA_M2M_Disable ;
DMA_Init (DMA1_Channel5,&DMA_InitStruct);
DMA_Cmd(DMA1_Channel5,ENABLE);
}
dma_ptm.h文件
#ifndef DMA_PTM_H_
#define DMA_PTM_H_
#include "stm32f10x.h"
#define SOR_ADDR (USART1_BASE+0x04)
#define DMA_DATASIZE 1000
#define PTM_DMA DMA1_Channel5
void DMA_PTM_Config(void);
#endif
main.c文件
#include "delay.h"
#include "sys.h"
#include "./dma/dma_mtm.h"
#include "./dma/dma_mtp.h"
#include "./dma/dma_ptm.h"
#include "./led/led.h"
#define MTM (1)
#define MTP (0)
#define PTM (0)
extern u32 DST_BUFFER[BUFFERSIZE];
extern const u32 RSC_CONST_BUFFER[BUFFERSIZE];
extern u8 SRC_BUFFER[USART_BUFFERSIZE];
extern u8 DST_Data[DMA_DATASIZE];
int main(void)
{
int i;
SystemInit();
delay_init();
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
LED_GPIO_Config();
uart_init(115200);
DMA_PTM_Config();
USART_DMACmd(USART1,USART_DMAReq_Rx,ENABLE);
while(1)
{
delay_ms(133);
LED_G_TOGGLE;
}
}
|