博客專欄

EEPW首頁(yè) > 博客 > 動(dòng)態(tài)庫(kù)的鏈接和鏈接選項(xiàng)-L,-rpath-link,-rpath

動(dòng)態(tài)庫(kù)的鏈接和鏈接選項(xiàng)-L,-rpath-link,-rpath

發(fā)布人:電子禪石 時(shí)間:2023-01-13 來(lái)源:工程師 發(fā)布文章
鏈接動(dòng)態(tài)庫(kù)

如何程序在連接時(shí)使用了共享庫(kù),就必須在運(yùn)行的時(shí)候能夠找到共享庫(kù)的位置。linux的可執(zhí)行程序在執(zhí)行的時(shí)候默認(rèn)是先搜索/lib和/usr/lib這兩個(gè)目錄,然后按照/etc/ld.so.conf里面的配置搜索絕對(duì)路徑。同時(shí),Linux也提供了環(huán)境變量LD_LIBRARY_PATH供用戶選擇使用,用戶可以通過(guò)設(shè)定它來(lái)查找除默認(rèn)路徑之外的其他路徑,如查找/work/lib路徑,你可以在/etc/rc.d/rc.local或其他系統(tǒng)啟動(dòng)后即可執(zhí)行到的腳本添加如下語(yǔ)句:LD_LIBRARY_PATH =/work/lib:$(LD_LIBRARY_PATH)。并且LD_LIBRARY_PATH路徑優(yōu)先于系統(tǒng)默認(rèn)路徑之前查找(詳細(xì)參考《使用LD_LIBRARY_PATH》)。

不過(guò)LD_LIBRARY_PATH的設(shè)定作用是全局的,過(guò)多的使用可能會(huì)影響到其他應(yīng)用程序的運(yùn)行,所以多用在調(diào)試。(LD_LIBRARY_PATH的缺陷和使用準(zhǔn)則,可以參考《Why LD_LIBRARY_PATH is bad》 )。通常情況下推薦還是使用gcc的-R或-rpath選項(xiàng)來(lái)在編譯時(shí)就指定庫(kù)的查找路徑,并且該庫(kù)的路徑信息保存在可執(zhí)行文件中,運(yùn)行時(shí)它會(huì)直接到該路徑查找?guī)欤苊饬耸褂肔D_LIBRARY_PATH環(huán)境變量查找。

鏈接選項(xiàng)和路徑

現(xiàn)代連接器在處理動(dòng)態(tài)庫(kù)時(shí)將鏈接時(shí)路徑(Link-time path)和運(yùn)行時(shí)路徑(Run-time path)分開(kāi),用戶可以通過(guò)-L指定連接時(shí)庫(kù)的路徑,通過(guò)-R(或-rpath)指定程序運(yùn)行時(shí)庫(kù)的路徑,大大提高了庫(kù)應(yīng)用的靈活性。比如我們做嵌入式移植時(shí)#arm-linux-gcc $(CFLAGS) –o target –L/work/lib/zlib/ -llibz-1.2.3 (work/lib/zlib下是交叉編譯好的zlib庫(kù)),將target編譯好后我們只要把zlib庫(kù)拷貝到開(kāi)發(fā)板的系統(tǒng)默認(rèn)路徑下即可。或者通過(guò)-rpath(或-R )、LD_LIBRARY_PATH指定查找路徑。

鏈接器ld的選項(xiàng)有 -L,-rpath 和 -rpath-link,看了下 man ld,大致是這個(gè)意思:

-L: “鏈接”的時(shí)候,去找的目錄,也就是所有的 -lFOO 選項(xiàng)里的庫(kù),都會(huì)先從 -L 指定的目錄去找,然后是默認(rèn)的地方。編譯時(shí)的-L選項(xiàng)并不影響環(huán)境變量LD_LIBRARY_PATH,-L只是指定了程序編譯連接時(shí)庫(kù)的路徑,并不影響程序執(zhí)行時(shí)庫(kù)的路徑,系統(tǒng)還是會(huì)到默認(rèn)路徑下查找該程序所需要的庫(kù),如果找不到,還是會(huì)報(bào)錯(cuò),類似cannot open shared object file。

-rpath-link:這個(gè)也是用于“鏈接”的時(shí)候的,例如你顯示指定的需要 FOO.so,但是 FOO.so 本身是需要 BAR.so 的,后者你并沒(méi)有指定,而是 FOO.so 引用到它,這個(gè)時(shí)候,會(huì)先從 -rpath-link 給的路徑里找。

-rpath: “運(yùn)行”的時(shí)候,去找的目錄。運(yùn)行的時(shí)候,要找 .so 文件,會(huì)從這個(gè)選項(xiàng)里指定的地方去找。對(duì)于交叉編譯,交叉編譯鏈接器需已經(jīng)配置 –with-sysroot 選項(xiàng)才能起作用。也就是說(shuō),-rpath指定的路徑會(huì)被記錄在生成的可執(zhí)行程序中,用于運(yùn)行時(shí)查找需要加載的動(dòng)態(tài)庫(kù)。-rpath-link 則只用于鏈接時(shí)查找。


鏈接搜索順序

直接man ld。The linker uses the following search paths to locate required shared libraries:




gcc和鏈接選項(xiàng)的使用

在gcc中使用ld鏈接選項(xiàng)時(shí),需要在選項(xiàng)前面加上前綴-Wl(是字母l,不是1),以區(qū)別不是編譯器的選項(xiàng)。

if the linker is being invoked indirectly, via a compiler driver (e.g. gcc) then all the linker command line options should be prefixed by -Wl, (or whatever is appropriate for the particular compiler driver) like this:

gcc  main.c  -lhello -L . -Wl.-rpath=./lib

This is important, because otherwise the compiler driver program may silently drop the linker options, resulting in a bad link.



*博客內(nèi)容為網(wǎng)友個(gè)人發(fā)布,僅代表博主個(gè)人觀點(diǎn),如有侵權(quán)請(qǐng)聯(lián)系工作人員刪除。



關(guān)鍵詞: gcc

相關(guān)推薦

技術(shù)專區(qū)

關(guān)閉
×

报名截止时间 12月22日
Let's do--DIY 液体流量检测仪让喝水更有乐趣,快来报名吧!