STM32 串口 (DMA + 空闲中断 + 环形缓冲区)

STM32 串口 (DMA + 空闲中断 + 环形缓冲区)


1. 基本概念

  • UART 空闲中断(IDLE)

    • 当串口 RX 线上 连续一段时间没有数据接收,USART 外设触发 空闲中断
    • 空闲中断的主要作用是通知数据传输完成或当前帧结束。
  • DMA 接收模式

    • DMA(Direct Memory Access) 自动将串口接收到的数据存储到指定缓冲区。
    • CPU 不再需要逐字节处理接收数据,提高效率。
    • HAL_UARTEx_ReceiveToIdle_DMA:启动 DMA 接收,支持接收数据直到触发 空闲中断
  • 环形缓冲区

    • 通过固定大小的缓冲区 + 读写指针 实现数据的循环存储。
    • 用于连续接收数据,解决 DMA 数据处理问题。
    • 读写指针逻辑:
      • 写指针:指向新接收数据的位置。
      • 读指针:指向待处理数据的位置。

2. 流程概述

  1. 初始化 UART 和 DMA

    • 配置 UART 和 DMA。
    • 启用 DMA 接收并启动 HAL_UARTEx_ReceiveToIdle_DMA
  2. 串口接收数据

    • 数据通过 DMA 存储到 DMA 缓冲区 uart_rx_dma_buffer
    • 串口数据未停止时,DMA 自动接收,CPU 不参与。
  3. 触发空闲中断

    • 当 RX 线上 超过一个字节时间没有数据接收,触发 空闲中断(IDLE)。
    • 调用 USARTx_IRQHandler
  4. 中断处理

    • USARTx_IRQHandler 中调用 HAL_UART_IRQHandler
    • HAL 库检测到 IDLE 中断,触发回调函数 HAL_UARTEx_RxEventCallback
  5. 回调函数处理接收数据

    • HAL_UARTEx_RxEventCallback 中:
      • 计算接收到的数据长度。
      • 将 DMA 缓冲区的数据拷贝到 环形缓冲区
      • 清空 DMA 缓冲区,准备下一次接收。
    • 重新启动 DMA 接收 HAL_UARTEx_ReceiveToIdle_DMA
  6. 主循环读取数据

    • 通过环形缓冲区的 读写指针 提取接收到的数据,进行处理。

3. 代码展示

初始化 UART 和 DMA
void MX_USART1_UART_Init(void) {huart1.Instance = USART1;huart1.Init.BaudRate = 115200;huart1.Init.WordLength = UART_WORDLENGTH_8B;huart1.Init.StopBits = UART_STOPBITS_1;huart1.Init.Parity = UART_PARITY_NONE;huart1.Init.Mode = UART_MODE_TX_RX;HAL_UART_Init(&huart1);// 启动 DMA 接收HAL_UARTEx_ReceiveToIdle_DMA(&huart1, uart_rx_dma_buffer, sizeof(uart_rx_dma_buffer));__HAL_DMA_DISABLE_IT(&hdma_usart1_rx, DMA_IT_HT);  // 禁用半满中断
}

中断服务函数
void USART1_IRQHandler(void) {HAL_UART_IRQHandler(&huart1);  // 处理 UART 中断
/*不同的触发,跳转不同的回调函数,若是空闲中断触发,跳转的是 *`HAL_UARTEx_RxEventCallback`*//* 调用HAL库的串口中断处理函数 */HAL_UARTEx_ReceiveToIdle_DMA(&huart1, uart_rx_dma_buffer, sizeof(uart_rx_dma_buffer));/* 重新启动DMA接收,确保下一次接收正常进行 */ __HAL_DMA_DISABLE_IT(&hdma_usart1_rx, DMA_IT_HT);
}

回调函数:处理空闲中断
void HAL_UARTEx_RxEventCallback(UART_HandleTypeDef *huart, uint16_t Size) {if (huart->Instance == USART1) {if (!ringbuffer_is_full(&usart_rb)) {ringbuffer_write(&usart_rb, uart_rx_dma_buffer, Size);  // 将数据写入环形缓冲区}memset(uart_rx_dma_buffer, 0, sizeof(uart_rx_dma_buffer));  // 清空 DMA 缓冲区}
}

4. 环形缓冲区实现

环形缓冲区结构
  • ringbuffer_t
    

    结构体用于管理缓冲区:

    • buffer:实际存储数据的数组。
    • r:读指针。
    • w:写指针。
    • itemCount:当前缓冲区内的数据量。
typedef struct {uint32_t w;                        // 写指针uint32_t r;                        // 读指针uint8_t buffer[RINGBUFFER_SIZE];   // 数据存储缓冲区uint32_t itemCount;                // 当前缓冲区数据量
} ringbuffer_t;

函数功能
  1. 初始化环形缓冲区

    • 清零缓冲区、读写指针和数据量。
    void ringbuffer_init(ringbuffer_t *rb);
    
  2. 写入数据

    • 检查缓冲区是否已满,未满时将数据写入。
    int8_t ringbuffer_write(ringbuffer_t *rb, uint8_t *data, uint32_t num);
    
  3. 读取数据

    • 检查缓冲区是否为空,非空时读取数据。
    int8_t ringbuffer_read(ringbuffer_t *rb, uint8_t *data, uint32_t num);
    
  4. 缓冲区状态检查

    • ringbuffer_is_full:检查缓冲区是否已满。
    • ringbuffer_is_empty:检查缓冲区是否为空。

5. 工作顺序总结

  1. HAL_UARTEx_ReceiveToIdle_DMA 启动 DMA 接收。
  2. UART 接收到数据,DMA 将数据存入缓冲区。
  3. 空闲中断(IDLE)触发,调用 USART1_IRQHandler
  4. HAL_UART_IRQHandler 检测到空闲中断,自动调用 HAL_UARTEx_RxEventCallback
  5. 在回调函数中
    • 处理接收到的数据(写入环形缓冲区)。
    • 重新启动 DMA 接收(调用 HAL_UARTEx_ReceiveToIdle_DMA)。
  6. 主循环处理数据:调用 uart_proc,从环形缓冲区中提取数据并处理

6. 图文总结

UART RX --> DMA 接收数据 --> RX 空闲中断 --> USART1_IRQHandler --> HAL_UARTEx_RxEventCallback|                                     ||---- 重新启动 DMA 接收 ----------------|
数据 --> 写入环形缓冲区 --> 主循环读取数据 --> 用户处理

本系统实现了 STM32 串口 DMA 接收 + 空闲中断 + 环形缓冲区,旨在高效接收和处理串口不定长数据,保证数据的完整性与实时性。其核心工作流程如下:

  1. DMA 接收:通过调用 HAL_UARTEx_ReceiveToIdle_DMA 函数启动 DMA 模式接收,将接收到的数据自动存储到 DMA 缓冲区,减少 CPU 干预,提高效率。
  2. 空闲中断触发:当串口 RX 线超过一个字节时间没有新数据输入时,触发 空闲中断(IDLE),并在中断中调用 HAL_UART_IRQHandler,进一步触发 HAL_UARTEx_RxEventCallback 回调函数。
  3. 数据存储
    在回调函数 HAL_UARTEx_RxEventCallback 中,将 DMA 缓冲区接收到的数据写入 环形缓冲区,通过 ringbuffer_write 函数实现数据的安全存储,防止数据丢失。随后清空 DMA 缓冲区,重新启动 DMA 接收,确保数据连续接收。
  4. 数据处理
    在主循环中,调用 uart_proc 函数,通过 ringbuffer_read 从环形缓冲区读取数据进行处理。环形缓冲区通过读写指针和数据计数机制实现数据的循环存储与读取,适用于不定长、连续数据接收的场景。
  5. 环形缓冲区管理
    • ringbuffer_is_fullringbuffer_is_empty 用于判断缓冲区状态。
    • ringbuffer_writeringbuffer_read 分别实现数据的写入与读取,确保缓冲区数据有序管理,防止数据丢失。

系统特点

  • 高效性:DMA 自动接收数据,减少 CPU 开销。
  • 实时性:通过 UART 空闲中断实时捕获数据接收完成状态。
  • 可靠性:使用环形缓冲区管理数据,确保数据存储稳定,避免数据丢失。
  • 适用性:适合高频率、不定长数据的串口通信场景。

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

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

相关文章

股指期货是什么?股指期货日内拐点有什么特征?

股指期货是一种金融衍生品,股指期货日内拐点就是在一天交易过程中,市场走势发生显著改变的那个点。 股指期货是什么? 股指是一个指数,比如上证50指数、沪深300指数、中证500指数以及中证1000指数,这是一堆股票的一个整…

Opensearch/ElasticSearch-ctx查询内容不全的问题

问题 在OpenSearch中,我希望在action中把一整条log作为报警内容发送出来,但是根据文档,配置为ctx.results.0.hits.hits.0._source.log,但是发现根本找不到这个值 经过查询,我发现在返回的ctx中仅存在如下的值 resul…

vue2和vue3插槽slot最通俗易懂的区别理解

在 Vue 的组件通信中,slot(插槽)的编译优化是一个重要的性能提升点。以下是 Vue2 和 Vue3 在 slot 处理上的差异及优化原理,用更直观的方式解释: Vue2 的 Slot 更新机制 想象一个父子组件场景: 父组件&am…

【16届蓝桥杯寒假刷题营】第1期DAY4

1.披萨和西蓝花 - 蓝桥云课 1. 披萨和西蓝花 问题描述 在接下来的 N 天里(编号从 1 到 N),坤坤计划烹饪披萨或西兰花。他写下一个长度为 N 的字符串 A,对于每个有效的 i,如果字符 Ai 是 1,那么他将在第 i…

你需要了解的远程登录协议——Telnet

你需要了解的远程登录协议——Telnet 一. 什么是Telnet?二. Telnet的优缺点三. Telnet vs SSH:哪一个更适合?四. Telnet的应用场景 前言 点个免费的赞和关注,有错误的地方请指出,看个人主页有惊喜。 作者:神…

本地部署【LLM-deepseek】大模型 ollama+deepseek/conda(python)+openwebui/docker+openwebui

通过ollama本地部署deepseek 总共两步 1.模型部署 2.[web页面] 参考官网 ollama:模型部署 https://ollama.com/ open-webui:web页面 https://github.com/open-webui/open-webui 设备参考 Mac M 芯片 windows未知 蒸馏模型版本:deepseek-r1:14b 运行情况macminim2 24256 本地…

PHP在线题库小程序

📚 在线题库小程序:学习提分新神器,轻松跃升学霸行列 这是一款专为追梦学子精心策划、基于ThinkPHPUniApp框架匠心打造的在线题库类微信小程序系统。它宛如一把✨智慧钥匙✨,为追求高效学习的你解锁🔓知识宝库的大门。…

Java开发中的连接池技术介绍

连接池技术是Java开发中用于管理数据库连接的重要技术,尤其在SSM(Spring、Spring MVC、MyBatis)架构中,连接池能够显著提升数据库操作的性能和资源利用率。下面我们将详细介绍连接池技术解决的问题、配置方案以及代码实现。 1. 连…

Unity-Mirror网络框架-从入门到精通之Pong示例

文章目录 前言示例介绍NetworkManagerPongBallPlayer总结前言 在现代游戏开发中,网络功能日益成为提升游戏体验的关键组成部分。本系列文章将为读者提供对Mirror网络框架的深入了解,涵盖从基础到高级的多个主题。Mirror是一个用于Unity的开源网络框架,专为多人游戏开发设计…

布隆过滤器到底是什么东西?它有什么用

布隆过滤器:用概率换空间的奇妙数据结构 引言:当空间成为奢侈品 在互联网每天产生2.5万亿字节数据的时代,Google每秒处理超过9万次搜索请求,Redis缓存系统支撑着百万级QPS的访问。面对如此海量的数据处理需求,传统的…

任务1 将单表中的单个rfid增加为多个rfid

方案 使用连表查询解决 单独创建一个rfid的表 让tool_id对应多个rfid 需要优化的表 1:tool_materials_stock 库存管理 已完成 数据迁移完成 原库rfid字段未删除 2:tool_borrow_return 借出借还管理 已完成 3:too…

OutSystems Platform Tools Platform Services

概述(Overview) outsystems是一整套低代码的企业级应用(WEB 和 移动端)的开发环境。 本文主要讲解outsystems的Platform Tools与Platform Services 平台工具(Platform Tools) 集成开发环境IDE&#xff0…

【深度解析】ETERM指令:离港系统的核心技术

在民航离港系统中,ETERM(中航信终端模拟系统)是广泛使用的指令操作系统,主要用于航班控制、旅客值机、登机等操作。以下是一些核心的ETERM指令及其功能分类: 1. 航班信息查询与操作 FLR:显示航班列表&…

ES的java操作

ES的java操作 一、添加依赖 在pom文件中添加依赖包 <dependencies><dependency><groupId>org.elasticsearch</groupId><artifactId>elasticsearch</artifactId><version>7.8.0</version></dependency><!-- elastic…

DeepSeek 从入门到精通学习指南,2025清华大学《DeepSeek从入门到精通》正式发布104页pdf版超全解析

DeepSeek 是一款强大的 AI 搜索引擎&#xff0c;广泛应用于企业级数据检索和分析。无论您是初学者还是有经验的用户&#xff0c;掌握 DeepSeek 的使用都能为您的工作带来极大的便利。本文将从入门到精通&#xff0c;详细介绍如何学习和使用 DeepSeek。 链接: https://pan.baid…

飞书专栏-TEE文档

CSDN学院课程连接&#xff1a;https://edu.csdn.net/course/detail/39573

2025.2.11——一、[极客大挑战 2019]PHP wakeup绕过|备份文件|代码审计

题目来源&#xff1a;BUUCTF [极客大挑战 2019]PHP 目录 一、打开靶机&#xff0c;整理信息 二、解题思路 step 1&#xff1a;目录扫描、爆破 step 2&#xff1a;代码审计 1.index.php 2.class.php 3.flag.php step 3&#xff1a;绕过__wakeup重置 ​编辑 三、小结…

AI大模型(DeepSeek)科研应用、论文写作、数据分析与AI绘图学习

【介绍】 在人工智能浪潮中&#xff0c;2024年12月中国公司研发的 DeepSeek 横空出世以惊艳全球的姿态&#xff0c;成为 AI领域不可忽视的力量!DeepSeek 完全开源&#xff0c;可本地部署&#xff0c;无使用限制&#xff0c;保护用户隐私。其次&#xff0c;其性能强大&#xff…

考研操作系统----操作系统的概念定义功能和目标(仅仅作为王道哔站课程讲义作用)

目录 操作系统的概念定义功能和目标 操作系统的四个特征 操作系统的分类 ​编辑 操作系统的运行机制 系统调用 操作系统体系结构 操作系统引导 虚拟机 操作系统的概念定义功能和目标 什么是操作系统&#xff1a; 操作系统是指控制和管理整个计算机系统的软硬件资源&…

DeepSeek 突然来袭,AI 大模型变革的危机与转机藏在哪?

随着人工智能技术的飞速发展&#xff0c;大模型领域不断涌现出具有创新性的成果。DeepSeek 的横空出世&#xff0c;为 AI 大模型领域带来了新的变革浪潮。本文将深入探讨 DeepSeek 出现后 AI 大模型面临的危机与转机。 冲冲冲&#xff01;&#xff01;&#xff01; 目录 一、…