2016-04-22 05:04:55 +08:00
|
|
|
// RUN: %clang_cc1 -verify -fsyntax-only -triple x86_64-pc-linux-gnu %s -Wno-literal-conversion -Wfloat-conversion -DFLOAT_CONVERSION -DZERO -DBOOL -DCONSTANT_BOOL -DOVERFLOW
|
|
|
|
// RUN: %clang_cc1 -verify -fsyntax-only -triple x86_64-pc-linux-gnu %s -Wno-conversion -Wfloat-overflow-conversion -DOVERFLOW
|
|
|
|
// RUN: %clang_cc1 -verify -fsyntax-only -triple x86_64-pc-linux-gnu %s -Wno-conversion -Wfloat-zero-conversion -DZERO
|
2014-04-22 09:01:05 +08:00
|
|
|
|
2016-04-22 05:04:55 +08:00
|
|
|
float ReturnFloat();
|
|
|
|
|
|
|
|
#ifdef FLOAT_CONVERSION
|
2014-04-22 09:01:05 +08:00
|
|
|
bool ReturnBool(float f) {
|
|
|
|
return f; //expected-warning{{conversion}}
|
|
|
|
}
|
|
|
|
|
|
|
|
char ReturnChar(float f) {
|
|
|
|
return f; //expected-warning{{conversion}}
|
|
|
|
}
|
|
|
|
|
|
|
|
int ReturnInt(float f) {
|
|
|
|
return f; //expected-warning{{conversion}}
|
|
|
|
}
|
|
|
|
|
|
|
|
long ReturnLong(float f) {
|
|
|
|
return f; //expected-warning{{conversion}}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Convert(float f, double d, long double ld) {
|
|
|
|
bool b;
|
|
|
|
char c;
|
|
|
|
int i;
|
|
|
|
long l;
|
|
|
|
|
|
|
|
b = f; //expected-warning{{conversion}}
|
|
|
|
b = d; //expected-warning{{conversion}}
|
|
|
|
b = ld; //expected-warning{{conversion}}
|
|
|
|
c = f; //expected-warning{{conversion}}
|
|
|
|
c = d; //expected-warning{{conversion}}
|
|
|
|
c = ld; //expected-warning{{conversion}}
|
|
|
|
i = f; //expected-warning{{conversion}}
|
|
|
|
i = d; //expected-warning{{conversion}}
|
|
|
|
i = ld; //expected-warning{{conversion}}
|
|
|
|
l = f; //expected-warning{{conversion}}
|
|
|
|
l = d; //expected-warning{{conversion}}
|
|
|
|
l = ld; //expected-warning{{conversion}}
|
|
|
|
}
|
|
|
|
|
2018-08-14 00:38:07 +08:00
|
|
|
void CompoundAssignment() {
|
|
|
|
int x = 3;
|
|
|
|
|
2019-02-15 06:48:01 +08:00
|
|
|
x += 1.234; // expected-warning {{implicit conversion turns floating-point number into integer: 'double' to 'int'}}
|
|
|
|
x -= -0.0; // expected-warning {{implicit conversion turns floating-point number into integer: 'double' to 'int'}}
|
|
|
|
x *= 1.1f; // expected-warning {{implicit conversion turns floating-point number into integer: 'float' to 'int'}}
|
|
|
|
x /= -2.2f; // expected-warning {{implicit conversion turns floating-point number into integer: 'float' to 'int'}}
|
2018-08-14 00:38:07 +08:00
|
|
|
|
2019-02-15 06:48:01 +08:00
|
|
|
int y = x += 1.4f; // expected-warning {{implicit conversion turns floating-point number into integer: 'float' to 'int'}}
|
2018-08-14 00:38:07 +08:00
|
|
|
|
|
|
|
float z = 1.1f;
|
|
|
|
double w = -2.2;
|
|
|
|
|
2019-02-15 06:48:01 +08:00
|
|
|
y += z + w; // expected-warning {{implicit conversion turns floating-point number into integer: 'double' to 'int'}}
|
2018-08-14 00:38:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
# 1 "foo.h" 3
|
|
|
|
// ^ the following text comes from a system header file.
|
|
|
|
#define SYSTEM_MACRO_FLOAT(x) do { (x) += 1.1; } while(0)
|
|
|
|
# 1 "warn-float-conversion.cpp" 1
|
|
|
|
// ^ start of a new file.
|
|
|
|
void SystemMacro() {
|
|
|
|
float x = 0.0f;
|
|
|
|
SYSTEM_MACRO_FLOAT(x);
|
|
|
|
}
|
|
|
|
|
2016-04-22 05:04:55 +08:00
|
|
|
void Test() {
|
|
|
|
int a1 = 10.0/2.0; //expected-warning{{conversion}}
|
|
|
|
int a2 = 1.0/2.0; //expected-warning{{conversion}}
|
|
|
|
bool a3 = ReturnFloat(); //expected-warning{{conversion}}
|
|
|
|
int a4 = 1e30 + 1; //expected-warning{{conversion}}
|
|
|
|
}
|
|
|
|
|
|
|
|
void TestConstantFloat() {
|
|
|
|
// Don't warn on exact floating literals.
|
|
|
|
int a1 = 5.0;
|
|
|
|
int a2 = 1e3;
|
|
|
|
|
|
|
|
int a3 = 5.5; // caught by -Wliteral-conversion
|
|
|
|
int a4 = 500.44; // caught by -Wliteral-convserion
|
|
|
|
|
|
|
|
int b1 = 5.0 / 1.0; //expected-warning{{conversion}}
|
|
|
|
int b2 = 5.0 / 2.0; //expected-warning{{conversion}}
|
|
|
|
|
|
|
|
const float five = 5.0;
|
|
|
|
|
|
|
|
int b3 = five / 1.0; //expected-warning{{conversion}}
|
|
|
|
int b4 = five / 2.0; //expected-warning{{conversion}}
|
2018-05-09 05:26:21 +08:00
|
|
|
|
|
|
|
int f = 2147483646.5 + 1; // expected-warning{{implicit conversion from 'double' to 'int' changes value from 2147483647.5 to 2147483647}}
|
|
|
|
unsigned g = -.5 + .01; // expected-warning{{implicit conversion from 'double' to 'unsigned int' changes non-zero value from -0.49 to 0}}
|
2016-04-22 05:04:55 +08:00
|
|
|
}
|
|
|
|
#endif // FLOAT_CONVERSION
|
|
|
|
|
|
|
|
#ifdef ZERO
|
|
|
|
void TestZero() {
|
|
|
|
const float half = .5;
|
|
|
|
int a1 = half; // expected-warning{{implicit conversion from 'const float' to 'int' changes non-zero value from 0.5 to 0}}
|
|
|
|
int a2 = 1.0 / 2.0; // expected-warning{{implicit conversion from 'double' to 'int' changes non-zero value from 0.5 to 0}}
|
|
|
|
int a3 = 5;
|
|
|
|
}
|
|
|
|
#endif // ZERO
|
|
|
|
|
|
|
|
#ifdef OVERFLOW
|
|
|
|
void TestOverflow() {
|
|
|
|
char a = 500.0; // caught by -Wliteral-conversion
|
|
|
|
char b = -500.0; // caught by -Wliteral-conversion
|
|
|
|
|
|
|
|
const float LargeNumber = 1024;
|
2018-05-08 04:52:56 +08:00
|
|
|
char c = LargeNumber; // expected-warning{{implicit conversion of out of range value from 'const float' to 'char' is undefined}}
|
|
|
|
char d = 400.0 + 400.0; // expected-warning{{implicit conversion of out of range value from 'double' to 'char' is undefined}}
|
2016-04-22 05:04:55 +08:00
|
|
|
|
2018-05-08 04:52:56 +08:00
|
|
|
char e = 1.0 / 0.0; // expected-warning{{implicit conversion of out of range value from 'double' to 'char' is undefined}}
|
2016-04-22 05:04:55 +08:00
|
|
|
}
|
2018-05-15 07:21:48 +08:00
|
|
|
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
class Check {
|
|
|
|
public:
|
|
|
|
static constexpr bool Safe();
|
|
|
|
};
|
|
|
|
|
|
|
|
template<>
|
|
|
|
constexpr bool Check<char>::Safe() { return false; }
|
|
|
|
|
|
|
|
template<>
|
|
|
|
constexpr bool Check<float>::Safe() { return true; }
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
T run1(T t) {
|
|
|
|
const float ret = 800;
|
|
|
|
return ret; // expected-warning {{implicit conversion of out of range value from 'const float' to 'char' is undefined}}
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
T run2(T t) {
|
|
|
|
const float ret = 800;
|
|
|
|
if (Check<T>::Safe())
|
|
|
|
return ret;
|
|
|
|
else
|
|
|
|
return t;
|
|
|
|
}
|
|
|
|
|
|
|
|
void test() {
|
|
|
|
float a = run1(a) + run2(a);
|
|
|
|
char b = run1(b) + run2(b); // expected-note {{in instantiation of function template specialization 'run1<char>' requested here}}
|
|
|
|
}
|
|
|
|
|
2016-04-22 05:04:55 +08:00
|
|
|
#endif // OVERFLOW
|