历史上的今天
今天是:2025年03月24日(星期一)
2019年03月24日 | STM32 硬件错误HardFault_Handler的真凶
2019-03-24 来源:eefocus
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、多使用指针解决数据的复制,同时减少内存的占用。
史海拾趣
|
本人做过技术开发工作多年,从焊电路板的小工程师逐渐做到项目经理、研发经理,现在做到总工程师,作为工程师有亲身的感受,作为研发主管,对工程师的性格、心理和知识结构有非常深入的了解,现在把自己的一点感悟与大家分享,希望大家批评指正,欢 ...… 查看全部问答> |
|
从70年代末的简单控制发展到今天的高端应用,嵌入式系统已经变成一个复杂的高技术系统,要在短时间开发出所需功能的难度大大提升,但是市场竞争又要求产品能够快速面市同时必须确保产品的质量和性能,这里面工具就起着很重要的作用。这其中,对工具 ...… 查看全部问答> |
|
EXTERN XBYTE gxbUartBitNum _AT_ (0x990F);/////当前接受数据的位数 EXTERN XBYTE gxbUart2s _AT_ (0x991F); //ie. SBUF2 EXTERN XBYTE gxbUart2f ...… 查看全部问答> |
|
BeagleBone 硬件性能测试(1) ---- 我是出来打酱油的 今天惊问BeagleBone的测试最终名单居然有我, 有点受宠若惊,同时诚惶诚恐的感觉。.最近确实事情比较多,怕耽误测试的效果;同时也希望有人能够关注一下硬件的实际性能。本来只想做条鲶鱼,趟趟浑水,打打酱油。没想到居然就上了,汗颜啊!亚历山大 ...… 查看全部问答> |
|
上电(mp/mc=0)-->bootrom(判断引脚状态)-->flash(3f7ff6.跳转指令)-->_c_init00-->main()-->memcopy()[此函数的作用是将不能在flash里之行的程序段复制到rom里]-->initflash()[此函数必须在rom里执行]-->while(1);… 查看全部问答> |




