可视化的网站开发工具,网站价值排行,电子商务网站建设与管理期末试题,wap网站怎么做转载#xff1a;https://blog.csdn.net/sinc00/article/details/44833839
今天在用g编译代码时#xff0c;提示说.rdata错误#xff0c;然后网上找了一堆资料#xff0c;最后明白了一个要重新编译对应的链接库。
在生成动态库时#xff0c;常常习惯性的加上fPIC选项https://blog.csdn.net/sinc00/article/details/44833839
今天在用g编译代码时提示说.rdata错误然后网上找了一堆资料最后明白了一个要重新编译对应的链接库。
在生成动态库时常常习惯性的加上fPIC选项fPIC有什么作用和意义加不加有什么区别这里做下小结
fPIC的全称是 Position Independent Code 用于生成位置无关代码。什么是位置无关代码个人理解是代码无绝对跳转跳转都为相对跳转。
1、不加fPIC选项
即使不加fPIC也可以生成.so文件但是对于源文件有要求例如
因为不加fPIC编译的so必须要在加载到用户程序的地址空间时重定向所有表目所以在它里面不能引用其它地方的代码
如下
#include stdio.h
int func1(int a) { printf(haha a%d\n, 2); a; return a; }
使用 gcc -shared -o libb3.so c.c 编译将报错
/usr/bin/ld: /tmp/ccCViivC.o: relocation R_X86_64_32 against .rodata can not be used when making a shared object; recompile with -fPIC /tmp/ccCViivC.o: could not read symbols: Bad value
将上述代码改为 int func1(int a) { // printf(haha a%d\n, 2); a; return a; }
则可以编译通过。
对于不加 -fPIC生成的动态库“ 生成动态库时假定它被加载在地址 0 处。加载时它会被加载到一个地址base这时要进行一次重定位relocation把代码、数据段中所有的地址加上这个 base 的值。这样代码运行时就能使用正确的地址了。”
2、加fPIC选项
加上fPIC选项生成的动态库显然是位置无关的
“这样的代码本身就能被放到线性地址空间的任意位置无需修改就能正确执行。通常的方法是获取指令指针的值加上一个偏移得到全局变量/函数的地址。”
加fPIC选项的 源文件对于它引用的函数头文件编写有很宽松的尺度。
比如只需要包含个声明的函数的头文件即使没有相应的C文件来实现编译成so库照样可以通过。
在内存引用上加不加fPIC的异同
加了fPIC实现真正意义上的多个进程共享so文件。
多个进程引用同一个 PIC 动态库时可以共用内存。这一个库在不同进程中的虚拟地址不同但操作系统显然会把它们映射到同一块物理内存上。
对于不加-fPIC的
不加fPIC则加载so文件时需要对代码段引用的数据对象需要重定位重定位会修改代码段的内容,这就造成每个使用这个.so文件代码段的进程在内核里都会生成这个.so文件代码段的copy.每个copy都不一样,取决于这个.so文件代码段和数据段内存映射的位置。
可见这种方式更消耗内存。
但是不加fPIC编译的 so文件的优点是加载速度比较快。 题外话能不能使用so库来静态编译(-static)一个可执行程序答案是否定的会出现错误提示
attempted static link of dynamic object