我想启用GCC拥有的所有警告。(你会认为这很容易…)
你可能会想-墙可能会管用,但不是!你还需要-沃特。 你可能会想-Wextra可能会奏效,但不是!并不是这里列出的所有警告(例如,-Wshadow)都可以通过此命令启用。我仍然不知道这个列表是否全面。
我如何告诉GCC启用(没有if's, and's, or but's!)它所具有的所有警告?
我想启用GCC拥有的所有警告。(你会认为这很容易…)
你可能会想-墙可能会管用,但不是!你还需要-沃特。 你可能会想-Wextra可能会奏效,但不是!并不是这里列出的所有警告(例如,-Wshadow)都可以通过此命令启用。我仍然不知道这个列表是否全面。
我如何告诉GCC启用(没有if's, and's, or but's!)它所具有的所有警告?
当前回答
我仍然不知道这个列表是否全面。
它可能是,但唯一100%全面的列表是编译器的实际源代码。然而,GCC很大!我不知道所有命令行参数是收集在一个地方还是分散在几个源文件中。还要注意,一些警告是针对预处理器的,一些是针对实际编译器的,还有一些是针对链接器的(链接器是一个完全独立的程序,可以在binutils包中找到),因此它们很可能是分散的。
其他回答
GCC 4.3+现在有-Q——help=警告,你甚至可以指定——help=警告,打印出与C相关的警告。
我刚刚写了一个m4模块来利用这一点(它也支持Clang的-Weverything);看到wget_manywarnings.m4。
如何使用它很简单。基本上,模块会打开每个警告标志。您还可以根据需要删除警告—有些警告非常冗长。
例如:configure.ac
如果你不使用autotools,你会发现在m4模块中打开所有禁用警告的代码,这基本上是通过AWK管道的GCC调用:
旗帜= " - wall -Wextra -Wformat = 2”$ (gcc - wall -Wextra -Wformat = 2 q——帮助=警告,C | awk '{如果(($ 2 = =“(残疾人)”| | 2美元 == "") && $ 1 !~/=/ && $1~/^-W/&& $1!="-Wall") print $1} `
在所有警告都启用的情况下编程是不可能的(除非您打算忽略它们,但那又何必呢?)例如,让我们假设您使用以下标志集:- wstrict - prototyping -Wtraditional。
即使启用了两个警告,下面的程序也会报错。
/tmp $ cat main.c
int main(int argc, char **argv) {
return 0;
}
/tmp $ gcc -Wstrict-prototypes -Wtraditional main.c
main.c: In function ‘main’:
main.c:1:5: warning: traditional C rejects ISO C style function definitions [-Wtraditional]
int main(int argc, char **argv) {
^
你可能会想“好吧,那我就用旧样式的原型吧”。不,这行不通。
/tmp $ cat main.c
int main(argc, argv)
int argc;
char **argv;
{
return 0;
}
/tmp $ gcc -Wstrict-prototypes -Wtraditional main.c
main.c:1:5: warning: function declaration isn’t a prototype [-Wstrict-prototypes]
int main(argc, argv)
^
不,不指定任何原型也是错误的,因为编译器也会报错。
/tmp $ cat main.c
int main() {
return 0;
}
/tmp $ gcc -Wstrict-prototypes -Wtraditional main.c
main.c:1:5: warning: function declaration isn’t a prototype [-Wstrict-prototypes]
int main() {
^
如果你在程序中定义任何函数,你不能使用所有的标志,因为编译器会抱怨任何可以想象到的函数定义。
对于c++,这是可能的(-Wtraditional标志不存在),并且可以编译非常简单的程序。要启用所有警告,请使用以下警告列表(可能有些警告是重复的,因为我没有费心过滤-Wall启用的警告)。
-Wabi -Wctor-dtor-privacy -Wnon-virtual-dtor -Wreorder -Weffc++ -Wstrict-null-sentinel -Wno-non-template-friend -Wold-style-cast -Woverloaded-virtual -Wno-pmf-conversions -Wsign-promo -Wextra -Wall -Waddress -Waggregate-return -Warray-bounds -Wno-attributes -Wno-builtin-macro-redefined -Wc++0x-compat -Wcast-align -Wcast-qual -Wchar-subscripts -Wclobbered -Wcomment -Wconversion -Wcoverage-mismatch -Wno-deprecated -Wno-deprecated-declarations -Wdisabled-optimization -Wno-div-by-zero -Wempty-body -Wenum-compare -Wno-endif-labels -Wfatal-errors -Wfloat-equal -Wformat -Wformat=2 -Wno-format-contains-nul -Wno-format-extra-args -Wformat-nonliteral -Wformat-security -Wformat-y2k -Wignored-qualifiers -Winit-self -Winline -Wno-int-to-pointer-cast -Wno-invalid-offsetof -Winvalid-pch -Wunsafe-loop-optimizations -Wlogical-op -Wlong-long -Wmain -Wmissing-braces -Wmissing-field-initializers -Wmissing-format-attribute -Wmissing-include-dirs -Wmissing-noreturn -Wno-mudflap -Wno-multichar -Wnonnull -Wno-overflow -Woverlength-strings -Wpacked -Wpacked-bitfield-compat -Wpadded -Wparentheses -Wpointer-arith -Wredundant-decls -Wreturn-type -Wsequence-point -Wshadow -Wsign-compare -Wsign-conversion -Wstack-protector -Wstrict-aliasing=1 -Wstrict-overflow=5 -Wswitch -Wswitch-default -Wswitch-enum -Wsync-nand -Wsystem-headers -Wtrigraphs -Wtype-limits -Wundef -Wuninitialized -Wunknown-pragmas -Wno-pragmas -Wunreachable-code -Wunused -Wunused-function -Wunused-label -Wunused-parameter -Wunused-value -Wunused-variable -Wvariadic-macros -Wvla -Wvolatile-register-var -Wwrite-strings
你不能。
GCC 4.4.0的手册只是该版本的综合手册,但它确实列出了4.4.0所有可能的警告。不过,它们并不都在你链接的页面上。例如,c++选项或Objective-C选项页面上有一些特定于语言的选项。要找到所有选项,最好查看选项摘要
打开所有选项将包括-Wdouble-promotion,它只与带有32位单精度浮点单元的cpu相关,该单元在硬件中实现浮点运算,但在软件中模拟double。以double方式进行计算将使用软件模拟,并且速度较慢。这与某些嵌入式cpu有关,但与硬件支持64位浮点的现代桌面cpu完全无关。
另一个通常不常用的警告是-Wtraditional,它警告在传统C中具有不同含义(或不起作用)的完美格式的代码,例如“字符串”“连接”或ISO C函数定义!你真的关心与30年前的编译器的兼容性吗?你真的想要一个写int inc(int i) {return i+1的警告吗?} ?
I think -Weffc++ is too noisy to be useful. It's based on the outdated first edition of Effective C++ and warns about constructs which are perfectly valid C++ (and for which the guidelines changed in later editions of the book). I don't want to be warned that I haven't initialized a std::string member in my constructor; it has a default constructor that does exactly what I want. Why should I write m_str() to call it? The -Weffc++ warnings that would be helpful are too difficult for the compiler to detect accurately (giving false negatives), and the ones that aren't useful, such as initializing all members explicitly, just produce too much noise, giving false positives.
Luc Danton提供了一个来自-Waggregate-return的无用警告的好例子,几乎可以肯定,这些警告对c++代码毫无意义。
也就是说,你并不需要所有的警告;你只是觉得你知道。
通读手册,阅读它们,决定您可能想要启用哪些,并尝试它们。无论如何,阅读编译器的手册是一个好东西,走捷径并启用你不理解的警告不是一个很好的主意,特别是如果它是为了避免不得不RTFM。
那些把所有东西都打开的人,可能要么是因为他们毫无头绪,要么是因为一头尖尖头发的老板说“没有警告”。
有些警告很重要,有些则不然。你必须有辨别力,否则你的程序就会一团糟。以-Wdouble-promotion为例。如果你在嵌入式系统上工作,你可能需要这个;如果您使用的是桌面系统,则可能不需要。你想要- wtradition吗?我对此表示怀疑。
参见-Wall-all启用所有警告,关闭为WONTFIX。
DevSolar抱怨makefile需要根据编译器版本使用不同的警告,如果-Wall -Wextra不适合,那么使用特定于编译器和特定于版本的CFLAGS并不困难:
compiler_name := $(notdir $(CC))
ifeq ($(compiler_name),gcc)
compiler_version := $(basename $(shell $(CC) -dumpversion))
endif
ifeq ($(compile_name),clang)
compiler_version := $(shell $(CC) --version | awk 'NR==1{print $$3}')
endif
# ...
wflags.gcc.base := -Wall -Wextra
wflags.gcc.4.7 := -Wzero-as-null-pointer-constant
wflags.gcc.4.8 := $(wflags.gcc.4.7)
wflags.clang.base := -Wall -Wextra
wflags.clang.3.2 := -Weverything
CFLAGS += $(wflags.$(compiler_name).base) $(wflags.$(compiler_name).$(compiler_version))
我仍然不知道这个列表是否全面。
它可能是,但唯一100%全面的列表是编译器的实际源代码。然而,GCC很大!我不知道所有命令行参数是收集在一个地方还是分散在几个源文件中。还要注意,一些警告是针对预处理器的,一些是针对实际编译器的,还有一些是针对链接器的(链接器是一个完全独立的程序,可以在binutils包中找到),因此它们很可能是分散的。
有人创建了一组工具,用于为给定的GCC或Clang版本确定完整的警告集。
对于GCC,从该工具提供的完整警告列表复制到编译器版本似乎是确保所有警告都被打开的唯一方法,因为(与Clang不同)GCC不提供-Weverything。
该工具似乎解析了GCC源代码中的实际c.opt文件,因此其结果应该是确定的。
存储库还包含为大多数GCC和Clang版本(目前是Clang 3.2到3.7以及GCC 3.4到5.3)生成的警告列表的文本文件。