dynamic_cast.pass.cpp
1 //===------------------------- dynamic_cast.pass.cpp ----------------------===// 2 // 3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4 // See https://llvm.org/LICENSE.txt for license information. 5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6 // 7 //===----------------------------------------------------------------------===// 8 9 // XFAIL: gcc-7, gcc-8, gcc-9 10 11 // PR33425 and PR33487 are not fixed until the dylib shipped with macOS 10.15 12 // XFAIL: use_system_cxx_lib && target={{.+}}-apple-macosx10.14 13 14 // PR33439 isn't fixed until the dylib shipped with macOS 10.14 15 // XFAIL: use_system_cxx_lib && target={{.+}}-apple-macosx10.{{9|10|11|12|13}} 16 17 #include <cassert> 18 19 // This test explicitly tests dynamic cast with types that have inaccessible 20 // bases. 21 #if defined(__clang__) 22 # pragma clang diagnostic ignored "-Winaccessible-base" 23 #elif defined(__GNUC__) && (__GNUC__ >= 10) 24 # pragma GCC diagnostic ignored "-Winaccessible-base" 25 #endif 26 27 typedef char Pad1[43981]; 28 typedef char Pad2[34981]; 29 typedef char Pad3[93481]; 30 typedef char Pad4[13489]; 31 typedef char Pad5[81349]; 32 typedef char Pad6[34819]; 33 typedef char Pad7[3489]; 34 35 namespace t1 36 { 37 38 // PR33425 39 struct C3 { virtual ~C3() {} Pad1 _; }; 40 struct C5 : protected virtual C3 { Pad2 _; }; 41 struct C6 : virtual C5 { Pad3 _; }; 42 struct C7 : virtual C3 { Pad4 _; }; 43 struct C9 : C6, C7 { Pad5 _; }; 44 45 C9 c9; 46 C3 *c3 = &c9; 47 48 void test() 49 { 50 assert(dynamic_cast<C3*>(c3) == static_cast<C3*>(&c9)); 51 assert(dynamic_cast<C5*>(c3) == static_cast<C5*>(&c9)); 52 assert(dynamic_cast<C6*>(c3) == static_cast<C6*>(&c9)); 53 assert(dynamic_cast<C7*>(c3) == static_cast<C7*>(&c9)); 54 assert(dynamic_cast<C9*>(c3) == static_cast<C9*>(&c9)); 55 } 56 57 } // t1 58 59 namespace t2 60 { 61 62 // PR33425 63 struct Src { virtual ~Src() {} Pad1 _; }; 64 struct Mask : protected virtual Src { Pad2 _; }; 65 struct Dest : Mask { Pad3 _; }; 66 struct Root : Dest, virtual Src { Pad4 _; }; 67 68 Root root; 69 Src *src = &root; 70 71 void test() 72 { 73 assert(dynamic_cast<Src*>(src) == static_cast<Src*>(&root)); 74 assert(dynamic_cast<Mask*>(src) == static_cast<Mask*>(&root)); 75 assert(dynamic_cast<Dest*>(src) == static_cast<Dest*>(&root)); 76 assert(dynamic_cast<Root*>(src) == static_cast<Root*>(&root)); 77 } 78 79 } // t2 80 81 namespace t3 82 { 83 84 // PR33487 85 struct Class1 { virtual ~Class1() {} Pad1 _; }; 86 struct Shared : virtual Class1 { Pad2 _; }; 87 struct Class6 : virtual Shared { Pad3 _; }; 88 struct Left : Class6 { Pad4 _; }; 89 struct Right : Class6 { Pad5 _; }; 90 struct Main : Left, Right { Pad6 _; }; 91 92 Main m; 93 Class1 *c1 = &m; 94 95 void test() 96 { 97 assert(dynamic_cast<Class1*>(c1) == static_cast<Class1*>(&m)); 98 assert(dynamic_cast<Shared*>(c1) == static_cast<Shared*>(&m)); 99 assert(dynamic_cast<Class6*>(c1) == 0); 100 assert(dynamic_cast<Left*>(c1) == static_cast<Left*>(&m)); 101 assert(dynamic_cast<Right*>(c1) == static_cast<Right*>(&m)); 102 assert(dynamic_cast<Main*>(c1) == static_cast<Main*>(&m)); 103 } 104 105 } // t3 106 107 namespace t4 108 { 109 110 // PR33439 111 struct C2 { virtual ~C2() {} Pad1 _; }; 112 struct C3 { virtual ~C3() {} Pad2 _; }; 113 struct C4 : C3 { Pad3 _; }; 114 struct C8 : C2, virtual C4 { Pad4 _; }; 115 struct C9 : C4, C8 { Pad5 _; }; 116 117 C9 c9; 118 C2 *c2 = &c9; 119 120 void test() 121 { 122 assert(dynamic_cast<C2*>(c2) == static_cast<C2*>(&c9)); 123 assert(dynamic_cast<C3*>(c2) == 0); 124 assert(dynamic_cast<C4*>(c2) == 0); 125 assert(dynamic_cast<C8*>(c2) == static_cast<C8*>(&c9)); 126 assert(dynamic_cast<C9*>(c2) == static_cast<C9*>(&c9)); 127 } 128 129 } // t4 130 131 namespace t5 132 { 133 134 // PR33439 135 struct Dummy { virtual ~Dummy() {} Pad1 _; }; 136 struct Src { virtual ~Src() {} Pad2 _; }; 137 struct Dest : Dummy { Pad3 _; }; 138 struct A1 : Dest { Pad4 _; }; 139 struct A2 : Dest { Pad5 _; }; 140 struct Root : Src, A1, A2 { Pad6 _; }; 141 142 Root root; 143 Src *src = &root; 144 145 void test() 146 { 147 assert(dynamic_cast<Dummy*>(src) == 0); 148 assert(dynamic_cast<Src*>(src) == static_cast<Src*>(&root)); 149 assert(dynamic_cast<Dest*>(src) == 0); 150 assert(dynamic_cast<A1*>(src) == static_cast<A1*>(&root)); 151 assert(dynamic_cast<A2*>(src) == static_cast<A2*>(&root)); 152 } 153 154 } // t5 155 156 int main(int, char**) 157 { 158 t1::test(); 159 t2::test(); 160 t3::test(); 161 t4::test(); 162 t5::test(); 163 164 return 0; 165 }