STM32F407的串口采用DMA收发数据

发布者:素雅之韵最新更新时间:2016-06-03 来源: eefocus关键字:STM32F407  串口  DMA  收发数据 手机看文章 扫描二维码
随时随地手机看文章
环境:

主机:WIN8

开发环境:MDK5.13

mcu: stm32f407VGT6

 

说明:

之前用STM32F103实现DMA收发串口数据,现在项目中采用STM32F407,所以将此机制移植到F4上。

STM32F103上用DMA收发串口数据文章:

STM32的串口采用DMA方式发送数据测试

STM32的串口采用DMA方式接收数据测试

 

源代码:

串口初始化代码:

 

/*
*							初始化串口
/

static void init_uart(void)
{
	//定义中断结构体
	NVIC_InitTypeDef NVIC_InitStructure ;
	//定义IO初始化结构体
 	GPIO_InitTypeDef GPIO_InitStructure;
	//定义串口结构体  
	USART_InitTypeDef USART_InitStructure;
	//定义DMA结构体
	DMA_InitTypeDef DMA_InitStructure;

	//打开串口对应的外设时钟  
    RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2, ENABLE);
	 
	//串口发DMA配置  
	//启动DMA时钟
    RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA1, ENABLE);
	//DMA发送中断设置
	NVIC_InitStructure.NVIC_IRQChannel = DMA1_Stream6_IRQn;
	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 2;
	NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
	NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
	NVIC_Init(&NVIC_InitStructure);
	//DMA通道配置
	DMA_DeInit(DMA1_Stream6);
	DMA_InitStructure.DMA_Channel = DMA_Channel_4; 
	//外设地址
	DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)(&USART2->DR);
	//内存地址
	DMA_InitStructure.DMA_Memory0BaseAddr = (uint32_t)Tx_Buf_Gsm;
	//dma传输方向
	DMA_InitStructure.DMA_DIR = DMA_DIR_MemoryToPeripheral;
	//设置DMA在传输时缓冲区的长度
	DMA_InitStructure.DMA_BufferSize = TX_LEN_GSM;
	//设置DMA的外设递增模式,一个外设
	DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
	//设置DMA的内存递增模式
	DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
	//外设数据字长
	DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;
	//内存数据字长
	DMA_InitStructure.DMA_MemoryDataSize = DMA_PeripheralDataSize_Byte;
	//设置DMA的传输模式
	DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;
	//设置DMA的优先级别
	DMA_InitStructure.DMA_Priority = DMA_Priority_High;
	
	//指定如果FIFO模式或直接模式将用于指定的流 : 不使能FIFO模式  
	DMA_InitStructure.DMA_FIFOMode = DMA_FIFOMode_Disable;    
	//指定了FIFO阈值水平
	DMA_InitStructure.DMA_FIFOThreshold = DMA_FIFOThreshold_HalfFull;        
	//指定的Burst转移配置内存传输 
	DMA_InitStructure.DMA_MemoryBurst = DMA_MemoryBurst_Single;       
	//指定的Burst转移配置外围转移 */  
	DMA_InitStructure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single; 

	//配置DMA1的通道         
	DMA_Init(DMA1_Stream6, &DMA_InitStructure);  
	//使能中断
	DMA_ITConfig(DMA1_Stream6,DMA_IT_TC,ENABLE);   

	//串口收DMA配置  
	//启动DMA时钟
    RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA1, ENABLE);
	//DMA通道配置
	DMA_DeInit(DMA1_Stream5);
	DMA_InitStructure.DMA_Channel = DMA_Channel_4;
	//外设地址
	DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)(&USART2->DR);
	//内存地址
	DMA_InitStructure.DMA_Memory0BaseAddr = (uint32_t)Rx_Buf_Gsm;
	//dma传输方向
	DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralToMemory;
	//设置DMA在传输时缓冲区的长度
	DMA_InitStructure.DMA_BufferSize = RX_LEN_GSM;
	//设置DMA的外设递增模式,一个外设
	DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
	//设置DMA的内存递增模式
	DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
	//外设数据字长
	DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;
	//内存数据字长
	DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;
	//设置DMA的传输模式
	DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;
	//设置DMA的优先级别
	DMA_InitStructure.DMA_Priority = DMA_Priority_VeryHigh;
	
	//指定如果FIFO模式或直接模式将用于指定的流 : 不使能FIFO模式  
	DMA_InitStructure.DMA_FIFOMode = DMA_FIFOMode_Disable;    
	//指定了FIFO阈值水平
	DMA_InitStructure.DMA_FIFOThreshold = DMA_FIFOThreshold_HalfFull;        
	//指定的Burst转移配置内存传输 
	DMA_InitStructure.DMA_MemoryBurst = DMA_MemoryBurst_Single;       
	//指定的Burst转移配置外围转移 */  
	DMA_InitStructure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single; 
	
	//配置DMA1的通道         
	DMA_Init(DMA1_Stream5, &DMA_InitStructure);  
	//使能通道
	DMA_Cmd(DMA1_Stream5,ENABLE);
	
    //初始化串口参数  
    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_InitStructure.USART_BaudRate = DEFAULT_BAUD_GSM; 
	//初始化串口 
    USART_Init(USART2,&USART_InitStructure);  
	
	//中断配置
	USART_ITConfig(USART2,USART_IT_TC,DISABLE);
	USART_ITConfig(USART2,USART_IT_RXNE,DISABLE);
	USART_ITConfig(USART2,USART_IT_IDLE,ENABLE);  

	//配置中断  
	//NVIC_PriorityGroupConfig(NVIC_PriorityGroup_3);
    NVIC_InitStructure.NVIC_IRQChannel = USART2_IRQn;               //通道设置为串口中断  
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;       //中断占先等级
    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;              //中断响应优先级 
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;                 //打开中断  
    NVIC_Init(&NVIC_InitStructure);   
        
	//采用DMA方式发送
	USART_DMACmd(USART2,USART_DMAReq_Tx,ENABLE);
	//采用DMA方式接收
	USART_DMACmd(USART2,USART_DMAReq_Rx,ENABLE);

	//中断配置
	USART_ITConfig(USART2,USART_IT_TC,DISABLE);
	USART_ITConfig(USART2,USART_IT_RXNE,DISABLE);
	USART_ITConfig(USART2,USART_IT_TXE,DISABLE);
	USART_ITConfig(USART2,USART_IT_IDLE,ENABLE);  
    //启动串口  
    USART_Cmd(USART2, ENABLE);    

    //设置IO口时钟      
	RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE); 
	GPIO_PinAFConfig(GPIOA,GPIO_PinSource2,GPIO_AF_USART2);  
	GPIO_PinAFConfig(GPIOA,GPIO_PinSource3,GPIO_AF_USART2);

    //管脚模式:输出口
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;	
    //类型:推挽模式
	GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;	
    //上拉下拉设置
	GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;	
	//IO口速度
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;
    //管脚指定
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;
    //初始化
	GPIO_Init(GPIOA, &GPIO_InitStructure);
	
    //管脚模式:输入口
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;	
    //上拉下拉设置
	GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;	
    //管脚指定
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3;
    //初始化
	GPIO_Init(GPIOA, &GPIO_InitStructure);      
}

 

 

发送数据:

 

/*
*							接口函数:向gsm模块发送数据
*参数:data:发送数据存放地址
*	  size:发送数据字节数
/

void drv_gsm_tx(uint8_t *data,uint16_t size)
{
	//等待空闲
	while (Flag_Tx_Gsm_Busy);
	Flag_Tx_Gsm_Busy = 1;
	//数据
	memcpy(Tx_Buf_Gsm,data,size);
	//设置传输数据长度
	DMA_SetCurrDataCounter(DMA1_Stream6,size);
	//打开DMA,开始发送
 	DMA_Cmd(DMA1_Stream6,ENABLE);
}

 

 

 

 

 

 

中断处理函数:

/*
*							接口函数:DMA发送中断处理函数
/

void drv_gsm_deal_irq_dma_tx(void)
{
	if(DMA_GetITStatus(DMA1_Stream6,DMA_IT_TCIF6) != RESET) 
	{
		//清除标志位
		DMA_ClearFlag(DMA1_Stream6,DMA_FLAG_TCIF6);
		//关闭DMA
		DMA_Cmd(DMA1_Stream6,DISABLE);
		//打开发送完成中断,发送最后两个字节
		USART_ITConfig(USART2,USART_IT_TC,ENABLE);
	}
}

/*
*							处理发送完成中断
*返回:0:未产生,1:已经产生
/

uint8_t drv_gsm_deal_irq_tx_end(void)
{
	if(USART_GetITStatus(USART2, USART_IT_TC) != RESET)
    {
		//关闭发送完成中断
		USART_ITConfig(USART2,USART_IT_TC,DISABLE);
		//发送完成
        Flag_Tx_Gsm_Busy = 0;
		
		return 1;
    } 
	
	return 0;
}

/*
*							处理接收完成中断
*参数:buf:接收的数据
*     len:接收的数据长度
*返回:0:未产生,其他:已经产生,此值为接收的数据长度
/

uint8_t drv_gsm_deal_irq_rx_end(uint8_t *buf)
{	
	uint16_t len = 0;
	
	//接收完成中断
	if(USART_GetITStatus(USART2, USART_IT_IDLE) != RESET)
    {
    	USART2->SR;
    	USART2->DR; //清USART_IT_IDLE标志
		//关闭DMA
    	DMA_Cmd(DMA1_Stream5,DISABLE);
		//清除标志位
		DMA_ClearFlag(DMA1_Stream5,DMA_FLAG_TCIF5);
		
		//获得接收帧帧长
		len = RX_LEN_GSM - DMA_GetCurrDataCounter(DMA1_Stream5);
		memcpy(buf,Rx_Buf_Gsm,len);
		
		//设置传输数据长度
		DMA_SetCurrDataCounter(DMA1_Stream5,RX_LEN_GSM);
    	//打开DMA
		DMA_Cmd(DMA1_Stream5,ENABLE);

		return len;
    } 
	
	return 0;
}


 

 

 

 

中断函数:

 

/*
*							GSM模块:DMA发送中断处理函数
/

void DMA1_Stream6_IRQHandler(void)
{
    gsm_dma_tx_irq_handler();
}
   
/*
*							GSM模块:串口中断处理函数
/

void USART2_IRQHandler(void) 
{
	gsm_irq_handler();
}

 

 

 

 

 

注意:

接收完成中断中必须要清除标志位

 

//清除标志位
		DMA_ClearFlag(DMA1_Stream5,DMA_FLAG_TCIF5);
并且必须在关闭DMA之后,这个语句在F103中并非必须的

 


中断处理函数:

 

/*
*							接口函数:DMA发送中断处理函数
/

void gsm_dma_tx_irq_handler(void)
{
	inf_gsm_deal_irq_dma_tx();
}
   
/*
*							接口函数:串口中断处理函数
*参数:data:接收数据存放地址
*返回:接收数据长度
/

void gsm_irq_handler(void)                              
{   
	struct _Gsm_Rx rx;
	uint8_t i = 0;
	
	//发送完成中断处理
	inf_gsm_deal_irq_tx_end();
	
	//接收完成中断处理
	rx.len = inf_gsm_deal_irq_rx_end(rx.buf);
	if (rx.len != 0)
	{
		//通知观察者
		for (i = 0;i < Len_Observer;i++)
		{
			Observer[i](rx);
		}
    } 
} 

 

 

 

 

 


接口函数定义:

 

/*
*							接口函数:DMA发送中断处理函数
/

void inf_gsm_deal_irq_dma_tx(void)
{
	drv_gsm_deal_irq_dma_tx();
}

/*
*							处理发送完成中断
*返回:0:未产生,1:已经产生
/

uint8_t inf_gsm_deal_irq_tx_end(void)
{
	return drv_gsm_deal_irq_tx_end();
}

/*
*							处理接收完成中断
*参数:buf:接收的数据
*     len:接收的数据长度
*返回:0:未产生,其他:已经产生,此值为接收的数据长度
/

uint8_t inf_gsm_deal_irq_rx_end(uint8_t *buf)
{
	return drv_gsm_deal_irq_rx_end(buf);
}

关键字:STM32F407  串口  DMA  收发数据 引用地址:STM32F407的串口采用DMA收发数据

上一篇:STM32F4 SPI2初始化及收发数据使用库函数
下一篇:ARM裸机串口UART

推荐阅读最新更新时间:2024-03-16 14:56

单片机简单的串口通信
1 /*------------------------------------------- 2 简单的串口通信 3 -------------------------------------------*/ 4 #include reg51.h 5 6 #define uint unsigned int 7 #define uchar unsigned char 8 9 10 uchar flag,a,i; //标志接收串口中断标志,接收SBUF内容 11 uchar code table = I get ; 12 /*------------------------------------
[单片机]
STM32串口用中断还是用轮询
1.从轮询到中断 很多同学都不喜欢用中断,而偏爱用轮询的操作方式。 这是不是和我们的天性有关呢?每个人都喜欢一切尽在掌握中,肯定都不喜欢被打断。我们常常都有这样的经验:正在跟别人说一件事,然后突然有个电话打进来,Call打完后突然记不起来刚才讲到哪了!这种糟糕的体验对我们影响是如此深刻,以至于我们认定机器可能也是这样吧,频繁的中断会不会把事情搞乱呢?好在机器虽然大部分时间都比人笨一些,但在处理这种问题上却能做到一丝不苟。机器在中断来的时候总会老老实实地先把当前正在做的记录下来,然后转去处理中断事件,中断处理完后分毫不差地恢复原来的工作。 仔细想一想,我们是不是也可以在接电话前先用个小本儿记录一下正在讲的事情呢?我们为什么没有这么做
[单片机]
STM32 ADC用DMA方式传输数据出错的解决方法
在ADC高速采集数据时,通过DMA一次性获取大量的转换数据。比如5个通道,一次获取1000组数据。获得1000组数据后再统一处理,然后触发下一次转换。在调试过程中发现获取的通道数据序号出错。经过一番折腾终于调通,程序如下 定义 #define adNum 10000 u16 adsample ; //#define DMA1_Channel4_IRQn_EN 1//DMA 电压采集中断 #ifdef DMA1_Channel4_IRQn_EN #define DMA1_Channel4_IRQn_PreemptionPriority 1 #define DMA1_Channel4_IRQn_SubPriority
[单片机]
STM32F407的TCP编程出现客户端无法连接上服务器
单写一篇文章记录这些问题,因为有的问题实在是困扰了我太久太久了,终于解决了!!! 1.STM32F407的TCP编程,TCP_SERVER测试完全正常,TCP_CLIENT测试过程中,开发板作为客户端,出现客户端无法连接上服务器的问题 解决方法:这个问题不会出现在用路由器接入开发板和电脑的情况,只会出现在电脑和开发板用网线直连的情况,解决方法很简单,关闭电脑防火墙!!! 2.DHCP获取IP地址一致失败、获取超时等问题,只能使用静态IP地址 解决方法:DHCP,又名动态主机配置服务,它的作用是用来分配IP地址,使网络环境中的主机动态的获得IP地址、Gateway地址、DNS服务器地址等信息,但是分配IP地址这东西笔记本电脑
[单片机]
<font color='red'>STM32F407</font>的TCP编程出现客户端无法连接上服务器
STM32串口通信之Hello(STM32_11)
一、开发板硬件介绍 1、MCU串口管脚 本程序使用STM32F103ZET6芯片,芯片有5个串口,其中有3个USART和两个UART,本程序使用异步方式,5个串口的管脚。 2、卡发板串口电路 ① USART1的电路连接 开发板中将USART1的TX(USART1_TXD, PA9)和RX(USART1_RXD, PA10)连接至CH340G (USB转串口芯片)的TXD和RXD,在TXD引脚串联一个二极管是为了防止CH340G给单片机供电而使单片机不能正常断电,从而导致程序下载失败。DTR连RSET可以实现程序下载完成后自动复位系统。下载程序时通过RTS将BOOT0拉低。 ② USART3的电路连接
[单片机]
STM32<font color='red'>串口</font>通信之Hello(STM32_11)
STC12C5A60S2双串口通信
STC12C5A60S2单片机是一款功能比较强大的单片机,它拥有两个全双工串行通信接口,串口1的功能及操作与传统51单片机串行口相同;特殊的是STC12C5A60S2单片机内部有一个独立波特率发生器,串口1可以使用定时器1作为波特率发生器,也可以使用独立波特率发生器作为波特率发生器;而串口2只能使用独立波特率发生器作为波特率发生器。 下面是一段双串口通信的程序: /*********************************************************************** 时间:2012.11.24 芯片:STC12C5A60S2 晶振:22.1184MHz 波特率:9600bps 引
[单片机]
STM32定时TIM2触发ADC采样,使用DMA保存结果
1.adc.h文件 //ADC-------------------------------------------------------------------------// #ifndef __EVAL_ADC_H #define __EVAL_ADC_H // Includes ------------------------------------------------------------------// #include stm32f10x.h #include eval.h // Exported types --------------------------------
[单片机]
基于串口透明传输的无线射频收发系统设计
目前市场上各类无线产品种类及应用越来越广泛,如何让广大学生、电子研发人员及一些小公司能快速在自己的产品中使用无线技术、降低无线技术的门槛,是本设计的初衷。对一些没有接触无线技术的人,从学到最后的设计应用需要比较长的时间。本文设计了一种基于无线收发芯片Si4432和单片机C8051F340的无线射频收发系统。基于串口透明传输,用户只需要了解一些串口指令即可方便实现无线收发。多种标准接口方便用户把模块嵌入到自己的产品中去。模块经过大量的试验、改进,能实现较远距离的稳定传输。 1、系统总体方案 无线射频收发系统结构框图如图1所示,由单片机C8051F340控制Si4432实现无线数据的收发。 发送模块中的C8051F340将数据
[单片机]
基于<font color='red'>串口</font>透明传输的无线射频<font color='red'>收发</font>系统设计
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved