ARM程序的RO段、RW段和ZI段 --Image

Limit 含义
了解RO,RW和ZI需要首先了解以下知识:
ARM程序的组成
此处所说的“ARM程序”是指在ARM系统中正在执行的程序,而非保存在ROM中的bin映像(image)文件,这一点清注意区别。
一个ARM程序包含3部分:RO,RW和ZI
RO是程序中的指令和常量
RW是程序中的已初始化变量
ZI是程序中的未初始化的变量
由以上3点说明可以理解为:
RO就是readonly,
RW就是read/write,
ZI就是zero

ARM映像文件的组成
所谓ARM映像文件就是指烧录到ROM中的bin文件,也成为image文件。以下用Image文件来称呼它。
Image文件包含了RO和RW数据。
之所以Image文件不包含ZI数据,是因为ZI数据都是0,没必要包含,只要程序运行之前将ZI数据所在的区域一律清零即可。包含进去反而浪费存储空间。
Q:为什么Image中必须包含RO和RW?
A:因为RO中的指令和常量以及RW中初始化过的变量是不能像ZI那样“无中生有”的。

注意:如果一个变量被初始化为0,则该变量的处理方法与未初始化华变量一样放在ZI区域。
即:ARM C程序中,所有的未初始化变量都会被自动初始化为0。
总结:
1; C中的指令以及常量被编译后是RO类型数据。
2; C中的未被初始化或初始化为0的变量编译后是ZI类型数据。
3; C中的已被初始化成非0值的变量编译后市RW类型数据。


ARM程序的执行过程
从以上两点可以知道,烧录到ROM中的image文件与实际运行时的ARM程序之间并不是完全一样的。因此就有必要了解ARM程序是如何从ROM中的image到达实际运行状态的。
实际上,ROM中的指令至少应该有这样的功能:
1. 将RW从ROM中搬到RAM中,因为RW是变量,变量不能存在ROM中。
2. 将ZI所在的RAM区域全部清零,因为ZI区域并不在Image中,所以需要程序根据编译器给出的ZI地址及大小来将相应得RAM区域清零。ZI中也是变量,同理:变量不能存在ROM中
在程序运行的最初阶段,RO中的指令完成了这两项工作后C程序才能正常访问变量。否则只能运行不含变量的代码。

 

Image$$RW_IRAM1$$ZI$$Limit

是一个链接器导出的符号,代表ZI段的结束,也就是程序执行区的RAM结束后的(注意这个‘的’,有点i++和++i的意思)地址,反过来也就是我们执行区的RAM未使用的区域的起始地址(其实这里有点牵强,因为这样理解往往只是一个准寻的标准,以为在RAM的使用上ZI区往往是整个程序的最末尾,也许这里我理解错了)。

 rt_system_heap_init((void*)&Image$$RW_IRAM1$$ZI$$Limit, (void*)STM32_SRAM_END);

用于初始化RT-thread系统的堆区域。

rt_system_heap_init((void*)&Image$$RW_RAM1$$ZI$$Limit, (void*)STM32_SRAM_END);

Image$$ RW_RAM1$$ZI$$Limit,这是一个链接器导出的符号,代表ZI段的结束,也就是程序执行区的RAM结束后的(注意这个‘的’,有点i++和++i的意思)地址,反过来也就是我们执行区的RAM未使用的区域的起始地址(其实这里有点牵强,因为这样理解往往只是一个准寻的标准,以为在RAM的使用上ZI区往往是整个程序的最末尾,也许这里我理解错了)。


&Image
RWIRAM1
RWIRAM1
ZI$$Limit 取地址。

例如:
|Image
RO
RO
Limit|:表示RO区末地址后面的地址,即RW数据源的起始地址
|Image
RW
RW
Base|:RW区在RAM里的执行区起始地址,也就是编译器选项RW_Base指定的地址
|Image
ZI
ZI
Base|:ZI区在RAM里面的起始地址
|Image
ZI
ZI
Limit|:ZI区在RAM里面的结束地址后面的一个地址

程序先把ROM里|Image
RO
RO
Limt|开始的RW初始数据拷贝到RAM里面
| Image
RW
RW
Base|开始的地址,当RAM这边的目标地址到达|Image
ZI
ZI
Base|后就表示RW区的结束和ZI区的开始,接下去就对这片ZI区进行清零操作,直到遇到结束地址。

|Imag开机运行程序设置e
ZI
ZI
Limit|

 

当把程序编写好以后,就要进行编译和链接了,在ADS1.2中选择MAKE按钮,会出现一个Errors and Warnings 的对话框,在该栏中显示编译和链接的结果,如果没有错误,在文件的最后应该能看到Image component sizes,后面紧跟的依次是Code,RO Data ,RW Data ,ZI Data ,Debug 各个项目的字节数,最后会有他们的一个统计数据:

Code 163632 ,RO Data 20939 ,RW Data 53 ,ZI Data 17028

Tatal RO size (Code+ RO Data) 184571 (180.25kB)

Tatal RW size(RW Data+ ZI Data) 17081(16.68 kB)

Tatal ROM size(Code+ RO Data+ RW Data) 184624(180.30 kB)

有了上面这些信息我们就可以完全知道这几个变量是怎么来的了:
|Image
RO
RO
Base| = Image entry point = 0x0c100000 ;表示程序代码存放的起始地址
|Image
RO
RO
Limit|=程序代码起始地址+代码长度+1=0x0c100000+Tatal RO size+1
                               = 0x0c100000 + 184571 + 1 = 0x0c100000 +0x2D0FB + 1
                               = 0x0c12d0fc

|Image
RW
RW
Base| = 0x0c200000 ;由RW base 地址指定

|Image
RW
RW
Limit| =|Image
RW
RW
Base|+ RW Data 53 = 0x0c200000+0x37(4的倍数,0到55,共56个单元)
                                                                                           =0x0c200037

|Image
ZI
ZI
Base| = |Image
RW
RW
Limit| + 1 =0x0c200038

|Image
ZI
ZI
Limit| = |Image
ZI
ZI
Base| + ZI Data 17028
                              =0x0c200038 + 0x4284
                             =0x0c2042bc

也可以由此计算:
|Image
ZI
ZI
Limit| = |Image
RW
RW
Base| +TatalRWsize(RWData+ZIData) 17081
                            =0x0c200000+0x42b9+3(要满足4的倍数)
                            =0x0c2042bc
————————————————
版权声明:本文为CSDN博主「minyuanxiani」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/minyuanxiani/article/details/24042831

上一篇:时间戳,秒级,毫秒级转换DateTime格式


下一篇:mdk编译时的内存分析