STM32F407的串口采用DMA收发数据

??

STM32F407的串口采用DMA收发数据

本文博客链接:http://blog.csdn.net/jdh99,作者:jdh,转载请注明.


环境:

主机: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(DMA_Channel_4);
	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(DMA2_Channel3);
	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_Cmd(DMA1_Stream5,DISABLE);

		//获得接收帧帧长
		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发送中断处理函数
**********************************************************************/

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);
}
时间: 2024-10-29 10:26:53

STM32F407的串口采用DMA收发数据的相关文章

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

STM32的串口采用DMA方式接收数据测试 本文博客链接:http://blog.csdn.net/jdh99,作者:jdh,转载请注明.   参考链接:http://www.amobbs.com/forum.PHP?mod=viewthread&tid=5511863&highlight=dma%E6%8E%A5%E6%94%B6   环境: 主机:WINXP 开发环境:MDK4.23 MCU:STM32F103CBT6 说明: 串口可以配置成用DMA的方式接收数据,不过DMA需要定长才

android4.2串口jni收发数据(基于自定义协议)

代码已经验证过,没问题 ! 代码层次结构: |-----serial_communication_class--- |                     |-------src--------- |                                  |------com------ |                                             |----object----- |                                

串口收发数据时字符、十六进制、二进制格式详细区分

在使用串口调试助手时发送和接收数据都是以字节 (Byte) 为单位,并且可以选择字符.十六进制.二进制三种收发格式 ,那么这三种格式究竟怎样区分呢? 首先我们来明确一个概念 :串口收发数据的单位 '' 字节 (Byte) '' ,   1Byte = 8 bits , 串口收发数据格式一般为  1bit起始位(一般为0) + 8bits 数据位(一字节) +1bit校验位(可有可无) +1bit结束位(一般为1) 下面逐一区分三种收发格式: (1) 十六进制:   由于1位十六进制数位宽为 4b

Arduino通过串口监视器收发数据

在串口监视器中发送数据,板子收到数据并打印出来. 不需要额外电路,但是板子必须连接电脑,Arduino IDE的串口监视器也需要被打开. 代码 /* 串口事件 当新的串口数据到来时,我们会将它添加到一个缓存字符串中.当收到换行符时就将缓存字符串 输出到串口监视器并将字符串清空. 本例程的最好测试方式是使用不断发送NMEA 0183语句的GPS接收器模块 代码公开. */ String inputString = ""; // 缓存字符串 boolean stringComplete =

STM32的PWM输入模式设置并用DMA接收数据

STM32的PWM输入模式设置并用DMA接收数据 项目中需要进行红外学习,如果采用输入捕获的方式,因为定时器只能捕获上升沿或者下降沿, 所以只能获得周期,而不能得到具体的红外波的高低电平的时间. 所以采用PWM输入的方式进行捕获. 采用的是PA8脚,对应TIM1的通道1. /********************************************************************* * 函数 *************************************

iOS开发之音频口通信-通过方波来收发数据

之前做过的项目有需要通过音频口通信用方波来收发数据,由于这方面的资料比较少,下面就介绍下其原理,希望能给大家帮助. 一. 音频通信简介大家应该都知道支付宝声波支付和拉卡拉吧,它们都是利用手机的音频口(手机耳机口)来实现全双工的通信(手机与设备之间的双向通信).其优点是低成本,编码芯片成本低,手机的 3.5mm 通信接口广泛. 二. 市场应用支付宝声波支付手机刷卡器皮肤检测仪检测如甲醛.气压.温度.湿度等等心率.血压等等....................... 三. 通信原理手机上用的耳机大

收发数据的原理(下)

因为网络原理不是三言两语可以讲完,如果读者很忙,可以直接拉到最底下,看总结,知道个大概,再回头细读此文章.感谢关注.废话不多说,直接进入主题.在上篇我们已经讲了TCP收发数据的前两步,接下来是最后两步. 将HTTP消息传给协议栈 上篇讲到控制流程从 connect 回到应用程序之后,就到了数据收发阶段. 数据收发数据是从应用程序调用write将要发送的数据交给协议栈开始的,协议栈收到数据后执行发送操作,这一操作包含如下要点. 首先,协议栈并不关心应用程序传来的数据是什么内容.应用程序调用writ

dsp28377控制DM9000收发数据

首先感谢上一篇转载文章的作者给出的参考,下面是一些自己在调试过程中的一些步骤: 首先把代码贴上来: //---------------------------------------------------------------------------------------------//DSP28377 利用EMIF控制网口DM9000芯片收发数据//----------------------------------------------------------------------

openVswitch(OVS)源代码分析之工作流程(收发数据包)

前面已经把分析openVswitch源代码的基础(openVswitch(OVS)源代码分析之数据结构)写得非常清楚了,虽然访问的人比较少,也因此让我看到了一个现象:第一篇,openVswitch(OVS)源代码分析之简介其实就是介绍了下有关于云计算现状和openVswitch的各个组成模块,还有笼统的介绍了下其工作流程,个人感觉对于学习openVswitch源代码来说没有多大含金量.云计算现状是根据公司发展得到的个人体会,对学习openVswitch源代码其实没什么帮助:openVswitch