1 //===------------------------- incomplete_type.cpp --------------------------===// 2 // 3 // The LLVM Compiler Infrastructure 4 // 5 // This file is dual licensed under the MIT and the University of Illinois Open 6 // Source Licenses. See LICENSE.TXT for details. 7 // 8 //===----------------------------------------------------------------------===// 9 // http://mentorembedded.github.io/cxx-abi/abi.html#rtti-layout 10 11 // Two abi::__pbase_type_info objects can always be compared for equality 12 // (i.e. of the types represented) or ordering by comparison of their name 13 // NTBS addresses. In addition, unless either or both have either of the 14 // incomplete flags set, equality can be tested by comparing the type_info 15 // addresses. 16 17 // UNSUPPORTED: libcxxabi-no-exceptions 18 19 // RUN: %cxx %flags %compile_flags -c %s -o %t.one.o 20 // RUN: %cxx %flags %compile_flags -c %s -o %t.two.o -DTU_ONE 21 // RUN: %cxx %flags %t.one.o %t.two.o %link_flags -o %t.exe 22 // RUN: %exec %t.exe 23 24 #include <stdio.h> 25 #include <cstring> 26 #include <cassert> 27 #include <typeinfo> 28 29 // Check that the addresses of the typeinfo differ but still compare equal 30 // via their NTBS. 31 inline void 32 AssertIncompleteTypeInfoEquals(std::type_info const& LHS, std::type_info const& RHS) 33 { 34 assert(&LHS != &RHS); 35 assert(strcmp(LHS.name(), RHS.name()) == 0); 36 } 37 38 struct NeverDefined; 39 void ThrowNeverDefinedMP(); 40 std::type_info const& ReturnTypeInfoNeverDefinedMP(); 41 42 struct IncompleteAtThrow; 43 void ThrowIncompleteMP(); 44 void ThrowIncompletePP(); 45 void ThrowIncompletePMP(); 46 std::type_info const& ReturnTypeInfoIncompleteMP(); 47 std::type_info const& ReturnTypeInfoIncompletePP(); 48 49 struct CompleteAtThrow; 50 void ThrowCompleteMP(); 51 void ThrowCompletePP(); 52 void ThrowCompletePMP(); 53 std::type_info const& ReturnTypeInfoCompleteMP(); 54 std::type_info const& ReturnTypeInfoCompletePP(); 55 56 void ThrowNullptr(); 57 58 #ifndef TU_ONE 59 60 void ThrowNeverDefinedMP() { throw (int NeverDefined::*)nullptr; } 61 std::type_info const& ReturnTypeInfoNeverDefinedMP() { return typeid(int NeverDefined::*); } 62 63 void ThrowIncompleteMP() { throw (int IncompleteAtThrow::*)nullptr; } 64 void ThrowIncompletePP() { throw (IncompleteAtThrow**)nullptr; } 65 void ThrowIncompletePMP() { throw (int IncompleteAtThrow::**)nullptr; } 66 std::type_info const& ReturnTypeInfoIncompleteMP() { return typeid(int IncompleteAtThrow::*); } 67 std::type_info const& ReturnTypeInfoIncompletePP() { return typeid(IncompleteAtThrow**); } 68 69 struct CompleteAtThrow {}; 70 void ThrowCompleteMP() { throw (int CompleteAtThrow::*)nullptr; } 71 void ThrowCompletePP() { throw (CompleteAtThrow**)nullptr; } 72 void ThrowCompletePMP() { throw (int CompleteAtThrow::**)nullptr; } 73 std::type_info const& ReturnTypeInfoCompleteMP() { return typeid(int CompleteAtThrow::*); } 74 std::type_info const& ReturnTypeInfoCompletePP() { return typeid(CompleteAtThrow**); } 75 76 void ThrowNullptr() { throw nullptr; } 77 78 #else 79 80 struct IncompleteAtThrow {}; 81 82 int main() { 83 AssertIncompleteTypeInfoEquals(ReturnTypeInfoNeverDefinedMP(), typeid(int NeverDefined::*)); 84 try { 85 ThrowNeverDefinedMP(); 86 assert(false); 87 } catch (int IncompleteAtThrow::*) { 88 assert(false); 89 } catch (int CompleteAtThrow::*) { 90 assert(false); 91 } catch (int NeverDefined::*p) { 92 assert(!p); 93 } 94 AssertIncompleteTypeInfoEquals(ReturnTypeInfoIncompleteMP(), typeid(int IncompleteAtThrow::*)); 95 try { 96 ThrowIncompleteMP(); 97 assert(false); 98 } catch (CompleteAtThrow**) { 99 assert(false); 100 } catch (int CompleteAtThrow::*) { 101 assert(false); 102 } catch (IncompleteAtThrow**) { 103 assert(false); 104 } catch (int IncompleteAtThrow::*p) { 105 assert(!p); 106 } 107 108 AssertIncompleteTypeInfoEquals(ReturnTypeInfoIncompletePP(), typeid(IncompleteAtThrow**)); 109 try { 110 ThrowIncompletePP(); 111 assert(false); 112 } catch (int IncompleteAtThrow::*) { 113 assert(false); 114 } catch (IncompleteAtThrow** p) { 115 assert(!p); 116 } 117 118 try { 119 ThrowIncompletePMP(); 120 assert(false); 121 } catch (int IncompleteAtThrow::*) { 122 assert(false); 123 } catch (IncompleteAtThrow**) { 124 assert(false); 125 } catch (int IncompleteAtThrow::**p) { 126 assert(!p); 127 } 128 129 AssertIncompleteTypeInfoEquals(ReturnTypeInfoCompleteMP(), typeid(int CompleteAtThrow::*)); 130 try { 131 ThrowCompleteMP(); 132 assert(false); 133 } catch (IncompleteAtThrow**) { 134 assert(false); 135 } catch (int IncompleteAtThrow::*) { 136 assert(false); 137 } catch (CompleteAtThrow**) { 138 assert(false); 139 } catch (int CompleteAtThrow::*p) { 140 assert(!p); 141 } 142 143 AssertIncompleteTypeInfoEquals(ReturnTypeInfoCompletePP(), typeid(CompleteAtThrow**)); 144 try { 145 ThrowCompletePP(); 146 assert(false); 147 } catch (IncompleteAtThrow**) { 148 assert(false); 149 } catch (int IncompleteAtThrow::*) { 150 assert(false); 151 } catch (int CompleteAtThrow::*) { 152 assert(false); 153 } catch (CompleteAtThrow**p) { 154 assert(!p); 155 } 156 157 try { 158 ThrowCompletePMP(); 159 assert(false); 160 } catch (IncompleteAtThrow**) { 161 assert(false); 162 } catch (int IncompleteAtThrow::*) { 163 assert(false); 164 } catch (int CompleteAtThrow::*) { 165 assert(false); 166 } catch (CompleteAtThrow**) { 167 assert(false); 168 } catch (int CompleteAtThrow::**p) { 169 assert(!p); 170 } 171 172 #if __cplusplus >= 201103L 173 // Catch nullptr as complete type 174 try { 175 ThrowNullptr(); 176 } catch (int IncompleteAtThrow::*p) { 177 assert(!p); 178 } 179 180 // Catch nullptr as an incomplete type 181 try { 182 ThrowNullptr(); 183 } catch (int CompleteAtThrow::*p) { 184 assert(!p); 185 } 186 // Catch nullptr as a type that is never complete. 187 try { 188 ThrowNullptr(); 189 } catch (int NeverDefined::*p) { 190 assert(!p); 191 } 192 #endif 193 } 194 #endif 195