|
使用ld0 @! k1 {, s* l3 r
********
/ x& r$ y3 L4 u; z本文档介绍GNU连接器ld的2.14版本.
/ E" m! c8 r0 {1 e! y' v$ @
c) V8 [" o r本文档在GNU自由文档许可证下发行.在"GNU自由文档许可证"一章中有关于本许可证的一份拷贝.. F) j# q+ v: ~
8 w! Y+ q1 W. w c3 E5 v" C
概述' {% M& i/ z0 z0 t- _% k e& J
********2 }4 r/ P" l; i: P5 ?7 K1 n- d- T
; X4 ?/ U8 E; G Q8 u( O'ld'把一定量的目标文件跟档案文件连接起来,并重定位它们的数据,连接符号引用.一般,在编译一个程序& Q' j5 Z e9 ?7 j! y! `, d
时,最后一步就是运行'ld'.
# e N- i* G6 J: ~" p; _7 t1 j' V7 d3 w+ ~
'ld'能接受连接命令语言文件,这是一种用AT&T的连接编辑命令语言的超集写成的文件,用来在连接的整个
: P( J R+ a& s5 ~$ @3 y0 d: f过程中提供显式的,全局的控制.
( a: _; [5 S5 t7 n9 B4 r
1 H, U* V: T& E# \! ?0 \( C3 o本版本的'ld'使用通用BFD库来操作目标文件.这就允许'ld'读取,合并,写入目标文件时,可以使用各种不同
3 C* ]1 g% n3 a7 u7 `; ?的格式,比如,COFF或'a.out'. 不同的格式可以被连接到一起产生一个有效的目标文件.6 n: q5 u6 a1 E* G
: {- ~# h& Z6 `( U: X; {除了它的灵活性,GNU连接器比其它连接器更有用的地方在于它提供了诊断信息. 许多连接器在碰到一个错误# w M% Z. k; b( C1 o
的时候立即放弃执行;但'ld'却能够继续执行,以让你发现其他的错误(或者,在某些情况下,得到一个带有错误
- g5 |: ]$ @3 F0 \2 `的输出文件)
, G* {% U# ?2 H. E9 ~3 s- L6 W1 \+ l1 |/ w, l
引用' T6 K0 P2 B$ ?0 ]3 \$ B: R
**********
2 l- h$ P+ `5 _5 d: K/ N
* x4 X- ]+ U i! t8 O9 y6 v2 pGNU连接器'ld'能够处理大量的不同情况,并且跟其他的连接器保持尽可能的兼容.这样,你就拥有更多的选择来+ v5 d1 J, l: A# e. G% y
控制它的行为.
& l- W g# g' g
$ H6 i( v4 D) J# _. U: L& H命令行选项, x; E( J ?* l( {
==================== l$ d% W. u7 }7 |
/ A q* E. F- e$ c* ~7 r+ G6 w$ h, P连接器提供大量的命令行选项,但是,在实际使用中,只有少数被经常使用.比如,'ld'的一个经常的使用场合是在
' ~% q" v/ |" _4 M# Q一个标准的Unix系统上连接标准的Unix目标文件.在这样的一个系统上,连接文件'hello.o'如下:
; r. J' |, R( J% G }# k' C1 |8 X. `1 F. w% m) i% e/ k% [, k
ld -o OUTPUT /lib/crt0.o hello.o -lc; R; p( v) j! r1 R5 T/ L
2 [! Q" S$ M* G+ o% B. D
这告诉'ld'产生一个叫OUTPUT的文件,作为连接文件'/lib/crt0.o'和'hello.o'和库'libc.a'的结果.'libc.a'5 A. U+ W" j2 w4 c# B- K5 L H+ p
来自标准的搜索路径.(参阅下文的关于'-l'选项的讨论).7 h& P1 ?5 \. d$ j' q3 g
1 n/ v* b9 M% G! o" ], G7 [* R
有些命令行选项可以在命令行的任何位置出现.但是,那些带有文件名的选项,比如'-l'或者'-T',会让文件在选+ N% r3 q% E" @
项出现的位置上被读取. 对于非文件选项,以带不同的参数重复它,不会有进一步的效果,或者覆盖掉前面的相同
, Z" {" m5 Z- \% ]- P9 v9 s$ b- n8 Y项.那些多次出现时具有特殊含义的选项会在下文的描述中指出.
6 f' H4 Y' y. y7 U
* T- S4 M1 J. H y4 `无参数选项是那些被连接的目标文件和档案文件.它们可能紧随命令行选项,或在它们前面,或者跟它们夹杂在一' q7 k( m" l" }4 O8 u
起,但是一个目标文件参数是不会出现在一个选项跟它的参数之间的.$ j, i5 \. p r, {+ W. }
. i0 v n( r1 _- [# l
通常,连接器至少引用一个目标文件,但是你可指定其它形式的二进制输入文件,这可以通过'-l','-R'或者脚本
; e2 M$ L- T/ O. C5 I命令语言来实现.如果没有任何二进制文件被指定,连接器不会产生任何输出,并给出信息:"缺少输入文件."# f8 G6 F/ M, ]5 T
% R' v4 e) S- i, o0 q( Y% S, b& ?; [如果连接器不能识别目标文件的格式,它会假设这些只是连接脚本.以这种方式指定的脚本增加了连接用的主连
t3 _% L, J1 @6 t接脚本的内容(主连接脚本即缺省连接脚本或使用'-T'指定的脚本). 这个特性可以允许连接器连接一些文件,8 R: I1 {1 x: h7 i6 ~# U @
它们看上去既像目标文件,又像档案文件,但实际上只是定义了一些符号值,或者使用'INPUT'或'GROUP'来载入其$ v1 f2 Z( F; M
它的目标文件.需要注意的是,用这种方式指定一个脚本只是增加了主连接脚本的内容;要完全替换掉主连接脚本* V$ E6 t2 S. j; _" z; e
,需要使用'-T'.
0 H/ [' z/ c9 q0 t: n0 G
" j6 u7 A8 m" H, f+ X1 M. V; n. r: J对于名称是单个字符的选项,选项参数必须紧跟在选项字母后面,中间不留空,或者也可留有一个空格.0 @4 ~, M4 i o2 z6 M
- q+ b1 i+ N8 h- T6 [$ j对于名称是多个字符的选项,选项前可以有一个或两个破折号;比如,'-trace-symbol'和`--trace-symbol'是等价- l" M6 [7 X2 J
的. 注意,对于这条规则有一个例外.那些以小写字母'o'开头的多字符选项前面只能是两个破折号,这是为了避免/ ^- R$ f, Y% L; y8 K/ y7 I
跟选项'-o'混淆. 比如'-omagic'把输出文件的名字定为'magic',而'--omagic'在输出文件中设置NMAGIC标志.
, ~* I0 ]. h1 U; C8 r6 V+ m3 o: J" E! G# c2 ?6 U3 h
多字符选项的参数必须跟选项名间以一个等于号分开,或者以一个空格分开.比如:`--trace-symbol foo'和
+ Z% ~; k7 Z2 d9 Z`--trace-symbol=foo'是等价的. 多字符选项的名字唯一缩写符也是可以被接受的." r2 ~2 t$ {8 d$ I- z* q: o
5 |' t. `8 l. G+ i+ Y5 i4 J0 Y注意,如果连接器通过被编译器驱动来间接引用(比如gcc), 那所有的连接器命令行选项前必须加上前缀'-Wl'8 ^ M i8 B3 q! @# N5 {
(或者能被特定编译器驱动接受的其他前缀),就像下面这样:
! v0 I! x' N9 u7 C! U( i9 C: d, m& h1 Q+ F9 i& [
gcc -Wl,--startgroup foo.o bar.o -Wl,--endgroup6 Y; K" H/ E. \2 r7 ~# L3 ~
4 Q. J2 F1 S! E/ L( z* K
这很重要,因为否则的话,编译器驱动程序会默认丢掉这些连接选项,产生一个错误的连接.
; j8 {& B! J3 c" s
( p7 v- c/ E6 X! V9 y, o( c. x6 H下面是关于被GNU连接器接受的常用命令行开关的一个列表:, V5 f0 F7 b4 A2 m0 B- I: Z
- m8 q% R& X$ z- I! x8 u \: \
`-aKEYWORD'7 f5 Q! q7 ^& ~( H( m
这个选项在HP/UX兼容系统上被支持. 参数KEYWORD必须是下面字符串中的一个:`archive', : \5 K9 T! O$ ^0 ^8 c
`shared', or `default'. `-aarchive'在功能上跟`-Bstatic'相同,而另外两个关键字功能上跟1 n* }7 {3 q% y3 w v9 y P- z
`-Bdynamic'相同. 这个选项可被多次使用.
* Z, H- ]3 _; g7 |2 D5 n. C& p' B$ Z- I* N3 m* c; z h5 \
`-AARCHITECTURE'
% N/ j; j6 \3 D5 T1 W`--architecture=ARCHITECTURE'
, `# X4 U: [7 ^$ ~3 O: }' ?9 }在最近发行版本的'ld'中,这个选项只在Intel 960系列架构上有用. 在那种'ld'配置中,参数
2 @% g/ D, w- B; b ARCHITECTURE确定960系列的某一特定架构,启用某些安全措施,并修改档案库的搜索路径.' T" @- p+ L% i' s
/ M, J- J" w; m3 @# r- K. A5 _+ O
将来的'ld'发行版可能为其它架构系列支持相似的功能.
" F! l7 P" b* L! N) E# @9 p( I# R
* Y3 G6 M Q* ^7 u1 y8 P`-b INPUT-formAT' h9 b8 x8 i' Z+ t9 [$ f9 e/ x# {
`--format=INPUT-formAT'
, U D4 _3 p! K1 m9 F3 Q8 ?5 u'ld'可以被配置为支持多于一种的目标文件.如果你的'ld'以这种方式被配置,你可以使用'-b'选6 B! e% k/ N% |
项为输入目标文件指定二进制格式. 就算'ld'被配置为支持可选目标格式,你不必经常指定这一项,, L1 y/ |* \* J0 N* X* {# G
因为'ld'被配置为在每一台机子上把最常用的格式作为默认输入格式. INPUT-formAT是一个字符串,
( k( u5 W3 S9 w ~$ f5 I8 E J+ B- P8 s( L4 d+ ]
你可能在连接一个不常用的二进制格式文件时需要这个参数.你也可使用'-b'来显式切换格式(在连接
( U3 c1 H. q( r; m0 I不同格式的目标文件时),方法是在每一组特定格式的目标前使用'-b INPUT-formAT'.
5 x# h! m, y! j2 ~9 I; _, l
Z2 {( r' ~/ x9 R/ W5 Z缺省的格式是从环境变量'GNUTARGET'中得到的.你也可以从一个脚本中定义输入格式,使用的命令是
b1 W U$ \' d+ q) i/ |/ ^% _; ['TARGET'.. S( Y+ \6 u f* H
* z' }# G9 Q& `' x`-c MRI-COMMANDFILE'
" r! W ]" q, D`--mri-script=MRI-COMMANDFILE'$ W: ] Y' U- @) C3 @
为了跟MRI生产的连接器兼容,'ld'接受另一种用受限命令语言写成的脚本文件,通过选项'-c'引入MRI2 S% M* \- a, e# z& X
脚本文件;使用'-T'选项是运行用普通'ld'脚本语言写的连接脚本.如果MRI-CMDFILE不存在,'ld'在'-L'
4 [+ P- _3 m* U: x$ k指定的目录中寻找. h6 \5 F! n# K m, ]- X1 X% ^
2 K# {! w! |7 I" X+ n4 y4 \6 `
`-d'
, F* g) I# n" B9 C' P4 c`-dc'# v. F; N5 R$ I( u8 x; z
`-dp'4 t) y- u. v% M. T7 n& m
这三个选项是等价的; 多字符形式是为了跟其他连接器兼容才被支持的.它们给普通符号分配空间,即+ I, ?5 I+ i/ \ { ]* L
使一个重定位输出文件已经被指定(通过'-r'). 脚本命令`FORCE_COMMON_ALLOCATION'具有同样的效果.! Q+ N3 r+ v& p/ }7 q. W) \) d2 j
- y$ [% }- e* k7 N; z8 ~0 _1 d`-e ENTRY'; e' B$ {" ^7 m4 b( c) _3 |" R! l
`--entry=ENTRY'
% E4 m' k- ] ^ R: @) N使用符号ENTRY作为你的程序的开始执行点,而不是使用缺省的进入点.如果没有叫做ENTRY的符号,连接器
9 g8 P: w4 K& s1 p会企图把ENTRY作为一个数字进行分析,并使用它作为入口地址(数字会被解释为10进制的;你可以使用前- A: B( ^6 Z/ U/ R+ g& x8 Y. o- N
导的'0x'强制为16进制,或'0'作为8进制.)/ W; }9 Y/ u$ z% Z% x8 |
- L9 a& Y& u) l% h7 U`-E'3 K1 a; c2 i7 c/ G
`--export-dynamic'; t( `7 O" v- x
当创建一个动态连接的可执行程序时, 把所有的符号加到动态符号表中.动态符号表是一个符号集,这
# x% ?5 @# `: O4 a* v; C% {- Y些符号对于运行时的动态对象是可见的.1 r1 h, k# R$ Q8 f$ U5 P# S
5 d- l) F4 g% I8 F# G* D" S2 b
如果你不使用这个选项,动态符号表中就会只含有那些连接进来的动态对象中用到的符号
! ~) p7 s2 w1 p* S
) m& H& w, L) m3 L! g3 n如果你使用'dlopen'来载入动态对象,它需要引用程序中的符号,那你可能需要在连接程序时用到这个& V) J( R0 d/ Z6 n9 J' x4 D% O
选项.
6 u1 m- k# f/ }9 \# c) n) j0 x2 ?) C2 s
你也可以使用版本脚本来控制哪些符号应当被加到动态符号表中.7 x v1 t( U. I. o& S
. A6 ]+ b! _" D' l' u`-EB'9 x0 p; n/ L/ N/ O3 x* v1 r
连接big-endian对象. 这会影响缺省输出格式.8 \3 U) l9 ]! C0 d: O( E
+ y/ V$ t0 g- c" j
`-EL') F$ N/ s s+ N. a$ k2 N
连接little-endian对象. 这会影响缺省输出格式.
6 R* p7 ?2 a" T
$ g2 Y( F( K m6 Z5 N2 O" I`-g'7 a, d6 m( g/ P+ i
忽略. 为了跟其它工具兼容而提供. k* O6 A& h9 W3 |8 k+ z
! q: l. b, l# |1 q0 ^$ Q`-i'
; \! ~! p: \% l3 v4 A6 C- S( }执行一个增量连接(跟'-r'等同)
- V! k% B3 U- m# O) w7 M& t( w/ K/ H+ |7 y; m! X% h. A4 S! o
`-init NAME'
1 J% t7 Q& H3 W9 D; j' R当创建一个ELF可执行文件或共享对象时,当可执行文件或共享对象被加载时,调用NAME, 这是通过把
8 X- Y1 x" o) _; @, SDT_INIT设置成函数的地址实现的. 缺省情况下,连接器使用'_init'作为调用的函数.
* q2 ^7 {8 p# P+ B# n
* r [2 b% l+ B! {3 C+ o/ O# f* s5 T`-lARCHIVE'
2 \ l D2 c4 U- g`--library=ARCHIVE'% p/ L9 q" y. u) M" S, n
增加一个档案文件ARCHIVE到连接的文件列表中.这个选项可以被多次使用. 'ld'会为每一个指定的
6 c$ G7 Y8 N& F! _4 s; i' Q PARCHIVE搜索它的路径列表,寻找`libARCHIVE.a'
$ \5 l) Z, @3 [: B1 {0 t9 q% a, p' G) e& B( E0 G; b* _( U& | x
对于支持共享库的系统, 'ld'可能还会搜索扩展名不是'.a'库.特别的,在ELF和SunOS系统上,'ld'会$ S3 [. R+ s" c3 T
在搜索带有'.a'扩展名的库前搜索带'.so'扩展名的库.
* P& s5 u0 c( |9 |3 F1 \" L3 |6 \# _8 ? @0 T
`-M') w3 a2 `3 t' s Z
`--print-map'. A" s8 k8 w( |, c# A
打印一个连接位图到标准输出.一个连接位图提供的关于连接的信息有如下一些:
?1 b1 V% P+ L8 K3 }8 `& l9 ~9 l1 r+ L8 l: i) t
* 目标文件和符号被映射到内存的哪些地方.( M0 N# k5 L$ K( F( M1 Y, E2 Y
! p; u& {+ k3 M9 h; V# o) y
* 普通符号如何被分配空间.
8 y9 \' h j ^4 y! L) K
1 }. i8 }( _8 s* a* ]7 j* 所有被连接进来的档案文件,还有导致档案文件被包含进来的那个符号." G# ?9 v0 g4 ?& ^
9 x. L9 P3 X. U. ^
`-n'
' I5 f9 M; D3 d& D6 o: i u`--nmagic'
/ t! u" z3 k9 j; Y5 `6 v G* u# `, a关闭所有节的页对齐,如果可能,把输出格式标识为'NMAGIC'.4 M% B& p% W9 B4 S0 D4 }
1 k! `5 Q% t% W' d`-N'
9 N: o+ _6 Z7 o" i* K`--omagic'
$ R& j2 n- I7 t$ m, q! ^把text和data节设置为可读写.同时,取消数据节的页对齐,同时,取消对共享库的连接.如果输出格式8 B" V% p | d& p% [1 `& B2 p
支持Unix风格的magic number, 把输出标志为'OMAGIC'.
4 n2 U+ \" m8 a% @) K ~8 s A6 f# H& q' U& U* ?
`--no-omagic'
+ v9 H+ W( `" s这个选项执行的操作大部分正好跟'-N'相反.它设置text节只读,强制data节页对齐. 但是,这个选项
* w2 T- k/ C# B+ @7 k3 E/ l2 O2 H并不开启连接共享库的功能. 使用'-Bdynamic'开启这个功能.% z0 R( \8 d; U+ X
% m4 l, S) F% Y`-o OUTPUT'
! M* n, Z! {; S6 j% C! Q`--output=OUTPUT'( U9 J5 a i" x) O8 `: v
使用OUTPUT作为'ld'产生的程序的名字;如果这个选项没有指定,缺省的输出文件名是'a.out'.脚本命
1 u; K: ~5 G5 d Y( ^, p令'OUTPUT'也可以被用来指定输出文件的文件名.3 D' r# X8 B7 J7 R& M1 s* c
' s3 }0 k4 v8 R' `' X. D`-O LEVEL'$ p! A- N$ |5 L$ E9 [# M7 [7 y5 V! a
如果LEVEL是一个比0大的数值, 'ld'优化输出.这可能会明显多占用时间,所以只有在生成最后的文件
- d1 k% |4 Y }" {# F时使用.
f$ ^1 y- X5 r! Q+ T$ E/ m* t/ E8 T; d$ _' S: {4 o
`-q'5 P& h1 O5 H* M2 u9 |* k* _5 w& J; H
`--emit-relocs'
2 t- U0 j7 r2 D5 Y把重定位节和内容留在完全连接后的可执行文件中. 连接分析和优化工具可能需要这些信息用来进行8 r5 k9 }! _8 X" m' U5 e; `. W( ]
正确的修改与执行. 这在大的可执行文件中有用.
2 E2 S% o; I9 p! a! M' g
7 d( ]- D7 p- F2 m这个选项目前只支持ELF平台.
b; z7 Y' m+ m: g7 B5 C& G`-r'6 p3 ^8 O/ ?2 N
`--relocateable'% S i, U1 X6 g0 w. v5 _
产生可重定位的输出, 比如,产生一个输出文件它可再次作为'ld'的输入.这经常被叫做"部分连接".
$ F9 r' J6 _ I+ r* C' G作为一个副作用,在支持标准Unix魔数的环境中,这个选项会把输出文件的魔数设置为'OMAGIC'. 如
+ L2 a0 J/ P: a4 P, w" C5 g; q果这个选项没有被指定,一个绝对文件就会被产生.当连接C++程序时,这个选项就不会解析构造函数的1 Y7 j) r4 \) R6 r9 S% v
引用;要解析,必须使用'-Ur', X# r) q) n8 l4 s& O2 u
: W5 Y8 ~; [/ m" t$ j
如果输入文件跟输出文件的格式不同,只有在输入文件不含有重定位信息的时候部分连接才被支持.输
9 U$ M7 B3 w9 i出格式不同的时候会有更多的限制.比如,有些'a.out'的格式在输入文件是其他格式的时候完全不支" ^# _" R+ ^8 [- E0 K* g' d* q
持部分连接.& W# e$ m! U6 q$ K8 z
9 X5 i; h) \5 B# U" P& i+ z这个选项跟'-i'等效.( K" L: h# }1 v" F& z; A0 P9 i
+ w) i1 L( \% i" ``-R FILENAME'3 y% \6 x" ]. _7 l6 [
`--just-symbols=FILENAME'
" d- q2 C1 v: {从FILENAME中读取符号名跟它们的值,但不重位这个文件,或者根本不把它包含在输出文件中.这就允
" P2 T& `8 i5 r许你的输出文件引用其它程序中定义的绝对内存地址.你可以多次使用这个选项.
0 `, _8 l' _# f o
7 Q" ~( g" V2 R1 {" g* \为了跟其他ELF连接器兼容,如果'-R'选项后面跟有一个目录名,而不是一个文件名,它会被处理成/ X7 A, |! O# _
'-rpath'选项.- F# t. s9 C- V' E$ V' N
( Y6 E; z" U, P
`-s'+ F" b3 }2 z/ I! c8 b; w
`--strip-all'
- H& X+ o; u5 n+ N. |忽略输出文件中所有的符号信息.
; ~+ q) u9 I( B8 ]6 W( d {- w
; q; Q8 |" T: w1 V" j8 D`-S'5 D3 V% M. Y/ _1 u
`--strip-debug'. x" ?4 ]1 T3 B: S5 }
忽略输出文件中所有的调试符号信息(但不是所有符号).. d" H# r, _* v- o; L
- u; ?5 P" ]" Y B6 y/ R`-t'
" V v+ v* \5 ~`--trace'
- Z3 U: Z3 c0 A0 I, q8 o( r" `* y打印'ld'处理的所有输入文件的名字.9 ]- @" m# D" S% `. N
0 I! \. Q$ J% D! r
`-T SCRIPTFILE'
0 t$ H" @+ N& X, }( w" ~`--script=SCRIPTFILE'0 k7 K: ~: \. G* K
把SCRIPTFILE作为连接脚本使用. 这个脚本会替代'ld'的缺省连接脚本(而不是增加它的内容),所以
) `& f$ v! O7 S" q2 S命令文件必须指定所有需要的东西以精确描述输出文件. 如果SCRIPTFILE在当前目录下不存在,'ld'
- T, s0 P4 ^; |0 ^& Y; l& c, S会在'-L'选项指定的所有目录下去寻找.多个'-T'选项会使内容累积.* B0 K& J- Z( V
+ R" u1 }' F$ B5 [: {& L4 W
`-u SYMBOL'3 W+ I$ y6 V5 J$ { B
`--undefined=SYMBOL'/ }, q- r/ L( X
强制SYMBOL在输出文件中作为一个无定义的符号被输入.这样做会有一些效果,比如,会引发从标准库- `: e1 H1 k* G$ u, e
中连接更多的模块. '-u'可以以不同的参数反复使用,以输入多个无定义的符号.这个选项跟连接脚
8 z4 f* @! B9 W7 ^本命令中的'EXTERN'是等效的. @9 {, N! ]& [" j' V
4 |, U8 i5 E4 S* l. o2 J! s6 N& `: J0 P`-Ur'! I) `$ P0 Y/ G3 n
对于不是C++的程序,这个选项跟'-r'是等效的: 它产生可重定位的输出,比如,一个输出文件它可以再: `# d F% S3 I9 N0 O) U
次作为'ld'的输入. 当连接C++程序时,'-Ur'解析构造函数的引用,跟'-r'不同. 但如果在一些用'-Ur'
! o. M ]8 [0 U连接过的文件上再次使用'-Ur',它不会工作,因为一旦构造函数表被建立,它不能被添加内容.请只在
* X; C" P4 S# u4 t7 s最后一遍连接的时候使用'-Ur', 对其它的,只使用'-r'.
5 w. d! p) [. V) Z) q* x2 b1 A3 r
`--unique[=SECTION]'+ \3 v; v1 }8 |# q3 j
对于所有匹配SECTION的输入节,在输出文件中都各自创建单独的节,或者,如果可选的通配符SECTION/ n5 h2 o2 E* e2 L+ N8 I
参数丢失了,为每一个孤儿输入节创建一个输出节. 一个孤儿节是一个连接脚本中没有指定的节.你! J& N! K$ H4 i' O# b1 e
可以在命令行上多次使用这个选项; 它阻止对同名输入节的合并,在连接脚本中重载输出节分配.
9 J/ ~$ T8 F% n1 m# k
* t6 G! z- E& g2 R! |`-v'( |3 `, N L* X
`--version'
2 M0 V9 g0 M1 T3 Z, ^`-V'! T0 f# _( D1 J3 v
显示'ld'的版本. '-V'选项同时会列出支持的模拟器.
- J/ @. y$ `) N/ A4 l
% K# N* T, M, h k; n6 ``-x'7 ?1 z) c7 g& l
`--discard-all', n/ k% V+ \' B8 Y' q1 F! f
删除所有的本地符号./ [1 y! J6 k9 y+ F! t5 w
' d6 @3 b1 u7 _0 Y! M`-X'9 M' \! e/ y4 d0 l4 y
`--discard-locals'
2 ^; h/ b% G0 C/ @' G删除所有的临时本地符号.对于大多数目标平台,就是所有的名字以'L'开头的本地符号.
; S# x, h5 e1 @+ {& R9 U
, @" f3 o+ ~5 C7 o% E% i`-y SYMBOL'
5 r/ {' r) K( X! @`--trace-symbol=SYMBOL'
, W# z; D4 X! f3 S9 z8 V9 n) ]打印出所有SYMBOL出现的被连接文件的名字. 这个选项可以被多次使用. 在很多系统中,这在预先确定底6 M/ C5 v% v% z/ |+ D, m4 ]
线时很有必要.
" r5 F) x' k3 \- H" J! j$ f* @
# ~' v3 e; E" {$ {8 k+ S; K当你拥有一个未定义的符号,但不知道这个引用出自哪里的时候,这个选项很有用.- f6 s' C8 @2 j/ Y- [
/ Z8 k4 ?6 t7 X. A`-Y PATH'
/ i& w+ Z/ w6 o8 b为缺省的库搜索路径增加一条路径.这个选项是为了跟Solaris兼容.
+ j. A. O! @ m }- I
, a: K9 r' J ~0 W% z6 G`-z KEYWORD'
& ~$ @4 x5 _, Q- Y( S* j! P能被识别的关键字包括'initfirst', 'interpose', 'loadfltr',`nodefaultlib', `nodelete',
' y9 U2 O, Y% k8 \$ ~* @9 Z`nodlopen', `nodump', `now', `origin',`combreloc', `nocombreloc' and `nocopyreloc'. 为了跟
! N) j( v7 P1 X v: b# ~- U4 nSolaris兼容,所有其它的关键字都被忽略. 'initfirst'标志一个对象,使它在运行时,在所有其他对象之
4 I, z( n1 J7 @" _% r前被初始化. 'interpose'标志一个对象,使它的符号表放在所有其他符号之前,作为主要的执行者.
' F$ b9 e% |5 l( n# l3 e- S6 n4 v'loadfltr'标志一个对象, 使它的过滤器在运行时立即被处理.'nodefaultlib'标志一个对象,使在搜索$ a9 W- b& I& [
本对象所依赖的库时,忽略所有缺省库搜索路径. 'nodelete'标志一个对象,使它在运行时不会被从内存4 \9 E$ g1 h" S% N
中删除.'nodlopen'标志一个对象,使这个对象不可以通过'dlopen'载入.'nodump'标志一个对象,使它不能
' d7 y+ D, k$ s: E: j6 S被'dldump'转储. 'now'标志一个对象,使它成为非懒惰运行时绑定对象. 'origin'标志一些可能含有* H1 V" n. w5 \2 V
$ORIGIN的对象,'defs'不允许无定义符号. 'muldefs'允许重定义. 'comberloc'组合多个重定位节,重新% I2 c9 f/ b {
排布它们,让动态符号可见. 'nocomberloc'使多个重定位节组合无效. 'nocopyreloc'使重定位拷贝后的* \* j# i. [! @9 k0 E1 Z! S: E
结果无效.4 n2 h# R) Q1 z( Z
" o9 W: ?5 p, B% S$ t. k`-( ARCHIVES -)'
9 e# D6 G5 ?/ @`--start-group ARCHIVES --end-group'3 G1 B( g7 V4 I# v) {, ~% x3 v/ Z# L
ARCHIVES应当是一个关于档案文件的列表. 它们可以是显式的文件名,或者'-l'选项.$ e$ T# Q- [4 r& x
# c1 X5 d( v+ F
这些指定的档案文件会被多遍搜索,直到没有新的无定义引用被创建. 通常,一个档案文件只会被搜索一4 N6 i" L7 a) m, j' B; R
次. 但如果这个档案文件中的一个符号需要被用来解析一个档案中的目标引用到的无定义的符号,而这个
9 @+ ?# E, M$ u% i# \: ~符号在命令行上的后面某个档案文件中出现, 连接器不能解析这个引用. 把这些档案文件分组后,它们都! P% @7 s" a0 w3 M
可被反复搜索直到所有可能的引用都被解析了为止.
' u& _4 p5 k" M6 T0 m. s- [) ^: V" ^0 i' o; Y- X: U
使用这个选项有一个很大的运行开销. 只有在无法避免在多个档案文件中使用循环引用时才用它.$ t- }3 Y4 |" I7 t
6 h: \) o* T) \2 C6 w
`--accept-unknown-input-arch'
" }" W% S( e$ i" j w& \`--no-accept-unknown-input-arch'0 q) l1 ?5 A" f+ a6 h
告诉连接器接受那些架构不能被识别的输入文件. 但前提假设是用户知道他们在做什么,并且是故意要连
2 g" {0 @/ U7 m) A接这些未知的输入文件. 在版本2.14之前,这个是连接器的缺省行为. 从版本2.14以后的,缺省行为是拒
' k4 t1 k$ m5 f. d& R4 ~绝这类输入文件, 所以`--accept-unknown-input-arch'选项被用来恢复旧的行为.! S( }2 `% v7 f6 R, w Q
+ b. |$ l, E4 g+ \" x5 A
`-assert KEYWORD'# `6 n0 ?- R/ V& Z6 _
这个选项被忽略,只是用来跟SunOS保持兼容.; s% n% M: i8 V
8 y7 g0 Y$ ^3 |) `, N {" J
`-Bdynamic') q& P5 d) F+ }5 U
`-dy'3 F% F* \/ V9 Q4 Y1 }9 h
`-call_shared'/ s! Y# A" u7 ~5 c7 V
连接动态链接库. 这个仅仅在支持共享库的平台上有用.在这些平台上,这个选项通常是默认行为. 这个选
3 E% n$ V8 B5 d; Q6 B项的不同形式是为了跟不同的系统保持兼容. 你可以在命令行上多次使用这个选项:它影响紧随其后的'-l'
- G D' k/ a l+ F& z! b3 e! q; s选项的库搜索.
8 I+ W) T3 @- X$ `9 |+ o3 {7 v; q# H) D& y9 w* @
`-Bgroup'2 ]0 u! Y7 v; B
在动态节的'DT_FLAGS_1'入口上设置'DF_1_GROUP'标志.这会让运行时连接器在处理在这个对象和它的相
2 }9 P1 c: f& V( O# y7 @( [关部分搜索时只在组中. '--no-undefined'是隐式的. 这个选项只在支持共享库的ELF平台上有用.
/ }4 p+ c. b3 Y& H, t+ g" g
, a' v/ Y) e8 `1 Y`-Bstatic'
2 E$ t6 \# o/ g y`-dn'; t2 c+ m; T6 o; j2 s" K! b8 v4 |5 e7 U
`-non_shared' d h) t2 T0 `' y% f+ b
`-static'7 Y) @* U6 a: X
不连接共享库. 这个仅仅在支持共享库的平台上有用. 这个选项的不同形式是为了跟不同的系统保持兼
9 L( O: Y% O; O' Q7 t& D容. 你可以在命令行上多次使用这个选项:它影响紧随其后的'-l'选项的库搜索.: [5 D7 Z5 { q! d' J' Z1 c. C1 T
+ j+ o" Z" S1 T) u
`-Bsymbolic'- E l# k9 h. M. R. d3 C( c. b
当创建一个共享库时, 把对全局符号的引用绑定到共享库中的定义(如果有), 通常, 一个连接共享库的程
2 R0 g+ B4 R+ V6 Y0 q" r- s+ k序重载共享库中的定义是可能的. 这个选项只在支持共享库的ELF平台上有用.
6 v+ E& S3 z% W! H2 G+ ?) [ y' o4 j; e
`--check-sections'# C: H2 C/ I) |9 G5 d
`--no-check-sections'- w6 X1 E+ W( D0 I0 C. j
让连接器在节地址被分配后不要去检查节地址是否重叠.通常,连接器会执行这种检查,如果它发现了任何& S1 ]/ U4 B8 T/ r
重叠,它会产生相应的错误信息. 连接器知道也允许节的重叠. 缺省的行为可以使用命令行开关
- H2 \4 c$ j0 P`--check-sections'来恢复.; p0 |+ \. e+ y
& ` g. I4 H# n9 K`--cref'
4 ^' _2 x7 v( K/ `. V6 u2 K输出一个交叉引用表. 如果一个连接器位图文件被产生, 交叉引用表被打印到位图文件. 否则, 它被打印
& R T# h7 f1 e( o2 U7 m到标准输出.
8 ^" h' H7 j! a7 _: ]3 z8 L. c s2 ?3 \# e9 |5 A
表的格式相当的简单, 所以,如果需要,可以通过一个脚本很轻易地处理它. 符号是以名字被打印输出,存
& X% ^7 }3 L' s1 r% x储. 对于每一个符号,给出一个文件名列表. 如果符号被定义了, 列出的第一个文件是符号定义的所在.
# k8 N6 d+ V3 n9 v) z接下来的文件包含符号的引用.
) p" A( c! j, U5 p& W9 R. L& ^, ?5 w' y/ \1 n
`--no-define-common'" a2 c% ~ K* J* \& j! Z
这个选项限制对普通符号的地址分配. 脚本命令`INHIBIT_COMMON_ALLOCATION'具有同等的效果.
1 t9 b; V5 F* [: m9 ^6 m; Y: v. h0 `" J' s# E3 L, a0 s
`--no-define-common'选项允许从输出文件的类型选择中确定对普通符号的地址分配; 否则, 一个非重定% G- u; G( I) B6 X4 {: A
位输出类型强制为普通符号分配地址. 使用'--no-define-common'允许那些从共享库中引用的普通符号只# ~: x$ ]& p1 f8 \3 a* o
在主程序中被分配地址. 这会消除在共享库中的无用的副本的空间, 同时,也防止了在有多个指定了搜索
1 U' w0 B* x! v2 n9 R路径的动态模块在进行运行时符号解析时引起的混乱.8 C2 }$ y8 d: w
9 U0 T# |% B$ g1 y" N( ^6 O( D`--defsym SYMBOL=EXPRESSION'
; a9 j. ~1 i$ r在输出文件中建立一个全局符号,这个符号拥有一个EXPRESSION指定的绝对地址. 你可以多次使用这个选8 Z/ w* n: ]& |. q5 O# R
项定义多个符号. EXPRESSION支持一个受限形式的算术运算:你可以给出一个十六进制常数或者一个已存1 w1 a; b K* ^% M8 x. r; `
在符号的名字,或者使用'+'和'-'来加或减十六进制常数或符号. 如果你需要更多的表达式,可以考虑在脚4 n& p6 I; H' e' D
本中使用连接器命令语言, 注意在SYMBOL,=和EXPRESSION之间不允许有空格.
5 u" b" }& y7 Z+ T3 E/ Y* C6 b8 I( e/ x
`--demangle[=style]'
' o5 \6 O% A+ e5 V. |& P1 V`--no-demangle'
& i* C' \1 E6 I0 G0 _( V* z: \这些选项控制是否在错误信息和其它的输出中重组符号名. 当连接器被告知要重组, 它会试图把符号名以
! L) U T: M$ p, B/ m' |( C. X) j一种可读的形式的展现: 如果符号被以目标文件格式使用,它剥去前导的下划线,并且把C++形式的符号名
: j9 w' q5 ]' q7 i5 g/ U2 m转换成用户可读的名字. 不同的编译器有不同的重组形式. 可选的重组形式参数可以被用来为你的编译器% ]! x; |3 i" {" { D
选择一个相应的重组形式. 连接器会以缺省形式重组直至环境变量`COLLECT_NO_DEMANGLE'被设置. 这些
. ^" w6 H" t; k [5 I/ ]: e c0 q选项可以被用来重载缺省的设置. C9 k: C' F4 Q# o
, D+ l/ `* \: ^`--dynamic-linker FILE'
1 B! Q8 E+ P) j: p( J$ E; r设置动态连接器的名字. 这个只在产生动态连接的ELF可执行文件时有效. 缺省的动态连接器通常是正确
C" P: K2 u7 K1 Z. W0 m( w的; 除非你知道你在干什么,不要使用这个选项.
( G O+ }3 x" b6 U e7 m! d, h
; ]. ~7 i% w( E& P`--embedded-relocs'
5 y1 E9 l t$ v' {& `9 i这个选项只在连接MIPS嵌入式PIC代码时有效, 这些代码必须是由GNU的编译器跟汇编器通过-membedded-pic
) I2 G5 {9 M6 _; V" Z选项生成的. 它导致连接器产生一个表,这个表被用来在运行时重定位所有的被静态初始化为指针值的数1 f/ E/ v% K. Y
据. 7 h/ F G/ ]& W+ [
3 ~5 V* A" i& h: K+ L* o5 Z3 X
`--fatal-warnings'
: T* c- G5 J8 D2 L4 F! N! V1 u把所有的警告视为错误.4 e+ {0 w5 q) [8 K: {: _
) `6 E9 @3 @( R- E`--force-exe-suffix'
7 ^& }. X {8 ?5 R# X确保输出文件有一个.exe后缀.
" f0 H" l- ^ q5 y
4 ~/ _; |. s) w如果一个被成功完整连接的输出文件不带有一个'.exe'或'.dll'后缀, 这个选项确保连接器把输出文件 `2 l9 l, X% V- b* z" F- L! i
拷贝成带有'.exe'后缀的同名文件. 这个选项在使用微软系统来编译未经修改的Unix的makefile时很有
6 |9 |2 C7 h+ X+ h6 l9 G用, 因为有些版本的windows不会运行一个不带有'.exe'后缀的映像.
: u# `0 f1 u8 b; ]/ _' O! v) E' F( w! i
`--no-gc-sections') v5 i( S1 u- T0 F7 t
`--gc-sections'
+ C( j; V- Z; M8 s( w允许对未使用的输入节的碎片收集. 在不支持这个选项的平台上,被忽略. 这个选项不能跟 '-r'选项共存
# U1 b+ i* N) a4 n. I7 P0 @也不能被用来进行动态连接. 缺省行为可以用`--no-gc-sections'进行恢复.
0 q2 {% T8 q, p3 B5 i3 p1 I( n
0 _& Q8 ^# z+ G6 C2 d$ D`--help'
" h& d/ H" u! k) R# r p! N在标准输出上打印一个命令行选项概要,然后退出.
/ f! Q) `4 I' e9 q8 t7 G+ i. [: B0 \( W
`--target-help'6 b/ |% a6 g6 c- G
打印一个所有目标平台相关的选项的概要,然后退出.
+ V* n6 r8 b6 _7 H8 \$ Z/ J3 e( }4 X7 ` [2 s6 M9 y
`-Map MAPFILE'8 D( |8 a( I; E- ^; b
打印一个连接位图到文件MAPFILE中. 参阅上面关于'-M'选项的描述.. v) l* Q7 Z/ W3 O
+ J/ q( s% @8 l0 f`--no-keep-memory'
, \$ o1 u* Y7 n3 \" V/ V'ld'通常会以速度优先于内存使用的方式优化程序,这是通过把输入文件的符号表放在内存缓冲中实现的,+ t/ c' x, B0 t2 o
这个选项告诉'ld'以内存使用优先来优化, 尽可能的减小符号表的重读. 这在'ld'在连接一个大文件时
1 N8 a8 _8 V/ e, y" k* o超出内存限制时有用.
# z3 s3 K4 o: y; n* A. K, E# m$ e8 G& w6 m
`--no-undefined'
9 X, ^# E5 v1 u: Y`-z defs'
' w) H7 Q2 `$ f: B1 |通常,当创建一个非符号共享库时, 无定义的符号允许出现,并留待运行时连接器去解决. 这个选项关闭这
: L1 o( E+ Q) S- o1 s* x样的无定义符号的使用. 开关`--no-allow-shlib-undefined'控制共享对象被连接进共享库时的行为.# R3 N# z2 | u2 M
6 r; R' w/ U8 e`--allow-multiple-definition'. v% x7 z. \9 @
`-z muldefs'
+ n$ A8 c9 N/ `3 X( J7 G通常,当一个符号被定义多次时, 连接器会报告一个致命错误. 这些选项允许重定义并且第一个定义被使* M) K2 F" f; {1 _ H7 T8 C' _
用
7 V' t& T) N0 c; |6 I$ K6 Z) j8 ^
& n* c1 J2 \. }0 \3 }. B+ t
2 m6 j& E0 M, {! N$ p`--allow-shlib-undefined': A5 f, M, u; v( b% C0 Z
`--no-allow-shlib-undefined'
: I4 b0 j- l3 ~( b允许(缺省)或不允许无定义符号存在于共享对象中. 这个开关的设置会重载'--no-undefined',这里只关. ?" T) y5 O' u
注共享对象. 这样,如果'--no-undefined'被设置,但'--no-allow-shlib-undefined'未被设置, 连锁反应
7 q! Q( R% T1 h% ]" C/ Y8 b7 |是存在于规则对象文件中的无定义的符号会引起一个错误,但是在共享对象中的未定义的符号会被忽略.
; ?. w! p& K3 [- ]/ j3 H/ H/ e. g% x' u, i0 X6 t
把`--allow-shlib-undefined'设置为缺省的原因是在连接时指定的共享对象并不一定是载入时可载入的( Z2 r5 m6 f1 V" A4 c" W
那个,所以,符号可能要到载入时间才被解析.
/ P9 x c: S. A* ^* O
3 f6 z# [9 h- W0 \/ B`--no-undefined-version'4 U: w' |: v# L1 p4 ?
通常当一个符号有一个未定义的版本时,连接器会忽略它. 这个选项不允许符号有未定义的版本,并且碰: e+ T" k( w0 s1 _. @7 k/ b
到这种情况,会报告一个严重错误.& W: Y' w# m% Y" P7 [ g4 I
! s, q" s, O; t3 l9 ``--no-warn-mismatch', p9 j6 t b9 H6 M- ^, I
通常, 如果你因为一些原因,企图把一些不匹配的输入文件连接起来的时候,'ld'会给出一个错误,可能这
- i" B: R; q. t) E9 Z5 F些文件是因为由不同的处理器编译. 这个选项告诉'ld'应当对这样的错误默认允许. 这个选项必须小心$ k1 r: m: i# G3 M+ l1 Q# f
使用.
3 s0 q5 e, [# Z
$ d* M) R' @) ?: U7 h9 Z`--no-whole-archive'5 n0 Q. Z: O6 P* k4 G: l- o% N4 c _
为后面的档案文件关闭'--whole-archive'选项的影响./ Q- j( U5 r' h; x
5 P2 k1 ~ E( J`--noinhibit-exec'
& ?0 r8 Y5 u2 Z0 x+ b0 x当一个可执行文件还可以使用时,就保留它. 通常,连接器如果在连接过程中遇到了错误,就不会产生输出
( l: j! `9 x2 F, p文件;当它遇上错误时,它会退出而不写输出文件.
# k: j- M1 P0 ^) c$ J* m/ s3 @9 |: V7 j3 M' ] B
`-nostdlib'( p# j, v8 c' K8 c$ f* a
仅搜索那些在命令行上显式指定的库路径. 在连接脚本中(包含在命令行上指定的连接脚本)指定的库路
c1 }, P( v9 p' N* y* `( _径都被忽略.
9 N5 Y8 J' L7 z$ s2 j, K- J# X9 d k3 d, U& T9 Q$ w
`--oformat OUTPUT-formAT'
+ ^' r( \+ P& p& u1 u, w: v'ld'可以被配置为支持多于一种的目标文件. 如果你的'ld'以这种方式被配置,你可以使用'--oformat'0 I C$ \: t0 ]' e" h" `
选项来指定输出目标文件的二进制格式.就算'ld'被配置为支持多种目标格式,你也不必指定这个项,因
& d1 y$ Y, b0 e% ]为'ld'应当被配置为把最常用的输出格式作为默认格式. OUTPUT-formAT是一个文本串,是被BFD库支持
X/ ^: Y6 u a的一个特定格式的名字.脚本命令'OUTPUT_formAT'也可以指定输出格式,但这个选项可以覆盖它.1 H. u% Z7 v5 M9 ?) l9 C$ ]
) K! y/ G; d4 T& e
`-qmagic'
5 e+ s4 I! X1 a6 M/ @" S) ~这个选项被忽略,只是为了跟Linux保持兼容.+ [( |" |2 n& U0 d6 Z& x2 o( v
; {1 ~4 [( _- K( f2 {! s! B I`-Qy'% V5 Q6 r+ x$ K! k( J
这个选项被忽略,只是为了跟SVR4保持兼容.. j9 `) J R, E
, L( _3 }; B& b`--relax'
) l s+ S: o( n9 k; O一个机器相关的选项. 只有在少数平台上,这个选项被支持. 8 Q. P/ t$ d" [- [" b* z
0 b3 T4 c- F* n- p% O在某些平台上,'--relax'选项在连接器解析程序中的地址时执行可能的全局优化, 比如松散地址模式和在输出文件! d' ~0 T) f$ X- m6 u* ?
中合成新的指令.
3 u6 O6 h5 D: z
2 Q2 S$ K- [/ K3 }2 y" w在某些平台上,连接时全局优化会进行符号调试导致程序不能运行.
) U7 `4 |* w7 \& r* x+ F# Q2 q; u5 h% v. t
在不支持这个选项的平台上,'--relax'被接受,但被忽略.
( }" j- E! R+ D+ j) u# b% v% b
7 n0 |. N1 N, F/ D4 E( ^/ o`--retain-symbols-file FILENAME'/ Y/ b5 O! _2 j' T) m$ E: `) v
只保留在FILENAME中列出的那些符号,丢弃所有其他的. FILENAME是一个简单地平坦模式文件, 一个符号占一行.
7 X6 Z2 i! K2 {9 ^0 u这个选项在那些会逐步积累起一个大的全局符号表的系统中(比如 VxWorks)会很有用,它能有效地节约内存空间.
' i3 b- @5 W0 |3 E( b
& H6 |! C. Z; \" B% n" I+ p'--retain-symbols-file'不丢弃未定义的符号,和需要重定位的符号.
/ z+ r/ J' N& \0 \4 c) N& `/ n2 {& ` b1 K H x' d2 j$ v* ] T
你可能在命令行上只指定'--retain-symbol-file'一次, 它覆盖'-s'和'-S'的功能.4 ?- g3 f" l! w' Q
' O) k. P) X! H; N: ?
`-rpath DIR'7 R. z& Y3 ?1 \* m$ j! ?9 D; p# e
为运行时库的搜索路径增加一个目录. 这个在连接带有共享库的ELF可执行文件时有用. '-rpath'的所有参数会被# c" f: x' z6 j! ?
连接起来传递给运行时连接器, 运行时连接器在运行时用它们定位共享对象. '-rpath'选项在定位那些在连接参数
6 i, m, N4 X; l \; S0 L- K' R! S% E, H指定的共享对象需要的共享对象时也很有用; 参阅关于'-rpath-link'选项的描述, 如果在连接一个ELF可执行文件8 f9 v( e% u4 v1 g
时不使用'-rpath'选项,那些环境变量'LD_RUN_PATH'选项就会被使用.
+ `/ t i+ T; e( r2 S
1 Q/ P4 i3 L" X8 y! ~5 P2 k. L* U'-rptah'选项也可以使用在SunOS上. 缺省地,在SunOS上,连接器会从所有的'-L'选项中形成一个运行时搜索路径.2 y0 U/ u; C4 g! V4 G4 G
如果使用了'-rpath'选项, 那运行时搜索路径就只从'-rpath'选项中得到, 忽略'-L'选项. 这在使用GCC时非常有
: r, }5 e% P+ k) v4 m用, 它会用上很多的'-L'选项,而这些路径很可能就是NFS挂上去的文件系统中.) |* ^7 e$ B8 f/ F( u( |. E2 b
9 E6 J8 J# d9 m, w! _; K2 r- ^为了同ELF的连接器兼容, 如果'-R'选面后面跟有一个目录名, 而不是一个文件名,那它也会被处理成'-rpath'选
/ j6 \0 n; V, t4 p项.
1 ^' b8 ] a, j" L5 H& \% _: E& V+ A8 F1 Z
`-rpath-link DIR'
% m1 M! T+ T) j9 n# E当在SunOS上使用ELF时,一个共享库可能会用到另一个共享库. 当'ld -share'把一个共享库作为一个输入文件连接
$ R1 v, o1 z8 X时就有可能发生这种情况.. y4 c( j& u3 Y1 B8 o; V
' ]; j( k+ n' x5 T
当一个连接器在作非共享,不可重定位连接时,如果遇上这种依赖情况,它会自动定位需要的共享库,然后把它包含在
% g1 _* _7 e- p4 q z连接中, 如果在这种情况中,它没有被显式包含, 那'-rpath-link'选项指定优先搜索的一组路径名.
* _+ g" a+ [" C$ N- o8 E7 ]/ b5 G( j) \& _" d3 }) H0 l
这个选项必须小心使用,因为它会覆盖那些可能已经被编译进共享库中的搜索路径. 在这种情况下,它就有可能使用
5 a& f+ S }& @9 D8 Y/ n一个非内部的不同的搜索路径.
. f4 @+ D$ N2 {8 L4 K+ i0 P, P4 @0 B+ Q2 Z3 d
连接器使用下面的搜索路径来定位需要的共享库:- G i6 P0 X, l& F& F0 n; [
U% F5 B+ t5 S5 W
1. 所有由'-rpath-link'选项指定的搜索路径.
4 ~8 R+ E/ w7 H% W0 e
, w! @4 B$ @# H5 o 2. 所有由'-rpath'指定的搜索路径. '-rpath'跟'-rpath_link'的不同之处在于,由'-rpath'指定的路径被包含在可; Z; o7 M2 ?4 ^" H$ s m. d. \
执行文件中,并在运行时使用, 而'-rpath-link'选项仅仅在连接时起作用. 它只用于本地连接器.
, N3 c; V8 o, [5 G$ @+ f
$ [4 o% }% y7 ] v1 H6 u6 y( P4 y1 q 3. 在一个ELF系统中, 如果'-rpath'和'rpath-link'选项没有被使用, 会搜索环境变量'LD_RUN_PATH'的内容.它也只9 n' w$ J: q1 E; t$ y
对本地连接器起作用.
" l2 v+ ~8 s G, w- L
. ]8 b1 Z& z" I" H, r 4. 在SunOS上, '-rpath'选项不使用, 只搜索所有由'-L'指定的目录.8 ^7 Q& F, \+ k; S: ^/ Y' H
0 @ q- h( q. e 5. 对于一个本地连接器,环境变量'LD_LIBRARY_PATH'的内容被搜索.
J) i& g( a2 }6 p4 S! f
% A' D! T t; [& X# a( B* ~ 6. 对于一个本地ELF连接器,共享库中的`DT_RUNPATH'和`DT_RPATH'操作符会被需要它的共享库搜索. 如果'DT_RUNPATH'
% L, v- Y$ O& U$ v 存在了, 那'DT_RPATH'就会被忽略.* E' J' r% C* X4 [7 L
8 }4 j: B. Y# V& h. u* z* h7 c 7. 缺省目录, 常规的,如'/lib'和'/usr/lib'.: d3 T: h' s. [1 M9 z+ K( _$ b
6 z* s. X- }. J1 v' l- e
8. 对于ELF系统上的本地连接器, 如果文件'/etc/ld.so.conf'存在, 这个文件中有的目录会被搜索.
/ Q: G# b9 I4 y) A c0 x, ^+ {6 t# z" N( |: p
如果需要的共享库没有被找到, 那连接器会发出一条警告信息,并继续执行连接.5 Y. U9 }5 M1 ]( `2 u' U3 q9 r
2 e6 l9 ~; S/ F. S
`-shared'
8 M5 t4 b$ `" q3 _, F% l`-Bshareable'
: _% ~8 g! w/ B) q1 y创建一个共享库. 这个选项只在ELF, XCOFF和SunOS平台上有用。 在SunOS上,如果'-e'选项没有被使用,并在连接
: `* R; m8 {7 d& t" [- | r' W& K* l中有未定义的符号,连接器会自动创建一个共享库,
# Y+ a, ^2 ]/ g; u; r! ?7 u$ N6 d/ ]0 x( y+ n3 c I
`--sort-common'( T+ J3 s# R/ ^
这个选项告诉'ld'当它把普通符号放到相应的输出节中时按大小进行排序。排在最前面的是所有的一字节符号,然% D, P2 Q- O* C! @
后是所有的二字节,然后是所有的四字节, 然后是其它的。 这是为了避免因为对齐约束而在符号间产生的断裂; V9 R( a9 P/ A+ H! p8 S$ Z
J' f) G' O0 p3 C; i, B" `6 v`--split-by-file [SIZE]'
$ }) @7 l- Z2 o" H8 }1 l. t跟'--split-by-reloc'相似,但在SIZE达到时,为每一个输入文件创建一个新的输出节。如果没有给出,SIZE缺省6 |9 V- n% C% M: x; | T
地设置为19 C, W, ?1 T& p7 P" k
, i( Y1 _* |1 I7 I' g
`--split-by-reloc [COUNT]'
7 S$ v& e2 a, ?试图在输出文件中创建节外的节,这样就没有单个的输出节含有多于COUNT个重定位符。这在产生巨大的用于COFF格
1 p6 u# }0 V' n2 `6 Z8 {6 `式的实时内核的可重定位文件时非常有用;因为COFF不能在一个节中表示多于65535个重定位。 注意,这在不支持
" F8 W; a% d. a4 i/ S1 h; A/ ]专有节的目标文件格式中会失败,连接器不会把单个输入节分割进行重分配, 所以,如果单个输入节含有多于COUNT
k- ` W1 w7 V; ]个重定位符, 那一个输出节会含有同样多的可重定位符。COUNT缺省被设为32768.
: k+ O6 V1 }2 X* g1 T& g" I V! e3 o+ `
`--stats': p1 e* s% C1 f) z2 i$ Q6 U; D
计算并显示关于连接器操作的统计信息, 比如执行时间,内存占用等.4 @& q- H3 K+ ~3 \7 U2 G6 k
; m" \! [/ X0 U5 f X`--traditional-format'9 K7 o( R M6 V, r' r
对于某些目标平台, 'ld'的输出会跟某些面有的连接器的输出有所不同. 这个开关要求'ld'使用传
+ H. [( d: y9 N统的格式.
2 n' S. P. s6 r/ O9 k, b
, E" J& a2 L) Z/ U; w0 A: l/ b- ?/ Y比如, 在SunOS上, 'ld'会把符号串表中的两上完全相同的入口合并起来. 这可以把一个带有调试信息
( ~/ A# I! p+ K/ m/ Y的输出文件的大小减小百发之三十. 不幸地是, SunOS的'dbx'程序不能读取这个输出的程序(gdb就没) C" ~+ U) A" Z9 a! z1 u% ], Y3 G
有问题).'--trafitinal-format'开关告诉'ld'不要把相同的入口合并起来.
, h/ a9 i- y- B! b7 B1 i! U: A% R6 _' T* \, j. ^
`--section-start SECTIONNAME=ORG'8 u& r# R# J# a m) l" E5 ^# {
通过指定ORG, 指定节在输出文件中的绝对地址. 你可以多次使用这个选项来定位多个节. ORG必须是
3 R4 U) m, y, H一个十六进制整数; 为了跟基他的连接器兼容,你可以忽略前导'0x'. 注意,在SECTIONNAME,等号,ORG
* @1 X7 N3 m+ H" h! `- O之间不允许有空格出现. s/ l" \4 q0 t# k# K
1 H; e2 X5 Y s' j' M
`-Tbss ORG'
1 Q7 D) D# U" B, i`-Tdata ORG'
% ]4 O" l8 l) C1 o7 N( U( T`-Ttext ORG'0 n8 X* a$ w/ t4 Y2 g/ F
跟-section-start同义, 不过把SECTIONNAME替换为'.bss', '.data'或'.text'.
% G' ]! L! A5 v, q# m
2 L; |5 I6 B# O$ f9 Y2 H4 ?`--dll-verbose'
# j5 [$ T# x" o8 N8 s`--verbose'$ ?, e. j, B" I3 T9 c, |: y1 C
显示'ld'的版本号,并列出支持的连接器模拟. 显示哪些输入文件能被打开,而哪些不能. 显示连接器
) x/ k$ C$ _ Q0 `- L. R使用的连接脚本.. C- J6 T5 n% D, H
. w! W; I4 {6 Z) x; E
`--version-script=VERSION-SCRIPTFILE'
1 n. X4 b6 u, N/ `8 x2 S指定连接器的脚本的版本名. 这个常在创建一个需要指定附加的关于版本层次的信息的共享库时使用,
! i6 K, O0 b: P' i/ i# P7 L这个选项只有支持共享库的ELF平台上有效.4 ^- j5 ]2 Z# p
1 v. P/ g( R6 t7 Y. A6 l
`--warn-common'
# g7 N% H& O# a( ^当一个普通符号跟另一个普通符号或会号定义合并起来时,警告. 类Unix连接器允许这个选项,有时比
1 G0 a/ K" v3 c) w较实用, 但是在其他的操作系统上的连接器不允许这个. 这个选项可以让你在合并全局符号时发现某
" f8 H, j1 T9 z$ m& H& D' `些潜在的问题. 不幸的是,有些C库使用这项特性,所以你可能会像在你的程序中一样,在库中得到一些0 F$ G( M1 W( N! H& y8 b
警告信息.
' R! L0 O! \! r w
6 G# ~* H1 H& L* U6 E这里给出三种类型的全局符号的解释(用C语言):
w& K' p% V3 \* i' X+ H6 {
x% `" l5 Q1 N' H2 y* C6 C) k8 N `int i = 1;'6 _! s4 P3 U3 o S( h0 d
一个定义, 它会存在于输出文件中的已初始化数据节.2 r+ U6 z) _" }, `
' U4 J3 s( Z8 E
`extern int i;'
6 X$ \% Q6 m% N1 c" u" U$ A8 m- E8 \ 一个未定义符号,它不占用空间. 必须在另外某一处对它有一个定义,或一个普通符号0 u7 c6 K( |+ B. W" H0 d2 p" k2 n$ g
8 ^& [) E# h; O8 P* T* Z
`int i;' o1 Z$ P! a* A
一个普通符号.如果对于一个变量只有(一个或多个)普通符号, 它进入输出文件的未初始化数据域. 连
& t1 e5 U2 u& Z# U 接器会把同一变量的多个普通符号合并成一个单一的符号. 如果他们有不同的大小, 它采用最大的一
4 i9 C; Z0 R/ ~9 l% G8 { 个. 如果是对同一变量的定义,连接器把一个普通符号转化为一个声明.8 D% U. c% Y2 e& i' N; M8 D
' b# q/ i4 [0 W- x
'--warn-common'选项可以产生五种类型的警告. 每种警告由两行组成: 第一行描述遇到的符号, 第二
5 M% I- @# C& G; c/ q9 O8 S/ F& C行描述遇到的前一个具有相同名字的符号. 一个或两个都可能成为普通符号.3 [+ D5 p* c9 }/ S
: D( l5 T$ g ^; l 1. 把一个普通符号转化为一个引用, 因为这个符号已经有一个定义了.
j! |9 N$ u9 X, f FILE(SECTION): warning: common of `SYMBOL'
4 p) k+ ]% _$ S* b( ^ overridden by definition
( y5 _9 S, }5 P( n" J, f FILE(SECTION): warning: defined here2 H$ F; T ^4 n t0 L* l
6 q. K# c) j' N6 Q1 R9 ~2 B 2. 把一个普通符号转化为一个引用,因为遇到了第二个关于符号的定义. 这跟前一种情况相同,除了符
2 `6 O: M" F. c. w8 B1 P 号遇到的顺序相反./ a% ~9 X, U3 ]# R
FILE(SECTION): warning: definition of `SYMBOL'3 h @ F6 t, {, ]8 I
overriding common3 N( A( D. x; ^+ C" t, A
FILE(SECTION): warning: common is here
5 h: L' v8 L5 v" f# W) \9 R; V4 @( ^9 ^" Q$ @
3. 把一个普通符号跟前一个相同大小的普通符号合并.8 _0 b# n$ O" ~. h% \) Y6 x- f
FILE(SECTION): warning: multiple common9 ^6 B: |* V. R7 f
of `SYMBOL'! j( u5 B* ~7 O" }0 F
FILE(SECTION): warning: previous common is here8 @9 d' C) @3 \
' d. L% m$ d# j& C
4. 把一个普通符号跟前一个更大的普通符号合并.
2 t8 E3 ?7 _2 J, d5 @9 q& e+ s FILE(SECTION): warning: common of `SYMBOL'' c5 G" T% n% p5 M% S8 N1 H, l! {
overridden by larger common
5 c8 f# n$ Z9 P: Q$ J, {: R0 z FILE(SECTION): warning: larger common is here7 K" [6 q' h: q# x
1 J; i8 o2 o7 H- O 5. 把一个普通符号跟前一个更小的普通符号合并. 这跟前一种情况相同, 除了遇到的符号的顺序不同.
9 X9 y$ w: q7 D% W FILE(SECTION): warning: common of `SYMBOL'
' o* v: i% E! v6 g6 Q$ v% _3 i; r overriding smaller common
% Z+ J6 k" p, C$ @$ n2 ?# D1 |: | FILE(SECTION): warning: smaller common is here
0 b+ h+ w. @/ x- b7 z; x6 \! L( x1 k
`--warn-constructors'
, n0 W8 z' ]+ c( W/ r如果有全局结构体被使用到了,警告. 这只对很少的一些目标文件格式有用. 对于COFF或ELF格式, 连
' B$ A9 [6 |% c0 z) }5 N& }: X0 M: `接器不同探测到全局结构体的使用./ i2 @& P6 H) }" b
6 m4 N8 `5 x9 r* w- s9 C6 K`--warn-multiple-gp': G! x5 ]7 |% h
如果在输出文件中,需要多个全局指针值,警告. 这只对特定的处理器有意义, 比如Alpha. 特别的,有
; S6 p( ?' P6 E( B1 Y些处理器在特定的节中放入很大的常数值. 一个特殊的寄存器(全局指针)指向这个节的中间部分, 所
, k; c* I3 c1 Q" o& `3 G4 r/ |以通过一个基地址寄存器相关的地址模式,这个常数可以很容易地被载入. 因为这个基寄存器相关模式 T+ m' ^" J3 V) X- H8 O, J2 c3 b
的偏移值是固定的而且很小(比如,16位), 这会限制常量池的最大尺寸. 所以,一个很大的问题是,为了
) c* }6 B% w2 p能够定位所有可能的常数,经常需要使用多个全局指针值. 这个选项在这种情况发生时产生一条警告.3 a3 v4 Q$ I3 c# E; c& ~1 h# @
: w3 ?0 O p0 _7 p! d`--warn-once'- L. n( Y, g S" S
对于每一个未定义符号只警告一次, 而不是在每一个用到它的模块中警告一次.
3 E9 k( i, t1 V& q: o- }3 r' M2 `/ Z3 {( d% [4 W
`--warn-section-align'3 S& Y7 b) d5 L1 w
如果输出节的地址因为对齐被改变了,警告. 通常, 对齐会被输入节设置. 如果'SECTION'命令没有指( }9 U6 y1 g- \ D- ^4 k4 H7 A7 Y
定节的起始地址, 地址就会被隐式改变.
7 K# l6 _/ o1 K. M" s) C+ A; V( X8 k+ k* U/ D, b+ U2 s
`--whole-archive'
7 m2 _5 ~& {8 D7 x( S5 c1 d# c% Z对于每一个在命令行中'--whole-archive'选项后面出现的档案文件, 在连接中包含档案文件中的所有9 W+ s5 n7 h, P1 C7 g2 d7 T/ Y
目标文件, 而不是为需要的目标文件搜索档案文件. 这在把一个档案文件转化为一个共享库时使用, 把9 |+ T+ R Q3 r; l7 M( _% J
所有的目标放到最终的共享库中. 这个选项可以被多次使用.
6 n2 R# I* \) }1 j2 W. ~6 }- K4 U* u
在GCC中使用这个选项需要注意两点: 首先,GCC不知道这个选项, 所以,你必须使用'-Wl, -whole-archive'.. i; @% W3 {# X: X9 x# \% }
第二, 不要忘了在你的档案文件列表的后面使用'-Wl, -no-whole-archive',因为GCC会把它自己的档
6 b& c1 S' B, `; W4 ^# E4 A, X案列表加到你的连接后面, 而这可能并不是你所预期的.
& s1 J, q, x3 J" g2 k
( {3 |2 V3 q' A" q5 E`--wrap SYMBOL'* P3 y) b8 B0 \% Y/ R3 r: ^
对SYMBOL符号使用包装函数. 任何未定义的对SYMBOL符号的引用会被解析成'_wrap_SYMBOL'. 而任何6 `4 T# k _" B2 W. @
未定义的对'_real_SYMBOL'的引用会被解析成SYMBOL.
$ g, C5 c% V/ x6 j9 F' Q& G- v9 P" V! r- a' a. l
这可以用来为系统函灵敏提供一个包装. 包装函灵敏应当被叫做'__wrap_SYMBOL'. 如果需要调用这个+ p" q; `+ o( P
函数, 那就应该调用'__real_SYMBOL'
! A) s0 ]8 ?8 l" w0 p5 b9 I
1 r$ o4 `4 L" G 这里是一个没什么实用价值的例子:
1 U% l8 e5 b& E" M7 _3 S) |4 D- E
void *- K7 ^8 L* y- D; X) ]2 P$ y
__wrap_malloc (int c)/ o: Y' j" A, S0 Z2 F/ t
{
5 m$ g% P' b1 l0 e7 y printf ("malloc called with %ld\n", c);
: O) u1 E9 q! p9 p2 o return __real_malloc (c);$ s. K5 {1 \( S# V1 C" u
}, U4 G8 ^" C! ?5 Q
! o( N" s. I" u; @% f如果你使用'--wrap malloc'把这节代码跟其他的代码连接, 那所有的对'malloc'的调用都会调用& @- Y* t9 w) w7 i
'__wrap_malloc'函数. 而在'__wrap_malloc'中的'__real_malloc'会调用真正的'malloc'函数.: L# u7 U+ O5 o
3 z/ M# \; P( A( I1 i2 H你有可能也希望提供一个'__real_malloc'函数, 这样,不带有'--wrap'的连接器也会成功连接.如果
6 s9 ?& c; D6 D1 ~4 _* D- m你这样做了, 你不能把'__real_malloc'的定义跟'__wrap_malloc'放到同一个文件中;如果放在一起
C5 Z6 T% r3 F% X1 k汇编器会在连接器之前把调用解析成真正的'malloc'.' f: q6 e" K* l: C( s/ `
; h$ P* X# Y; m( U* t$ h* s`--enable-new-dtags'- d% d7 ~2 [. j' o, n/ T( J
`--disable-new-dtags'
! c' W p+ r6 |% D连接器可以在ELF中创建一个新的动态标签. 但是旧的ELF系统可能不理解这个. 如果你指定了
Q; }- u( V2 z) J: R'--enable-new-dtags',动态标签会按需要被创建. 如果你指定了'--disable-new-dtags',那不会有
+ n/ @0 @( q0 B: h: Y1 P! M! r新的动态标签被创建. 缺省地,新的动态标签不会被创建. 注意这些选项只在ELF系统中有效.
9 x j% P, c9 a w* I. ]! J' ^ o4 q" v1 w) S/ i$ T V9 }
i386 PE平台的特定选项.: _8 e3 b0 `; @! v
-----------------------------------
/ c/ F/ G3 E) C1 ?
# S5 F! v# C; H0 {3 z( Ii386 PE连接器支持'-shared'选项, 它使输出文件为一个动态链接库(DLL),而不是一个普通的可执行文件. 在" ^/ y/ C3 ~6 ~* t7 v
使用这个选项的时候,你应当为输出文件取名'*.dll',另外, 连接器完全支持标准的'*.def'文件, 这类文件可
3 F# L9 p8 z3 Y) q6 H以在连接器命令行上象一个目标文件一样被指定(实际上, 它应当被放在它从中导出符号的那个档案文件前面,# Y2 Y/ d, s: X* I0 F: d
以保证它们象一个普通的目标文件一样被连接进去.)* z Z: N! n% f& P- [
9 Q7 Y* O' y5 D, H' m% }
除了对所有平台通用的那些选项外,i386 PE连接器支持一些只对i386平台专有的命令行选面. 带有值的选项应- P3 q% r$ V! e& Z- h
当用空格或等号把它跟值分隔开.: Q7 _' f- ^+ X( K% j/ K
$ n3 B' k" a! ]! g- f`--add-stdcall-alias'* n& M0 N3 k& m: |- J& J S
如果给出这个选项, 带有标准调用后缀(@NN)的符号会被剥掉后缀后导出.
$ a9 W0 s5 K; T9 u" A( `2 }( k- q- X0 Z* ^, s. V+ t- @7 T
`--base-file FILE'
3 \. E* m* i5 n! w使用FILE作为文件名,该文件是存放用'dlltool'产生 DLL文件时所需的所有重定位符的基地址的.(这
2 k, R. X- _* E9 L( V7 v! E1 R个选面是i386 PE平台所专有的]$ X R9 X0 D7 o# T3 S* J
: T7 a$ I+ _) L`--dll'
/ z. Y6 k2 J) x6 Y( O7 O创建一个DLL文件而不是一个常规可执行文件. 你可能在一个给出的'.def'文件中使用'-shared'或指. m* U0 L: k0 n8 D8 |! o2 l( |5 I
定'LIBRARY'.
" H0 S8 }9 s# z' x5 G5 N7 ^- Z+ {/ ^- c( Y' {( n5 U+ E! _( A
`--enable-stdcall-fixup'
2 ^1 C5 k% g! f: t3 j`--disable-stdcall-fixup'
+ K6 u' x: @7 l1 z& f' d, y如果连接器发现有符号不能解析, 它会试图进行'失真连接',即寻找另一个定义的符号,它们只是在
; ^ V* I3 T9 X9 [" j# o符号名的格式上不同(cdecl vs stdcall),并把符号解析为找到的这个符号. 比如, 一个未定义的符7 e7 T: o+ d2 B, V
号'_foo'可能被连接到函数'_foo@12', 或者一个未定义的符号'_bar@16'可能被连接到函数'_bar'.5 t) t }3 y! U3 s; G4 |- o$ X
如果连接器这么做了, 它会打印出一条警告信息, 因为在正常情况下,这会连接失败, 但有时,由第三; x$ a1 a" z, u* p- Z+ @* s/ s1 s
方库产生的导入库可能需要这个特性. 如果你指定了'--enable-stdcall-fixup', 这个特性会被完全" S3 A; A, r4 J. F3 R$ Z
开启,警告信息也不会打印出来. 如果你指定了'--disable-stdcall-fixup',这个特性被关闭,而且这
; d: f& e" q9 k" q: q$ d* J% Y9 F$ u样的错误匹配会被认为是个错误.* L& ?( F* T, u/ k+ y/ {
7 {- i1 Q, e8 l% |$ j
`--export-all-symbols'
. ~( l4 B. u" O# g& c( q% G2 O如果给出这个选项,目标中所有由DLL建立的全局符号会被DLL导出. 注意这是缺省情况,否则没有任何1 G: k' X! R4 ?" t7 m
符号被导出. 如果符号由DEF文件显式地导出,或由函数本身的属性隐式地导出, 缺省情况是除非选项
. K7 k8 z- b, o4 ]& a给出,否则不导出任何其他的符号. 注意符号`DllMain@12',`DllEntryPoint@0',
, z$ F- x ]" o' p$ I2 I3 A9 S$ e% k`DllMainCRTStartup@12'和`impure_ptr'不会自动被导出.而且,由其他的DLL导入的符号也不会被再
& p( @7 z# c1 Q! V: u7 w次导出, 还有指定DLL内部布局的符号,比如那些以'_head_'开头,或者以'_iname'结尾的符号也不会7 r& B5 v9 O+ ]& i- E1 D5 F$ p
被导出.还有,'libgcc','libstd++','libmingw32'或'crtX.o'中的符号也不会被导出. ......3 G% h% `. i' d" E" J( A' {
- B1 [& J$ V8 j+ s5 E( g7 ~* s
环境变量0 ^/ b# ^3 x1 Z7 \& e- l" Q
=====================: S _! b/ f+ Q5 ~1 M
) [, l* J- y9 y# Z你可以通过环境变量`GNUTARGET', `LDEMULATION'和`COLLECT_NO_DEMANGLE'改变'ld'的行为./ n! z& T: n+ D& z
* X9 q3 P5 |. A8 w`GNUTARGET'在你没有使用'-b'(或者它的同义词'--format')的时候,决定输入文件的格式. 它的值应当是BFD3 V3 a8 t6 D' X* A& O- W% Y7 @ w
中关于输入格式的一个名字. 如果环境中没有'GNUTARGET'变量, 'ld'使用目标平台的缺省格式. 如果
- ?+ G l+ ~0 r'GNUTARGET'被设为'default', 那BFD就会通过检查二进制的输入文件来找到输入格式; 这个方法通常会成功,
( C7 k: U& A2 Q5 }/ Z- m1 T6 `但会有潜在的不明确性, 因为没有办法保证指定一个目标文件格式的魔数总是唯一的. 但是, 在每一个系统上
! l# r* O6 k* p的BFD配置程序会把这个系统的常规格式放在搜索列表的首位, 所以不明确性可以通过这种惯列来解决.
: ]) p4 V4 {5 S0 P3 `, p
+ R$ S6 ~ H; L8 A( p`LDEMULATION'在你没有使用'-m'选项的时候决定缺省的模拟器. 模拟器可以影响到连接器行为的很多方面,
/ Y. E y: l1 t$ a$ K' @特别是连接器的缺省连接脚本. 你可以通过'--verbose'或'-V'选项列出所有可用的模拟器. 如果'-m'选项没5 R/ i! _, y+ S
有使用, 而且`LDEMULATION'环境变量没有定义, 缺省的模拟器跟连接器如何被配置有关.
/ |5 H8 x8 D; k _# r% a8 v. G/ e) _3 ?% g
一般地,连接器缺省状况下会重构符号.但是,如果在环境中设置了`COLLECT_NO_DEMANGLE', 那缺省状态下就不
5 j- q' ^* F5 _会重构符号.这个环境变量在GCC的连接包装程序中会以相似的方式被使用. 这个缺省的行为可以被'--demangle'
K" G9 t& `8 o) P0 h9 Y或'--no-demangle'选项覆盖.
, G2 f' x% u5 q4 q# }1 E
3 U& @* _3 ^+ ]! f& F连接脚本
6 ?1 t, E" U8 m N) \" B/ r**************: O5 V, O) _& y8 T+ Q# e' R
4 A( F, q4 A. d
每个连接都被一个'连接脚本'所控制. 这个脚本是用连接命令语言书写的.
. E2 v7 V/ [. r6 t2 e/ h* \
! V* U2 M& j' L1 B6 g4 c4 I连接脚本的一个主要目的是描述输入文件中的节如何被映射到输出文件中,并控制输出文件的内存排布. 几乎
0 k" r' b+ J; z j所有的连接脚本只做这两件事情. 但是,在需要的时候,连接器脚本还可以指示连接器执行很多其他的操作.这
4 f3 u) x; _! _+ ?7 e通过下面描述的命令实现.
$ m% I, z8 s5 g' k: I' f$ ?9 @8 r q9 _; u
连接器总是使用连接器脚本的.如果你自己不提供, 连接器会使用一个缺省的脚本,这个脚本是被编译进连接器
: j0 q3 l) m) S( I8 o$ W$ o! e可执行文件的. 你可以使用'--verbose'命令行选项来显示缺省的连接器脚本的内容. 某些命令行选项,比如
9 ?8 E, M- q! g'-r'或'-N', 会影响缺省的连接脚本./ T q5 s* Q }( Q7 c
4 J c2 X# J( Z" A/ I
你可以过使用'-T'命令行选项来提供你自己的连接脚本. 当你这么做的时候, 你的连接脚本会替换缺省的连9 N% |7 [/ a' d' r- Y4 ~
接脚本.
. x! H W# T- f9 z$ W7 `' ^: |) U* O# g
你也可以通过把连接脚本作为一个连接器的输入文件来隐式地使用它,就象它们是一个被连接的文件一样.
/ Z# Q- O6 W/ m8 n/ E Y
! ~1 i2 V) E' v4 Z! |' q基本的连接脚本的概念
, H. g/ Y" v) H/ ?============================$ @/ H i7 ^# i! ^: T% z) Y
4 Z* v2 v+ F0 L0 m2 C/ ]/ V
我们需要定义一些基本的概念与词汇以描述连接脚本语言.: a& W' z" W* O9 Z& k+ x/ J
0 Q! R8 `% w0 A5 e1 C连接器把多个输入文件合并成单个输出文件. 输出文件和输入文件都以一种叫做'目标文件格式'的数据格式形
" ]7 E, n. x5 ^4 q式存在. 每一个文件被叫做'目标文件'. 输出文件经常被叫做'可执行文件',但是由于需要,我们也把它叫做目8 U6 t9 Y n/ p0 r/ G
标文件. 每一个目标文件中,在其它东西之间,有一个节列表.我们有时把输入文件的节叫做输入节; 相似的,输/ }$ @9 s" l# R
出文件中的一个节经常被叫做输出节.
( s7 P) B- K; D' j) F! `8 Y( X- e% ?- S8 ?
一个目标文件中的每一个节都有一个名字和一个大小尺寸. 大多数节还有一个相关的数据块, 称为节内容. 某
6 \# X% R& O9 a一个节可能被标式讵'loadable',含义是在输出文件被执行时,这个节应当被载入到内存中去. 一个没有内容的
; j* P/ M; ]7 ~节可能是'allocatable', 含义是内存中必须为这个节开辟一块空间,但是没有实际的内容载入到这里(在某些
8 q6 E4 ^; j+ q情况下,这块内存必须被标式讵零). 一个既不是loadable也不是allocatable的节一般含有一些调试信息.
7 d9 R9 \' x6 f5 f4 p M0 K1 m" V4 |, d7 w5 l" i# T0 t- r! Y4 ^
每一个loadable或allocatable的输出节有两个地址. 第一个是'VMA'或称为虚拟内存地址. 这是当输出文件运
' d# E; _" D) q, k9 S行时节所拥有的地址. 第二个是"LMA', 或称为载入内存地址. 这个节即将要载入的内存地址. 这大多数情况下5 E- L+ f& H; e3 E9 Q8 N/ R
这两个地址是相同的. 它们两个有可能不同的一个例子是当一个数据节在ROM中时, 当程序启动时,被拷贝到RAM" C8 a' ?. D7 b" n* C: l3 a" R
中(这个技术经常被用在基于ROM的系统中进行全局变量的初始化). 在这种情况下, ROM地址就是LMA, 而RAM地( @* V7 }' J- h1 }; O: y
址就是VMA.
; L: Y8 Y; g; Y
" x# H& h0 A n; Y# G你可以通过使用带有'-h'选项的'objdump'来察看目标文件中的节.7 a1 g$ {; P5 k W- X- v' V
r1 R0 Y. C" y- E$ m6 T每一个目标文件还有一个关于符号的列表, 被称为'符号表'. 一个符号可能是定义过了的,也可能是未定义的.
& {2 y( _) ]( I$ }+ X每一个符号有一个名字, 而且每一个定义的符号有一个地址. 如果你把一个C/C++程序编译为一个目标文件,对
" \8 I4 P* ]& u2 p) `于每一个定义的函数和全局或静态变量,你为得到一个定义的符号. 每一个在输入文件中只是一个引用而未定义
; ]) b: K$ h2 i% F0 [6 R4 L的函数或全局变量会变成一个未定义的符号./ T7 @; T% {# B' T- {& j6 G
+ a% f! ^/ h N; P) N7 j你可以使用'nm'程序来看一个目标文件中的符号, 或者使用'objdump'程序带有'-t'选项.
" e! Z- x4 P }( _; _- y% y4 A, ^* R1 g1 E
连接脚本的格式
p2 E2 k! r; o! [5 z2 t====================3 k+ ~4 b! e0 m: Y
2 F! N# c3 E {3 }5 K5 r8 {2 A/ X
连接脚本是文本文件.2 D' Z; m6 ]1 ]
, c7 k( E- X# O9 g你写了一系列的命令作为一个连接脚本. 每一个命令是一个带有参数的关键字,或者是一个对符号的赋值. 你可
" `2 e; g' i0 s4 d3 V以用分号分隔命令. 空格一般被忽略.8 P. N7 B" @2 d6 Q' {
1 C) N8 K& {0 I# e3 }! q5 a文件名或格式名之类的字符串一般可以被直接键入. 如果文件名含有特殊字符,比如一般作为分隔文件名用的逗& J. x* c( D* J# b
号, 你可以把文件名放到双引号中. 文件名中间无法使用双引号.- ~. H# J1 |: f7 B' i2 U% ^% ^$ P
, P" v3 q3 ?" O' ~; {你可以象在C语言中一样,在连接脚本中使用注释, 用'/*'和'*/'隔开. 就像在C中,注释在语法上等同于空格.
7 u- ~+ b2 j( }9 h6 F
w5 L& b8 m; M i6 j简单的连接脚本示例
+ D& {3 k' I$ ^! W============================' _+ K$ |: z7 ?3 v* b3 o, i
6 A c/ X ~' X2 H5 d
许多脚本是相当的简单的.- T8 ^/ A$ t- d/ p$ b
9 f% K/ p( o7 T1 f! ~9 @
可能的最简单的脚本只含有一个命令: 'SECTIONS'. 你可以使用'SECTIONS'来描述输出文件的内存布局.
0 @- \9 [$ _. @) R0 _* I+ E7 x; u6 Q1 W5 E* R# M9 `0 q6 f1 X# I! ~
'SECTIONS'是一个功能很强大的命令. 这里这们会描述一个很简单的使用. 让我们假设你的程序只有代码节,
( C: h" v2 V- y9 N# F初始化过的数据节, 和未初始化过的数据节. 这些会存在于'.text','.data'和'.bss'节, 另外, 让我们进一
* z$ L+ \) z# S( n8 z步假设在你的输入文件中只有这些节.$ V* E+ s/ t5 m2 ]/ l3 G4 P
r& Z% m1 V8 o
对于这个例子, 我们说代码应当被载入到地址'0x10000'处, 而数据应当从0x8000000处开始. 下面是一个实现" \( D2 Y5 z: o; _/ J" q, ]* v3 y! S6 z/ Q
这个功能的脚本:7 l) T8 C. i4 X) B
% F( O* k/ h- r% j9 O4 R SECTIONS: B3 {2 P) {! \# T& M( E
{
3 U- V7 d1 [& K . = 0x10000;* l( K- N0 M4 d5 k
.text : { *(.text) }% ~7 j0 t" l" J! \
. = 0x8000000;* p& L3 I# q% C D3 q5 S: O
.data : { *(.data) }
# L- w% f( q2 r( l .bss : { *(.bss) }- V, i# w( k Q; F
}6 p8 G' I5 n7 D2 Q1 S+ | l
5 [" |( P7 R# {1 d你使用关键字'SECTIONS'写了这个SECTIONS命令, 后面跟有一串放在花括号中的符号赋值和输出节描述的内容.1 u! J6 a# F" m0 N& g; w
5 f! O4 y) G5 W5 j' q
上例中, 在'SECTIONS'命令中的第一行是对一个特殊的符号'.'赋值, 这是一个定位计数器. 如果你没有以其1 f+ J" W' Q: r5 C9 F% C. Y
它的方式指定输出节的地址(其他方式在后面会描述), 那地址值就会被设为定位计数器的现有值. 定位计数器* }) R6 V5 G6 M6 B2 j m
然后被加上输出节的尺寸. 在'SECTIONS'命令的开始处, 定位计数器拥有值'0'.
, V3 @- C, v( L1 x" D$ q- x
& {( F* D1 ^; c+ g7 F- j第二行定义一个输出节,'.text'. 冒号是语法需要,现在可以被忽略. 节名后面的花括号中,你列出所有应当被
& a3 U. l, [ Y/ G放入到这个输出节中的输入节的名字. '*'是一个通配符,匹配任何文件名. 表达式'*(.text)'意思是所有的输
' f" F7 l8 }6 U+ A/ C# y入文件中的'.text'输入节. m8 `$ J5 U3 O; ]" a0 q% Q
1 L# W$ Q2 ]$ U因为当输出节'.text'定义的时候, 定位计数器的值是'0x10000',连接器会把输出文件中的'.text'节的地址设" F! a& Y$ X) D" r9 B! h
为'0x10000'.
( L- E) @1 c R# Q
/ f* N' J! c( i6 W+ P) Z6 @余下的内容定义了输出文件中的'.data'节和'.bss'节. 连接器会把'.data'输出节放到地址'0x8000000'处. 连接' V# K0 \: L' ?# U
器放好'.data'输出节之后, 定位计数器的值是'0x8000000'加上'.data'输出节的长度. 得到的结果是连接器会
+ F! Q5 j" U: z/ V8 J把'.bss'输出节放到紧接'.data'节后面的位置.
* |& F6 q0 r5 [5 A
5 u9 I g$ K6 h7 `连接器会通过在必要时增加定位计数器的值来保证每一个输出节具有它所需的对齐. 在这个例子中, 为'.text') r! x4 O, q, Z4 G& `3 w/ m- I
和'.data'节指定的地址会满足对齐约束, 但是连接器可能会需要在'.data'和'.bss'节之间创建一个小的缺口.
' x/ R4 d A* K3 d) O/ w
" L2 L" C3 z/ O5 h就这样,这是一个简单但完整的连接脚本.* h/ ^! C/ O" ~- `" y3 D
* M$ G. s4 Y, t! U
简单的连接脚本命令.2 k6 Y. g, V0 s G3 E" r( x. x9 y
=============================
: I r6 u. ~& Z3 u0 Z1 a
( ^& i/ g2 I7 E& e) K在本章中,我们会描述一些简单的脚本命令.
; a3 c ^" ]5 u
3 _: A% Y/ A6 E, N- h设置入口点.
& a0 W5 f6 g B( y `-----------------------) P8 T3 ^" X! J# \! N
) f' S" `- k2 ~1 m- a) P在运行一个程序时第一个被执行到的指令称为"入口点". 你可以使用'ENTRY'连接脚本命令来设置入口点.参数
' h5 A+ n2 Z @是一个符号名:4 m0 J2 I9 r+ F" {1 i, r' M
ENTRY(SYMBOL)
- |$ @ b$ ]8 D+ x
1 a1 v- L- J5 T, _ K) r$ |有多种不同的方法来设置入口点.连接器会通过按顺序尝试以下的方法来设置入口点, 如果成功了,就会停止.+ I% V) V6 U/ d$ ~
$ V2 Z( {) g$ Z4 N7 a. \$ {3 g
* `-e'入口命令行选项;& u9 t* j9 l. Y: k& j- }; w
, m( h' e9 O3 S6 f0 U; u
* 连接脚本中的`ENTRY(SYMBOL)'命令;1 B& y! a$ [6 R8 u: X$ [1 u
9 t4 q) z B; @$ E6 T0 Z
* 如果定义了start, 就使用start的值;
& p" O8 F# n5 G- r; F: ]8 _" f) v+ r/ N
* 如果存在,就使用'.text'节的首地址;1 c R; \+ }% I/ o( S% h
$ z- R" }' l7 Y6 Q: h" L# B
* 地址`0'.
- \9 G+ q5 _* [3 _ v
D5 O6 B) o/ z处理文件的命令.
& K/ ?- T+ M8 t* m$ M---------------------------
; {- l. d7 m0 `! {( Q* \
" K) r( M, ~& E! F* Q7 \有几个处理文件的连接脚本命令.7 Y. M& @4 `! k, N2 g! _
7 ~; d( O2 k# `: K9 P' o`INCLUDE FILENAME'
6 s" V' Q* ~$ n4 f5 W- `在当前点包含连接脚本文件FILENAME. 在当前路径下或用'-L'选项指定的所有路径下搜索这个文件,
/ v: R" ^/ z: x7 w你可以嵌套使用'INCLUDE'达10层.8 E) y A# A! [6 i
9 }2 j. p$ \8 k0 M
`INPUT(FILE, FILE, ...)') D7 L9 u+ m( t. w+ f
`INPUT(FILE FILE ...)'
( K, L% G r9 r1 Z q( ?' v'INPUT'命令指示连接器在连接时包含文件, 就像它们是在命令行上指定的一样.7 t" W6 K' G: T% P/ z
0 j/ S! G% L) @2 w8 l7 ]* G: y4 G比如,如果你在连接的时候总是要包含文件'subr.o',但是你对每次连接时要在命令行上输入感到厌烦
& Z% y" Y2 y1 i+ `/ R, 你就可以在你的连接脚本中输入'INPUT (subr.o).; m% |5 |& u% k3 v8 I6 {
2 n) x5 ?/ ]( r& _( C0 ~# q
事实上,如果你喜欢,你可以把你所有的输入文件列在连接脚本中, 然后在连接的时候什么也不需要,
/ U) R; A D$ L1 C) q2 c- h只要一个'-T'选项就够了.( N3 A! f/ e2 B' d
# A! u5 ]+ U0 Z- ]% G- Q/ O在一个'系统根前缀'被配置的情况下, 一个文件名如果以'/'字符打头, 并且脚本也存放在系统根+ \' A1 u4 ~/ ]3 w7 q7 W
前缀的某个子目录下, 文件名就会被在系统根前缀下搜索. 否则连接器就会企图打开当前目录下的文
- L( C! ?3 p4 J3 I7 O" [/ h件. 如果没有发现, 连接器会通过档案库搜索路径进行搜索.
+ ~7 {' Z. O: b) N/ ?
0 S8 H9 E' j6 D! C5 ?如果你使用了'INPUT (-lFILE)', 'ld'会把文件名转换为'libFILE.a', 就象命令行参数'-l'一样.
9 N& j$ \) ~( J. T% x# s. T; ] B
: @( C, u/ c) |当你在一个隐式连接脚本中使用'INPUT'命令的时候, 文件就会在连接时连接脚本文件被包含的点上
9 o# U' @# Z3 H1 c# U4 f1 B被包含进来. 这会影响到档案搜索.
( G) n1 T# Q3 p: J G8 C9 i
) C1 J5 Q5 C+ ~: q& ^`GROUP(FILE, FILE, ...)'
5 E, i( V2 K- J. S. w' i2 _`GROUP(FILE FILE ...)'
6 @ v/ l- L9 G+ o+ X+ t除了文件必须全是档案文件之外, 'GROUP'命令跟'INPUT'相似, 它们会被反复搜索,直至没有未定义
/ G& _' @ l) X的引用被创建.
) q5 O$ Q: l* f5 g6 w2 R' ]& W
6 P/ V L `0 j8 K, l* s1 L`OUTPUT(FILENAME)'6 B& b0 e/ X {0 H7 c, Q
'OUTPUT'命令命名输出文件. 在连接脚本中使用'OUTPUT(FILENAME)'命令跟在命令行中使用'-o
1 {( r/ s& k8 Y$ \! s8 EFILENAME'命令是完全等效的. 如果两个都使用了, 那命令行选项优先.
, Q9 h9 \1 @- s: s6 j2 B, h5 Y6 W7 H
; u' e! I* M$ M# _你可以使用'OUTPUT'命令为输出文件创建一个缺省的文件名,而不是常用的'a.out'.
. ^# m9 p' j* ^/ ^. G* c- r
8 _* b2 R u* }% f( X& A& ? n`SEARCH_DIR(PATH)'
2 j3 ]" F% h' H) T9 O`SEARCH_DIR'命令给'ld'用于搜索档案文件的路径中再增加新的路径. 使用`SEARCH_DIR(PATH)'跟在$ w6 [4 x9 { C. }
命令行上使用'-L PATH'选项是完全等效的. 如果两个都使用了, 那连接器会两个路径都搜索. 用命. @ p" z% ]& J9 R0 O
令行选项指定的路径首先被搜索.
. Y6 j/ N4 c1 \5 t b- g& r Q
7 u: w1 o, z. X, G`STARTUP(FILENAME)'% B' I# l. b, l- m
除了FILENAME会成为第一个被连接的输入文件, 'STARTUP'命令跟'INPUT'命令完全相似, 就象这个文
6 d3 F' X" d" d$ u件是在命令行上第一个被指定的文件一样. 如果在一个系统中, 入口点总是存在于第一个文件中,那
; u0 W8 P1 O. f( G这个就很有用.
, a. U; u/ s6 ~# R
; R5 d: u# ^' m: r4 O, R3 U处理目标文件格式的命令.
1 G, w% t" s3 L/ j7 _* ^* T) }-----------------------------------------
& G7 ?4 k# A% E
3 R* s* B& _- o/ j8 L( q3 J有两个处理目标文件格式的连接脚本命令.& [/ \% M6 L, {4 K, y2 F
5 F& W+ j8 f3 y. U- f r3 \
`OUTPUT_formAT(BFDNAME)'' F: g# T8 b5 ]! S0 p
`OUTPUT_formAT(DEFAULT, BIG, LITTLE)'
* J) A3 v$ K+ w/ I$ o8 u`OUTPUT_formAT'命令为输出文件使用的BFD格式命名. 使用`OUTPUT_formAT(BFDNAME)'跟在命令行上
+ d6 f: A1 X5 c使用'-oformat BFDNAME'是完全等效的. 如果两个都使用了, 命令行选项优先.
" A3 Z6 d4 N; ?3 \- R
b( Z1 G7 t9 w# C: M你可在使用`OUTPUT_formAT'时带有三个参数以使用不同的基于'-EB'和'-EL'的命令行选项的格式.
7 Q X# _% q: C+ H( R: T0 ~1 D# K6 V, `' m
如果'-EB'和'-EL'都没有使用, 那输出格式会是第一个参数DEFAULT, 如果使用了'-EB',输出格式会是# D# z7 z: F4 O: x; i' G) J
第二个参数BIG, 如果使用了'-EL', 输出格式会是第三个参数, LITTLE.' O" L8 c! m0 E
# h3 E \& s7 ~% H比如, 缺省的基于MIPS ELF平台连接脚本使用如下命令:% o5 m. f& |# j3 ]% a) E' N' B& i9 E! `
6 ~6 S7 f1 j* ^7 T
OUTPUT_formAT(elf32-bigmips, elf32-bigmips, elf32-littlemips)
% E ~0 Z. L5 y" s) A 这表示缺省的输出文件格式是'elf32-bigmips', 但是当用户使用'-EL'命令行选项的时候, 输出文件就会# P# X: A. A7 _ z2 z6 i5 x
被以`elf32-littlemips'格式创建.
) ? S" o) J6 Q! h; f$ P5 _- }$ R* }4 o
`TARGET(BFDNAME)'" Z; y' T9 O' f/ z/ _2 Y& x
'TARGET'命令在读取输入文件时命名BFD格式. 它会影响到后来的'INPUT'和'GROUP'命令. 这个命令跟! q: H' L0 t* i3 t: F
在命令行上使用`-b BFDNAME'相似. 如果使用了'TARGET'命令但`OUTPUT_formAT'没有指定, 最后的
1 p2 v5 I1 Y G$ y6 ]4 n'TARGET'命令也被用来设置输出文件的格式.% m$ {' V7 e2 ]
* I+ Z/ o7 T$ |! a- P5 p+ S
其它的连接脚本命令.2 X- m5 \) W# z0 y" \9 y7 n4 g ?+ p
----------------------------
* \9 W& M6 N \) Q9 T, p% x/ @, U* M0 p6 g
还有一些其它的连接脚本命令.0 D2 f1 i3 G7 ]7 C
4 a& X7 _( v4 f( j# s& o
`ASSERT(EXP, MESSAGE)'
1 k$ c9 a9 [8 T4 V4 G% n确保EXP不等于零,如果等于零, 连接器就会返回一个错误码退出,并打印出MESSAGE.; H8 \$ m. @$ p1 E; x
, ?8 d- [( a' l9 r9 ]$ T' Y
`EXTERN(SYMBOL SYMBOL ...)'
: u7 Z0 }& O ?5 B5 s2 S强制SYMBOL作为一个无定义的符号输入到输出文件中去. 这样做了,可能会引发从标准库中连接一些
' P6 q% V+ M) B节外的库. 你可以为每一个EXTERN'列出几个符号, 而且你可以多次使用'EXTERN'. 这个命令跟'-u'
/ T0 l0 S2 f8 e, Q" O }命令行选项具有相同的效果.* E& y Q U! }. q
) `* m# b9 \4 z# I0 }`FORCE_COMMON_ALLOCATION'
5 f1 y( k" X/ M1 ~ A5 N! f这个命令跟命令行选项'-d'具有相同的效果: 就算指定了一个可重定位的输出文件('-r'),也让'ld'& k/ L% g h( a
为普通符号分配空间.
& N* S; i) R3 @! Z3 w$ i) M- _5 ^) Q, }1 e* y/ T/ ]
`INHIBIT_COMMON_ALLOCATION'3 I4 h- X- S9 S0 |, M. t8 Q9 _/ H
这个命令跟命令行选项`--no-define-common'具有相同的效果: 就算是一个不可重位输出文件, 也让
: b. A: @6 v" ^'ld'忽略为普通符号分配的空间.
- n; e; X% F( |1 p, g% G2 d% q7 g
`NOCROSSREFS(SECTION SECTION ...)'
; q4 f$ X. t% X) S! S+ F& k% K这个命令在遇到在某些特定的节之间引用的时候会产生一条错误信息.- I7 h" U, Z Z
4 F! s3 k- T9 B+ R在某些特定的程序中, 特别是在使用覆盖技术的嵌入式系统中, 当一个节被载入内存时,另外一个节
8 |9 A/ A' Y, U: S# D9 V就不会在内存中. 任何在两个节之间的直接引用都会是一个错误. 比如, 如果节1中的代码调用了另/ {! N- f3 i `9 [2 ?: t% ?
一个节中的一个函数,这就会产生一个错误./ ~" t3 w+ K* T% R! G5 }7 j
+ S% D$ ~2 ]) K. \9 [( d. |`NOCROSSREFS'命令带有一个输出节名字的列表. 如果'ld'遇到任何在这些节之间的交叉引用, 它就. Q. Y) q6 Y, h; {# `
会报告一个错误,并返回一个非零退出码. 注意, `NOCROSSREFS'命令使用输出节名,而不是输入节名.! H c0 o! B, Z/ M$ @( J& p$ j
' H, |, n5 j- k4 a8 F- {
`OUTPUT_ARCH(BFDARCH)'# B1 I+ t1 q- V* `6 m9 t1 x5 @# o6 V
指定一个特定的输出机器架构. 这个参数是BFD库中使用的一个名字. 你可以通过使用带有'-f'选项
2 `. E+ y; u3 f; x$ H+ P的'objdump'程序来查看一个目标文件的架构.- s [) }2 P7 }
: v" x; i1 z+ J为符号赋值.4 a; Y8 A* b. w* A/ t! j: r
===========================
$ m* b3 R N; s5 k2 ~' y- o; Y$ k, h7 T4 ?$ G) V$ i3 G* u6 w
你可以在一个连接脚本中为一个符号赋一个值. 这会把一个符号定义为一个全局符号.* |% O1 D2 Y& H
- `* t6 p% p: |' H/ D0 s简单的赋值.
( }4 z' W7 N, |------------------! h- I* k; j" t. k* B. f
2 P7 b) h) Z: n8 T你可以使用所有的C赋值符号为一个符号赋值.
y0 v0 u+ V+ l5 k# ]( J- G2 X6 D* B( y* F# ~ D
`SYMBOL = EXPRESSION ;'
3 C( ]# [4 k) D1 Y8 C& B* K& c`SYMBOL += EXPRESSION ;'
* [5 C5 f; a3 ]# ~`SYMBOL -= EXPRESSION ;'' T" Y, s/ s( `
`SYMBOL *= EXPRESSION ;'
2 I6 [: X! p5 M$ ^`SYMBOL /= EXPRESSION ;'
, i* d$ ?+ x) B9 t, k$ z`SYMBOL <<= EXPRESSION ;'6 q. T3 e/ R6 z4 E0 E( c; M
`SYMBOL >>= EXPRESSION ;'
3 d6 `( q6 r4 u, x6 Q`SYMBOL &= EXPRESSION ;'4 J6 p& Q& s7 v6 ~9 y7 W$ A
`SYMBOL = EXPRESSION ;'
! }# J) m/ X& w. P: ~( t0 n6 J7 g# g
第一个情况会把SYMBOL定义为值EXPRESSION. 其它情况下, SYMBOL必须是已经定义了的, 而值会作出相应的调5 ^; e7 h7 s, A; W- W
整.
0 e$ Y( ~6 |5 D$ d# l; L/ @2 E4 X/ @3 T$ D
特殊符号名'.'表示定位计数器. 你只可以在'SECTIONS'命令中使用它.! ]+ R, u3 M3 J i8 d
* ]( w0 S/ M9 a- Q" R- I9 M# e+ w; T
EXPRESSION后面的分号是必须的.
4 u/ d: b* w$ q6 X( I$ y8 W" Q. n- c9 A8 R
表达式下面会定义.3 h0 i" T7 |2 h% G
3 R0 R N/ B% z8 X% g0 p你在写表达式赋值的时候,可以把它们作为单独的部分,也可以作为'SECTIONS'命令中的一个语句,或者作为
3 e5 c4 p* Q' z$ {0 o2 c/ q' s'SECTIONS'命令中输出节描述的一个部分.
8 l2 H/ }0 f) ~9 Y P& h' k' @8 y6 N$ I! N7 Z) F3 ]+ O M
符号所在的节会被设置成表达式所在的节.1 t4 b$ y. t6 T+ C5 t2 j6 S; z
& w% m* |3 u4 K下面是一个关于在三处地方使用符号赋值的例子:
! K1 H( x. J7 Q* Y( \8 {8 R7 M/ j/ P" v, X% X
floating_point = 0;
% J1 s/ A$ C _+ ?& @3 H2 c SECTIONS
6 l: n8 P" L% E {
. E4 q. c8 b9 S+ g, t8 p( J$ X! F .text :2 R% R# Z6 x4 X& A; `6 G9 j5 T
{
J3 E3 B' ?: h Q, J8 Y, Z *(.text)
' c5 M# ?! r( }& | ` _etext = .;
0 H# Q5 x( @, H' F, E }
) E) u0 h: v' ], t _bdata = (. + 3) & ~ 3;
3 ]! H O, s5 ~, e$ D5 t# u" t1 X .data : { *(.data) }- P* H8 |4 F- m. c; l* m0 @
}3 S! F- ^$ Q- i; C$ f4 g, G! ^
3 u5 R7 U" ~: y2 @7 m# r# ~在这个例子中, 符号`floating_point'被定义为零. 符号'-etext'会被定义为前面一个'.text'节尾部的地址.
+ q/ ?! M. V. A; E& ~而符号'_bdata'会被定义为'.text'输出节后面的一个向上对齐到4字节边界的一个地址值.6 ^; b2 j+ M5 m) W) y; b/ M" U
) ?8 q$ g% ]% L
PROVIDE/ |: ]$ ]1 X! g: w. }3 A$ r, w
-------
! s3 T+ X) u) D5 K( |/ @2 {
1 H: W4 ~ \8 Y5 Y6 V& O在某些情况下, 一个符号被引用到的时候只在连接脚本中定义,而不在任何一个被连接进来的目标文件中定" u9 o6 r0 M$ ^$ X/ l' ?3 t/ g) J
义. 这种做法是比较明智的. 比如, 传统的连接器定义了一个符号'etext'. 但是, ANSI C需要用户能够把5 e$ _) p1 e' q7 ~2 h
'etext'作为一个函数使用而不会产生错误. 'PROVIDE'关键字可以被用来定义一个符号, 比如'etext', 这个" k1 c. M6 U- ~: }; @# P
定义只在它被引用到的时候有效,而在它被定义的时候无效.语法是 `PROVIDE(SYMBOL = EXPRESSION)'.. n0 r4 | T- N! h$ v2 e
( n& p% b' D: Q" N0 n8 y4 F下面是一个关于使用'PROVIDE'定义'etext'的例子:
- h* Z" P/ }9 i% u) r
7 y8 h: i' t# s SECTIONS
* S% O5 K% S7 e4 U* \ {3 w0 S. X1 D, w9 p$ \
.text :
& D3 X3 v: f8 s7 C p {
' ]* c' u4 j( P *(.text)
( x k" z5 M, F3 s _etext = .;
1 N4 l* i& Q+ I% z* Q PROVIDE(etext = .);
/ F- h/ }) V4 D; W }: S" }/ [3 j9 j9 k* K) d4 ~/ E
}' k# {! R( C4 @2 g) E z, u
9 R, m4 B" x: i* z7 `
在这个例子中, 如果程序定义了一个'_etext'(带有一个前导下划线), 连接器会给出一个重定义错误. 如果,/ u( y6 |# x1 M% H
程序定义了一个'etext'(不带前导下划线), 连接器会默认使用程序中的定义. 如果程序引用了'etext'但不
0 C7 d0 p0 d& A8 D* f+ A% `; x定义它, 连接器会使用连接脚本中的定义.& P! L. _$ E( a* T) M- p1 N2 u
4 v2 J+ H) f/ S. y' h' i+ v, W
SECTIONS命令
' {0 g4 G5 Y" C/ }6 {================+ _, }3 n+ R! Z2 C; U9 n+ [9 ^$ c
* D& @0 t5 r w1 @, _& J
'SECTIONS'命令告诉连接器如何把输入节映射到输出节, 并如何把输出节放入到内存中." |- M {! q% ?7 [# s# _
( O) c. Q s9 t" c, f* S. ]
'SECTIONS'命令的格式如下:
" P. ~1 E9 O9 q; T6 `
+ A7 ?# P+ W7 L- n/ v SECTIONS
( j% t+ }6 R& U: K/ c {
2 s% c. S9 C3 _/ ]' }8 c8 S SECTIONS-COMMAND6 Z. X" t6 o- {
SECTIONS-COMMAND4 }. `, Y# S% I* \" I
...
8 Y6 b* v3 E0 r. R* b/ u* D: w3 i }
( ]! c; p* K& z/ [. Z
3 x# C: N, M* ?2 X8 M' H6 |, F每一个SECTIONS-COMMAND可能是如下的一种:
* M8 G* ~# V( [/ b# c& V G& s+ u! O
* 一个'ENTRY'命令.
" K1 e+ _8 e) S+ U# h
; k" V- z+ L% K * 一个符号赋值.
$ s9 m' v4 c# a( T: f- |0 ]
! W+ o; `7 w" K, _) N0 B * 一个输出节描述.; h% U/ x/ D+ B% o
3 f, |4 T4 ^8 ]) c3 H# ?/ q: B* [) \ * 一个重叠描述.5 B; C4 [' x$ n0 H& I0 B/ n
1 o, u8 ?# [7 X: P'ENTRY'命令和符号赋值在'SECTIONS'命令中是允许的, 这是为了方便在这些命令中使用定位计数器. 这也可, K% ~; a& J ~/ X( u
以让连接脚本更容易理解, 因为你可以在更有意义的地方使用这些命令来控制输出文件的布局.
# y2 ~" s5 ]8 m" g- Y% w& @# S8 A6 C7 N2 S- m' C( W% u) J# z
输出节描述和重叠描述在下面描述.
# z: [3 h# S. D& {( [: M
1 M4 X2 M3 @8 w如果你在连接脚本中不使用'SECTIONS'命令, 连接器会按在输入文件中遇到的节的顺序把每一个输入节放到同
5 J1 S) `2 L' D# ^- n名的输出节中. 如果所有的输入节都在第一个文件中存在,那输出文件中的节的顺序会匹配第一个输入文件中- s$ v& t. N% c! p9 P% a9 X
的节的顺序. 第一个节会在地址零处.) [$ A1 x- y/ e# \2 W- d1 Y
1 s' m9 ?+ Z( {# a7 L8 r输出节描述
7 j8 Z1 \2 d( ~' Y--------------------------4 N, o2 J0 k- s
; T: k6 n, F7 x6 k- ^- R
一个完整的输出节的描述应该是这个样子的:
$ c3 x' r& ^4 h1 n6 T1 N% a& l
) }, d& s6 B- _/ N6 j* _) d SECTION [ADDRESS] [(TYPE)] : [AT(LMA)]3 ` O& m0 E2 ~3 V9 K' W6 X
{* P3 {5 Z, [: n1 D; T
OUTPUT-SECTION-COMMAND
! Y" N; D e- b% Q; R OUTPUT-SECTION-COMMAND
+ C3 c/ W w* @- V. y) H( P5 p ...
/ D4 d0 Q; j2 N& a% w. u } [>REGION] [AT>LMA_REGION] [:PHDR :PHDR ...] [=FILLEXP]* P* z$ m2 ?* a2 M- U
7 f5 ]# ]. p6 W/ R' F5 S大多数输出节不使用这里的可选节属性.! B" n7 Z$ _) k% |5 ^: R" U
6 i4 P0 u8 y" |SECTION边上的空格是必须的, 所以节名是明确的. 冒号跟花括号也是必须的. 断行和其他的空格是可选的.5 Q, {( z- R7 m
& Q$ h; e' U& ]% R4 A6 K% F5 p每一个OUTPUT-SECTION-COMMAND可能是如下的情况:' r& Q: t/ @5 \: l3 f ?$ D
9 q2 e. f% u N4 r7 o$ v * 一个符号赋值.
$ Y& S7 w, i1 i* Q6 d9 f3 z, K3 [: R( |( E5 ^$ r9 z5 t, @
* 一个输入节描述.
& p: Q* R L3 ^1 n$ `* f( }1 D' X; \5 w A4 s" G' R
* 直接包含的数据值.
2 c0 ]5 x3 D& h B6 o- `
9 i4 ^$ c) T! a b * 一个特定的输出节关键字.
2 W1 `) f$ Y: U }' k 0 T8 U7 Z& J4 f9 r, A* J
输出节名.
4 ~, X2 \# ]+ Z2 K8 \-------------------" Y& R% E( q2 h9 ^/ C9 U2 U3 l+ J
4 ]" A. z: Z$ ] Y, @) Q" L输出节的名字是SECTION. SECTION必须满足你的输出格式的约束. 在一个只支持限制数量的节的格式中,比如5 c) z/ t- M1 h0 A4 p% E
'a.out',这个名字必须是格式支持的节名中的一个(比如, 'a.out'只允许'.text', '.data'或'.bss').如果) h8 k: @: i8 p' }4 e5 P
输出格式支持任意数量的节, 但是只支持数字,而没有名字(就像Oasys中的情况), 名字应当以一个双引号中的
5 V% ?; Z4 Q' ~3 C& l5 j数值串的形式提供.一个节名可以由任意数量的字符组成,但是一个含有任意非常用字符(比如逗号)的字句必须
- [1 P% R/ o% L( a用双引号引起来.7 b( H! \4 n$ U9 F
" c+ a& c4 T' b6 I# v: P7 j
输出节描述
8 V2 _) s, }* Q. c- H% g--------------------------2 p5 L6 k5 O& G+ }# D% q
& `5 j. u; {6 O# I; P$ RADDRESS是关于输出节中VMS的一个表达式. 如果你不提供ADDRESS, 连接器会基于REGION(如果存在)设置它,或1 p' \+ t! T0 i) x: E
者基于定位计数器的当前值.: e2 a4 M2 E7 k" }$ R2 E& E/ M
% Q% H R$ s7 {* @; O# T
如果你提供了ADDRESS, 那输出节的地址会被精确地设为这个值. 如果你既不提供ADDRESS也不提供REGION, 那
8 N- W' Y: O/ Q) l8 { ^输出节的地址会被设为当前的定位计数器向上对齐到输出节需要的对齐边界的值. 输出节的对齐要求是所有输: Z% O. P+ t, ^3 D/ A n0 S, Q
入节中含有的对齐要求中最严格的一个./ j; o- ~' @" g. d/ x0 U% n
+ C% c! \, `, R) H6 C
比如:4 H9 a7 w: C7 K1 w# C
.text . : { *(.text) }
9 i" L) }( ]8 T5 V k+ }0 K3 j1 H+ ?8 r+ a3 A, N
和8 Y* s$ Z5 {% n8 C: p T0 U" r
.text : { *(.text) }
7 [) t% H \ x6 g: E$ j8 ?) X5 S* w+ H) o+ W) B& e7 X9 T
有细微的不同. 第一个会把'.text'输出节的地址设为当前定位计数器的值. 第二个会把它设为定位计数器的4 q7 [: X" T; X6 F
当前值向上对齐到'.text'输入节中对齐要求最严格的一个边界.4 h+ q. ]* `+ t U
+ W& J' m# U4 X! _% P X) L7 ~ADDRESS可以是任意表达式; 比如,如果你需要把节对齐对0x10字节边界,这样就可以让低四字节的节地址值为
7 O' M" g) Y9 Q3 C: W* S1 Z7 y零, 你可以这样做:
: y4 Q' }. l$ e& {6 t
) }5 k4 |/ x: F% _! K .text ALIGN(0x10) : { *(.text) }
+ g- F( ~, B8 H9 F! C0 w$ h2 e4 I+ G' L. G m0 \8 d/ ^3 X
这个语句可以正常工作,因为'ALIGN'返回当前的定位计数器,并向上对齐到指定的值./ C( J" H4 u: m# G5 ^
5 B+ ?' I5 b4 o k
指定一个节的地址会改变定位计数器的值.
5 ^6 S; c+ u6 \# W, p; K# [ W7 I( l) k( q# d9 H: @
输入节描述
. H* {. S* t6 p9 y. g+ }-------------------------
3 Q2 ?+ t- E1 \* w; N, }3 c% \. V& q- G" ^
最常用的输出节命令是输入节描述.5 h+ J h$ Y; l- l1 }+ v: E
+ D. M. L5 |9 x输入节描述是最基本的连接脚本操作. 你使用输出节来告诉连接器在内存中如何布局你的程序. 你使用输入节
R# x2 w/ r# V/ [3 z- K' V来告诉连接器如何把输入文件映射到你的内存中.
* _: p+ g9 Y5 f
) M7 k+ Y, s6 u! Z输入节基础
+ c8 _, ]; L9 _, K6 u/ J' y4 |---------------------------+ |: M. o7 P; ]; y
0 [ h+ ^' N2 U% L+ Z1 m" ]一个输入节描述由一个文件名后跟有可选的括号中的节名列表组成.
! p, N* e" v4 Q% Y0 H$ Q7 P' U; W0 B- x
文件名和节名可以通配符形式出现, 这个我们以后再介绍.) D# X6 B+ h# ]* A/ L
6 \* n/ U: ? h# u2 H8 b1 Q" j4 o最常用的输入节描述是包含在输出节中的所有具有特定名字的输入节. 比如, 包含所有输入'.text'节,你可以
! g ]* `2 ~* G, T' y这样写:
: y4 j& S4 j0 b
$ j' G$ w' ?2 p( X0 v7 o$ X *(.text)
* m/ T y9 N- H) G+ F6 ^7 }, G G! y
这里,'*'是一个通配符,匹配所有的文件名. 为把一部分文件排除在匹配的名字通配符之外, EXCLUDE_FILE可. H- I' P4 l/ ?, ~- d7 k4 F
以用来匹配所有的除了在EXCLUDE_FILE列表中指定的文件.比如:
4 E8 m2 |4 w; q: Q9 r4 A3 H6 Z& s+ P. _6 x5 R" p
(*(EXCLUDE_FILE (*crtend.o *otherfile.o) .ctors))
_0 X; e" E: u- `) q/ w) B1 u7 b$ C
会让除了`crtend.o'文件和`otherfile.o'文件之外的所有的文件中的所有的.ctors节被包含进来.. Y |& [, L( A
6 R, e+ ]0 e& I+ v有两种方法包含多于一个的节:
2 `/ t; b x4 V$ |+ t- M
' m8 N( S4 M- J; } *(.text .rdata)
/ }; O; m7 a& B( [ *(.text) *(.rdata)
+ C- @, O1 R/ }6 @1 O% A9 e( H) v4 ~! z/ V: \
上面两句的区别在于'.text'和'.rdata'输入节的输出节中出现的顺序不同. 在第一个例子中, 两种节会交替, C5 R) c- |) \3 b; ]
出现,并以连接器的输入顺序排布. 在第二个例子中,所有的'.text'输入节会先出现,然后是所有的'.rdata'节.( c' {: a. H f7 j c8 a+ J% A/ w. [
; x; ^' r5 ?, j( b- s2 n" L3 c
你可以指定文件名,以从一个特定的文件中包含节. 如果一个或多个你的文件含有特殊的数据在内存中需要特8 u h' T, \- L: K; R
殊的定位,你可以这样做. 比如:! {; b4 L$ a2 s7 ~) d9 e d4 e+ W
6 z0 }- M+ f' L; }
data.o(.data)
?! X' `' g! J- `
" H4 s* h8 ?- @ u如果你使用一个不带有节列表的文件名, 那输入文件中的所有的节会被包含到输出节中. 通常不会这样做, 但
5 L' \7 { h4 ~$ a是在某些场合下这个可能非常有用. 比如:& b c4 F4 q' U9 i1 \3 |/ o
$ H |3 ^& |2 L1 B: b7 }: A
data.o8 B) A' P3 j2 `! C
0 U+ R4 _* B6 p0 O
当你使用一个不含有任何通配符的文件名时, 连接器首先会查看你是否在连接命令行上指定了文件名或者在
0 ]7 o# A _ f'INPUT'命令中. 如果你没有, 连接器会试图把这个文件作为一个输入文件打开, 就像它在命令行上出现一样.
5 ~( W7 B1 L$ q; G) M: d注意这跟'INPUT'命令不一样, 因为连接器会在档案搜索路径中搜索文件.9 W1 l& c3 F" ?' k
( u8 G3 E( P! n1 a. M+ @0 ~$ Z
输入节通配符; O/ k1 Q$ J) b2 N* {0 I
---------------------------------
# F/ x: B: K! u5 w" z9 z, I. `/ `1 V9 p# U. _
在一个输入节描述中, 文件名或者节名,或者两者同时都可以是通配符形式.
3 L# X( b. A( T5 j, S) Y3 E' D& t. n, B3 L+ V; p, h7 g
文件名通配符'*'在很多例子中都可以看到,这是一个简单的文件名通配符形式.
, K# `' R# N* p
$ B3 r8 ]0 Q. }( p4 T9 w; z通配符形式跟Unix Shell中使用的一样.
/ ?. L/ A" ]7 C. q
4 g# P- `2 _; _2 y, M( g`*'
: ]1 y0 E9 `2 R) g H5 Z" }- s8 J匹配任意数量的字符.
) K. K5 @; C+ b/ M8 Z
6 I/ q. r( W0 {- I0 J |0 A`?'
8 ]* v$ u) j; C6 |/ G匹配单个字符.
0 N; @0 T/ b. @# d5 x9 J, ~) o/ p! c
`[CHARS]'1 j( H4 C+ K' p* H
匹配CHARS中的任意单个字符; 字符'-'可以被用来指定字符的方讧, 比如[a-z]匹配任意小字字符.; M$ C+ W3 G1 _- T7 g9 U# `# F) n
/ d) A1 Y8 z) t5 w+ j7 G`\'" p1 ], Z0 ]$ P( a, P' I$ W
转义其后的字符.
$ u1 W, J2 |. V, q7 E3 u+ [) e
当一个文件名跟一个通配符匹配时, 通配符字符不会匹配一个'/'字符(在UNIX系统中用来分隔目录名), 一个# p" q8 H- T- f/ ]) P
含有单个'*'字符的形式是个例外; 它总是匹配任意文件名, 不管它是否含有'/'. 在一个节名中, 通配符字1 I* I& u ~0 m5 s( |6 o0 p6 s
符会匹配'/'字符.
: G V: Y! }$ e6 r( J
) B# @: s( y5 ~6 @! B; _ Q5 E( l文件名通配符只匹配那些在命令行或在'INPUT'命令上显式指定的文件. 连接器不会通过搜索目录来展开通配2 S( N" F- ]! Z/ B. A( U
符.
5 m& p1 ?; t+ F7 I
; l: _( A: P6 ]* p如果一个文件名匹配多于一个通配符, 或者如果一个文件名显式出现同时又匹配了一个通配符, 连接器会使用
5 j8 \: U% V; u3 P0 m! q7 G第一次匹配到的连接脚本. 比如, 下面的输入节描述序列很可能就是错误的,因为'data.o'规则没有被使用:
6 `2 J& p7 Y" d) d3 P! M3 o g9 a! [0 t
.data : { *(.data) }5 B9 H8 _# u: N
.data1 : { data.o(.data) }
/ Q5 v8 \- b9 }
R! `7 s3 X3 D% z; R4 @* j- _* |通常, 连接器会把匹配通配符的文件和节按在连接中被看到的顺序放置. 你可以通过'SORT'关键字改变它, 它
/ v2 W% R% J6 l$ i3 l' t出现在括号中的通配符之前(比如, 'SORT(.text*)'). 当'SORT'关键字被使用时, 连接器会在把文件和节放到
6 Z% n/ v+ \0 ]8 @% @% P# @# Y输出文件中之前按名字顺序重新排列它们.' f/ I6 d& X* f3 L5 {7 P
: c$ h+ M' v2 o0 K; C如果你对于输入节被放置到哪里去了感到很困惑, 那可以使用'-M'连接选项来产生一个位图文件. 位图文件会. @' N) S2 X, ~1 @9 s
精确显示输入节是如何被映射到输出节中的.
6 m& H7 D% @* O6 D7 i
5 H5 J& e+ U( S! s4 W! c) w这个例子显示了通配符是如何被用来区分文件的. 这个连接脚本指示连接器把所有的'.text'节放到'.text'中, 把所有的'.bss'节放到'.bss'. 连接器会把所有的来自文件名以一个大写字母开始的文件中的'.data'节放进'.DATA'节中; 对于所有其他文件, 连接器会把'.data'节放进'.data'节中.
0 g( i) I: X" i% E4 l& ]
/ x! D7 {8 {% w% s& e SECTIONS {" D8 E! s. H. F) w6 Y( r2 r
.text : { *(.text) }
7 \9 n# q5 W- B; x2 z .DATA : { [A-Z]*(.data) }3 A8 i: a& [; `
.data : { *(.data) }0 X6 i7 `, j. s) F( B0 ~$ K
.bss : { *(.bss) }
$ X, |' r: c& R3 u }
: l9 w9 {7 Z0 K$ o
" a) U" {: e# R( ?. }输入节中的普通符号.
4 V1 Z8 F( @& c, h: o5 v% w' h4 A' E9 V-----------------------------------1 W t. W# ]" ~9 e' J* x2 l
$ ~" Y( Z: v: ], ]" d5 `& V! ~: i对于普通符号,需要一个特殊的标识, 因为在很多目标格式中, 普通符号没有一个特定的输入节. 连接器会把1 K4 x# ~, v2 o6 r0 G7 W, I/ Y
普通符号处理成好像它们在一个叫做'COMMON'的节中.
& h% d; G% h/ d8 c6 T; b4 b, y6 v
, L! U/ r& {% P# {你可能像使用带有其他输入节的文件名一样使用带有'COMMON'节的文件名。你可以通过这个把来自一个特定输
. }7 |5 N0 h, N+ Y: N入文件的普通符号放入一个节中,同时把来自其它输入文件的普通符号放入另一个节中。% t( b9 `, ^- X8 x$ s4 Y# |) J
4 I: u; K! o8 D% O
在大多数情况下,输入文件中的普通符号会被放到输出文件的'.bss'节中。比如:' Z. {, q4 J# h+ p
7 T, E1 b- G9 E$ P, k2 [/ } .bss { *(.bss) *(COMMON) }
% Z" K0 V, W! x8 H, z
3 w2 f$ B- l$ b% [6 z有些目标文件格式具有多于一个的普通符号。比如,MIPS ELF目标文件格式区分标准普通符号和小普通符号。3 J/ t' Z0 d- q
在这种情况下,连接器会为其他类型的普通符号使用一个不同的特殊节名。 在MIPS ELF的情况中, 连接器
0 ]& [# q0 W7 K( F) `+ @( ^7 V为标准普通符号使用'COMMON',并且为小普通符号使用'.common'。这就允许你把不同类型的普通符号映射到4 m8 }$ D' b0 c. W
内存的不同位置。
! c) o4 L, H! ~9 l, d0 Y; y, Y* U- v( V3 a" O, g8 u0 t- s
在一些老的连接脚本上,你有时会看到'[COMMON]'。这个符号现在已经过时了, 它等效于'*(COMMON)'。
# F a- M: f5 {( z) q$ K
# c b. l1 b( q* S& ]% i输入节和垃圾收集/ ^. e" q5 M) t
--------------------------------------- v3 v) a) y" X2 b- G6 k! F
: l8 I5 L; Z) d, V8 n当连接时垃圾收集正在使用中时('--gc-sections'),这在标识那些不应该被排除在外的节时非常有用。这% p, _/ f$ P1 z- h( @ \6 W. n
是通过在输入节的通配符入口外面加上'KEEP()'实现的,比如'KEEP(*(.init))'或者'KEEP(SORT(*)(.sorts))
! P2 _1 o+ z" r7 }8 T b'。
0 T- m3 u+ }3 N1 `, I7 L& K- B0 B! p
输入节示例
) s, z4 [4 c! `$ M$ T---------------------1 `* S, [" \: x$ V4 ]9 U
3 B: `) h- J/ z1 U+ ]% K! z接下来的例子是一个完整的连接脚本。它告诉连接器去读取文件'all.o'中的所有节,并把它们放到输出节
: `: w- l8 h3 Z! x'outputa'的开始位置处, 该输出节是从位置'0x10000'处开始的。 从文件'foo.o'中来的所有节'.input1'
& I: q" n {5 H7 Q在同一个输出节中紧密排列。 从文件'foo.o'中来的所有节'.input2'全部放入到输出节'outputb'中,后面
" D. S. G+ n* p! ^8 E) J跟上从'foo1.o'中来的节'.input1'。来自所有文件的所有余下的'.input1'和'.input2'节被写入到输出节1 m2 a" i( G, t% C: r1 e
'outputc'中。
" e$ F# ~# ^& e& n% M
/ i* z4 |. p- x) } SECTIONS {$ W7 m O0 Y6 ?- ~5 n2 i! _
outputa 0x10000 :4 M7 |; x, O; Z
{( f+ m+ A1 S- G; u" g
all.o
# M8 f& u! J$ i7 o# u foo.o (.input1)
: J. A: _( j! D }% E3 }! b$ l$ v$ E
outputb :
8 ?) i6 Y0 |0 m4 {0 m) W6 k {
/ j. E# u: @9 Q4 L: v foo.o (.input2)
. q2 I& i* u1 m$ u. Z1 q foo1.o (.input1)+ b7 A; k0 P& E3 S/ B( h7 I! _' N
}
- E* q& M+ H c) Z/ a- f) j outputc :
# z) q; d: d& S {$ ~$ O: U; @5 S# A& S) B
*(.input1)
' j. Q# E/ {5 B7 C *(.input2)5 c# k1 x1 w6 K1 ~1 F% z
}
; C0 ~* b: G6 y }$ R5 |: a @) S4 Y$ v6 @: I7 S
% P$ H4 Z, Q* B
输出节数据
3 K; I: Y+ t( _0 r-------------------
; m D% T7 S9 P6 D; N) S# b
8 u5 c0 n$ }1 K7 W5 h6 f3 d) s你可以通过使用输出节命令'BYTE','SHORT','LONG','QUAD',或者'SQUAD'在输出节中显式包含几个字节的数据
8 i8 q; |9 g$ |( e每一个关键字后面都跟上一个圆括号中的要存入的值。表达式的值被存在当前的定位计数器的值处。
5 U& J/ ]/ Z1 }8 T0 l' D2 P- o8 j" N5 j- J6 \. b5 y
‘BYTE’,‘SHORT’,‘LONG’‘QUAD’命令分别存储一个,两个,四个,八个字节。存入字节后,定位计% }0 C. C! q) A$ i' ?
数器的值加上被存入的字节数。& t; c7 K- E9 {7 O3 |, Y: F3 F
3 f& f' Q. r( X& C比如,下面的命令会存入一字节的内容1,后面跟上四字节,其内容是符号'addr'的值。) u6 K' C2 S2 @6 D n# F& D
, k& @0 F& ]( W1 Q6 F( a, t
BYTE(1)
9 w) t5 u' w" x& q4 U$ H7 q* P LONG(addr)4 g6 A D7 q4 C
1 ?# Y, }& {% Q) s: D8 A' n
当使用64位系统时,‘QUAD’和‘SQUAD’是相同的;它们都会存储8字节,或者说是64位的值。而如果软硬件
6 N" s+ e, U }6 U系统都是32位的,一个表达式就会被作为32位计算。在这种情况下,‘QUAD’存储一个32位值,并把它零扩展2 L/ y% t( r9 x+ Y7 W, y4 U+ ?
到64位, 而‘SQUAD’会把32位值符号扩展到64位。
2 R/ O3 n8 p) J
' t( d% N$ ~$ u) |如果输出文件的目标文件格式有一个显式的endianness,它在正常的情况下,值就会被以这种endianness存储' S% W/ |0 t/ ]3 |
当一个目标文件格式没有一个显式的endianness时, 值就会被以第一个输入目标文件的endianness存储。- d8 W! b# a# h+ D& {
4 q0 M/ F# q/ v/ d+ J3 |. a- q% Z注意, 这些命令只在一个节描述内部才有效,而不是在它们之间, 所以,下面的代码会使连接器产生一个错
& h$ ~: P' I+ x5 x _误信息:: P* q1 K; V$ C% ^- G5 Q& B* J
6 A, z& k$ H- C0 |' m* ^ SECTIONS { .text : { *(.text) } LONG(1) .data : { *(.data) } }
: X: ^! \$ K5 O5 R& i3 s# o7 N1 T1 x) w, c2 X H! H& w j
而这个才是有效的:
|% E( B+ g' N5 _7 d6 ]
9 \: O' N& x5 E7 p SECTIONS { .text : { *(.text) ; LONG(1) } .data : { *(.data) } }
5 v7 r, k: l+ j+ z* q5 x7 d) Z% n X" p c+ u1 F- I5 H! G
你可能使用‘FILL’命令来为当前节设置填充样式。它后面跟有一个括号中的表达式。任何未指定的节内内存
5 n8 D: x2 c6 t! y. ~0 k8 k区域(比如,因为输入节的对齐要求而造成的裂缝)会以这个表达式的值进行填充。一个'FILL'语句会覆盖到
2 y; d4 i7 [' f$ |1 P! X m) k3 v它本身在节定义中出现的位置后面的所有内存区域;通过引入多个‘FILL’语句,你可以在输出节的不同位置
1 a' C# u$ K7 j3 A" l- y; _4 S/ m+ {拥有不同的填充样式。0 G( L0 F6 P3 w. t5 g
5 n4 i( o0 T/ [) {. ]4 u5 L这个例子显示如何在未被指定的内存区域填充'0x90':+ l7 g0 J2 @8 O A
, T: C9 y6 h* r! B' B$ H3 ~
FILL(0x90909090)
: K$ c1 a& _% V! j' T+ T! v+ `8 q- B
7 D5 I2 u4 x- o‘FILL’命令跟输出节的‘=FILLEXP’属性相似,但它只影响到节内跟在‘FILL’命令后面的部分,而不是 G$ o" K% l; b: H
整个节。如果两个都用到了,那‘FILL’命令优先。1 W$ N- k6 s. n; k/ f) o+ U7 B( x
; \' l0 a* q" Q& R- E6 {, Q/ ?输出节关键字
- T: G3 \$ Z% _, B( B8 \-----------------------
' P1 m9 P$ K; P: C2 m
% ~$ t8 e! ?" [有两个关键字作为输出节命令的形式出现。
* e( ]! Y4 v/ x( p. I! D/ [( O* T4 B* l. Q. `
`CREATE_OBJECT_SYMBOLS'1 r& f" f; c" q$ k# y# I
这个命令告诉连接器为每一个输入文件创建一个符号。而符号的名字正好就是相关输入文件的名字。
8 l! z V* t8 W$ c2 k2 W" l" \而每一个符号的节就是`CREATE_OBJECT_SYMBOLS'命令出现的那个节。
: P0 C$ X8 }& I! M# e& g
. @( B8 Z! W7 X( \# T这个命令一直是a.out目标文件格式特有的。 它一般不为其它的目标文件格式所使用。
5 b& U0 E1 G9 u* n. o6 j/ F/ K5 {+ W! p; n6 V) W6 f
`CONSTRUCTORS'
4 t. P: [' O% _5 v* V, O当使用a.out目标文件格式进行连接的时候, 连接器使用一组不常用的结构以支持C++的全局构造函+ O, P, _, ~0 [: k, c. u
数和析构函数。当连接不支持专有节的目标文件格式时, 比如ECOFF和XCOFF,连接器会自动辩识C++
! ^+ ?* g( c: |全局构造函数和析构函数的名字。对于这些目标文件格式,‘CONSTRUCTORS’命令告诉连接器把构造" m) M( Y8 v9 n7 r
函数信息放到‘CONSTRUCTORS’命令出现的那个输出节中。对于其它目标文件格式,‘CONSTRUCTORS’2 P) e d. p+ u4 x* @/ N. i u6 q. m
命令被忽略。
; Y( Q& L8 {8 u" {! m6 Q5 K& \; l5 \+ T$ r+ ^. |* V0 {' @4 `6 R
符号`__CTOR_LIST__'标识全局构造函数的开始,而符号`__DTOR_LIST'标识结束。这个列表的第一个/ W( R. F. C7 j9 Q
WORD是入口的数量,紧跟在后面的是每一个构造函数和析构函数的地址,再然后是一个零WORD。编译
+ V( R# _5 t* L$ s+ f3 a4 u% a器必须安排如何实际运行代码。对于这些目标文件格式,GNU C++通常从一个`__main'子程序中调用
/ K8 S4 }; H' A# x) y) L构造函数,而对`__main'的调用自动被插入到`main'的启动代码中。GNU C++通常使用'atexit'运行
* f/ c9 y1 O+ \析构函数,或者直接从函数'exit'中运行。8 V4 ~- F' p! K" |! F2 `
+ i9 e ~1 n% |* D# l/ \5 ?
对于像‘COFF’或‘ELF’这样支持专有节名的目标文件格式,GNU C++通常会把全局构造函数与析构
% c2 O& B0 |* y2 M2 F: m函数的地址值放到'.ctors'和'.dtors'节中。把下面的代码序列放到你的连接脚本中去,这样会构建
/ ?. Z6 R7 v2 v, x; } P出GNU C++运行时代码希望见到的表类型。6 w1 n1 ]; s P
# c; Z1 a# C+ @ __CTOR_LIST__ = .;
8 y# L Z# v6 o+ v7 |2 g LONG((__CTOR_END__ - __CTOR_LIST__) / 4 - 2)
' S/ j6 M4 o! w3 _& P *(.ctors)# s' n1 t( o0 ^; A$ {3 L
LONG(0)/ w/ G' W) l5 E! ^) b
__CTOR_END__ = .;# C! s, F" a: ]8 K- t9 t
__DTOR_LIST__ = .;
2 _6 A/ h$ ?" [# g5 P LONG((__DTOR_END__ - __DTOR_LIST__) / 4 - 2). ]6 f9 I% V- z* E; h7 f8 K* D
*(.dtors)% f/ O. a- B7 A8 R" p2 u2 ]
LONG(0)
4 Z7 ] ?% K/ J/ |" y __DTOR_END__ = .;0 f1 j1 n8 [1 S1 f3 ^3 [$ {: q- y8 _
" C" _0 c/ ^$ K如果你正使用GNU C++支持来进行优先初始化,那它提供一些可以控制全局构造函数运行顺序的功能,
( n; j8 y, o; D- a0 s, Y你必须在连接时给构造函数排好序以保证它们以正确的顺序被执行。当使用'CONSTRUCTORS'命令时,
* O' ~2 D$ ]5 y2 T( p! I: _替代为`SORT(CONSTRUCTORS)'。当使用'.ctors'和'dtors'节时,使用`*(SORT(.ctors))'和
9 h' J8 \5 @6 ]' a" L4 W5 e`*(SORT(.dtors))' 而不是`*(.ctors)'和`*(.dtors)'。% j$ L) V) L1 F" z8 u
! j$ Y; l! N4 Y( H% W通常,编译器和连接器会自动处理这些事情,并且你不必亲自关心这些事情。但是,当你正在使用
6 b& B5 F- z! s& C" xC++,并自己编写连接脚本时,你可能就要考虑这些事情了。* b5 ?( j! E! p9 ]0 W$ E
1 O* K1 f2 m: }. C
输出节的丢弃。1 C5 `1 D T* l9 [: L+ E
-------------------------3 |) x% x+ ?5 y H& f2 A
( _9 f9 s- Y; S8 s+ _; \
连接器不会创建那些不含有任何内容的输出节。这是为了引用那些可能出现或不出现在任何输入文件中的输入* V4 w m5 S4 n
节时方便。比如:
9 B0 L/ |, u) z- \+ b5 P+ c8 W7 i0 J* d4 Z+ j) U$ k2 p& }
.foo { *(.foo) }) J& y- X, ?, o$ T3 b. u1 k& [! q
2 N7 u- F, A7 U6 o9 o8 A$ P+ I如果至少在一个输入文件中有'.foo'节,它才会在输出文件中创建一个'.foo'节
6 P1 y7 F7 O8 s* Q$ m- s4 n
, r6 S* _/ K( P9 A+ m如果你使用了其它的而不是一个输入节描述作为一个输出节命令,比如一个符号赋值,那这个输出节总是被
& u/ ~ Q) A6 f4 X6 E创建,即使没有匹配的输入节也会被创建。6 C' y ^% L4 S$ e0 q5 c: |
5 Y8 ]: x$ Z4 i `一个特殊的输出节名`/DISCARD/'可以被用来丢弃输入节。任何被分配到名为`/DISCARD/'的输出节中的输入; D/ @* Q, o' i, b5 |/ ]$ o9 P
节不包含在输出文件中。
3 s ]7 \0 F; ^+ m) T2 w x: H2 m+ n# Q0 X, u
输出节属性0 }* ?, ~' S: ^4 \: J
-------------------------2 z7 m5 t" i% P! u1 P4 R
' {- R7 W' v" P4 ?% c. d& n7 }
上面,我们已经展示了一个完整的输出节描述,看下去就象这样:; F( [% o0 p+ P
+ w3 e! x0 B9 {) f# n! T( C7 e, D
SECTION [ADDRESS] [(TYPE)] : [AT(LMA)]; V1 L4 k3 X- S& ~% Y, d2 b
{
6 k: T( }, e# s* Y4 T1 x OUTPUT-SECTION-COMMAND
* S" @# C- h* g* S+ a6 u" w OUTPUT-SECTION-COMMAND4 X$ [1 J9 w2 ]4 E6 T
...
. F1 Q0 J6 p" A( N } [>REGION] [AT>LMA_REGION] [:PHDR :PHDR ...] [=FILLEXP]# I3 _+ @/ z2 P
* K2 ~4 t) x1 [6 ^& L' A我们已经介绍了SECTION, ADDRESS, 和OUTPUT-SECTION-COMMAND. 在这一节中,我们将介绍余下的节属性。
( c" t% d% Z6 ~) r& q. h- v) {4 `3 E* i3 u% }
输出节类型
6 [3 s7 p& Z' G...................
Q' n) g0 s$ ~; x+ X9 t( Q' X U# S- }4 q# |
每一个输出节可以有一个类型。类型是一个放在括号中的关键字,已定义的类型如下所示:
# k* U, z& o3 N$ Z7 Z
" R* E9 l. O% s`NOLOAD' Q u+ Q. r6 v) [5 Z$ b
这个节应当被标式讵不可载入,所以当程序运行时,它不会被载入到内存中。& k- i4 X; \+ r/ o2 ~8 p
) _; a# C. O, R- c; S
`DSECT'/ Q" U+ I+ f/ j0 O2 z" [/ Q
`COPY'
) _ M$ a! l# g& @- p`INFO'
8 d, }9 b# @0 L8 [9 ^' Y: b2 u`OVERLAY'
8 ], a0 Y# b3 A, i) v2 |支持这些类型名只是为了向下兼容,它们很少使用。它们都具有相同的效果:这个节应当被标式讵不8 i S0 o* M, f8 A
可分配,所以当程序运行时,没有内存为这个节分配。6 }. S' w% ]0 U0 C
7 a8 u. n$ h+ V) g c8 k# x连接器通常基于映射到输出节的输入节来设置输出节的属性。你可以通过使用节类型来重设这个属性,: Z: I* j5 ^8 o+ j/ i1 ]4 Y
比如,在下面的脚本例子中,‘ROM’节被定址在内存地址零处,并且在程序运行时不需要被载入。
) Q7 k" ~' d# M+ e1 n‘ROM’节的内容会正常出现在连接输出文件中。! n/ B; f& O$ u) w+ Z1 q, I
) r! f( K- q. R4 k( S% L( o3 n1 `1 i SECTIONS { P8 @3 ~9 P% }: |! i2 D
ROM 0 (NOLOAD) : { ... }
3 D5 L2 @% p2 Y1 l# t5 ` ..., Q1 Q# x% w8 Y$ V) o
}
0 K2 P$ E' f5 C
1 V/ h& h8 P1 a8 q4 H. c. G! f输出节LMA- C/ k" `" [& e; d5 `
..................
. g% r+ w$ E8 }, z v
1 m4 }& B) O" j每一个节有一个虚地址(VMA)和一个载入地址(LMA);出现在输出节描述中的地址表达式设置VMS Z" K2 W. W+ b5 d0 y# a
/ q* s1 L- F* Y) |; X2 C连接器通常把LMA跟VMA设成相等。你可以通过使用‘AT’关键字改变这个。跟在关键字‘AT’后面的表达式
# a8 q! m# P$ k. n; m0 LLMA指定节的载入地址。或者,通过`AT>LMA_REGION'表达式, 你可以为节的载入地址指定一个内存区域。
: |( X. s5 ]6 ^8 Y. U/ x. j4 T/ k
4 Q C4 j9 }! x这个特性是为了便于建立ROM映像而设计的。比如,下面的连接脚本创建了三个输出节:一个叫做‘.text’7 k3 P3 m. S% A5 Q6 @3 Q
从地址‘0x1000’处开始,一个叫‘.mdata’,尽管它的VMA是'0x2000',它会被载入到'.text'节的后面,最: l4 C& x4 p/ m* l2 l( X* n
后一个叫做‘.bss’是用来放置未初始化的数据的,其地址从'0x3000'处开始。符号'_data'被定义为值: P. @9 a6 M! U1 V T
'0x2000', 它表示定位计数器的值是VMA的值,而不是LMA。
6 h9 C l, Y, L* [3 G6 L1 ^$ v, k/ m. w/ o, H, A
SECTIONS
! t2 \) P" _7 s {
5 q2 q# I; q5 L: {: w .text 0x1000 : { *(.text) _etext = . ; }7 e a$ _, f0 Y
.mdata 0x2000 :
( p+ o- B( j) i+ O! E0 ]& y: Q7 o AT ( ADDR (.text) + SIZEOF (.text) )' ]% m$ e9 T- I2 H: P
{ _data = . ; *(.data); _edata = . ; }
' T- l$ K* t; [+ F/ H .bss 0x3000 :& X, v9 c' Z3 i: S8 N, B, C
{ _bstart = . ; *(.bss) *(COMMON) ; _bend = . ;}3 G1 h' {( r: Z( s: c. I
}8 S7 x8 u/ |* E3 @2 u$ `1 o% q! h
% Z' W* V8 x0 ~$ g2 ]( D
这个连接脚本产生的程序使用的运行时初始化代码会包含象下面所示的一些东西,以把初始化后的数据从ROM! {+ C. v3 f9 E% s9 h7 p
映像中拷贝到它的运行时地址中去。注意这节代码是如何利用好连接脚本定义的符号的。9 J2 M" j* u" z9 `1 s
7 @* U5 o' A, j+ R+ }: T
extern char _etext, _data, _edata, _bstart, _bend;- U* W2 j4 l4 w' t
char *src = &_etext;' s7 }: i1 K3 ^8 s3 g1 q/ S
char *dst = &_data; b" `; `' w1 ]7 b
# Y; E% [% q1 o6 {1 U5 j1 H7 e
/* ROM has data at end of text; copy it. */
5 H s# z/ O! s4 p! C while (dst < &_edata) {5 W7 O$ E- E, O+ m0 S j: Y
*dst++ = *src++;& H+ y$ o: O' {3 j! G
}
& O/ [* l( D% ^* B; A) c # L0 I; T; |% i0 g9 R# L( _
/* Zero bss */
* H/ R6 y u' ~% p# f for (dst = &_bstart; dst< &_bend; dst++)
3 m" ]8 z% N: {7 X& Z& u$ N- o *dst = 0;( b6 K3 Y; p; g! |" H9 ?
2 a( q( |* S4 H" G输出节区域. h+ N' [! q5 C* s7 A) o
.....................
$ ]4 Q$ ]. C% r. v" K8 D% T/ b
% \, N3 R$ e8 R ] \+ x你可以通过使用`>REGION'把一个节赋给前面已经定义的一个内存区域。
7 x) p @) X4 R. N4 j$ U# D/ Z' m9 R! S5 q; ~: t2 l) C
这里有一个简单的例子:6 b5 ]7 K4 M7 t2 a9 P# L7 \
2 E O5 _# M4 S$ m/ v4 z MEMORY { rom : ORIGIN = 0x1000, LENGTH = 0x1000 }
/ q. ~* f! E; x, B/ E SECTIONS { ROM : { *(.text) } >rom }4 B0 W6 k" U7 t# d
; \" V( ~ ?% b& }( U; N
输出节Phdr4 Q2 t* [+ }8 d/ ]: R4 F& b
...................
$ @+ ~. t, d, @$ o1 p6 m( P5 K8 S$ f2 W! I: ^
你可以通过使用`:PHDR'把一个节赋给前面已定义的一个程序段。如果一个节被赋给一个或多个段,那后来分# j6 P( k. }6 y0 g9 g
配的节都会被赋给这些段,除非它们显式使用了':PHDR'修饰符。你可以使用':NONE'来告诉连接器不要把节' `/ {5 o$ x' N
放到任何一个段中。2 }' C) b: m: u. H( h, g
b: ]7 j& u2 l) z8 h- i这儿有一个简单的例子:
) Y v+ A/ i' T$ }0 M3 z) ?# V* e( f" L2 r
PHDRS { text PT_LOAD ; }
2 r; Q3 U+ I# T c9 W8 o+ l+ o' P+ L7 t SECTIONS { .text : { *(.text) } :text }
$ U1 ~% }+ U8 I) m" ]/ n4 @3 s6 q( `4 }7 k
输出段填充/ z2 e9 E- p" e8 ^
...................8 ~2 T, B1 p1 V2 W
5 q3 L" c% n: y. S! L' a% N1 h你可以通过使用'=FILLEXP'为整个节设置填充样式。FILLEXP是一个表达式。任何没有指定的输出段内的内存5 E1 g7 g. {+ M! t3 n( d: Y
区域(比如,因为输入段的对齐要求而产生的裂缝)会被填入这个值。如果填充表达式是一个简单的十六进制' j# C C6 C! E/ B1 B
值,比如,一个以'0x'开始的十六进制数字组成的字符串,并且尾部不是'k'或'M',那一个任意的十六进制数
; Q2 B ^7 E$ ]: }2 d$ S5 k字长序列可以被用来指定填充样式;前导零也变为样式的一部分。对于所有其他的情况,包含一个附加的括号
9 Z/ n1 W! O3 L6 x1 {4 A或一元操作符'+',那填充样式是表达式的最低四字节的值。在所有的情况下,数值是big-endian., i3 R. `4 u7 f# `, w( O
- {7 N7 z0 ~. H: k& C& E, X你还可以通过在输出节命令中使用'FILL'命令来改变填充值。( x: q% M; Q0 V! f d3 {& ]
+ C+ U; B7 M% U
这里是一个简单的例子:
2 ^* D- x+ ~8 d$ L1 I! \' O$ { SECTIONS { .text : { *(.text) } =0x90909090 }: o- e3 z3 |. x4 n" g& Z. W+ g$ t
! b; `- }+ Y' c4 I0 D p
覆盖描述6 ]" f9 k/ H( s! P# g
-------------------
2 L, h9 g4 _$ o# _ L; J
, J6 @5 b5 H, R一个覆盖描述提供一个简单的描述办法,以描述那些要被作为一个单独内存映像的一部分载入内存,但是却要" T' G2 k# g# G$ m
在同一内存地址运行的节。在运行时,一些覆盖管理机制会把要被覆盖的节按需要拷入或拷出运行时内存地址,
( W5 n. b4 S, K* `# a5 O并且多半是通过简单地处理内存位。 这个方法可能非常有用,比如在一个特定的内存区域比另一个快时。$ F! T. f# i) ?2 F- U/ A: d
% j( D- U( B4 E" V" q4 p( ~覆盖是通过‘OVERLAY’命令进行描述。‘OVERLAY’命令在‘SECTIONS’命令中使用,就像输出段描述一样。$ E J. u5 J3 b6 F; V' x
‘OVERLAY’命令的完整语法如下:
; H; P8 m# Z. q# |6 K4 I( y! [' q$ N. j/ p, ?
OVERLAY [START] : [NOCROSSREFS] [AT ( LDADDR )]2 D, s8 z; E. F8 _% G
{+ u. B+ Y/ O1 r, H
SECNAME1
% T/ V2 Z. r" L! ~' Z! X {
" ^& p2 g8 ^* i. d' Z& V) W OUTPUT-SECTION-COMMAND
& L/ }" a" L- L7 S7 t6 f/ s OUTPUT-SECTION-COMMAND; b% C2 e" M# }. _1 V) U5 O
...
$ c. V5 H3 N o } [:PHDR...] [=FILL]
7 e0 n a8 N7 j3 _ SECNAME2
5 H/ _4 ^/ l- j& L: r {
# ^/ E& ^ k- N2 k4 X8 U OUTPUT-SECTION-COMMAND
# v9 m0 x2 o8 S/ a" Y' F OUTPUT-SECTION-COMMAND
; y3 F n7 m1 { ...
6 v! |7 l' e; b, Z7 ~+ [; U } [:PHDR...] [=FILL]
- ~# u% Z$ C* }7 q2 Z ...3 g# m) G$ u' V+ Z! w' E5 g) u
} [>REGION] [:PHDR...] [=FILL]7 R) T; {: V; H6 Q* i; r- b
1 F( J; ?: ? [除了‘OVERLAY’关键字,所有的都是可选的,每一个节必须有一个名字(上面的SECNAME1和SECNAME2)。在 e7 o% o* }+ ^- H; c4 Z
‘OVERLAY’结构中的节定义跟通常的‘SECTIONS’结构中的节定义是完全相同的,除了一点,就是在‘OVERLAY’2 v7 t1 d: c/ U
中没有地址跟内存区域的定义。1 }. j+ I5 f) \# \2 O
1 f: E. `: H* }- S$ F6 b p节都被定义为同一个开始地址。所有节的载入地址都被排布,使它们在内存中从整个'OVERLAY'的载入地址开
$ z$ s, k+ V1 N1 |0 B; ?) j始都是连续的(就像普通的节定义,载入地址是可选的,缺省的就是开始地址;开始地址也是可选的,缺省的+ o5 M( Y( ~/ V, b* L" f
是当前的定位计数器的值。)' Y2 x% f3 `8 k2 b
6 N# E" ]% m: Q( K1 g- |如果使用了关键字`NOCROSSREFS', 并且在节之间存在引用,连接器就会报告一个错误。因为节都运行在同一1 e/ C) V3 c6 a/ \
个地址上,所以一个节直接引用另一个节中的内容是错误的。: C, d+ H# u: G. ]' c' E
4 V* n8 Y8 U) @3 x' a/ `$ E/ I对于'OVERLAY'中的每一个节,连接器自动定义两个符号。符号`__load_start_SECNAME'被定义为节的开始载8 G2 [# [# d" j, U- U
入地址。符号`__load_stop_SECNAME'被定义为节的最后载入地址。SECNAME中的不符合C规定的任何字符都将
: [/ c' ^0 R; V; S0 g被删除。C(或者汇编语言)代码可能使用这些符号在必要的时间搬移覆盖代码。3 a; X' l. X Q, z. O [' q; S2 v
6 M, m+ V9 \" a6 K* j7 t
在覆盖区域的最后,定位计数器的值被设为覆盖区域的开始地址加上最大的节的长度。
/ x" b6 q1 {" R3 F- C1 D
* s" H! t. @, H这里是一个例子。记住这只会出现在‘SECTIONS’结构的内部。: o2 r( g+ x/ Y; N$ }* N
, M/ _( K$ m5 E. {" r! d4 |# A/ x OVERLAY 0x1000 : AT (0x4000)
$ ]' \$ b. s- D% X. Z5 l3 J {+ P7 U2 Y5 X* t% s0 S+ X
.text0 { o1/*.o(.text) }3 I0 r/ W( }2 m- W' F- h* a* e
.text1 { o2/*.o(.text) }3 l7 y2 J5 h% q; {9 ?4 w2 K1 O
}
& h; v$ @5 x+ g$ Q9 {1 |+ E4 y8 I" g
这段代码会定义'.text0'和'.text1',它们都从地址0x1000开始。‘.text0'会被载入到地址0x4000处,而- o! `) ]1 n* l: S5 }$ j
'.text1'会被载入到紧随'.text0'后的位置。下面的几个符号会被定义:`__load_start_text0', # ?! q& [- O9 K; f
`__load_stop_text0', `__load_start_text1', `__load_stop_text1'.
2 n5 \. N+ H( T( M3 f% s# u; p8 l* ~0 M5 O) K
拷贝'.text1'到覆盖区域的C代码看上去可能会像下面这样:' `. N$ v ]+ t' c
* E- C! k$ b7 u' e {& X6 U& k extern char __load_start_text1, __load_stop_text1;
* }" ~6 K% l; }. V memcpy ((char *) 0x1000, &__load_start_text1,
" Z9 ~. k# K/ Q" s* W &__load_stop_text1 - &__load_start_text1);9 ]5 g/ s; }9 F# \+ D
+ w. c, f' R( a/ a' Z
注意'OVERLAY'命令只是为了语法上的便利,因为它所做的所有事情都可以用更加基本的命令加以代替。上面) g, h4 V4 } D) i$ _% j% Y
的例子可以用下面的完全特效的写法:1 U7 U: F- G) Z: V" q
: R2 ^1 w1 i ?& b, r7 P" _
.text0 0x1000 : AT (0x4000) { o1/*.o(.text) }
( g( V+ _/ C! b __load_start_text0 = LOADADDR (.text0);
6 h Y1 k& f( ~8 Z, l __load_stop_text0 = LOADADDR (.text0) + SIZEOF (.text0);: {9 D3 l9 K& }( \/ f. t) L; d
.text1 0x1000 : AT (0x4000 + SIZEOF (.text0)) { o2/*.o(.text) }# @$ Q+ g+ n1 M. y) c! m
__load_start_text1 = LOADADDR (.text1);
# r- u7 w/ X2 m5 s% W __load_stop_text1 = LOADADDR (.text1) + SIZEOF (.text1);
5 ?4 K X1 n6 y . = 0x1000 + MAX (SIZEOF (.text0), SIZEOF (.text1));" U5 {' R1 C/ e1 A$ Q5 p
; e, |% w3 L1 v) V2 kMEMORY命令6 ?, b6 d& b4 |7 P
==============
4 O' m+ `3 t6 P* ^! j: [9 _. o% u1 w( a
连接器在缺省状态下被配置为允许分配所有可用的内存块。你可以使用‘MEMORY’命令重新配置这个设置。
7 i; q5 t* L$ X: \( {. M! u; j. G b, B: j+ }& k
‘MEMORY’命令描述目标平台上内存块的位置与长度。你可以用它来描述哪些内存区域可以被连接器使用,& ^2 m# g$ ]! q. \/ B/ c
哪些内存区域是要避免使用的。然后你就可以把节分配到特定的内存区域中。连接器会基于内存区域设置节+ c. {3 V2 F6 B* \
的地址,对于太满的区域,会提示警告信息。连接器不会为了适应可用的区域而搅乱节。
1 u# f3 }' M/ n- |3 J
% Y! v! J& z* [$ O一个连接脚本最多可以包含一次'MEMORY'命令。但是,你可以在命令中随心所欲定义任意多的内存块,语法
2 N* a2 w% x# |3 N6 N如下:$ r1 r4 \ U2 N; n
, X, m* o) a- j; @; R3 q) H MEMORY
6 D' W' k0 p9 c% H* i {7 X: c+ j, }, B
NAME [(ATTR)] : ORIGIN = ORIGIN, LENGTH = LEN
- H. g6 ], _# {: ~ ...
+ g- R1 k1 {* B _( L1 B! n }
8 @5 _; D( @7 Q0 N7 Q
9 L" Q! N4 O$ D6 w$ I- E' SNAME是用在连接脚本中引用内存区域的名字。出了连接脚本,区域名就没有任何实际意义。区域名存储在一个
6 ?, S1 N: ?8 T6 S+ z3 ]单独的名字空间中,它不会和符号名,文件名,节名产生冲突,每一块内存区域必须有一个唯一的名字。' ?8 p" G* f; k* K
6 D1 d/ Y8 d4 }" ^; H0 ^
ATTR字符串是一个可选的属性列表,它指出是否为一个没有在连接脚本中进行显式映射地输入段使用一个特定
- i6 J8 I# B q ]5 M( Z2 G的内存区域。如果你没有为某些输入段指定一个输出段,连接器会创建一个跟输入段同名的输出段。如果你定
3 V. V; X0 A" v. |! H$ Q# j& I义了区域属性,连接器会使用它们来为它创建的输出段选择内存区域。
/ F2 ]- q( C# [5 J3 ~
0 y/ ~$ S* @$ wATTR字符串必须包含下面字符中的一个,且必须只包含一个:
& z9 T0 d( J( y5 g: }`R'
5 _5 {$ ^; b! C" t. t7 G只读节。3 `# u9 Q% V9 G7 K2 v
`W'( p% y+ a- u) ~0 [
可读写节。
" s8 ~! r" t% j$ e; R# H! j`X'
" R4 H9 X/ r/ ?3 a可执行节。
7 d. I3 k" c! r- _' T5 ?1 _" D`A'# J" a- \- `% J6 J! D
可分配节。
( }' E/ G: U$ |" E`I'
! P0 o- P& |+ N$ z, V/ ^+ [已初始化节。
( X, G4 a/ d' |, H( J# m`L'! @' ^6 z& @3 |, d% o6 Z
同‘I’
3 n: _3 C& Z+ E3 f' E. R`!'" E& q H. N; Z( L) J( f
对前一个属性值取反。
/ |0 U- Y; j G* i- H& d+ X% Q
4 N3 A0 s# [* p) z+ z4 @. K6 ~$ ^6 y如果一个未映射节匹配了上面除'!'之外的一个属性,它就会被放入该内存区域。'!'属性对该测试取反,所以
# i: _8 A2 o, }) X只有当它不匹配上面列出的行何属性时,一个未映射节才会被放入到内存区域。
- j' i3 Y( Q1 s+ Y" b F. n7 j$ u9 f3 L4 b
ORIGIN是一个关于内存区域地始地址的表达式。在内存分配执行之前,这个表达式必须被求值产生一个常数,
) f. X9 W# W/ |- h! [7 m$ Y这意味着你不可以使用任何节相关的符号。关键字'ORIGIN'可以被缩写为'org'或'o'(但是,不可以写为,比
W6 q9 l! v3 H/ |& W1 U7 |如‘ORG’)" c, |, Z$ J/ l
: W( }0 J8 U- ~/ B4 K8 K1 X
LEN是一个关于内存区域长充(以字节为单位)的表达式。就像ORIGIN表达式,这个表达式在分配执行前也
. w! Q! B n1 p5 S* b4 o/ p( v$ d必须被求得为一个常数值。关键字'LENGTH'可以被简写为‘len'或'l'。" Y9 W7 \; V* F
5 ^5 P; ^9 J. r7 E在下面的例子中,我们指定两个可用于分配的内存区域:一个从0开始,有256kb长度,另一个从0x4000000
0 M+ ?$ i# n$ O0 z- W6 z! p: W开始,有4mb长度。连接器会把那些没有进行显式映射且是只读或可执行的节放到'rom'内存区域。并会把另- N4 ]/ ?# k6 F: X$ d* S
外的没有被显式映射地节放入到'ram'内存区域。/ Q; `: h1 G( G) b" D* y
" N0 @$ H1 D& y U8 `' H( f
MEMORY
6 O5 d; Y: M( c! l {) m& ^' K5 m" u5 @
rom (rx) : ORIGIN = 0, LENGTH = 256K' a4 F" k# T& v" L9 {
ram (!rx) : org = 0x40000000, l = 4M$ p. p# r" `; a
}7 j# y% m5 y0 K: q ?% Y
0 h1 p3 i( I: M0 t( k一旦你定义了一个内存区域,你也可以指示连接器把指定的输出段放入到这个内存区域中,这可以通过使用
, j6 E0 z9 W8 o9 S) _$ c( X'>REGION'输出段属性。比如,如果你有一个名为'mem'的内存区域,你可以在输出段定义中使用'>mem'。如
- O( d! [. b* M" o0 Y果没有为输出段指定地址,连接器就会把地址设置为内存区域中的下一个可用的地址。如果总共的映射到一& [+ f: u% R- n: l# x
个内存区域的输出段对于区域来说太大了,连接器会提示一条错误信息。
& n; {9 {+ E2 ?
8 f/ m4 G# L) x$ |% }* }PHDRS命令
# M, Z1 G# A" `=============
" `& Z& @( f$ d$ v6 F' ^6 G6 D
5 J! B4 J) @1 a! G; ]3 A; EELF目标文件格式使用“程序头”,它也就是人们熟知的“节”。程序头描述了程序应当如何被载入到内存中。
5 h; F" ~- W0 y( E1 o/ h你可以通过使用带有'-p'选项的‘objdump’命令来打印出这个程序头。, k% W# ?3 T- g5 ~
2 i9 F) Q- r; Z5 _; f
当你在一个纯ELF系统上运行ELF程序时,系统的载入程序通过读取文件头来计算得到如何来 |
|