当前位置: 首页 > news >正文

【KEIL5】HardFault问题DEBUG排查方式

【KEIL5】HardFault问题DEBUG排查方式


文章目录

    • @[TOC](文章目录)
  • 前言
  • 一、Cortex-M3 权威指南——寄存器组概念
    • 1.通用寄存器
    • 2.堆栈指针——重要
    • 3.连接寄存器
    • 4.程序计数寄存器
    • 5.特殊功能寄存器
  • 二、内存区域概念
    • 1.内存区域解析
    • 2.代码例程
  • 三、实验
    • 1.复现BUG
    • 2.查找MSP
    • 3.定位代码段
    • 4.排查原因
  • 三、参考文章
  • 总结

实验工具:
1.MCU
2.DEBUG工具
3.KEIL5

前言


提示:以下是本篇文章正文内容,下面案例可供参考

一、Cortex-M3 权威指南——寄存器组概念

在这里插入图片描述

1.通用寄存器

在这里插入图片描述

2.堆栈指针——重要

在这里插入图片描述
在这里插入图片描述

在这里插入图片描述
在这里插入图片描述

3.连接寄存器

在这里插入图片描述

4.程序计数寄存器

在这里插入图片描述

5.特殊功能寄存器

在这里插入图片描述
在这里插入图片描述

二、内存区域概念

1.内存区域解析

栈区(Stack)
local_var(局部变量)、param(函数参数)
特点:自动分配/释放,生命周期与函数执行周期一致。
堆区(Heap)
heap_ptr 指向的内存(通过 malloc 分配)
特点:需手动 malloc/free,生命周期由程序员控制。
全局区(静态区)
global_var(已初始化全局变量)、static_var(静态变量)、uninitialized_global(未初始化全局变量)
特点:程序启动时分配,结束时释放,静态变量仅初始化一次。
文字常量区
“Hello, World!”(字符串常量)
特点:存储常量字符串,生命周期贯穿整个程序运行期。
代码区
example_function 和 main 的二进制代码
特点:存储函数指令,只读且不可修改。

2.代码例程

#include <stdio.h>
#include <stdlib.h>

// 全局区(静态区)
int global_var = 10;          // 已初始化的全局变量(数据段)
int uninitialized_global;     // 未初始化的全局变量(BSS段)

// 文字常量区
const char* str_literal = "Hello, World!"; // 字符串常量存储在文字常量区

// 代码区
void example_function(int param) {
    // 栈区:局部变量和函数参数
    int local_var = 20;       // 局部变量存储在栈区
    static int static_var = 30; // 静态变量存储在全局区(静态区)
    
    // 堆区:动态分配内存
    int* heap_ptr = (int*)malloc(sizeof(int)); // 通过malloc分配堆内存
    *heap_ptr = 40;
    
    printf("Stack param: %d\n", param);      // 访问栈区的函数参数
    printf("Stack local: %d\n", local_var);  // 访问栈区的局部变量
    printf("Static var: %d\n", static_var);  // 访问全局区的静态变量
    printf("Heap var: %d\n", *heap_ptr);     // 访问堆区的动态内存
    printf("Global var: %d\n", global_var);  // 访问全局区的全局变量
    printf("String literal: %s\n", str_literal); // 访问文字常量区的字符串
    
    free(heap_ptr); // 释放堆内存(需手动管理)
}

int main() {
    example_function(5); // 传递栈参数
    return 0;
}

三、实验

1.复现BUG

在进入HardFault中断处,打断点
在这里插入图片描述

2.查找MSP

对应左侧寄存器组,找到堆栈指针寄存器中的MSP和PSP,如果是 0XFFFFFFF9,那么中断前使用的是 MSP,如果是 0XFFFFFFFD,那么中断前使用的是 PSP
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

3.定位代码段

在这里插入图片描述

4.排查原因

1. 内存访问错误
未初始化指针:访问未分配或已释放的内存(野指针)。
数组越界:操作超出数组边界。
栈溢出:局部变量过多、递归过深或中断嵌套过深。
内存对齐问题:某些架构要求数据按特定对齐方式访问(如STM32的LDRD/STRD指令)。
2. 未定义指令或非法操作
执行非法指令:如尝试执行未被处理器支持的指令。
错误的函数指针:调用未初始化的函数指针或错误地址。
中断向量表损坏:中断服务例程(ISR)地址被意外修改。
3. 外设配置错误
未初始化外设:直接操作未使能的外设寄存器。
寄存器误配置:写入保留位或非法值到外设寄存器。
DMA配置冲突:源/目标地址冲突或DMA通道未正确配置。
4. 中断相关问题
中断优先级冲突:高优先级中断被低优先级中断打断。
中断服务例程(ISR)错误:在ISR中调用阻塞函数或访问非法内存。
未清除中断标志:导致中断重复触发。
5. 硬件故障
电源不稳定:电压波动或供电不足。
时钟配置错误:PLL未锁定或时钟源配置错误。
硬件损坏:RAM/Flash故障或总线错误。
6. 软件逻辑错误
多线程竞争:未加锁的共享资源访问(如RTOS任务间)。
死循环或死锁:导致看门狗超时(若启用)。
库函数误用:如错误使用标准库或HAL库API。

三、参考文章

ARM嵌入式软件开发 - 寄存器组 (SP, LR, PC)[0x04]
【STM32】HardFault问题详细分析及调试笔记
STM32单片机编程调试常见问题(一) HardFault_Handler故障分析与解决

总结

本文仅仅简单介绍了【KEIL5】HardFault问题DEBUG排查方式,评论区欢迎讨论。

相关文章:

  • 爱普生FC1610AN5G手机中替代传统晶振的理想之选
  • Hyperlane 文件分块上传服务端
  • 解决java使用easyexcel填充模版后,高度不一致问题
  • 【人工智能】DeepSeek 的上下文窗口扩展:解锁长文本处理的理论与实践
  • 【力扣hot100题】(088)最长有效括号
  • VS Code 的 .S 汇编文件里面的注释不显示绿色
  • 在spark中,窄依赖算子map和filter会组合为一个stage,这种情况下,map和filter是在一个task内进行的吗?
  • 玄机靶场-webshell查杀WP
  • viewmodel协程中执行耗时操作,导致viewmodel创建两次,导致observer失效
  • Linux 网络基础知识总结
  • 供应S620 支持 PD 的多协议双向快充移动电源解决方案
  • 保护PCBA的不同方法:喷三防漆 vs 镀膜
  • Ajax------免刷新地前后端交互
  • 力扣DAY46-50 | 热100 | 二叉树:展开为链表、pre+inorder构建、路径总和、最近公共祖先、最大路径和
  • 英伟达开源253B语言模型:Llama-3.1-Nemotron-Ultra-253B-v1 模型情况
  • #Hash 模式 vs History 模式
  • MCP基础学习四:MCP在AI应用中的集成(MCP在AI应用中的完整架构图)
  • 备赛蓝桥杯-Python-考前突击
  • EPLAN许可证更新教程
  • FTPClient开发遇到的坑
  • 做视频网站教程/武汉千锋教育培训机构怎么样
  • 房产门户网站建设/推广app赚佣金平台
  • 什么网站做家具外贸好呢/免费的个人网页
  • 套用别人产品图片做网站/企业网站seo多少钱
  • 尤溪网站开发/关键词排名软件官网
  • 天津网站建设工作室/google应用商店