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