keil使用命令行编译_被Keil坑了一天!实在太意外了!用Keil的来瞅瞅看

博主在搭建基于GNU ARM+Cygwin+VSCODE的开发环境时遇到hardfault中断问题,通过分析发现是Keil中RAM大小设置错误。Keil自动填充的RAM大小与实际不同,导致栈空间未充分利用,而在开源环境下手动配置连接文件时,栈顶地址设置正确,解决了问题。Keil的这种设置可能导致RAM浪费,作者建议注意内存分布初始化的差异。
摘要由CSDN通过智能技术生成

在前面的文章和视频里,我一直是以STM32F303VCTx为基础给大家介绍的,项目上也一直用着,没出现任何问题。但是前几天在探索建立基于GNU ARM+Cygwin+VSCODE的免费开发环境的时候,调试程序时老是发生hardfault中断,昨天终于发现问题之所在。

1.问题描述

使用Keil新建一个工程,选择STM32F303VCTx,看target的内容,这个内容是选好MCU后系统自动填充的,自从使用Keil以来,从来未曾想过这个数据会有错误。

b549bd32ee0b93e56c46dd46a2de5f87.png

看RAM大小

我的工程项目里也是这样的默认设置,也一直运行的很好,根本没有理由让我去怀疑。但是为什么在配置开源开发环境时会出现问题呢?因为使用开源软件时,要自己修改连接文件(*.ld)(其源文件模板在Keil_v5ARMPACKARMCMSIS5.6.0DeviceARMARMCM4SourceGCC文件夹里能找到),部分参数设置有误!

0917a2d1820c3011d858f86c42c54e1c.png

编译连接

我按Keil的参数将栈顶地址变量_estact初始化为0x2000C000,编译连接过程没有发现错误,但是在调试的时候,发生硬件中断错误(当系统无法匹配到合理的错误风类型时,产生这种错误,多半是因为SP设置错误),百思不得其解!终于在试用System Workbench for STM32时,发现其生成的连接文件RAM大小是0xA000,才恍然大悟。修改后调试成功,至此,基于ARM+Cygwin+VSCODE的开源免费项目工程环境完全搭建成功。

2.问题分析

那么,Keil的错误设置为什么运行时不会出错呢?

我在《一分钟看透“堆“和"栈"》中说到RAM的空间分布,是根据Keil这个环境来写的,当时我就充满疑问,这种定义方法,还有一部分空间可能是没有用到的--即高于SP初始化地址的空间,因为实际设置栈大小时,不太可能刚刚把空间大小设置的刚刚好和除变量和堆之外的剩余空间大小相等。

Keil设置的是栈空间大小,根据栈空间大小来初始化SP地址,这种情况往往会有空闲RAM没用到;而自己写连接文件时,恰好相反,我们设置的是栈地址,而不管栈空间是多少,实际上是实现了栈空间的最大化!当把SP值初始化为RAM顶时,全部RAM都被用上了。不会再有空余的RAM了!

9d7aacc0c65e60b5a940f6d98e656836.png

堆、栈

在Keil上实际测试了一下,在0x2000B000(超过实际RAM的最大地址)定义一变量,在读写此变量时发生hardfault,验证成功!

//把变量存到指定位置int var__attribute__((at(0x2000B000)));

3.小结

这次BUG的排查,说明KEIL和GNU GCC对内存的堆、栈分布的初始化流程和设计思想是不一样的。 也要注意到,在Keil里,很可能会有RAM空间被浪费了!!那如何才能避免浪费呢?

每一个BUG,都是一个阶梯!嵌入式学习更一步!!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值