blob: 1551bf593df192885884120d358e69bcb8fd23b8 [file] [log] [blame]
Pirama Arumuga Nainar799172d2016-03-03 15:50:30 -08001// RUN: %clangxx -fsanitize=float-cast-overflow %s -o %t
Stephen Hines2d1fdb22014-05-28 23:58:16 -07002// RUN: %run %t _
Pirama Arumuga Nainar799172d2016-03-03 15:50:30 -08003// RUN: %env_ubsan_opts=print_summary=1:report_error_type=1 %run %t 0 2>&1 | FileCheck %s --check-prefix=CHECK-0
Stephen Hines2d1fdb22014-05-28 23:58:16 -07004// RUN: %run %t 1 2>&1 | FileCheck %s --check-prefix=CHECK-1
5// RUN: %run %t 2 2>&1 | FileCheck %s --check-prefix=CHECK-2
6// RUN: %run %t 3 2>&1 | FileCheck %s --check-prefix=CHECK-3
7// RUN: %run %t 4 2>&1 | FileCheck %s --check-prefix=CHECK-4
8// RUN: %run %t 5 2>&1 | FileCheck %s --check-prefix=CHECK-5
9// RUN: %run %t 6 2>&1 | FileCheck %s --check-prefix=CHECK-6
10// FIXME: %run %t 7 2>&1 | FileCheck %s --check-prefix=CHECK-7
Stephen Hines6d186232014-11-26 17:56:19 -080011// FIXME: not %run %t 8 2>&1 | FileCheck %s --check-prefix=CHECK-8
12// RUN: not %run %t 9 2>&1 | FileCheck %s --check-prefix=CHECK-9
Stephen Hines2d1fdb22014-05-28 23:58:16 -070013
14// This test assumes float and double are IEEE-754 single- and double-precision.
15
Stephen Hines6d186232014-11-26 17:56:19 -080016#if defined(__APPLE__)
17# include <machine/endian.h>
18# define BYTE_ORDER __DARWIN_BYTE_ORDER
19# define BIG_ENDIAN __DARWIN_BIG_ENDIAN
20# define LITTLE_ENDIAN __DARWIN_LITTLE_ENDIAN
21#elif defined(__FreeBSD__)
22# include <sys/endian.h>
23# define BYTE_ORDER _BYTE_ORDER
24# define BIG_ENDIAN _BIG_ENDIAN
25# define LITTLE_ENDIAN _LITTLE_ENDIAN
Pirama Arumuga Nainar799172d2016-03-03 15:50:30 -080026#elif defined(_WIN32)
27# define BYTE_ORDER 0
28# define BIG_ENDIAN 1
29# define LITTLE_ENDIAN 0
Stephen Hines6d186232014-11-26 17:56:19 -080030#else
31# include <endian.h>
32# define BYTE_ORDER __BYTE_ORDER
33# define BIG_ENDIAN __BIG_ENDIAN
34# define LITTLE_ENDIAN __LITTLE_ENDIAN
35#endif // __APPLE__
Stephen Hines2d1fdb22014-05-28 23:58:16 -070036#include <stdint.h>
37#include <stdio.h>
38#include <string.h>
39
40float Inf;
41float NaN;
42
43int main(int argc, char **argv) {
44 float MaxFloatRepresentableAsInt = 0x7fffff80;
45 (int)MaxFloatRepresentableAsInt; // ok
46 (int)-MaxFloatRepresentableAsInt; // ok
47
48 float MinFloatRepresentableAsInt = -0x7fffffff - 1;
49 (int)MinFloatRepresentableAsInt; // ok
50
51 float MaxFloatRepresentableAsUInt = 0xffffff00u;
52 (unsigned int)MaxFloatRepresentableAsUInt; // ok
53
54#ifdef __SIZEOF_INT128__
55 unsigned __int128 FloatMaxAsUInt128 = -((unsigned __int128)1 << 104);
56 (void)(float)FloatMaxAsUInt128; // ok
57#endif
58
59 float NearlyMinusOne = -0.99999;
60 unsigned Zero = NearlyMinusOne; // ok
61
62 // Build a '+Inf'.
Stephen Hines6d186232014-11-26 17:56:19 -080063#if BYTE_ORDER == LITTLE_ENDIAN
Stephen Hines2d1fdb22014-05-28 23:58:16 -070064 char InfVal[] = { 0x00, 0x00, 0x80, 0x7f };
Stephen Hines6d186232014-11-26 17:56:19 -080065#else
66 char InfVal[] = { 0x7f, 0x80, 0x00, 0x00 };
67#endif
Stephen Hines2d1fdb22014-05-28 23:58:16 -070068 float Inf;
69 memcpy(&Inf, InfVal, 4);
70
71 // Build a 'NaN'.
Stephen Hines6d186232014-11-26 17:56:19 -080072#if BYTE_ORDER == LITTLE_ENDIAN
Stephen Hines2d1fdb22014-05-28 23:58:16 -070073 char NaNVal[] = { 0x01, 0x00, 0x80, 0x7f };
Stephen Hines6d186232014-11-26 17:56:19 -080074#else
75 char NaNVal[] = { 0x7f, 0x80, 0x00, 0x01 };
76#endif
Stephen Hines2d1fdb22014-05-28 23:58:16 -070077 float NaN;
78 memcpy(&NaN, NaNVal, 4);
79
80 double DblInf = (double)Inf; // ok
81
82 switch (argv[1][0]) {
83 // FIXME: Produce a source location for these checks and test for it here.
84
85 // Floating point -> integer overflow.
Pirama Arumuga Nainar259f7062015-05-06 11:49:53 -070086 case '0': {
Stephen Hines2d1fdb22014-05-28 23:58:16 -070087 // Note that values between 0x7ffffe00 and 0x80000000 may or may not
88 // successfully round-trip, depending on the rounding mode.
Pirama Arumuga Nainar799172d2016-03-03 15:50:30 -080089 // CHECK-0: {{.*}}cast-overflow.cpp:[[@LINE+1]]:27: runtime error: value 2.14748{{.*}} is outside the range of representable values of type 'int'
Pirama Arumuga Nainar259f7062015-05-06 11:49:53 -070090 static int test_int = MaxFloatRepresentableAsInt + 0x80;
Pirama Arumuga Nainar799172d2016-03-03 15:50:30 -080091 // CHECK-0: SUMMARY: {{.*}}Sanitizer: float-cast-overflow {{.*}}cast-overflow.cpp:[[@LINE-1]]
Pirama Arumuga Nainar259f7062015-05-06 11:49:53 -070092 return 0;
93 }
Pirama Arumuga Nainarcdce50b2015-07-01 12:26:56 -070094 case '1': {
Pirama Arumuga Nainar799172d2016-03-03 15:50:30 -080095 // CHECK-1: {{.*}}cast-overflow.cpp:[[@LINE+1]]:27: runtime error: value -2.14748{{.*}} is outside the range of representable values of type 'int'
Pirama Arumuga Nainarcdce50b2015-07-01 12:26:56 -070096 static int test_int = MinFloatRepresentableAsInt - 0x100;
97 return 0;
98 }
Stephen Hines6d186232014-11-26 17:56:19 -080099 case '2': {
Pirama Arumuga Nainar799172d2016-03-03 15:50:30 -0800100 // CHECK-2: {{.*}}cast-overflow.cpp:[[@LINE+2]]:37: runtime error: value -1 is outside the range of representable values of type 'unsigned int'
Stephen Hines6d186232014-11-26 17:56:19 -0800101 volatile float f = -1.0;
102 volatile unsigned u = (unsigned)f;
103 return 0;
104 }
Pirama Arumuga Nainar259f7062015-05-06 11:49:53 -0700105 case '3': {
Pirama Arumuga Nainar799172d2016-03-03 15:50:30 -0800106 // CHECK-3: {{.*}}cast-overflow.cpp:[[@LINE+1]]:37: runtime error: value 4.2949{{.*}} is outside the range of representable values of type 'unsigned int'
Pirama Arumuga Nainar259f7062015-05-06 11:49:53 -0700107 static int test_int = (unsigned)(MaxFloatRepresentableAsUInt + 0x100);
108 return 0;
109 }
Stephen Hines2d1fdb22014-05-28 23:58:16 -0700110
Pirama Arumuga Nainar259f7062015-05-06 11:49:53 -0700111 case '4': {
Pirama Arumuga Nainar799172d2016-03-03 15:50:30 -0800112 // CHECK-4: {{.*}}cast-overflow.cpp:[[@LINE+1]]:27: runtime error: value {{.*}} is outside the range of representable values of type 'int'
Pirama Arumuga Nainar259f7062015-05-06 11:49:53 -0700113 static int test_int = Inf;
114 return 0;
115 }
Pirama Arumuga Nainarcdce50b2015-07-01 12:26:56 -0700116 case '5': {
Pirama Arumuga Nainar799172d2016-03-03 15:50:30 -0800117 // CHECK-5: {{.*}}cast-overflow.cpp:[[@LINE+1]]:27: runtime error: value {{.*}} is outside the range of representable values of type 'int'
Pirama Arumuga Nainarcdce50b2015-07-01 12:26:56 -0700118 static int test_int = NaN;
119 return 0;
120 }
Stephen Hines2d1fdb22014-05-28 23:58:16 -0700121
122 // Integer -> floating point overflow.
Pirama Arumuga Nainar259f7062015-05-06 11:49:53 -0700123 case '6': {
Pirama Arumuga Nainar799172d2016-03-03 15:50:30 -0800124 // CHECK-6: cast-overflow.cpp:[[@LINE+2]]:{{34: runtime error: value 0xffffff00000000000000000000000001 is outside the range of representable values of type 'float'| __int128 not supported}}
125#if defined(__SIZEOF_INT128__) && !defined(_WIN32)
Pirama Arumuga Nainar259f7062015-05-06 11:49:53 -0700126 static int test_int = (float)(FloatMaxAsUInt128 + 1);
127 return 0;
Stephen Hines2d1fdb22014-05-28 23:58:16 -0700128#else
Pirama Arumuga Nainar799172d2016-03-03 15:50:30 -0800129 // Print the same line as the check above. That way the test is robust to
130 // line changes around it
131 printf("%s:%d: __int128 not supported", __FILE__, __LINE__ - 5);
Stephen Hines2d1fdb22014-05-28 23:58:16 -0700132 return 0;
133#endif
Pirama Arumuga Nainar259f7062015-05-06 11:49:53 -0700134 }
Stephen Hines2d1fdb22014-05-28 23:58:16 -0700135 // FIXME: The backend cannot lower __fp16 operations on x86 yet.
136 //case '7':
137 // (__fp16)65504; // ok
138 // // CHECK-7: runtime error: value 65505 is outside the range of representable values of type '__fp16'
139 // return (__fp16)65505;
140
141 // Floating point -> floating point overflow.
142 case '8':
Pirama Arumuga Nainar799172d2016-03-03 15:50:30 -0800143 // CHECK-8: {{.*}}cast-overflow.cpp:[[@LINE+1]]:19: runtime error: value 1e+39 is outside the range of representable values of type 'float'
Stephen Hines2d1fdb22014-05-28 23:58:16 -0700144 return (float)1e39;
145 case '9':
146 volatile long double ld = 300.0;
Pirama Arumuga Nainar799172d2016-03-03 15:50:30 -0800147 // CHECK-9: {{.*}}cast-overflow.cpp:[[@LINE+1]]:14: runtime error: value 300 is outside the range of representable values of type 'char'
Stephen Hines2d1fdb22014-05-28 23:58:16 -0700148 char c = ld;
149 return c;
150 }
151}