STM32F407ZET6 USART DMA方式收发数据

发布者:chinapxf最新更新时间:2018-06-08 来源: eefocus关键字:STM32F407ZET6  USART  DMA方式  收发数据 手机看文章 扫描二维码
随时随地手机看文章

串口采用DMA方式收发数据有两种不同的方式,第一种方式:采用DMA传输完成中断进行发送和接收;第二种方式:采用串口总线空闲方式收发数据。这两种方式第二种方式更好一些,因为第二种方式可以收发不定长度的数据帧,然而第一种方式不能。但是第二种方式的逻辑复杂一些,收发过程之前都要判断总线是否是空闲。

在此,以USART2的DMA收发方式举例:

一、使用DMA传输完成中断收发

整体思路:上位机发送四个字节的数据,STM32接收完成后进入DMA中断中,发送下位机STM32定义好的数据给上位机并且清除DMA传输完成中断标志位,最后进入发送完成中断,关闭发送通道,清除DMA发送完成标志位。

在上述思路之前,要进行的自然是串口配置、DMA配置以及中断配置。此处的配置函数如下:

/****************************************** 

**函数名称:UpperUsart2Init 

**函数参数:baudRate 波特率 

**函数作用:初始化与上位机通讯的串口Usart2 

**硬件引脚:TX--PD5  RX--PD6  

******************************************/  

void UpperUsart2Init(int baudRate)  

/****************************************** 

**函数名称:UpperUsart2Init 

**函数参数:baudRate 波特率 

**函数作用:初始化与上位机通讯的串口Usart2 

**硬件引脚:TX--PD5  RX--PD6  

******************************************/  

void UpperUsart2Init(int baudRate)  

{  


//开启串口时钟、DMA时钟以及相应GPIO时钟  

    RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2, ENABLE);    

    RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA1, ENABLE);   

    RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOD, ENABLE);    

  

    GPIO_PinAFConfig(GPIOD,GPIO_PinSource5,GPIO_AF_USART2);  

    GPIO_PinAFConfig(GPIOD,GPIO_PinSource6,GPIO_AF_USART2);   

    //PD8(TX)设置成复用推挽输出  

    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5; //| GPIO_Pin_9;  

    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;  

    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;  

    GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;  

    GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;  

    GPIO_Init(GPIOD, &GPIO_InitStructure);  

    //PD9(RX)设置成浮空输入  

    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6;  

    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;  

    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;  

    GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;  

    GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;  

    GPIO_Init(GPIOD, &GPIO_InitStructure);    

      

    // USART2_DMA_RX DMA1_Stream5 DMA_Channel_4   

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

    DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralToMemory;  

    DMA_InitStructure.DMA_BufferSize = UPPERRBSIZE;  

    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_Circular;//如果是Normal只能接受一次,故采用循环模式  

     DMA_InitStructure.DMA_Priority = DMA_Priority_High;  

     DMA_InitStructure.DMA_FIFOMode = DMA_FIFOMode_Disable;           

     DMA_InitStructure.DMA_FIFOThreshold = DMA_FIFOThreshold_HalfFull;  

     DMA_InitStructure.DMA_MemoryBurst = DMA_MemoryBurst_Single;  

     DMA_InitStructure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single;  

    DMA_Init(DMA1_Stream5, &DMA_InitStructure);  

      

    // USART2_DMA_TX DMA1_Stream6 DMA_Channel_4   

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

    DMA_InitStructure.DMA_DIR = DMA_DIR_MemoryToPeripheral;  

    DMA_InitStructure.DMA_BufferSize = UPPERTBSIZE;  

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

    DMA_InitStructure.DMA_Priority = DMA_Priority_High;  

    DMA_InitStructure.DMA_FIFOMode = DMA_FIFOMode_Disable;           

    DMA_InitStructure.DMA_FIFOThreshold = DMA_FIFOThreshold_HalfFull;  

    DMA_InitStructure.DMA_MemoryBurst = DMA_MemoryBurst_Single;  

    DMA_InitStructure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single;  

    DMA_Init(DMA1_Stream6, &DMA_InitStructure);  

  

    //USART2设置 115200 8 1 0 NONE  

    USART_InitStructure.USART_BaudRate = baudRate;  

    USART_InitStructure.USART_WordLength = USART_WordLength_8b;  

    USART_InitStructure.USART_StopBits = USART_StopBits_1;  

    USART_InitStructure.USART_Parity = USART_Parity_No ;  

    USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;  

    USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;  

    USART_Init(USART2,&USART_InitStructure);  

      

    // Configure one bit for preemption priority   

    NVIC_PriorityGroupConfig(NVIC_PriorityGroup_0);  

    // Enable DMA1_Stream5 Interrupt   

    NVIC_InitStructure.NVIC_IRQChannel = DMA1_Stream5_IRQn;  

     NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;  

    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 4;  

    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;  

    NVIC_Init(&NVIC_InitStructure);  

    // Enable DMA1_Stream6 Interrupt   

    NVIC_InitStructure.NVIC_IRQChannel = DMA1_Stream6_IRQn;  

    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;  

    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;  

    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;  

    NVIC_Init(&NVIC_InitStructure);  

    // Enable USART2 Interrupt   

    NVIC_InitStructure.NVIC_IRQChannel = USART2_IRQn;  

    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;  

    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 5;  

    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;  

    NVIC_Init(&NVIC_InitStructure);    

      

    //开启串口、DMA和串口总线空闲中断  

    DMA_Cmd(DMA1_Stream5,ENABLE);  

    DMA_Cmd(DMA1_Stream6,DISABLE);  

    USART_DMACmd(USART2,USART_DMAReq_Tx,ENABLE);  

    USART_DMACmd(USART2,USART_DMAReq_Rx,ENABLE);    

    DMA_ITConfig(DMA1_Stream5, DMA_IT_TC, ENABLE);  

    DMA_ITConfig(DMA1_Stream6, DMA_IT_TC, ENABLE);  

    DMA_ClearITPendingBit(DMA1_Stream5, DMA_IT_TCIF5); //标志位设置为默认值  

    DMA_ClearITPendingBit(DMA1_Stream6, DMA_IT_TCIF6);        

    USART_ITConfig(USART2,USART_IT_IDLE,ENABLE);   

//  USART_ITConfig(USART2, USART_IT_RXNE, ENABLE);  

    USART_Cmd(USART2, ENABLE);    

}  

上述程序段功能是配置串口、DMA以及相应中断。(一定要注意开启时钟)

在本程序中STM32发送给上位机的数据为uint8_t data[4] = {0x01,0x03,0x04,0x06};


下面这个中断是DMA接收完成中断,完成功能是:清除接收完成标志位,并且向上位机发送预定义数据。


/****************************************** 

**函数名称:DMA1_Stream5_IRQHandler 

**函数参数:无 

**函数作用:串口2 DMA接受完成时发送数据给上位机 

******************************************/  

void DMA1_Stream5_IRQHandler(void)  

{  

if(SET ==  DMA_GetITStatus(DMA1_Stream5, DMA_IT_TCIF5))  

{   

    DMA_Cmd(DMA1_Stream5,DISABLE);  

    DMA_ClearFlag(DMA1_Stream5,DMA_FLAG_TCIF5);  

    DMA_Cmd(DMA1_Stream5,ENABLE);  

     

    DMA_SetCurrDataCounter(DMA1_Stream6,UPPERTBSIZE);   

    memcpy(upperTxBuffer,data,UPPERTBSIZE);        

    DMA_Cmd(DMA1_Stream6,ENABLE);      

}  

}  

 下面这个中断是DMA传输完成中断,完成功能是:清除发送完成标志位。


/****************************************** 

**函数名称:DMA1_Stream6_IRQHandler 

**函数参数:无 

**函数作用:串口2发送完成时中断入口函数,关闭传输通道并且清除标志 

******************************************/  

void DMA1_Stream6_IRQHandler(void) //UART2_TX  

{  

    if(SET == DMA_GetITStatus(DMA1_Stream6,DMA_IT_TCIF6))  

    {  

        DMA_Cmd(DMA1_Stream6,DISABLE);  

        DMA_ClearFlag(DMA1_Stream6, DMA_FLAG_TCIF6);  

    }  

}  


关键字:STM32F407ZET6  USART  DMA方式  收发数据 引用地址:STM32F407ZET6 USART DMA方式收发数据

上一篇:STM32 串口采用DMA方式接收数据
下一篇:STM32F429串口IDLE中断+DMA接收串口数据

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

tiny4412 UART 收发数据
Exynos4412 UART 的特性 Exynos4412 中UART,有4 个独立的通道,每个通道都可以工作于中断模式或DMA 模式,即 UART 可以发出中断或 DMA 请求以便在UART 、CPU 间传输数据。UART 由波特率发生器、发送器、接收器和控制逻辑组成。 使用系统时钟时,Exynos4412 的 UART 波特率可以达到 4Mbps 。波特率可以通过编程进行 。 Exynos4412 UART 的通道 0有 256 字节的发送 FIFO 和 256 字节的接收FIFO ;通道 1、4有 64 字节的发送 FIFO 和 64 字节的接收FIFO;通道 2、3有 16 字节的发送FIFO 和 16 字节 的
[单片机]
tiny4412 UART <font color='red'>收发</font><font color='red'>数据</font>
STM32F103--(三) USART实践
GPIO的后面很容易想到的应该就是通用同步/异步接受发送器(USART) 了。对于比较复杂点的程序而言,用led来调试显然是有点不太科学。所以,把USART口调试好后,有助于之后其它部分的调试。(把USART当成是调试输出口来用 ) 调试USART花了我一些时间,其实问题主要出现在一些很小的细节方面。比如发现发送的数据中夹杂这乱码,后来通过数据的二进制分析发现是奇偶校验位不小心打开了。如果排除这些小问题的话,整个工作应该是很容易的。 先上例程,然后标注,最后分析 USART的初始化 void UartInit(void) { USART_InitTypeDef USART_InitStructure;//声明一个USART初
[单片机]
第24章 STM32F429的USART应用之八个串口FIFO实现
24.1 初学者重要提示 学习本章节前,务必优先学习第23章。 串口FIFO的实现跟前面章节按键FIFO的机制是一样的。 本章节比较重要,因为后面的ESP8266,GPS,RS485,GPRS等试验都是建立在这个驱动的基础上实现。 大家自己做的板子,测试串口收发是乱码的话,重点看stm32h7xx_hal_conf.h文件中的HSE_VALUE的大小跟板子上实际晶振大小是否一致,然后再看PLL配置。 CH340/CH341的USB转串口Windows驱动程序的安装包,支持32/64位 Windows 10/8.1/8/7。http://www.armbbs.cn/forum.php?mod=viewthread&t
[单片机]
第24章 STM32F429的<font color='red'>USART</font>应用之八个串口FIFO实现
STM32f103的数电采集电路的USART的使用与蓝牙的连接的程序
STM32 的串口资源相当丰富的,功能也相当强劲。本项目所使用的 STM32F103C8T6 最多可提供 3 路串口,有分数波特率发生器、支持同步单线通信和半双工单线通讯、支持 LIN、 支持调制解调器操作、 智能卡协议和 IrDA SIR ENDEC 规范、具有 DMA等。 串口设置的一般步骤可以总结为如下几个步骤: 1.串口时钟使能,GPIO 时钟使能 2.串口复位 3.GPIO 端口模式设置 4.串口参数初始化 5.开启中断并且初始化 NVIC(如果需要开启中断才需要这个步骤) 6.使能串口 7.编写中断处理函数 HC-05 嵌入式蓝牙串口通讯模块(以下简称模块)具有两种工作模式:命令响应工作模式和自动连接工作模式,在自动连
[单片机]
STM32f103的数电采集电路的<font color='red'>USART</font>的使用与蓝牙的连接的程序
USART test---ATMEGA128
#include avr/io.h #define fosc 16000000UL //晶振16MHZ #define baud 9600//波特率 #define com0 void uart0_init(void) { UCSR0B = 0x00; //disable while setting baud rate UCSR0A = 0x00; UCSR0C =(1 UCSZ01)|(1 UCSZ00);//8bit+1bit stop UBRR0L=(fosc/16/(baud+1))%256; UBRR0H=(fosc/16/(baud+1))/256; UCSR0B =(1 RXE
[单片机]
stm32的USART_IT_TXE和USART_IT_TC
一般来说,串口的发送中断传输数据,对于我目前的应用来说,应用的要求并不是很高,因此,因此就一直没有对其进行一个比较好的实验与认识。然而,在一次串口程序升级(IAP)升级实验中,发现有人使用了这个发送中断方式进行的,所以特别的进行了一个个步骤的实验来进一步知道其运行机理。 首先,串口发送数据,可以使用的方式有: 1、 发送一个数据,然后读取USART_IT_TXE或者USART_IT_TC寄存器的状态。 2、 使用串口发送中断 3、 使用dma发送完成中断 在效率上,肯定是3比较好。其次是2。 网上关于USART_IT_TXE和USART_IT_TC是怎么个用法,各家有各家的言论,在我这里只通过实验来了解
[单片机]
STM32CubeMx(Keil5)开发之路——4采用DMA方式收发数据
运行环境 Windows10 STM32CubeMX___Version 5.0.0 Keil5(MDK5)___Version 5.15 简介 本例程主要讲解如何通过串口发送数据和重定向printf STM32CubeMx基本配置 基础配置过程请参考 STM32CubeMx(Keil5)开发之路—配置第一个项目 STM32CubeMx USART1配置 1——点击USART1进行设置 2——模式选择Asynchronous异步传输 3——可以看到右边自动出现了Tx和Rx 4——可以自行设置波特率,停止位,校验位等参数 1——选择DMA设置 2——添加DMA 3——添加DMA的Tx和Rx通道 1——选择NVIC设
[单片机]
STM32CubeMx(Keil5)开发之路——4采用<font color='red'>DMA</font><font color='red'>方式</font><font color='red'>收发</font><font color='red'>数据</font>
STM32的USART发送数据时如何使用TXE和TC标志
在USART的发送端有2个寄存器,一个是程序可以看到的USART_DR寄存器(下图中阴影部分的TDR),另一个是程序看不到的移位寄存器(下图中阴影部分Transmit Shift Register)。 对应USART数据发送有两个标志,一个是TXE=发送数据寄存器空,另一个是TC=发送结束;对照下图,当TDR中的数据传送到移位寄存器后,TXE被设置,此时移位寄存器开始向TX信号线按位传输数据,但因为TDR已经变空,程序可以把下一个要发送的字节(操作USART_DR)写入TDR中,而不必等到移位寄存器中所有位发送结束,所有位发送结束时(送出停止位后)硬件会设置TC标志。 另一方面,在刚刚初始化好USART还没有发送任何数据时,也会
[单片机]
STM32的<font color='red'>USART</font>发送<font color='red'>数据</font>时如何使用TXE和TC标志
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

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

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

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