這篇文章主要介紹了linux 下同名符號衝突問題解決方案的相關資料,需要的朋友可以參考下
linux 下同名符號衝突問題解決方案
最近的工作中遇到如下令人蛋疼的問題:
Linux 下有三個模組aa、bb、cc,基本情況如下:
cc 編譯連接得到cc. so 動態函式庫,cc 中有如下介面:
cc_fun { …… do();//调用名为do的cc模块内部函数 …… }
bb 編譯連線得到bb.a 靜態函式庫,bb 中有以下介面:
bb_fun { …… handle = dlopen(cc.so, RTLD_LAZY);//加载cc.so pccfun = dlsym(handle, “cc_fun”);//获取cc_fun函数指针 (*pccfun)();//调用cc_fun函数,此时应该会调用cc模块中的do()函数 do();//调用名为do的bb模块内部函数(与cc模块中的do()函数同名,实现却不相同) …… }
aa 編譯後透過-lbb 連結選項的方式連接bb.a 得到aa 可執行程序,並呼叫bb.a 的介面函數bb_fun( ):
main { …… bb_fun();//调用bb_fun函数 …… }
工作中發現aa 在運行時行為異常,總是有記憶體洩漏和功能異常,透過定位發現問題集中在同名的do() 函數上。透過輸出列印發現程式中兩次呼叫 do() 函數都呼叫了 bb 模組中的 do() 函數,而 cc 模組中的 do() 函數從未被呼叫到,導致程式行為異常和記憶體洩漏。
後經多方查證了解到因為linux 程式中各個函式庫中的符號表最終都會載入到程式所在的全域符號表中,此時如果有同名符號就只能呼叫到第一個載入進來的符號,也就是說後邊加載的同名符號都會被之前的覆蓋。 cc 模組中的 do() 函數被 bb 模組中的 do() 函數覆寫了,所以無法被呼叫到。
廢話不多說。 。 。
在試驗過很多不滿意的方法之後,最終的解決方法如下:
1.在cc 的makefile 中加入-Wl,-Bsymbolic -Wl, --version-script,version 的連線選項,意思是用version 檔案中的腳本指定其要匯出哪些函數。
2.version 檔案的實作如下:
VERS{ global: cc_fun; local: *; };
意思是指定cc 模組只匯出介面函數cc_fun,其餘函數都設為local 不做匯出。
將該檔案儲存在 makefile 所在目錄即可。
3.重新編譯連接三個模組,問題解決。
以上是linux 下同名符衝突的問題及解決方法的詳細內容。更多資訊請關注PHP中文網其他相關文章!