[分享]Keil编译后的Code,RO,RW,ZI分别表示什么以及和芯片Flash、SRAM的对应关系

2019-07-20 15:45发布

在使用keil开发STM32应用程序时,点击Build后在Build Output窗口中经常会有如下信息:



以前一直好奇这几个参数和实际使用的STM32芯片中Flash和SRAM的对应关系,于是上网搜了一圈,做如下总结:
  • 这些参数的单位是Byte
  • 图中几个参数分别代表
    • Code:代码的大小
    • RO:常量所占空间
    • RW:程序中已经初始化的变量所占空间
    • ZI:未初始化的static变量和全局变量以及堆栈所占的空间
  • 上述参数和芯片Flash以及SRAM的对应关系是
    • Flash占用大小=Code+RO+RW
    • SRAM占用大小=RW+ZI
不知道有没有人会像我一样好奇为什么RW参数同时参与了Flash和SRAM占用量的计算。这是因为Flash部分的属性是Read-Only的,而SRAM虽然是Read-Write但里面数据不能掉电保存,所以只能把已经初始化的值保存到ROM里,上电后再拷贝到SRAM中进行读写操作,即两部分都需要留出RW变量所占用的空间。这里给出的解释比较详细,在此我参照着用STM32F407ZGT6图解一下。
STM32F407ZGT6的Flash大小为1MB,SRAM大小为(128KB+64KB)。这里SRAM之所以分开表示是因为在芯片内部前面的128KB和后面的64KB地址不是连续的,后面的64KB在ST官方叫做CCM (core coupled memory) ,据说是由内核直接访问的,不能由外设访问(见原帖2楼)。下面给出的示意图中只标出了前面的128KB的SRAM空间。红 {MOD}部分表示STM32F407ZGT6提供的Flash和SRAM大小。



结合上图说下STM32F407ZGT6的启动过程(采用Cortex-M4、Cortex-M3内核的芯片基本都是这个过程):上电后首先从 0x00000000(映射到0x08000000,这里只考虑从内部Flash启动)处获得中断向量表,然后在运行用户代码之前会在标号2处有一段引导代码负责把存在Flash中的初始化变量的值Copy到SRAM中对应的变量位置(标号3),之后把ZI区全部清零(标号4),之后才正式开始运行用户代码(标号5)。

对于详细的启动过程,这个帖子写得比较详细,可以试试~

参考资料:
http://stackoverflow.com/questions/5430284/rom-and-ram-in-arm
http://blog.csdn.net/he_ning/article/details/35226125
http://infocenter.arm.com/help/index.jsp?topic=/com.arm.doc.faqs/4027.html
http://www.amobbs.com/thread-5517029-1-1.html
http://anlx27.iteye.com/blog/1575848
友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
36条回答
k0becheng
2019-07-20 21:21
回复【2楼】龙之谷:
---------------------------------
ZI的意思是Zero Initialized,从字面的理解应该是用0进行初始化。我之前知道static类型的值不初始化的话会自动初始化为0,而局部变量的值不初始化则是未定义的。
不过刚刚搜了一些文档,在ARM官方的文档中心有这样一段说明"The ANSI C specification states that static data that is not initialized explicitly should be initialized to zero. The compiler therefore places both zero-initialized and uninitialized data in the same ZI section, which is zero-filled at run time by the C library initialization code."大意是ANSI C规范规定没有进行初始化的static型变量应该被初始化为0。编译器本身把需要初始化为0的变量(static型变量)和不需要初始化的变量统一放在了ZI区,并在C语言库的初始化代码中对ZI区的所有内容进行了初始化为0的操作。
文档地址:http://infocenter.arm.com/help/index.jsp?topic=/com.arm.doc.faqs/4027.html
最后我用一个简单的串口打印程序进行了验证:
static int s_value;
int g_value;
int main(void)
{
    int l_value;
    printf("The static value is %d ", s_value);
    printf("The global value is %d ", g_value);
    printf("The local  value is %d ", l_value);
}
最终输出的值都为0,所以ZI区域的内容应该是全部被初始化为0的(堆栈也在ZI区),定义在堆栈中的局部变量会随着堆栈的使用而呈现不同的值(但一开始也是0),因此局部变量不初始化使用是危险的。

一周热门 更多>