blob: 7128c4e4d07bc5966f1b8f76bd4067d6f9d3886b [file] [log] [blame]
Anders Carlssone8820a82009-12-17 04:57:25 +00001// RUN: %clang_cc1 %s -I%S -triple=x86_64-apple-darwin10 -emit-llvm -O3 -o - | FileCheck %s
Anders Carlsson0a4f1862009-12-17 04:41:05 +00002#include <typeinfo>
3
Anders Carlssonf64531a2009-12-30 01:00:12 +00004// vtables.
5extern "C" {
6 const void *_ZTVN10__cxxabiv123__fundamental_type_infoE;
7 const void *_ZTVN10__cxxabiv117__class_type_infoE;
8 const void *_ZTVN10__cxxabiv120__si_class_type_infoE;
Anders Carlsson2c0769e2009-12-30 01:25:42 +00009 const void *_ZTVN10__cxxabiv121__vmi_class_type_infoE;
Anders Carlssonf64531a2009-12-30 01:00:12 +000010 const void *_ZTVN10__cxxabiv119__pointer_type_infoE;
11 const void *_ZTVN10__cxxabiv129__pointer_to_member_type_infoE;
12};
13#define fundamental_type_info_vtable _ZTVN10__cxxabiv123__fundamental_type_infoE
14#define class_type_info_vtable _ZTVN10__cxxabiv117__class_type_infoE
15#define si_class_type_info_vtable _ZTVN10__cxxabiv120__si_class_type_infoE
Anders Carlsson2c0769e2009-12-30 01:25:42 +000016#define vmi_class_type_info_vtable _ZTVN10__cxxabiv121__vmi_class_type_infoE
Anders Carlssonf64531a2009-12-30 01:00:12 +000017#define pointer_type_info_vtable _ZTVN10__cxxabiv119__pointer_type_infoE
18#define pointer_to_member_type_info_vtable _ZTVN10__cxxabiv129__pointer_to_member_type_infoE
19
Anders Carlsson0a4f1862009-12-17 04:41:05 +000020class __pbase_type_info : public std::type_info {
21public:
22 unsigned int __flags;
23 const std::type_info *__pointee;
24
25 enum __masks {
26 __const_mask = 0x1,
27 __volatile_mask = 0x2,
28 __restrict_mask = 0x4,
29 __incomplete_mask = 0x8,
30 __incomplete_class_mask = 0x10
31 };
32};
33
Anders Carlssonbeb80192009-12-30 01:29:05 +000034class __class_type_info : public std::type_info { };
35
36class __si_class_type_info : public __class_type_info {
37public:
38 const __class_type_info *__base_type;
39};
40
Anders Carlsson08148092009-12-30 23:47:56 +000041struct __base_class_type_info {
42public:
43 const __class_type_info *__base_type;
44 long __offset_flags;
45
46 enum __offset_flags_masks {
47 __virtual_mask = 0x1,
48 __public_mask = 0x2,
49 __offset_shift = 8
50 };
51};
52
53class __vmi_class_type_info : public __class_type_info {
54public:
55 unsigned int __flags;
56 unsigned int __base_count;
57 __base_class_type_info __base_info[1];
58
59 enum __flags_masks {
60 __non_diamond_repeat_mask = 0x1,
61 __diamond_shaped_mask = 0x2
62 };
63};
64
Anders Carlsson0a4f1862009-12-17 04:41:05 +000065template<typename T> const T& to(const std::type_info &info) {
66return static_cast<const T&>(info);
67}
68struct Incomplete;
69
Anders Carlsson2c0769e2009-12-30 01:25:42 +000070struct A { int a; };
71struct Empty { };
72
73struct SI1 : A { };
74struct SI2 : Empty { };
75struct SI3 : Empty { virtual void f() { } };
76
77struct VMI1 : private A { };
78struct VMI2 : virtual A { };
79struct VMI3 : A { virtual void f() { } };
80struct VMI4 : A, Empty { };
Anders Carlsson17fa6f92009-12-20 23:37:55 +000081
Anders Carlsson08148092009-12-30 23:47:56 +000082struct VMIBase1 { int a; };
83struct VMIBase2 : VMIBase1 { int a; };
84struct VMI5 : VMIBase1, VMIBase2 { int a; };
85
86struct VMIBase3 : virtual VMIBase1 { int a; };
87struct VMI6 : virtual VMIBase1, VMIBase3 { int a; };
88
89struct VMI7 : VMIBase1, VMI5, private VMI6 { };
90
Anders Carlssonf64531a2009-12-30 01:00:12 +000091#define CHECK(x) if (!(x)) return __LINE__
Anders Carlsson08148092009-12-30 23:47:56 +000092#define CHECK_VTABLE(type, vtable) CHECK(&vtable##_type_info_vtable + 2 == (((void **)&(typeid(type)))[0]))
93#define CHECK_BASE_INFO_TYPE(type, index, base) CHECK(to<__vmi_class_type_info>(typeid(type)).__base_info[(index)].__base_type == &typeid(base))
94#define CHECK_BASE_INFO_OFFSET_FLAGS(type, index, offset, flags) CHECK(to<__vmi_class_type_info>(typeid(type)).__base_info[(index)].__offset_flags == (((offset) << 8) | (flags)))
Anders Carlsson8d145152009-12-20 22:30:54 +000095
Anders Carlssonabd6b092010-06-02 15:44:35 +000096struct B {
97 static int const volatile (*a)[10];
98 static int (*b)[10];
99
100 static int const volatile (B::*c)[10];
101 static int (B::*d)[10];
102};
103
Anders Carlsson0a4f1862009-12-17 04:41:05 +0000104// CHECK: define i32 @_Z1fv()
105int f() {
Anders Carlssonf64531a2009-12-30 01:00:12 +0000106 // Vectors should be treated as fundamental types.
107 typedef short __v4hi __attribute__ ((__vector_size__ (8)));
108 CHECK_VTABLE(__v4hi, fundamental);
Anders Carlsson2c0769e2009-12-30 01:25:42 +0000109
110 // A does not have any bases.
111 CHECK_VTABLE(A, class);
112
113 // SI1 has a single public base.
114 CHECK_VTABLE(SI1, si_class);
Anders Carlsson08148092009-12-30 23:47:56 +0000115 CHECK(to<__si_class_type_info>(typeid(SI1)).__base_type == &typeid(A));
Anders Carlsson2c0769e2009-12-30 01:25:42 +0000116
117 // SI2 has a single public empty base.
118 CHECK_VTABLE(SI2, si_class);
Anders Carlsson08148092009-12-30 23:47:56 +0000119 CHECK(to<__si_class_type_info>(typeid(SI2)).__base_type == &typeid(Empty));
Anders Carlsson2c0769e2009-12-30 01:25:42 +0000120
121 // SI3 has a single public empty base. SI3 is dynamic whereas Empty is not, but since Empty is
122 // an empty class, it will still be at offset zero.
123 CHECK_VTABLE(SI3, si_class);
Anders Carlsson08148092009-12-30 23:47:56 +0000124 CHECK(to<__si_class_type_info>(typeid(SI3)).__base_type == &typeid(Empty));
Anders Carlsson2c0769e2009-12-30 01:25:42 +0000125
126 // VMI1 has a single base, but it is private.
127 CHECK_VTABLE(VMI1, vmi_class);
128
129 // VMI2 has a single base, but it is virtual.
130 CHECK_VTABLE(VMI2, vmi_class);
131
132 // VMI3 has a single base, but VMI3 is dynamic whereas A is not, and A is not empty.
133 CHECK_VTABLE(VMI3, vmi_class);
134
135 // VMI4 has two bases.
136 CHECK_VTABLE(VMI4, vmi_class);
Anders Carlsson08148092009-12-30 23:47:56 +0000137
138 // VMI5 has non-diamond shaped inheritance.
139 CHECK_VTABLE(VMI5, vmi_class);
140 CHECK(to<__vmi_class_type_info>(typeid(VMI5)).__flags == __vmi_class_type_info::__non_diamond_repeat_mask);
141 CHECK(to<__vmi_class_type_info>(typeid(VMI5)).__base_count == 2);
142 CHECK_BASE_INFO_TYPE(VMI5, 0, VMIBase1);
143 CHECK_BASE_INFO_OFFSET_FLAGS(VMI5, 0, 0, __base_class_type_info::__public_mask);
144 CHECK_BASE_INFO_TYPE(VMI5, 1, VMIBase2);
145 CHECK_BASE_INFO_OFFSET_FLAGS(VMI5, 1, 4, __base_class_type_info::__public_mask);
Anders Carlssonf64531a2009-12-30 01:00:12 +0000146
Anders Carlsson08148092009-12-30 23:47:56 +0000147 // VMI6 has diamond shaped inheritance.
148 CHECK_VTABLE(VMI6, vmi_class);
149 CHECK(to<__vmi_class_type_info>(typeid(VMI6)).__flags == __vmi_class_type_info::__diamond_shaped_mask);
150 CHECK(to<__vmi_class_type_info>(typeid(VMI6)).__base_count == 2);
151 CHECK_BASE_INFO_TYPE(VMI6, 0, VMIBase1);
152 CHECK_BASE_INFO_OFFSET_FLAGS(VMI6, 0, -24, __base_class_type_info::__public_mask | __base_class_type_info::__virtual_mask);
153 CHECK_BASE_INFO_TYPE(VMI6, 1, VMIBase3);
154 CHECK_BASE_INFO_OFFSET_FLAGS(VMI6, 1, 0, __base_class_type_info::__public_mask);
155
156 // VMI7 has both non-diamond and diamond shaped inheritance.
157 CHECK_VTABLE(VMI7, vmi_class);
158 CHECK(to<__vmi_class_type_info>(typeid(VMI7)).__flags == (__vmi_class_type_info::__non_diamond_repeat_mask | __vmi_class_type_info::__diamond_shaped_mask));
159 CHECK(to<__vmi_class_type_info>(typeid(VMI7)).__base_count == 3);
160 CHECK_BASE_INFO_TYPE(VMI7, 0, VMIBase1);
161 CHECK_BASE_INFO_OFFSET_FLAGS(VMI7, 0, 16, __base_class_type_info::__public_mask);
162 CHECK_BASE_INFO_TYPE(VMI7, 1, VMI5);
163 CHECK_BASE_INFO_OFFSET_FLAGS(VMI7, 1, 20, __base_class_type_info::__public_mask);
164 CHECK_BASE_INFO_TYPE(VMI7, 2, VMI6);
165 CHECK_BASE_INFO_OFFSET_FLAGS(VMI7, 2, 0, 0);
Anders Carlssonbeb80192009-12-30 01:29:05 +0000166
Anders Carlsson8d145152009-12-20 22:30:54 +0000167 // Pointers to incomplete classes.
Anders Carlssonf64531a2009-12-30 01:00:12 +0000168 CHECK_VTABLE(Incomplete *, pointer);
Anders Carlsson17fa6f92009-12-20 23:37:55 +0000169 CHECK(to<__pbase_type_info>(typeid(Incomplete *)).__flags == __pbase_type_info::__incomplete_mask);
170 CHECK(to<__pbase_type_info>(typeid(Incomplete **)).__flags == __pbase_type_info::__incomplete_mask);
171 CHECK(to<__pbase_type_info>(typeid(Incomplete ***)).__flags == __pbase_type_info::__incomplete_mask);
Anders Carlsson8d145152009-12-20 22:30:54 +0000172
173 // Member pointers.
Anders Carlssonf64531a2009-12-30 01:00:12 +0000174 CHECK_VTABLE(int Incomplete::*, pointer_to_member);
Anders Carlsson17fa6f92009-12-20 23:37:55 +0000175 CHECK(to<__pbase_type_info>(typeid(int Incomplete::*)).__flags == __pbase_type_info::__incomplete_class_mask);
176 CHECK(to<__pbase_type_info>(typeid(Incomplete Incomplete::*)).__flags == (__pbase_type_info::__incomplete_class_mask | __pbase_type_info::__incomplete_mask));
177 CHECK(to<__pbase_type_info>(typeid(Incomplete A::*)).__flags == (__pbase_type_info::__incomplete_mask));
Anders Carlsson8d145152009-12-20 22:30:54 +0000178
Anders Carlssonabd6b092010-06-02 15:44:35 +0000179 // Check that when stripping qualifiers off the pointee type, we correctly handle arrays.
180 CHECK(to<__pbase_type_info>(typeid(B::a)).__flags == (__pbase_type_info::__const_mask | __pbase_type_info::__volatile_mask));
181 CHECK(to<__pbase_type_info>(typeid(B::a)).__pointee == to<__pbase_type_info>(typeid(B::b)).__pointee);
182 CHECK(to<__pbase_type_info>(typeid(B::c)).__flags == (__pbase_type_info::__const_mask | __pbase_type_info::__volatile_mask));
183 CHECK(to<__pbase_type_info>(typeid(B::c)).__pointee == to<__pbase_type_info>(typeid(B::d)).__pointee);
184
Anders Carlsson0a4f1862009-12-17 04:41:05 +0000185 // Success!
Anders Carlsson8d145152009-12-20 22:30:54 +0000186 // CHECK: ret i32 0
Anders Carlsson0a4f1862009-12-17 04:41:05 +0000187 return 0;
188}
189
190#ifdef HARNESS
191extern "C" void printf(const char *, ...);
192
193int main() {
194 int result = f();
195
196 if (result == 0)
197 printf("success!\n");
198 else
Anders Carlsson8d145152009-12-20 22:30:54 +0000199 printf("test on line %d failed!\n", result);
Anders Carlsson0a4f1862009-12-17 04:41:05 +0000200
201 return result;
202}
203#endif
204
205