小猫爪:嵌入式小知识03-程序的组成、存储与运行(转自野火教程)

小猫爪:嵌入式小知识03-程序的组成、存储与运行(转自野火教程)

1 前言

这一章我们主要学习我们编写的C最终是怎么在单片机里跑起来的,野火的教程里写的很全,所以我就照搬过来了。

2 程序的组成

在工程的编译提示输出信息中有一个语句“Program Size:Code=xx RO-data=xx RWdata=xx ZI-data=xx”,它说明了程序各个域的大小,编译后,应用程序中所有具有同一性质的数据(包括代码)被归到一个域,程序在存储或运行的时候,不同的域会呈现不同的状态,这些域的意义如下:

①Code:即代码域,它指的是编译器生成的机器指令,这些内容被存储到ROM区。

②RO-data:Read Only data,即只读数据域,它指程序中用到的只读数据,这些 数据被存储在ROM区,因而程序不能修改其内容。例如C 语言中const 关键字定义的变量就是典型的RO-data。

③RW-data:Read Write data,即可读写数据域,它指初始化为“非0 值”的可读写数据,程序刚运行时,这些数据具有非0 的初始值,且运行的时候它们会常驻在RAM 区,因而应用程序可以修改其内容。例如C 语言中使用定义的全局变量,且定义时赋予“非0 值”给该变量进行初始化。

④ZI-data:Zero Initialie data,即0初始化数据,它指初始化为“0 值”的可读写数据域,它与RW-data 的区别是程序刚运行时这些数据初始值全都为0,而后续运行过程与RW-data 的性质一样,它们也常驻在RAM 区,因而应用程序可以更改其内容。例如C 语言中使用定义的全局变量,且定义时赋予“0 值”给该变量进行初始化(若定义该变量时没有赋予初始值,编译器会把它当ZI-data 来对待,初始化为0);

⑤ZI-data 的栈空间(Stack)及堆空间(Heap):在C 语言中,函数内部定义的局部变量属于栈空间,进入函数的时候从向栈空间申请内存给局部变量,退出时释放局部变量,归还内存空间。而使用malloc 动态分配的变量属于堆空间。在程序中的栈空间和堆空间都是属于ZI-data 区域的,这些空间都会被初始值化为0 值。编译器给出的ZI-data 占用的空间值中包含了堆栈的大小(经实际测试,若程序中完全没有使用malloc 动态申请堆空间,编译器会优化,不把堆空间计算在内)。

综上所述,以程序的组成构件为例,它们所属的区域类别见表 39-1。

程序组件所属类别
机器代码指令Code
常量RO-data
初值非0 的全局变量RW-data
初值为0 的全局变量ZI-data
局部变量 ZI-data栈空间
使用malloc 动态分配的空间ZI-data 堆空间

编译器所干的事情就是把我们写的源文件按照上面说的类别分解、组合成一个个这样的块,专业词汇section(.o),链接器接下来再将这些块按照以上类别再进行组合,最后按照链接文件定义的地址信息给这些块加地址信息生成最终的elf文件,elf文件经过格式转换生成镜像。

3 程序的存储和运行

RW-data 和ZI-data 它们仅仅是初始值不一样而已,为什么编译器非要把它们区分开?这就涉及到程序的存储状态了,应用程序具有静止状态和运行状态。静止态的程序被存储在非易失存储器中,如RT1052 的外部FLASH,因而系统掉电后也能正常保存。但是当程序在运行状态的时候,程序常常需要修改一些暂存数据,由于运行速度的要求,这些数据往往存放在内存中(RAM),掉电后这些数据会丢失。因此,程序在静止与运行的时候它在存储器中的表现是不一样的。
在这里插入图片描述图中的左侧是应用程序的存储状态,右侧是运行状态,而上方是RAM存储器区域,下方是ROM存储器区域。程序在存储状态时,RO 节(RO section)及RW 节都被保存在ROM区。当程序开始运行时,内核直接从ROM中读取代码,并且在执行主体代码前,会先执行一段加载代码,它把RW 节数据从ROM复制到RAM, 并且在RAM加入ZI 节,ZI 节的数据都被初始化为0。加载完后RAM区准备完毕,正式开始执行主体程序。

编译生成的RW-data 的数据属于图中的RW 节,ZI-data 的数据属于图中的ZI节。是否需要掉电保存,这就是把RW-data 与ZI-data 区别开来的原因,因为在RAM创建数据的时候,默认值为0,但如果有的数据要求初值非0,那就需要使用ROM记录该初始值,运行时再复制到RAM。

对于XIP执行的芯片来说,RO 区域不需要加载到SRAM,内核直接从FLASH 读取指令运行。计算机系统的应用程序运行过程很类似,不过计算机系统的程序在存储状态时位于硬盘,执行的时候甚至会把上述的RO 区域(代码、只读数据)加载到内存,加快运行速度,还有虚拟内存管理单元(MMU)辅助加载数据,使得可以运行比物理内存还大的应用程序。所以对于没有MMU的MCU来说,所以无法支持Linux 和Windows 系统。当程序存储到外部部FLASH 时(即ROM区),它占用的空间是Code、RO-data 及RW-data 的总和,所以如果这些内容比MCU芯片的FLASH 空间大,程序就无法被正常保存了。当程序在执行的时候,需要占用内部SRAM空间(即RAM区),占用的空间包括RW-data 和ZI-data。应用程序在各个状态时各区域的组成见下表。

程序状态与区域组成
程序执行时的只读区域(RO)Code + RO data
程序执行时的可读写区域(RW)RW data + ZI data
程序存储时占用的ROM区Code + RO data + RW data

END

  • 7
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

小猫爪

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值