共有ライブラリの再配置を調査していて、奇妙なことに遭遇しました。次のコードを検討してください。
int myglob;
int ml_util_func(int p)
{
return p + 2;
}
int ml_func2(int a, int b)
{
int c = ml_util_func(a);
return c + b + myglob;
}
で非PIC共有ライブラリにコンパイルしgcc -shared
ます。x86で実行されている32ビットのUbuntuでこれを行います。
結果には、 in.so
への呼び出しの再配置エントリがあります。onの出力は次のとおりです。ml_util_func
ml_func2
objdump -dR -Mintel
ml_func2
0000050d <ml_func2>:
50d: 55 push ebp
50e: 89 e5 mov ebp,esp
510: 83 ec 14 sub esp,0x14
513: 8b 45 08 mov eax,DWORD PTR [ebp+0x8]
516: 89 04 24 mov DWORD PTR [esp],eax
519: e8 fc ff ff ff call 51a <ml_func2+0xd>
51a: R_386_PC32 ml_util_func
51e: 89 45 fc mov DWORD PTR [ebp-0x4],eax
521: 8b 45 0c mov eax,DWORD PTR [ebp+0xc]
524: 8b 55 fc mov edx,DWORD PTR [ebp-0x4]
527: 01 c2 add edx,eax
529: a1 00 00 00 00 mov eax,ds:0x0
52a: R_386_32 myglob
52e: 8d 04 02 lea eax,[edx+eax*1]
531: c9 leave
532: c3 ret
533: 90 nop
命令のR_386_PC32
再配置に注意してください。call
さて、私の質問は、なぜこの移転が必要なのですか? e8
x86では「相対呼び出し...」でありml_util_func
、同じオブジェクトで定義されているため、リンカーは動的ローダーに任せることなく、それと呼び出しの間の相対オフセットを計算できますか?
興味深いことに、ml_util_func
が宣言されている場合static
、再配置はなくなり、リンカーはオフセットを正しく計算して挿入します。ml_util_func
リンカーがそれについて怠惰になるのは、エクスポートされていることについて何ですか?
PS: ロード時の再配置を理解するために、意図的に非 PIC コードで遊んでいます。