久久久久久久999_99精品久久精品一区二区爱城_成人欧美一区二区三区在线播放_国产精品日本一区二区不卡视频_国产午夜视频_欧美精品在线观看免费

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 5333|回復: 0
打印 上一主題 下一主題
收起左側

GCC的符號可見性——解決多個庫同名符號沖突問題

[復制鏈接]
跳轉到指定樓層
樓主
ID:90014 發表于 2015-9-14 16:06 | 只看該作者 回帖獎勵 |倒序瀏覽 |閱讀模式

問題

最近項目遇到一些問題,場景如下

主程序依賴了兩個庫libA的funcA函數和libB的funcB函數。示意的代碼(main.cpp)如下:


#include <cstdio>int funcA(int, int);int funcB(int, int);int main() {    printf("%d,", funcA(2, 1));    printf("%d\n", funcB(2, 1));    return 0;}

libA示意實現(libA.cpp)如下:

int subfunc(int a, int b) {    return a + b;}int funcA(int a, int b) {    return subfunc(a, b);}

libB示意實現(libB.cpp)如下:

int subfunc(int a, int b) {    return a - b;}int funcB(int a, int b) {    return subfunc(a, b);}

可見funcA調用了libA中的內部函數subfunc,funcB調用了libB中的內部函數subfunc,這兩個subfunc實現不同,但不幸的是名字不小心起得一樣了

這時我們嘗試編譯并運行:

g++ -fPIC libA.cpp -shared -o libA.sog++ -fPIC libB.cpp -shared -o libB.sog++ main.cpp libA.so libB.so -o mainexport LD_LIBRARY_PATH=../main

我們期望的結果是3,1(funcA和funcB各自調用不同的subfunc實現),

實際得到的結果是3,3(funcA和funcB都調用了libA中的subfunc實現)

原因

我們通過readelf來查看符號:

$ readelf -a libA.so | grep subfunc000000200a60  000200000007 R_X86_64_JUMP_SLO 0000000000000708 _Z7subfuncii + 0     2: 0000000000000708    20 FUNC    GLOBAL DEFAULT   10 _Z7subfuncii    45: 0000000000000708    20 FUNC    GLOBAL DEFAULT   10 _Z7subfuncii$ readelf -a libB.so | grep subfunc 000000200a60  000200000007 R_X86_64_JUMP_SLO 0000000000000708 _Z7subfuncii + 0     2: 0000000000000708    22 FUNC    GLOBAL DEFAULT   10 _Z7subfuncii    45: 0000000000000708    22 FUNC    GLOBAL DEFAULT   10 _Z7subfuncii

可見libA和libB里面都有subfunc符號,名字完全一樣,而且都是GLOBAL的

GLOBAL的符號即全局的符號,同名的全局符號會被認為是同一個符號,由于main先加載了libA,得到了libA中的subfunc符號,再加載libB時,就把libB中的subfunc忽略了。

解決方案

這其實是符號的可見性(Symbol Visibility)問題,既然有GLOBAL符號,那自然會有LOCAL符號,LOCAL的符號只在當前lib可見,全局不可見。

如何將符號變成LOCAL的呢,最直接的就是加上visibility為hidden的標志,修改后的libA.cpp:

__attribute__ ((visibility ("hidden"))) int subfunc(int a, int b) {    return a + b;}int funcA(int a, int b) {    return subfunc(a, b);}

再重新編譯執行,可以得到結果為3,1,成功!這里再查看一下libA的符號:

$ readelf -a libA.so | grep subfunc    40: 00000000000006a8    20 FUNC    LOCAL  DEFAULT   10 _Z7subfuncii

可見subfunc符號已經變成了LOCAL

默認LOCAL

上面的方法可以解決問題,但是,實際情況往往是,libA里面有很多的內部函數,而暴露給外部的只有少數,能不能指定少數符號為GLOBAL,其它的都是LOCAL呢?答案是肯定的,修改libA.cpp如下:

int subfunc(int a, int b) {    return a + b;}__attribute__ ((visibility ("default"))) int funcA(int a, int b) {    return subfunc(a, b);}

這時,libA的編譯參數需要加上-fvisibility=hidden:

g++ -fPIC libA.cpp -shared -fvisibility=hidden -o libA.so

同樣可以解決問題。

跨平臺兼容性

windows平臺對于符號的行為是不一樣的,windows默認動態庫里符號是LOCAL的,通過__declspec(dllexport)來聲明GLOBAL符號,所以可以用下面的方式來兼容:

#if defined _WIN32 || defined __CYGWIN__  #ifdef BUILDING_DLL    #ifdef __GNUC__      #define DLL_PUBLIC __attribute__ ((dllexport))    #else      #define DLL_PUBLIC __declspec(dllexport) // Note: actually gcc seems to also supports this syntax.    #endif  #else    #ifdef __GNUC__      #define DLL_PUBLIC __attribute__ ((dllimport))    #else      #define DLL_PUBLIC __declspec(dllimport) // Note: actually gcc seems to also supports this syntax.    #endif  #endif  #define DLL_LOCAL#else  #if __GNUC__ >= 4    #define DLL_PUBLIC __attribute__ ((visibility ("default")))    #define DLL_LOCAL  __attribute__ ((visibility ("hidden")))  #else    #define DLL_PUBLIC    #define DLL_LOCAL  #endif#endif

隱藏外部依賴的符號

我遇到的實際情況比上面更復雜一些,subfunc并不是在libA中實現的,而是在另一個外部庫libsubfunc.a中實現的。libA通過包含頭文件來獲取到這個函數:

#include "subfunc.h"int funcA(int a, int b) {    return subfunc(a, b);}

上面的-fvisibility僅對實現生效,不能對聲明生效。但libsubfunc.a是第三方庫,我們不能去改它的代碼,也不能改它的頭文件,對于這種情況,gcc提供了下面方式來支持:

#pragma GCC visibility push(hidden)#include "subfunc.h"#pragma GCC visibility popint funcA(int a, int b) {    return subfunc(a, b);}

這種方式更方便靈活。

如果是dlopen載入so可以通過參數控制。


參考文檔

GCC Visibility

source:https://github.com/wwbmmm/blog/wiki/gcc_visibility






分享到:  QQ好友和群QQ好友和群 QQ空間QQ空間 騰訊微博騰訊微博 騰訊朋友騰訊朋友
收藏收藏 分享淘帖 頂 踩
回復

使用道具 舉報

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規則

手機版|小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術交流QQ群281945664

Powered by 單片機教程網

快速回復 返回頂部 返回列表
主站蜘蛛池模板: 欧美日韩精品一区二区三区视频 | 国产成人精品一区二 | 日本成人中文字幕在线观看 | 亚洲精品区 | 91久久久久久久 | 国产成人一区 | 欧美一区二区三区在线视频 | 久久久久av| 成人片网址 | 午夜精品久久久久久久久久久久 | 国产一级片在线播放 | 久久久一区二区三区 | www.久久影视 | 欧洲av在线 | 久久精品视频在线观看 | 日韩免费在线观看视频 | 日韩欧美不卡 | 日韩国产一区二区三区 | 国产农村一级国产农村 | 精品久久久久久久久久久久 | 91精品国产一区二区 | 91精品国产麻豆 | 亚洲精品乱码久久久久久按摩观 | 国产成人免费视频 | 亚洲精品久久久久久久久久久久久 | 欧美阿v| 久草福利 | 久久国产成人 | 久久久青草婷婷精品综合日韩 | 日韩在线综合网 | 波多野结衣中文视频 | 亚洲高清在线观看 | 日韩精品一区二区三区中文在线 | 欧美一区二区视频 | 免费精品久久久久久中文字幕 | 久草视频观看 | 嫩草视频免费 | 亚洲免费av一区 | 在线观看av网站 | 亚洲综合色婷婷 | 日日天天 |