new-overflow.cpp revision 88c1dfccf8c5d487408e3365c752ed24c867c7ce
1// RUN: %clang_cc1 -triple i386-unknown-unknown %s -emit-llvm -o - | FileCheck %s 2 3// rdar://problem/9246208 4 5// CHECK: [[OVR_T:%.*]] = type { i32, i1 } 6 7// Basic test. 8namespace test0 { 9 struct A { 10 A(); 11 int x; 12 }; 13 14 typedef A elt; 15 16 // CHECK: define [[A:%.*]]* @_ZN5test04testEs(i16 signext 17 // CHECK: [[N:%.*]] = sext i16 {{%.*}} to i32 18 // CHECK-NEXT: [[T0:%.*]] = call [[OVR_T]] @llvm.umul.with.overflow.i32(i32 [[N]], i32 4) 19 // CHECK-NEXT: [[T1:%.*]] = extractvalue [[OVR_T]] [[T0]], 1 20 // CHECK-NEXT: [[T2:%.*]] = extractvalue [[OVR_T]] [[T0]], 0 21 // CHECK-NEXT: [[T3:%.*]] = select i1 [[T1]], i32 -1, i32 [[T2]] 22 // CHECK-NEXT: call noalias i8* @_Znaj(i32 [[T3]]) 23 // CHECK: icmp ult i32 {{.*}}, [[N]] 24 elt *test(short s) { 25 return new elt[s]; 26 } 27} 28 29// test0 with a nested array. 30namespace test1 { 31 struct A { 32 A(); 33 int x; 34 }; 35 36 typedef A elt[100]; 37 38 // CHECK: define [100 x [[A:%.*]]]* @_ZN5test14testEs(i16 signext 39 // CHECK: [[N:%.*]] = sext i16 {{%.*}} to i32 40 // CHECK-NEXT: [[T0:%.*]] = call [[OVR_T]] @llvm.umul.with.overflow.i32(i32 [[N]], i32 400) 41 // CHECK-NEXT: [[T1:%.*]] = extractvalue [[OVR_T]] [[T0]], 1 42 // CHECK-NEXT: [[T2:%.*]] = extractvalue [[OVR_T]] [[T0]], 0 43 // CHECK-NEXT: [[T3:%.*]] = mul i32 [[N]], 100 44 // CHECK-NEXT: [[T4:%.*]] = select i1 [[T1]], i32 -1, i32 [[T2]] 45 // CHECK-NEXT: call noalias i8* @_Znaj(i32 [[T4]]) 46 // CHECK: icmp ult i32 {{.*}}, [[T3]] 47 elt *test(short s) { 48 return new elt[s]; 49 } 50} 51 52// test1 with an array cookie. 53namespace test2 { 54 struct A { 55 A(); 56 ~A(); 57 int x; 58 }; 59 60 typedef A elt[100]; 61 62 // CHECK: define [100 x [[A:%.*]]]* @_ZN5test24testEs(i16 signext 63 // CHECK: [[N:%.*]] = sext i16 {{%.*}} to i32 64 // CHECK-NEXT: [[T0:%.*]] = call [[OVR_T]] @llvm.umul.with.overflow.i32(i32 [[N]], i32 400) 65 // CHECK-NEXT: [[T1:%.*]] = extractvalue [[OVR_T]] [[T0]], 1 66 // CHECK-NEXT: [[T2:%.*]] = extractvalue [[OVR_T]] [[T0]], 0 67 // CHECK-NEXT: [[T3:%.*]] = mul i32 [[N]], 100 68 // CHECK-NEXT: [[T4:%.*]] = call [[OVR_T]] @llvm.uadd.with.overflow.i32(i32 [[T2]], i32 4) 69 // CHECK-NEXT: [[T5:%.*]] = extractvalue [[OVR_T]] [[T4]], 1 70 // CHECK-NEXT: [[T6:%.*]] = or i1 [[T1]], [[T5]] 71 // CHECK-NEXT: [[T7:%.*]] = extractvalue [[OVR_T]] [[T4]], 0 72 // CHECK-NEXT: [[T8:%.*]] = select i1 [[T6]], i32 -1, i32 [[T7]] 73 // CHECK-NEXT: call noalias i8* @_Znaj(i32 [[T8]]) 74 // CHECK: icmp ult i32 {{.*}}, [[T3]] 75 elt *test(short s) { 76 return new elt[s]; 77 } 78} 79 80// test0 with a 1-byte element. 81namespace test4 { 82 struct A { 83 A(); 84 }; 85 86 typedef A elt; 87 88 // CHECK: define [[A:%.*]]* @_ZN5test44testEs(i16 signext 89 // CHECK: [[N:%.*]] = sext i16 {{%.*}} to i32 90 // CHECK-NEXT: [[T0:%.*]] = icmp slt i32 [[N]], 0 91 // CHECK-NEXT: [[T1:%.*]] = select i1 [[T0]], i32 -1, i32 [[N]] 92 // CHECK-NEXT: call noalias i8* @_Znaj(i32 [[T1]]) 93 // CHECK: icmp ult i32 {{.*}}, [[N]] 94 elt *test(short s) { 95 return new elt[s]; 96 } 97} 98 99// test4 with no sext required. 100namespace test5 { 101 struct A { 102 A(); 103 }; 104 105 typedef A elt; 106 107 // CHECK: define [[A:%.*]]* @_ZN5test54testEi(i32 108 // CHECK: [[N:%.*]] = load i32* 109 // CHECK-NEXT: [[T0:%.*]] = icmp slt i32 [[N]], 0 110 // CHECK-NEXT: [[T1:%.*]] = select i1 [[T0]], i32 -1, i32 [[N]] 111 // CHECK-NEXT: call noalias i8* @_Znaj(i32 [[T1]]) 112 // CHECK: icmp ult i32 {{.*}}, [[N]] 113 elt *test(int s) { 114 return new elt[s]; 115 } 116} 117 118// test0 with an unsigned size. 119namespace test6 { 120 struct A { 121 A(); 122 int x; 123 }; 124 125 typedef A elt; 126 127 // CHECK: define [[A:%.*]]* @_ZN5test64testEt(i16 zeroext 128 // CHECK: [[N:%.*]] = zext i16 {{%.*}} to i32 129 // CHECK-NEXT: [[T0:%.*]] = call [[OVR_T]] @llvm.umul.with.overflow.i32(i32 [[N]], i32 4) 130 // CHECK-NEXT: [[T1:%.*]] = extractvalue [[OVR_T]] [[T0]], 1 131 // CHECK-NEXT: [[T2:%.*]] = extractvalue [[OVR_T]] [[T0]], 0 132 // CHECK-NEXT: [[T3:%.*]] = select i1 [[T1]], i32 -1, i32 [[T2]] 133 // CHECK-NEXT: call noalias i8* @_Znaj(i32 [[T3]]) 134 // CHECK: icmp ult i32 {{.*}}, [[N]] 135 elt *test(unsigned short s) { 136 return new elt[s]; 137 } 138} 139 140// test1 with an unsigned size. 141namespace test7 { 142 struct A { 143 A(); 144 int x; 145 }; 146 147 typedef A elt[100]; 148 149 // CHECK: define [100 x [[A:%.*]]]* @_ZN5test74testEt(i16 zeroext 150 // CHECK: [[N:%.*]] = zext i16 {{%.*}} to i32 151 // CHECK-NEXT: [[T0:%.*]] = call [[OVR_T]] @llvm.umul.with.overflow.i32(i32 [[N]], i32 400) 152 // CHECK-NEXT: [[T1:%.*]] = extractvalue [[OVR_T]] [[T0]], 1 153 // CHECK-NEXT: [[T2:%.*]] = extractvalue [[OVR_T]] [[T0]], 0 154 // CHECK-NEXT: [[T3:%.*]] = mul i32 [[N]], 100 155 // CHECK-NEXT: [[T4:%.*]] = select i1 [[T1]], i32 -1, i32 [[T2]] 156 // CHECK-NEXT: call noalias i8* @_Znaj(i32 [[T4]]) 157 // CHECK: icmp ult i32 {{.*}}, [[T3]] 158 elt *test(unsigned short s) { 159 return new elt[s]; 160 } 161} 162 163// test0 with a signed type larger than size_t. 164namespace test8 { 165 struct A { 166 A(); 167 int x; 168 }; 169 170 typedef A elt; 171 172 // CHECK: define [[A:%.*]]* @_ZN5test84testEx(i64 173 // CHECK: [[N:%.*]] = load i64* 174 // CHECK-NEXT: [[T0:%.*]] = icmp uge i64 [[N]], 4294967296 175 // CHECK-NEXT: [[T1:%.*]] = trunc i64 [[N]] to i32 176 // CHECK-NEXT: [[T2:%.*]] = call [[OVR_T]] @llvm.umul.with.overflow.i32(i32 [[T1]], i32 4) 177 // CHECK-NEXT: [[T3:%.*]] = extractvalue [[OVR_T]] [[T2]], 1 178 // CHECK-NEXT: [[T4:%.*]] = or i1 [[T0]], [[T3]] 179 // CHECK-NEXT: [[T5:%.*]] = extractvalue [[OVR_T]] [[T2]], 0 180 // CHECK-NEXT: [[T6:%.*]] = select i1 [[T4]], i32 -1, i32 [[T5]] 181 // CHECK-NEXT: call noalias i8* @_Znaj(i32 [[T6]]) 182 // CHECK: icmp ult i32 {{.*}}, [[T1]] 183 elt *test(long long s) { 184 return new elt[s]; 185 } 186} 187 188// test8 with an unsigned type. 189namespace test9 { 190 struct A { 191 A(); 192 int x; 193 }; 194 195 typedef A elt; 196 197 // CHECK: define [[A:%.*]]* @_ZN5test94testEy(i64 198 // CHECK: [[N:%.*]] = load i64* 199 // CHECK-NEXT: [[T0:%.*]] = icmp uge i64 [[N]], 4294967296 200 // CHECK-NEXT: [[T1:%.*]] = trunc i64 [[N]] to i32 201 // CHECK-NEXT: [[T2:%.*]] = call [[OVR_T]] @llvm.umul.with.overflow.i32(i32 [[T1]], i32 4) 202 // CHECK-NEXT: [[T3:%.*]] = extractvalue [[OVR_T]] [[T2]], 1 203 // CHECK-NEXT: [[T4:%.*]] = or i1 [[T0]], [[T3]] 204 // CHECK-NEXT: [[T5:%.*]] = extractvalue [[OVR_T]] [[T2]], 0 205 // CHECK-NEXT: [[T6:%.*]] = select i1 [[T4]], i32 -1, i32 [[T5]] 206 // CHECK-NEXT: call noalias i8* @_Znaj(i32 [[T6]]) 207 // CHECK: icmp ult i32 {{.*}}, [[T1]] 208 elt *test(unsigned long long s) { 209 return new elt[s]; 210 } 211} 212