- 预编译
- 将所有的“#define”删除,并且展开所有的宏定义
- 处理“#include”预编译指令,将被包含的文件插入到该预编译指令的位置。注意,这个过程是递归进行的,也就是说被包含的文件可能还包含其他文件
- 保留所有的#pragma编译器指令
- 编译
- 汇编
gcc –c hello.c –o hello.o - 链接
-
重定位
比如我们在第1条指令之后、第5条指令之前插入了一条或多条指令,那么第5条指令及后面的指令的位置将会相应地往后移动 -
使用符号来标记位置
-
符号
符号(Symbol)这个概念随着汇编语言的普及迅速被使用,它用来表示一个地址,这个地址可能是一段子程序(后来发展成函数)的起始地址,也可以是一个变量的起始地址。 -
寻找符号地址
静态链接
链接过程主要包括了地址和空间分配(Address and Storage Allocation)、符号决议(Symbol Resolution)和重定位(Relocation)等这些步骤。
模块main.c中使用func.c的函数foo()。调用foo的时候必须确切知道foo的地址,但在编译main.c的时候它并不知道foo的地址,所以它暂时把这些调用foo的指令的目标地址搁置,等待最后链接的时候由链接器去将这些指令的目标地址修正。
变量也有同样问题
https://zhuanlan.zhihu.com/p/111682188