STM32 HAL库USART串口DMA IDLE中断编程:避坑指南

HAL_UART_Receive接收最容易丢数据了,STM32 HAL库UART查询方式实例

可以考虑用中断来实现,但是HAL_UART_Receive_IT还不能直接用,容易数据丢失,实际工作中不会这样用,STM32 HAL库USART串口中断编程:演示数据丢失,

需要在此基础优化一下. STM32F103 HAL库USART串口中断,利用环形缓冲区来防止数据丢失.

本文介绍STM32F103 HAL库USART串口DMA IDLE中断.

IDLE 中断 在串口通信里,IDLE 代表空闲状态,其定义为:总线在一个字节的传输时间内未再接收到新数据。

或许有人会有疑问:UART 的DMA RxD 引脚初始状态就是空闲的,那 IDLE 中断会一直触发吗?其实并非如此。当我们使能 IDLE 中断后,它不会立即产生。只有在至少接收到 1 个数据后,若发现接下来一个字节的时间里都没有新数据到来,才会触发 IDLE 中断。

使用 DMA 进行数据接收时,虽然能显著提升 CPU 的使用效率,但也存在一个问题,即“无法预先知晓要接收的数据量”。然而,我们往往希望能尽快处理接收到的数据。举个例子,假设我们打算读取 150 字节的数据,但在接收到 50 字节后,对方停止了数据发送,这种情况下该如何判断数据传输已经中止呢?此时,IDLE 中断就起作用了。

坑在HAL_UARTEx_ReceiveToIdle_DMA(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size)的DMA的半满中断-UART_DMARxHalfCplt,关闭 DMA 的半传输完成中断    __HAL_DMA_DISABLE_IT(&hdma_usart1_rx, DMA_IT_HT);   

目录

 一、开发环境

二、配置STM32CubeMX

三、代码实现与部署

四、运行结果:

​五、注意事项


主要是5个函数的调用和实现.

HAL_UARTEx_ReceiveToIdle_DMA(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size);__HAL_DMA_DISABLE_IT(&hdma_usart1_rx, DMA_IT_HT);   void circle_buf_init(p_circle_buf pCircleBuf, uint32_t len, uint8_t *buf);//初始化环形缓冲区int circle_buf_read(p_circle_buf pCircleBuf, uint8_t *pVal);//读取环形缓冲区int circle_buf_write(p_circle_buf pCircleBuf, uint8_t val);//写环形缓冲区

1.HAL_UARTEx_ReceiveToIdle_DMA(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size);调用此函数后,务必关闭 DMA 的半传输完成中断。若未关闭,当接收到超过数据总长度一半的数据时,系统不仅会触发一次空闲中断,还会触发一次半传输完成中断。而这两个中断均会调用一次HAL_UARTEx_RxEventCallback 回调函数,导致该回调函数总共被调用两次。

2.__HAL_DMA_DISABLE_IT(&hdma_usart1_rx, DMA_IT_HT); 禁用与 USART1 接收相关的 DMA 通道的半传输完成中断。

__HAL_DMA_DISABLE_IT 是一个 HAL 库提供的宏,专门用于禁用 DMA 的特定中断。

&hdma_usart1_rx是指向 USART1 接收所用 DMA 句柄的指针,明确要操作的是哪个 DMA 通道。

DMA_IT_HT代表半传输完成中断(Half Transfer Complete Interrupt),指定要禁用的具体中断类型。

3.从源头到调用回调函数的调用过程,  HAL_UARTEx_ReceiveToIdle_DMA->UART_Start_Receive_DMA(huart, pData, Size)->huart->hdmarx->XferHalfCpltCallback = UART_DMARxHalfCplt或  huart->hdmarx->XferCpltCallback = UART_DMAReceiveCplt;

DMA1_Channel5_IRQHandler->HAL_DMA_IRQHandler(&hdma_usart1_rx)->  hdma->XferHalfCpltCallback(hdma)或hdma->XferCpltCallback(hdma);

 一、开发环境

硬件:正点原子精英版 V2 STM32F103开发板

单片机:STM32F103ZET6

Keil版本:5.32

STM32CubeMX版本:6.9.2

STM32Cube MCU Packges版本:STM32F1xx_DFP.2.4.1
串口:USART1(PA9,PA10)

二、配置STM32CubeMX

1.启动STM32CubeMX,新建STM32CubeMX项目

2.选择MCU:在软件中选择你的STM32型号-STM32F103ZET6。

3.选择时钟源:

4.配置时钟:

​5.使能Debug功能:Serial Wire

​6.HAL库时基选择:SysTick

7.USART1配置:选择异步模式,使能中断。

8.配置工程参数:在Project标签页中,配置项目名称和位置,选择工具链MDK-ARM。9.生成代码:在Code Generator标签页中,配置工程外设文件与HAL库,勾选头文件.c和.h文件分开,然后点击Project > Generate Code生成代码。 

三、代码实现与部署

 main.c增加代码

/* USER CODE BEGIN Header */
/********************************************************************************* @file           : main.c* @brief          : Main program body******************************************************************************* @attention** Copyright (c) 2025 STMicroelectronics.* All rights reserved.** This software is licensed under terms that can be found in the LICENSE file* in the root directory of this software component.* If no LICENSE file comes with this software, it is provided AS-IS.********************************************************************************/
/* USER CODE END Header */
/* Includes ------------------------------------------------------------------*/
#include "main.h"
#include "dma.h"
#include "usart.h"
#include "gpio.h"/* Private includes ----------------------------------------------------------*/
/* USER CODE BEGIN Includes */
#include <string.h>
#include"circle_buffer.h"/* USER CODE END Includes *//* Private typedef -----------------------------------------------------------*/
/* USER CODE BEGIN PTD */
extern UART_HandleTypeDef huart1;
extern DMA_HandleTypeDef hdma_usart1_rx;
static circle_buf g_CircleBuf;
uint32_t buf_len;
static uint8_t g_RecvChar;
static uint8_t g_RecvTempBuf[10];
static uint8_t g_RecvBuf[150];
int UART1_read(uint8_t *pVal)
{return circle_buf_read(&g_CircleBuf, pVal);
}/* USER CODE END PTD *//* Private define ------------------------------------------------------------*/
/* USER CODE BEGIN PD *//* USER CODE END PD *//* Private macro -------------------------------------------------------------*/
/* USER CODE BEGIN PM *//* USER CODE END PM *//* Private variables ---------------------------------------------------------*//* USER CODE BEGIN PV *//* USER CODE END PV *//* Private function prototypes -----------------------------------------------*/
void SystemClock_Config(void);
/* USER CODE BEGIN PFP */
char *str= "hello\r\n";
char c;
/* USER CODE END PFP *//* Private user code ---------------------------------------------------------*/
/* USER CODE BEGIN 0 *//* USER CODE END 0 *//*** @brief  The application entry point.* @retval int*/
int main(void)
{/* USER CODE BEGIN 1 *//* USER CODE END 1 *//* MCU Configuration--------------------------------------------------------*//* Reset of all peripherals, Initializes the Flash interface and the Systick. */HAL_Init();/* USER CODE BEGIN Init *//* USER CODE END Init *//* Configure the system clock */SystemClock_Config();/* USER CODE BEGIN SysInit *//* USER CODE END SysInit *//* Initialize all configured peripherals */MX_GPIO_Init();MX_DMA_Init();MX_USART1_UART_Init();/* USER CODE BEGIN 2 */HAL_UART_Transmit(&huart1,str,strlen(str),1000);circle_buf_init(&g_CircleBuf, 150, g_RecvBuf);//初始化环形缓冲区HAL_UARTEx_ReceiveToIdle_DMA (&huart1,g_RecvTempBuf,10);//一开始就打开中断__HAL_DMA_DISABLE_IT(&hdma_usart1_rx, DMA_IT_HT);    //关闭DMA本身的半传输中断/* USER CODE END 2 *//* Infinite loop *//* USER CODE BEGIN WHILE */while (1){/* USER CODE END WHILE *//* USER CODE BEGIN 3 */while (0 != UART1_read(&c)); HAL_UART_Transmit(&huart1, &c, 1, 1000);HAL_UART_Transmit(&huart1, "\r\n", 2, 1000);}/* USER CODE END 3 */
}/*** @brief System Clock Configuration* @retval None*/
void SystemClock_Config(void)
{RCC_OscInitTypeDef RCC_OscInitStruct = {0};RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};/** Initializes the RCC Oscillators according to the specified parameters* in the RCC_OscInitTypeDef structure.*/RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE;RCC_OscInitStruct.HSEState = RCC_HSE_ON;RCC_OscInitStruct.HSEPredivValue = RCC_HSE_PREDIV_DIV1;RCC_OscInitStruct.HSIState = RCC_HSI_ON;RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON;RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE;RCC_OscInitStruct.PLL.PLLMUL = RCC_PLL_MUL9;if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK){Error_Handler();}/** Initializes the CPU, AHB and APB buses clocks*/RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK|RCC_CLOCKTYPE_PCLK1|RCC_CLOCKTYPE_PCLK2;RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK;RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV2;RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1;if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_2) != HAL_OK){Error_Handler();}
}/* USER CODE BEGIN 4 */
static volatile int g_rx_cplt = 0;void HAL_UARTEx_RxEventCallback(UART_HandleTypeDef *huart, uint16_t Size)
{for (int i = 0; i < Size; i++){circle_buf_write(&g_CircleBuf, g_RecvTempBuf[i]);}HAL_UARTEx_ReceiveToIdle_DMA(&huart1, g_RecvTempBuf, 10);//重新开中断__HAL_DMA_DISABLE_IT(&hdma_usart1_rx, DMA_IT_HT);    //关闭DMA本身的半传输中断}
/* USER CODE END 4 *//*** @brief  This function is executed in case of error occurrence.* @retval None*/
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 */
}#ifdef  USE_FULL_ASSERT
/*** @brief  Reports the name of the source file and the source line number*         where the assert_param error has occurred.* @param  file: pointer to the source file name* @param  line: assert_param error line source number* @retval None*/
void assert_failed(uint8_t *file, uint32_t line)
{/* USER CODE BEGIN 6 *//* User can add his own implementation to report the file name and line number,ex: printf("Wrong parameters value: file %s on line %d\r\n", file, line) *//* USER CODE END 6 */
}
#endif /* USE_FULL_ASSERT */

2.增加代码circle_buffer.h,circle_buffer.c

#ifndef _CIRCLE_BUF_H
#define _CIRCLE_BUF_H#include <stdint.h>typedef struct circle_buf {uint32_t r;uint32_t w;uint32_t len;uint8_t *buf;
}circle_buf, *p_circle_buf;void circle_buf_init(p_circle_buf pCircleBuf, uint32_t len, uint8_t *buf);int circle_buf_read(p_circle_buf pCircleBuf, uint8_t *pVal);int circle_buf_write(p_circle_buf pCircleBuf, uint8_t val);#endif /* _CIRCLE_BUF_H */
#include <stdint.h>
#include "circle_buffer.h"void circle_buf_init(p_circle_buf pCircleBuf, uint32_t len, uint8_t *buf)
{pCircleBuf->r = pCircleBuf->w = 0;pCircleBuf->len = len;pCircleBuf->buf = buf;
}int circle_buf_read(p_circle_buf pCircleBuf, uint8_t *pVal)
{if (pCircleBuf->r != pCircleBuf->w){*pVal = pCircleBuf->buf[pCircleBuf->r];pCircleBuf->r++;if (pCircleBuf->r == pCircleBuf->len)pCircleBuf->r = 0;return 0;}else{return -1;}
}int circle_buf_write(p_circle_buf pCircleBuf, uint8_t val)
{uint32_t next_w;next_w = pCircleBuf->w + 1;if (next_w == pCircleBuf->len)next_w = 0;if (next_w != pCircleBuf->r){pCircleBuf->buf[pCircleBuf->w] = val;pCircleBuf->w = next_w;return 0;}else{return -1;}
}

  3.连接USART1:用USB转TTL工具连接当前硬件USART1的PA9、PA10,GND。​​​

 4.打开串口助手:​​​

 5.编译代码:Keil编译生成的代码。

 6.烧录程序:将编译好的程序用ST-LINK烧录到STM32微控制器中。

四、运行结果

1. 程序烧录完成并运行,复位打印hello,发送"abcdefghij"时候,打印"abcdefghij",数据没有丢失.  2.可以把两处关闭半满中断的程序屏蔽__HAL_DMA_DISABLE_IT(&hdma_usart1_rx, DMA_IT_HT); ,观察现象,编译烧录完成并运行,复位打印hello,发送"abcdefghij"时候,打印"abcdeabcdefghij","abcde"重复了,不正确. 

​五、注意事项

1.确保你的开发环境和工具已经正确安装和配置。

2.如果没有打印,按一下复位键,检查连接和电源是否正确,注意根据你所用的硬件来接线,不要接错线。
3.在串口打印数据时,要确保波特率等参数与串口助手设置一致。

通过上述步骤,讲解STM32F103 HAL库USART串口DMA IDLE中断.坑在HAL_UARTEx_ReceiveToIdle_DMA()的DMA的半满中断-UART_DMARxHalfCplt.仅供参考,有任何问题,欢迎在评论区留言讨论!

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

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

相关文章

sql注入中information_schema被过滤的问题

目录 一、information_schema库的作用 二、获得表名 2.1 sys.schema_auto_increment_columns 2.2 schema_table_statistics 三、获得列名 join … using … order by盲注 子查询 在进行sql注入时&#xff0c;我们经常会使用information_schema来进行爆数据库名、表名、…

Jenkins 给任务分配 节点(Node)、设置工作空间目录

Jenkins 给任务分配 节点(Node)、设置工作空间目录 创建 Freestyle project 类型 任务 任务配置 Node 打开任务-> Configure-> General 勾选 Restrict where this project can be run Label Expression 填写一个 Node 的 Label&#xff0c;输入有效的 Label名字&#x…

Electron:使用electron-react-boilerplate创建一个react + electron的项目

使用 electron-react-boilerplate git clone --depth 1 --branch main https://github.com/electron-react-boilerplate/electron-react-boilerplate.git your-project-name cd your-project-name npm install npm start 安装不成功 在根目录加上 .npmrc文件 内容为 electron_…

数控机床设备分布式健康监测与智能维护系统MTAgent

数控机床设备分布式健康监测与智能维护系统MTAgent-v1.1融合了目前各种先进的信号处理以及信息分析算法以算法工具箱的方式&#xff0c;采用了一种开发的、模块化的结构实现信号各种分析处理&#xff0c;采用Python编程语言&#xff0c;满足不同平台需求(包括Windows、Linux)。…

FPGA VIVADO:axi-lite 从机和主机

FPGA VIVADO:axi-lite 从机和主机 TOC在这里插入代码片 前言 协议就不详细讲解了&#xff0c;直接看手册即可。下面主要如何写代码和关键的时序。 此外下面的代码可以直接用于实际工程 一、AXI-LITE 主机 数据转axi lite接口&#xff1a; 读/写数据FIFO缓存 仲裁&#xff1a…

1. 对比 LVS 负载均衡群集的 NAT 模式和 DR 模式,比较其各自的优势 。2. 基于 openEuler 构建 LVS-DR 群集。

DR 模式 * 负载各节点服务器通过本地网络连接&#xff0c;不需要建立专用的IP隧道 原理&#xff1a;首先负载均衡器接收到客户的请求数据包时&#xff0c;根据调度算法决定将请求发送给哪个后端的真实服务器&#xff08;RS&#xff09;。然后负载均衡器就把客户端发送的请求数…

ollama server启动服务后如何停止

要停止 Ollama 服务器服务&#xff0c;取决于如何启动该服务的。以下是几种常见的启动方法和相应的停止服务的步骤&#xff1a; 1. 直接在命令行中启动 如果是在命令行中直接启动 Ollama 服务器的&#xff0c;例如使用以下命令&#xff1a; ollama serve 可以通过以下方式停…

【设计模式】03-理解常见设计模式-行为型模式(专栏完结)

前言 前面我们介绍完创建型模式和创建型模式&#xff0c;这篇介绍最后的行为型模式&#xff0c;也是【设计模式】专栏的最后一篇。 一、概述 行为型模式主要用于处理对象之间的交互和职责分配&#xff0c;以实现更灵活的行为和更好的协作。 二、常见的行为型模式 1、观察者模…

mapbox基础,使用geojson加载line线图层,实现纯色填充、图片填充、虚线和渐变效果

👨‍⚕️ 主页: gis分享者 👨‍⚕️ 感谢各位大佬 点赞👍 收藏⭐ 留言📝 加关注✅! 👨‍⚕️ 收录于专栏:mapbox 从入门到精通 文章目录 一、🍀前言1.1 ☘️mapboxgl.Map 地图对象1.2 ☘️mapboxgl.Map style属性1.3 ☘️line线图层样式二、🍀使用geojson加载…

深入浅出:CUDA是什么,如何利用它进行高效并行计算

在当今这个数据驱动的时代&#xff0c;计算能力的需求日益增加&#xff0c;特别是在深度学习、科学计算和图像处理等领域。为了满足这些需求&#xff0c;NVIDIA推出了CUDA&#xff08;Compute Unified Device Architecture&#xff09;&#xff0c;这是一种并行计算平台和编程模…

LNMP+Zabbix安装部署(Zabbix6.0 Lnmp+Zabbix Installation and Deployment)

LNMPZabbix安装部署&#xff08;Zabbix6.0&#xff09; 简介 LNMP&#xff08;Linux Nginx MySQL PHP&#xff09;是一种流行的Web服务器架构&#xff0c;广泛用于搭建高性能的网站和应用程序。Zabbix 是一个开源的监控软件&#xff0c;可以用来监控网络、服务器和应用程序…

Docker 部署 Dify:轻松集成 Ollama 和 DeepSeek

1 Ollama的安装及使用 1.1 什么是Ollama&#xff1f; Ollama 是一个用于本地部署和运行大型语言模型的框架。 Ollama 的作用包括&#xff1a; 本地模型运行&#xff1a;Ollama 允许在本地机器上运行大型语言模型&#xff08;如 LLaMA、DeepSeek 等&#xff09;&#xff0c;无…

C++笔记之标准库中用于处理迭代器的`std::advance`和`std::distance`

C++笔记之标准库中用于处理迭代器的std::advance和std::distance code review! 文章目录 C++笔记之标准库中用于处理迭代器的`std::advance`和`std::distance`一.`std::advance`函数原型参数说明使用场景示例代码示例 1:移动 `std::vector` 的随机访问迭代器示例 2:移动 `st…

工业制造能耗管理新突破,漫途MTIC-ECM平台助力企业绿色转型!

在工业制造领域&#xff0c;能源消耗一直是企业运营成本的重要组成部分。随着“双碳”目标的推进&#xff0c;如何实现高效能耗管理&#xff0c;成为制造企业亟待解决的问题。漫途MTIC-ECM能源能耗在线监测平台&#xff0c;结合其自研的硬件产品&#xff0c;为工业制造企业提供…

C语言——深入理解指针(2)(数组与指针)

文章目录 数组名的理解使用指针访问数组一维数组传参的本质冒泡排序二级指针指针数组指针数组模拟二维数组 数组名的理解 之前我们在使用指针访问数组内容时&#xff0c;有这样的代码&#xff1a; int arr[10]{1,2,3,4,5,6,7,8,9,10}; int* p&arr[0];这里我们使用&ar…

在Windows系统中安装Open WebUI并连接Ollama

Open WebUI是一个开源的大语言模型&#xff08;LLM&#xff09;交互界面&#xff0c;支持本地部署与离线运行。通过它&#xff0c;用户可以在类似ChatGPT的网页界面中&#xff0c;直接操作本地运行的Ollama等大语言模型工具。 安装前的核心要求&#xff1a; Python 3.11&#…

Day4:强化学习之Qlearning走迷宫

一、迷宫游戏 1.环境已知 迷宫环境是定义好的&#xff0c;障碍物位置和空位置是已知的&#xff1b; # 定义迷宫 grid [[0, 0, 0, 1, 0],[0, 1, 0, 1, 0],[0, 1, 0, 0, 0],[0, 0, 0, 1, 0],[0, 1, 1, 1, 0] ] 2.奖励方式已知 如果碰到障碍物则得-1&#xff0c;如果到终点则…

家里WiFi信号穿墙后信号太差怎么处理?

一、首先在调制解调器&#xff08;俗称&#xff1a;猫&#xff09;测试网速&#xff0c;网速达不到联系运营商&#xff1b; 二、网线影响不大&#xff0c;5类网线跑500M完全没问题&#xff1b; 三、可以在卧室增加辅助路由器&#xff08;例如小米AX系列&#xff09;90~200元区…

视点开场动画实现(九)

这个相对比较简单&#xff1a; void COSGObject::FlyTo(double lon, double lat, double hei) {theApp.bNeedModify TRUE;while(!theApp.bCanModify)Sleep(1);em->setViewpoint(osgEarth::Viewpoint("0",lon, lat, 0, 0, -45, hei), 2);theApp.bNeedModify FAL…

保姆级GitHub大文件(100mb-2gb)上传教程

GLF&#xff08;Git Large File Storage&#xff09;安装使用 使用GitHub desktop上传大于100mb的文件时报错 The following files are over 100MB. lf you commit these files, you will no longer beable to push this repository to GitHub.com.term.rarWe recommend you a…