cast-overflow.cpp revision 909fff81b83df049ecc6e02407394640435d7bef
1// FIXME: run this (and other) UBSan tests in both 32- and 64-bit modes (?).
2// RUN: %clangxx -fsanitize=float-cast-overflow %s -o %t
3// RUN: %run %t _
4// RUN: %run %t 0 2>&1 | FileCheck %s --check-prefix=CHECK-0
5// RUN: %run %t 1 2>&1 | FileCheck %s --check-prefix=CHECK-1
6// RUN: %run %t 2 2>&1 | FileCheck %s --check-prefix=CHECK-2
7// RUN: %run %t 3 2>&1 | FileCheck %s --check-prefix=CHECK-3
8// RUN: %run %t 4 2>&1 | FileCheck %s --check-prefix=CHECK-4
9// RUN: %run %t 5 2>&1 | FileCheck %s --check-prefix=CHECK-5
10// RUN: %run %t 6 2>&1 | FileCheck %s --check-prefix=CHECK-6
11// FIXME: %run %t 7 2>&1 | FileCheck %s --check-prefix=CHECK-7
12// FIXME: not %run %t 8 2>&1 | FileCheck %s --check-prefix=CHECK-8
13// RUN: not %run %t 9 2>&1 | FileCheck %s --check-prefix=CHECK-9
14
15// This test assumes float and double are IEEE-754 single- and double-precision.
16
17#if defined(__APPLE__)
18# include <machine/endian.h>
19# define BYTE_ORDER __DARWIN_BYTE_ORDER
20# define BIG_ENDIAN __DARWIN_BIG_ENDIAN
21# define LITTLE_ENDIAN __DARWIN_LITTLE_ENDIAN
22#elif defined(__FreeBSD__)
23# include <sys/endian.h>
24# define BYTE_ORDER _BYTE_ORDER
25# define BIG_ENDIAN _BIG_ENDIAN
26# define LITTLE_ENDIAN _LITTLE_ENDIAN
27#else
28# include <endian.h>
29# define BYTE_ORDER __BYTE_ORDER
30# define BIG_ENDIAN __BIG_ENDIAN
31# define LITTLE_ENDIAN __LITTLE_ENDIAN
32#endif  // __APPLE__
33#include <stdint.h>
34#include <stdio.h>
35#include <string.h>
36
37float Inf;
38float NaN;
39
40int main(int argc, char **argv) {
41  float MaxFloatRepresentableAsInt = 0x7fffff80;
42  (int)MaxFloatRepresentableAsInt; // ok
43  (int)-MaxFloatRepresentableAsInt; // ok
44
45  float MinFloatRepresentableAsInt = -0x7fffffff - 1;
46  (int)MinFloatRepresentableAsInt; // ok
47
48  float MaxFloatRepresentableAsUInt = 0xffffff00u;
49  (unsigned int)MaxFloatRepresentableAsUInt; // ok
50
51#ifdef __SIZEOF_INT128__
52  unsigned __int128 FloatMaxAsUInt128 = -((unsigned __int128)1 << 104);
53  (void)(float)FloatMaxAsUInt128; // ok
54#endif
55
56  float NearlyMinusOne = -0.99999;
57  unsigned Zero = NearlyMinusOne; // ok
58
59  // Build a '+Inf'.
60#if BYTE_ORDER == LITTLE_ENDIAN
61  char InfVal[] = { 0x00, 0x00, 0x80, 0x7f };
62#else
63  char InfVal[] = { 0x7f, 0x80, 0x00, 0x00 };
64#endif
65  float Inf;
66  memcpy(&Inf, InfVal, 4);
67
68  // Build a 'NaN'.
69#if BYTE_ORDER == LITTLE_ENDIAN
70  char NaNVal[] = { 0x01, 0x00, 0x80, 0x7f };
71#else
72  char NaNVal[] = { 0x7f, 0x80, 0x00, 0x01 };
73#endif
74  float NaN;
75  memcpy(&NaN, NaNVal, 4);
76
77  double DblInf = (double)Inf; // ok
78
79  switch (argv[1][0]) {
80    // FIXME: Produce a source location for these checks and test for it here.
81
82    // Floating point -> integer overflow.
83  case '0': {
84    // Note that values between 0x7ffffe00 and 0x80000000 may or may not
85    // successfully round-trip, depending on the rounding mode.
86    // CHECK-0: runtime error: value 2.14748{{.*}} is outside the range of representable values of type 'int'
87    static int test_int = MaxFloatRepresentableAsInt + 0x80;
88    return 0;
89    }
90  case '1':
91    // CHECK-1: runtime error: value -2.14748{{.*}} is outside the range of representable values of type 'int'
92    return MinFloatRepresentableAsInt - 0x100;
93  case '2': {
94    // CHECK-2: runtime error: value -1 is outside the range of representable values of type 'unsigned int'
95    volatile float f = -1.0;
96    volatile unsigned u = (unsigned)f;
97    return 0;
98  }
99  case '3': {
100    // CHECK-3: runtime error: value 4.2949{{.*}} is outside the range of representable values of type 'unsigned int'
101    static int test_int = (unsigned)(MaxFloatRepresentableAsUInt + 0x100);
102    return 0;
103  }
104
105  case '4': {
106    // CHECK-4: runtime error: value {{.*}} is outside the range of representable values of type 'int'
107    static int test_int = Inf;
108    return 0;
109  }
110  case '5':
111    // CHECK-5: runtime error: value {{.*}} is outside the range of representable values of type 'int'
112    return NaN;
113
114    // Integer -> floating point overflow.
115  case '6': {
116    // CHECK-6: {{runtime error: value 0xffffff00000000000000000000000001 is outside the range of representable values of type 'float'|__int128 not supported}}
117#ifdef __SIZEOF_INT128__
118    static int test_int = (float)(FloatMaxAsUInt128 + 1);
119    return 0;
120#else
121    puts("__int128 not supported");
122    return 0;
123#endif
124  }
125  // FIXME: The backend cannot lower __fp16 operations on x86 yet.
126  //case '7':
127  //  (__fp16)65504; // ok
128  //  // CHECK-7: runtime error: value 65505 is outside the range of representable values of type '__fp16'
129  //  return (__fp16)65505;
130
131    // Floating point -> floating point overflow.
132  case '8':
133    // CHECK-8: runtime error: value 1e+39 is outside the range of representable values of type 'float'
134    return (float)1e39;
135  case '9':
136    volatile long double ld = 300.0;
137    // CHECK-9: runtime error: value 300 is outside the range of representable values of type 'char'
138    char c = ld;
139    return c;
140  }
141}
142