gcc 编译器 编译过程中产生的 *.d 文件 详解

原文:https://blog.youkuaiyun.com/zhawk/article/details/53291428 

gcc 编译器 编译过程中产生的 *.d 文件 详解

.d 文件 dependencies 依赖文件 里面包含了 依赖的头文件

.d dependencies

依赖文件。

是给Makefile用的。内容和Makefile的target相似:

假设hello.c里有

#include "hello.h"

 

hello.h里有

#include "foo.h"

#include "bar.h"

 

gcc -c -MMD hello.c就会产生hello.d

 

hello.o: hello.c hello.h foo.h bar.h

 

在Makefile里用-include 进来后

即使hello.c没有修改,即使hello.c没有直接include foo.h bar.h 只要hello.h foo.h bar.h中的任意一个修改都会重新编译hello.c

 

自动生成依赖关系

大多数c/c++编译器提供了-M选项,可自动寻找源文件依赖的头文件,并生成依赖规则。对于gcc,需要使用-MM选项,否则它会把系统依赖的头文件也包含进来。

 

初次编译时,.d 不存在:

 

    调用隐含规则生成 .d

    包含 .d(注意:该文件定义 .d 依赖于 .cpp .h,由于 .d 是新建文件,肯定不过时)

    从最终目标开始推导依赖关系链

 

后续编译时,.d 已存在:

    包含 .d(注意:该文件定义 .d 依赖于 .cpp .h)

    如果 .d 过时,再次调用隐含规则重新生成 .d,并重新包含

    从最终目标开始推导依赖关系链

 

以下用一个简化后的例子进行说明。

 

Makefile文件内容:

 

final: abc.obj

    @echo "【$@】V1=$(V1)"

%.d: %.cpp

    @echo "【隐含规则1】$^ --> $@"

    echo -e "V1=$$$$\n$*.obj $@: $*.cpp $*.h" > $@

%.obj: %.cpp

    @echo "【隐含规则2】$^ --> $@"

include abc.d

 

输入下列命令准备测试环境:

    touch abc.cpp abc.h

 

初始时abc.d不存在。输入命令:

    make

 

处理流程:

- 调用“隐含规则1”生成abc.d。注意在此规则的执行过程中,abc.d的依赖文件($^)为abc.cpp。

- 包含abc.d,其中定义了abc.d依赖abc.cpp abc.h,由于abc.d是新建文件,肯定不过时。

- 从最终目标开始推导依赖关系链并处理。

 

更新abc.h,然后再次make。命令如下:

    touch abc.h

    make

 

处理流程:

- 包含abc.d,其中定义了abc.d依赖abc.cpp abc.h。

- 由于abc.h已更新,导致abc.d过时,因此调用“隐含规则1”重新生成abc.d。注意虽然规则只列出.cpp,但其依赖文件($^)仍然为abc.cpp abc.h。

- 重新包含abc.d,这一点可以从变量V1发生了改变得到证明。

- 从最终目标开始推导依赖关系链并处理。

 

 

 

-MF很好理解,就是输出依赖文件名

关于-MF和-MT,这些选项是用来生成依赖文件列表的,而这个列表又能以-include <dlist>或者include <dlist>的形式添加到Makefile中。其中MF指定文件名,MT指定依赖目标名。

### GCC编译器编译过程详解 #### 预处理阶段 预处理阶段主要负责处理源代码中的宏定义、条件编译指令以及头文件包含等内容。此阶段会生成一个扩展后的源文件,通常以`.i`作为后缀名[^2]。 ```bash gcc -E hello.c -o hello.i ``` 上述命令展示了如何仅执行预处理操作并将结果保存到`hello.i`文件中。 #### 编译阶段 经过预处理之后进入编译阶段,在这个过程中,预处理器产生的中间文件被转换成汇编语言形式的代码,即生成带有`.s`后缀的目标文件。这一步骤涉及复杂的语义分析和优化工作,最终产出适合后续步骤进一步加工的数据结构表示。 ```bash gcc -S hello.i -o hello.s ``` 这里展示的是单独运行编译器前端的方式,它接受由预处理器输出的结果并将其转化为人类可读但机器能理解的形式——汇编代码。 #### 汇编阶段 当拥有了汇编级别的描述以后就可以继续前进至汇编环节了;在此期间,汇编器把之前获得的人类易懂却计算机难解的文字表述转变为了二进制格式的对象模块(.o),这些对象模块包含了可以直接加载入内存供CPU执行的具体指令序列。 ```bash gcc -c hello.s -o hello.o ``` 这条命令用于指示GCC只做汇编而不进行链接动作,从而得到未链接的状态下的目标文件(`.o`)。 #### 链接阶段 最后是链接部分,这是整个构建链上的最后一环。链接器的任务在于收集所有必要的输入片段(可能是多个不同来源处生产出来的`.o`),再加上一些标准库函数的支持材料一起打包组合起来形成完整的可执行映像。对于C/C++项目而言,默认情况下还会引入启动代码以便于操作系统能够正确初始化环境进而顺利启动应用程序主体逻辑[^4]。 ```bash gcc -o hello hello.o ``` 以上就是从原始源码直至最终成品之间所经历的主要几个关键节点概述。值得注意之处在于实际应用当中往往并不需要如此细致地区分每一步具体行为,因为大多数时候开发者更倾向于依赖工具链本身所提供的自动化机制来简化日常作业流程。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值