根(ROOT)区域是具有相同加载和执行地址的区域。
2.1 根(ROOT)执行区域和程序加载入口
Image的程序加载入口必须在根(ROOT)区域。
如果初始入口点不在根(ROOT)区域中,则链接失败,链接器会给出错误消息。
2.2 根(ROOT)执行区域和绝对(ABSOLUTE) 属性
你可以使用ABSOLUTE属性来指定根执行区域。给执行区域指定ABSOLUT属性,不管是显式的指明、还是默认的,都将为第一个执行区域和紧挨的加载区域设为相同地址。
为了给执行区域和加载区域设置相同的地址,有下面两种方法:
1、给执行区域和加载区域设置相同的地址值。
2、使用 +0 着用便宜地址给加载区域中的第一个执行区域。
若 +0 被指定给加载区域内的所有执行区域,那么所有 ZI 之前的执行区域都属于 root 执行区域。
下图定义了一个root区域的实现:
2.3 根(ROOT)执行区域和固定(FIXED) 属性
你可以使用 FIXED 属性来创建一个具有固定地址的ROOT 执行区域。使用FIX属性必须保证该区域的加载地址和执行地址是相同的,如下图所示:
下图展示了相应的分散加载文件:
你可以使用这个属性来放置一个函数或者一块数据(如常量表或者checksum),在Fixed 地址中的内容能够很方便用指针访问。
如果你希望指定一些初始化代码放在ROM的开头,一些checksum内容放在ROM的结尾,中间的ROM内容就可能没有使用到,请使用 * 或 .ANY 符号来填充这个区域。
为了方便你的代码调试,建议在分散加载文件中尽量少使用指定分配,将剩余的函数和数据的分配权留给链接器。
注意:在某些情况下,单个加载区域使用FIXED 是不合适的,其它指定固定区域的技术有:
1 如果你的加载器Loader能处理多个多个加载区域,请将RO代码或data放到它们自己的加载区域。
2 如果你不需要函数或数据放在ROM中固定的位置,请使用 ABSOLUTE 代替了 FIXED。加载器会把数据从加载区域复制到RAM中的特定地址,ABSOLUTE 是默认属性。
3 若想放置一个数据结构到IO映射的内存地址中,请使用2个加载区域,并指定UNINT属性,该属性可以保证这块区域不会被初始化成0。
下面是一个FIXED乱用的示例,是个反面示例,请注意错误。
2.4 放置函数或数据到指定位置的方法
有很多方法可以实现放置函数或数据到指定位置,编译器一般会给单个源代码文件产生 RO, RW, ZI 和 XO四个分段。这些分段包含了该源代码中的所有代码和数据。
若想放置单个函数或者数据到固定地址,你必须让链接器单独的、与输入文件的其它部分区分开来处理该函数或数据。Linker允许你使用如下方式了指定某段的链接地址:
1、 你可以创建一个分散加载文件,在其中定义一个指定地址的执行区域,并只将该段的描述符号放入其中。
2、 对于特定名字的分段,链接器能根据分段的名称获取存放地址,这种特定的分段名称被叫做 __at 段。
要将函数或变量放在特定的地址,必须将其放在自己的分段中。有几种方法可以做到这一点:
1、将函数和数据放在自己的源代码文件中。
2、使用 __attribute__((at(address))) 放置变量到指定地址的单独分段。
3、使用 __attribute__((section("name"))) 放置函数或变量到指定名称的分段中。
4、 在汇编代码中直接使用 AREA , 在汇编中 AREA 是最小定位分配单元。
5、使用编译器选型 --split_sections 来给每个函数产生单独的 ELF文件。这个选项后一些函数的代码量会少量增加,这是因为它会减少函数间一些潜在的共享地址、数据和字符串。但是最终生成程序映像时也可以使用 armlink --remove 来删除未使用的函数,以此节约代码量。
2.5 使用 __attribute__((section("name"))) 来指定 Code 和 Data 的位置
未完待续……