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

STM32一个地址未对齐引起的 HardFault 异常

1. 概述

客户在使用 STM32G070 的时候,KEIL MDK 为编译工具,当编译优化选项设置为Level0 的时候,程序会出现 Hard Fault 异常,而当编译优化选项设置为 Level1 的时候,则程序运行正常。表面上看,这似乎是 KEIL MDK 的问题,通过分析,导致这个问题的本质原因是内存地址没有对齐引起的,下面章节将详细分析该问题的来龙去脉以及解决方法。

2. 问题描述与分析

根据客户的反馈,引起问题的代码很简单,客户定义了几个全局数组,在主程序中访问这几个数组就会出现 Hard Fault 异常,参考代码如下。

图1.导致异常的代码片段
在这里插入图片描述

把客户提供的代码片段移植到 NUCLEO-G070RB 开发板上,问题很容易就复现了,代码本身功能简单,写法上也没有错误,所以从代码片段本身上看,无法确定问题出在哪里,通过 KEIL 调试器,在汇编窗口单步调试下,最终发现导致 HardFault 异常的语句为下图所示语句。

图2.导致 Hard Fault 异常的语句
在这里插入图片描述

根据单步调试得知出现问题的语句为 LDR 指令,参考 Cortex M0 编程手册 PM0223 得知 LDR 指令的作用是从内存地址中加载一个 WORD 数据到目的寄存器 Rt 中,其中内存地址根据 Rn 或者 SP 寄存器的值以及立即数 imm 得到。

图3.LDR 指令描述
在这里插入图片描述
根据指令的描述,使用 LDR 指令的时候,通过 Rn 和 imm 计算得到的内存地址必须是读取字节数的倍数,LDR 每次读取一个 WORD,所以使用 LDR 指令时,内存地址必须 4字节对齐。如果地址没有对齐,则会导致 HardFault 异常。

结合 LDR 指令的描述,在调试状态下,通过查看寄存器值,图 2 出错语句中根据 Rn和 imm 计算得到的内存地址为 R0=0x2000000B,imm=4 所以内存地址为 0x2000000F,很显然这个地址不是 4 字节对齐的。

图4.内存地址不对齐
在这里插入图片描述
而当我们改变编译优化选项为 Level1 时,得到的内存地址为R0=0x20000000,imm=0x04 显然这个地址是按照 4 字节对齐的,所以这种情况下是不会出现 HardFault 异常的,印证了客户的问题现象。

图5.内存地址对齐
在这里插入图片描述

3. 问题解决

通过上一节的分析,明确了导致该问题的本质原因是内存地址没有对齐,这个内存地址实际上是代码中定义的全局变量 g_curPlaySound_app 指向的地址,也就是全局数组变量 SoundFile 的地址,在编译器不同的优化选项下,分配给 SoundFile 变量的地址是不一样的,在本案例中,编译优化选项 Level0 条件下,SoundFile 分配的地址没有按照WORD 对齐,而在优化选项 Level1 条件下,SoundFile 分配的地址是 WORD 对齐,所以在两种优化选项下,出现了不一样的运行结果。所以要保证程序不出错,当通过指针访问变量的时候,要确保指针指向的地址是 4 字节对齐的,在 Keil 环境下,可以通过__attribute__((aligned (4))) 关键字实现,如下图所示,通过该关键字,对齐了地址,也就不会出现 HardFault 异常了。

图6.确保地址对齐
在这里插入图片描述

4. 总结

地址未对齐是嵌入式系统中容易忽视的一个细节,忽视这点往往会导致一些奇怪的问题,所以在开发过程中,注意这些细节还是很有必要的。

参考文献

在这里插入图片描述
文档中所用到的工具及版本
Keil MDK V5.29


本文档参考ST官方的《【应用笔记】LAT1185+一个地址未对齐引起的+HardFault+异常》文档。

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • Golang | Leetcode Golang题解之第8题字符串转换整数atoi
  • 【游戏逆向】游戏全屏捡物的实现
  • 【运输层】网络数据报协议 UDP
  • 指针的深入理解(六)
  • Prime Ring Problem(UVA 524)
  • 基于Springboot+Vue实现前后端分离酒店管理系统
  • 常规的k8s的监控指标
  • 微信小程序 电影院售票选座票务系统5w7l6
  • fakebook-攻防世界
  • JVM字节码与类加载——字节码指令集与解析
  • Java | Leetcode Java题解之第13题罗马数字转整数
  • I2C协议介绍
  • Vue - 你知道Vue中computed和watch的区别吗
  • RabbitMQ系统监控、问题排查和性能优化实践
  • 大话设计模式——六大基本设计原则(SOLID原则)
  • 【vuex入门系列02】mutation接收单个参数和多个参数
  • - C#编程大幅提高OUTLOOK的邮件搜索能力!
  • gops —— Go 程序诊断分析工具
  • LeetCode算法系列_0891_子序列宽度之和
  • Lucene解析 - 基本概念
  • miaov-React 最佳入门
  • Objective-C 中关联引用的概念
  • Python_网络编程
  • 阿里云爬虫风险管理产品商业化,为云端流量保驾护航
  • 得到一个数组中任意X个元素的所有组合 即C(n,m)
  • 记一次和乔布斯合作最难忘的经历
  • 使用 @font-face
  • 手写一个CommonJS打包工具(一)
  • 微信小程序上拉加载:onReachBottom详解+设置触发距离
  • 云大使推广中的常见热门问题
  • 字符串匹配基础上
  • 《码出高效》学习笔记与书中错误记录
  • 仓管云——企业云erp功能有哪些?
  • ​如何防止网络攻击?
  • #1014 : Trie树
  • #70结构体案例1(导师,学生,成绩)
  • #Js篇:单线程模式同步任务异步任务任务队列事件循环setTimeout() setInterval()
  • #免费 苹果M系芯片Macbook电脑MacOS使用Bash脚本写入(读写)NTFS硬盘教程
  • #数学建模# 线性规划问题的Matlab求解
  • #我与Java虚拟机的故事#连载17:我的Java技术水平有了一个本质的提升
  • $HTTP_POST_VARS['']和$_POST['']的区别
  • (~_~)
  • (51单片机)第五章-A/D和D/A工作原理-A/D
  • (C++)八皇后问题
  • (Matalb时序预测)PSO-BP粒子群算法优化BP神经网络的多维时序回归预测
  • (PWM呼吸灯)合泰开发板HT66F2390-----点灯大师
  • (二十六)Java 数据结构
  • (附源码)ssm失物招领系统 毕业设计 182317
  • (附源码)小程序儿童艺术培训机构教育管理小程序 毕业设计 201740
  • (三十五)大数据实战——Superset可视化平台搭建
  • (十三)Flink SQL
  • (学习日记)2024.04.04:UCOSIII第三十二节:计数信号量实验
  • (一)Dubbo快速入门、介绍、使用
  • (转)EOS中账户、钱包和密钥的关系
  • (转)Java socket中关闭IO流后,发生什么事?(以关闭输出流为例) .