在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

如何鏈接兩個(gè)名字一樣動(dòng)態(tài)庫(kù)

Linux閱碼場(chǎng) ? 來(lái)源:IOT物聯(lián)網(wǎng)小鎮(zhèn) ? 作者:道哥 ? 2021-10-08 14:58 ? 次閱讀

Linux應(yīng)用的開發(fā)過(guò)程中,直接利用現(xiàn)成的第三方庫(kù)(俗稱:輪子)來(lái)完成自己的業(yè)務(wù)功能,是很常見(jiàn)的事情。

不知道你是否遇到這樣的場(chǎng)景:應(yīng)用程序中需要使用兩個(gè)動(dòng)態(tài)庫(kù)里的不同功能的函數(shù),但是這兩個(gè)動(dòng)態(tài)庫(kù)的作者發(fā)生心靈感應(yīng)了,居然起了完全一樣的動(dòng)態(tài)庫(kù)名字,這該如何是好?

具體來(lái)說(shuō)面對(duì)的問(wèn)題是:在編譯可執(zhí)行程序的時(shí)候,通過(guò)gcc編譯參數(shù)的-lXXX就可以動(dòng)態(tài)鏈接一個(gè)動(dòng)態(tài)庫(kù)。

但是,現(xiàn)在你想鏈接兩個(gè)動(dòng)態(tài)庫(kù),它們的名字是一樣的!!怎么辦?

第一個(gè)動(dòng)態(tài)庫(kù)文件現(xiàn)在,假設(shè)我們?cè)陂_發(fā)一個(gè)機(jī)器人應(yīng)用程序,需要用到一個(gè)第三方動(dòng)態(tài)庫(kù)中的算法

這個(gè)庫(kù)的源碼很簡(jiǎn)單,如下:

// 第一個(gè)動(dòng)態(tài)庫(kù) 源文件 RobotMath.c:

double func0(double arg)

{

double ret = arg + arg;

return ret;

}

double func1(double arg1, double arg2)

{

double ret = arg1 + arg2;

return ret;

}

動(dòng)態(tài)庫(kù)的編譯命令是:

$ gcc -m32 -fPIC --shared -o libRobotMath.so -Wl,--soname,libRobotMath.so RobotMath.c

以上這些屬性都比較常見(jiàn),請(qǐng)注意其中的 -Wl,--soname,libRobotMath.so,它用來(lái)指定生成的動(dòng)態(tài)庫(kù)的 SONAME,一般用于動(dòng)態(tài)庫(kù)的版本管理中。

為了方便起見(jiàn),這里就不加版本信息了。

執(zhí)行了 gcc 指令之后,就得到了一個(gè)動(dòng)態(tài)庫(kù)文件:libRobotMath.so。

可以通過(guò) patchelf 這個(gè)工具(在Ubuntu系統(tǒng)中,可以通過(guò)apt-get直接安裝),來(lái)查看一下這個(gè)動(dòng)態(tài)庫(kù)文件的 SONAME :

$ patchelf --print-soname libRobotMath.so

libRobotMath.so // SONAME

第2行打印出來(lái)的就是所謂的 SONAME。

你也可以測(cè)試一下,指定其他的 SONAME,例如:

$ gcc -m32 -fPIC --shared -o libRobotMath.so -Wl,--soname,libRobotMath-1.2.3.so RobotMath.c

$ patchelf --print-soname libRobotMath.so

libRobotMath-1.2.3.so // SONAME

以上就是第一個(gè)動(dòng)態(tài)庫(kù),已經(jīng)交代清楚了,下面再來(lái)看一下最簡(jiǎn)單的應(yīng)用程序。

應(yīng)用程序// 可執(zhí)行程序 源文件: main.c

extern double func0(double arg);

extern double func1(double arg1, double arg2);

int main(int argc, char *agv[])

{

double arg = 1.1;

double result0 = func0(arg);

printf(“result0 = %lf

”, result0);

double arg1 = 1.1, arg2 = 2.2;

double result1 = func1(arg1, arg2);

printf(“result1 = %lf

”, result1);

return 0;

}

這個(gè)代碼簡(jiǎn)直是幼兒園水平,不多解釋,直接編譯(假設(shè)已經(jīng)把動(dòng)態(tài)庫(kù)復(fù)制到main.c同一個(gè)文件夾中了):

$ gcc -m32 -o main main.c -lRobotMath -L./ -Wl,-rpath=。/

執(zhí)行:

$ 。/main

result0 = 2.200000

result1 = 3.300000

完美!

第二個(gè)動(dòng)態(tài)庫(kù)文件問(wèn)題來(lái)了:現(xiàn)在應(yīng)用程序還需要實(shí)現(xiàn)另外一個(gè)復(fù)雜的算法,本著偷懶的精神,終于在另外一個(gè)機(jī)器人算法相關(guān)的庫(kù)中找到了這個(gè)算法。

// 第二個(gè)動(dòng)態(tài)庫(kù) 源文件 RobotMath.c:

double func2(double arg1, double arg2, double arg3)

{

double ret = arg1 * arg2 * arg3;

return ret;

}

// 編譯指令

$ gcc -m32 -fPIC --shared -o libRobotMath.so -Wl,--soname,libRobotMath.so RobotMath.c

但是坑爹的是,這個(gè)算法庫(kù)輸出的動(dòng)態(tài)庫(kù)名稱居然也是 libRobotMath.so !

與第一個(gè)算法庫(kù)的文件名同名同姓,看來(lái)這個(gè)名字太招人喜歡了。

如果這個(gè)作者直接起一個(gè)其它的名字,那就啥事都沒(méi)有了。

假如: 名字叫 libRobotUltra.so,那么只需要直接復(fù)制過(guò)來(lái),然后在編譯執(zhí)行程序時(shí),直接鏈接 -lRobotUltra 就可以了。

錯(cuò)誤做法:直接給它改名既然如此,我們是否可以直接給它改名呢?嘗試一下:

$ mv libRobotMath.so libRobotMath2.so

然后把libRobotMath2.so復(fù)制到應(yīng)用程序的目錄下,并在main.c中,調(diào)用這個(gè)庫(kù)中的算法函數(shù) func2。

extern double func2(double arg1, double arg2, double arg3);

int main(int argc, char *agv[])

{

// 之前的其它代碼

// 。..

double arg3 = 1.1, arg4 = 2.2, arg5 = 3.3;

double result2 = func2(arg3, arg4, arg5);

printf(“result2 = %lf

”, result2);

return 0;

}

編譯一下試試:

$ gcc -m32 -o main main.c -lRobotMath -lRobotMath2 -L./ -Wl,-rpath=。/

/tmp/ccDGqFkl.o: In function `main‘:

main.c undefined reference to `func2’

collect2: error: ld returned 1 exit status

報(bào)錯(cuò):找不到 func2 這個(gè)函數(shù)。

但是libRobotMath2.so這個(gè)庫(kù)中明明已經(jīng)有這個(gè)函數(shù)啊,不信你看:

$ readelf -s libRobotMath2.so | grep func2

8: 0000052a 69 FUNC GLOBAL DEFAULT 11 func2

51: 0000052a 69 FUNC GLOBAL DEFAULT 11 func2

為啥 gcc 還找不到呢?

看來(lái),很粗魯?shù)刂苯咏o第二個(gè)動(dòng)態(tài)庫(kù)文件強(qiáng)行改名,不是解決問(wèn)題的正確思路!

正解:patchelf 工具還記得在第一個(gè)庫(kù)中,我們使用 patchelf 這個(gè)小工具來(lái)查看動(dòng)態(tài)庫(kù)的 SONAME 嗎?

繼續(xù)用它來(lái)查看下被我們改名后的 libRobotMath2.so:

$ patchelf --print-soname libRobotMath2.so

libRobotMath.so

SONAME 依然是原來(lái)的名稱,說(shuō)明通過(guò)mv指令改名,只是改變了外表,并沒(méi)有改變它的內(nèi)心。

如果你熟悉文件系統(tǒng),就會(huì)知道:mv 指令只是修改了庫(kù)文件在 inode 節(jié)點(diǎn)中的名字,而庫(kù)文件實(shí)際內(nèi)容所存儲(chǔ)的 block 存儲(chǔ)空間中,一點(diǎn)都沒(méi)有變化。

動(dòng)態(tài)庫(kù)是一個(gè)ELF格式的文件,操作系統(tǒng)在加載動(dòng)態(tài)庫(kù)的時(shí)候,是根據(jù)ELF格式的標(biāo)準(zhǔn),對(duì)文件的內(nèi)容進(jìn)行一層一層解析的。

可以參考很久之前寫的一篇文章:Linux系統(tǒng)中編譯、鏈接的基石-ELF文件:扒開它的層層外衣,從字節(jié)碼的粒度來(lái)探索。

patchelf 這個(gè)工具,就提供了這樣的功能:查看或修改動(dòng)態(tài)庫(kù)文件的內(nèi)部信息,包括:SONAME, 依賴的其他動(dòng)態(tài)庫(kù),rpath 路徑信息等等。

$ patchelf -h

syntax: patchelf

[--set-interpreter FILENAME]

[--page-size SIZE]

[--print-interpreter]

[--print-soname]Prints ‘DT_SONAME’ entry of .dynamic section. Raises an error if DT_SONAME doesn‘t exist

[--set-soname SONAME]Sets ’DT_SONAME‘ entry to SONAME.

[--set-rpath RPATH]

[--remove-rpath]

[--shrink-rpath]

[--print-rpath]

[--force-rpath]

[--add-needed LIBRARY]

[--remove-needed LIBRARY]

[--replace-needed LIBRARY NEW_LIBRARY]

[--print-needed]

[--no-default-lib]

[--debug]

[--version]

FILENAME

我們可以使用--set-soname這個(gè)參數(shù),來(lái)把它的 SONAME 修改一下:

$ patchelf --set-soname libRobotMath2.so libRobotMath2.so

第一個(gè) libRobotMath2.so,是設(shè)置的 SONAME 名稱;

第二個(gè) libRobotMath2.so,是指定修改哪一個(gè)動(dòng)態(tài)庫(kù)文件的 SONAME;

修改之后,再檢查一下是否修改正確了:

$ patchelf --print-soname libRobotMath2.so

libRobotMath2.so

Bingo!SONAME 已經(jīng)被正確修改了。

再次編譯一下可執(zhí)行程序:

$ gcc -m32 -o main main.c -lRobotMath -lRobotMath2 -L./ -Wl,-rpath=。/

沒(méi)有報(bào)錯(cuò)!

執(zhí)行一下:

$ 。/main

result0 = 2.200000

result1 = 3.300000

result2 = 7.986000

問(wèn)題解決了!

One More Thing什么?你說(shuō)這樣的問(wèn)題是千年等一回?是為賦新詞強(qiáng)說(shuō)愁?那說(shuō)明走過(guò)的路還不是足夠的長(zhǎng)。

記得大概是2015年的時(shí)候,開發(fā)一個(gè)網(wǎng)關(guān),在硬件出來(lái)之前需要在Ubuntu (x86)平臺(tái)上進(jìn)行模擬

為了便于跨平臺(tái),選擇了 glib 庫(kù),但是對(duì)其中的小部分源碼進(jìn)行了二次開發(fā)。

但是Ubuntu的桌面系統(tǒng)是基于GTK的(底層使用的就是glib庫(kù)),也就是說(shuō)操作系統(tǒng)在啟動(dòng)時(shí)已經(jīng)加載了系統(tǒng)目錄下的 glib庫(kù)。

那么我們的應(yīng)用程序在編譯時(shí),的確可以鏈接到自己二次開發(fā)的glib庫(kù)(放在本地文件夾),但是在執(zhí)行時(shí),一直加載不成功,就是因?yàn)閯?dòng)態(tài)庫(kù)的名字沖突問(wèn)題導(dǎo)致的。

最后沒(méi)辦法,只好利用 patchelf 工具,對(duì)動(dòng)態(tài)庫(kù)的名稱,包括 SONAME 進(jìn)行改寫,這樣才解決問(wèn)題。

責(zé)任編輯:haq

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • Linux
    +關(guān)注

    關(guān)注

    87

    文章

    11465

    瀏覽量

    212817
  • 代碼
    +關(guān)注

    關(guān)注

    30

    文章

    4887

    瀏覽量

    70266

原文標(biāo)題:鏈接兩個(gè)"名字完全一樣"的【動(dòng)態(tài)庫(kù)】,你會(huì)怎么處理?

文章出處:【微信號(hào):LinuxDev,微信公眾號(hào):Linux閱碼場(chǎng)】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦
    熱點(diǎn)推薦

    請(qǐng)問(wèn)如何鏈接動(dòng)態(tài)庫(kù)

    是否有可參考的工程? 鏈接成功后動(dòng)態(tài)庫(kù)應(yīng)該放在哪里啊?SDK是RTOS_ONLY
    發(fā)表于 04-25 08:15

    DLP4710和DLP4710LC,在兩個(gè)套件中的DMD是不是一樣的呢?

    ,DLPDLCR4710EVM-G2全高清 DLP4710 芯片組評(píng)估模塊,我對(duì)比了兩個(gè)套件,發(fā)現(xiàn)它們用的方案基本是一樣的,除了DLP驅(qū)動(dòng)芯片,
    發(fā)表于 02-19 07:13

    深入探討Linux系統(tǒng)中的動(dòng)態(tài)鏈接庫(kù)機(jī)制

    本文將深入探討Linux系統(tǒng)中的動(dòng)態(tài)鏈接庫(kù)機(jī)制,這其中包括但不限于全局符號(hào)介入、延遲綁定以及地址無(wú)關(guān)代碼等內(nèi)容。 引言 在軟件開發(fā)過(guò)程中,動(dòng)態(tài)庫(kù)鏈接
    的頭像 發(fā)表于 12-18 10:06 ?514次閱讀
    深入探討Linux系統(tǒng)中的<b class='flag-5'>動(dòng)態(tài)</b><b class='flag-5'>鏈接庫(kù)</b>機(jī)制

    DS90UB941AS是否支持帶兩個(gè)屏? 兩個(gè)屏是要求一樣的參數(shù)才行嗎?

    DS90UB941AS是否支持帶兩個(gè)屏? 兩個(gè)屏是要求一樣的參數(shù)才行么?
    發(fā)表于 12-12 06:07

    求助,PCM3070怎么設(shè)置LOL和LOR都是一樣的輸出,為什么?

    現(xiàn)在設(shè)置的是LDAC輸出到LOL(PAGE 1 REG14 D3=1), RADC輸出到LOR(PAGE 1 REG15 D3=1). I2S把兩個(gè)聲道混音,即兩個(gè)聲道輸入一樣的聲音的時(shí)候,單個(gè)
    發(fā)表于 10-21 06:46

    正電壓轉(zhuǎn)負(fù)電壓之后兩個(gè)電壓地一樣

    當(dāng)正電壓轉(zhuǎn)換為負(fù)電壓時(shí),兩個(gè)電壓的“地”(即參考點(diǎn)或零電位點(diǎn))在理論上應(yīng)該是相同的,但在實(shí)際操作中可能會(huì)存在些差異。 首先,我們明確點(diǎn):電壓是相對(duì)的,它表示兩個(gè)點(diǎn)之間的電勢(shì)差。因此
    的頭像 發(fā)表于 10-09 10:33 ?776次閱讀

    定影熱敏電阻兩個(gè)都是一樣的嗎?

    定影熱敏電阻在兩個(gè)不同的情境或應(yīng)用中可能不完全一樣,這主要取決于具體的設(shè)備、設(shè)計(jì)需求以及熱敏電阻的類型和功能。以下是對(duì)定影熱敏電阻兩個(gè)是否一樣的分析:
    的頭像 發(fā)表于 09-06 15:39 ?875次閱讀

    同相兩個(gè)輸入放大電路的作用一樣

    同相兩個(gè)輸入放大電路的作用,在定程度上是相似的,但也有著顯著的差異,主要體現(xiàn)在它們的放大特性、輸入輸出阻抗、共模抑制能力以及應(yīng)用場(chǎng)景等方面。 、同相放大器的基本工作原理 同相放大器,也稱為“反饋
    的頭像 發(fā)表于 09-05 11:32 ?1065次閱讀

    嵌入式學(xué)習(xí)-靜態(tài)鏈接動(dòng)態(tài)鏈接

    目標(biāo)文件,般是以.so文件形式存在。GCC默認(rèn)是動(dòng)態(tài)鏈接不需要加參數(shù)。舉例子看動(dòng)態(tài)鏈接庫(kù)
    發(fā)表于 08-28 09:33

    靜態(tài)鏈接動(dòng)態(tài)鏈接

    目標(biāo)文件,般是以.so文件形式存在。GCC默認(rèn)是動(dòng)態(tài)鏈接不需要加參數(shù)。舉例子看動(dòng)態(tài)鏈接庫(kù)
    發(fā)表于 08-27 10:20

    外熱釋電傳感器兩個(gè)濾波電路U2A,U2B(紅圈部分)電阻電容參數(shù)為什么不能用一樣的?

    外熱釋電傳感器兩個(gè)濾波電路U2A,U2B(紅圈部分)電阻電容參數(shù)為什么不能用一樣
    發(fā)表于 08-20 07:32

    OPA817跟opa818一樣的封裝,為什么者的熱阻不一樣呢?

    我用了opa818,但是有個(gè)通道的Cf需要變大,需要換成opa817才能穩(wěn)定,否則震蕩。 但是我在對(duì)比兩個(gè)運(yùn)放的參數(shù)時(shí)候,有個(gè)疑問(wèn),
    發(fā)表于 07-30 06:16

    opencv-python和opencv一樣

    一樣。OpenCV(Open Source Computer Vision Library)是個(gè)開源的計(jì)算機(jī)視覺(jué)和機(jī)器學(xué)習(xí)軟件庫(kù),它提供了大量的圖像和視頻處理功能。OpenCV-P
    的頭像 發(fā)表于 07-16 10:38 ?1936次閱讀

    示波器兩個(gè)通道的接地探頭電壓是一樣的嗎?

    問(wèn)題:示波器兩個(gè)通道的接地探頭電壓是一樣的嗎?這個(gè)問(wèn)題涉及到電子測(cè)量中的基礎(chǔ)知識(shí)和實(shí)際應(yīng)用,讓我們來(lái)深入探討下。 首先,讓我們來(lái)了解下示波器的基本結(jié)構(gòu)。示波器通常有多個(gè)通道,每個(gè)通
    的頭像 發(fā)表于 06-24 10:19 ?893次閱讀
    示波器<b class='flag-5'>兩個(gè)</b>通道的接地探頭電壓是<b class='flag-5'>一樣</b>的嗎?

    esp32-s3全雙工需要兩個(gè)iis組合,這樣全雙工模式下兩個(gè)iis使用的引腳是否可以配置成一樣呢?

    api參考說(shuō)單個(gè)iis只能半雙工,全雙工需要兩個(gè)iis組合,這樣全雙工模式下兩個(gè)iis使用的引腳是否可以配置成一樣呢? 另lcd外設(shè)會(huì)占用iis么
    發(fā)表于 06-19 07:58
    主站蜘蛛池模板: 日韩无| 欧美刺激午夜性久久久久久久 | 香蕉久久高清国产精品免费 | 成人最新午夜免费视频 | 亚洲综合国产一区二区三区 | 日本一区二区三区在线网 | 黄色三级在线看 | 欧美夜夜| 亚洲欧美日韩另类精品一区二区三区 | 亚洲啊v在线 | 免费人成a大片在线观看动漫 | 亚洲444444在线观看 | 日本黄色三级视频 | 午夜精品久视频在线观看 | 狠狠狠狠狠操 | 日韩欧美一级 | 亚欧有色亚欧乱色视频 | 欧美在线视频免费播放 | 亚洲va中文字幕无码 | 偷偷操不一样的久久 | 亚洲黄色官网 | 福利片免费一区二区三区 | 奇米色88欧美一区二区 | 久久亚洲国产精品五月天 | 最新大黄网站免费 | 久久久久国产精品免费免费 | jlzzjlzz亚洲大全 | 日本精品一在线观看视频 | 精品久久久久久国产免费了 | 亚洲国产精品热久久2022 | 天天曰夜夜曰 | 国产大乳孕妇喷奶水在线观看 | 午夜三级影院 | 网站大全黄免费 | 视频一本大道香蕉久在线播放 | 色一情一乱一乱91av | 深点再深一点好爽好多水 | 在线女同免费观看网站 | 中文字幕一区二区三区在线不卡 | 色婷婷综合久久久中文字幕 | 黄色国产在线视频 |