版主
主题
回帖0
积分10609
阅读权限200
注册时间2008-11-22
最后登录1970-1-1
在线时间 小时
|
楼主 |
发表于 2016-12-6 16:30
|
显示全部楼层
Linux链接脚本学习--lds
6 z, p! o5 H2 Q$ H) Q
; ] L3 }9 }7 j) A$ N- ]( v, T, Q& o+ G+ d/ `1 J8 {
一、概论
$ b5 D3 d! j p. c/ m/ k7 W0 ~! T, V, O4 y
ld:0 ]9 @4 F) u1 W& x7 ^
2 B( y6 ~! C! k3 P! @; F, P
GNU的链接器.
8 J1 @2 ?9 I3 O6 ~ W/ l/ {% \0 @) y2 S, D
用来把一定量的目标文件跟档案文件链接在一起,并重新定位它们的数据,链接符号引用.
/ Q( u3 |& E1 f+ |4 [8 ?# p* Q' n5 R1 @6 C+ l+ l
一般编译一个程序时,最后一步就是运行ld进行链接
G5 ]3 E: Q O5 A; Z
6 j# Z& q" S0 i1 P每一个链接都被一个链接脚本所控制,这个脚本是用链接命令语言书写的.
3 ?! H9 G+ K9 l8 K
$ r0 _5 c/ Z, c8 C二、链接脚本8 V1 n* } S% {" H. A+ A
4 n- v# B) s7 [
链接脚本的一个主要目的是描述输入文件中的各个段(数据段,代码段,堆,栈,bss)如何被映射到输出文件中,并控制输出文件的内存排布.7 o4 y0 t- O9 K7 y/ Z
$ ?2 l6 A0 X1 x: n链接器总是使用链接脚本的,如果你不提供,则链接器会使用一个缺省的脚本,这个脚本是被编译进链接器可执行文件的.
2 C; y7 j1 S& P* {6 S0 [4 t8 ]4 X& f6 m# ^( [6 B
可以使用--verbose命令行显示缺省的链接器脚本的内容., {7 O7 P# }! B% d& x6 Z5 f
/ H2 Q8 G7 b8 ?; n* h' H) m你可以使用-T命令行来提供你自己的链接脚本来替换缺省的链接脚本.
: X5 z6 q N3 g4 F* [$ m; J5 H8 m/ T3 T E' l- B
三、简单的链接脚本示例. p/ P L5 I7 p) ], J P7 T
0 d4 w& i! G2 t8 H许多脚本是相当简单的.
6 |6 R+ Z1 H$ u3 E+ R! ^& f0 M p1 I! q& F0 L& |$ D t' ^
可能最简单的脚本只含有一个命令:’SECTIONS’.
4 @4 \% q) S( Y/ H
4 ^9 z' s2 j3 y( ?% K _你可以使用’SECTIONS’来描述输出文件的内存布局.
1 X" _; x* b s# O
$ l. r5 s; y9 W+ a‘SECTIONS’是一个功能很强大的命令.& H x, @/ z' J3 e' z/ o8 D* {% o/ H
% H/ Y" i( A9 P
假设你的程序只有代码段,初始化过的数据段,和未初始化过的数据段.这些会存在于’.text’,’data’,’bss’段中./ z5 x# N0 f; I& B! m& e0 J
# T6 G& O% b0 a7 U2 Y( x对于这个例子,假设代码应该被载入到地址0x1000处,而数据应该从0x8000000开始,如下是实现这个功能的脚本:" ~! ]- L/ \' c6 A
" B# K* w) P2 R7 K, O0 p- \3 k# N
SECTIONS
3 \9 D. o5 z A! L: Z o4 K8 C% Z6 v' L, @) ?9 w/ E6 w2 u7 b c
{; o6 {4 Z' k, F W5 Y
+ j" w& X, h0 ?7 e
.=0x1000;+ G8 K. `! R% w$ D5 ?& ^
% n/ F2 Z* Q, q M* @8 C.text:{*(.text)}
% ~3 |; _2 w7 b
1 c9 W6 l2 G4 w6 o s( s.=0x8000000;/ }' F* N8 q$ L" v0 b4 @- k6 \
4 R+ _( P: i2 s X6 i# `- ]" }.data:{*(.data)}5 S/ y) w( i1 |# T' k8 m
: P, W- t! l. |
.bss:{*(.bss)}
& Q, n4 Z" b1 x
+ p) |6 G/ _7 w6 h' p. O}) R5 H% Q7 v3 |7 C
, p: J/ L0 K1 i3 q$ Z
具体分析: |) n- M8 y2 @' X, j
/ x% p& ^4 H2 `3 e* m2 {6 B关键字’SECTIONS’开始于这个配置.后面跟有一串放在花括号中的符号赋值和输出端描述的内容.
9 v: f/ |' h3 Z: m0 y
$ @' }0 F8 `5 Q$ \8 ~) W& s7 L第一行是对一个特殊的符号’.’赋值,这是一个定位标识器.如果你没有以其他的方式制定输出段的地址,那地址值就会被设为定位标识器的现有值,即0x1000. I+ t) r- T, b2 l
& v& ^% f' o3 d+ C f
第二行定义一个输出段,’.text’.冒号’:’是语法需要,现在可以被忽略.段后面的花括号中,应该列出所有应该放入这个输出段中的输入端的名字.’*’是通配符,匹配所有文件名.即将所有输入文件中的.text段都保存在此段中.
9 c+ a+ c6 j- M% u" S
) X9 @+ E+ h" d o7 U" C2 S9 W% i余下的是.data和.bss段,同理,链接器会把所有.data段从地址0x8000000开始处放置.
5 |7 }4 h3 g& T5 r4 q! N4 q# o- I3 p' j# C/ j
最后,定位标识器的值变为0x8000000加上所有.data段的地址.此时链接器把所有.bss放在此处开始的地址.# F. V/ Q+ e8 D/ @* Y5 a) ]% K
" L" ]' X3 [0 h2 c ]+ }# @
. l" @: L' `: p9 u6 |+ w3 {
/ e& Q& Q( _" O, g. V, [3 J四、简单的链接脚本命令
6 k0 M$ k9 h) G! D- L8 m7 ` \2 ^; C. R9 j, ^* p
设置入口点
* s% }; h- W+ C8 P0 E
9 M8 j+ B$ M) w, m在运行一个程序时,第一个被执行到的指令成为”入口点”.你可以使用”ENTRY”链接脚本命令来设置入口点.参数是一个符号名,如下:6 T! W! R" _% @# L1 ]. n+ I
+ a* _$ W G* O8 p3 ]/ v
ENTRY(SYMBOL)
8 `: V: g* g8 T) ~ T, E
" q8 [# S, ]7 u2 D2 m" b- l$ L: P9 r有很多不同的方法来设置入口点.链接器会通过按顺序尝试一下方法来设置入口点,如果成功了,就会停止.( v ^0 \& L9 M5 V; K% F
3 h, {& P5 {( W: ] ?) t1,’-e’ 入口命令行选项
2 k% S2 O S& V2 M A; C# E; p; w4 W$ f, k9 S
2,链接脚本中的ENTRY(SYMBOL)命令
$ X w& ?& Q, E! d8 c# l# _0 y2 d& Q7 z9 E: r: r) v
3,如果定义了start,就使用start的值' r" @* l3 m+ c% y3 Z1 A7 p# b
9 |/ l. N) h/ }! B* k2 ]+ F$ n( k% V4,如果存在就使用’.text’段的首地址9 X% {4 S! A$ h- E& Q# S: P) R0 W
3 p" C& x( @. c/ [0 [
5,地址’0’
; u, i! g4 w8 V. r6 N' z# [! ]$ j8 {* A. l" d7 I, J9 c5 [
1 |0 E4 W& X F4 F8 Z$ L7 `7 ]9 l5 a
4 E! o4 ], `, V; h) L
" S0 o! a0 o# Z+ P0 ?5 n a$ S" L: {# m) a
. o5 k6 W, C7 b9 o6 {2 Y
0 ]; Q7 }1 V" F. V$ h' _五、命令行设置链接地址. Y) H0 a( z- h- Z5 T3 E
* {/ V1 }+ A1 e n3 Sld用于将多个obj或者so(库)文件链接成可执行文件.
3 _( }: o8 Q0 R! q) O' {+ q( A; p! S: q# O1 M ~; m- K; n
使用-T选项可以指定数据段,代码段,bss段起始位置.(-T只用于链接bootloader、内核等没有底层软件支持的软件.链接运行于操作系统之上的应用程序时,一般使用默认方式链接)., x3 z7 @/ M# B5 g6 n) K0 _1 w
! j/ n" E- o; l0 F1 n. X1,直接指定代码段、数据段、bss段起始地址, I4 A- e2 v! }8 I- b
! d) }, |6 S' I. ?
如下:- u# \/ B" Y' w% Y0 m j
# F" S* z5 k# ~0 A$ T# Q! C-Ttext startaddr% s' S- K5 C; h' @
- S, y$ r' f4 e/ T2 F' x9 B3 f
-Tdata startaddr( z+ a& V" k4 L; X7 A( s% [
6 y! I# [) a, o( c9 x s-Tbss startaddr
# c9 e7 {& ~0 n0 ]3 p" y2 `" Y, T, T; d1 c6 i R
例如:$ c; Z" U) |+ a! M' v4 p
/ w0 M/ c4 P6 ?3 [- D4 J
ld –Ttext 0x00000000 –g led_on.o –o led_on_elf
/ r: t! ~# q8 G& a& D, I: ~
, V: `2 ]) `* Y& u, L8 w% G- i2,直接使用链接脚本来设置起始地址: Q0 c% ~5 {+ a4 U
& |0 y8 z3 d1 k; [, e
ld –Ttimer.lds –o timer_elf a.o b.o
; ]8 R; C9 _7 i0 P/ F4 V9 K1 }4 K7 l0 D5 I1 B" q+ C
链接脚本timer.lds内容如下:/ ]/ s! W* R9 M6 p& V9 [ c7 a$ }
4 g5 w( m4 p. Y% m$ P/ g1 r% ^
SECTIONS{
$ l* Y4 T6 a* N9 h7 l8 u. ?* H/ @+ ]$ W& ]% g' L1 I6 O0 t, J9 e
.=0x30000000;
, x3 t( Y3 f+ W- \! f0 }' f6 ]! { p) o; K- a2 L) F7 ^, P# g9 H8 r. ?
.text : {*(.text)}
% Z) \4 E3 ~" c, M7 x, j* Q1 y( L) I/ E) X
.rodata ALIGN(4) : {*(.rodata)}
6 s) v) j n1 Q: u, X& ?( }
( S- h% R( y2 e8 h.data ALIGN(4) : {*(.data)}
6 I4 C% }/ N* l' ?' V' z |
. a1 B& L& y ?/ f' p.bss ALIGN(4) : {*.(.bss) *(COMMON)}
, x8 a- f6 U2 Z! M% l' R. p' Z. Q' q4 u
}
, o5 H U; X' k6 z J! ^9 K! N+ ]: i, Y
一个SECTIONS命令内部包含一个或多个段,段(section)是连接脚本的基本单元,它表示输入文件中的某部分怎么放置.
; X9 U4 C8 k5 X( X! m+ F |
|