美文网首页gcc编译器入门
GCC编译过程和原理浅析

GCC编译过程和原理浅析

作者: 王泽0106 | 来源:发表于2020-10-22 12:20 被阅读0次

什么是GCC

  • GCC(GNU C Compiler)编译器的作者是Richard Stallman,也是GNU项目的奠基者。
  • GCC是GNU Compiler Collection的缩写。最初是作为C语言的编译器,现在已经支持多种语言了,如C、C++、Java、Pascal、Ada、COBOL语言等。
  • GCC支持多种硬件平台,甚至对Don Knuth设计的MMIX这类不常见的计算机都提供了完善的支持。

GCC的主要特征

  • GCC是一个可移植的编译器,支持多种硬件平台。
  • GCC不仅仅是本地编译器,它还能跨平台交叉编译。
  • GCC有多种语言前段,用于解析不同的语言。
  • GCC是按模块化设计的,可以加入新的语言和新CPU架构的支持。
  • GCC是自由软件。

GCC编译程序的过程

例如使用一个hello.c文件编译的过程如下图所示: 编译过程
  1. 预处理(Pre-Processing):主要包括宏定义,文件包含,条件编译三部分。预处理过程读入源代码,检查包含预处理指令的语句和宏定义,并对其进行响应和替换。预处理过程还会删除程序中的注释和多余空白字符。最后会生成 .i 文件。
  2. 编译器(Compiling):编译器会将预处理完的 .i 文件进行一些列的语法分析,并优化后生成对应的汇编代码。会生成 .s 文件。
  3. 汇编器(Assembling):汇编器会将编译器生成的 .s 汇编程序汇编为机器语言或指令,也就是可以机器可以执行的二进制程序。会生成 .o 文件。
  4. 链接器(Linking):链接器会来链接程序运行的所需要的目标文件,以及依赖的库文件,最后生成可执行文件,以二进制形式存储在磁盘中。

GCC的编译过程

GCC常用选项

  • -o:生成目标( .i 、 .s 、 .o 、可执行文件等)
  • -c:通知 gcc 取消链接步骤,即编译源码并在最后生成目标文件。
  • -E:只运行 C 预编译器
  • -S:告诉编译器产生汇编语言文件后停止编译,产生的汇编语言文件扩展名为 .s
  • -Wall:使 gcc 对源文件的代码有问题的地方发出警告
    • Idir:将dir目录加入搜索头文件的目录路径
  • -Ldir:将dir目录加入搜索库的目录路径
  • -llib:连接lib库
  • -g:在目标文件中嵌入调试信息,以便gdb之类的调试程序调试

预处理过程

我们以 hello.c 程序为例:

#include <stdio.h>

#define HELLOWORLD ("hello world\n")

int main(void)
{
    printf(HELLOWORLD); 
    return 0;
}

使用gcc -E hello.c -o hello.i命令,将 hello.c 文件预处理并且生成 hello.i 目标文件。

之前说道,预处理会将头文件包含进来并且会将宏定义进行替换,因此替换后的 hello.i 文件如下:

# 1 "hello.c"
# 1 "<built-in>"
# 1 "<命令行>"
# 1 "hello.c"
# 1 "/usr/include/stdio.h" 1 3 4
# 28 "/usr/include/stdio.h" 3 4
# 1 "/usr/include/features.h" 1 3 4
# 361 "/usr/include/features.h" 3 4
# 1 "/usr/include/sys/cdefs.h" 1 3 4
# 365 "/usr/include/sys/cdefs.h" 3 4
# 1 "/usr/include/bits/wordsize.h" 1 3 4
# 366 "/usr/include/sys/cdefs.h" 2 3 4
# 362 "/usr/include/features.h" 2 3 4
# 385 "/usr/include/features.h" 3 4
# 1 "/usr/include/gnu/stubs.h" 1 3 4

......

extern void funlockfile (FILE *__stream) __attribute__ ((__nothrow__));
# 938 "/usr/include/stdio.h" 3 4

# 2 "hello.c" 2


int main()
{
printf(("hello world!\n"));
return 0;
}

可以看到将 stdio.h 文件包含进来,并且原封不动的将 HELLOWORLD 宏进行了替换。

编译过程

使用gcc -S hello.i -o hello.s,将生成的hello.i文件编译为汇编程序hello.s。

    .file   "hello.c"
    .section    .rodata
.LC0:
    .string "hello world!"
    .text
.globl main
    .type   main, @function
main:
    pushl   %ebp
    movl    %esp, %ebp
    andl    $-16, %esp
    subl    $16, %esp
    movl    $.LC0, (%esp)
    call    puts
    movl    $0, %eax
    leave
    ret
    .size   main, .-main
    .ident  "GCC: (GNU) 4.4.7 20120313 (Red Hat 4.4.7-23)"
    .section    .note.GNU-stack,"",@progbits

可以看到hello.s文件中全部都是汇编指令,说明已经生成成功了。

汇编过程

汇编就是要将hello.s文件中的汇编指令全部转换为二进制的机器指令。
执行gcc -c hello.s -o hello.o命令。而生成的hello.o文件是二进制文件,我们用od -b hello.o命令看一下该二进制文件的八进制表示。

0000000 177 105 114 106 001 001 001 000 000 000 000 000 000 000 000 000
0000020 002 000 003 000 001 000 000 000 
......

链接过程

链接hello.o程序运行的所需要的目标文件,以及依赖的库文件,最后生成可执行文件。
执行gcc hello.o -o hello不需要选项,生成hello二进制的可执行文件。同样可以使用od命令来查看。执行hello文件:./hello

hello world

以上编译过程的分步骤进行,还可以直接执行gcc hello.c -o hello直接生成可执行文件。
至此,使用gcc编译程序的过程就介绍完毕。

待更新...

相关文章

  • GCC编译过程和原理浅析

    什么是GCC GCC(GNU C Compiler)编译器的作者是Richard Stallman,也是GNU项目...

  • iOS App编译和启动原理

    编译原理: 将编译过程分为前端和后端两部分处理 Clang(编译器前端):GCC的替代品,Clang的编译速度比G...

  • gcc和编译过程分解

    编译过程 预处理->编译->汇编->链接 gcc需要调用预处理程序cpp,由它负责展开在源文件中定义的宏,并向其中...

  • gcc 编译过程

    在执行上面语句时,-o实际上分多步来处理.c文件。下面我们讲一下gcc的编译过程。 预处理器处理 首先gcc使用预...

  • GCC编译过程

    第一步:进行预编译,使用 -E 参数 gcc-Etest.c-o test.i 查看 test.i 文件中的内容,...

  • make与configure

    因为gcc的编译器来进行编译过程过于复杂,所以源码编译软件包安装时会通过configure和make进行编译过程简...

  • 编译原理4个步骤

    编译原理 编译原理(把人写的代码,变成,机器可以认识的指令)Linux(gcc)Mac (Clang) 1.预处理...

  • GCC的编译过程和链接

    编译步骤 GCC 编译器在编译一个C语言程序时需要经过以下 4 步: 1. 预处理 将C语言源程序预处理,生成.i...

  • iOS静态库及动态库

    2018-06-19 浅析静态库链接原理编译原理以及什么是静态库和动态库 转之: iOS 静态库和动态库(库详解)...

  • gcc如何编译.c 文件、gcc编译过程以及编译的主要参数

    gcc如何编译.c 文件、gcc编译过程以及编译的主要参数如何编译.c 文件假如在当前目录下存在两个文件hello...

网友评论

    本文标题:GCC编译过程和原理浅析

    本文链接:https://www.haomeiwen.com/subject/ekeumktx.html