别再只会用轮询了!GD32F103 USART中断与DMA传输实战对比(附代码)

news2026/5/1 22:37:39
GD32F103 USART通信三剑客轮询、中断与DMA的实战抉择在嵌入式开发中串口通信就像空气一样无处不在——调试信息输出、传感器数据采集、设备间通信都离不开它。但很多开发者停留在最基础的轮询方式就像只会用螺丝刀却面对一整套精密工具。本文将带你深入GD32F103的USART模块通过实测对比轮询、中断和DMA三种通信方式的性能差异帮你找到不同场景下的最优解。1. USART通信基础与三种模式概览USART通用同步异步收发器是嵌入式系统中使用最广泛的通信接口之一。在GD32F103系列中除了UART4外其他USART模块都支持三种数据传输模式轮询模式CPU不断查询状态寄存器等待数据传输完成中断模式数据传输完成后触发中断CPU在中断服务程序中处理数据DMA模式由DMA控制器自动搬运数据完全解放CPU这三种模式在代码复杂度、CPU占用率和传输效率上存在显著差异。下面这个表格直观展示了它们的核心特点特性轮询模式中断模式DMA模式CPU占用率100%忙等按需处理接近0%代码复杂度最简单中等较复杂延迟确定性低中高适用场景简单调试中低速常规应用高速大数据量传输最大理论波特率约500kbps约1Mbps可达4.5Mbps在实际项目中选择哪种模式需要综合考虑数据量、实时性要求和系统资源占用等因素。比如一个每秒钟只需要传输几十字节的温湿度传感器用轮询可能反而更简单直接而高速ADC采集系统则必须考虑DMA方案。2. 轮询模式简单但低效的实现轮询模式是大多数开发者最先接触的USART使用方式它的实现直白得像打开水龙头喝水——发送数据时等待发送缓冲区空接收数据时检查接收缓冲区非空标志。// 轮询方式发送字符串 void UART_SendString_Polling(uint32_t usart_periph, char* str) { while(*str) { usart_data_transmit(usart_periph, (uint8_t)*str); while(RESET usart_flag_get(usart_periph, USART_FLAG_TBE)); } } // 轮询方式接收数据 uint8_t UART_ReceiveByte_Polling(uint32_t usart_periph) { while(RESET usart_flag_get(usart_periph, USART_FLAG_RBNE)); return usart_data_receive(usart_periph); }这种模式的优点显而易见代码逻辑简单直观易于理解和调试不需要配置中断或DMA硬件依赖小适合在初始化阶段或简单任务中使用但它的缺点同样明显CPU资源浪费在等待传输完成期间CPU只能空转无法执行其他任务实时性差无法及时响应数据到达可能导致数据丢失效率低下在高波特率下CPU可能来不及处理连续到达的数据实测数据显示在115200波特率下轮询模式发送1KB数据需要约90ms期间CPU利用率始终维持在100%。而在处理接收数据时如果主程序有其他耗时操作很容易错过数据接收时机导致溢出错误。提示即使使用轮询模式也建议启用溢出错误中断至少能在数据丢失时及时发现问题。3. 中断模式平衡性能与复杂度的选择中断模式是轮询的升级版它通过硬件中断机制让CPU从不断查询中解放出来。当USART事件如数据接收完成、发送缓冲区空发生时硬件自动触发中断CPU暂停当前任务去处理数据传输。3.1 中断模式配置要点配置USART中断需要关注以下几个关键点NVIC设置配置中断优先级避免被其他高优先级中断阻塞中断事件选择根据需要启用发送完成、接收缓冲区非空等中断中断服务程序优化尽量保持ISR简短避免嵌套中断// 中断模式初始化示例 void USART_Interrupt_Init(uint32_t usart_periph) { // ...GPIO和USART基本配置与轮询模式相同 // 配置NVIC nvic_irq_enable(USART0_IRQn, 1, 0); // 抢占优先级1子优先级0 // 使能接收缓冲区非空中断 usart_interrupt_enable(usart_periph, USART_INT_RBNE); // 可选使能错误中断 usart_interrupt_enable(usart_periph, USART_INT_ERR); } // 中断服务程序 void USART0_IRQHandler(void) { if(RESET ! usart_interrupt_flag_get(USART0, USART_INT_FLAG_RBNE)) { uint8_t data usart_data_receive(USART0); // 将数据存入环形缓冲区 ring_buffer_put(rx_buf, data); } // 处理发送中断 if(RESET ! usart_interrupt_flag_get(USART0, USART_INT_FLAG_TBE)) { if(!ring_buffer_empty(tx_buf)) { uint8_t data ring_buffer_get(tx_buf); usart_data_transmit(USART0, data); } else { // 发送缓冲区空禁用发送中断避免持续触发 usart_interrupt_disable(USART0, USART_INT_TBE); } } }3.2 中断模式的性能特点中断模式相比轮询有了质的飞跃CPU利用率大幅降低实测在相同115200波特率下传输1KB数据CPU利用率降至约15%响应及时数据到达后立即触发中断几乎没有延迟系统吞吐量提升主程序可以并行处理其他任务但中断模式也有其局限性中断开销每次中断都有上下文保存/恢复的开销高频中断下可能成为瓶颈优先级管理不当的中断优先级设置可能导致关键任务被阻塞数据缓冲需要实现环形缓冲区等机制来处理突发数据在波特率超过1Mbps时中断模式开始显得力不从心。此时每个字节的传输时间不足10μs而典型的中断响应和处理时间可能达到1-2μsCPU大部分时间都在处理中断。4. DMA模式极致性能的终极方案DMA直接内存访问是USART通信的性能巅峰它通过专用硬件控制器在内存和外设间直接搬运数据完全不需要CPU介入。GD32F103的DMA控制器支持多达7个通道其中USART0_TX对应DMA0通道4USART0_RX对应DMA0通道5。4.1 DMA模式配置详解配置USART DMA传输需要同时设置DMA控制器和USART模块// DMA发送初始化 void USART_DMA_Tx_Init(uint32_t usart_periph, uint32_t dma_periph, uint32_t dma_channel) { // 启用DMA时钟 rcu_periph_clock_enable(RCU_DMA0); // 配置DMA通道 dma_parameter_struct dma_init_struct; dma_struct_para_init(dma_init_struct); dma_init_struct.direction DMA_MEMORY_TO_PERIPHERAL; dma_init_struct.memory_addr (uint32_t)send_buffer; dma_init_struct.memory_inc DMA_MEMORY_INCREASE_ENABLE; dma_init_struct.memory_width DMA_MEMORY_WIDTH_8BIT; dma_init_struct.number BUFFER_SIZE; dma_init_struct.periph_addr (uint32_t)USART_DATA(usart_periph); dma_init_struct.periph_inc DMA_PERIPH_INCREASE_DISABLE; dma_init_struct.periph_width DMA_PERIPHERAL_WIDTH_8BIT; dma_init_struct.priority DMA_PRIORITY_HIGH; dma_init(dma_periph, dma_channel, dma_init_struct); // 使能DMA通道 dma_channel_enable(dma_periph, dma_channel); // 配置USART使用DMA usart_dma_transmit_config(usart_periph, USART_DENT_ENABLE); } // DMA接收初始化类似主要区别在direction和periph/memory配置4.2 DMA模式性能实测DMA模式的性能优势令人印象深刻CPU占用趋近于零数据传输过程完全由硬件处理超高吞吐量实测在4.5Mbps波特率下仍能稳定传输支持大数据块单次传输可达65535字节下表对比了三种模式在115200波特率下传输1KB数据的性能表现指标轮询模式中断模式DMA模式传输时间(ms)89.288.788.5CPU占用率(%)100151最大中断次数010241代码复杂度★☆☆☆☆★★★☆☆★★★★☆虽然在小数据量时三种模式的传输时间相近但随着数据量增大DMA的优势会越来越明显。特别是在需要同时处理多个外设或复杂算法的系统中DMA解放的CPU资源可以显著提升整体性能。4.3 DMA使用中的坑与技巧内存对齐问题GD32的DMA对内存地址有对齐要求4字节传输时地址必须4字节对齐传输完成判断建议使用DMA中断而非USART TC标志来判断传输完成双缓冲技巧在高速连续传输时可以设置双缓冲区交替使用错误处理必须处理DMA传输错误中断特别是内存访问错误// DMA传输完成中断处理示例 void DMA0_Channel4_IRQHandler(void) { if(dma_interrupt_flag_get(DMA0, DMA_CH4, DMA_INT_FLAG_FTF)) { dma_interrupt_flag_clear(DMA0, DMA_CH4, DMA_INT_FLAG_FTF); // 处理传输完成逻辑 transfer_complete 1; } if(dma_interrupt_flag_get(DMA0, DMA_CH4, DMA_INT_FLAG_ERR)) { dma_interrupt_flag_clear(DMA0, DMA_CH4, DMA_INT_FLAG_ERR); // 处理传输错误 error_occurred 1; } }5. 实战选型指南与混合应用策略了解了三种模式的特性后如何在项目中做出正确选择以下是几个典型场景的建议5.1 单模式应用场景轮询模式适用初始化阶段的简单配置极低频率的调试信息输出资源极度受限的场合中断模式适用中低速常规通信1Mbps需要及时响应的控制指令不定长数据包处理DMA模式适用高速数据流1Mbps大块数据传输64字节低功耗要求的系统5.2 混合模式高级应用在实际项目中经常需要混合使用多种模式以获得最佳效果。以下是几种常见组合DMA发送中断接收适合命令响应型应用发送使用DMA提高效率接收用中断保证实时性中断控制DMA数据传输用中断处理控制指令大数据块传输切换到DMA模式双DMA通道全双工收发均使用独立DMA通道适合高速双向数据流// 混合模式示例DMA发送中断接收 void USART_Mixed_Init(void) { // 初始化USART和GPIO... // 配置DMA发送 USART_DMA_Tx_Init(USART0, DMA0, DMA_CH4); // 配置中断接收 nvic_irq_enable(USART0_IRQn, 1, 0); usart_interrupt_enable(USART0, USART_INT_RBNE); } // 发送函数改用DMA void UART_Send_DMA(uint8_t *data, uint16_t length) { // 等待上次传输完成 while(dma_flag_get(DMA0, DMA_CH4, DMA_FLAG_FTF) RESET); // 配置DMA传输 dma_channel_disable(DMA0, DMA_CH4); dma_memory_address_config(DMA0, DMA_CH4, (uint32_t)data); dma_transfer_number_config(DMA0, DMA_CH4, length); dma_channel_enable(DMA0, DMA_CH4); }5.3 GD32F103的特殊考量GD32F103的USART模块有几个需要特别注意的地方UART4不支持DMA只有USART0-3支持DMA功能时钟源差异USART0在APB2总线最高108MHz其他在APB1最高54MHz中断优先级DMA中断优先级通常应低于USART中断在资源受限的情况下可以巧妙利用USART的TXE发送缓冲区空和TC发送完成不同中断来实现高效传输。比如在中断模式下可以只在缓冲区空时装载数据而在最后使用TC中断来判断整个传输完成。6. 性能优化进阶技巧6.1 波特率极限挑战GD32F103的USART理论上支持最高4.5Mbps的波特率。要达到这个极限需要注意时钟配置确保系统时钟和APB总线时钟正确配置IO速度将USART引脚设置为最高速度模式GPIO_OSPEED_50MHZ信号完整性高频下需要关注PCB布线必要时添加终端电阻// 配置4.5Mbps波特率 usart_baudrate_set(USART0, 4500000U);6.2 低功耗优化在电池供电设备中USART通信的低功耗优化尤为重要空闲时关闭时钟不使用时禁用USART和DMA时钟DMA唤醒配置DMA传输完成中断唤醒MCU自动波特率检测某些型号支持自动波特率可节省配置时间6.3 错误处理与鲁棒性稳定的USART通信需要完善的错误处理机制启用所有错误中断溢出错误、噪声错误、帧错误等超时机制为DMA传输设置硬件或软件超时数据校验添加CRC或校验和验证数据完整性// 全面的错误中断配置 usart_interrupt_enable(USART0, USART_INT_ERR); usart_interrupt_enable(USART0, USART_INT_ORERR); usart_interrupt_enable(USART0, USART_INT_NERR); usart_interrupt_enable(USART0, USART_INT_FERR); usart_interrupt_enable(USART0, USART_INT_PERR);在GD32F103项目中使用USART时我曾遇到一个棘手问题在高波特率下偶尔会出现数据错位。最终发现是GPIO速度配置不足导致的将USART引脚设置为50MHz速度后问题解决。这个案例告诉我们高性能通信需要每个环节都优化到位。

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

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

相关文章

SpringBoot-17-MyBatis动态SQL标签之常用标签

文章目录 1 代码1.1 实体User.java1.2 接口UserMapper.java1.3 映射UserMapper.xml1.3.1 标签if1.3.2 标签if和where1.3.3 标签choose和when和otherwise1.4 UserController.java2 常用动态SQL标签2.1 标签set2.1.1 UserMapper.java2.1.2 UserMapper.xml2.1.3 UserController.ja…

wordpress后台更新后 前端没变化的解决方法

使用siteground主机的wordpress网站,会出现更新了网站内容和修改了php模板文件、js文件、css文件、图片文件后,网站没有变化的情况。 不熟悉siteground主机的新手,遇到这个问题,就很抓狂,明明是哪都没操作错误&#x…

网络编程(Modbus进阶)

思维导图 Modbus RTU(先学一点理论) 概念 Modbus RTU 是工业自动化领域 最广泛应用的串行通信协议,由 Modicon 公司(现施耐德电气)于 1979 年推出。它以 高效率、强健性、易实现的特点成为工业控制系统的通信标准。 包…

UE5 学习系列(二)用户操作界面及介绍

这篇博客是 UE5 学习系列博客的第二篇,在第一篇的基础上展开这篇内容。博客参考的 B 站视频资料和第一篇的链接如下: 【Note】:如果你已经完成安装等操作,可以只执行第一篇博客中 2. 新建一个空白游戏项目 章节操作,重…

IDEA运行Tomcat出现乱码问题解决汇总

最近正值期末周,有很多同学在写期末Java web作业时,运行tomcat出现乱码问题,经过多次解决与研究,我做了如下整理: 原因: IDEA本身编码与tomcat的编码与Windows编码不同导致,Windows 系统控制台…

利用最小二乘法找圆心和半径

#include <iostream> #include <vector> #include <cmath> #include <Eigen/Dense> // 需安装Eigen库用于矩阵运算 // 定义点结构 struct Point { double x, y; Point(double x_, double y_) : x(x_), y(y_) {} }; // 最小二乘法求圆心和半径 …

使用docker在3台服务器上搭建基于redis 6.x的一主两从三台均是哨兵模式

一、环境及版本说明 如果服务器已经安装了docker,则忽略此步骤,如果没有安装,则可以按照一下方式安装: 1. 在线安装(有互联网环境): 请看我这篇文章 传送阵>> 点我查看 2. 离线安装(内网环境):请看我这篇文章 传送阵>> 点我查看 说明&#xff1a;假设每台服务器已…

XML Group端口详解

在XML数据映射过程中&#xff0c;经常需要对数据进行分组聚合操作。例如&#xff0c;当处理包含多个物料明细的XML文件时&#xff0c;可能需要将相同物料号的明细归为一组&#xff0c;或对相同物料号的数量进行求和计算。传统实现方式通常需要编写脚本代码&#xff0c;增加了开…

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器的上位机配置操作说明

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器专为工业环境精心打造&#xff0c;完美适配AGV和无人叉车。同时&#xff0c;集成以太网与语音合成技术&#xff0c;为各类高级系统&#xff08;如MES、调度系统、库位管理、立库等&#xff09;提供高效便捷的语音交互体验。 L…

(LeetCode 每日一题) 3442. 奇偶频次间的最大差值 I (哈希、字符串)

题目&#xff1a;3442. 奇偶频次间的最大差值 I 思路 &#xff1a;哈希&#xff0c;时间复杂度0(n)。 用哈希表来记录每个字符串中字符的分布情况&#xff0c;哈希表这里用数组即可实现。 C版本&#xff1a; class Solution { public:int maxDifference(string s) {int a[26]…

【大模型RAG】拍照搜题技术架构速览:三层管道、两级检索、兜底大模型

摘要 拍照搜题系统采用“三层管道&#xff08;多模态 OCR → 语义检索 → 答案渲染&#xff09;、两级检索&#xff08;倒排 BM25 向量 HNSW&#xff09;并以大语言模型兜底”的整体框架&#xff1a; 多模态 OCR 层 将题目图片经过超分、去噪、倾斜校正后&#xff0c;分别用…

【Axure高保真原型】引导弹窗

今天和大家中分享引导弹窗的原型模板&#xff0c;载入页面后&#xff0c;会显示引导弹窗&#xff0c;适用于引导用户使用页面&#xff0c;点击完成后&#xff0c;会显示下一个引导弹窗&#xff0c;直至最后一个引导弹窗完成后进入首页。具体效果可以点击下方视频观看或打开下方…

接口测试中缓存处理策略

在接口测试中&#xff0c;缓存处理策略是一个关键环节&#xff0c;直接影响测试结果的准确性和可靠性。合理的缓存处理策略能够确保测试环境的一致性&#xff0c;避免因缓存数据导致的测试偏差。以下是接口测试中常见的缓存处理策略及其详细说明&#xff1a; 一、缓存处理的核…

龙虎榜——20250610

上证指数放量收阴线&#xff0c;个股多数下跌&#xff0c;盘中受消息影响大幅波动。 深证指数放量收阴线形成顶分型&#xff0c;指数短线有调整的需求&#xff0c;大概需要一两天。 2025年6月10日龙虎榜行业方向分析 1. 金融科技 代表标的&#xff1a;御银股份、雄帝科技 驱动…

观成科技:隐蔽隧道工具Ligolo-ng加密流量分析

1.工具介绍 Ligolo-ng是一款由go编写的高效隧道工具&#xff0c;该工具基于TUN接口实现其功能&#xff0c;利用反向TCP/TLS连接建立一条隐蔽的通信信道&#xff0c;支持使用Let’s Encrypt自动生成证书。Ligolo-ng的通信隐蔽性体现在其支持多种连接方式&#xff0c;适应复杂网…

铭豹扩展坞 USB转网口 突然无法识别解决方法

当 USB 转网口扩展坞在一台笔记本上无法识别,但在其他电脑上正常工作时,问题通常出在笔记本自身或其与扩展坞的兼容性上。以下是系统化的定位思路和排查步骤,帮助你快速找到故障原因: 背景: 一个M-pard(铭豹)扩展坞的网卡突然无法识别了,扩展出来的三个USB接口正常。…

未来机器人的大脑:如何用神经网络模拟器实现更智能的决策?

编辑&#xff1a;陈萍萍的公主一点人工一点智能 未来机器人的大脑&#xff1a;如何用神经网络模拟器实现更智能的决策&#xff1f;RWM通过双自回归机制有效解决了复合误差、部分可观测性和随机动力学等关键挑战&#xff0c;在不依赖领域特定归纳偏见的条件下实现了卓越的预测准…

Linux应用开发之网络套接字编程(实例篇)

服务端与客户端单连接 服务端代码 #include <sys/socket.h> #include <sys/types.h> #include <netinet/in.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <arpa/inet.h> #include <pthread.h> …

华为云AI开发平台ModelArts

华为云ModelArts&#xff1a;重塑AI开发流程的“智能引擎”与“创新加速器”&#xff01; 在人工智能浪潮席卷全球的2025年&#xff0c;企业拥抱AI的意愿空前高涨&#xff0c;但技术门槛高、流程复杂、资源投入巨大的现实&#xff0c;却让许多创新构想止步于实验室。数据科学家…

深度学习在微纳光子学中的应用

深度学习在微纳光子学中的主要应用方向 深度学习与微纳光子学的结合主要集中在以下几个方向&#xff1a; 逆向设计 通过神经网络快速预测微纳结构的光学响应&#xff0c;替代传统耗时的数值模拟方法。例如设计超表面、光子晶体等结构。 特征提取与优化 从复杂的光学数据中自…