|
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、多使用指针解决数据的复制,同时减少内存的占用。
|
|