是否有一种方法可以让gcc/g++从命令行转储其默认的预处理器定义? 我的意思是像__GNUC__, __STDC__,等等。


当前回答

一种在Linux或Windows上同样有效的可移植方法(其中没有/dev/null):

echo | gcc -dM -E -

对于c++,您可以使用(将c++11替换为您使用的任何版本):

echo | gcc -x c++ -std=c++11 -dM -E -

它的工作原理是告诉gcc预处理stdin(由echo生成)并打印所有预处理器定义(搜索-dletters)。如果你想知道当你包含头文件时添加了什么定义,你可以使用-dD选项,它类似于-dM,但不包括预定义的宏:

echo "#include <stdlib.h>" | gcc -x c++ -std=c++11 -dD -E -

但是请注意,空输入仍然会产生许多带有-dD选项的定义。

其他回答

我通常是这样做的:

$ gcc -dM -E - < /dev/null

请注意,一些预处理器定义依赖于命令行选项——您可以通过向上面的命令行添加相关选项来测试这些定义。例如,要查看哪些SSE3/SSE4选项默认启用:

$ gcc -dM -E - < /dev/null | grep SSE[34]
#define __SSE3__ 1
#define __SSSE3__ 1

然后在指定-msse4时进行比较:

$ gcc -dM -E -msse4 - < /dev/null | grep SSE[34]
#define __SSE3__ 1
#define __SSE4_1__ 1
#define __SSE4_2__ 1
#define __SSSE3__ 1

类似地,你可以看到在两组不同的命令行选项中哪些选项不同,例如比较优化级别的预处理器定义-O0(无)和-O3(满):

$ gcc -dM -E -O0 - < /dev/null > /tmp/O0.txt
$ gcc -dM -E -O3 - < /dev/null > /tmp/O3.txt
$ sdiff -s /tmp/O0.txt /tmp/O3.txt 
#define __NO_INLINE__ 1        <
                               > #define __OPTIMIZE__ 1

一种在Linux或Windows上同样有效的可移植方法(其中没有/dev/null):

echo | gcc -dM -E -

对于c++,您可以使用(将c++11替换为您使用的任何版本):

echo | gcc -x c++ -std=c++11 -dM -E -

它的工作原理是告诉gcc预处理stdin(由echo生成)并打印所有预处理器定义(搜索-dletters)。如果你想知道当你包含头文件时添加了什么定义,你可以使用-dD选项,它类似于-dM,但不包括预定义的宏:

echo "#include <stdlib.h>" | gcc -x c++ -std=c++11 -dD -E -

但是请注意,空输入仍然会产生许多带有-dD选项的定义。

当工作在一个具有复杂构建系统的大项目中,并且很难直接获得(或修改)gcc/g++命令时,还有另一种方法可以查看宏扩展的结果。 简单地重新定义宏,你会得到类似如下的输出:

file.h: note: this is the location of the previous definition
#define MACRO current_value

是的,使用-E -dM选项而不是-c。 示例(输出到stdout):

 echo | gcc -dM -E -
 echo | clang -dM -E -

美国小妞C + +

 echo | g++ -dM -E -x c++ -
 echo | clang++ -dM -E -x c++ -

来自gcc手册:

而不是正常的输出,生成 所有“#define”指令的列表 过程中定义的宏 预处理器的执行, 包括预定义的宏。这 给你一种方法来找出什么是 的版本中预定义的 预处理器。假设你没有 文件foo.h,命令 触摸foo;cpp -dM foo.h 将显示所有预定义宏。 如果使用-dM而不带-E选项, -dM被解释为-fdump-rtl-mach的同义词。

晚回答——我发现其他答案很有用——想再补充一点。


如何转储来自特定头文件的预处理器宏?

echo "#include <sys/socket.h>" | gcc -E -dM -

或者(感谢@mymedia的建议):

gcc -E -dM -include sys/socket.h - < /dev/null

特别是,我想知道在我的系统上SOMAXCONN定义为什么。我知道我可以直接打开标准头文件,但有时我必须搜索一下才能找到头文件的位置。相反,我可以使用以下一行代码:

$ gcc -E -dM -include sys/socket.h - < /dev/null | grep SOMAXCONN
#define SOMAXCONN 128
$