找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 42060|回复: 2
打印 上一主题 下一主题
收起左侧

STM32 硬件错误HardFault_Handler的真凶

[复制链接]
跳转到指定楼层
楼主
ID:94349 发表于 2015-11-9 14:54 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
STM32出现HardFault_Handler故障的原因主要有两个方面:

1、内存溢出或者访问越界。
2、堆栈溢出。
最近遇到的问题是栈溢出,情况是这样的,举例说明:

static char data[10000];
void fun1(unsigned char *buf)
{
int  i=0;
for(i=0; i<5000; i++)
{
data = buf;
}
}

void fun2(void)
{
unsigned char buf[5000];
.........;
fun1(buf); //执行完毕此函数出现硬件错误HardFault_Handler
printf("data: %s\r\n",buf);
}

int main()
{
.........();
.........();
.........();
fun2();
.........();
.........();
.........();
while();
}

问题分析,通过断点代码跟踪,在进入fun1(buf);函数时,发现SP指向了数组data所开辟的空间,同时PC、等寄存器值压入栈,在循环执行data =buf;的时候修改了压入栈的数据,导致在退出函数fun1(buf);时PC指向了错误的位置。
问题:为什么SP会指向数组data所开辟的空间?原因是发生了栈溢出。
问题:那里导致了堆栈溢出呢? 下面我们看下面的网络资料,认识一下堆栈。

**************************************************************************************************
int main()
{
while(1);
}
BUILD://Program Size: Code=340 RO-data=252 RW-data=0ZI-data=1632  
编译后,就会发现这么个程序已用了1600多的RAM,这1600多的RAM跑哪儿去了,分析map,你会发现是堆和栈占用的
在startup_stm32f10x_md.s文件中,它的前面几行就有以上定义,这下该明白了吧。
Stack_Size     EQU    0x00000400
Heap_Size      EQU    0x00000200

理解堆和栈的区别

(1)栈区(stack):由编译器自动分配和释放,存放函数的参数值、局部变量的值等,其操作方式类似
     于数据结构中的栈。
(2)堆区(heap):一般由程序员分配和释放,若程序员不释放,程序结束时可能由操作系统回收。分配
     方式类似于数据结构中的链表。
(3)全局区(静态区)(static):全局变量和静态变量的存储是放在一块的,初始化的全局变量和静态
     变量在一块区域,未初始化的全局变量和未初始化的静态变量在相邻的另一块区域。程序结束后由系
     统自动释放。
(4)文字常量区:常量字符串就是存放在这里的。
(5)程序代码区:存放函数体的二进制代码。

例如:
    int a=0;                    //全局初始化区
    char *p1;                   //全局未初始化区
    main()
    {
     int b;                    //栈
     char s[]="abc";              //栈
     char *p3= "1234567";          //在文字常量区
     static int c =0 ;            //静态初始化区
     p1= (char *)malloc(10);         //堆区
     strcpy(p1,"123456");          //"123456"放在常量区
    }
所以堆和栈的区别:
    stack的空间由操作系统自动分配/释放,heap上的空间手动分配/释放。
    stack的空间有限,heap是很大的自由存储区。
    程序在编译期和函数分配内存都是在栈上进行,且程序运行中函数调用时参数的传递也是在栈上进行。
**************************************************************************************************
明白堆栈的分配原理后,我们也就明白了为什么说是栈溢出了,而没有说是堆栈溢出或者堆溢出,我们接下来再来分析什么导致了栈溢出,这会不难发现真凶是unsignedcharbuf[5000];,buf的开辟占用了很大的栈空间,超出了startup_stm32f10x_md.s文件中定义的空间大小,导致了栈的溢出。
问题总结:
1、函数内部变量占用空间较大时,定义为全局变量或者静态变量,减少堆栈的占用。
2、多使用指针解决数据的复制,同时减少内存的占用。
分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏1 分享淘帖 顶1 踩
回复

使用道具 举报

沙发
ID:103743 发表于 2018-9-8 13:29 | 只看该作者
赞,懂了,我也遇到了这种情况,
回复

使用道具 举报

板凳
ID:487048 发表于 2020-11-2 19:04 | 只看该作者
我的也是把局部变量改为全局变量就OK啦。真是一个容易忽略的问题。
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表