gcc 预编译

最近在做一个关于重写符号表的 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 的命令,以及利用预编译参数重写符号表。

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容