Выборочно отключить предупреждения GCC только для части единицы перевода?


85

Какой GCC наиболее близок к этому коду препроцессора MSVC?

#pragma warning( push )                    // Save the current warning state.
#pragma warning( disable : 4723 )          // C4723: potential divide by 0
// Code which would generate warning 4723.
#pragma warning( pop )                     // Restore warnings to previous state.

У нас есть код в часто включаемых заголовках, который мы не хотим генерировать конкретное предупреждение. Однако мы хотим, чтобы файлы, содержащие эти заголовки, продолжали выдавать это предупреждение (если в проекте включено это предупреждение).


Если заголовки установлены в / usr / include или что у вас есть, gcc по умолчанию не генерирует для них предупреждения.
Spudd86

Ответы:


95

Это возможно в GCC, начиная с версии 4.6, или примерно в июне 2010 года в магистрали.

Вот пример:

#pragma GCC diagnostic push
#pragma GCC diagnostic error "-Wuninitialized"
    foo(a);         /* error is given for this one */
#pragma GCC diagnostic push
#pragma GCC diagnostic ignored "-Wuninitialized"
    foo(b);         /* no diagnostic for this one */
#pragma GCC diagnostic pop
    foo(c);         /* error is given for this one */
#pragma GCC diagnostic pop
    foo(d);         /* depends on command line options */

8
Функциональные возможности push и pop были добавлены в gcc 4.6 ( gcc.gnu.org/onlinedocs/gcc-4.6.0/gcc/Diagnostic-Pragmas.html ).
Дэйв Йохансен

1
Вы, вероятно, захотите нажать дважды, если собираетесь лопнуть дважды.
Дэн

2
@Dan: Прочтите руководство и прокомментируйте. Обратите внимание на происхождение примера.
Matt Joiner

К вашему сведению, для более старых версий, таких как 4.4.7, вы все еще можете использовать, #pragma GCC diagnostic [error|warning|ignored]но popон не реализован / не поддерживается.
Тревор Бойд Смит

36

Ближе всего является диагностическим Прагма GCC , #pragma GCC diagnostic [warning|error|ignored] "-Wwhatever". Это не очень похоже на то, что вы хотите, и см. Ссылку для получения подробностей и предупреждений.


1
Знаете ли вы, что и где может быть основание для отказа от добавления этой функции? (Мне не удалось его найти.) Я считаю полезным предупреждение push-disable-pop.

1
Я действительно не думаю, что «отказ от добавления функций» в gcc имеет такое объяснение, как отсутствие кого-либо, отправляющего рабочий патч.
хаос

14
Дело не в том, что никто не желает выполнять работу для такого рода детализированного контроля предупреждений в gcc или отправлять код - я знаю одну крупную корпорацию Кремниевой долины, которая уже сделала это, а другая была бы рада заплатить кому-то сделать это и получить код в потоке. Скорее, согласно обсуждению с парнем, который (как один из сопровождающих gdb) подключился к этому материалу, у сопровождающих gcc есть философия: «Если есть предупреждение, это ошибка, и вы должны ее исправить». Итак (имо) это религиозный аргумент, и они контролируют код, поэтому они выигрывают.
Боб Мерфи

Чтобы добавить к комментарию Боба, разработчики GCC не любили эту #pragmaдирективу, поэтому все, что специфично для GCC, вероятно, будет реализовано как __attribute__((foo)).
Tom

10
новый gcc (> = 4.4) имеет, #pragma GCC push_optionsтак что вы можете возиться
Spudd86

33

Я сделал нечто подобное. Для стороннего кода я вообще не хотел видеть никаких предупреждений. Поэтому вместо того, чтобы указывать -I/path/to/libfoo/include, я использовал-isystem /path/to/libfoo/include . Это заставляет компилятор обрабатывать эти файлы заголовков как «системные заголовки» для предупреждений, и пока вы не включите их -Wsystem-headers, вы в большей степени в безопасности. Я все еще видел утечку нескольких предупреждений, но это сокращает большую часть мусора.

Обратите внимание, что это поможет вам только в том случае, если вы можете изолировать вредоносный код с помощью include-directory. Если это просто подмножество вашего собственного проекта или смешанное с другим кодом, вам не повезло.


1
Отличный совет. При использовании LLVM добавьте флаг -isystem в разделе «Другие флаги C» в разделе «Компилятор Apple LLVM - Язык».
Nestor

@Tom Спасибо, что поделились. Я не могу понять, где использовать ваше решение. Можете сказать еще немного?
Lorenzo B

1

Это расширение ответа Мэтта Джойнера .

Если вы не хотите создавать прагмы по всему коду, вы можете использовать оператор _Pragma :

#ifdef __GNUC__
#  define DIAGNOSTIC_ERROR(w) _Pragma("GCC diagnostic error \"" w "\"")
#  define DIAGNOSTIC_IGNORE(w) _Pragma("GCC diagnostic ignore \"" w "\"")
#  define DIAGNOSTIC_PUSH _Pragma("GCC diagnostic push")
#  define DIAGNOSTIC_POP _Pragma("GCC diagnostic pop")
#endif
// (...)

DIAGNOSTIC_ERROR("-Wuninitialized")
foo(a); // Error

DIAGNOSTIC_PUSH
DIAGNOSTIC_IGNORE("-Wuninitialized")
foo(a); // No error

DIAGNOSTIC_POP
foo(a); // Error
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.