f4a2713ac8
Change-Id: Ia40e9ffdf29b5dab2f122f673ff6802a58bc690f
205 lines
7.5 KiB
C++
205 lines
7.5 KiB
C++
// RUN: %clang_cc1 %s -I%S -triple=x86_64-apple-darwin10 -emit-llvm -O3 -o - | FileCheck %s
|
|
#include <typeinfo>
|
|
|
|
// vtables.
|
|
extern "C" {
|
|
const void *_ZTVN10__cxxabiv123__fundamental_type_infoE;
|
|
const void *_ZTVN10__cxxabiv117__class_type_infoE;
|
|
const void *_ZTVN10__cxxabiv120__si_class_type_infoE;
|
|
const void *_ZTVN10__cxxabiv121__vmi_class_type_infoE;
|
|
const void *_ZTVN10__cxxabiv119__pointer_type_infoE;
|
|
const void *_ZTVN10__cxxabiv129__pointer_to_member_type_infoE;
|
|
};
|
|
#define fundamental_type_info_vtable _ZTVN10__cxxabiv123__fundamental_type_infoE
|
|
#define class_type_info_vtable _ZTVN10__cxxabiv117__class_type_infoE
|
|
#define si_class_type_info_vtable _ZTVN10__cxxabiv120__si_class_type_infoE
|
|
#define vmi_class_type_info_vtable _ZTVN10__cxxabiv121__vmi_class_type_infoE
|
|
#define pointer_type_info_vtable _ZTVN10__cxxabiv119__pointer_type_infoE
|
|
#define pointer_to_member_type_info_vtable _ZTVN10__cxxabiv129__pointer_to_member_type_infoE
|
|
|
|
class __pbase_type_info : public std::type_info {
|
|
public:
|
|
unsigned int __flags;
|
|
const std::type_info *__pointee;
|
|
|
|
enum __masks {
|
|
__const_mask = 0x1,
|
|
__volatile_mask = 0x2,
|
|
__restrict_mask = 0x4,
|
|
__incomplete_mask = 0x8,
|
|
__incomplete_class_mask = 0x10
|
|
};
|
|
};
|
|
|
|
class __class_type_info : public std::type_info { };
|
|
|
|
class __si_class_type_info : public __class_type_info {
|
|
public:
|
|
const __class_type_info *__base_type;
|
|
};
|
|
|
|
struct __base_class_type_info {
|
|
public:
|
|
const __class_type_info *__base_type;
|
|
long __offset_flags;
|
|
|
|
enum __offset_flags_masks {
|
|
__virtual_mask = 0x1,
|
|
__public_mask = 0x2,
|
|
__offset_shift = 8
|
|
};
|
|
};
|
|
|
|
class __vmi_class_type_info : public __class_type_info {
|
|
public:
|
|
unsigned int __flags;
|
|
unsigned int __base_count;
|
|
__base_class_type_info __base_info[1];
|
|
|
|
enum __flags_masks {
|
|
__non_diamond_repeat_mask = 0x1,
|
|
__diamond_shaped_mask = 0x2
|
|
};
|
|
};
|
|
|
|
template<typename T> const T& to(const std::type_info &info) {
|
|
return static_cast<const T&>(info);
|
|
}
|
|
struct Incomplete;
|
|
|
|
struct A { int a; };
|
|
struct Empty { };
|
|
|
|
struct SI1 : A { };
|
|
struct SI2 : Empty { };
|
|
struct SI3 : Empty { virtual void f() { } };
|
|
|
|
struct VMI1 : private A { };
|
|
struct VMI2 : virtual A { };
|
|
struct VMI3 : A { virtual void f() { } };
|
|
struct VMI4 : A, Empty { };
|
|
|
|
struct VMIBase1 { int a; };
|
|
struct VMIBase2 : VMIBase1 { int a; };
|
|
struct VMI5 : VMIBase1, VMIBase2 { int a; };
|
|
|
|
struct VMIBase3 : virtual VMIBase1 { int a; };
|
|
struct VMI6 : virtual VMIBase1, VMIBase3 { int a; };
|
|
|
|
struct VMI7 : VMIBase1, VMI5, private VMI6 { };
|
|
|
|
#define CHECK(x) if (!(x)) return __LINE__
|
|
#define CHECK_VTABLE(type, vtable) CHECK(&vtable##_type_info_vtable + 2 == (((void **)&(typeid(type)))[0]))
|
|
#define CHECK_BASE_INFO_TYPE(type, index, base) CHECK(to<__vmi_class_type_info>(typeid(type)).__base_info[(index)].__base_type == &typeid(base))
|
|
#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)))
|
|
|
|
struct B {
|
|
static int const volatile (*a)[10];
|
|
static int (*b)[10];
|
|
|
|
static int const volatile (B::*c)[10];
|
|
static int (B::*d)[10];
|
|
};
|
|
|
|
// CHECK-LABEL: define i32 @_Z1fv()
|
|
int f() {
|
|
// Vectors should be treated as fundamental types.
|
|
typedef short __v4hi __attribute__ ((__vector_size__ (8)));
|
|
CHECK_VTABLE(__v4hi, fundamental);
|
|
|
|
// A does not have any bases.
|
|
CHECK_VTABLE(A, class);
|
|
|
|
// SI1 has a single public base.
|
|
CHECK_VTABLE(SI1, si_class);
|
|
CHECK(to<__si_class_type_info>(typeid(SI1)).__base_type == &typeid(A));
|
|
|
|
// SI2 has a single public empty base.
|
|
CHECK_VTABLE(SI2, si_class);
|
|
CHECK(to<__si_class_type_info>(typeid(SI2)).__base_type == &typeid(Empty));
|
|
|
|
// SI3 has a single public empty base. SI3 is dynamic whereas Empty is not, but since Empty is
|
|
// an empty class, it will still be at offset zero.
|
|
CHECK_VTABLE(SI3, si_class);
|
|
CHECK(to<__si_class_type_info>(typeid(SI3)).__base_type == &typeid(Empty));
|
|
|
|
// VMI1 has a single base, but it is private.
|
|
CHECK_VTABLE(VMI1, vmi_class);
|
|
|
|
// VMI2 has a single base, but it is virtual.
|
|
CHECK_VTABLE(VMI2, vmi_class);
|
|
|
|
// VMI3 has a single base, but VMI3 is dynamic whereas A is not, and A is not empty.
|
|
CHECK_VTABLE(VMI3, vmi_class);
|
|
|
|
// VMI4 has two bases.
|
|
CHECK_VTABLE(VMI4, vmi_class);
|
|
|
|
// VMI5 has non-diamond shaped inheritance.
|
|
CHECK_VTABLE(VMI5, vmi_class);
|
|
CHECK(to<__vmi_class_type_info>(typeid(VMI5)).__flags == __vmi_class_type_info::__non_diamond_repeat_mask);
|
|
CHECK(to<__vmi_class_type_info>(typeid(VMI5)).__base_count == 2);
|
|
CHECK_BASE_INFO_TYPE(VMI5, 0, VMIBase1);
|
|
CHECK_BASE_INFO_OFFSET_FLAGS(VMI5, 0, 0, __base_class_type_info::__public_mask);
|
|
CHECK_BASE_INFO_TYPE(VMI5, 1, VMIBase2);
|
|
CHECK_BASE_INFO_OFFSET_FLAGS(VMI5, 1, 4, __base_class_type_info::__public_mask);
|
|
|
|
// VMI6 has diamond shaped inheritance.
|
|
CHECK_VTABLE(VMI6, vmi_class);
|
|
CHECK(to<__vmi_class_type_info>(typeid(VMI6)).__flags == __vmi_class_type_info::__diamond_shaped_mask);
|
|
CHECK(to<__vmi_class_type_info>(typeid(VMI6)).__base_count == 2);
|
|
CHECK_BASE_INFO_TYPE(VMI6, 0, VMIBase1);
|
|
CHECK_BASE_INFO_OFFSET_FLAGS(VMI6, 0, -24, __base_class_type_info::__public_mask | __base_class_type_info::__virtual_mask);
|
|
CHECK_BASE_INFO_TYPE(VMI6, 1, VMIBase3);
|
|
CHECK_BASE_INFO_OFFSET_FLAGS(VMI6, 1, 0, __base_class_type_info::__public_mask);
|
|
|
|
// VMI7 has both non-diamond and diamond shaped inheritance.
|
|
CHECK_VTABLE(VMI7, vmi_class);
|
|
CHECK(to<__vmi_class_type_info>(typeid(VMI7)).__flags == (__vmi_class_type_info::__non_diamond_repeat_mask | __vmi_class_type_info::__diamond_shaped_mask));
|
|
CHECK(to<__vmi_class_type_info>(typeid(VMI7)).__base_count == 3);
|
|
CHECK_BASE_INFO_TYPE(VMI7, 0, VMIBase1);
|
|
CHECK_BASE_INFO_OFFSET_FLAGS(VMI7, 0, 16, __base_class_type_info::__public_mask);
|
|
CHECK_BASE_INFO_TYPE(VMI7, 1, VMI5);
|
|
CHECK_BASE_INFO_OFFSET_FLAGS(VMI7, 1, 20, __base_class_type_info::__public_mask);
|
|
CHECK_BASE_INFO_TYPE(VMI7, 2, VMI6);
|
|
CHECK_BASE_INFO_OFFSET_FLAGS(VMI7, 2, 0, 0);
|
|
|
|
// Pointers to incomplete classes.
|
|
CHECK_VTABLE(Incomplete *, pointer);
|
|
CHECK(to<__pbase_type_info>(typeid(Incomplete *)).__flags == __pbase_type_info::__incomplete_mask);
|
|
CHECK(to<__pbase_type_info>(typeid(Incomplete **)).__flags == __pbase_type_info::__incomplete_mask);
|
|
CHECK(to<__pbase_type_info>(typeid(Incomplete ***)).__flags == __pbase_type_info::__incomplete_mask);
|
|
|
|
// Member pointers.
|
|
CHECK_VTABLE(int Incomplete::*, pointer_to_member);
|
|
CHECK(to<__pbase_type_info>(typeid(int Incomplete::*)).__flags == __pbase_type_info::__incomplete_class_mask);
|
|
CHECK(to<__pbase_type_info>(typeid(Incomplete Incomplete::*)).__flags == (__pbase_type_info::__incomplete_class_mask | __pbase_type_info::__incomplete_mask));
|
|
CHECK(to<__pbase_type_info>(typeid(Incomplete A::*)).__flags == (__pbase_type_info::__incomplete_mask));
|
|
|
|
// Check that when stripping qualifiers off the pointee type, we correctly handle arrays.
|
|
CHECK(to<__pbase_type_info>(typeid(B::a)).__flags == (__pbase_type_info::__const_mask | __pbase_type_info::__volatile_mask));
|
|
CHECK(to<__pbase_type_info>(typeid(B::a)).__pointee == to<__pbase_type_info>(typeid(B::b)).__pointee);
|
|
CHECK(to<__pbase_type_info>(typeid(B::c)).__flags == (__pbase_type_info::__const_mask | __pbase_type_info::__volatile_mask));
|
|
CHECK(to<__pbase_type_info>(typeid(B::c)).__pointee == to<__pbase_type_info>(typeid(B::d)).__pointee);
|
|
|
|
// Success!
|
|
// CHECK: ret i32 0
|
|
return 0;
|
|
}
|
|
|
|
#ifdef HARNESS
|
|
extern "C" void printf(const char *, ...);
|
|
|
|
int main() {
|
|
int result = f();
|
|
|
|
if (result == 0)
|
|
printf("success!\n");
|
|
else
|
|
printf("test on line %d failed!\n", result);
|
|
|
|
return result;
|
|
}
|
|
#endif
|
|
|
|
|