GCC c选项

GCC -c选项:生成目标文件

我们知道,从 CC++ 源代码生成可执行文件要经历 4 个过程,分别为预处理、编译、汇编和链接。

[root@haicoder demo]# ls demo.c [root@haicoder demo]# cat demo.c #include <stdio.h> int main(){ puts("Hello,World!"); return 0; } [root@haicoder demo]# gcc -E demo.c -o demo.i [root@haicoder demo]# ls demo.c demo.i [root@haicoder demo]# gcc -S demo.i [root@haicoder demo]# ls demo.c demo.i demo.s

在此基础上,本节继续讲解如何对已得到的 dmeo.s 执行汇编操作,并得到相应的目标文件。所谓目标文件,其本质为二进制文件,但由于尚未经过链接操作,所以无法直接运行。

gcc -c指令

简单地理解,汇编其实就是将汇编代码转换成可以执行的机器指令。大部分汇编语句对应一条机器指令,有的汇编语句对应多条机器指令。

相对于编译操作,汇编过程会简单很多,它并没有复杂的语法,也没有语义,也不需要做指令优化,只需要根据汇编语句和机器指令的对照表一一翻译即可。

通过为 gcc 指令添加 -c 选项(注意是小写字母 c),即可让 GCC 编译器将指定文件加工至汇编阶段,并生成相应的目标文件。例如:

[root@haicoder demo]# gcc -c demo.s [root@haicoder demo]# ls demo.c demo.i demo.o demo.s

可以看到,该指令生成了和 demo.s 同名但后缀名为 .o 的文件,这就是经过汇编操作得到的目标文件。当然如果必要的话,还可以为 gcc -c 指令在添加一个 -o 选项,用于将汇编操作的结果输入到指定文件中,例如:

[root@haicoder demo]# gcc -c demo.s -o test.o [root@haicoder demo]# ls demo.c demo.i demo.o demo.s test.o

需要强调的一点是,和 gcc -S 类似,gcc -c 选项并非只能用于加工 .s 文件。事实上,-c 选项只是令 GCC 编译器将指定文件加工至汇编阶段,但不执行链接操作。这也就意味着:

  • 如果指定文件为源程序文件(例如 demo.c),则 gcc -c 指令会对 demo.c 文件执行预处理、编译以及汇编这 3 步操作;
  • 如果指定文件为刚刚经过预处理后的文件(例如 demo.i),则 gcc -c 指令对 demo.i 文件执行编译和汇编这 2 步操作;
  • 如果指定文件为刚刚经过编译后的文件(例如 demo.s),则 gcc -c 指令只对 demo.s 文件执行汇编这 1 步操作。

注意,如果指定文件已经经过汇编,或者 GCC 编译器无法识别,则 gcc -c 指令不做任何操作。这里以 demo.c、demo.i、demo.s 为例,演示 gcc -c 指令的作用:

[root@haicoder demo]# ls demo.c demo.i demo.s [root@haicoder demo]# gcc -c demo.c -o democ.o [root@haicoder demo]# ls demo.c demo.i democ.o demo.s [root@haicoder demo]# gcc -c demo.i -o demoi.o [root@haicoder demo]# ls demo.c demo.i demoi.o democ.o demo.s [root@haicoder demo]# gcc -c demo.s -o demos.o [root@haicoder demo]# ls demo.c demo.i demoi.o democ.o demo.s demos.o

以上操作分别生成的 democ.o、demoi.o 以及 demos.o,其包含的二进制内容是完全一样的,只是文件名不同而已。

gcc执行链接操作

得到生成目标文件之后,接下来就可以直接使用 gcc 指令继续执行链接操作,例如:

[root@haicoder demo]# gcc democ.o -o democ.exe [root@haicoder demo]# ./democ.exe Hello,World! [root@haicoder demo]# gcc demoi.o -o demoi.exe [root@haicoder demo]# ./demoi.exe Hello,World! [root@haicoder demo]# gcc demos.o -o demos.exe [root@haicoder demo]# ./demos.exe Hello,World!

gcc 会根据所给文件的后缀名 .o,自行判断出此类文件为目标文件,仅需要进行链接操作,所以这里的 gcc 指令只会对 democ.o、demoi.o、demos.i 执行链接操作,并分别生成 democ.exe、demoi.exe 以及 demos.exe 这 3 个可执行文件。

通过分别执行这 3 个可执行文件,其执行结果完全相同,从侧面验证了 democ.o、demo.i.o、demos.o 包含二进制的内容相同。