[STM32 HAL库]串口空闲中断+DMA接收不定长数据
一、空闲中断
STM32的串口具有空闲中断,什么叫做空闲呢?如何触发空闲中断呢?
- 空闲:串口发送的两个字符之间间隔非常短,所以在两个字符之间不叫空闲。空闲的定义是总线上在一个字节的时间内没有再接收到数据。
- 触发条件:空闲中断是检测到有数据被接收后,总线上在一个字节的时间内没有再接收到数据的时候发生的。而总线在什么情况时,会有一个字节时间内没有接收到数据呢?一般就只有一个数据帧发送完成的情况,所以串口的空闲中断也叫帧中断。
开启空闲中断后,要重写对应的回调函数HAL_UARTEx_RxEventCallback,在函数里做些处理。
二、实验内容
使用USART1外设,接收电脑发来的数据,然后将接收到的数据发送给电脑。

二、STM32CUBEMX配置
USART1的模式为异步通信、115200波特率、数据长度8位、无校验位、停止位1位。

DMA Settings的配置,开启串口接收的DMA。

NVIC Settings的配置,开启USART1的全局中断。

这里两个中断优先级我都给了1,根据不同情况修改中断优先级

三、keil代码
首先确保魔术棒中的Use MicroLIB这个选项勾选上,不然串口发送数据会不正常

添加头文件,因为要使用memset函数
#include "string.h"
定义串口接收数据数组
#define BUFF_SIZE 128 //接收缓存大小
uint8_t rx_buffer[BUFF_SIZE]; // 创建接收缓存,大小为BUFF_SIZE
手动在usart.h外部声明hdma_usart1_rx,在main函数中要使用
extern UART_HandleTypeDef huart1;/* USER CODE BEGIN Private defines */
extern DMA_HandleTypeDef hdma_usart1_rx; //手动外部声明
/* USER CODE END Private defines */void MX_USART1_UART_Init(void);
在main函数初始化添加这两个函数,不然串口首次无法进入中断
HAL_UARTEx_ReceiveToIdle_DMA(&huart1,rx_buffer,BUFF_SIZE); //手动开启串口DMA模式接收数据
__HAL_DMA_DISABLE_IT(&hdma_usart1_rx, DMA_IT_HT); //手动关闭DMA_IT_HT中断
重定向HAL_UARTEx_RxEventCallback串口接收完成回调函数
/* 串口接收完成回调函数 */
void HAL_UARTEx_RxEventCallback(UART_HandleTypeDef *huart, uint16_t Size)
{if (huart->Instance == USART1){HAL_UARTEx_ReceiveToIdle_DMA(&huart1, rx_buffer, BUFF_SIZE); // 接收完毕后重启串口DMA模式接收数据HAL_UART_Transmit(&huart1, rx_buffer, Size, 0xffff); // 将接收到的数据再发出__HAL_DMA_DISABLE_IT(&hdma_usart1_rx, DMA_IT_HT); // 手动关闭DMA_IT_HT中断memset(rx_buffer, 0, BUFF_SIZE); // 清除接收缓存}
}
重定向HAL_UART_ErrorCallback串口错误回调函数
/* 串口错误回调函数 */
void HAL_UART_ErrorCallback(UART_HandleTypeDef * huart)
{if(huart->Instance == USART1){HAL_UARTEx_ReceiveToIdle_DMA(&huart1, rx_buffer, BUFF_SIZE); //手动开启串口DMA模式接收数据__HAL_DMA_DISABLE_IT(&hdma_usart1_rx, DMA_IT_HT); // 手动关闭DMA_IT_HT中断memset(rx_buffer, 0, BUFF_SIZE); // 清除接收缓存}
}
以下是main函数完整代码
#include "main.h"
#include "dma.h"
#include "usart.h"
#include "gpio.h"#include "stdio.h"
#include "string.h"void SystemClock_Config(void);#define BUFF_SIZE 128 //接收缓存大小
uint8_t rx_buffer[BUFF_SIZE]; // 创建接收缓存,大小为BUFF_SIZEint main(void)
{HAL_Init();SystemClock_Config();MX_GPIO_Init();MX_DMA_Init();MX_USART1_UART_Init();HAL_UARTEx_ReceiveToIdle_DMA(&huart1,rx_buffer,BUFF_SIZE); //手动开启串口DMA模式接收数据__HAL_DMA_DISABLE_IT(&hdma_usart1_rx, DMA_IT_HT); //手动关闭DMA_IT_HT中断 while (1){}
}
void SystemClock_Config(void)
{//...
}
/* USER CODE BEGIN 4 */
/* 串口接收完成回调函数 */
void HAL_UARTEx_RxEventCallback(UART_HandleTypeDef *huart, uint16_t Size)
{if (huart->Instance == USART1){HAL_UARTEx_ReceiveToIdle_DMA(&huart1, rx_buffer, BUFF_SIZE); /// 接收完毕后重启串口DMA模式接收数据HAL_UART_Transmit(&huart1, rx_buffer, Size, 0xffff); // 将接收到的数据再发出__HAL_DMA_DISABLE_IT(&hdma_usart1_rx, DMA_IT_HT); // 手动关闭DMA_IT_HT中断memset(rx_buffer, 0, BUFF_SIZE); // 清除接收缓存}
}
/* 串口错误回调函数 */
void HAL_UART_ErrorCallback(UART_HandleTypeDef * huart)
{if(huart->Instance == USART1){HAL_UARTEx_ReceiveToIdle_DMA(&huart1, rx_buffer, BUFF_SIZE);//手动开启串口DMA模式接收数据__HAL_DMA_DISABLE_IT(&hdma_usart1_rx, DMA_IT_HT); // 手动关闭DMA_IT_HT中断memset(rx_buffer, 0, BUFF_SIZE); // 清除接收缓存}
}
/* USER CODE END 4 */
void Error_Handler(void)
{/* USER CODE BEGIN Error_Handler_Debug *//* User can add his own implementation to report the HAL error return state */__disable_irq();while (1){}/* USER CODE END Error_Handler_Debug */
}
四、原理讲解
我们使用串口空闲中断的目的是为了获取完整的数据,并且是由软件自动判断是否接收完整。当软件判断接收到完整的数据时,就会产生中断进入回调函数HAL_UARTEx_RxEventCallback。
软件判断接收到完整数据有两种情况:
- 1、数据接收后,一个字节未接收到数据;
- 2、当前的
数据接收长度与预设接收长度相等;预设接收长度为HAL_UARTEx_ReceiveToIdle_DMA的参数BUFF_SIZE;
预设接收长度要考虑好,若数据接收长度>预设接受长度,就会出现数据的丢失情况。因为数据接收长度是随着接收不断累加的,其大小等于预设接收长度时就会触发中断,程序就会判断成接收到完整数据,就是第二情况。
1、HAL_UARTEx_ReceiveToIdle_DMA
在main函数里我们调用了HAL_UARTEx_ReceiveToIdle_DMA,该函数会开启USART1的空闲中断,并启用串口以DMA方式接收数据,当数据接收完成之后进入HAL_UARTEx_RxEventCallback回调函数。随后,我们在回调函数里进行数据处理即可。
2、__HAL_DMA_DISABLE_IT
为什么要手动关闭这个中断,如果不关闭这个中断,程序在接收一串完整的数据时会进入执行两次HAL_UARTEx_RxEventCallback。一次是数据传输一半时,一次数数据传输完成时,这与我们预期的不符。我们只希望在数据传输完成时进入回调函数进行数据处理。
__HAL_DMA_DISABLE_IT的回调函数是UART_DMARxHalfCplt,该回调函数会执行一次HAL_UARTEx_RxEventCallback
3、HAL_UARTEx_RxEventCallback
在该回调函数里,需要手动开启中断,才能进入下一次的中断。
4、HAL_UART_ErrorCallback
重写这个回调函数主要是为了防止一些错误情况的发生。若产生错误中断进入HAL_UART_ErrorCallback,则无法进入HAL_UARTEx_RxEventCallback回调函数,也就无法开启下一次空闲中断。所以,这里在HAL_UART_ErrorCallback里手动开启了空闲中断,做了一些恢复处理。
错误示例:开发板设置115200波特率,电脑串口用9600波特率,电脑发送数据之后,程序会进入串口错误中断,而进入不了空闲中断。若程序当中未在错误中断进行错误处理,即使电脑串口修改成115200波特率进行通信,程序也无法进入空闲中断。
五、参考文章
配置和代码链接,里面详解了相关一些函数的源码
链接: 【STM32 HAL库实战】串口DMA + 空闲中断 实现不定长数据接收
空闲中断讲解
链接: STM32串口之空闲中断
相关文章:
[STM32 HAL库]串口空闲中断+DMA接收不定长数据
一、空闲中断 STM32的串口具有空闲中断,什么叫做空闲呢?如何触发空闲中断呢? 空闲:串口发送的两个字符之间间隔非常短,所以在两个字符之间不叫空闲。空闲的定义是总线上在一个字节的时间内没有再接收到数据。触发条件…...
三、华为交换机 Hybrid
一、Hybrid功能 Hybrid口既可以连接普通终端的接入链路(类似于Access接口),又可以连接交换机间的干道链路(类似于Trunk接口)。它允许多个VLAN的帧通过,并可以在出接口方向将某些VLAN帧的标签剥掉࿰…...
如何通过 Apache Airflow 将数据导入 Elasticsearch
作者:来自 Elastic Andre Luiz 了解如何通过 Apache Airflow 将数据导入 Elasticsearch。 Apache Airflow Apache Airflow 是一个旨在创建、安排(schedule)和监控工作流的平台。它用于编排 ETL(Extract-Transform-Load࿰…...
Android Studio:Linux环境下安装与配置
更多内容:XiaoJ的知识星球 Android Studio:Linux环境下安装与配置 1.安装JDK2.安装Android Studio2.1 获取安装包2.2 安装(1)配置环境变量:(2)运行安装:(3)配…...
token是用来鉴权的,那session是用来干什么的?
在Web应用和API设计中,鉴权与会话管理是两个核心概念,它们对于确保用户身份的安全性和维护用户会话状态至关重要。Token和Session是两种常用的鉴权与会话管理机制,它们各自具有独特的工作原理和适用场景。下面是对Token和Session的详细解析及…...
基于 WEB 开发的二手车辆销售管理系统设计与实现
标题:基于 WEB 开发的二手车辆销售管理系统设计与实现 内容:1.摘要 摘要:随着互联网技术的不断发展,电子商务在各个领域得到了广泛的应用。本文以二手车辆销售管理系统为例,探讨了基于 WEB 开发的销售管理系统的设计与实现。通过对系统需求的…...
wordpress的火车头商品发布接口
<?php require ../wp-load.php; ini_set(memory_limit, 1024M); set_time_limit(180);$top_cat ; # 图片链接域名替换 $image_host ;$start_time microtime(true);$counter 0; // 临时缓存 $products $skus $categories []; $var_sku_index 1;$rowData$_POST;// if…...
浙江安吉成新照明电器:Acrel-1000DP 分布式光伏监控系统应用探索
安科瑞吕梦怡 18706162527 摘 要:分布式光伏发电站是指将光伏发电组件安装在用户的建筑物屋顶、空地或其他适合的场地上,利用太阳能进行发电的一种可再生能源利用方式,与传统的大型集中式光伏电站相比,分布式光伏发电具有更灵活…...
总结3..
#include<stdio.h> int n,m; int a[1002][1002]; int b[1002][1002];//判断该空的八连通图是否被走过 int gg0; int dd0; int xz[8]{-1,-1,-1,0,0,1,1,1},yz[8]{-1,0,1,-1,1,-1,0,1};//八个方向 void dfs(int x,int y) { int dx,dy; for(int i0;i<8;i) { …...
信息奥赛一本通 1168:大整数加法
这道题是一道大整数加法,涉及到高精度的算法,比如说有两个数要进行相加,1111111111111111111111111111111111111112222222222222222222222222222222,那么如果这两个数很大的话我们常用的数据类型是不能进行计算的,那么…...
3.3 OpenAI GPT-4, GPT-3.5, GPT-3 模型调用:开发者指南
OpenAI GPT-4, GPT-3.5, GPT-3 模型调用:开发者指南 OpenAI 的 GPT 系列语言模型,包括 GPT-4、GPT-3.5 和 GPT-3,已经成为自然语言处理领域的标杆。无论是文本生成、对话系统,还是自动化任务,开发者都可以通过 API 调用这些强大的模型来增强他们的应用。本文将为您详细介…...
横盘出击的三种经典走势形态,买点以及需要注意的问题技术详解
龙头股在横盘整理过程中,也会出现几种不同的形态,比如矩形整理形态,或者在某一趋势线下方运行。 第一种形态:突破横盘趋势线 突破横盘趋势线时识别横盘龙头启动的关键点位。股价经过一段时间的横盘后,突然出现快速上…...
处理没有提示的字符串、计算相隔天数应用题
正常情况下,小云每天跑 1 千米。如果某天是周一或者月初(1 日),为了激励自己,小云要跑 2 千米。如果同时是周一或月初,小云也是跑 2 千米。 小云跑步已经坚持了很长时间,从 1990 年 1 月 1 日周…...
【LeetCode】力扣刷题热题100道(31-35题)附源码 搜索二维矩阵 岛屿数量 腐烂的橙子 课程表 实现 Trie (前缀树)(C++)
一、搜索二维矩阵 编写一个高效的算法来搜索 m x n 矩阵 matrix 中的一个目标值 target 。该矩阵具有以下特性: 每行的元素从左到右升序排列。每列的元素从上到下升序排列。 可以使用 从右上角开始搜索 的方法来有效地找到目标值。 选择起始位置: 从矩…...
react使用react-redux状态管理
1、安装 npm install react-redux2、创建store.js import { createStore } from redux;// 定义初始状态 const initialState {counter: 888 };// 定义 reducer 函数,根据 action 类型更新状态 function reducer(state initialState, action) {switch (action.ty…...
04_角色创建窗口
将上文的登录窗口隐藏 创建空节点 作为创建角色窗口 命名为CreateWnd 创建输入的名字的输入框 再创建一个按钮用来随机角色名字 创建开始游戏按钮 End....
Dockerfile -> Docker image -> Docker container
1. Dockfile -> Docker image docker build -t shuai_image -f xxx/xxx/Dockerfile . (.不能少)出现: [] Building xxx(10/17) > [internal] load build definition from Dockerfile > > transferring dockerfile: … > > transferring context …...
LDN的蓝牙双模键盘帮助文档
文档索引 已支持的PCB列表(仅列出少部分):键盘特性硬件软件键盘以及驱动蓝牙模式USB模式 驱动功能介绍主界面键盘列表页面键盘配置(使用双模键盘的请务必细看本说明)功能层配置(改键)触发层配置(改FN键等触发功能)功能选择(重要&a…...
搭建一个基于Spring Boot的驾校管理系统
搭建一个基于Spring Boot的驾校管理系统可以涵盖多个功能模块,例如学员管理、教练管理、课程管理、考试管理、车辆管理等。以下是一个简化的步骤指南,帮助你快速搭建一个基础的系统。 1. 项目初始化 使用 Spring Initializr 生成一个Spring Boot项目&am…...
运动相机拍视频过程中摔了,导致录视频打不开怎么办
3-11 在使用运动相机拍摄激烈运动的时候,极大的震动会有一定概率使得保存在存储卡中的视频出现打不开的情况,原因是存储卡和相机在极端情况下,可能会出现接触不良的问题,如果遇到这种问题,就不得不进行视频修复了。 本…...
2026届最火的十大AI写作平台解析与推荐
Ai论文网站排名(开题报告、文献综述、降aigc率、降重综合对比) TOP1. 千笔AI TOP2. aipasspaper TOP3. 清北论文 TOP4. 豆包 TOP5. kimi TOP6. deepseek 在学术研究里,人工智能辅助撰写开题报告已然成了重要工具的一种,用户…...
国内贸易商选工商业储能代工厂需要关注哪些核心细节?
近两年工商业储能的海内外需求持续攀升,不少贸易商在筛选合作工厂时,经常遇到证货不符、交付延期、性能不达标等问题,本文从工艺、合规、成本三个维度拆解核心考察标准。一、合规认证的核验标准核心认证清单:内销产品需具备 GB/T …...
软件竞争管理中的差异化策略
在当今高度数字化的商业环境中,软件行业的竞争日益激烈。企业若想在市场中脱颖而出,差异化策略成为关键。通过独特的价值主张和创新的产品设计,软件公司能够有效区分自身与竞争对手,吸引目标用户并建立长期竞争优势。本文将探讨软…...
LFM2.5-1.2B-Thinking-GGUF应用案例:工厂巡检报告自动生成+隐患关键词高亮标注
LFM2.5-1.2B-Thinking-GGUF应用案例:工厂巡检报告自动生成隐患关键词高亮标注 1. 项目背景与需求 在工业生产环境中,设备巡检是保障安全生产的重要环节。传统的人工巡检报告撰写存在以下痛点: 效率低下:巡检员需要花费大量时间…...
WordPress 站长自查手册:手把手教你用 WPScan 给自己的网站做一次免费“安全体检”
WordPress 站长安全自查指南:用 WPScan 给网站做专业级体检 作为 WordPress 站长,你是否经常担心网站存在安全隐患却无从下手?就像定期体检能预防疾病一样,网站也需要定期安全检查。WPScan 就是专为 WordPress 设计的"体检仪…...
升级 IntelliJ IDEA 编辑器到2026.1
提示升级。提示插件升级。右下角提示升级下载信息。遇到安装后提示试用期的问题;私信。...
从零搞懂Transformer,从位置编码到自注意力,大模型的核心逻辑全拆解
平时我们用ChatGPT聊天、用翻译软件做中英互译、用AI写文案,甚至让AI帮忙编代码,背后最核心的“功臣”,都是2017年Google团队提出的Transformer模型。这个看起来复杂的模型,其实打破了传统AI的局限,靠着高效的计算能力…...
【C++原创开源】formort.h:一行头文件,实现比JS模板字符串更爽的链式拼接+响应式变量
文章目录一、先看效果:比JS模板字符串更爽的写法二、核心功能:不止是拼接,还有响应式1. 任意类型|链式拼接2. Cast响应式变量绑定三、核心实现:几十行代码,看懂原理四、使用方法:零门槛,直接上手…...
LTspice FFT 仿真实战:从基础操作到高级参数调优
1. LTspice FFT功能入门指南 第一次接触LTspice的FFT功能时,我完全被那些频谱图搞懵了。后来才发现,这其实是电路仿真中最实用的分析工具之一。简单来说,FFT(快速傅里叶变换)就像给电路信号做"体检"…...
Translumo:打破语言壁垒的终极屏幕翻译解决方案 - 如何实现游戏、视频、软件的实时翻译
Translumo:打破语言壁垒的终极屏幕翻译解决方案 - 如何实现游戏、视频、软件的实时翻译 【免费下载链接】Translumo Advanced real-time screen translator for games, hardcoded subtitles in videos, static text and etc. 项目地址: https://gitcode.com/gh_mi…...
