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

发布者:EtherealBeauty最新更新时间:2024-12-04 来源: elecfans关键字:GD32F303  串口收发 手机看文章 扫描二维码
随时随地手机看文章

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)

wKgZomZCzTOAdz_OAAAuTeg2cA0146.png?imageView2/2/w/1000

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

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

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

状态寄存器0(USART_STAT0 )

wKgaomZCzUOAbIljAAA9A6YgC4w762.png?imageView2/2/w/1000

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

  1. TBE(发送空):这个位置“1”表示现在可以往数据寄存器中写数据了,当移位寄存器空闲时,写入到数据寄存器中的数据则会转移到移位寄存器中,串口开始对外发送数据;

  2. TC(发送完成):发送数据时,当数据寄存器和移位寄存器都为空时,表示所有的数据都已经完成了,则TC置“1”,所以当连续发数据时,最后一个字节从移位寄存器中发送完,TC才会置起。

  3. RBNE(接受非空):当串口接受到一个字节数据,RBNE置“1”,此时CPU可以去数据寄存器中取数据,当使用了DMA接受,DMA自动将数据寄存器中数据搬走,当数据寄存器数据被读走/搬走,RBNE位自动清“0”;

  4. 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.3main函数实现

以下为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.rn');

//配置UART接受,最长100byte
driver_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);
}
//callback
if(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会将这帧数据回发到串口调试助手中。

wKgZomZs8_GAHWEFAAAEB2USNVY682.png?imageView2/2/w/1000wKgZomZs8_aAZSKSAAAIwyq5TP0056.png?imageView2/2/w/1000

关键字:GD32F303  串口收发 引用地址:【GD32F303红枫派开发板使用手册】第十六讲 USART-DMA串口收发实验

上一篇:GD32330C-START开发板试用体验:软件配置及程序烧写调试
下一篇:【GD32F303红枫派开发板使用手册】第二讲 GPIO-流水灯实验

推荐阅读最新更新时间:2026-03-25 09:23

GD32F303红枫开发板使用手册】第十七讲 USART-中断串口收发实验
17.1实验内容 通过本实验主要学习以下内容: 使用中断进行串口收发 17.2实验原理 前面章节中我们已经学习了串口的状态标志位,本实验就是使用TBE中断和RBNE中断来实现中断收发数据,实验原理是RBNE中断用来接受数据,IDLE中断用于判断发送方数据结束,TBE中断用于发送数据。 17.3硬件设计 本实验仍然使用USB转UART接口,硬件设计见前面章节。 17.4代码解析 17.4.1串口中断发送函数 在driver_uart.c中定义了串口中断发送函数: C Drv_Err driver_uart_int_transmit(typdef_uart_struct *uartx,uint8_t *pbuff,uint16_t
[单片机]
【<font color='red'>GD32F303</font><font color='red'>红枫</font><font color='red'>派</font><font color='red'>开发板</font>使用手册】第十七讲 USART-中断<font color='red'>串口</font><font color='red'>收发</font>实验
GD32F303红枫开发板使用手册】第二十四讲 DHT11温湿度传感器检测实验
24.1实验内容 通过本实验主要学习以下内容: DHT11操作原理 单总线GPIO模拟操作原理 24.2实验原理 HT11是一款已校准数字信号输出的温湿度一体化数字传感器。该产品具有品质卓越、超快响应、抗干扰能力强、性价比极高等优点信号,传输距离可达20米以上。 其具体参数如下: 工作电压:3.3V-5.5V 工作电流:0.5mA 控制方式:单总线 输出方式:数字量 湿度精度:±5% 温度精度:±2℃ 湿度量程:5%~95% 温度量程:-20℃~+60℃ DHT11引脚定义和封装如下图所示 DHT11采用单总线的方式进行数据传输,下面对其通信时序以及传输数据构成进行介绍。 DHT11通信时序可分为:建立
[单片机]
【<font color='red'>GD32F303</font><font color='red'>红枫</font><font color='red'>派</font><font color='red'>开发板</font>使用手册】第二十四讲 DHT11温湿度传感器检测实验
GD32F303红枫开发板使用手册】第十一讲 ADC-电源电压单通道ADC检测实验
通过本实验主要学习以下内容: ADC的简介 GD32F303 ADC工作原理 查询方式实现ADC单通道采样 11.2实验原理 11.2.1ADC原理 我们知道,自然界中有非常多的模拟信号,比如上一节提到的光照强度,还有其他的例如温度、声音等等,那么人们是怎么来衡量一个模拟信号的呢? 我们通常会说今天光照度达到了3万Lux(照度单位),现在测量到的体温是36.5℃,我们所处的环境是40分贝,没错,人们就是通过将这些模拟信号数字化,从而达到衡量这些模拟信号的目的。那对于MCU来说,如果要测量一个模拟量,可以通过自带的ADC(Analog-to-Digital converters)模块,即模-数转换器将模拟量转化为可以被M
[单片机]
【<font color='red'>GD32F303</font><font color='red'>红枫</font><font color='red'>派</font><font color='red'>开发板</font>使用手册】第十一讲 ADC-电源电压单通道ADC检测实验
GD32F303红枫开发板使用手册】第二十一讲 I2C-EEPROM读写实验
21.1实验内容 通过本实验主要学习以下内容: AT24C16 EEPROM的工作原理; IIC模块原理以及IIC驱动原理。 21.2实验原理 21.2.1AT24C16 EEPROM的工作原理 下图为AT24CXX系列EEPROM相关参数,由该图可知,AT24C16的存储容量为16Kbit,共2048字节,共128页,每页为16字节。 由下图可知,AT24C16由8块组成,每块256字节。 I2C开始信号后,第一个字节为器件地址,由1010+3位块地址+1位读写标志组成,3位块地址刚好可以表示8个块, 所以一次写完256字节,换到下一下块的时候,要重新更改器件地址。 AT24C16支持页写入模式,一次最多可支持
[单片机]
【<font color='red'>GD32F303</font><font color='red'>红枫</font><font color='red'>派</font><font color='red'>开发板</font>使用手册】第二十一讲 I2C-EEPROM读写实验
GD32F303红枫开发板使用手册】第十五讲 USART-printf打印实验
15.1实验内容 通过本实验主要学习以下内容: 串口简介 GD32F303串口工作原理 使用printf打印信息 15.2实验原理 15.2.1串口简介 串口,从广义上看,指所有串行通信接口,比如RS232、RS422、RS485、SPI、IIC等。串行通讯是指仅用一根接收线和一根发送线就能将数据以位进行传输的通讯方式。和串行通讯相对应的是并行通讯,并行通信指一个传输接口可以传输8个bit即一个byte(有时甚至更多),虽然串行通信比并行通信慢,但是串口可以在仅仅使用两根线的情况下就能实现数据的传输。 对于GD32F303来说,串口一般特指USART(通用同步异步收发器 )和UART(通用异步收发器 )。USART/U
[单片机]
【<font color='red'>GD32F303</font><font color='red'>红枫</font><font color='red'>派</font><font color='red'>开发板</font>使用手册】第十五讲 USART-printf打印实验
GD32F303红枫开发板使用手册】第二十五讲 EXMC-外部SRAM读写实验
25.1实验内容 通过本实验主要学习以下内容: EXMC外设原理和配置 EXMC NOR/SRAM模式介绍 外部SRAM接口时序和操作方式 25.2实验原理 MCU的片内SRAM空间有限,在做一些大量数据处理、GUI显示等应用中片内SRAM容量无法满足应用需求,而外部SRAM器件读写速度快,不需要自刷新,工作稳定,是性能最优的外扩RAM选择之一。MCU通过EXMC接口可以实现外部SRAM的接口通信协议,同时可映射到内部地址实现和内部ram相同的操作方式。 25.2.1EXMC外设原理 EXMC是MCU的外部存储控制器,可以配置实现各类片外设备的通信协议,包括SRAM、PSRAM、NOR FLASH、NAND FLASH等,
[单片机]
【<font color='red'>GD32F303</font><font color='red'>红枫</font><font color='red'>派</font><font color='red'>开发板</font>使用手册】第二十五讲 EXMC-外部SRAM读写实验
GD32F303红枫开发板使用手册】第七讲 TIMER-蜂鸣器PWM
7.1实验内容 通过本实验主要学习以下内容: PWM输出功能实现; 定时器基本原理; 蜂鸣器驱动原理; 7.2实验原理 7.2.1蜂鸣器驱动原理 蜂鸣器是一种一体化结构的电子讯响器,采用直流电压供电,广泛应用于计算机、打印机、复印机、报警器、电子玩具、汽车电子设备、电话机、定时器等电子产品中作发声器件。比如台式电脑的主机开机会 滴 一声、洗衣机按下按键及洗衣完成都会有声响,以上这些声音都是通过蜂鸣器来发出的。蜂鸣器的驱动方式可分为:有源蜂鸣器(内有驱动线路)和无源蜂鸣器(使用外部驱动)。这里的“源”不是指电源。而是指震荡源。 也就是说,有源蜂鸣器内部带震荡源,所以只要一通电就会叫。而无源内部不带震荡源,所以如果用直流信号无
[单片机]
【<font color='red'>GD32F303</font><font color='red'>红枫</font><font color='red'>派</font><font color='red'>开发板</font>使用手册】第七讲 TIMER-蜂鸣器PWM
GD32F303红枫开发板使用手册】第六讲 PMU-低功耗实验讲
6.1实验内容 通过本实验主要学习以下内容: PMU原理; 低功耗的进入以及退出操作; 6.2实验原理 6.2.1PMU结构原理 PMU即电源管理单元,其内部结构下图所示,由该图可知,GD32F303系列MCU具有三个电源域,包括VDD/VDDA电源域、1.2V电源域以及电池备份域,其中,VDD /VDDA域由电源直接供电。在VDD/VDDA域中嵌入了一个LDO,用来为1.2V域供电。在备份域中有一个电源切换器,当VDD/VDDA电源关闭时,电源切换器可以将备份域的电源切换到VBAT引脚,此时备份域由VBAT引脚(电池)供电。 VDD/VDDA电源域 VDD 域为数字电源域包括HXTAL(高速外部晶体振荡器)、LDO
[单片机]
【<font color='red'>GD32F303</font><font color='red'>红枫</font><font color='red'>派</font><font color='red'>开发板</font>使用手册】第六讲 PMU-低功耗实验讲
小广播
最新单片机文章
何立民专栏 单片机及嵌入式宝典

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

厂商技术中心

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

 
机器人开发圈

电子工程世界版权所有 京ICP证060456号 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2026 EEWORLD.com.cn, Inc. All rights reserved