简单来说就是gcc与ld。
我也是没完全搞懂。
先记录下来:
#编写一个c代码文件,编译
echo 'main(){}' > dummy.c
cc dummy.c -v -Wl,--verbose &> dummy.log //日志信息输出到dummy.log
# 此时dummy.c编译之后会生成一个a.out的可执行文件
readelf -l a.out | grep ': /lib'
# readelf命令,了解源码编译,汇编过程最后一个步骤类似elf文件处理
# elf文件包括{可重定位文件lib*.a}{可执行文件a.out}{共享目标文件lib*.so}
# readelf命令输出结果:/lib64/ld-linux-x86-64.so.2就是动态链接器
[Requesting program interpreter: /lib64/ld-linux-x86-64.so.2]
确认启动文件设置正确:
grep -o '/usr/lib.*/crt[1in].*succeeded' dummy.log
# 输出结果
/usr/lib/gcc/x86_64-redhat-linux/4.4.7/../../../../lib64/crt1.o succeeded
/usr/lib/gcc/x86_64-redhat-linux/4.4.7/../../../../lib64/crti.o succeeded
/usr/lib/gcc/x86_64-redhat-linux/4.4.7/../../../../lib64/crtn.o succeeded
验证编译器能搜索正确的头文件:
grep -B4 '^ /usr/include' dummy.log // -B 找到某行且显示此行的前4行,-A是显示此行的后#行
# 输出结果
#include "..." 搜索从这里开始:
#include <...> 搜索从这里开始:
/usr/local/include
/usr/lib/gcc/x86_64-redhat-linux/4.4.7/include
/usr/include
验证链接器使用了正确的搜索路径:
grep 'SEARCH.*/usr/lib' dummy.log |sed 's|; |\n|g'
# 输出结果:
SEARCH_DIR("/usr/x86_64-redhat-linux/lib64")
SEARCH_DIR("/usr/local/lib64")
SEARCH_DIR("/lib64")
SEARCH_DIR("/usr/lib64")
SEARCH_DIR("/usr/x86_64-redhat-linux/lib")
SEARCH_DIR("/usr/lib64")
SEARCH_DIR("/usr/local/lib")
SEARCH_DIR("/lib")
SEARCH_DIR("/usr/lib");
确认使用正确的libc:
# grep "/lib.*/libc.so.6 " dummy.log
#输出结果:
attempt to open /lib64/libc.so.6 succeeded
确认GCC使用的正确的动态编译器:有些程序习惯用cc编译,其实就是gcc名字不一样而已
grep found dummy.log
# 输出结果:
found ld-linux-x86-64.so.2 at /lib64/ld-linux-x86-64.so.2