[Sema] -Wtautological-constant-compare is too good. Cripple it.
Summary:
The diagnostic was mostly introduced in D38101 by me, as a reaction to wasting a lot of time, see [[ https://lists.llvm.org/pipermail/cfe-commits/Week-of-Mon-20171009/206427.html | mail ]].
However, the diagnostic is pretty dumb. While it works with no false-positives,
there are some questionable cases that are diagnosed when one would argue that they should not be.
The common complaint is that it diagnoses the comparisons between an `int` and
`long` when compiling for a 32-bit target as tautological, but not when
compiling for 64-bit targets. The underlying problem is obvious: data model.
In most cases, 64-bit target is `LP64` (`int` is 32-bit, `long` and pointer are
64-bit), and the 32-bit target is `ILP32` (`int`, `long`, and pointer are 32-bit).
I.e. the common pattern is: (pseudocode)
```
#include <limits>
#include <cstdint>
int main() {
using T1 = long;
using T2 = int;
T1 r;
if (r < std::numeric_limits<T2>::min()) {}
if (r > std::numeric_limits<T2>::max()) {}
}
```
As an example, D39149 was trying to fix this diagnostic in libc++, and it was not well-received.
This *could* be "fixed", by changing the diagnostics logic to something like
`if the types of the values being compared are different, but are of the same size, then do diagnose`,
and i even attempted to do so in D39462, but as @rjmccall rightfully commented,
that implementation is incomplete to say the least.
So to stop causing trouble, and avoid contaminating upcoming release, lets do this workaround:
* move these three diags (`warn_unsigned_always_true_comparison`, `warn_unsigned_enum_always_true_comparison`, `warn_tautological_constant_compare`) into it's own `-Wtautological-constant-in-range-compare`
* Disable them by default
* Make them part of `-Wextra`
* Additionally, give `warn_tautological_constant_compare` it's own flag `-Wtautological-type-limit-compare`.
I'm not happy about that name, but i can't come up with anything better.
This way all three of them can be enabled/disabled either altogether, or one-by-one.
Reviewers: aaron.ballman, rsmith, smeenai, rjmccall, rnk, mclow.lists, dim
Reviewed By: aaron.ballman, rsmith, dim
Subscribers: thakis, compnerd, mehdi_amini, dim, hans, cfe-commits, rjmccall
Tags: #clang
Differential Revision: https://reviews.llvm.org/D41512
llvm-svn: 321691
2018-01-03 16:45:19 +08:00
|
|
|
// RUN: %clang_cc1 -triple x86_64-linux-gnu -fsyntax-only -Wtautological-constant-in-range-compare -DTEST -verify %s
|
|
|
|
// RUN: %clang_cc1 -triple x86_64-linux-gnu -fsyntax-only -Wtautological-constant-in-range-compare -DTEST -verify -x c++ %s
|
|
|
|
// RUN: %clang_cc1 -triple x86_64-linux-gnu -fsyntax-only -Wtautological-type-limit-compare -DTEST -verify %s
|
|
|
|
// RUN: %clang_cc1 -triple x86_64-linux-gnu -fsyntax-only -Wtautological-type-limit-compare -DTEST -verify -x c++ %s
|
Remove TautologicalInRangeCompare from Extra and TautologicalCompare.
This removes the following (already default-off) warnings from -Wextra:
-Wtautological-type-limit-compare,
-Wtautological-unsigned-zero-compare
-Wtautological-unsigned-enum-zero-compare
On the thread "[cfe-dev] -Wtautological-constant-compare issues", clang
code owners Richard Smith, John McCall, and Reid Kleckner as well as
libc++ code owner Marshall Clow stated that these new warnings are not
yet ready for prime time and shouldn't be part of -Wextra.
Furthermore, Vedant Kumar (Apple), Peter Hosek (Fuchsia), and me (Chromium)
expressed the same concerns (Vedant on that thread, Peter on
https://reviews.llvm.org/D39462, me on https://reviews.llvm.org/D41512).
So remove them from -Wextra, and remove TautologicalInRangeCompare from
TautologicalCompare too until they're usable with real-world code.
llvm-svn: 322901
2018-01-19 05:40:27 +08:00
|
|
|
// RUN: %clang_cc1 -triple x86_64-linux-gnu -fsyntax-only -Wextra -Wno-sign-compare -verify %s
|
|
|
|
// RUN: %clang_cc1 -triple x86_64-linux-gnu -fsyntax-only -Wextra -Wno-sign-compare -verify -x c++ %s
|
[Sema] -Wtautological-constant-compare is too good. Cripple it.
Summary:
The diagnostic was mostly introduced in D38101 by me, as a reaction to wasting a lot of time, see [[ https://lists.llvm.org/pipermail/cfe-commits/Week-of-Mon-20171009/206427.html | mail ]].
However, the diagnostic is pretty dumb. While it works with no false-positives,
there are some questionable cases that are diagnosed when one would argue that they should not be.
The common complaint is that it diagnoses the comparisons between an `int` and
`long` when compiling for a 32-bit target as tautological, but not when
compiling for 64-bit targets. The underlying problem is obvious: data model.
In most cases, 64-bit target is `LP64` (`int` is 32-bit, `long` and pointer are
64-bit), and the 32-bit target is `ILP32` (`int`, `long`, and pointer are 32-bit).
I.e. the common pattern is: (pseudocode)
```
#include <limits>
#include <cstdint>
int main() {
using T1 = long;
using T2 = int;
T1 r;
if (r < std::numeric_limits<T2>::min()) {}
if (r > std::numeric_limits<T2>::max()) {}
}
```
As an example, D39149 was trying to fix this diagnostic in libc++, and it was not well-received.
This *could* be "fixed", by changing the diagnostics logic to something like
`if the types of the values being compared are different, but are of the same size, then do diagnose`,
and i even attempted to do so in D39462, but as @rjmccall rightfully commented,
that implementation is incomplete to say the least.
So to stop causing trouble, and avoid contaminating upcoming release, lets do this workaround:
* move these three diags (`warn_unsigned_always_true_comparison`, `warn_unsigned_enum_always_true_comparison`, `warn_tautological_constant_compare`) into it's own `-Wtautological-constant-in-range-compare`
* Disable them by default
* Make them part of `-Wextra`
* Additionally, give `warn_tautological_constant_compare` it's own flag `-Wtautological-type-limit-compare`.
I'm not happy about that name, but i can't come up with anything better.
This way all three of them can be enabled/disabled either altogether, or one-by-one.
Reviewers: aaron.ballman, rsmith, smeenai, rjmccall, rnk, mclow.lists, dim
Reviewed By: aaron.ballman, rsmith, dim
Subscribers: thakis, compnerd, mehdi_amini, dim, hans, cfe-commits, rjmccall
Tags: #clang
Differential Revision: https://reviews.llvm.org/D41512
llvm-svn: 321691
2018-01-03 16:45:19 +08:00
|
|
|
// RUN: %clang_cc1 -triple x86_64-linux-gnu -fsyntax-only -Wall -verify %s
|
|
|
|
// RUN: %clang_cc1 -triple x86_64-linux-gnu -fsyntax-only -Wall -verify -x c++ %s
|
|
|
|
// RUN: %clang_cc1 -triple x86_64-linux-gnu -fsyntax-only -verify %s
|
|
|
|
// RUN: %clang_cc1 -triple x86_64-linux-gnu -fsyntax-only -verify -x c++ %s
|
2017-10-16 04:13:17 +08:00
|
|
|
|
|
|
|
int value(void);
|
|
|
|
|
|
|
|
#define macro(val) val
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
template<typename T>
|
|
|
|
void TFunc() {
|
|
|
|
// Make sure that we do warn for normal variables in template functions !
|
|
|
|
unsigned char c = value();
|
|
|
|
#ifdef TEST
|
|
|
|
if (c > 255) // expected-warning {{comparison 'unsigned char' > 255 is always false}}
|
|
|
|
return;
|
|
|
|
#else
|
|
|
|
if (c > 255)
|
|
|
|
return;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (c > macro(255))
|
|
|
|
return;
|
|
|
|
|
|
|
|
T v = value();
|
|
|
|
if (v > 255)
|
|
|
|
return;
|
|
|
|
if (v > 32767)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
int main()
|
|
|
|
{
|
|
|
|
#ifdef __cplusplus
|
|
|
|
TFunc<unsigned char>();
|
|
|
|
TFunc<signed short>();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
short s = value();
|
|
|
|
|
|
|
|
#ifdef TEST
|
|
|
|
if (s == 32767)
|
|
|
|
return 0;
|
|
|
|
if (s != 32767)
|
|
|
|
return 0;
|
|
|
|
if (s < 32767)
|
|
|
|
return 0;
|
|
|
|
if (s <= 32767) // expected-warning {{comparison 'short' <= 32767 is always true}}
|
|
|
|
return 0;
|
|
|
|
if (s > 32767) // expected-warning {{comparison 'short' > 32767 is always false}}
|
|
|
|
return 0;
|
|
|
|
if (s >= 32767)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (32767 == s)
|
|
|
|
return 0;
|
|
|
|
if (32767 != s)
|
|
|
|
return 0;
|
|
|
|
if (32767 < s) // expected-warning {{comparison 32767 < 'short' is always false}}
|
|
|
|
return 0;
|
|
|
|
if (32767 <= s)
|
|
|
|
return 0;
|
|
|
|
if (32767 > s)
|
|
|
|
return 0;
|
|
|
|
if (32767 >= s) // expected-warning {{comparison 32767 >= 'short' is always true}}
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
// FIXME: assumes two's complement
|
|
|
|
if (s == -32768)
|
|
|
|
return 0;
|
|
|
|
if (s != -32768)
|
|
|
|
return 0;
|
|
|
|
if (s < -32768) // expected-warning {{comparison 'short' < -32768 is always false}}
|
|
|
|
return 0;
|
|
|
|
if (s <= -32768)
|
|
|
|
return 0;
|
|
|
|
if (s > -32768)
|
|
|
|
return 0;
|
|
|
|
if (s >= -32768) // expected-warning {{comparison 'short' >= -32768 is always true}}
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (-32768 == s)
|
|
|
|
return 0;
|
|
|
|
if (-32768 != s)
|
|
|
|
return 0;
|
|
|
|
if (-32768 < s)
|
|
|
|
return 0;
|
|
|
|
if (-32768 <= s) // expected-warning {{comparison -32768 <= 'short' is always true}}
|
|
|
|
return 0;
|
|
|
|
if (-32768 > s) // expected-warning {{comparison -32768 > 'short' is always false}}
|
|
|
|
return 0;
|
|
|
|
if (-32768 >= s)
|
|
|
|
return 0;
|
|
|
|
|
2017-12-09 06:57:11 +08:00
|
|
|
// Note: both sides are promoted to unsigned long prior to the comparison, so
|
|
|
|
// it is perfectly possible for a short to compare greater than 32767UL.
|
2017-10-16 04:13:17 +08:00
|
|
|
if (s == 32767UL)
|
|
|
|
return 0;
|
|
|
|
if (s != 32767UL)
|
|
|
|
return 0;
|
|
|
|
if (s < 32767UL)
|
|
|
|
return 0;
|
2017-12-09 06:57:11 +08:00
|
|
|
if (s <= 32767UL)
|
2017-10-16 04:13:17 +08:00
|
|
|
return 0;
|
2017-12-09 06:57:11 +08:00
|
|
|
if (s > 32767UL)
|
2017-10-16 04:13:17 +08:00
|
|
|
return 0;
|
|
|
|
if (s >= 32767UL)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (32767UL == s)
|
|
|
|
return 0;
|
|
|
|
if (32767UL != s)
|
|
|
|
return 0;
|
2017-12-09 06:57:11 +08:00
|
|
|
if (32767UL < s)
|
2017-10-16 04:13:17 +08:00
|
|
|
return 0;
|
|
|
|
if (32767UL <= s)
|
|
|
|
return 0;
|
|
|
|
if (32767UL > s)
|
|
|
|
return 0;
|
2017-12-09 06:57:11 +08:00
|
|
|
if (32767UL >= s)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
enum { ULONG_MAX = (2UL * (unsigned long)__LONG_MAX__ + 1UL) };
|
|
|
|
if (s == 2UL * (unsigned long)__LONG_MAX__ + 1UL)
|
|
|
|
return 0;
|
|
|
|
if (s != 2UL * (unsigned long)__LONG_MAX__ + 1UL)
|
|
|
|
return 0;
|
|
|
|
if (s < 2UL * (unsigned long)__LONG_MAX__ + 1UL)
|
|
|
|
return 0;
|
|
|
|
if (s <= 2UL * (unsigned long)__LONG_MAX__ + 1UL) // expected-warning-re {{comparison 'short' <= {{.*}} is always true}}
|
|
|
|
return 0;
|
|
|
|
if (s > 2UL * (unsigned long)__LONG_MAX__ + 1UL) // expected-warning-re {{comparison 'short' > {{.*}} is always false}}
|
|
|
|
return 0;
|
|
|
|
if (s >= 2UL * (unsigned long)__LONG_MAX__ + 1UL)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (2UL * (unsigned long)__LONG_MAX__ + 1UL == s)
|
|
|
|
return 0;
|
|
|
|
if (2UL * (unsigned long)__LONG_MAX__ + 1UL != s)
|
|
|
|
return 0;
|
|
|
|
if (2UL * (unsigned long)__LONG_MAX__ + 1UL < s) // expected-warning-re {{comparison {{.*}} < 'short' is always false}}
|
|
|
|
return 0;
|
|
|
|
if (2UL * (unsigned long)__LONG_MAX__ + 1UL <= s)
|
|
|
|
return 0;
|
|
|
|
if (2UL * (unsigned long)__LONG_MAX__ + 1UL > s)
|
|
|
|
return 0;
|
|
|
|
if (2UL * (unsigned long)__LONG_MAX__ + 1UL >= s) // expected-warning-re {{comparison {{.*}} >= 'short' is always true}}
|
2017-10-16 04:13:17 +08:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
// FIXME: assumes two's complement
|
|
|
|
if (s == -32768L)
|
|
|
|
return 0;
|
|
|
|
if (s != -32768L)
|
|
|
|
return 0;
|
|
|
|
if (s < -32768L) // expected-warning {{comparison 'short' < -32768 is always false}}
|
|
|
|
return 0;
|
|
|
|
if (s <= -32768L)
|
|
|
|
return 0;
|
|
|
|
if (s > -32768L)
|
|
|
|
return 0;
|
|
|
|
if (s >= -32768L) // expected-warning {{comparison 'short' >= -32768 is always true}}
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (-32768L == s)
|
|
|
|
return 0;
|
|
|
|
if (-32768L != s)
|
|
|
|
return 0;
|
|
|
|
if (-32768L < s)
|
|
|
|
return 0;
|
|
|
|
if (-32768L <= s) // expected-warning {{comparison -32768 <= 'short' is always true}}
|
|
|
|
return 0;
|
|
|
|
if (-32768L > s) // expected-warning {{comparison -32768 > 'short' is always false}}
|
|
|
|
return 0;
|
|
|
|
if (-32768L >= s)
|
|
|
|
return 0;
|
|
|
|
#else
|
|
|
|
// expected-no-diagnostics
|
|
|
|
if (s == 32767)
|
|
|
|
return 0;
|
|
|
|
if (s != 32767)
|
|
|
|
return 0;
|
|
|
|
if (s < 32767)
|
|
|
|
return 0;
|
|
|
|
if (s <= 32767)
|
|
|
|
return 0;
|
|
|
|
if (s > 32767)
|
|
|
|
return 0;
|
|
|
|
if (s >= 32767)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (32767 == s)
|
|
|
|
return 0;
|
|
|
|
if (32767 != s)
|
|
|
|
return 0;
|
|
|
|
if (32767 < s)
|
|
|
|
return 0;
|
|
|
|
if (32767 <= s)
|
|
|
|
return 0;
|
|
|
|
if (32767 > s)
|
|
|
|
return 0;
|
|
|
|
if (32767 >= s)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
// FIXME: assumes two's complement
|
|
|
|
if (s == -32768)
|
|
|
|
return 0;
|
|
|
|
if (s != -32768)
|
|
|
|
return 0;
|
|
|
|
if (s < -32768)
|
|
|
|
return 0;
|
|
|
|
if (s <= -32768)
|
|
|
|
return 0;
|
|
|
|
if (s > -32768)
|
|
|
|
return 0;
|
|
|
|
if (s >= -32768)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (-32768 == s)
|
|
|
|
return 0;
|
|
|
|
if (-32768 != s)
|
|
|
|
return 0;
|
|
|
|
if (-32768 < s)
|
|
|
|
return 0;
|
|
|
|
if (-32768 <= s)
|
|
|
|
return 0;
|
|
|
|
if (-32768 > s)
|
|
|
|
return 0;
|
|
|
|
if (-32768 >= s)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (s == 32767UL)
|
|
|
|
return 0;
|
|
|
|
if (s != 32767UL)
|
|
|
|
return 0;
|
|
|
|
if (s < 32767UL)
|
|
|
|
return 0;
|
|
|
|
if (s <= 32767UL)
|
|
|
|
return 0;
|
|
|
|
if (s > 32767UL)
|
|
|
|
return 0;
|
|
|
|
if (s >= 32767UL)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (32767UL == s)
|
|
|
|
return 0;
|
|
|
|
if (32767UL != s)
|
|
|
|
return 0;
|
|
|
|
if (32767UL < s)
|
|
|
|
return 0;
|
|
|
|
if (32767UL <= s)
|
|
|
|
return 0;
|
|
|
|
if (32767UL > s)
|
|
|
|
return 0;
|
|
|
|
if (32767UL >= s)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
// FIXME: assumes two's complement
|
|
|
|
if (s == -32768L)
|
|
|
|
return 0;
|
|
|
|
if (s != -32768L)
|
|
|
|
return 0;
|
|
|
|
if (s < -32768L)
|
|
|
|
return 0;
|
|
|
|
if (s <= -32768L)
|
|
|
|
return 0;
|
|
|
|
if (s > -32768L)
|
|
|
|
return 0;
|
|
|
|
if (s >= -32768L)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (-32768L == s)
|
|
|
|
return 0;
|
|
|
|
if (-32768L != s)
|
|
|
|
return 0;
|
|
|
|
if (-32768L < s)
|
|
|
|
return 0;
|
|
|
|
if (-32768L <= s)
|
|
|
|
return 0;
|
|
|
|
if (-32768L > s)
|
|
|
|
return 0;
|
|
|
|
if (-32768L >= s)
|
|
|
|
return 0;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (s == 0)
|
|
|
|
return 0;
|
|
|
|
if (s != 0)
|
|
|
|
return 0;
|
|
|
|
if (s < 0)
|
|
|
|
return 0;
|
|
|
|
if (s <= 0)
|
|
|
|
return 0;
|
|
|
|
if (s > 0)
|
|
|
|
return 0;
|
|
|
|
if (s >= 0)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (0 == s)
|
|
|
|
return 0;
|
|
|
|
if (0 != s)
|
|
|
|
return 0;
|
|
|
|
if (0 < s)
|
|
|
|
return 0;
|
|
|
|
if (0 <= s)
|
|
|
|
return 0;
|
|
|
|
if (0 > s)
|
|
|
|
return 0;
|
|
|
|
if (0 >= s)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
unsigned short us = value();
|
|
|
|
|
|
|
|
#ifdef TEST
|
|
|
|
if (us == 65535)
|
|
|
|
return 0;
|
|
|
|
if (us != 65535)
|
|
|
|
return 0;
|
|
|
|
if (us < 65535)
|
|
|
|
return 0;
|
|
|
|
if (us <= 65535) // expected-warning {{comparison 'unsigned short' <= 65535 is always true}}
|
|
|
|
return 0;
|
|
|
|
if (us > 65535) // expected-warning {{comparison 'unsigned short' > 65535 is always false}}
|
|
|
|
return 0;
|
|
|
|
if (us >= 65535)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (65535 == us)
|
|
|
|
return 0;
|
|
|
|
if (65535 != us)
|
|
|
|
return 0;
|
|
|
|
if (65535 < us) // expected-warning {{comparison 65535 < 'unsigned short' is always false}}
|
|
|
|
return 0;
|
|
|
|
if (65535 <= us)
|
|
|
|
return 0;
|
|
|
|
if (65535 > us)
|
|
|
|
return 0;
|
|
|
|
if (65535 >= us) // expected-warning {{comparison 65535 >= 'unsigned short' is always true}}
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (us == 65535UL)
|
|
|
|
return 0;
|
|
|
|
if (us != 65535UL)
|
|
|
|
return 0;
|
|
|
|
if (us < 65535UL)
|
|
|
|
return 0;
|
|
|
|
if (us <= 65535UL) // expected-warning {{comparison 'unsigned short' <= 65535 is always true}}
|
|
|
|
return 0;
|
|
|
|
if (us > 65535UL) // expected-warning {{comparison 'unsigned short' > 65535 is always false}}
|
|
|
|
return 0;
|
|
|
|
if (us >= 65535UL)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (65535UL == us)
|
|
|
|
return 0;
|
|
|
|
if (65535UL != us)
|
|
|
|
return 0;
|
|
|
|
if (65535UL < us) // expected-warning {{comparison 65535 < 'unsigned short' is always false}}
|
|
|
|
return 0;
|
|
|
|
if (65535UL <= us)
|
|
|
|
return 0;
|
|
|
|
if (65535UL > us)
|
|
|
|
return 0;
|
|
|
|
if (65535UL >= us) // expected-warning {{comparison 65535 >= 'unsigned short' is always true}}
|
|
|
|
return 0;
|
|
|
|
#else
|
|
|
|
// expected-no-diagnostics
|
|
|
|
if (us == 65535)
|
|
|
|
return 0;
|
|
|
|
if (us != 65535)
|
|
|
|
return 0;
|
|
|
|
if (us < 65535)
|
|
|
|
return 0;
|
|
|
|
if (us <= 65535)
|
|
|
|
return 0;
|
|
|
|
if (us > 65535)
|
|
|
|
return 0;
|
|
|
|
if (us >= 65535)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (65535 == us)
|
|
|
|
return 0;
|
|
|
|
if (65535 != us)
|
|
|
|
return 0;
|
|
|
|
if (65535 < us)
|
|
|
|
return 0;
|
|
|
|
if (65535 <= us)
|
|
|
|
return 0;
|
|
|
|
if (65535 > us)
|
|
|
|
return 0;
|
|
|
|
if (65535 >= us)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (us == 65535UL)
|
|
|
|
return 0;
|
|
|
|
if (us != 65535UL)
|
|
|
|
return 0;
|
|
|
|
if (us < 65535UL)
|
|
|
|
return 0;
|
|
|
|
if (us <= 65535UL)
|
|
|
|
return 0;
|
|
|
|
if (us > 65535UL)
|
|
|
|
return 0;
|
|
|
|
if (us >= 65535UL)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (65535UL == us)
|
|
|
|
return 0;
|
|
|
|
if (65535UL != us)
|
|
|
|
return 0;
|
|
|
|
if (65535UL < us)
|
|
|
|
return 0;
|
|
|
|
if (65535UL <= us)
|
|
|
|
return 0;
|
|
|
|
if (65535UL > us)
|
|
|
|
return 0;
|
|
|
|
if (65535UL >= us)
|
|
|
|
return 0;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (us == 32767)
|
|
|
|
return 0;
|
|
|
|
if (us != 32767)
|
|
|
|
return 0;
|
|
|
|
if (us < 32767)
|
|
|
|
return 0;
|
|
|
|
if (us <= 32767)
|
|
|
|
return 0;
|
|
|
|
if (us > 32767)
|
|
|
|
return 0;
|
|
|
|
if (us >= 32767)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (32767 == us)
|
|
|
|
return 0;
|
|
|
|
if (32767 != us)
|
|
|
|
return 0;
|
|
|
|
if (32767 < us)
|
|
|
|
return 0;
|
|
|
|
if (32767 <= us)
|
|
|
|
return 0;
|
|
|
|
if (32767 > us)
|
|
|
|
return 0;
|
|
|
|
if (32767 >= us)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (us == 32767UL)
|
|
|
|
return 0;
|
|
|
|
if (us != 32767UL)
|
|
|
|
return 0;
|
|
|
|
if (us < 32767UL)
|
|
|
|
return 0;
|
|
|
|
if (us <= 32767UL)
|
|
|
|
return 0;
|
|
|
|
if (us > 32767UL)
|
|
|
|
return 0;
|
|
|
|
if (us >= 32767UL)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (32767UL == us)
|
|
|
|
return 0;
|
|
|
|
if (32767UL != us)
|
|
|
|
return 0;
|
|
|
|
if (32767UL < us)
|
|
|
|
return 0;
|
|
|
|
if (32767UL <= us)
|
|
|
|
return 0;
|
|
|
|
if (32767UL > us)
|
|
|
|
return 0;
|
|
|
|
if (32767UL >= us)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
#if __SIZEOF_INT128__
|
[Sema] -Wtautological-constant-compare is too good. Cripple it.
Summary:
The diagnostic was mostly introduced in D38101 by me, as a reaction to wasting a lot of time, see [[ https://lists.llvm.org/pipermail/cfe-commits/Week-of-Mon-20171009/206427.html | mail ]].
However, the diagnostic is pretty dumb. While it works with no false-positives,
there are some questionable cases that are diagnosed when one would argue that they should not be.
The common complaint is that it diagnoses the comparisons between an `int` and
`long` when compiling for a 32-bit target as tautological, but not when
compiling for 64-bit targets. The underlying problem is obvious: data model.
In most cases, 64-bit target is `LP64` (`int` is 32-bit, `long` and pointer are
64-bit), and the 32-bit target is `ILP32` (`int`, `long`, and pointer are 32-bit).
I.e. the common pattern is: (pseudocode)
```
#include <limits>
#include <cstdint>
int main() {
using T1 = long;
using T2 = int;
T1 r;
if (r < std::numeric_limits<T2>::min()) {}
if (r > std::numeric_limits<T2>::max()) {}
}
```
As an example, D39149 was trying to fix this diagnostic in libc++, and it was not well-received.
This *could* be "fixed", by changing the diagnostics logic to something like
`if the types of the values being compared are different, but are of the same size, then do diagnose`,
and i even attempted to do so in D39462, but as @rjmccall rightfully commented,
that implementation is incomplete to say the least.
So to stop causing trouble, and avoid contaminating upcoming release, lets do this workaround:
* move these three diags (`warn_unsigned_always_true_comparison`, `warn_unsigned_enum_always_true_comparison`, `warn_tautological_constant_compare`) into it's own `-Wtautological-constant-in-range-compare`
* Disable them by default
* Make them part of `-Wextra`
* Additionally, give `warn_tautological_constant_compare` it's own flag `-Wtautological-type-limit-compare`.
I'm not happy about that name, but i can't come up with anything better.
This way all three of them can be enabled/disabled either altogether, or one-by-one.
Reviewers: aaron.ballman, rsmith, smeenai, rjmccall, rnk, mclow.lists, dim
Reviewed By: aaron.ballman, rsmith, dim
Subscribers: thakis, compnerd, mehdi_amini, dim, hans, cfe-commits, rjmccall
Tags: #clang
Differential Revision: https://reviews.llvm.org/D41512
llvm-svn: 321691
2018-01-03 16:45:19 +08:00
|
|
|
__int128 i128 = value();
|
2017-10-16 04:13:17 +08:00
|
|
|
if (i128 == -1) // used to crash
|
|
|
|
return 0;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
enum E {
|
|
|
|
yes,
|
|
|
|
no,
|
|
|
|
maybe
|
|
|
|
};
|
[Sema] -Wtautological-constant-compare is too good. Cripple it.
Summary:
The diagnostic was mostly introduced in D38101 by me, as a reaction to wasting a lot of time, see [[ https://lists.llvm.org/pipermail/cfe-commits/Week-of-Mon-20171009/206427.html | mail ]].
However, the diagnostic is pretty dumb. While it works with no false-positives,
there are some questionable cases that are diagnosed when one would argue that they should not be.
The common complaint is that it diagnoses the comparisons between an `int` and
`long` when compiling for a 32-bit target as tautological, but not when
compiling for 64-bit targets. The underlying problem is obvious: data model.
In most cases, 64-bit target is `LP64` (`int` is 32-bit, `long` and pointer are
64-bit), and the 32-bit target is `ILP32` (`int`, `long`, and pointer are 32-bit).
I.e. the common pattern is: (pseudocode)
```
#include <limits>
#include <cstdint>
int main() {
using T1 = long;
using T2 = int;
T1 r;
if (r < std::numeric_limits<T2>::min()) {}
if (r > std::numeric_limits<T2>::max()) {}
}
```
As an example, D39149 was trying to fix this diagnostic in libc++, and it was not well-received.
This *could* be "fixed", by changing the diagnostics logic to something like
`if the types of the values being compared are different, but are of the same size, then do diagnose`,
and i even attempted to do so in D39462, but as @rjmccall rightfully commented,
that implementation is incomplete to say the least.
So to stop causing trouble, and avoid contaminating upcoming release, lets do this workaround:
* move these three diags (`warn_unsigned_always_true_comparison`, `warn_unsigned_enum_always_true_comparison`, `warn_tautological_constant_compare`) into it's own `-Wtautological-constant-in-range-compare`
* Disable them by default
* Make them part of `-Wextra`
* Additionally, give `warn_tautological_constant_compare` it's own flag `-Wtautological-type-limit-compare`.
I'm not happy about that name, but i can't come up with anything better.
This way all three of them can be enabled/disabled either altogether, or one-by-one.
Reviewers: aaron.ballman, rsmith, smeenai, rjmccall, rnk, mclow.lists, dim
Reviewed By: aaron.ballman, rsmith, dim
Subscribers: thakis, compnerd, mehdi_amini, dim, hans, cfe-commits, rjmccall
Tags: #clang
Differential Revision: https://reviews.llvm.org/D41512
llvm-svn: 321691
2018-01-03 16:45:19 +08:00
|
|
|
enum E e = (enum E)value();
|
2017-10-16 04:13:17 +08:00
|
|
|
|
|
|
|
if (e == yes)
|
|
|
|
return 0;
|
|
|
|
if (e != yes)
|
|
|
|
return 0;
|
|
|
|
if (e < yes)
|
|
|
|
return 0;
|
|
|
|
if (e <= yes)
|
|
|
|
return 0;
|
|
|
|
if (e > yes)
|
|
|
|
return 0;
|
|
|
|
if (e >= yes)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (yes == e)
|
|
|
|
return 0;
|
|
|
|
if (yes != e)
|
|
|
|
return 0;
|
|
|
|
if (yes < e)
|
|
|
|
return 0;
|
|
|
|
if (yes <= e)
|
|
|
|
return 0;
|
|
|
|
if (yes > e)
|
|
|
|
return 0;
|
|
|
|
if (yes >= e)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (e == maybe)
|
|
|
|
return 0;
|
|
|
|
if (e != maybe)
|
|
|
|
return 0;
|
|
|
|
if (e < maybe)
|
|
|
|
return 0;
|
|
|
|
if (e <= maybe)
|
|
|
|
return 0;
|
|
|
|
if (e > maybe)
|
|
|
|
return 0;
|
|
|
|
if (e >= maybe)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (maybe == e)
|
|
|
|
return 0;
|
|
|
|
if (maybe != e)
|
|
|
|
return 0;
|
|
|
|
if (maybe < e)
|
|
|
|
return 0;
|
|
|
|
if (maybe <= e)
|
|
|
|
return 0;
|
|
|
|
if (maybe > e)
|
|
|
|
return 0;
|
|
|
|
if (maybe >= e)
|
|
|
|
return 0;
|
|
|
|
|
2017-12-09 06:57:11 +08:00
|
|
|
// For the time being, use the declared type of bit-fields rather than their
|
|
|
|
// length when determining whether a value is in-range.
|
|
|
|
// FIXME: Reconsider this.
|
|
|
|
struct A {
|
|
|
|
int a : 3;
|
|
|
|
unsigned b : 3;
|
|
|
|
long c : 3;
|
|
|
|
unsigned long d : 3;
|
|
|
|
} a;
|
|
|
|
if (a.a < 3) {}
|
|
|
|
if (a.a < 4) {}
|
|
|
|
if (a.b < 7) {}
|
|
|
|
if (a.b < 8) {}
|
|
|
|
if (a.c < 3) {}
|
|
|
|
if (a.c < 4) {}
|
|
|
|
if (a.d < 7) {}
|
|
|
|
if (a.d < 8) {}
|
|
|
|
|
2017-10-16 04:13:17 +08:00
|
|
|
return 1;
|
|
|
|
}
|