哈哈哈哈哈操欧洲电影,久草网在线,亚洲久久熟女熟妇视频,麻豆精品色,久久福利在线视频,日韩中文字幕的,淫乱毛视频一区,亚洲成人一二三,中文人妻日韩精品电影

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

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

如何鏈接兩個名字一樣動態(tài)庫

Linux閱碼場 ? 來源:IOT物聯(lián)網(wǎng)小鎮(zhèn) ? 作者:道哥 ? 2021-10-08 14:58 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

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

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

具體來說面對的問題是:在編譯可執(zhí)行程序的時候,通過gcc編譯參數(shù)的-lXXX就可以動態(tài)鏈接一個動態(tài)庫。

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

第一個動態(tài)庫文件現(xiàn)在,假設我們在開發(fā)一個機器人應用程序,需要用到一個第三方動態(tài)庫中的算法。

這個庫的源碼很簡單,如下:

// 第一個動態(tài)庫 源文件 RobotMath.c:

double func0(double arg)

{

double ret = arg + arg;

return ret;

}

double func1(double arg1, double arg2)

{

double ret = arg1 + arg2;

return ret;

}

動態(tài)庫的編譯命令是:

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

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

為了方便起見,這里就不加版本信息了。

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

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

$ patchelf --print-soname libRobotMath.so

libRobotMath.so // SONAME

第2行打印出來的就是所謂的 SONAME。

你也可以測試一下,指定其他的 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

以上就是第一個動態(tài)庫,已經(jī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;

}

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

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

執(zhí)行:

$ 。/main

result0 = 2.200000

result1 = 3.300000

完美!

第二個動態(tài)庫文件問題來了:現(xiàn)在應用程序還需要實現(xiàn)另外一個復雜的算法,本著偷懶的精神,終于在另外一個機器人算法相關的庫中找到了這個算法。

// 第二個動態(tài)庫 源文件 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

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

與第一個算法庫的文件名同名同姓,看來這個名字太招人喜歡了。

如果這個作者直接起一個其它的名字,那就啥事都沒有了。

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

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

$ mv libRobotMath.so libRobotMath2.so

然后把libRobotMath2.so復制到應用程序的目錄下,并在main.c中,調用這個庫中的算法函數(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

報錯:找不到 func2 這個函數(shù)。

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

$ readelf -s libRobotMath2.so | grep func2

8: 0000052a 69 FUNC GLOBAL DEFAULT 11 func2

51: 0000052a 69 FUNC GLOBAL DEFAULT 11 func2

為啥 gcc 還找不到呢?

看來,很粗魯?shù)刂苯咏o第二個動態(tài)庫文件強行改名,不是解決問題的正確思路!

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

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

$ patchelf --print-soname libRobotMath2.so

libRobotMath.so

SONAME 依然是原來的名稱,說明通過mv指令改名,只是改變了外表,并沒有改變它的內心。

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

動態(tài)庫是一個ELF格式的文件,操作系統(tǒng)在加載動態(tài)庫的時候,是根據(jù)ELF格式的標準,對文件的內容進行一層一層解析的。

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

patchelf 這個工具,就提供了這樣的功能:查看或修改動態(tài)庫文件的內部信息,包括:SONAME, 依賴的其他動態(tài)庫,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這個參數(shù),來把它的 SONAME 修改一下:

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

第一個 libRobotMath2.so,是設置的 SONAME 名稱;

第二個 libRobotMath2.so,是指定修改哪一個動態(tài)庫文件的 SONAME;

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

$ patchelf --print-soname libRobotMath2.so

libRobotMath2.so

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

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

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

沒有報錯!

執(zhí)行一下:

$ 。/main

result0 = 2.200000

result1 = 3.300000

result2 = 7.986000

問題解決了!

One More Thing什么?你說這樣的問題是千年等一回?是為賦新詞強說愁?那說明走過的路還不是足夠的長。

記得大概是2015年的時候,開發(fā)一個網(wǎng)關,在硬件出來之前需要在Ubuntu (x86)平臺上進行模擬。

為了便于跨平臺,選擇了 glib 庫,但是對其中的小部分源碼進行了二次開發(fā)。

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

那么我們的應用程序在編譯時,的確可以鏈接到自己二次開發(fā)的glib庫(放在本地文件夾),但是在執(zhí)行時,一直加載不成功,就是因為動態(tài)庫的名字沖突問題導致的。

最后沒辦法,只好利用 patchelf 工具,對動態(tài)庫的名稱,包括 SONAME 進行改寫,這樣才解決問題。

責任編輯:haq

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

    關注

    88

    文章

    11806

    瀏覽量

    219481
  • 代碼
    +關注

    關注

    30

    文章

    4975

    瀏覽量

    74326

原文標題:鏈接兩個"名字完全一樣"的【動態(tài)庫】,你會怎么處理?

文章出處:【微信號:LinuxDev,微信公眾號:Linux閱碼場】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評論

    相關推薦
    熱點推薦

    AI “拆彈專家”!如何打造一樣的“劉德華”

    正如劉德華演繹的《拆彈專家》電影中一樣,現(xiàn)實中的防爆排爆,需要拆彈專家穿著防護服,進行近距離拆除。這種非常危險的職業(yè),稍有不慎就容易對拆彈人員的生命造成威脅。隨著機器人技術應用的發(fā)展,采用機器人
    的頭像 發(fā)表于 03-31 17:59 ?296次閱讀
    AI “拆彈專家”!如何打造<b class='flag-5'>一</b><b class='flag-5'>個</b>不<b class='flag-5'>一樣</b>的“劉德華”

    兩個RS485-Modbus主站如何通訊

    本產品能很好解決Master-1主站向模塊寫入數(shù)據(jù),Master-2主站讀取數(shù)據(jù);Master-2主站向模塊寫入數(shù)據(jù),Master-1主站讀取數(shù)據(jù)。由此解決兩個主站之間的互相讀通信難題。
    發(fā)表于 02-08 15:32 ?0次下載

    曙光存儲連續(xù)斬獲兩個行業(yè)獎項

    近期,曙光存儲連續(xù)斬獲兩個行業(yè)獎項,自研技術產品在國產突破、AI行業(yè)應用等方面的成果獲得廣泛關注。
    的頭像 發(fā)表于 01-15 16:28 ?2694次閱讀

    連載|開發(fā)工具,易安卓讓系統(tǒng)功能調用像寫應用邏輯一樣簡單

    通過自研系統(tǒng)封裝,易安卓將復雜的系統(tǒng)控制能力以接口形式開放給開發(fā)者,讓系統(tǒng)功能調用像寫應用邏輯一樣簡單。
    的頭像 發(fā)表于 11-27 11:40 ?87次閱讀
    連載|開發(fā)工具,易安卓讓系統(tǒng)功能調用像寫應用邏輯<b class='flag-5'>一樣</b>簡單

    一樣的展會,不一樣的精彩 2025灣芯展圓滿收官

    10月17日,2025灣區(qū)半導體產業(yè)生態(tài)博覽會(2025灣芯展)在深圳會展中心(福田)圓滿收官。全球600多家展商、超30場論壇,在6萬平方米的展區(qū)內,打造場“不一樣的展會”,呈現(xiàn)出“不一樣的精彩
    的頭像 發(fā)表于 10-29 16:56 ?1465次閱讀
    不<b class='flag-5'>一樣</b>的展會,不<b class='flag-5'>一樣</b>的精彩 2025灣芯展圓滿收官

    一樣的展會,不一樣的精彩:2025灣芯展順利收官

    內,打造場"不一樣的展會",呈現(xiàn)出"不一樣的精彩"。本屆展會人氣火爆,展期三天累計接待總量達到11.23萬人次。參展企業(yè)集中發(fā)布年度新品數(shù)約2500件,新品發(fā)布與商業(yè)合作收獲頗豐。 ? 不
    的頭像 發(fā)表于 10-19 20:27 ?868次閱讀
    不<b class='flag-5'>一樣</b>的展會,不<b class='flag-5'>一樣</b>的精彩:2025灣芯展順利收官

    飛凌嵌入式ElfBoard-Vim編輯器之靜態(tài)鏈接動態(tài)鏈接

    文件,般是以.so文件形式存在。GCC默認是動態(tài)鏈接不需要加參數(shù)。舉例子看動態(tài)鏈接庫的使用
    發(fā)表于 10-17 09:07

    硬件SPI兩個CS操作兩個norflash,怎么互斥操作兩個norflash?

    硬件SPI兩個CS操作兩個norflash,怎么互斥操作兩個norflash,有
    發(fā)表于 09-26 06:18

    種TVS有啥不同?

    當我們查看TVS二極管的規(guī)格書,常會看到有以下種種引腳功能標識圖:對于初學者,看到感到疑惑,他們一樣嗎?他們有啥區(qū)別?為啥有的兩個尖頭往外,陽極連在起,有的
    的頭像 發(fā)表于 09-15 20:27 ?946次閱讀
    這<b class='flag-5'>兩</b>種TVS有啥不同?

    基本半導體連獲兩個行業(yè)獎項

    近日,基本半導體憑借在碳化硅模塊領域的突出表現(xiàn),連獲“國產SiC模塊TOP企業(yè)獎”和“年度優(yōu)秀功率器件產品獎”兩個行業(yè)獎項。
    的頭像 發(fā)表于 09-05 16:31 ?1216次閱讀

    現(xiàn)場解決EMC電磁輻射干擾:“望聞問切”,像中醫(yī)一樣

    南柯電子|現(xiàn)場解決EMC電磁輻射干擾:“望聞問切”,像中醫(yī)一樣
    的頭像 發(fā)表于 09-04 09:47 ?1010次閱讀

    看到STM8L152用兩個IO用兩個或非門檢測兩個通斷,是什么原理呢?

    圖中兩個按鍵開關是兩個干簧管,為什么不直接對GND設計來檢測這個干簧管通斷呢? 這樣設計的原理是什么?
    發(fā)表于 06-12 06:25

    圖像采集卡和顯卡是一樣的嗎?從核心差異、工作原理與應用全解析

    不少朋友在做系統(tǒng)集成或設備選型時,經(jīng)常會問看似簡單但又容易混淆的問題:圖像采集卡和顯卡一樣嗎?一個是“采圖”的,一個是“顯圖”的,聽起來
    的頭像 發(fā)表于 05-14 09:52 ?1751次閱讀
    圖像采集卡和顯卡是<b class='flag-5'>一樣</b>的嗎?從核心差異、工作原理與應用全解析

    貼片電容和瓷片電容一樣嗎?

    貼片電容和瓷片電容并不完全一樣,它們在結構、材料、特點和應用等方面存在些差異。以下是對這種電容器的詳細比較: 、結構差異 貼片電容: 結構上,貼片電容是
    的頭像 發(fā)表于 04-30 15:05 ?1029次閱讀
    貼片電容和瓷片電容<b class='flag-5'>一樣</b>嗎?

    請問如何鏈接動態(tài)?

    是否有可參考的工程? 鏈接成功后動態(tài)應該放在哪里???SDK是RTOS_ONLY
    發(fā)表于 04-25 08:15
    霍城县| 清丰县| 巴楚县| 修水县| 渝北区| 水富县| 那曲县| 阳东县| 南丰县| 东乌珠穆沁旗| 农安县| 大城县| 塔城市| 儋州市| 泉州市| 宜兰市| 潼南县| 靖江市| 聂荣县| 长泰县| 海原县| 洛宁县| 五峰| 临湘市| 顺昌县| 沾益县| 牡丹江市| 榆中县| 岱山县| 德江县| 晋城| 清新县| 闻喜县| 博爱县| 祁连县| 辉县市| 恭城| 合肥市| 城口县| 哈密市| 肇州县|