成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

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

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

[[425670]]

【目錄】

  • 第一個動態(tài)庫文件
  • 應(yīng)用程序
  • 第二個動態(tài)庫文件
  • 錯誤做法:直接給它改名
  • 正解:patchelf 工具
  • One More Thing

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

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

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

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

第一個動態(tài)庫文件

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

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

  1. // 第一個動態(tài)庫 源文件 RobotMath.c: 
  2. double func0(double arg) 
  3.     double ret = arg + arg; 
  4.     return ret; 
  5.  
  6. double func1(double arg1, double arg2) 
  7.     double ret = arg1 + arg2; 
  8.     return ret; 

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

  1. $ 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 :

  1. $ patchelf --print-soname libRobotMath.so  
  2. 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)交代清楚了,下面再來看一下最簡單的應(yīng)用程序。

應(yīng)用程序

  1. // 可執(zhí)行程序 源文件: main.c 
  2. extern double func0(double arg); 
  3. extern double func1(double arg1, double arg2); 
  4.  
  5. int main(int argc, char *agv[]) 
  6.     double arg = 1.1; 
  7.     double result0 = func0(arg); 
  8.     printf("result0 = %lf \n", result0); 
  9.  
  10.     double arg1 = 1.1, arg2 = 2.2; 
  11.     double result1 = func1(arg1, arg2); 
  12.     printf("result1 = %lf \n", result1); 
  13.  
  14.     return 0; 

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

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

執(zhí)行:

  1. $ ./main  
  2. result0 = 2.200000  
  3. result1 = 3.300000 

完美!

第二個動態(tài)庫文件

問題來了:現(xiàn)在應(yīng)用程序還需要實(shí)現(xiàn)另外一個復(fù)雜的算法,本著偷懶的精神,終于在另外一個機(jī)器人算法相關(guān)的庫中找到了這個算法。

  1. // 第二個動態(tài)庫 源文件 RobotMath.c: 
  2. double func2(double arg1, double arg2, double arg3) 
  3.     double ret = arg1 * arg2 * arg3; 
  4.     return ret; 
  5.  
  6. // 編譯指令 
  7. $ gcc -m32 -fPIC --shared -o libRobotMath.so -Wl,--soname,libRobotMath.so RobotMath.c 

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

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

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

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

錯誤做法:直接給它改名

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

  1. $ mv libRobotMath.so libRobotMath2.so 

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

  1. extern double func2(double arg1, double arg2, double arg3); 
  2.  
  3. int main(int argc, char *agv[]) 
  4.     // 之前的其它代碼 
  5.     // ... 
  6.  
  7.     double arg3 = 1.1, arg4 = 2.2, arg5 = 3.3; 
  8.     double result2 = func2(arg3, arg4, arg5); 
  9.     printf("result2 = %lf \n", result2); 
  10.  
  11.     return 0; 

編譯一下試試:

  1. $ gcc -m32 -o main main.c -lRobotMath -lRobotMath2 -L./ -Wl,-rpath=./ 
  2. /tmp/ccDGqFkl.o: In function `main': 
  3. main.c:(.text+0xb4): undefined reference to `func2' 
  4. collect2: error: ld returned 1 exit status 

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

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

  1. $ readelf -s libRobotMath2.so | grep func2 
  2.      8: 0000052a    69 FUNC    GLOBAL DEFAULT   11 func2 
  3.     51: 0000052a    69 FUNC    GLOBAL DEFAULT   11 func2 

為啥 gcc 還找不到呢?

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

正解:patchelf 工具

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

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

  1. $ patchelf --print-soname libRobotMath2.so  
  2. libRobotMath.so 

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

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

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

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

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

  1. $ patchelf -h 
  2. syntax: patchelf 
  3.   [--set-interpreter FILENAME] 
  4.   [--page-size SIZE] 
  5.   [--print-interpreter] 
  6.   [--print-soname]      Prints 'DT_SONAME' entry of .dynamic section. Raises an error if DT_SONAME doesn't exist 
  7.   [--set-soname SONAME]     Sets 'DT_SONAME' entry to SONAME. 
  8.   [--set-rpath RPATH] 
  9.   [--remove-rpath] 
  10.   [--shrink-rpath] 
  11.   [--print-rpath] 
  12.   [--force-rpath] 
  13.   [--add-needed LIBRARY] 
  14.   [--remove-needed LIBRARY] 
  15.   [--replace-needed LIBRARY NEW_LIBRARY] 
  16.   [--print-needed] 
  17.   [--no-default-lib] 
  18.   [--debug] 
  19.   [--version] 
  20.   FILENAME 

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

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

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

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

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

  1. $ patchelf --print-soname libRobotMath2.so  
  2. libRobotMath2.so 

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

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

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

沒有報(bào)錯!

執(zhí)行一下:

  1. $ ./main  
  2. result0 = 2.200000  
  3. result1 = 3.300000  
  4. result2 = 7.986000 

問題解決了!

One More Thing

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

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

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

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

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

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

本文轉(zhuǎn)載自微信公眾號「IOT物聯(lián)網(wǎng)小鎮(zhèn)」,可以通過以下二維碼關(guān)注。轉(zhuǎn)載本文請聯(lián)系IOT物聯(lián)網(wǎng)小鎮(zhèn)公眾號。

 

責(zé)任編輯:武曉燕 來源: IOT物聯(lián)網(wǎng)小鎮(zhèn)
相關(guān)推薦

2014-06-05 11:25:10

2023-11-23 08:30:16

2022-12-07 08:47:28

2024-02-28 08:10:15

Linux靜態(tài)庫動態(tài)庫

2010-08-18 17:06:02

DB2數(shù)據(jù)庫編譯

2016-11-10 19:10:09

唯品會雙11

2020-11-13 07:16:09

線程互斥鎖死循環(huán)

2012-03-21 10:15:48

RIM越獄

2012-06-14 09:48:11

OpenStackLinux

2022-11-24 12:22:39

2011-02-28 10:38:13

Windows 8

2012-03-07 17:24:10

戴爾咨詢

2012-12-20 10:17:32

IT運(yùn)維

2009-06-12 15:26:02

2017-08-25 15:08:24

深度學(xué)習(xí)機(jī)器人

2021-10-20 20:02:47

字符變量函數(shù)

2021-07-14 10:00:28

數(shù)據(jù)庫SQL查詢Linux

2015-07-15 09:29:27

UI設(shè)計(jì)

2021-11-05 10:59:06

元編程語言工具

2022-09-09 18:59:28

Vue類型枚舉
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號

主站蜘蛛池模板: 欧美在线视频不卡 | 亚洲精品久久久久久国产精华液 | 久久久久久久综合 | 欧美日韩精品 | 欧美在线观看黄色 | 精品毛片在线观看 | 蜜桃av一区二区三区 | 国产欧美一区二区三区日本久久久 | 日韩精品一区二区三区中文字幕 | 成人av在线大片 | 国产视频中文字幕在线观看 | 日韩精品免费在线观看 | 日韩精品1区2区3区 爱爱综合网 | 亚洲国产一区在线 | 亚洲国产一区二区视频 | 国产传媒在线观看 | 不卡视频一区二区三区 | 国产精品99久久久久久久久久久久 | 欧美日韩综合精品 | 欧美午夜精品久久久久免费视 | 亚洲成人av在线播放 | 国产女人第一次做爰毛片 | 九九热视频这里只有精品 | 中文在线一区二区 | 真人女人一级毛片免费播放 | 日本不卡一区 | aⅴ色国产 欧美 | 精品国产一区二区三区久久狼黑人 | 国产精品永久免费 | 色偷偷噜噜噜亚洲男人 | 国产在线中文字幕 | 国产精品视频播放 | 三区在线 | 免费看国产片在线观看 | 91精品一区二区三区久久久久久 | 韩日在线观看视频 | 中文字幕1区2区3区 亚洲国产成人精品女人久久久 | 免费一级黄色 | av高清 | 欧美一级小视频 | 亚洲激情在线视频 |