0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

【GD32H757Z海棠派开发板使用手册】第十讲 USART-中断串口收发实验

聚沃科技 ? 2024-05-16 10:30 ? 次阅读
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

wKgZomYgeJOAUiXJAB6mQrDJGEg027.png

10.1实验内容

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

  • 使用中断进行串口收发

10.2实验原理

10.2.1串口寄存器介绍

串口有几个非常重要的寄存器需要读者理解。

数据接收寄存器(USART_RDATA)

wKgaomZFbTOAXPmzAAAwMAxUG-I889.png

数据发送寄存器(USART_RDATA)

wKgZomZFbUCAHOXZAAAvvuPzh50222.png

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

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

状态寄存器(USART_STAT )

wKgZomZFbUuAFDIOAABSvI18o7s614.png

我们需要特别理解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中断)来判断发送方一帧数据发送结束了。

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

以上就是串口寄存器的介绍。本实验就是使用TBE中断和RBNE中断来实现中断收发数据,实验原理是RBNE中断用来接受数据,IDLE中断用于判断发送方数据结束,TBE中断用于发送数据。

10.3硬件设计

本实验使用P1接口的PA9和PA10实现串口功能,硬件设计请见上一章。

10.4代码解析

10.4.1串口中断发送函数

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

C
Drv_Err driver_uart_int_transmit(typdef_uart_struct *uartx,uint8_t *pbuff,uint16_t length)
{
__IO uint64_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.SendSuccess=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;

usart_flag_clear(uartx->uart_x,USART_FLAG_TC);
usart_interrupt_enable(uartx->uart_x,USART_INT_TBE);

return DRV_SUCCESS;
}

10.4.2串口中断接受函数

在driver_uart.c中定义了串口中断接受函数:

C
Drv_Err driver_uart_int_receive(typdef_uart_struct *uartx,uint8_t *pbuff,uint16_t length)
{
__IO uint64_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;
}
}

if(usart_flag_get(uartx->uart_x,USART_FLAG_ORERR))
{
usart_flag_clear(uartx->uart_x,USART_FLAG_ORERR);
}

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_flag_clear(uartx->uart_x,USART_FLAG_IDLE);

usart_interrupt_enable(uartx->uart_x,USART_INT_RBNE);
usart_interrupt_enable(uartx->uart_x,USART_INT_IDLE);

return DRV_SUCCESS;
}

10.4.3main函数实现

以下为main函数代码:

C
int main(void)
{
//延时、共用驱动部分初始化
driver_init();
//初始化UART为中断模式,注册接受完成(IDLE)回调函数
BOARD_UART.uart_mode_tx=MODE_INT;
BOARD_UART.uart_mode_rx=MODE_INT;
BOARD_UART.uart_idle_callback=user_receive_complete_callback;
bsp_uart_init(&BOARD_UART);

bsp_led_init(&LED2);
bsp_led_init(&LED1);
bsp_led_on(&LED2);
bsp_led_off(&LED1);
//使能UART中断
nvic_irq_enable(USART0_IRQn,2,0);
delay_ms(100);
printf_log("uart interrupt mode sends and receives loopback packets of indefinite length.\r\n");
//启动UART中断接受,最长100byte
driver_uart_int_receive(&BOARD_UART,uart_rec_buff,100);

while (1)
{
//查询到接受完成回调函数标志
if(uart_receive_complete_flag==SET)
{
uart_receive_complete_flag=RESET;
//启动中断方式发送刚接受到的数据
driver_uart_int_transmit(&BOARD_UART,uart_send_buff,uart_receive_count);
printf_log("\r\n The received data is %s\r\n",uart_send_buff);
memset(uart_send_buff,0,100);
}
}
}

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

10.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);
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.SendSuccess=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);
uart_state=DRV_ERROR;
}

return uart_state;

}

10.5实验结果

使用串口调试助手发送一帧数据到MCU,MCU会将这帧数据回发到串口调试助手中。

wKgZomZCzhqAd1-uAAAEQNN_QRI502.pngwKgZomZCziKAWcJsAAAIXPeWoWw536.png

教程GD32 MCU方案商聚沃科技原创发布,了解更多GD32 MCU教程,关注聚沃科技官网

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 单片机
    +关注

    关注

    6068

    文章

    45032

    浏览量

    652192
  • 串口
    +关注

    关注

    15

    文章

    1591

    浏览量

    80240
  • 开发板
    +关注

    关注

    25

    文章

    5725

    浏览量

    105630
  • USART
    +关注

    关注

    1

    文章

    201

    浏览量

    32069
  • GD32
    +关注

    关注

    7

    文章

    422

    浏览量

    25650
收藏 人收藏
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

    评论

    相关推荐
    热点推荐

    基于小凌RK2206开发板:OpenHarmony如何使用IoT接口控制GPIO中断

    1、实验简介本实验将演示如何在小凌-RK2206开发板上使用IOT库的GPIO中断模式,进行GPIO编程
    的头像 发表于 04-21 10:39 ?497次阅读
    基于小凌<b class='flag-5'>派</b>RK2206<b class='flag-5'>开发板</b>:OpenHarmony如何使用IoT接口控制GPIO<b class='flag-5'>中断</b>

    GD32H737/757/759 MCU规格书

    电子发烧友网站提供《GD32H737/757/759 MCU规格书.pdf》资料免费下载
    发表于 02-07 14:45 ?107次下载
    <b class='flag-5'>GD32H</b>737/<b class='flag-5'>757</b>/759 MCU规格书

    GD32H757xx数据表

    电子发烧友网站提供《GD32H757xx数据表.pdf》资料免费下载
    发表于 01-17 15:30 ?0次下载
    <b class='flag-5'>GD32H757</b>xx数据表

    GD32H737/757/759用户手册

    电子发烧友网站提供《GD32H737/757/759用户手册.pdf》资料免费下载
    发表于 01-16 14:44 ?36次下载
    <b class='flag-5'>GD32H</b>737/<b class='flag-5'>757</b>/759用户<b class='flag-5'>手册</b>

    ADS8422使用手册单端转差分电路读出的值有几mv的波动,为什么?

    ADS8422: 使用手册单端转差分电路读出的值有几mv的波动,用万用表测差分输出是很稳的,附上电路图,求大神指点,感谢
    发表于 11-29 07:51

    野火GD32H759开发板产品概述

    野火GD32H759开发板基于中国首款Arm Cortex-M7内核的超高性能微控制器——兆易创新GD32H759IMK6芯片进行设计,主频高达600MHz,采用BGA176封装,配备3840KB的片上FLASH,以及1024K
    的头像 发表于 11-13 17:31 ?4334次阅读
    野火<b class='flag-5'>GD32H</b>759<b class='flag-5'>开发板</b>产品概述

    ARM开发板与树莓的比较

    ARM开发板和树莓都是基于ARM架构的单板计算机,它们在教育、工业控制、物联网等领域有着广泛的应用。 硬件配置 ARM开发板和树莓的硬件配置各有特点,具体取决于不同的型号和制造商。
    的头像 发表于 11-05 11:11 ?1458次阅读

    迅为itop-3568开发板AMP双系统使用手册之烧写AMP镜像

    迅为itop-3568开发板AMP双系统使用手册之烧写AMP镜像
    的头像 发表于 11-04 15:00 ?1328次阅读
    迅为itop-3568<b class='flag-5'>开发板</b>AMP双系统<b class='flag-5'>使用手册</b>之烧写AMP镜像

    迅为瑞芯微RK3568开发板/核心《iTOP-3568开发板实时系统使用手册

    iTOP-RK3568开发板使用手册上新,后续资料会不断更新,不断完善,帮助用户快速入门,大大提升研发速度。 iTOP-RK3568开发板支持了Preemption和Xenomai实时系统。 实时
    发表于 10-31 09:53

    GD32H757Z海棠开发板使用手册第十 TFT-8080电阻屏屏触摸实验

    通过本实验主要学习以下内容: LCD触控原理 SPI外设功能
    的头像 发表于 09-19 09:30 ?915次阅读
    【<b class='flag-5'>GD32H757Z</b><b class='flag-5'>海棠</b><b class='flag-5'>派</b><b class='flag-5'>开发板</b><b class='flag-5'>使用手册</b>】<b class='flag-5'>第十</b>五<b class='flag-5'>讲</b> TFT-8080电阻屏屏触摸<b class='flag-5'>实验</b>

    GD32H757Z海棠开发板使用手册第十四讲 TFT-8080口液晶显示

    通过本实验主要学习以下内容: LCD显示原理 EXMC NOR/SRAM模式时序和8080并口时序 LCD显示控制
    的头像 发表于 09-19 09:21 ?1862次阅读
    【<b class='flag-5'>GD32H757Z</b><b class='flag-5'>海棠</b><b class='flag-5'>派</b><b class='flag-5'>开发板</b><b class='flag-5'>使用手册</b>】<b class='flag-5'>第十</b>四讲 TFT-8080口液晶显示

    【北京迅为】iTOP-i.MX6开发板使用手册第四部分固件编译第十四章非设备树Android4.4系统编译

    【北京迅为】iTOP-i.MX6开发板使用手册第四部分固件编译第十四章非设备树Android4.4系统编译
    的头像 发表于 09-12 15:43 ?996次阅读
    【北京迅为】iTOP-i.MX6<b class='flag-5'>开发板</b><b class='flag-5'>使用手册</b>第四部分固件编译<b class='flag-5'>第十</b>四章非设备树Android4.4系统编译

    树莓和arm开发板的区别

    树莓(Raspberry Pi)和ARM开发板都是基于ARM架构的微型计算机,但它们之间存在一些关键区别。 一、历史背景 树莓(Raspberry Pi) 树莓是由英国树莓
    的头像 发表于 08-30 15:36 ?2033次阅读

    linux开发板与树莓的区别

    定义和用途 Linux开发板:Linux开发板是一种基于Linux操作系统的嵌入式开发板,通常用于工业自动化、物联网、智能家居等领域。 树莓:树莓
    的头像 发表于 08-30 15:34 ?2039次阅读

    【「ARM MCU嵌入式开发 | 基于国产GD32F10x芯片」阅读体验】+书籍整体概况

    。可以说是内容相当新颖,结合了当代ARM MCU嵌入式开发市场需求。 二、书籍封面 书籍封面采用国产GD32F10x EVK开发板做为背景图片,更易吸引了业内嵌入式开发者的兴趣,本书还
    发表于 08-25 22:48