2011-02-15 14:40:56 +08:00
|
|
|
// RUN: %clang_cc1 %s -emit-llvm -o %t.ll -triple=x86_64-apple-darwin10
|
|
|
|
// RUN: FileCheck %s < %t.ll
|
|
|
|
// RUN: FileCheck -check-prefix=CHECK-GLOBAL %s < %t.ll
|
|
|
|
|
2010-02-02 11:37:46 +08:00
|
|
|
struct A { int a; int b; };
|
2009-11-27 12:56:40 +08:00
|
|
|
struct B { int b; };
|
|
|
|
struct C : B, A { };
|
|
|
|
|
2010-02-02 11:32:35 +08:00
|
|
|
// Zero init.
|
|
|
|
namespace ZeroInit {
|
2011-02-15 14:40:56 +08:00
|
|
|
// CHECK-GLOBAL: @_ZN8ZeroInit1aE = global i64 -1
|
2010-02-02 11:32:35 +08:00
|
|
|
int A::* a;
|
|
|
|
|
2011-02-15 14:40:56 +08:00
|
|
|
// CHECK-GLOBAL: @_ZN8ZeroInit2aaE = global [2 x i64] [i64 -1, i64 -1]
|
2010-02-02 11:32:35 +08:00
|
|
|
int A::* aa[2];
|
|
|
|
|
2011-02-15 14:40:56 +08:00
|
|
|
// CHECK-GLOBAL: @_ZN8ZeroInit3aaaE = global [2 x [2 x i64]] {{\[}}[2 x i64] [i64 -1, i64 -1], [2 x i64] [i64 -1, i64 -1]]
|
2010-02-02 11:32:35 +08:00
|
|
|
int A::* aaa[2][2];
|
|
|
|
|
2011-02-15 14:40:56 +08:00
|
|
|
// CHECK-GLOBAL: @_ZN8ZeroInit1bE = global i64 -1,
|
2010-02-02 11:32:35 +08:00
|
|
|
int A::* b = 0;
|
2010-02-02 13:17:25 +08:00
|
|
|
|
2011-07-12 13:53:08 +08:00
|
|
|
// CHECK-GLOBAL: @_ZN8ZeroInit2saE = internal global %struct.anon { i64 -1 }
|
2010-02-02 13:17:25 +08:00
|
|
|
struct {
|
|
|
|
int A::*a;
|
|
|
|
} sa;
|
2010-10-23 05:05:15 +08:00
|
|
|
void test_sa() { (void) sa; } // force emission
|
2010-02-02 13:17:25 +08:00
|
|
|
|
2011-02-15 14:40:56 +08:00
|
|
|
// CHECK-GLOBAL: @_ZN8ZeroInit3ssaE = internal
|
|
|
|
// CHECK-GLOBAL: [2 x i64] [i64 -1, i64 -1]
|
2010-02-02 13:17:25 +08:00
|
|
|
struct {
|
|
|
|
int A::*aa[2];
|
|
|
|
} ssa[2];
|
2010-10-23 05:05:15 +08:00
|
|
|
void test_ssa() { (void) ssa; }
|
2010-02-02 13:17:25 +08:00
|
|
|
|
2011-07-12 13:53:08 +08:00
|
|
|
// CHECK-GLOBAL: @_ZN8ZeroInit2ssE = internal global %struct.anon.1 { %struct.anon.2 { i64 -1 } }
|
2010-02-02 13:17:25 +08:00
|
|
|
struct {
|
|
|
|
struct {
|
|
|
|
int A::*pa;
|
|
|
|
} s;
|
|
|
|
} ss;
|
2010-10-23 05:05:15 +08:00
|
|
|
void test_ss() { (void) ss; }
|
2010-05-19 00:51:41 +08:00
|
|
|
|
|
|
|
struct A {
|
|
|
|
int A::*a;
|
|
|
|
int b;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct B {
|
|
|
|
A a[10];
|
|
|
|
char c;
|
|
|
|
int B::*b;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct C : A, B { int j; };
|
2014-09-28 14:39:30 +08:00
|
|
|
// CHECK-GLOBAL: @_ZN8ZeroInit1cE = global {{%.*}} <{ %"struct.ZeroInit::A" { i64 -1, i32 0 }, %"struct.ZeroInit::B" { [10 x %"struct.ZeroInit::A"] [%"struct.ZeroInit::A" { i64 -1, i32 0 }, %"struct.ZeroInit::A" { i64 -1, i32 0 }, %"struct.ZeroInit::A" { i64 -1, i32 0 }, %"struct.ZeroInit::A" { i64 -1, i32 0 }, %"struct.ZeroInit::A" { i64 -1, i32 0 }, %"struct.ZeroInit::A" { i64 -1, i32 0 }, %"struct.ZeroInit::A" { i64 -1, i32 0 }, %"struct.ZeroInit::A" { i64 -1, i32 0 }, %"struct.ZeroInit::A" { i64 -1, i32 0 }, %"struct.ZeroInit::A" { i64 -1, i32 0 }], i8 0, i64 -1 }, i32 0, [4 x i8] zeroinitializer }>, align 8
|
2010-05-19 00:51:41 +08:00
|
|
|
C c;
|
2010-02-02 11:37:46 +08:00
|
|
|
}
|
2010-02-02 11:32:35 +08:00
|
|
|
|
2010-02-02 11:37:46 +08:00
|
|
|
// PR5674
|
|
|
|
namespace PR5674 {
|
2011-02-15 14:40:56 +08:00
|
|
|
// CHECK-GLOBAL: @_ZN6PR56742pbE = global i64 4
|
2010-02-02 11:37:46 +08:00
|
|
|
int A::*pb = &A::b;
|
2010-02-02 11:32:35 +08:00
|
|
|
}
|
|
|
|
|
2010-02-02 11:23:38 +08:00
|
|
|
// Casts.
|
|
|
|
namespace Casts {
|
|
|
|
|
2009-11-27 12:56:40 +08:00
|
|
|
int A::*pa;
|
|
|
|
int C::*pc;
|
|
|
|
|
|
|
|
void f() {
|
2010-08-23 09:21:21 +08:00
|
|
|
// CHECK: store i64 -1, i64* @_ZN5Casts2paE
|
2009-11-27 12:56:40 +08:00
|
|
|
pa = 0;
|
|
|
|
|
2015-02-28 05:19:58 +08:00
|
|
|
// CHECK-NEXT: [[TMP:%.*]] = load i64, i64* @_ZN5Casts2paE, align 8
|
2010-08-23 09:21:21 +08:00
|
|
|
// CHECK-NEXT: [[ADJ:%.*]] = add nsw i64 [[TMP]], 4
|
|
|
|
// CHECK-NEXT: [[ISNULL:%.*]] = icmp eq i64 [[TMP]], -1
|
|
|
|
// CHECK-NEXT: [[RES:%.*]] = select i1 [[ISNULL]], i64 [[TMP]], i64 [[ADJ]]
|
|
|
|
// CHECK-NEXT: store i64 [[RES]], i64* @_ZN5Casts2pcE
|
2009-11-27 12:56:40 +08:00
|
|
|
pc = pa;
|
|
|
|
|
2015-02-28 05:19:58 +08:00
|
|
|
// CHECK-NEXT: [[TMP:%.*]] = load i64, i64* @_ZN5Casts2pcE, align 8
|
2010-08-23 09:21:21 +08:00
|
|
|
// CHECK-NEXT: [[ADJ:%.*]] = sub nsw i64 [[TMP]], 4
|
|
|
|
// CHECK-NEXT: [[ISNULL:%.*]] = icmp eq i64 [[TMP]], -1
|
|
|
|
// CHECK-NEXT: [[RES:%.*]] = select i1 [[ISNULL]], i64 [[TMP]], i64 [[ADJ]]
|
|
|
|
// CHECK-NEXT: store i64 [[RES]], i64* @_ZN5Casts2paE
|
2009-11-27 12:56:40 +08:00
|
|
|
pa = static_cast<int A::*>(pc);
|
|
|
|
}
|
2010-02-02 11:23:38 +08:00
|
|
|
|
|
|
|
}
|
2010-02-02 11:37:46 +08:00
|
|
|
|
|
|
|
// Comparisons
|
|
|
|
namespace Comparisons {
|
|
|
|
void f() {
|
|
|
|
int A::*a;
|
|
|
|
|
|
|
|
// CHECK: icmp ne i64 {{.*}}, -1
|
|
|
|
if (a) { }
|
|
|
|
|
|
|
|
// CHECK: icmp ne i64 {{.*}}, -1
|
|
|
|
if (a != 0) { }
|
|
|
|
|
|
|
|
// CHECK: icmp ne i64 -1, {{.*}}
|
|
|
|
if (0 != a) { }
|
|
|
|
|
|
|
|
// CHECK: icmp eq i64 {{.*}}, -1
|
|
|
|
if (a == 0) { }
|
|
|
|
|
|
|
|
// CHECK: icmp eq i64 -1, {{.*}}
|
|
|
|
if (0 == a) { }
|
|
|
|
}
|
|
|
|
}
|
2010-05-14 23:05:19 +08:00
|
|
|
|
|
|
|
namespace ValueInit {
|
|
|
|
|
|
|
|
struct A {
|
|
|
|
int A::*a;
|
|
|
|
|
|
|
|
char c;
|
|
|
|
|
|
|
|
A();
|
|
|
|
};
|
|
|
|
|
2018-10-15 23:43:00 +08:00
|
|
|
// CHECK-LABEL: define void @_ZN9ValueInit1AC2Ev(%"struct.ValueInit::A"* %this) unnamed_addr
|
2010-05-14 23:05:19 +08:00
|
|
|
// CHECK: store i64 -1, i64*
|
|
|
|
// CHECK: ret void
|
|
|
|
A::A() : a() {}
|
|
|
|
|
|
|
|
}
|
2010-05-19 00:51:41 +08:00
|
|
|
|
2010-05-28 02:51:01 +08:00
|
|
|
namespace VirtualBases {
|
|
|
|
|
|
|
|
struct A {
|
|
|
|
char c;
|
|
|
|
int A::*i;
|
|
|
|
};
|
|
|
|
|
2011-04-18 05:56:13 +08:00
|
|
|
// CHECK-GLOBAL: @_ZN12VirtualBases1bE = global %"struct.VirtualBases::B" { i32 (...)** null, %"struct.VirtualBases::A" { i8 0, i64 -1 } }, align 8
|
2010-05-28 02:51:01 +08:00
|
|
|
struct B : virtual A { };
|
|
|
|
B b;
|
|
|
|
|
2011-04-18 05:56:13 +08:00
|
|
|
// CHECK-GLOBAL: @_ZN12VirtualBases1cE = global %"struct.VirtualBases::C" { i32 (...)** null, i64 -1, %"struct.VirtualBases::A" { i8 0, i64 -1 } }, align 8
|
2010-05-28 02:51:01 +08:00
|
|
|
struct C : virtual A { int A::*i; };
|
|
|
|
C c;
|
|
|
|
|
2011-04-18 05:56:13 +08:00
|
|
|
// CHECK-GLOBAL: @_ZN12VirtualBases1dE = global %"struct.VirtualBases::D" { %"struct.VirtualBases::C.base" { i32 (...)** null, i64 -1 }, i64 -1, %"struct.VirtualBases::A" { i8 0, i64 -1 } }, align 8
|
2010-05-28 02:51:01 +08:00
|
|
|
struct D : C { int A::*i; };
|
|
|
|
D d;
|
|
|
|
|
|
|
|
}
|
2010-06-03 23:36:07 +08:00
|
|
|
|
|
|
|
namespace Test1 {
|
|
|
|
|
|
|
|
// Don't crash when A contains a bit-field.
|
|
|
|
struct A {
|
|
|
|
int A::* a;
|
|
|
|
int b : 10;
|
|
|
|
};
|
|
|
|
A a;
|
|
|
|
|
|
|
|
}
|
2010-09-02 23:00:29 +08:00
|
|
|
|
|
|
|
namespace BoolPtrToMember {
|
|
|
|
struct X {
|
|
|
|
bool member;
|
|
|
|
};
|
|
|
|
|
2020-05-19 02:29:11 +08:00
|
|
|
// CHECK-LABEL: define nonnull align {{[0-9]+}} dereferenceable({{[0-9]+}}) i8* @_ZN15BoolPtrToMember1fERNS_1XEMS0_b
|
2010-09-02 23:00:29 +08:00
|
|
|
bool &f(X &x, bool X::*member) {
|
|
|
|
// CHECK: {{bitcast.* to i8\*}}
|
2015-02-28 03:18:17 +08:00
|
|
|
// CHECK-NEXT: getelementptr inbounds i8, i8*
|
2010-09-02 23:00:29 +08:00
|
|
|
// CHECK-NEXT: ret i8*
|
|
|
|
return x.*member;
|
|
|
|
}
|
|
|
|
}
|
2010-10-31 09:21:47 +08:00
|
|
|
|
|
|
|
namespace PR8507 {
|
|
|
|
|
|
|
|
struct S;
|
|
|
|
void f(S* p, double S::*pm) {
|
|
|
|
if (0 < p->*pm) {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2011-02-15 14:40:56 +08:00
|
|
|
namespace test4 {
|
|
|
|
struct A { int A_i; };
|
|
|
|
struct B : virtual A { int A::*B_p; };
|
|
|
|
struct C : virtual B { int *C_p; };
|
|
|
|
struct D : C { int *D_p; };
|
|
|
|
|
2011-07-10 01:41:47 +08:00
|
|
|
// CHECK-GLOBAL: @_ZN5test41dE = global %"struct.test4::D" { %"struct.test4::C.base" zeroinitializer, i32* null, %"struct.test4::B.base" { i32 (...)** null, i64 -1 }, %"struct.test4::A" zeroinitializer }, align 8
|
2011-02-15 14:40:56 +08:00
|
|
|
D d;
|
|
|
|
}
|
2011-12-07 09:30:11 +08:00
|
|
|
|
|
|
|
namespace PR11487 {
|
|
|
|
union U
|
|
|
|
{
|
|
|
|
int U::* mptr;
|
|
|
|
char x[16];
|
|
|
|
} x;
|
|
|
|
// CHECK-GLOBAL: @_ZN7PR114871xE = global %"union.PR11487::U" { i64 -1, [8 x i8] zeroinitializer }, align 8
|
|
|
|
|
|
|
|
}
|
2012-06-16 07:51:06 +08:00
|
|
|
|
|
|
|
namespace PR13097 {
|
|
|
|
struct X { int x; X(const X&); };
|
|
|
|
struct A {
|
|
|
|
int qq;
|
|
|
|
X x;
|
|
|
|
};
|
|
|
|
A f();
|
|
|
|
X g() { return f().*&A::x; }
|
2013-08-15 14:47:53 +08:00
|
|
|
// CHECK-LABEL: define void @_ZN7PR130971gEv
|
2012-06-16 07:51:06 +08:00
|
|
|
// CHECK: call void @_ZN7PR130971fEv
|
|
|
|
// CHECK-NOT: memcpy
|
|
|
|
// CHECK: call void @_ZN7PR130971XC1ERKS0_
|
|
|
|
}
|
2013-02-20 15:22:19 +08:00
|
|
|
|
2014-09-28 14:39:30 +08:00
|
|
|
namespace PR21089 {
|
|
|
|
struct A {
|
|
|
|
bool : 1;
|
|
|
|
int A::*x;
|
|
|
|
bool y;
|
|
|
|
A();
|
|
|
|
};
|
|
|
|
struct B : A {
|
|
|
|
};
|
|
|
|
B b;
|
|
|
|
// CHECK-GLOBAL: @_ZN7PR210891bE = global %"struct.PR21089::B" { %"struct.PR21089::A.base" <{ i8 0, [7 x i8] zeroinitializer, i64 -1, i8 0 }>, [7 x i8] zeroinitializer }, align 8
|
|
|
|
}
|
|
|
|
|
2014-10-15 15:57:41 +08:00
|
|
|
namespace PR21282 {
|
|
|
|
union U {
|
|
|
|
int U::*x;
|
|
|
|
long y[2];
|
|
|
|
};
|
|
|
|
U u;
|
|
|
|
// CHECK-GLOBAL: @_ZN7PR212821uE = global %"union.PR21282::U" { i64 -1, [8 x i8] zeroinitializer }, align 8
|
|
|
|
}
|
|
|
|
|
2015-05-27 05:28:50 +08:00
|
|
|
namespace FlexibleArrayMember {
|
|
|
|
struct S {
|
|
|
|
int S::*x[];
|
|
|
|
};
|
|
|
|
S s;
|
|
|
|
// CHECK-GLOBAL: @_ZN19FlexibleArrayMember1sE = global %"struct.FlexibleArrayMember::S" zeroinitializer, align 8
|
|
|
|
}
|
|
|
|
|
2015-05-30 17:12:07 +08:00
|
|
|
namespace IndirectPDM {
|
|
|
|
union U {
|
|
|
|
union {
|
|
|
|
int U::*m;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
U u;
|
|
|
|
// CHECK-GLOBAL: @_ZN11IndirectPDM1uE = global %"union.IndirectPDM::U" { %union.anon { i64 -1 } }, align 8
|
|
|
|
}
|