Linux 共享库中函数的可见性
Linux共享库中的函数,默认都是可以外部可见的,也就是不需要像Windows下通过DEF文件或者dllexport指令来导出。先写一个源文件:MyExportFunctions.cpp,内容如下:
1 2 3 4 5 6 | int Add(int a, int b) {
return a + b;
}
int Sub(int a, int b) {
return a - b;
} |
用gcc编译成共享库:
1 | gcc -shared -o libExportFunctions.so MyExportFunctions.cpp |
用nm -g命令看查看所有导出的函数,Add和Sub都已经导出了。
1 2 3 4 5 6 7 8 9 10 11 12 13 | nm -g libExportFunctions.so
0000000000201028 B __bss_start
w __cxa_finalize@@GLIBC_2.2.5
0000000000201028 D _edata
0000000000201030 B _end
00000000000006ac T _fini
w __gmon_start__
0000000000000550 T _init
w _ITM_deregisterTMCloneTable
w _ITM_registerTMCloneTable
w _Jv_RegisterClasses
0000000000000680 T _Z3Addii
0000000000000694 T _Z3Subii |
不过为什么函数名不是Add和Sub,而是_Z3Addii和_Z3Subii (可以给nm 加上参数 –demangle来把c++修饰后的名字翻译回来,也可以用c++filt这个命令来转换),这是因为这两个函数是在c++源文件里声明的,而c++编译器会给函数名自动加上一些修饰符,解决方法就是加extern “C”.
1 2 3 4 5 6 7 8 | extern "C"{
int Add(int a, int b) {
return a + b;
}
int Sub(int a, int b) {
return a - b;
}
} |
nm -g 的结果是这样
1 2 3 4 5 6 7 8 9 10 11 12 13 | nm -g libExportFunctions.so
0000000000000670 T Add
0000000000201028 B __bss_start
w __cxa_finalize@@GLIBC_2.2.5
0000000000201028 D _edata
0000000000201030 B _end
000000000000069c T _fini
w __gmon_start__
0000000000000540 T _init
w _ITM_deregisterTMCloneTable
w _ITM_registerTMCloneTable
w _Jv_RegisterClasses
0000000000000684 T Sub |
这样Add和Sub都是原来的名字了。
那怎么样只导出需要的函数呢?gcc的一篇关于可见性的wiki说得挺详细gcc函数可见性
先在编译时用-fvisibility=hidden把函数都改成默认不可见 ,然后在需要导出函数的声明前加attribute ((visibility (“default”)))。修改后的代码
1 2 3 4 5 6 7 8 | extern "C"{
__attribute__ ((visibility ("default"))) int Add(int a, int b) {
return a + b;
}
int Sub(int a, int b) {
return a - b;
}
} |
然后在编译时加上-fvisibility=hidden
1 | gcc -shared -fvisibility=hidden -o libExportFunctions.so MyExportFunctions.cpp |
这次nm -g的输出中就只有Add,没有Sub了
1 2 3 4 5 6 7 8 9 10 11 12 | nm -g libExportFunctions.so
0000000000000660 T Add
0000000000201028 B __bss_start
w __cxa_finalize@@GLIBC_2.2.5
0000000000201028 D _edata
0000000000201030 B _end
000000000000068c T _fini
w __gmon_start__
0000000000000528 T _init
w _ITM_deregisterTMCloneTable
w _ITM_registerTMCloneTable
w _Jv_RegisterClasses |
上面说的这种方式类似Windows下的dllexport指令,那有没有跟DEF文件类似的东西可以在配置文件里配置哪些函数是导出的呢?gcc也有类似的配置文件,可以看GNU Export Maps。用法很简单,创建一个文本文件,叫exportmap,内容是这样
1 2 3 4 | {
global: Sub;
local: *;
}; |
然后编译时带–version-script参数
1 | gcc -shared -Wl,--version-script=exportmap -o libExportFunctions.so MyExportFunctions.cpp |
用nm -g查看,结果是
1 2 3 4 5 6 7 | nm -g libExportFunctions.so
w __cxa_finalize@@GLIBC_2.2.5
w __gmon_start__
w _ITM_deregisterTMCloneTable
w _ITM_registerTMCloneTable
w _Jv_RegisterClasses
00000000000005b4 T Sub |