一、背景
由于Linux内核需要支持各种硬件平台、架构和特性,因此通过使用宏,开发者可以在不同的编译条件下包含或排除特定的代码段,从而确保内核能够在不同的环境中正确运行。
然而这也带来了一个挑战:理解和维护这些宏定义可能会变得相当复杂。由于宏是在预处理阶段进行处理的,所以将引用宏的源文件进行预编译,也不失为一个理解源码的好办法。
但是由于内核编译开关太多,依赖关系复杂,因此要先找出在当前的内核配置下编译目标源文件的编译命令。
二、方法
这里可以通过编译 Linux 内核的日志信息,来找出目标源文件的编译命令。
编译内核的 make 指令,应添加参数 V=1 或 VERBOSE=1 以打印更全面的编译日志,参考下例:
make V=1 > all.log 2>&1 #其中的2>&1表示标准错误的打印输出到标准输出中,而重定向符默认将标准输出中的打印重定向到all.log中
接着在 all.log 中找到目标源文件的 gcc 编译命令,这条编译命令是将目标源文件直接编译为 .o 文件,因此要参考下例做参数替换。
参数替换前:
gcc -c hello.c -o hello.o
参数替换后:
gcc -E hello.c -o hello.i # hello.i就是预编译文件了,这里面引用的宏都已展开
三、完整示例
以 Linux-4.9.88\drivers\base\platform.c 为目标源文件,进行预编译。
预编译前的源代码中部分的宏引用示例:
- 在 Linux-4.9.88 目录下执行 make 命令,编译内核
make zImage -j4 V=1 > ../zImage_log.txt 2>&1
- 在 zImage_log.txt 中找到编译 platform.c 的命令,注意红框部分。
- 修改编译 platform.c 的命令,并在 Linux-4.9.88 目录下执行,注意红框部分。
打开预编译后生成的 platform.i 文件,源代码中宏引用的示例部分,现如下所示: