1.前言
这个部分我们描述了简单的链接脚本命令
2.设置entry point
程序中第一条运行的指令被称为入口点entry point,可以使用ENTRY链接脚本命令设置entry point,参数是一个符号名:
ENTRY(symbol)
有几种方法可以设置entry point,链接器会按照如下的顺序来try各种方法,只要任何一种方法成功则会停止:
- the ‘-e’ entry command-line option;
- the ENTRY(symbol) command in a linker script;
- the value of the symbol start, if defined;
- the address of the first byte of the ‘.text’ section, if present;
- The address 0
3.Commands Dealing with Files
有几种处理文件的链接脚本命令:
- INCLUDE filename:这里包含了链接脚本文件名
这个文件会在当前目录搜索,或者在"-L"指定的目录搜索,可以嵌套调用INCLUDE达10级
即: 文件1内INCLUDE文件2, 文件2内INCLUDE文件3… , 文件10内INCLUDE文件11. 那么文件11内不能再出现 INCLUDE指令了.
可以把INCLUDE命令放到文件的最前面,也可以放到MEMORY或SECTIONS命令里面,或者放到输出section描述里面。
- INPUT(file, file, ...)
- INPUT(file file ...)
INPUT命令指导链接器在链接时包含文件,否则它们必须出现在命令行
举例来说,如果你链接的时候总是想包含"subr.o",但是不想在每行命令行里输入,可以把‘INPUT (subr.o)’放到链接脚本里
事实上可以把所有的输入文件放到链接脚本里,然后通过-T选项来调用链接器
这种情况下,sysroot前缀被配置,文件名以“/”开始,被处理的脚本位于sysroot前缀,文件名将在sysroot前缀搜索。
否则链接器将在当前目录搜索,如果没有发现,将搜索archive库搜索路径,可以参考‘-L’ in Section 2.1 [Command Line Options], page 3.
如果使用‘INPUT (-lfile)’,则ld将转换文件名为libfile.a,与命令行参数"-l"一样
- GROUP(files) : 指定需要重复搜索符号定义的多个输入文件
除了file必须是库文件以外,该命令与INPUT相似, 且file文件作为一组被ld重复扫描,直到不在有新的未定义的引用出现。
- OUTPUT(FILENAME) : 定义输出文件的名字
同ld的-o选项, 不过-o选项的优先级更高. 所以它可以用来定义默认的输出文件名. 如a.out
- SEARCH_DIR(PATH) :定义搜索路径,
同ld的-L选项, 不过由-L指定的路径要比它定义的优先被搜索。
- STARTUP(filename) : 指定filename为第一个输入文件
在链接过程中, 每个输入文件是有顺序的. 此命令设置文件filename为第一个输入文件。
就象这个文件是在命令行上第一个被指定的文件一样, 如果在一个系统中,,入口点总是存在于第一个文件中,那这个就很有用。
4.Commands Dealing with Object File Formats
- OUTPUT_FORMAT(BFDNAME) : 设置输出文件使用的BFD格式
同ld选项-o format BFDNAME, 不过ld选项优先级更高.
- OUTPUT_FORMAT(DEFAULT,BIG,LITTLE) : 定义三种输出文件的格式(大小端)
对于此命令,要在命令行中使用-EB或-EL选项来指定不同的输出文件格式
如果'-EB'和'-EL'都没有使用, 那输出格式会是第一个参数 DEFAULT,
如果使用了'-EB',输出格式会是第二个参数 BIG,
如果使用了'-EL', 输出格式会是第三个参数, LITTLE.
比如:缺省的基于 MIPS ELF 平台连接脚本使用如下命令:
OUTPUT_formAT(elf32-bigmips, elf32-bigmips, elf32-littlemips)
这表示缺省的输出文件格式是'elf32-bigmips', 但是当用户使用'-EL'命令行选项的时候, 输出文件就会被以`elf32-littlemips'格式创建.
- TARGET(BFDNAME):设置输入文件的BFD格式
同ld选项-b BFDNAME. 若使用了TARGET命令, 但未使用OUTPUT_FORMAT命令, 则最用一个TARGET命令设置的BFD格式将被作为输出文件的BFD格式.
5. Assign alias names to memory regions
别名可以被加到用MEMORY命令创建的存储区域.每个名称对应着最多一个存储区域
REGION_ALIAS函数创建了存储区域region的一个别名.这使得输出section可以灵活地映射到存储区域。后面有一个例子。
假设我们有一个具有各种存储设备的嵌入式系统的应用:
(1)各存储设备存储特性
易失性存储器RAM允许执行代码或存储数据;
非易失性存储器ROM允许执行代码和数据只读访问;
非易失性存储器ROM2,具有只读数据访问和不可运行代码的特性。
(2)我们有四个输出sections:
.text 程序代码
.rodata 只读数据
.data 可读写初始化数据
.bss 可读写初始化数据,但数据必须被初始化为0.
(3)目标是提供一个链接器命令文件,该文件包含定义输出sections的系统独立部分和映射输出sections到系统上有效的存储区域的系统非独立部分.我们的嵌入式系统带有三种不同的存储配置A,B和C:
Section | Variant A | Variant B | Variant C |
.text | RAM | ROM | ROM |
.rodata | RAM | ROM | ROM2 |
.data | RAM | RAM/ROM | RAM/ROM2 |
.bss | RAM | RAM | RAM |
注:符号RAM/ROM 或 RAM/ROM2 表示这个section会被加载到相应的区域ROM 或 ROM2.请注意data section的加载地址是基于三个变量中.rodata section的末尾开始的.
(4)基本链接器脚本处理了如下输出section.它包含了系统非独立的linkcmds.memory文件,该文件用于描述存储分布:
INCLUDE linkcmds.memory
SECTIONS
{
.text :
{
*(.text)
} > REGION_TEXT
.rodata :
{
*(.rodata)
rodata_end = .;
} > REGION_RODATA
.data : AT (rodata_end)
{
data_start = .;
*(.data)
} > REGION_DATA
data_size = SIZEOF(.data);
data_load_start = LOADADDR(.data);
.bss :
{
*(.bss)
} > REGION_BSS
}
(5)现在我们需要三个不同的 linkcmds.memory 文件来定义存储区域和别名。 针对A,B和C三种的linkcmds.memory的内容如下:
A Here everything goes into the
RAM
.- 这里所有的都进入到RAM
MEMORY { RAM : ORIGIN = 0, LENGTH = 4M } REGION_ALIAS("REGION_TEXT", RAM); REGION_ALIAS("REGION_RODATA", RAM); REGION_ALIAS("REGION_DATA", RAM); REGION_ALIAS("REGION_BSS", RAM);
B Program code and read-only data go into the
ROM
.Read-write data goes into the
RAM
.An image of the initialized data is loaded into the
ROM
and will be copied during system start into theRAM
.程序代码和只读数据进入的是ROM。可读写的数据进入的是RAM。初始化数据的一个镜像被加载到ROM并且当系统启动时会被拷贝到RAM。
MEMORY { ROM : ORIGIN = 0, LENGTH = 3M RAM : ORIGIN = 0x10000000, LENGTH = 1M
} REGION_ALIAS("REGION_TEXT", ROM); REGION_ALIAS("REGION_RODATA", ROM); REGION_ALIAS("REGION_DATA", RAM); REGION_ALIAS("REGION_BSS", RAM);
C Program code goes into the
ROM
. Read-only data goes into the ROM2
.
Read-write data goes into the
RAM
.An image of the initialized data is loaded into the
ROM2
and will be copied during system start into theRAM
.
程序代码进入的是ROM。只读数据进入的是ROM2。可读写的数据进入的是RAM。初始化数据的一个镜像被加载到ROM2并且当系统启动时被拷贝到RAM。
MEMORY { ROM : ORIGIN = 0, LENGTH = 2M ROM2 : ORIGIN = 0x10000000, LENGTH = 1M RAM : ORIGIN = 0x20000000, LENGTH = 1M } REGION_ALIAS("REGION_TEXT", ROM); REGION_ALIAS("REGION_RODATA", ROM2); REGION_ALIAS("REGION_DATA", RAM); REGION_ALIAS("REGION_BSS", RAM);
It is possible to write a common system initialization routine to copy the .data
section from ROM
or ROM2
into the RAM
if necessary:
如果需要,请尽可能地写一个通用的系统初始化例程,实现把.data section从ROM/ROM2拷贝到RAM。
#include <string.h> extern char data_start []; extern char data_size [];
extern char data_load_start []; void copy_data(void) { if (data_start != data_load_start) { memcpy(data_start, data_load_start, (size_t) data_size); } }
6. Other Linker Script Commands
- ASSERT(exp, message)
确保exp是非0的
如果是0则退出链接,并返回错误码,打印message
- EXTERN(symbol symbol ...)
强制未定义的符号链接进输出文件,这样可以触发链接器从标准库文件链接。
每个EXTERN可以列出几个symbol,也可以使用多个EXTERN,跟在命令行使用-u选项是一样的
- FORCE_COMMON_ALLOCATION
- INHIBIT_COMMON_ALLOCATION
- INSERT [ AFTER | BEFORE ] output_section
- NOCROSSREFS(section section ...)
让ld产生一个错误
- OUTPUT_ARCH(bfdarch)
指定一个特别的输出机器结构,参数用的是BFD库的名字。你可以使用objdump程序的‘-f’选项查看目标文件的机器结构
参考文献
[1] http://blog.csdn.net/han22647/article/details/64920623
[2] http://blog.csdn.net/huiyuyang_fish/article/details/16884593