210 lines
6.5 KiB
C++
210 lines
6.5 KiB
C++
|
// RUN: %clang_cc1 -triple i386-unknown-unknown %s -emit-llvm -o - | FileCheck %s
|
||
|
|
||
|
// rdar://problem/9246208
|
||
|
|
||
|
// Basic test.
|
||
|
namespace test0 {
|
||
|
struct A {
|
||
|
A();
|
||
|
int x;
|
||
|
};
|
||
|
|
||
|
typedef A elt;
|
||
|
|
||
|
// CHECK: define [[A:%.*]]* @_ZN5test04testEs(i16 signext
|
||
|
// CHECK: [[N:%.*]] = sext i16 {{%.*}} to i32
|
||
|
// CHECK-NEXT: [[T0:%.*]] = call { i32, i1 } @llvm.umul.with.overflow.i32(i32 [[N]], i32 4)
|
||
|
// CHECK-NEXT: [[T1:%.*]] = extractvalue { i32, i1 } [[T0]], 1
|
||
|
// CHECK-NEXT: [[T2:%.*]] = extractvalue { i32, i1 } [[T0]], 0
|
||
|
// CHECK-NEXT: [[T3:%.*]] = select i1 [[T1]], i32 -1, i32 [[T2]]
|
||
|
// CHECK-NEXT: call noalias i8* @_Znaj(i32 [[T3]])
|
||
|
// CHECK: getelementptr inbounds {{.*}}, i32 [[N]]
|
||
|
elt *test(short s) {
|
||
|
return new elt[s];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// test0 with a nested array.
|
||
|
namespace test1 {
|
||
|
struct A {
|
||
|
A();
|
||
|
int x;
|
||
|
};
|
||
|
|
||
|
typedef A elt[100];
|
||
|
|
||
|
// CHECK: define [100 x [[A:%.*]]]* @_ZN5test14testEs(i16 signext
|
||
|
// CHECK: [[N:%.*]] = sext i16 {{%.*}} to i32
|
||
|
// CHECK-NEXT: [[T0:%.*]] = call { i32, i1 } @llvm.umul.with.overflow.i32(i32 [[N]], i32 400)
|
||
|
// CHECK-NEXT: [[T1:%.*]] = extractvalue { i32, i1 } [[T0]], 1
|
||
|
// CHECK-NEXT: [[T2:%.*]] = extractvalue { i32, i1 } [[T0]], 0
|
||
|
// CHECK-NEXT: [[T3:%.*]] = mul i32 [[N]], 100
|
||
|
// CHECK-NEXT: [[T4:%.*]] = select i1 [[T1]], i32 -1, i32 [[T2]]
|
||
|
// CHECK-NEXT: call noalias i8* @_Znaj(i32 [[T4]])
|
||
|
// CHECK: getelementptr inbounds {{.*}}, i32 [[T3]]
|
||
|
elt *test(short s) {
|
||
|
return new elt[s];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// test1 with an array cookie.
|
||
|
namespace test2 {
|
||
|
struct A {
|
||
|
A();
|
||
|
~A();
|
||
|
int x;
|
||
|
};
|
||
|
|
||
|
typedef A elt[100];
|
||
|
|
||
|
// CHECK: define [100 x [[A:%.*]]]* @_ZN5test24testEs(i16 signext
|
||
|
// CHECK: [[N:%.*]] = sext i16 {{%.*}} to i32
|
||
|
// CHECK-NEXT: [[T0:%.*]] = call { i32, i1 } @llvm.umul.with.overflow.i32(i32 [[N]], i32 400)
|
||
|
// CHECK-NEXT: [[T1:%.*]] = extractvalue { i32, i1 } [[T0]], 1
|
||
|
// CHECK-NEXT: [[T2:%.*]] = extractvalue { i32, i1 } [[T0]], 0
|
||
|
// CHECK-NEXT: [[T3:%.*]] = mul i32 [[N]], 100
|
||
|
// CHECK-NEXT: [[T4:%.*]] = call { i32, i1 } @llvm.uadd.with.overflow.i32(i32 [[T2]], i32 4)
|
||
|
// CHECK-NEXT: [[T5:%.*]] = extractvalue { i32, i1 } [[T4]], 1
|
||
|
// CHECK-NEXT: [[T6:%.*]] = or i1 [[T1]], [[T5]]
|
||
|
// CHECK-NEXT: [[T7:%.*]] = extractvalue { i32, i1 } [[T4]], 0
|
||
|
// CHECK-NEXT: [[T8:%.*]] = select i1 [[T6]], i32 -1, i32 [[T7]]
|
||
|
// CHECK-NEXT: call noalias i8* @_Znaj(i32 [[T8]])
|
||
|
// CHECK: getelementptr inbounds {{.*}}, i32 [[T3]]
|
||
|
elt *test(short s) {
|
||
|
return new elt[s];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// test0 with a 1-byte element.
|
||
|
namespace test4 {
|
||
|
struct A {
|
||
|
A();
|
||
|
};
|
||
|
|
||
|
typedef A elt;
|
||
|
|
||
|
// CHECK: define [[A:%.*]]* @_ZN5test44testEs(i16 signext
|
||
|
// CHECK: [[N:%.*]] = sext i16 {{%.*}} to i32
|
||
|
// CHECK-NEXT: [[T0:%.*]] = icmp slt i32 [[N]], 0
|
||
|
// CHECK-NEXT: [[T1:%.*]] = select i1 [[T0]], i32 -1, i32 [[N]]
|
||
|
// CHECK-NEXT: call noalias i8* @_Znaj(i32 [[T1]])
|
||
|
// CHECK: getelementptr inbounds {{.*}}, i32 [[N]]
|
||
|
elt *test(short s) {
|
||
|
return new elt[s];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// test4 with no sext required.
|
||
|
namespace test5 {
|
||
|
struct A {
|
||
|
A();
|
||
|
};
|
||
|
|
||
|
typedef A elt;
|
||
|
|
||
|
// CHECK: define [[A:%.*]]* @_ZN5test54testEi(i32
|
||
|
// CHECK: [[N:%.*]] = load i32*
|
||
|
// CHECK-NEXT: [[T0:%.*]] = icmp slt i32 [[N]], 0
|
||
|
// CHECK-NEXT: [[T1:%.*]] = select i1 [[T0]], i32 -1, i32 [[N]]
|
||
|
// CHECK-NEXT: call noalias i8* @_Znaj(i32 [[T1]])
|
||
|
// CHECK: getelementptr inbounds {{.*}}, i32 [[N]]
|
||
|
elt *test(int s) {
|
||
|
return new elt[s];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// test0 with an unsigned size.
|
||
|
namespace test6 {
|
||
|
struct A {
|
||
|
A();
|
||
|
int x;
|
||
|
};
|
||
|
|
||
|
typedef A elt;
|
||
|
|
||
|
// CHECK: define [[A:%.*]]* @_ZN5test64testEt(i16 zeroext
|
||
|
// CHECK: [[N:%.*]] = zext i16 {{%.*}} to i32
|
||
|
// CHECK-NEXT: [[T0:%.*]] = call { i32, i1 } @llvm.umul.with.overflow.i32(i32 [[N]], i32 4)
|
||
|
// CHECK-NEXT: [[T1:%.*]] = extractvalue { i32, i1 } [[T0]], 1
|
||
|
// CHECK-NEXT: [[T2:%.*]] = extractvalue { i32, i1 } [[T0]], 0
|
||
|
// CHECK-NEXT: [[T3:%.*]] = select i1 [[T1]], i32 -1, i32 [[T2]]
|
||
|
// CHECK-NEXT: call noalias i8* @_Znaj(i32 [[T3]])
|
||
|
// CHECK: getelementptr inbounds {{.*}}, i32 [[N]]
|
||
|
elt *test(unsigned short s) {
|
||
|
return new elt[s];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// test1 with an unsigned size.
|
||
|
namespace test7 {
|
||
|
struct A {
|
||
|
A();
|
||
|
int x;
|
||
|
};
|
||
|
|
||
|
typedef A elt[100];
|
||
|
|
||
|
// CHECK: define [100 x [[A:%.*]]]* @_ZN5test74testEt(i16 zeroext
|
||
|
// CHECK: [[N:%.*]] = zext i16 {{%.*}} to i32
|
||
|
// CHECK-NEXT: [[T0:%.*]] = call { i32, i1 } @llvm.umul.with.overflow.i32(i32 [[N]], i32 400)
|
||
|
// CHECK-NEXT: [[T1:%.*]] = extractvalue { i32, i1 } [[T0]], 1
|
||
|
// CHECK-NEXT: [[T2:%.*]] = extractvalue { i32, i1 } [[T0]], 0
|
||
|
// CHECK-NEXT: [[T3:%.*]] = mul i32 [[N]], 100
|
||
|
// CHECK-NEXT: [[T4:%.*]] = select i1 [[T1]], i32 -1, i32 [[T2]]
|
||
|
// CHECK-NEXT: call noalias i8* @_Znaj(i32 [[T4]])
|
||
|
// CHECK: getelementptr inbounds {{.*}}, i32 [[T3]]
|
||
|
elt *test(unsigned short s) {
|
||
|
return new elt[s];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// test0 with a signed type larger than size_t.
|
||
|
namespace test8 {
|
||
|
struct A {
|
||
|
A();
|
||
|
int x;
|
||
|
};
|
||
|
|
||
|
typedef A elt;
|
||
|
|
||
|
// CHECK: define [[A:%.*]]* @_ZN5test84testEx(i64
|
||
|
// CHECK: [[N:%.*]] = load i64*
|
||
|
// CHECK-NEXT: [[T0:%.*]] = icmp uge i64 [[N]], 4294967296
|
||
|
// CHECK-NEXT: [[T1:%.*]] = trunc i64 [[N]] to i32
|
||
|
// CHECK-NEXT: [[T2:%.*]] = call { i32, i1 } @llvm.umul.with.overflow.i32(i32 [[T1]], i32 4)
|
||
|
// CHECK-NEXT: [[T3:%.*]] = extractvalue { i32, i1 } [[T2]], 1
|
||
|
// CHECK-NEXT: [[T4:%.*]] = or i1 [[T0]], [[T3]]
|
||
|
// CHECK-NEXT: [[T5:%.*]] = extractvalue { i32, i1 } [[T2]], 0
|
||
|
// CHECK-NEXT: [[T6:%.*]] = select i1 [[T4]], i32 -1, i32 [[T5]]
|
||
|
// CHECK-NEXT: call noalias i8* @_Znaj(i32 [[T6]])
|
||
|
// CHECK: getelementptr inbounds {{.*}}, i32 [[T1]]
|
||
|
elt *test(long long s) {
|
||
|
return new elt[s];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// test8 with an unsigned type.
|
||
|
namespace test9 {
|
||
|
struct A {
|
||
|
A();
|
||
|
int x;
|
||
|
};
|
||
|
|
||
|
typedef A elt;
|
||
|
|
||
|
// CHECK: define [[A:%.*]]* @_ZN5test94testEy(i64
|
||
|
// CHECK: [[N:%.*]] = load i64*
|
||
|
// CHECK-NEXT: [[T0:%.*]] = icmp uge i64 [[N]], 4294967296
|
||
|
// CHECK-NEXT: [[T1:%.*]] = trunc i64 [[N]] to i32
|
||
|
// CHECK-NEXT: [[T2:%.*]] = call { i32, i1 } @llvm.umul.with.overflow.i32(i32 [[T1]], i32 4)
|
||
|
// CHECK-NEXT: [[T3:%.*]] = extractvalue { i32, i1 } [[T2]], 1
|
||
|
// CHECK-NEXT: [[T4:%.*]] = or i1 [[T0]], [[T3]]
|
||
|
// CHECK-NEXT: [[T5:%.*]] = extractvalue { i32, i1 } [[T2]], 0
|
||
|
// CHECK-NEXT: [[T6:%.*]] = select i1 [[T4]], i32 -1, i32 [[T5]]
|
||
|
// CHECK-NEXT: call noalias i8* @_Znaj(i32 [[T6]])
|
||
|
// CHECK: getelementptr inbounds {{.*}}, i32 [[T1]]
|
||
|
elt *test(unsigned long long s) {
|
||
|
return new elt[s];
|
||
|
}
|
||
|
}
|