屏蔽静态库接口

分享屏蔽静态库接口的一种方法.

准备

hello.c:

1
2
3
4
5
#include <stdio.h>

__attribute__ ((visibility ("default"))) void hello() {
printf("Hello World!\n");
}

hello.h:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
#ifndef __HELLO__H
#define __HELLO__H

#ifdef __cplusplus
extern "C" {
#endif

void hello();

#ifdef __cplusplus
}
#endif

#endif

bye.c:

1
2
3
4
5
#include <stdio.h>

void bye() {
printf("Bye Bye!\n");
}

bye.h:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
#ifndef __BYE__H
#define __BYE__H

#ifdef __cplusplus
extern "C" {
#endif

void bye();

#ifdef __cplusplus
}
#endif

#endif

编译

编译时使用-fvisibility=hidden,可以默认将符号隐藏;需要对外的符号使用__attribute__ ((visibility ("default")))修饰即可:

1
2
$ gcc -fvisibility=hidden -I. -c hello.c -o hello.o
$ gcc -fvisibility=hidden -I. -c bye.c -o bye.o

其中hello()未被隐藏,bye()是被隐藏的.

链接

将生成的两个.o文件重定位到libt.o中:

1
$ ld -r hello.o bye.o -o libt.o

去除无用的符号

1
$ strip --strip-unneeded libt.o

隐藏的符号本地化(我也不知道中文怎么翻译了)

1
$ objcopy --localize-hidden libt.o libt_hidden.o

打包成静态库

1
$ ar crv libt.a libt_hidden.o

验证

调用未被隐藏的hello()

test1.c:

1
2
3
4
5
6
#include "hello.h"

int main(void) {
hello();
return 0;
}

编译并运行

1
2
3
$ gcc -I. test1.c -L. -lt -o test
$ ./test
Hello World!

调用隐藏的bye()

test2.c

1
2
3
4
5
6
#include "bye.h"

int main(void) {
bye();
return 0;
}

编译并运行

1
2
3
4
5
$ gcc -I. test2.c -L. -lt -o test
$ ./test
/tmp/ccdaJT7s.o: In function `main':
test2.c:(.text+0xa): undefined reference to `bye'
collect2: error: ld returned 1 exit status

Reference

About me

forthebadge

Creative Commons License This work is licensed under a Creative Commons Attribution-ShareAlike 4.0 International License.
本作品采用知识共享署名-相同方式共享 4.0 国际许可协议进行许可。