Hello基本操作
准备工作
#include <stdio.h>
int main(void) {
printf("Hello World! \n");
return 0;
}
data:image/s3,"s3://crabby-images/94a4d/94a4dcc41237f9cf491d48a8d062ecae576c79d7" alt="在这里插入图片描述" 预处理基本操作
- 将所有的#define 删除,并且展开所有的宏定义,并且处理所有的条件预编 译指令,比如#if #ifdef #elif #else #endif 等。
- 处理#include 预编译指令,将被包含的文件插入到该预编译指令的位置。
- 删除所有注释“//”和“/* */”。
- 添加行号和文件标识,以便编译时产生调试用的行号及编译错误警告行号。
- 保留所有的#pragma 编译器指令,后续编译过程需要使用它们。
data:image/s3,"s3://crabby-images/79f8f/79f8ff662017da880c8aba0e0d246926a1171204" alt="在这里插入图片描述" hello.i文件 data:image/s3,"s3://crabby-images/9b068/9b068a289000bdc06ed29b659b4a4d3da18faa7e" alt="在这里插入图片描述"
编译
将预处理生成的 hello.i 文件编译生成汇编程序 hello.s data:image/s3,"s3://crabby-images/6b450/6b450f73e826df6bf149811b0758d55edc3544bf" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/e9fd3/e9fd3df03e40a2f6cca411fab9ffd97122a5fc48" alt="在这里插入图片描述"
汇编
汇编过程调用对汇编代码进行处理,生成处理器能识别的指令,保存在后缀为.o 的目标文件中。 data:image/s3,"s3://crabby-images/35833/35833d6c3ba71e83d830dffe83dadc3de3a989bf" alt="在这里插入图片描述"
将编译生成的 hello.s 文件汇编生成目标文件 hello.o GCC 的选项-c 使 GCC 在执行完汇编后停止,生成目标文件
链接
-
静态链接: 静态链接是指在编译阶段直接把静态库加入到可执行文件中去,这样可执行 文件会比较大。 data:image/s3,"s3://crabby-images/728ac/728ac2451729989afc40c8857b0c07d20650c25b" alt="在这里插入图片描述" text链接静态库,代码尺寸变得极大 data:image/s3,"s3://crabby-images/a3249/a3249f4c08fb41259989922144e5883af52ed6a7" alt="在这里插入图片描述" 没有链接动态库 -
动态链接: 动态链接则是指链接阶段仅仅只加入一些描述信息,而程序执行时再从系统 中把相应动态库加载到内存中去。 gcc hello.c -o hello data:image/s3,"s3://crabby-images/0ed52/0ed5248da9537564052394c4c0b44383797d830c" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/ecf78/ecf78b528652c39ae6ed3948708e678df5d2186b" alt="在这里插入图片描述" 链接了许多其他的动态库,主要是linux系统glibc 动态库
ELF文件分析
链接器链接后生成的最终文件为 ELF 格式可执行文件,一个 ELF 可执行文件通常 被链接为不同的段,常见的段譬如.text、.data、.rodata、.bss 等段
.text:已编译程序的指令代码段。 .rodata:ro 代表 read only,即只读数据(譬如常数 const)。 .data:已初始化的 C 程序全局变量和静态局部变量。 .bss:未初始化的 C 程序全局变量和静态局部变量。 .debug:调试符号表,调试器用此段的信息帮助调试。 data:image/s3,"s3://crabby-images/79cc1/79cc1eb1d1cf1e099d7683cb1c3bb99288314e17" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/1a665/1a6650210d2872de2b58fc8784ec983ee86f3bc4" alt="在这里插入图片描述"
data:image/s3,"s3://crabby-images/13130/13130e25409343ae56a0a65fc57fadb00973bf46" alt="在这里插入图片描述"
反汇编ELF
由于 ELF 文件无法被当做普通文本文件打开,如果希望直接查看一个 ELF 文件包 含的指令和数据,需要使用反汇编的方法 data:image/s3,"s3://crabby-images/fbeb2/fbeb22b811b9a60fe09916da9a06ba01fde1d319" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/8aa19/8aa1965f3e7ac6a77e3f780f939789dedb03f290" alt="在这里插入图片描述" 使用 objdump -S 将其反汇编并且将其 C 语言源代码混合显示出来: data:image/s3,"s3://crabby-images/0fc03/0fc038a99b805ecdd1eb48a504dd28c6e7268cdb" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/77249/77249dcf9f719f666fde52da63b37882f515e19f" alt="在这里插入图片描述"
gcc命令基本操作
简单编译
data:image/s3,"s3://crabby-images/7bf48/7bf48a69648d185af3beab05d33363cdbeb9d09c" alt="在这里插入图片描述"
- 预处理:
gcc -E test.c -o test.i 或 gcc -E test.c gcc 的-E 选项,可以让编译器在预处理后停止,并输出预处理结果 - 编译为汇编代码
gcc -S test.i -o test.s data:image/s3,"s3://crabby-images/4db1f/4db1f39ce8ce66fe55367ad8dfba93d84209be0b" alt="在这里插入图片描述" gcc 的-S 选项,表示在程序编译期间,在生成汇编代码后,停止,-o 输出汇编代码文件。 - 汇编
gcc -c test.s -o test.o - 链接
gcc 连接器是 gas 提供的,负责将程序的目标文件与所需的所有附加的目标文件连接起来,最终生 成可执行文件。附加的目标文件包括静态连接库和动态连接库。 gcc test.o -o test data:image/s3,"s3://crabby-images/88e7d/88e7df65f81d7ce3094d8ba4c836961b52950377" alt="在这里插入图片描述"
多个程序文件的编译
data:image/s3,"s3://crabby-images/8d99d/8d99df8959695862f7677f0ecdd6109a4a2456e8" alt="在这里插入图片描述" gcc -c test1.c -o test1.o gcc -c test2.c -o test2.o gcc test1.o test2.o -o test
- 检错
a. gcc -pedantic illcode.c -o illcode -pedantic 选项能够帮助程序员发现一些不符合 ANSI/ISO C 标准的代码,但不是全部,事实上只有 ANSI/ISO C 语言标准中要求进行编译器诊断的 那些情况,才有可能被 GCC 发现并提出警告 b. gcc -Wall illcode.c -o illcode 使用它能够使 GCC 产生尽可能多的警告信息
总结
静态库链接时搜索路径顺序:
- ld 会去找 GCC 命令中的参数-L
- 再找 gcc 的环境变量 LIBRARY_PATH
- 再找内定目录 /lib /usr/lib /usr/local/lib 这是当初 compile gcc 时写在程序内的
动态链接时、执行时搜索路径顺序:
- 编译目标代码时指定的动态库搜索路径
- 环境变量 LD_LIBRARY_PATH 指定的动态库搜索路径
- 配置文件/etc/ld.so.conf 中指定的动态库搜索路径
- 默认的动态库搜索路径/lib
- 默认的动态库搜索路径/usr/lib
|