原创 STM32笔记--DMA(USART)的演示

2010-3-30 11:54 3367 12 13 分类: MCU/ 嵌入式
                这里有个小小的例子,来演示DMA模块与系统程序并行工作。
  用串口以低波特率发送一个10K的数据,花费近10s时间,此时按照以往方法,CPU要不断等待数据发送、送数据;或者送数据、进中断、送数据,处理起来比较消耗时间。
  使用了DMA功能以后,用户程序中只需配置好DMA,开启传输后,再也不需要操心,10K数据完成后会有标志位或中断产生,期间可以做任何想做的事,非常方便。
  这个是相应的代码例子,基于STM32F103VBT6


/******************************************************************************
* 本文件实现串口发送功能(通过重构putchar函数,调用printf;或者USART_SendData()
* 这里是一个用串口实现大量数据传输的例子,使用了DMA模块进行内存到USART的传输
* 每当USART的发送缓冲区空时,USART模块产生一个DMA事件,
* 此时DMA模块响应该事件,自动从预先定义好的发送缓冲区中拿出下一个字节送给USART
* 整个过程无需用户程序干预,用户只需启动DMA传输传输即可
* 在仿真器调试时,可以在数据传输过程中暂停运行,此时DMA模块并没有停止
* 串口依然发送,表明DMA传输是一个独立的过程。
* 同时开启接收中断,在串口中断中将数据存入缓冲区,在main主循环中处理
* 作者:jjldc(九九)
* 代码硬件基于万利199元的EK-STM32F开发板,CPU=STM32F103VBT6
*******************************************************************************/

/* Includes ------------------------------------------------------------------*/
#i nclude "stm32f10x_lib.h"
#i nclude "stdio.h"

/* Private typedef -----------------------------------------------------------*/
/* Private define ------------------------------------------------------------*/
#define USART1_DR_Base 0x40013804

/* Private macro -------------------------------------------------------------*/
/* Private variables ---------------------------------------------------------*/
#define SENDBUFF_SIZE 10240
vu8 SendBuff[SENDBUFF_SIZE];
vu8 RecvBuff[10];
vu8 recv_ptr;

/* Private prototypes -----------------------------------------------*/
void RCC_Configuration(void);
void GPIO_Configuration(void);
void NVIC_Configuration(void);
void DMA_Configuration(void);
void USART1_Configuration(void);

int fputc(int ch, FILE *f);
void Delay(void);

/* Private s ---------------------------------------------------------*/
/*******************************************************************************
* Name : main
* Deion : Main program.
* Input : None
* Output : None
* Return : None
*******************************************************************************/
int main(void)
{
 u16 i;
#ifdef DEBUG
 debug();
#endif
 recv_ptr = 0;

 RCC_Configuration();
 GPIO_Configuration();
 NVIC_Configuration();
 DMA_Configuration();
 USART1_Configuration();

 printf("\r\nSystem Start...\r\n");
 printf("Initialling SendBuff... \r\n");
 for(i=0;i {
 SendBuff = i&0xff;
 }
 printf("Initial success!\r\nWaiting for transmission...\r\n");
 //发送去数据已经准备好,按下按键即开始传输
 while(GPIO_ReadInputDataBit(GPIOD, GPIO_Pin_3));

 printf("Start DMA transmission!\r\n");

 //这里是开始DMA传输前的一些准备工作,将USART1模块设置成DMA方式工作
 USART_DMACmd(USART1, USART_DMAReq_Tx, ENABLE);
 //开始一次DMA传输!
 DMA_Cmd(DMA1_Channel4, ENABLE);

 //等待DMA传输完成,此时我们来做另外一些事,点灯
 //实际应用中,传输数据期间,可以执行另外的任务
 while(DMA_GetFlagStatus(DMA1_FLAG_TC4) == RESET)
 {
 LED_1_REV; //LED翻转
 Delay(); //浪费时间
 }
 //DMA传输结束后,自动关闭了DMA通道,而无需手动关闭
 //下面的语句被注释
 //DMA_Cmd(DMA1_Channel4, DISABLE);

 printf("\r\nDMA transmission successful!\r\n");


 /* Infinite loop */
 while (1)
 {
 }
}

/*******************************************************************************
* Name : 重定义系统putchar函数int fputc(int ch, FILE *f)
* Deion : 串口发一个字节
* Input : int ch, FILE *f
* Output :
* Return : int ch
* 这个是使用printf的关键
*******************************************************************************/
int fputc(int ch, FILE *f)
{
 //USART_SendData(USART1, (u8) ch);
 USART1->DR = (u8) ch;

 /* Loop until the end of transmission */
 while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET)
 {
 }

 return ch;
}

/*******************************************************************************
* Name : Delay
* Deion : 延时函数
* Input : None
* Output : None
* Return : None
*******************************************************************************/
void Delay(void)
{
 u32 i;
 for(i=0;i<0xF0000;i++);
 return;
}

/*******************************************************************************
* Name : RCC_Configuration
* Deion : 系统时钟设置
* Input : None
* Output : None
* Return : None
*******************************************************************************/
void RCC_Configuration(void)
{
 ErrorStatus HSEStartUpStatus;

 //使能外部晶振
 RCC_HSEConfig(RCC_HSE_ON);
 //等待外部晶振稳定
 HSEStartUpStatus = RCC_WaitForHSEStartUp();
 //如果外部晶振启动成功,则进行下一步操作
 if(HSEStartUpStatus==SUCCESS)
 {
 //设置HCLK(AHB时钟)=SYSCLK
 RCC_HCLKConfig(RCC_SYSCLK_Div1);

 //PCLK1(APB1) = HCLK/2
 RCC_PCLK1Config(RCC_HCLK_Div2);

 //PCLK2(APB2) = HCLK
 RCC_PCLK2Config(RCC_HCLK_Div1);

 //FLASH时序控制
 //推荐值:SYSCLK = 0~24MHz Latency="0"
 // SYSCLK = 24~48MHz Latency="1"
 // SYSCLK = 48~72MHz Latency="2"
 FLASH_SetLatency(FLASH_Latency_2);
 //开启FLASH预取指功能
 FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);

 //PLL设置 SYSCLK/1 * 9 = 8*1*9 = 72MHz
 RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_9);
 //启动PLL
 RCC_PLLCmd(ENABLE);
 //等待PLL稳定
 while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET);
 //系统时钟SYSCLK来自PLL输出
 RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);
 //切换时钟后等待系统时钟稳定
 while(RCC_GetSYSCLKSource()!=0x08);


 /*
 //设置系统SYSCLK时钟为HSE输入
 RCC_SYSCLKConfig(RCC_SYSCLKSource_HSE);
 //等待时钟切换成功
 while(RCC_GetSYSCLKSource() != 0x04);
 */
 }

 //下面是给各模块开启时钟
 //启动GPIO
 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_GPIOB | \
 RCC_APB2Periph_GPIOC | RCC_APB2Periph_GPIOD,\
 ENABLE);
 //启动AFIO
 RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO, ENABLE);
 //启动USART1
 RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1, ENABLE);
 //启动DMA时钟
 RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);

}



/*******************************************************************************
* Name : GPIO_Configuration
* Deion : GPIO设置
* Input : None
* Output : None
* Return : None
*******************************************************************************/
void GPIO_Configuration(void)
{
 GPIO_InitTypeDef GPIO_InitStructure;

 //PC口4567脚设置GPIO输出,推挽 2M
 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4 | GPIO_Pin_5 | GPIO_Pin_6 | GPIO_Pin_7;
 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_2MHz;
 GPIO_Init(GPIOC, &GPIO_InitStructure);

 //KEY2 KEY3 JOYKEY
 //位于PD口的3 4 11-15脚,使能设置为输入
 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3 | GPIO_Pin_4 | GPIO_Pin_11 | GPIO_Pin_12 |\
 GPIO_Pin_13 | GPIO_Pin_14 | GPIO_Pin_15;
 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
 GPIO_Init(GPIOD, &GPIO_InitStructure);

 //USART1_TX
 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);

 //USART1_RX
 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
 GPIO_Init(GPIOA, &GPIO_InitStructure);

}



/*******************************************************************************
* Name : NVIC_Configuration
* Deion : NVIC设置
* Input : None
* Output : None
* Return : None
*******************************************************************************/
void NVIC_Configuration(void)
{
 NVIC_InitTypeDef NVIC_InitStructure;

#ifdef VECT_TAB_RAM
 // Set the Vector Table base location at 0x20000000
 NVIC_SetVectorTable(NVIC_VectTab_RAM, 0x0);
#else /* VECT_TAB_FLASH */
 // Set the Vector Table base location at 0x08000000
 NVIC_SetVectorTable(NVIC_VectTab_FLASH, 0x0);
#endif

 //设置NVIC优先级分组为Group2:0-3抢占式优先级,0-3的响应式优先级
 NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
 //串口接收中断打开
 NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQChannel;
 NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
 NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;
 NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
 NVIC_Init(&NVIC_InitStructure);
}


/*******************************************************************************
* Name : USART1_Configuration
* Deion : NUSART1设置
* Input : None
* Output : None
* Return : None
*******************************************************************************/
void USART1_Configuration(void)
{
 USART_InitTypeDef USART_InitStructure;

 USART_InitStructure.USART_BaudRate = 9600;
 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_Tx | USART_Mode_Rx;
 USART_Init(USART1, &USART_InitStructure);

 USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);

 USART_Cmd(USART1, ENABLE);
}


void DMA_Configuration(void)
{
 DMA_InitTypeDef DMA_InitStructure;
 //DMA设置:
 //设置DMA源:内存地址&串口数据寄存器地址
 //方向:内存-->外设
 //每次传输位:8bit
 //传输大小DMA_BufferSize=SENDBUFF_SIZE
 //地址自增模式:外设地址不增,内存地址自增1
 //DMA模式:一次传输,非循环
 //优先级:中
 DMA_DeInit(DMA1_Channel4);
 DMA_InitStructure.DMA_PeripheralBaseAddr = USART1_DR_Base;
 DMA_InitStructure.DMA_MemoryBaseAddr = (u32)SendBuff;
 DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralDST;
 DMA_InitStructure.DMA_BufferSize = SENDBUFF_SIZE;
 DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
 DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
 DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;
 DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;
 DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;
 DMA_InitStructure.DMA_Priority = DMA_Priority_Medium;
 DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;
 DMA_Init(DMA1_Channel4, &DMA_InitStructure);
}
PARTNER CONTENT

文章评论1条评论)

登录后参与讨论

用户1275153 2012-8-2 22:51

TKS!
相关推荐阅读
用户89529 2010-09-25 17:03
stm32 malloc:动态分配内存函数
函数原型:void *malloc(unsigned size)头文件:#include<stdlib.h>是否是标准函数:是函数功能:动态分配一块内存空间,size为指定的分配空间的大小...
用户89529 2010-08-09 14:13
SD卡的读写和FatFS文件系统
因为要用,学习了一下SPI操作SD卡,同时移植了一个免费开源的FAT文件系统:FatFS。感觉挺好,在单片机上实现了读写文件的操作,接下来就可以解释我的G代码咯!  我的SD卡底层操作参考了网上几种常...
用户89529 2010-08-09 14:11
转一篇比较详细介绍FatFs文件系统移植的文章
FatFs文件系统的移植    因为需要,又不想自己写,所以就移植了一个文件系统。    说下我的硬件和开发工具:接成 TRUE IDE 模式下的CF卡(也就是相当于一块硬盘了),三星S3C2440的...
用户89529 2010-08-05 11:53
关于STM32的systick定时器的详细说明
我不得不说意法半导体确实有点风骚!甚至有点变态。我对ST文档 STM32F10XXX参考手册的编辑水平真是不敢恭维。手册中好多说明都是含糊不清,甚至将好多对初学者来说很重要的地方都一笔带过,让人着实摸...
用户89529 2010-05-11 19:40
stm32 fsmc数据地址线的关系
      stm32的fsmc接口设置的资料很少,所以现在对各个寄存器的具体应用还不是很明白,最近写了一个程序是把以前8位数据线读写sram的程序,改为16位数据线。有了一个小小的发现,现在把它写出...
用户89529 2010-05-07 11:59
stm32关于BOOT0和BOOT1
BOOT0和BOOT1STM32三种启动模式对应的存储介质均是芯片内置的,它们是:1)用户闪存 = 芯片内置的Flash。2)SRAM = 芯片内置的RAM区,就是内存啦。3)系统存储器 = 芯片内部...
EE直播间
更多
我要评论
1
12
关闭 站长推荐上一条 /3 下一条