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 |