最近在做一个关于重写符号表的 shell,但是其中用到了预编译的知识。所以先补一补 gcc 预编译的知识。明白整个编译过程,更是极好的。
简单提一下编译过程
- 预编译处理 -> 编译 -> 汇编 -> 连接
预编译
- 1.处理注释:删除所有注释,使用空格取代注释内容
- 2.处理宏定义:将 define 的值替换为宏定义对应的值
- 3.处理条件编译指令:处理 #if, #else, #elif, #endif 等条件编译指令
- 4.处理#include:将把#include 的代码复制到源代码中
- 5.处理#pragma:这个指定会被保留下来
#include<stdio.h>
int main(int argc, char **argv) {
printf("Hello, world!\n");
printf("This is define value is\n");
#pragma message("编译")
return 0;
}
这是示例代码
使用 gcc -E main.c -o main.s
定义的 pragma message 会被打印
接下来使用 gcc -S main.i -o main.s
也会被打印
编译
- 1.词法分析:分析关键字,标识符等的合法性
- 2.语法分析:检查,代码是否遵循 c 语音语法规范
- 3.语义分析:分析表达式是否合法
汇编器
- 1.使用汇编器将汇编代码转化为机器的可执行机器码
连接器
- 1.软件各个模块之前会相互调用,连接器就是处理各个模块调用的衔接
相关编译命令
- -E (预处理: gcc/clang -E main.c -o main.i)
- -S (编译: gcc/clang -S main.i -o main.s)
- -c (汇编: gcc/clang -c main.s -o main.o)
- -V 可以查看总体编译细节
- gcc main.c 直接生成可执行文件,不会有中间过程生成
- gcc -save-temps main.c 可以生成中间过程文件
- gcc main.o 直接把汇编文件生成可执行二进制文件
本文其实主要是为了说一个小重点,在预编译阶段的条件编译
命令行定义宏:可以使用 gcc -D
选型来定义宏,我就是使用这个预编译命令来修改符号表的
gcc -DMAX=1 main.c 等价于 #define MAX 1 语句
具体用法示例:
#include<stdio.h>
int main(int argc, char **argv) {
printf("Hello, world!\n");
printf("This is define value is %f\n", MAX);
return 0;
}
使用 gcc -DMAX=10 -E main.c -o main.i
编译一下,生成了很多代码,提取出来的如下
int main(int argc, char **argv) {
printf("Hello, world!\n");
printf("This is define value is %f\n", 10);
return 0;
}
可以看出,他已经把我在编译时加入的参数,编译到了源文件,也证明了在预编译极端,会把#define 的值替换为原始值的论点
主要命令参数是
gcc -D
下一篇就要说,如何用 llvm 的命令,以及利用预编译参数重写符号表。