【GD32F303红枫派使用手册】第十六节 USART-DMA串口收发实验

本文主要是介绍【GD32F303红枫派使用手册】第十六节 USART-DMA串口收发实验,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

16.1 实验内容

通过本实验主要学习以下内容:

  • 串口DMA工作原理
  • 使用DMA进行串口收发

16.2 实验原理

16.2.1 串口DMA工作原理

在前面ADC章节中,我们介绍了DMA的工作原理,这里就不多做介绍。从GD32F303用户手册中可以查到,各串口的TX和RX分别对应DMA的不同通道,比如USART0的TX对应DMA0的通道3,而RX对应DMA0的通道4。

当需要使用DMA发送时,需要配置DMA工作为内存到外设的模式,DMA目标地址需要设置为串口的数据寄存器,当DMA使能后,一旦串口的TBE(发送空)标志位为1,则DMA自动从内存中搬运数据到串口数据寄存器中。

当需要使用DMA接受时,需要配置DMA工作为外设到内存的模式,DMA的源地址需要设置为串口的数据寄存器,当DMA使能,一旦串口收到一个字节数据,RBNE(接受非空)标志位为1,则DMA自动将数据寄存器中的数据搬运到内存中。

16.2.2 串口寄存器介绍

串口有几个非常重要的寄存器需要读者理解,这里单独用一个章节来介绍。

数据寄存器(USART_DATA)

该寄存器虽然只有一个,但内部是映射为发送和接受两个寄存器。

发送时,除了发送数据寄存器,还有一个移位寄存器,当数据写入数据寄存器中,移位寄存器空闲的情况下,数据从数据寄存器中转移到移位寄存器,移位寄存器按照低bit——高bit的顺序将数据移位到IO口上。

接收时,接收到的数据保存在数据寄存器中,CPU或DMA可以从该寄存器中读接收到的数据。

状态寄存器0(USART_STAT0  )

我们需要特别理解TBE、TC、RBNE、IDLE、OREE这几位。

  1. TBE(发送空):这个位置“1”表示现在可以往数据寄存器中写数据了,当移位寄存器空闲时,写入到数据寄存器中的数据则会转移到移位寄存器中,串口开始对外发送数据;
  1. TC(发送完成):发送数据时,当数据寄存器和移位寄存器都为空时,表示所有的数据都已经完成了,则TC置“1”,所以当连续发数据时,最后一个字节从移位寄存器中发送完,TC才会置起。
  1. RBNE(接受非空):当串口接受到一个字节数据,RBNE置“1”,此时CPU可以去数据寄存器中取数据,当使用了DMA接受,DMA自动将数据寄存器中数据搬走,当数据寄存器数据被读走/搬走,RBNE位自动清“0”;
  1. IDLE(空闲):该标志位用于检测接受空闲,当串口接受最后一个字节后,再往后一个字节时间内,没有接受到新的数据,则该位置“1”;

IDLE一般用于串口DMA接受中,DMA接受中,MCU无法知道发送方的数据个数,所以可以通过判断IDLE位(或IDLE中断)来判断发送方一帧数据发送结束了。

  1. OREE(溢出错误):当RBNE置位的情况,又接收到一个字节数据,则OREE位置“1”。

16.3 硬件设计

本实验使用DMA进行串口发送和接收,仍然使用USB转UART接口,硬件设计见上一章。

16.4 代码解析

16.4.1 串口DMA发送函数

在driver_uart.c中定义了串口DMA发送函数driver_uart_dma_transmit:

C
Drv_Err driver_uart_dma_transmit(typdef_uart_struct *uartx,uint8_t *pbuff,uint16_t length)
{Drv_Err uart_state=DRV_ERROR;uint32_t timeout = driver_tick;    while(uartx->uart_control.Com_Flag.Bits.SendState==1){if((timeout+UART_TIMEOUT_MS) <= driver_tick) {              uartx->uart_control.Com_Flag.Bits.SendState=0;return DRV_ERROR;        } }      uartx->uart_control.Com_Flag.Bits.SendSucess=0;uartx->uart_control.Com_Flag.Bits.SendState=1;    uartx->uart_control.p_Send=pbuff;uartx->uart_control.SendSize=length;uartx->uart_control.SendCount=0;    uart_state=driver_dma_flag_wait_timeout(uartx->uart_tx_dma,DMA_FLAG_FTF,SET);     usart_dma_transmit_config(uartx->uart_x,USART_DENT_DISABLE);    driver_dma_start(uartx->uart_tx_dma,pbuff,length);    usart_flag_clear(uartx->uart_x,USART_FLAG_TC);    usart_dma_transmit_config(uartx->uart_x,USART_DENT_ENABLE);    usart_interrupt_enable(uartx->uart_x,USART_INT_TC);return uart_state;    
}

 16.4.2 串口DMA接收函数

在driver_uart.c中定义了串口DMA接收函数driver_uart_dma_receive:

C
Drv_Err driver_uart_dma_receive(typdef_uart_struct *uartx,uint8_t *pbuff,uint16_t length)
{Drv_Err uart_state=DRV_SUCCESS; uint32_t timeout = driver_tick; while(uartx->uart_control.Com_Flag.Bits.RecState==1){if((timeout+UART_TIMEOUT_MS) <= driver_tick) {              uartx->uart_control.Com_Flag.Bits.RecState=0;return DRV_ERROR;        } }    uartx->uart_control.Com_Flag.Bits.RecSuccess=0;uartx->uart_control.Com_Flag.Bits.RecState=1;uartx->uart_control.p_Rec=pbuff;uartx->uart_control.RecSize=length;uartx->uart_control.RecCount=0;       usart_dma_receive_config(uartx->uart_x,USART_DENR_DISABLE);    driver_dma_start(uartx->uart_rx_dma,pbuff,length);USART_STAT0(uartx->uart_x);usart_data_receive(uartx->uart_x);usart_interrupt_flag_clear(uartx->uart_x,USART_INT_FLAG_IDLE);    usart_interrupt_enable(uartx->uart_x,USART_INT_IDLE);        usart_dma_receive_config(uartx->uart_x,USART_DENR_ENABLE);  return uart_state;     
}

16.4.3 main函数实现

以下为main函数代码:

C
int main(void)
{delay_init(); //初始化UART为DMA模式,注册接受完成(IDLE)回调函数BOARD_UART.uart_mode_tx=MODE_DMA;BOARD_UART.uart_mode_rx=MODE_DMA;BOARD_UART.uart_idle_callback=user_receive_complete_callback;       bsp_uart_init(&BOARD_UART);nvic_irq_enable(USART0_IRQn,2,0); delay_ms(1000);printf("uart dma mode sends and receives loopback packets of indefinite length.\r\n"); //配置UART接受,最长100bytedriver_uart_dma_receive(&BOARD_UART,uart_rec_buff,100);while (1){//查询到接受完成回调函数标志if(uart_receive_complete_flag==SET){uart_receive_complete_flag=RESET;//发送刚接受到的数据driver_uart_dma_transmit(&BOARD_UART,uart_send_buff,uart_receive_count);             }}
}

 本例程main函数首先进行了延时函数初始化,再初始化UART为DMA模式,接着配置串口BOARD_UART,开启串口中断NVIC,这里使用到了IDLE中断,用来接受不定长数据,然后配置串口DMA接受,最长100个字节,所以我们可以给串口发送100个字节以下长度的数据。在while(1)循环中循环查询uart_receive_complete_flag标志位,当该标志位为“SET”时,表示IDLE中断被触发,一帧数据接受完,最后将接收到的帧数据通过DMA发送方式原封不动发送到串口上。

16.4.4 中断函数

在bsp_uart.c中定义了串口中断处理函数

C
void USART0_IRQHandler(void)
{driver_uart_int_handler(&BOARD_UART);
}

在driver_uart.c中定义了driver_uart_int_handler函数:

C
Drv_Err driver_uart_int_handler(typdef_uart_struct *uartx)
{   Drv_Err uart_state=DRV_SUCCESS;    if(usart_interrupt_flag_get(uartx->uart_x,USART_INT_FLAG_RBNE)!=RESET){if(uartx->uart_control.RecCount < uartx->uart_control.RecSize){uartx->uart_control.p_Rec[uartx->uart_control.RecCount]=usart_data_receive(uartx->uart_x); uartx->uart_control.RecCount++;            }else{usart_data_receive(uartx->uart_x);uart_state=DRV_ERROR;//err 溢出}if(uartx->uart_rbne_callback!=NULL){uartx->uart_rbne_callback(uartx);}        //callbackif(uartx->uart_control.RecCount == uartx->uart_control.RecSize){uartx->uart_control.Com_Flag.Bits.RecSuccess=1;            uartx->uart_control.Com_Flag.Bits.RecState=0; uartx->uart_control.RecCount=0;            }        }       if(usart_interrupt_flag_get(uartx->uart_x,USART_INT_FLAG_IDLE)!=RESET){usart_interrupt_flag_clear(uartx->uart_x,USART_INT_FLAG_IDLE);USART_STAT0(uartx->uart_x);USART_DATA(uartx->uart_x);if( (uartx->uart_mode_rx==MODE_INT && uartx->uart_control.RecCount>0) \||(uartx->uart_mode_rx==MODE_DMA && dma_transfer_number_get(uartx->uart_rx_dma->dmax,uartx->uart_rx_dma->dma_chx)!=uartx->uart_control.RecSize)){uartx->uart_control.Com_Flag.Bits.RecSuccess=1;uartx->uart_control.Com_Flag.Bits.RecState=0;              if(uartx->uart_mode_rx==MODE_DMA){uartx->uart_control.RecCount=uartx->uart_control.RecSize-dma_transfer_number_get(uartx->uart_rx_dma->dmax,uartx->uart_rx_dma->dma_chx);            }//callback            if(uartx->uart_idle_callback!=NULL){uartx->uart_idle_callback(uartx);}                                }        } if(usart_interrupt_flag_get(uartx->uart_x,USART_INT_FLAG_TBE)!=RESET){usart_data_transmit(uartx->uart_x,uartx->uart_control.p_Send[uartx->uart_control.SendCount]);uartx->uart_control.SendCount++;if(uartx->uart_tbe_callback!=NULL){uartx->uart_tbe_callback(uartx);} if(uartx->uart_control.SendCount >= uartx->uart_control.SendSize){uartx->uart_control.SendCount=0;            usart_interrupt_disable(uartx->uart_x, USART_INT_TBE);usart_interrupt_enable(uartx->uart_x, USART_INT_TC);}} if(usart_interrupt_flag_get(uartx->uart_x,USART_INT_FLAG_TC)!=RESET){usart_interrupt_disable(uartx->uart_x, USART_INT_TC);          usart_flag_clear(uartx->uart_x,USART_FLAG_TC);  if( !(uartx->uart_mode_rx==MODE_DMA && dma_transfer_number_get(uartx->uart_tx_dma->dmax,uartx->uart_tx_dma->dma_chx)!=0) ){    uartx->uart_control.Com_Flag.Bits.SendSucess=1;            uartx->uart_control.Com_Flag.Bits.SendState=0; if(uartx->uart_tc_callback!=NULL){uartx->uart_tc_callback(uartx);}         uartx->uart_control.SendCount=0; }}  if(usart_flag_get(uartx->uart_x,USART_FLAG_ORERR)==SET){usart_flag_clear(uartx->uart_x,USART_FLAG_ORERR);        USART_STAT0(uartx->uart_x);USART_DATA(uartx->uart_x);uart_state=DRV_ERROR;}return uart_state;     }

16.5 实验结果

使用USB-TypeC线,连接电脑和板上USB to UART口后,使用串口调试助手发送一帧数据到MCU,MCU会将这帧数据回发到串口调试助手中。

 

由聚沃科技原创,来源于【红枫派开发板】第十六讲 USART-DMA串口收发实验 - 苏州聚沃电子科技有限公司 (gd32bbs.com) 

这篇关于【GD32F303红枫派使用手册】第十六节 USART-DMA串口收发实验的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1063525

相关文章

高性能并行计算华为云实验五:

目录 一、实验目的 二、实验说明 三、实验过程 3.1 创建PageRank源码 3.2 makefile的创建和编译 3.3 主机配置文件建立与运行监测 四、实验结果与分析 4.1 采用默认的节点数量及迭代次数进行测试 4.2 分析并行化下节点数量与耗时的变化规律 4.3 分析迭代次数与耗时的变化规律 五、实验思考与总结 5.1 实验思考 5.2 实验总结 E

407串口01发送

实验一: 工程。 链接:https://pan.baidu.com/s/1g8DV4yZWOix0BbcZ08LYDQ?pwd=2176 提取码:2176 串口1的使用。发送功能。 单片机发送信息到电脑。 通过串口进行通信。 首先单片机这边。 单片机这边,需要对单片机的串口模块进行使能初始化,设置串口的格式。 单片机和电脑的串口收发格式要配置一致。不然A和B肯定通信不成功,鸡和鸭讲,

物联网系统运维——移动电商应用发布,Tomcat应用服务器,实验CentOS 7安装JDK与Tomcat,配置Tomcat Web管理界面

一.Tomcat应用服务器 1.Tomcat介绍 Tomcat是- -个免费的开源的Ser Ivet容器,它是Apache基金会的Jakarta 项目中的一个核心项目,由Apache, Sun和其他一 些公司及个人共同开发而成。Tomcat是一一个小型的轻量级应用服务器,在中小型系统和并发访问用户不是很多的场合下被普遍使用,是开发和调试JSP程序的首选。 在Tomcat中,应用程序的成部署很简

STM32HAL库--SDRAM实验(速记版)

STM32F429IGT6 自带了 256K 字节的 SRAM,对一般应用来说,已经足够了,不过在一些对内存要求高的场合, STM32F429 自带的这些内存就不够用了。比如使用 LTDC 驱动RGB 屏、跑算法或者跑 GUI 等,就可能不太够用,所以阿波罗 STM32F429 开发板板载了一颗 32M 字节容量的 SDRAM 芯片:W9825G6KH,满足大内存使用的需求。 1

xilinx FPGA 串口设计笔记

xilinx FPGA  串口设计笔记 在设计中,需要用 FPGA 读取 GPS 内部的信息,GPS 的通信方式为串口,所以在 FPGA 中移植了串口程序。 本次移植的程序源代码是特权的串口程序,本以为移植应该很快就能完成, 但其中还是 出了一写小问题,耽误了不少的时间,下面将问题进行一个总结! 以下是串口的时序: 在设计中,耽误时间最长的问题就是数据老是出错,为了找出问题

HCIA 19 结束 企业总部-分支综合实验(下)

3.6出口NAT配置可以访问互联网 配置NAT使内网可以访问公网8.8.8.8,当前总部PC1 PING不通公网地址8.8.8.8。 3.6.1总部配置NAT访问互联网 步骤1:配置NAT acl number 2000    rule 5 permit source 192.168.0.0 0.0.255.255 # interface GigabitEthernet0/0/2

TX2 关闭console serial使用该串口

由于项目变更,GPU型号从Tx1变成Tx2,原来的硬件接口没有变,但是把原来UART0的console serial功能关闭变成一个串口使用的方法没法使用了。这就导致了数据通过该串口发送的数据变的很有问题,无法使用了。 经过测试,终于找到了解决的方法。 首先在虚拟机上,安装编辑设备树的软件: $sudo apt-get update $sudo apt-get install device

JetSon Tx1 串口使用记录

最近在学习使用Jetson Tx1,下面是使用串口遇到的问题,做为我的第一篇博客。 Jetson Tx1串口是TTL电平。 Jetson Tx1的UART0在Linux上的设备号是“/dev/tty0”,但是在Tx1刷的系统中,UART0默认是做为console serail设备使用的,这个具体有什么作用,我也不大清楚了,百度一下也不是很懂。这就导致了在Linux上使用串口调试工具和PC机相连

STM32G030F6使用CubeMx配置PWM实验

1. 使用 CubeMx 创建 PWM 工程 打开 CubeMx 软件,选中我们此次使用的单片机型号 STM32G030F6P6 ,点击 StartProject. 配置定时器 配置定时器1的通道1和通道2 产生PWM; 设置定时器1的主频:设置了( 63 + 1) 分频即定时器主频为1M 设置PWM定时的周期计数为 1000 即 1000HZ 设置通道一 翻转的计数值为 500 即

STM32G030F6使用CubeMx配置DMA读取多通道ADC实验

1. 使用 CubeMx 创建 ADC 工程 打开 CubeMx 软件,选中我们此次使用的单片机型号 STM32G030F6P6 ,点击 StartProject. 先配置一下串口,用来打印相关信息 再来配置 ADC 配置DMA PS:DMA 需要要配置成循环模式,否则只填满一次缓存数组后就停止工作,需要重调用启动 DMA 的函数. 配置时钟 ps:本实验使用内部高速时钟