【沁恒蓝牙MESH】CH582串口中断内存溢出导致MCU频繁重启

本文主要是介绍【沁恒蓝牙MESH】CH582串口中断内存溢出导致MCU频繁重启,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本文主要记录了【沁恒蓝牙mesh】CH582串口中断内存溢出导致MCU频繁重启
由于开发疏忽,导致的数组内存溢出,是入门嵌入式开发经常忽视的错误,用以记录,共勉!!

目录

  • 1. 遇到问题描述以及解决
    • 1.1 问题一:串口中断导致MCU频繁重启
      • 1.1.1【场景描述】
      • 1.1.2 【问题描述】
      • 1.1.3 【问题定位】
      • 1.1.4 【问题解决】

  • 💖 作者简介:大家好,我是喜欢记录零碎知识点的小菜鸟。😎
  • 📝 个人主页:欢迎访问我的 Ethernet_Comm 博客主页🔥
  • 🎉 支持我:点赞👍+收藏⭐️+留言📝
  • 📣 系列专栏:沁恒蓝牙mesh二次开发🍁
  • 💬格言:写文档啊不是写文章,重要的还是直白!🔥

1. 遇到问题描述以及解决

1.1 问题一:串口中断导致MCU频繁重启

1.1.1【场景描述】

硬件:沁恒的 CH582F 单片机, 海凌科 24G 雷达模组

引脚:海凌科雷达模组5个引脚和 CH582F 连,单片机PB4引脚(普通GPIO,下拉输入,采用IO查询的方式判断电平)

CH582F雷达模组
5vvcc
gndgnd
PB4out
RX3—PA4tx
RX3—PA5rx

1.1.2 【问题描述】

烧录程序后,发现 单片机一直 周期性的重启。。。。下图左侧为串口3的接受中断
在这里插入图片描述

1.1.3 【问题定位】

雷达:每间隔100ms 从TX引脚输出的数据为

[2023-12-01 07:58:27.167]# RECV HEX>
F4 F3 F2 F1 0D 00 02 AA 03 1E 00 5A 1E 00 64 00 00 55 00 F8 F7 F6 F5 
[2023-12-01 07:58:27.267]# RECV HEX>
F4 F3 F2 F1 0D 00 02 AA 03 1E 00 64 1E 00 64 00 00 55 00 F8 F7 F6 F5 
[2023-12-01 07:58:27.367]# RECV HEX>
F4 F3 F2 F1 0D 00 02 AA 03 1E 00 64 1E 00 64 00 00 55 00 F8 F7 F6 F5 
[2023-12-01 07:58:27.469]# RECV HEX>
F4 F3 F2 F1 0D 00 02 AA 03 1E 00 64 1E 00 64 00 00 55 00 F8 F7 F6 F5 
[2023-12-01 07:58:27.562]# RECV HEX>
F4 F3 F2 F1 0D 00 02 AA 03 1E 00 64 1E 00 64 00 00 55 00 F8 F7 F6 F5 

将雷达模组拿掉,程序就正常了,因此一个个引脚查问题,最终定位问题为:

软件中开启了串口3的接收中断,雷达模组频繁的发送数据导致单片机频繁进入中断,造成重启

检查软件中的中断咋写的,,,,,

1.1.4 【问题解决】

曾经在ESP32 开发中翻过这样一个错误:【esp32】解决以太网+mqtt堆栈溢出问题 报错 no mem for receive buffer,这个是由于动态申请的内存没有是释放导致栈空间不足,MCU反复重启

数组溢出导致的栈内存被冲刷掉,导致内存溢出而使得MCU重启

问题定位如下, 我的软件串口接收中断竟然是这么写的::::::令人发指!!!!

串口中断将接收到的数据存储到 uart_rx_buffer[]数组中, 但是,,我定义了一个 100 大小的数组,却 user_rx_buffer_write_index & user_rx_buffer_length_mask 去接收数据。。。。

本意是这样的 定义一个 1024 大小的数组,每接收到一个自己 user_rx_buffer_write_index 自加1,然后通过 user_rx_buffer_write_index & user_rx_buffer_length_mask 将范围限制在 1-1024 范围内,防止数组溢出。。。。当时应该是担心占用内存太大,手动将uart_rx_buffer[1024]改为了 uart_rx_buffer[100],但是忘记修改 user_rx_buffer_length_mask变量的值了。。

造成的结果就是,我定义了100维的数组,缺往里边存了 最大1024个字节,那剩下的900字节空间就把其他栈空间占用了,,冲掉了哪些变量的值???不得而知,但是造成的结果就是单片机不断重启了。。。。。。。。

解决方式:

将数组改为 1024 维的,或者数组100维不变,将掩码的值改为99,别造成内存溢出,就可以了,

实测修改后,就不会重启了

uint8_t uart_rx_buffer[100]={0}; 
uint16_t user_rx_buffer_length_mask = 1024-1;
uint16_t user_rx_buffer_write_index = 0;
uint8_t trigB = 7 ;
uint8_t uart_rx_done = 0;__INTERRUPT
__HIGH_CODE
void UART3_IRQHandler(void)
{uint16_t error;int i = 0;switch(UART3_GetITFlag()){case UART_II_LINE_STAT:  // 线路状态错误UART3_GetLinSTA();break;case UART_II_RECV_RDY: // 数据达到设置触发点while( R8_UART3_RFC > 1) {  // 这个方式必须保证uart的接收触发中断是大于1字节的uart_rx_buffer[user_rx_buffer_write_index & user_rx_buffer_length_mask] = R8_UART3_RBR;user_rx_buffer_write_index += 1;}break;case UART_II_RECV_TOUT: // 接收超时,暂时一帧数据接收完成for(i = 0; i < R8_UART3_RFC; i++){iRxBuff[i] = UART3_RecvByte();//UART3_SendByte(iRxBuff[i]);}  break;case UART_II_THR_EMPTY:break;case UART_II_MODEM_CHG:break;default:break;}
}

知识点:

当 b = 2^n 的时候:a % b = a & (b-1)

这篇关于【沁恒蓝牙MESH】CH582串口中断内存溢出导致MCU频繁重启的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/444225

相关文章

Django开发时如何避免频繁发送短信验证码(python图文代码)

《Django开发时如何避免频繁发送短信验证码(python图文代码)》Django开发时,为防止频繁发送验证码,后端需用Redis限制请求频率,结合管道技术提升效率,通过生产者消费者模式解耦业务逻辑... 目录避免频繁发送 验证码1. www.chinasem.cn避免频繁发送 验证码逻辑分析2. 避免频繁

MySQL 内存使用率常用分析语句

《MySQL内存使用率常用分析语句》用户整理了MySQL内存占用过高的分析方法,涵盖操作系统层确认及数据库层bufferpool、内存模块差值、线程状态、performance_schema性能数据... 目录一、 OS层二、 DB层1. 全局情况2. 内存占js用详情最近连续遇到mysql内存占用过高导致

最新Spring Security的基于内存用户认证方式

《最新SpringSecurity的基于内存用户认证方式》本文讲解SpringSecurity内存认证配置,适用于开发、测试等场景,通过代码创建用户及权限管理,支持密码加密,虽简单但不持久化,生产环... 目录1. 前言2. 因何选择内存认证?3. 基础配置实战❶ 创建Spring Security配置文件

java内存泄漏排查过程及解决

《java内存泄漏排查过程及解决》公司某服务内存持续增长,疑似内存泄漏,未触发OOM,排查方法包括检查JVM配置、分析GC执行状态、导出堆内存快照并用IDEAProfiler工具定位大对象及代码... 目录内存泄漏内存问题排查1.查看JVM内存配置2.分析gc是否正常执行3.导出 dump 各种工具分析4.

怎样通过分析GC日志来定位Java进程的内存问题

《怎样通过分析GC日志来定位Java进程的内存问题》:本文主要介绍怎样通过分析GC日志来定位Java进程的内存问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、GC 日志基础配置1. 启用详细 GC 日志2. 不同收集器的日志格式二、关键指标与分析维度1.

Java内存分配与JVM参数详解(推荐)

《Java内存分配与JVM参数详解(推荐)》本文详解JVM内存结构与参数调整,涵盖堆分代、元空间、GC选择及优化策略,帮助开发者提升性能、避免内存泄漏,本文给大家介绍Java内存分配与JVM参数详解,... 目录引言JVM内存结构JVM参数概述堆内存分配年轻代与老年代调整堆内存大小调整年轻代与老年代比例元空

linux重启命令有哪些? 7个实用的Linux系统重启命令汇总

《linux重启命令有哪些?7个实用的Linux系统重启命令汇总》Linux系统提供了多种重启命令,常用的包括shutdown-r、reboot、init6等,不同命令适用于不同场景,本文将详细... 在管理和维护 linux 服务器时,完成系统更新、故障排查或日常维护后,重启系统往往是必不可少的步骤。本文

C++高效内存池实现减少动态分配开销的解决方案

《C++高效内存池实现减少动态分配开销的解决方案》C++动态内存分配存在系统调用开销、碎片化和锁竞争等性能问题,内存池通过预分配、分块管理和缓存复用解决这些问题,下面就来了解一下... 目录一、C++内存分配的性能挑战二、内存池技术的核心原理三、主流内存池实现:TCMalloc与Jemalloc1. TCM

Redis过期删除机制与内存淘汰策略的解析指南

《Redis过期删除机制与内存淘汰策略的解析指南》在使用Redis构建缓存系统时,很多开发者只设置了EXPIRE但却忽略了背后Redis的过期删除机制与内存淘汰策略,下面小编就来和大家详细介绍一下... 目录1、简述2、Redis http://www.chinasem.cn的过期删除策略(Key Expir

MySQL版本问题导致项目无法启动问题的解决方案

《MySQL版本问题导致项目无法启动问题的解决方案》本文记录了一次因MySQL版本不一致导致项目启动失败的经历,详细解析了连接错误的原因,并提供了两种解决方案:调整连接字符串禁用SSL或统一MySQL... 目录本地项目启动报错报错原因:解决方案第一个:第二种:容器启动mysql的坑两种修改时区的方法:本地