// RUN: %clang_cc1 -fsyntax-only -verify -Wtautological-negation-compare -Wno-constant-logical-operand %s // RUN: %clang_cc1 -fsyntax-only -verify -Wtautological-compare -Wno-constant-logical-operand %s // RUN: %clang_cc1 -fsyntax-only -verify -Wall -Wno-unused -Wno-loop-analysis -Wno-constant-logical-operand %s #define COPY(x) x void test_int(int x) { if (x || !x) {} // expected-warning {{'||' of a value and its negation always evaluates to true}} if (!x || x) {} // expected-warning {{'||' of a value and its negation always evaluates to true}} if (x && !x) {} // expected-warning {{'&&' of a value and its negation always evaluates to false}} if (!x && x) {} // expected-warning {{'&&' of a value and its negation always evaluates to false}} // parentheses are ignored if (x || (!x)) {} // expected-warning {{'||' of a value and its negation always evaluates to true}} if (!(x) || x) {} // expected-warning {{'||' of a value and its negation always evaluates to true}} // don't warn on macros if (COPY(x) || !x) {} if (!x || COPY(x)) {} if (x && COPY(!x)) {} if (COPY(!x && x)) {} // dont' warn on literals if (1 || !1) {} if (!42 && 42) {} // don't warn on overloads struct Foo{ int val; Foo operator!() const { return Foo{!val}; } bool operator||(const Foo other) const { return val || other.val; } bool operator&&(const Foo other) const { return val && other.val; } }; Foo f{3}; if (f || !f) {} if (!f || f) {} if (f.val || !f.val) {} // expected-warning {{'||' of a value and its negation always evaluates to true}} if (!f.val && f.val) {} // expected-warning {{'&&' of a value and its negation always evaluates to false}} }