前情提要
- 对于编译和链接的基本过程,这里只对链接过过程的符号重定位做了解释,因为个人认为在链接过程中符号的重定位是最重要的一步,也是其精华所在,知道了这一步的实现过程可以解决很多问题,包括面试中可能会问到的关于extern等的方面。所以这里只有对符号重定位进行了详细的说明,如果想了解更多可以参看《程序员的自我修养》第2,3,6 章节。里面有很详细的解释。
如果对虚拟地址空间的内存分布还不够了解的建议先看上一篇博客IA32位Linux内核中的虚拟地址映射
其中的前面一部分有对虚拟内存分布的分析。
一,编译和链接的总体流程:
预编译:处理预编译指令(包括宏替换等#指令),删除注释。
具体如下:
1)将所有的“#define”删除,并且展开所有的宏定义
2)处理所有的条件预编译指令,将被包含的文件插入到该预编译指令的位置,注意:这个过程是递归进行的,也就是说被包含的文件可能还包含其他文件。
3)删除所有的注释”//”,”/* */”。
4)添加行号和文件名标识。e.g #2 “hello” 2,以便于编译时编译器产生调试用的行号信息及用于编译时产生编译错误或警告时能够显示行号。
5)保留所有的#pragma 编译器指令。编译阶段:进行一系列词法,语法,语义分析及优化后生成相应的汇编代码文件。(gcc -01最高级别优化),