linux-编译连接原理

在linux系统下编写代码后,通过gcc进行编译和链接生成可执行文件 细分就是四大部分:预编译,编译,汇编,链接

在这里插入图片描述

预编译

gcc下,首先会将C文件生成 .i 文件,C++文件生成 .ii 文 件。

gcc -E 源文件 -o 生成文件

预编译做的事情:

a) 删除所有的“#define”,并且展开所有的宏定义;
b) 处理所有的条件预编译指令,“#if”、“#ifdef”、“#endif”等;
c) 处理“#include”预编译指令,将被包含的文件插入到该预编译指令的位置;
d) 删除所有的注释;
e) 添加行号和文件名标识,以便于编译器产生调试用的符号信息及编译时产生编译错
误和警告时显示行号;
f) 保留所有的#pragma 编译器指令,因为编译器需要使用它们。

预编译后的文件不包含任何宏的定义,所以可以通过预编译后的文件来检查宏定义是否正确。

C语言的宏替换和文件包含等工作一般不归入编译器的范围而交给一个独立的预处理器

编译

编译做的事情:

1、词法分析

2、语法分析

3、语义分析

4、源代码优化

5、目标代码生成(汇编)与优化

6.汇总符号

汇编

生成二进制可重定位的目标文件
将汇编指令翻译成二进制格式,生成各个 section,生成符号表。

链接

a) 合并各个 section,调整 section 的起始位移和段大小,合并符号表,进行符号解析,
给符号分配虚拟地址
b) 符号重定位

什么是符号解析?

符号解析:所有对符号的引用,都有找到该符号定义的地方。常见错误:符号重定义,符号未定义。

什么是重定位?

《程序员自我修养》这本书中给重定位的定义是:在编译多个目标文件时,无法确定变量/函数地址的情况下,会将其目标地址置为0,等到链接时将目标地址进行修改。这个地址修正的过程就被叫做重定位。就是给程序中的每个绝对地址引用的位置打补丁,使它们指向正确的位置。

因为在编译过程中符号是不分配虚拟地址的,但是编译过程指令已经生成。所以先把符号的地址都置为0,等到链接时符号解析成功以后,给所有的符号分配虚拟地址,写入指令中置0的地方,称之为符号的重定向。

可以发现目标文件和最终的可执行文件格式上来说是很像的,但是目标文件不能执行的原因之一就是因为其还不确定符号的地址。必须得等到链接时对其地址进行重定位。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值