`
fujinbing
  • 浏览: 238925 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

gcc 从语言编译全过程 预处理---->编译---->汇编----->链接

阅读更多
在Linux下进行C语言编程,必然要采用GNU GCC来编译C源代码生成可执行程序。

一、GCC快速入门

Gcc指令的一般格式为:Gcc [选项] 要编译的文件 [选项] [目标文件]

其中,目标文件可缺省,Gcc默认生成可执行的文件名为:编译文件.out

我们来看一下经典入门程序"Hello World!"

# vi hello.c

#include <stdlib.h>
#include <stdio.h>
void main(void)
{
printf("hello world!\r\n");
}

用gcc编译成执行程序。

#gcc hello.c

该命令将hello.c直接生成最终二进制可执行程序a.out

这条命令隐含执行了(1)预处理、(2)汇编、(3)编译并(4)链接形成最终的二进制可执行程序。这里未指定输出文件,默认输出为a.out。

如何要指定最终二进制可执行程序名,那么用-o选项来指定名称。比如需要生成执行程序hello.exe

那么

#gcc hello.c -o hello.exe

二、GCC的命令剖析--四步走

从上面我们知道GCC编译源代码生成最终可执行的二进制程序,GCC后台隐含执行了四个阶段步骤。

GCC编译C源码有四个步骤:

预处理-----> 编译 ----> 汇编 ----> 链接

现在我们就用GCC的命令选项来逐个剖析GCC过程。

1)预处理(Pre-processing)

在该阶段,编译器将C源代码中的包含的头文件如stdio.h编译进来,用户可以使用gcc的选项”-E”进行查看。

用法:#gcc -E hello.c -o hello.i

作用:将hello.c预处理输出hello.i文件。

[root]# gcc -E hello.c -o hello.i
[root]# ls
hello.c  hello.i
[root]# vi hello.i
# 1 "hello.c"
# 1 "<built-in>"
# 1 "<command line>"
# 1 "hello.c"
# 1 "/usr/include/stdlib.h" 1 3
# 25 "/usr/include/stdlib.h" 3
# 1 "/usr/include/features.h" 1 3
# 291 "/usr/include/features.h" 3
# 1 "/usr/include/sys/cdefs.h" 1 3
# 292 "/usr/include/features.h" 2 3
# 314 "/usr/include/features.h" 3
# 1 "/usr/include/gnu/stubs.h" 1 3
# 315 "/usr/include/features.h" 2 3
# 26 "/usr/include/stdlib.h" 2 3
# 3 "hello.c" 2
void main(void)
{
printf("hello world!\r\n");
}


2)编译阶段(Compiling)

第二步进行的是编译阶段,在这个阶段中,Gcc首先要检查代码的规范性、是否有语法错误等,以确定代码的实际要做的工作,在检查无误后,Gcc把代码翻译成汇编语言。用户可以使用”-S”选项来进行查看,该选项只进行编译而不进行汇编,生成汇编代码。

选项 -S

用法:[root]# gcc –S hello.i –o hello.s

作用:将预处理输出文件hello.i汇编成hello.s文件。

[root@richard hello-gcc]# ls

hello.c  hello.i  hello.s

如下为hello.s汇编代码

[root@richard hello-gcc]# vi hello.s
.file   "hello.c"
.section    .rodata
.LC0:
.string "hello world!\r\n"
.text
.globl main
.type   main,@function
main:
pushl   %ebp
movl    %esp, %ebp
subl    $8, %esp
andl    $-16, %esp
movl    $0, %eax
subl    %eax, %esp
subl    $12, %esp
pushl   $.LC0
call    printf
addl    $16, %esp
movl    $0, %eax
leave
ret
.Lfe1:
.size   main,.Lfe1-main
.ident  "GCC: (GNU) 3.2.2 20030222 (Red Hat Linux 3.2.2-5)"

3)汇编阶段(Assembling)

汇编阶段是把编译阶段生成的”.s”文件转成二进制目标代码.

选项 -c

用法:[root]# gcc –c hello.s –o hello.o

作用:将汇编输出文件test.s编译输出test.o文件。

[root]# gcc -c hello.s -o hello.o

[root]# ls

hello.c  hello.i  hello.o  hello.s


4)链接阶段(Link)

在成功编译之后,就进入了链接阶段。

无选项链接

用法:[root]# gcc hello.o –o hello.exe

作用:将编译输出文件hello.o链接成最终可执行文件hello.exe。

[root]# ls

hello.c  hello.exe  hello.i  hello.o  hello.s

运行该可执行文件,出现正确的结果如下。

[root@localhost Gcc]# ./hello

Hello World!

在这里涉及到一个重要的概念:函数库。

读者可以重新查看这个小程序,在这个程序中并没有定义”printf”的函数实现,且在预编译中包含进的”stdio.h”中也只有该函数的声明,而没有定义函数的实现,那么,是在哪里实现”printf”函数的呢?最后的答案是:系统把这些函数实现都被做到名为libc.so.6的库文件中去了,在没有特别指定时,gcc会到系统默认的搜索路径”/usr/lib”下进行查找,也就是链接到libc.so.6库函数中去,这样就能实现函数”printf” 了,而这也就是链接的作用。

你可以用ldd命令查看动态库加载情况:

[root]# ldd hello.exe

libc.so.6 => /lib/tls/libc.so.6 (0x42000000)

/lib/ld-linux.so.2 => /lib/ld-linux.so.2 (0x40000000)

函数库一般分为静态库和动态库两种。静态库是指编译链接时,把库文件的代码全部加入到可执行文件中,因此生成的文件比较大,但在运行时也就不再需要库文件了。其后缀名一般为”.a”。动态库与之相反,在编译链接时并没有把库文件的代码加入到可执行文件中,而是在程序执行时由运行时链接文件加载库,这样可以节省系统的开销。动态库一般后缀名为”.so”,如前面所述的libc.so.6就是动态库。gcc在编译时默认使用动态库。
分享到:
评论

相关推荐

    arm-linux-gcc编译选项.pdf

    以下是arm-linux-gcc编译选项的详细知识点说明: 1. 编译过程的四个阶段: - 预处理阶段:GCC会对源文件进行预处理,展开宏定义、处理条件编译指令、包含头文件等。 - 编译阶段:经过预处理的源文件会被转化为...

    gcc-arm-none-eabi-9-2019-q4-major-win32.zip

    标题中的"gcc-arm-none-eabi-9-2019-q4-major-win32.zip"是一个针对Windows操作系统的GCC交叉编译工具链压缩包。这个工具链是专门为ARM架构的微控制器和嵌入式系统设计的,使得开发者可以在运行Windows的个人电脑上...

    gcc-arm-none-eabi-9-2019-q4-major-x86_64-linux.tar.bz2.7z

    GCC(GNU Compiler Collection)是GNU项目的核心部分,它不仅包括C、C++、Fortran、Objective-C、Ada等编程语言的编译器,还提供了预处理、汇编、链接等工具。在本案例中,"arm-none-eabi"标识了这是一个针对ARM架构...

    gcc-core-3.4.5-20060117-3.tar.gz

    4. libexec:这个目录可能包含编译过程所需的辅助程序或者动态链接库,这些文件在编译过程中会被调用,帮助完成诸如预处理、汇编和链接等任务。 5. lib:这个目录可能包含编译器需要的库文件,比如支持不同语言特性...

    gcc-g++-4.2.1-sjlj-2.tar.gz

    3. **libexec**:通常存放辅助执行程序或库,这些文件可能在编译过程中被调用,执行一些特定的任务,比如预处理、汇编或优化。 4. **lib**:这个目录下的库文件可能是GCC和G++运行时所需的依赖。在编译和链接阶段,...

    linux中gcc4.8.5,下载解压即可直接使用,linux系统GCC编译

    Linux中的GCC(GNU Compiler Collection)是开源的、跨平台的编译器套件,用于将C、C++、Fortran、Objective-C等编程语言的源代码编译为可执行文件。GCC 4.8.5是该系列的一个稳定版本,发布于2015年,虽然不是最新版...

    GCC-----------中文手册

    预处理处理宏定义、条件编译等,编译将预处理后的文本转换成汇编代码,汇编将汇编代码转化为机器码,最后链接将多个目标文件和库合并成可执行文件。GCC通过源文件的后缀名来判断文件的语言类型,并决定后续处理方式...

    gcc-4.6.2-glibc-2.13-linaro-multilib-2011.12.tar.gz

    交叉编译的基本流程包括:预处理、编译、汇编和链接四个步骤。在这个过程中,编译器会根据指定的目标体系结构生成相应的机器码,而链接器则会结合各种静态和动态库,生成最终的可执行文件。 总之,“gcc-4.6.2-...

    gcc编译数据库1

    它通过一系列命令行参数来控制编译过程的不同方面,包括预处理、编译、汇编和链接等步骤。 ### GCC编译数据库的概念 GCC编译数据库并不是一个传统意义上的数据库,而是一种存储项目编译指令的数据结构或文件格式,...

    gcc-4.9.2-Ee500v2-eabispe

    4. `as`:汇编器,将预处理后的代码转换为机器语言。 5. `ld`:链接器,将编译后的对象文件合并成一个可执行文件或库。 6. 开发头文件和库:提供了Ee500v2平台上的系统调用、硬件接口和其他功能的接口。 在实际开发...

    gcc-linaro-6.3.1-2017.05-x86-64-aarch64-linux-gnu.7z

    5. 预处理器和汇编器:`cpp`和`as`,分别处理预处理指令和汇编代码。 6. 工具和脚本:用于辅助编译过程,如`make`,用于自动化构建过程。 在Linux环境下安装这个工具包后,开发者可以在本地x86_64系统上构建针对...

    GCC的开发编译配置

    在GCC的编译过程中,可以分为四个阶段:预处理(Pre-Processing)、编译(Compiling)、汇编(Assembling)和链接(Linking)。预处理阶段处理#include指令、宏定义和条件编译;编译阶段将预处理后的文本转换成汇编...

    gcc-c++-4.8.5-36.el7.x86_64_rpm.zip

    编译过程分为预处理、编译、汇编和链接四个步骤。预处理阶段处理宏定义、条件编译指令和包含文件;编译阶段将预处理后的代码转化为汇编语言;汇编阶段将汇编语言转换为机器码;链接阶段则将各个编译生成的目标文件...

    arm-fsl-linux-gnueabi-交叉编译工具链

    2. **编译器** (cc/g++): 编译器如gcc或g++,将预处理过的C/C++源代码转化为汇编代码。 3. **汇编器** (as): 汇编器将汇编代码转化为机器语言的二进制目标文件。 4. **链接器** (ld): 链接器负责合并多个目标文件...

    gcc编译过程概述.pdf

    ### GCC编译过程详解 #### 一、GCC简介与作用 GCC(GNU Compiler Collection)是GNU发布的一套编程语言编译器,它支持多种编程语言,如C、C++、Objective-C、Fortran等,并且可以针对不同的处理器架构生成优化过的...

    gcc g++ 中文编译选项详解 手册

    GCC 和 G++ 编译器的编译过程可以分为四个步骤:预处理、编译、汇编和链接。预处理阶段将源代码转换为中间代码,编译阶段将中间代码转换为汇编代码,汇编阶段将汇编代码转换为机器代码,链接阶段将机器代码链接成可...

    linux Gcc 编译详解

    本文将详细讲解Linux下GCC的编译过程,包括预处理、编译、汇编和链接四个阶段,并探讨相关的编译选项和优化原理。 1. 预处理 预处理是GCC处理C源代码的第一步,由cpp程序负责。在这个阶段,GCC执行以下操作: - ...

    GCC的编译流程

    预处理是GCC编译过程中最初始的阶段。在这个阶段,GCC会处理以`#`字符开头的预处理器指令。预处理器的主要任务包括: - **宏替换**:对宏进行展开,例如`#define`定义的宏会被相应的文本替换。 - **文件包含**:...

    gcc-g77-3.4.5-20060117-3.tar.gz

    G77的编译过程通常涉及预处理、编译、汇编和链接四个步骤,通过g77命令行工具可以完成这些操作。用户需要编写Fortran 77源代码,然后使用g77进行编译,生成可执行文件。 在编程过程中,G77支持Fortran 77标准的特性...

Global site tag (gtag.js) - Google Analytics