什么是链接
链接是将各种代码和数据片段收集并组合成为单一文件的过程
编译器驱动程序
int sum(int *a,int n);
int array[2]={1,2};
int main(){
int val=sum(array,2);
return val;
}
main.c
int sum(int *a,int n){
int i,s=0;
for(i=0;i<n;i++){
s+=a[i];
}
return s;
}
sum.c
我们可以通过下面的语句来调用gcc驱动程序
$ gcc -Og -o prog main.c sum.c
??这里我不知道怎么用-v 来查看步骤
我们可以来看一下这个过程(那main.c来说)
首先:
预处理器(cpp)将main.c翻译成一个ascII码的中间文件main.i:
cpp [other arguments] main.c /tmp/main.i
接下来,驱动程序运行c编译器(cc1)将main.i->main.s(ascll汇编语言文件)
cc1 /tmp/main.i -Og[other arguments ] -o /tmp/main.s
然后运行汇编器(as)将main.s转为main.o(可重定位目标文件)
as [other argument ] -o /tmp/main.o /tmp/main.s
最后运行连接器(ld),将main.o和sum.o和有关代码组合起来,形成一个可执行目标文件
ld -o prog [system object files and args] /tmp/main.o /tmp/sum.o
链接的主要任务
符号解析
将符号定义与引用对应起来
比如想上面的main.o和sum.o链接的过程中,链接器就得将main.o中对sum的引用与sum.o中的sum的定义对应起来。
这里的符号可以是一个函数,一个全局变量,一个静态变量
思考:这里为什么没有提到局部变量
重定位
因为我们的链接器和汇编器生成的是从0字节开始的代码段,链接器首先对把每一个符号定义与一个具体的内存位置关联起来,从而重定位这些节,然后修改所有对这些符号的引用,使他们指向这个内存位置,
目标文件
可重定位目标文件
包含二进制代码和数据,其形式可以在编译时与其他可重定位目标文件合并起来
可执行目标文件
包含二进制,代码和数据,其形式可以被直接复制到内存并执行
共享目标文件:
一种特殊类型的可重定位目标文件,可以在加载或者运行时被动态加载进内存并链接