【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) 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/29545.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Roboflow对YOLO数据集、标注、训练、下载

一、Roboflow 介绍 一个功能丰富的在线数据标注和深度学习平台&#xff0c;它提供了从数据标注 到模型训练的一站式服务&#xff0c;yolov5到v8官方推荐的数据集查找及下载网站。 主要功能 1、在线标注数据&#xff1a;Roboflow支持在线标注图片和视频数据&#xff0c;提供多…

如何查看k8s中service的负载均衡策略

在Kubernetes中&#xff0c;Service的负载均衡策略一般由kube-proxy负责&#xff0c;kube-proxy使用iptables或IPVS规则进行负载均衡。默认情况下&#xff0c;kube-proxy使用的是轮询&#xff08;Round Robin&#xff09;策略&#xff0c;但是在使用IPVS模式时&#xff0c;可以…

烂笔头笔记:为JDK安装Charles证书,让你的请求能够像在浏览器中那样被抓包

为什么要为JDK安装Charles证书 众所周知&#xff0c;https就是为了防止中间过程被拦截从而导致数据泄密的。若强行加入Charles代理&#xff0c;数据被解密后再被其重新加密&#xff0c;数据已经被“破坏”&#xff0c;客户端从而拒绝建立连接或解析内容。 #mermaid-svg-ksLo5W…

代码随想录算法训练营第42天(py)| 动态规划 |1049. 最后一块石头的重量 II、494. 目标和、474.一和零

1049. 最后一块石头的重量 II 力扣链接 有一堆石头&#xff0c;用整数数组 stones 表示。其中 stones[i] 表示第 i 块石头的重量。 每一回合&#xff0c;从中选出任意两块石头&#xff0c;然后将它们一起粉碎。假设石头的重量分别为 x 和 y&#xff0c;且 x < y。那么粉碎的…

【Linux】—Apache Hive 安装部署

文章目录 前言认识Metadata认识Metastoremetastore三种配置方式 一、安装前准备二、下载hive-3.1.2安装包三、下载完成后&#xff0c;通过xftp6上传到Linux服务器上四、解压Hive安装包五、配置Hive六、内嵌模型安装—Hive元数据配置到Derby七、本地模式安装—Hive元数据配置到M…

高等数学笔记(二):极限

一、数列极限的定义 以下符号表示 “对于任意给定的” 以下符号表示 “存在” 以下符号表示 “如果什么&#xff08;箭头左&#xff09;&#xff0c;则什么&#xff08;箭头右&#xff09;” 二、收敛数列的性质 2.1 唯一性 2.2 有界性 2.3 保号性 2.4 子数列收敛性 三、函数…

文字转语音在线怎么转?总结了三种快速转换

文字转语音在线怎么转&#xff1f;在数字化信息爆炸的时代&#xff0c;文字转语音的操作也越来越多。无论是为了制作视频配音、播客节目&#xff0c;还是为了方便视力障碍者阅读&#xff0c;文字转语音技术都提供了极大的便利。因此&#xff0c;本文将总结三种文字转语音的在线…

2024年【高压电工】模拟考试题及高压电工考试试题

题库来源&#xff1a;安全生产模拟考试一点通公众号小程序 2024年高压电工模拟考试题为正在备考高压电工操作证的学员准备的理论考试专题&#xff0c;每个月更新的高压电工考试试题祝您顺利通过高压电工考试。 1、【单选题】下列()安全用具是在电气操作中使用的基本安全用具。…

华为安全Security认证,你了解多少?

华为安全Security 认证包含HCIA-Security, HCIP-Security,HCIE-Security。HCIA-Security 掌握中小型网络信息安全基础知识与相关技术&#xff08;华为防火墙技术、加解密技术、PKI 证书体系等&#xff09;&#xff0c;具备搭建小型企业信息安全网络的能力&#xff0c;实现中小企…

文案提取小帮手如何提取文案或视频转文字呢?

文案提取小帮手通常是指专门用于从视频、音频中提取文本内容的工具。以下是一些常见的方法来提取文案或将视频转换为文字&#xff1a; 方法一&#xff1a;必应搜索引擎 打开必应输入【视频下载plus助手工具】 然后点击进入选择视频转文案按钮使用手机扫码即可体验视频转文字功…

threejs教程:绘制3D地图(广东省区划图)

一、效果展示&#xff1a; 二、开发准备 Three.js中文文档&#xff1a;Three.js中文网 Three.js文本渲染插件&#xff1a;Troika 3D Text - Troika JS 行政区划边界数据查询&#xff08;阿里云数据可视化平台&#xff09;&#xff1a;DataV.GeoAtlas地理小工具系列 1. 在项目…

DDL与DML语句

1、DDL语句 CREATE table user(user_id INT AUTO_INCREMENT PRIMARY KEY,userName VARCHAR(50) NOT NULL UNIQUE,password VARCHAR(255) NOT NULL,email VARCHAR(100) UNIQUE,phone VARCHAR(20) UNIQUE,create_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP ); CREATE TABLE user_w…

【Ubuntu开发入门之“orangepi H3 linux开发③kernel移植调试“】

Ubuntu开发入门之"orangepi H3 linux开发③kernel移植调试 问题描述解决方法获取源码内核适配和编译制作TF卡分区,以备存放各个分区和文件根文件系统拷贝郑重声明:本人原创博文,都是实战,均经过实际项目验证出货的 转载请标明出处:攻城狮2015 Platform: Allwinner H3 O…

Nginx和Tomcat负载均衡、动静分离集群1

一、Nginx负载均衡 1.负载均衡概念 服务端接受来自客户端的请求中&#xff0c;既有动态资源也有动态资源&#xff0c;静态资源由 Nginx 提供服务&#xff0c;动态资源 Nginx 转发至后端

【机器学习300问】123、什么是GRU?GRU网络的基本结构是怎样的?

在之前的文章中&#xff0c;我们谈到了标准RNN所面临的诸多困境&#xff0c;你也可以理解为RNN的缺点。其中最让人苦恼的就是梯度消失问题&#xff0c;正是由于梯度消失问题的存在&#xff0c;导致RNN无法获得上下文的长期依赖信息。那么就没有办法解决了吗&#xff1f;非也&am…

Linux实现: 客户端(cli01)通过TCP(或UDP)连接到聊天服务器(serv)进行聊天?(伪代码版本)

&#x1f3c6;本文收录于「Bug调优」专栏&#xff0c;主要记录项目实战过程中的Bug之前因后果及提供真实有效的解决方案&#xff0c;希望能够助你一臂之力&#xff0c;帮你早日登顶实现财富自由&#x1f680;&#xff1b;同时&#xff0c;欢迎大家关注&&收藏&&…

Flutter 小技巧之面试题里有意思的异步问题

很久没更新小技巧系列了&#xff0c;本次简单介绍一下 Flutter 面试里我认为比较有意思的异步基础知识点。 首先我们简单看一段代码&#xff0c;如下代码所示&#xff0c;是一个循环定时器任务&#xff0c;这段代码里&#xff1a; testFunc 循环每 1 秒执行一次 asyncWorkasy…

C++ 60 之 虚析构和纯虚析构

#include <iostream> #include <string> #include <cstring> using namespace std;class Animal13{ public:Animal13(){cout << "Animal的默认构造函数" << endl;}virtual void speak(){cout << "动物叫" << en…

?? 与 || 在 JavaScript 中的微妙差别

起初&#xff0c;你可能会认为你可以随意替换任何你喜欢的人&#xff0c;对吗&#xff1f; 错误。他们并非你所想的那样。 我们必须一劳永逸地学习这个区别&#xff0c;以避免日后出现痛苦的错误。 这个差别是什么&#xff1f; 这是他们对待真值和假值的令人难以置信的对比。这…

C++ 61 之 函数模版

#include <iostream> #include <string> using namespace std;void swapInt(int &a,int &b){int temp a;a b;b temp; }void swapDou(double& a, double& b){double temp a;a b;b temp; }// T代表通用数据类型&#xff0c;紧接着后面的代码&a…