该文章内容大致源于韦东山老师的课件,以下是本人自己的学习笔记
一个c/c++文件大致经过以下几个过程才能变为可执行文件。
完整的编译命令为
gcc main.c -o led
表示将main.c经过上面几个步骤编译为led可执行文件
常用选项
-E 预处理,开发过程中想快速确定某个宏可以使用“-E -dM”
-c 把预处理、编译、汇编都做了,但是不链接
-o 指定输出文件
-I 指定头文件目录
-L 指定链接时库文件目录
-l 指定链接哪一个库文件
-v 显示制作GCC工具自身时的配置命令;同时显示编译器驱动程序、预处理器、编译器的版本号
1、预处理
C/C++源文件中,以“#”开头的命令被称为预处理命令,如包含命令“#include”、宏定义命令“#define”、条件编译命令“#if”、“#ifdef”等。预处理就是将要包含(include)的文件插入原文件中、将宏定义展开、根据条件编译命令选择要使用的代码,最后将这些东西输出到一个“.i”文件中等待进一步处理。
命令行示例:gcc -E main.c -o main.i 或 gcc -E main.c
表示将main.c进行预处理后的结果存为main.i
2、编译
编译就是把C/C++代码(比如上述的“.i”文件)“翻译”成汇编代码,所用到的工具为cc1(它的名字就是cc1,x86有自己的cc1命令,ARM板也有自己的cc1命令)。
命令行示例:gcc -S main.i -o main.s 或 gcc -S -o main.s main.i
表示将预处理后的main.i进行编译后的结果输出为main.s
3、汇编
汇编就是将第二步输出的汇编代码翻译成符合一定格式的机器代码,在Linux系统上一般表现为ELF目标文件(OBJ文件),用到的工具为as。x86有自己的as命令,ARM版也有自己的as命令,也可能是xxxx-as(比如arm-linux-as)。“反汇编”是指将机器代码转换为汇编代码,这在调试程序时常常用到。
汇编用到的命令行选项是-c,下面对该选项进行说明:
预处理、编译和汇编源文件,但是不作链接,编译器根据源文件生成OBJ文件。缺省情况下,GCC通过用’.o’替换源文件名的后缀’.c’,’.i’,’.s’等,产生OBJ文件名。可以使用-o选项选择其他名字。GCC忽略-c选项后面任何无法识别的输入文件。
命令行示例:gcc -c main.s -o main.o 或 gcc -c -o main.o main.s
表示将编译后的main.s进行汇编后的结果输出为main.o
4、链接
链接就是将上步生成的OBJ文件和系统库的OBJ文件、库文件链接起来,这些附加文件包括动态链接库(通常以.so结尾)和静态链接库(通常以.a结尾),最终生成了可以在特定平台运行的可执行文件,用到的工具为ld或collect2。
命令行示例:gcc -o led main.o 或 gcc main.o -o led
表示将汇编后的main.o和所有附加的目标文件链接起来生成可执行文件输出程序为led
gcc命令在执行这些命令行的时候,其实就是把这些命令进行展开去执行cpp,ccl,collect2等内部命令。
5、编译多个文件
① 一起编译、链接:
gcc -o test main.c sub.c
② 分开编译,统一链接:
gcc -c -o main.o main.c
gcc -c -o sub.o sub.c
gcc -o test main.o sub.o
6、制作并使用动态库
制作、编译:
gcc -c -o main.o main.c
gcc -c -o sub.o sub.c
gcc -shared -o libsub.so sub.o sub2.o sub3.o(可以使用多个.o生成动态库)
gcc -o test main.o -lsub -L /libsub.so/所在目录/
运行:
① 先把libsub.so放到Ubuntu的/lib目录,然后就可以运行test程序。
② 如果不想把libsub.so放到/lib,也可以放在某个目录比如/a,然后如下执行:
export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/a
./test
7、制作并使用静态库
gcc -c -o main.o main.c
gcc -c -o sub.o sub.c
ar crs libsub.a sub.o sub2.o sub3.o(可以使用多个.o生成静态库)
gcc -o test main.o libsub.a (如果.a不在当前目录下,需要指定它的绝对或相对路径)
运行:
不需要把静态库libsub.a放到板子上。
注意:执行arm-linux-gnueabihf-gcc -c -o sub.o sub.c交叉编译需要在最后面加上 -fPIC参数
Linux里的库文件分为动态链接库(通常以.so结尾)和静态链接库(通常以.a结尾),二者的区别为程序执行时所需的代码是在运行时动态加载的,还是静态加载的。默认情况下, GCC在链接时优先使用动态链接库,只有当动态链接库不存在时才考虑使用静态链接库,如果需要的话可以在编译时加上-static选项,强制使用静态链接库,编出来的可执行程序大小相差很大。
8、其他一些有用的选项
gcc -E main.c // 查看预处理结果,比如头文件是哪个
gcc -E -dM main.c > 1.txt // 把所有的宏展开,存在1.txt里
gcc -Wp,-MD,abc.dep -c -o main.o main.c // 生成依赖文件abc.dep,Makefile会用
echo ‘main(){}’| gcc -E -v - // 它会列出头文件目录、库目录(LIBRARY_PATH)
韦老师的视频还有配套的书:《嵌入式Linux应用开发完全手册 升级版》,干货满满,有需要的可以自行去官网搜索。