gcc简介和命令行参数说明

Posted 清水寺扫地僧

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了gcc简介和命令行参数说明相关的知识,希望对你有一定的参考价值。


更多的相关资料见: GCC常用参数详解

其中链接部分的数据段合并(对只读数据进行合并,节省内存,如右下图)和地址回填(对应符号解析和地址重定位两部分,详见:链接器、链接过程及相关概念解析——3. 链接器的任务)。

gcc常用参数
数据合并示意图
栈向低地址使用,堆向高地址分配
中间是共享库位置

对于右上的图,详见:异常控制流(Exception Control Flow)——2.2 私有地址空间(虚拟内存)


(一) gcc 基本用法

使用gcc编译器时,必须给出一系列必要的调用参数和文件名称。不同参数的先后顺序对执行结果没有影响,只有在使用同类参数时的先后顺序才需要考虑。 如果使用了多个 -L 的参数来定义库目录,gcc会根据多个 -L 参数的先后顺序来执行相应的库目录。

因为很多gcc参数都由多个字母组成,所以gcc参数不支持单字母的组合,Linux中常被叫短参数(short options),如 -dr 与 -d -r 的含义不一样。gcc编译器的调用参数大约有100多个,其中多数参数我们可能根本就用不到,这里只介绍其中最基本、最常用的参数。

gcc最基本的用法是:gcc [options] [filenames]

其中,options就是编译器所需要的参数,filenames给出相关的文件名称,最常用的有以下参数

参数说明
− c -c c只编译,不链接成为可执行文件。 编译器只是由输入的 .c 等源代码文件生成 .o 为后缀的目标文件,通常用于编译不包含主程序的子程序文件。
− o o u t p u t _ f i l e n a m e -o \\quad output\\_filename ooutput_filename确定输出文件的名称为-o output_filename。 同时这个名称不能和源文件同名。如果不给出这个选项,gcc就给出默认的可执行文件 a.out 。
− g -g g产生符号调试工具(GNU的 gdb)所必要的符号信息。 想要对源代码进行调试,就必须加入这个选项。
− O -O O对程序进行优化编译、链接。 采用这个选项,整个源代码会在编译、链接过程中进行优化处理,这样产生的可执行文件的执行效率可以提高,但是编译、链接的速度就相应地要慢一些,而且对执行文件的调试会产生一定的影响,造成一些执行效果与对应源文件代码不一致等一些令人“困惑”的情况。因此,一般在编译输出软件发行版时使用此选项。
− O 2 -O2 O2比 -O 更好的优化编译、链接。当然整个编译链接过程会更慢。
− I d i r -Idir Idir即 Include dir,将 dirname 所指出的目录加入到程序头文件目录列表中,是在预编译过程中使用的参数。

说明:C程序中的头文件包含两种情况:
#include <stdio.h>
#include "stdio.h"
其中,使用尖括号(<>),预处理程序 cpp 在系统默认包含文件目录(如/usr/include)中搜索相应的文件;使用双引号,预处理程序 cpp 首先在当前目录中搜寻头文件,如果没有找到,就到指定的 dirname 目录中去寻找。在程序设计中,如果需要的这种包含文件分别分布在不同的目录中,就需要逐个使用 -I 选项给出搜索路径。
− L d i r -Ldir Ldir即 Link dir,将dirname所指出的目录加入到程序函数库文件的目录列表中,是在链接过程中使用的参数。 在默认状态下,链接程序 ld 在系统默认路径中(如 /usr/lib)寻找所需要的库文件。这个选项告诉链接程序,首先到 -L 指定的目录中去寻找,然后到系统默认路径中寻找;如果函数库存放在多个目录下,就需要依次使用这个选项,给出相应的存放目录。
− l n a m e -lname lname链接时装载名为 libname.a 的函数库。 该函数库位于系统默认的目录或者由 -L 选项确定的目录下。 Linux下的库文件在命名时有一个约定,就是应该以 lib 这3个字母开头,由于所有的库文件都遵循了同样的规范,因此在用 -l 选项指定链接的库文件名时可以省去 lib 这3个字母。例如,gcc 在对 -lfoo 进行处理时,会自动去链接名为 libfoo.so 的文件,又如,-lm 表示链接名为 libm.a 的数学函数库。

gcc使用案例:

假定有一个程序名为 test.c 的C语言源代码文件,要生成一个可执行文件。

#include <stdio.h>
int main(void)
{
    printf("Hello world/n");
    return 0;
}

最简单的办法:gcc test.c -o test

首先给出gcc将源代码转换为机器码的过程(以C语言为例)。默认情况下,预编译、编译链接一次完成:
在这里插入图片描述
相关内容可见:计算机系统漫游:1. 程序由源文件转换至机器码的过程

编译过程的分步执行:
为了更好地理解gcc的工作过程,我们可以让在gcc工作的4个阶段中的任何一个阶段中停止下来。相关的参数有:

  • ① -E:预编译后停下来,生成后缀为 .i 的预编译文件。gcc -E test.c -o test.i,查看 test.i 文件中的内容,会发现 stdio.h 的内容确实都插到文件里去了,而其他应当被预处理的宏定义也都做了相应的处理;
  • ② -S:汇编后停下来,生成后缀为 .s 的汇编源文件。生成汇编源文件,gcc -S test.c -o test.s
  • ③ -c:编译后停下来,生成后缀为 .o 的目标文件。gcc -c test.c -o test.o
  • ④ -o:将生成的目标文件链接成可执行文件,gcc test.o - o test

对于稍微复杂的情况,比如有多个源代码文件、需要链接库或有其他比较特别的要求,就要给定适当的调用选项参数。

例子:整个源代码程序由两个文件 testmain.c 和 testsub.c 组成,程序中使用了系统提供的数学库(所有与浮点相关的数学运算都必须使用数学库)。gcc testmain.c testsub.c -lm -o test。其中,-lm 表示链接系统的数学库 libm.a


说明: 在编译一个包含许多源文件的工程时,若只用一条gcc命令来完成编译是非常浪费时间的。假如项目中有100个源文件需要编译,并且每个源文件中都包含一万行代码,如果像上面那样仅用一条gcc命令来完成编译工作,那么gcc需要将每个源文件都重新编译一遍,然后再全部链接起来。很显然,这样浪费的时间相当多,尤其是当用户只是修改了其中某个文件的时候,完全没有必要将每个文件都重新编译一遍,因为很多已经生成的目标文件是不会发生改变的。要解决这个问题,需要借助像make这样的工具。



(二) 警告提示功能选项

gcc包含完整的出错检查和警告提示功能,它们可以帮助Linux程序员写出更加专业的代码。

参数说明
− v -v v输出 gcc 工作的详细过程
–target-help显示目前所用的gcc支持CPU类型
− Q -Q Q显示编译过程的统计数据和每一个函数名
− p e d a n t i c -pedantic pedantic当gcc在编译不符合ANSI/ISO C 语言标准的源代码时,将产生相应的警告信息。
− W a l l -Wall Wall
warning all
除了 -pedantic 之外,gcc 还有一些其他编译选项,也能够产生有用的警告信息。这些选项大多以 -W 开头。其中最有价值的当数 -Wall 了,使用它能够使 gcc 产生尽可能多的警告信息。
− W e r r o r -Werror Werror它要求 gcc 将所有的警告当成错误进行处理,这在使用自动编译工具(如 Make 等)时非常有用。如果编译时带上 -Werror 选项,那么 gcc 会在所有产生警告的地方停止编译,迫使程序员对自己的代码进行修改。只有当相应的警告信息消除时,才可能将编译过程继续朝前推进。
− W c a s t − a l i g n -Wcast-align Wcastalign当源程序中地址不需要对齐的指针指向一个地址需要对齐的变量地址时,则产生一个警告。例如,char * 指向一个 int * 地址,而通常在机器中 int 变量类型是需要地址能被2或4整除的对齐地址。

对于如下代码,使用-pedantic参数与不使用有下图的区别:

#include <stdio.h>  
 void main(void)  
 {  
     long long int var = 1;  
     printf("It is not standard C code!/n");  
 }  
/*它有以下问题:
> main 函数的返回值被声明为 void,但实际上应该是 int。
> 使用了 GNU 语法扩展,即使用 long long 来声明64位整数,不符合 ANSI/ISO C 语言标准。
> main 函数在终止前没有调用 return 语句。*/

此外还有-Wall-Werror参数的使用效果也见下图:

-pedantic 命令行参数效果
-Wall 命令行参数效果
-Werror 命令行参数效果

建议:gcc 给出的警告信息虽然从严格意义上说不能算作错误,但却和可能成为错误来源。一个优秀的程序员应该尽量避免产生警告信息,使自己的代码始终保持简洁、优美和健壮的特性。gcc 给出的警告信息是很有价值的,它们不仅可以帮助程序员写出更加健壮的程序,而且还是跟踪和调试程序的有力工具。建议在用 gcc 编译源代码时始终带上 -Wall 选项,并把它逐渐培养成一种习惯,这对找出常见的隐式编程错误很有帮助。



(三) 库操作选项

在Linux下开发软件时,完全不使用第三方函数库的情况是比较少见的,通常来讲都需要借助一个或多个函数库的支持才能够完成相应的功能。

从程序员的角度看,函数库实际上就是一些头文件(.h)和库文件(.so 或 .a)的集合。虽然Linux下的大多数函数都默认将头文件放到 /usr/include/ 目录下,而库文件则放到 /usr/lib/ 目录下,但并不是所有的情况都是这样。正因如此,gcc 在编译时必须有自己的办法来查找所需要的头文件和库文件。常用的方法有:

参数说明
− I -I I可以向 gcc 的头文件搜索路径中添加新的目录。同(一)gcc基本用法中的-Idir命令行参数
− L -L L如果使用了不在标准位置的库文件,那么可以通过 -L 选项向 gcc 的库文件搜索路径中添加新的目录。同(一)gcc基本用法中的-Ldir命令行参数
− l -l lLinux下的库文件在命名时有一个约定,就是应该以 lib 这3个字母开头,由于所有的库文件都遵循了同样的规范,因此在用 -l 选项指定链接的库文件名时可以省去 lib 这3个字母。同(一)gcc基本用法中的-lname命令行参数。例如,gcc 在对 -lfoo 进行处理时,会自动去链接名为 libfoo.so 的文件。
− s t a t i c -static staticLinux下的库文件分为两大类,分别是:动态链接库(通常以 .so 结尾)和静态链接库(通常以 .a 结尾)。两者的差别仅在程序执行时所需的代码是在运行时动态加载的,还是在编译时静态加载的。默认情况下,gcc 在链接时优先使用动态链接库,只有当动态链接库不存在时才考虑使用静态链接库。如果需要的话,可以在编译时加上 -static 选项,强制使用静态链接库
− s h a r e d -shared shared生成一个共享的目标文件,它能够与其他的目标一起链接生成一个可执行的文件。


(四) 调试选项

对于Linux程序员来讲,gdb(GNU Debugger)通过与 gcc 的配合使用,为基于Linux的软件开发提供了一个完善的调试环境。常用的有:

参数说明
− g \\quad-g g
\\quad
− g g d b -ggdb ggdb
默认情况下,gcc 在编译时不会将调试符号插入到生成的二进制代码中,因为这样会增加可执行文件的大小。如果需要在编译时生成调试符号信息,可以使用 gcc 的 -g 或 -ggdb 选项。

gcc 在产生调试符号时,同样采用了分级的思路,开发人员可以通过在 -g 选项后附加数字1、2、3指定在代码中加入调试信息的多少:
级别2(-g2),默认的级别。此时产生的调试信息包括:扩展的符号表、行号、局部或外部变量信息;
级别3(-g3)包含级别2中的所有调试信息以及源代码中定义的宏;
级别1(-g1)不包含局部变量和与行号有关的调试信息,因此只能够用于回溯跟踪和堆栈转储。
回溯追踪:指的是监视程序在运行过程中函数调用历史。
堆栈转储:则是一种以原始的十六进制格式保存程序执行环境的方法。
− p \\quad-p p
\\quad
− p g \\quad-pg pg
会将剖析(Profiling)信息加入到最终生成的二进制代码中。剖析信息对于找出程序的性能瓶颈很有帮助,是协助Linux程序员开发出高性能程序的有力工具。
− s a v e − t e m p s -save-temps savetemps保存编译过程中生成的一些列中间文件。gcc test.c -o test -save-temps。除了生成执行文件test之外,还保存了test.i 和 test.s 中间文件,供用户查询调试。

注意:使用任何一个调试选项都会使最终生成的二进制文件的大小急剧增加,同时增加程序在执行时的开销,因此,调试选项通常仅在软件的开发和调试阶段使用。



(五) 交叉编译选项

通常情况下使用 gcc 编译的目标代码都与使用的机器是一致的,但 gcc 也支持交叉编译的功能,能够编译其他不同CPU的目标代码

使用 gcc 开发嵌入式系统,我们几乎都是以通用的PC机(X86)平台来做宿主机,通过 gcc 的交叉编译功能对其他嵌入式CPU的开发任务。

(具体的选项设置,此处省略)



(六) gcc 和 g++ 比较

gcc 和 g++现在是GNU中最主要和最流行的 c/c++编译器。


gcc/g++在执行编译工作的时候,总共需要以下几步:

  • ①预处理,生成.i的文件[预处理器cpp];
  • ②将预处理后的文件不转换成汇编语言,生成文件.s[编译器egcs];
  • ③有汇编变为目标代码(机器代码)生成.o的文件[汇编器as];
  • ④连接目标代码,生成可执行程序[链接器ld];

实际过程与上边的流程图并无二致。


gcc能够处理的后缀有:

  • *.c *.C (C语言)
  • *.cxx *.cc (C++语言)
  • *.m (面向对象的C,objective C?)
  • *.i (预处理后的C语言源文件)
  • *.ii (预处理后的C++语言源文件)
  • *.s *.S (汇编语言)
  • *.h (头文件)

目标文件可以是:

  • *.o 编译连接后的目标文件
  • *.a 库文件

gcc与g++有什么区别?

gcc和g++都是GNU(组织)的一个编译器。

误区答案
gcc只能编译c代码,g++只能编译c++代码两者都可以,但是请注意:
1.后缀为.c的,gcc把它当作是C程序,而g++当作是c++程序;后缀为.cpp的,两者都会认为是c++程序,注意,虽然c++是c的超集,但是两者对语法的要求是有区别的。C++的语法规则更加严谨一些。
2. 编译阶段,g++会调用gcc,对于c++代码,两者是等价的,但是因为gcc命令不能自动和C++程序使用的库联接,所以通常用g++来完成链接,为了统一起见,干脆编译/链接统统用g++了,这就给人一种错觉,好像cpp程序只能用g++似的。
gcc不会定义__cplusplus宏,而g++会实际上,这个宏只是标志着编译器将会把代码按C还是C++语法来解释,如上所述,如果后缀为.c,并且采用gcc编译器,则该宏就是未定义的,否则,就是已定义。
编译只能用gcc,链接只能用g++严格来说,这句话不算错误,但是它混淆了概念,应该这样说:编译可以用gcc/g++,而链接可以用g++或者gcc -lstdc++。因为gcc命令不能自动和C++程序使用的库联接,所以通常使用g++来完成联接。但在编译阶段,g++会自动调用gcc,二者等价。

以上是关于gcc简介和命令行参数说明的主要内容,如果未能解决你的问题,请参考以下文章

gcc命令行选项说明

GCC 命令行详解 -L 指定库的路径 -l 指定需连接的库名

NASM汇编学习系列——获取命令行参数

用C语言实现Linux命令——模拟gcc

gcc编译程序时的命令行参数-I(大写i) -L -l (小写L) 2020-10-10

GCC 和 NASM 链接到外部库