2018-05-19 04:42:53 +08:00
|
|
|
//===------------------------- dynamic_cast.pass.cpp ----------------------===//
|
|
|
|
//
|
2019-01-19 18:56:40 +08:00
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
2018-05-19 04:42:53 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2020-07-09 04:38:54 +08:00
|
|
|
// PR33425 and PR33487 are not fixed until the dylib shipped with macOS 10.15
|
2021-06-19 01:33:14 +08:00
|
|
|
// XFAIL: use_system_cxx_lib && target={{.+}}-apple-macosx10.14
|
2020-07-09 04:38:54 +08:00
|
|
|
|
|
|
|
// PR33439 isn't fixed until the dylib shipped with macOS 10.14
|
2021-06-19 01:33:14 +08:00
|
|
|
// XFAIL: use_system_cxx_lib && target={{.+}}-apple-macosx10.{{9|10|11|12|13}}
|
2020-07-09 04:38:54 +08:00
|
|
|
|
2018-05-19 04:42:53 +08:00
|
|
|
#include <cassert>
|
|
|
|
|
|
|
|
// This test explicitly tests dynamic cast with types that have inaccessible
|
|
|
|
// bases.
|
|
|
|
#if defined(__clang__)
|
2020-10-31 05:33:02 +08:00
|
|
|
# pragma clang diagnostic ignored "-Winaccessible-base"
|
2021-08-20 23:39:46 +08:00
|
|
|
#elif defined(__GNUC__)
|
2020-10-31 05:33:02 +08:00
|
|
|
# pragma GCC diagnostic ignored "-Winaccessible-base"
|
2018-05-19 04:42:53 +08:00
|
|
|
#endif
|
|
|
|
|
|
|
|
typedef char Pad1[43981];
|
|
|
|
typedef char Pad2[34981];
|
|
|
|
typedef char Pad3[93481];
|
|
|
|
typedef char Pad4[13489];
|
|
|
|
typedef char Pad5[81349];
|
|
|
|
typedef char Pad6[34819];
|
|
|
|
typedef char Pad7[3489];
|
|
|
|
|
|
|
|
namespace t1
|
|
|
|
{
|
|
|
|
|
|
|
|
// PR33425
|
|
|
|
struct C3 { virtual ~C3() {} Pad1 _; };
|
|
|
|
struct C5 : protected virtual C3 { Pad2 _; };
|
|
|
|
struct C6 : virtual C5 { Pad3 _; };
|
|
|
|
struct C7 : virtual C3 { Pad4 _; };
|
|
|
|
struct C9 : C6, C7 { Pad5 _; };
|
|
|
|
|
|
|
|
C9 c9;
|
|
|
|
C3 *c3 = &c9;
|
|
|
|
|
|
|
|
void test()
|
|
|
|
{
|
|
|
|
assert(dynamic_cast<C3*>(c3) == static_cast<C3*>(&c9));
|
|
|
|
assert(dynamic_cast<C5*>(c3) == static_cast<C5*>(&c9));
|
|
|
|
assert(dynamic_cast<C6*>(c3) == static_cast<C6*>(&c9));
|
|
|
|
assert(dynamic_cast<C7*>(c3) == static_cast<C7*>(&c9));
|
|
|
|
assert(dynamic_cast<C9*>(c3) == static_cast<C9*>(&c9));
|
|
|
|
}
|
|
|
|
|
|
|
|
} // t1
|
|
|
|
|
|
|
|
namespace t2
|
|
|
|
{
|
|
|
|
|
|
|
|
// PR33425
|
|
|
|
struct Src { virtual ~Src() {} Pad1 _; };
|
|
|
|
struct Mask : protected virtual Src { Pad2 _; };
|
|
|
|
struct Dest : Mask { Pad3 _; };
|
|
|
|
struct Root : Dest, virtual Src { Pad4 _; };
|
|
|
|
|
|
|
|
Root root;
|
|
|
|
Src *src = &root;
|
|
|
|
|
|
|
|
void test()
|
|
|
|
{
|
|
|
|
assert(dynamic_cast<Src*>(src) == static_cast<Src*>(&root));
|
|
|
|
assert(dynamic_cast<Mask*>(src) == static_cast<Mask*>(&root));
|
|
|
|
assert(dynamic_cast<Dest*>(src) == static_cast<Dest*>(&root));
|
|
|
|
assert(dynamic_cast<Root*>(src) == static_cast<Root*>(&root));
|
|
|
|
}
|
|
|
|
|
|
|
|
} // t2
|
|
|
|
|
|
|
|
namespace t3
|
|
|
|
{
|
|
|
|
|
|
|
|
// PR33487
|
|
|
|
struct Class1 { virtual ~Class1() {} Pad1 _; };
|
|
|
|
struct Shared : virtual Class1 { Pad2 _; };
|
|
|
|
struct Class6 : virtual Shared { Pad3 _; };
|
|
|
|
struct Left : Class6 { Pad4 _; };
|
|
|
|
struct Right : Class6 { Pad5 _; };
|
|
|
|
struct Main : Left, Right { Pad6 _; };
|
|
|
|
|
|
|
|
Main m;
|
|
|
|
Class1 *c1 = &m;
|
|
|
|
|
|
|
|
void test()
|
|
|
|
{
|
|
|
|
assert(dynamic_cast<Class1*>(c1) == static_cast<Class1*>(&m));
|
|
|
|
assert(dynamic_cast<Shared*>(c1) == static_cast<Shared*>(&m));
|
|
|
|
assert(dynamic_cast<Class6*>(c1) == 0);
|
|
|
|
assert(dynamic_cast<Left*>(c1) == static_cast<Left*>(&m));
|
|
|
|
assert(dynamic_cast<Right*>(c1) == static_cast<Right*>(&m));
|
|
|
|
assert(dynamic_cast<Main*>(c1) == static_cast<Main*>(&m));
|
|
|
|
}
|
|
|
|
|
|
|
|
} // t3
|
|
|
|
|
2018-05-19 04:51:38 +08:00
|
|
|
namespace t4
|
|
|
|
{
|
|
|
|
|
|
|
|
// PR33439
|
|
|
|
struct C2 { virtual ~C2() {} Pad1 _; };
|
|
|
|
struct C3 { virtual ~C3() {} Pad2 _; };
|
|
|
|
struct C4 : C3 { Pad3 _; };
|
|
|
|
struct C8 : C2, virtual C4 { Pad4 _; };
|
|
|
|
struct C9 : C4, C8 { Pad5 _; };
|
|
|
|
|
|
|
|
C9 c9;
|
|
|
|
C2 *c2 = &c9;
|
|
|
|
|
|
|
|
void test()
|
|
|
|
{
|
|
|
|
assert(dynamic_cast<C2*>(c2) == static_cast<C2*>(&c9));
|
|
|
|
assert(dynamic_cast<C3*>(c2) == 0);
|
|
|
|
assert(dynamic_cast<C4*>(c2) == 0);
|
|
|
|
assert(dynamic_cast<C8*>(c2) == static_cast<C8*>(&c9));
|
|
|
|
assert(dynamic_cast<C9*>(c2) == static_cast<C9*>(&c9));
|
|
|
|
}
|
|
|
|
|
|
|
|
} // t4
|
|
|
|
|
|
|
|
namespace t5
|
|
|
|
{
|
|
|
|
|
|
|
|
// PR33439
|
|
|
|
struct Dummy { virtual ~Dummy() {} Pad1 _; };
|
|
|
|
struct Src { virtual ~Src() {} Pad2 _; };
|
|
|
|
struct Dest : Dummy { Pad3 _; };
|
|
|
|
struct A1 : Dest { Pad4 _; };
|
|
|
|
struct A2 : Dest { Pad5 _; };
|
|
|
|
struct Root : Src, A1, A2 { Pad6 _; };
|
|
|
|
|
|
|
|
Root root;
|
|
|
|
Src *src = &root;
|
|
|
|
|
|
|
|
void test()
|
|
|
|
{
|
|
|
|
assert(dynamic_cast<Dummy*>(src) == 0);
|
|
|
|
assert(dynamic_cast<Src*>(src) == static_cast<Src*>(&root));
|
|
|
|
assert(dynamic_cast<Dest*>(src) == 0);
|
|
|
|
assert(dynamic_cast<A1*>(src) == static_cast<A1*>(&root));
|
|
|
|
assert(dynamic_cast<A2*>(src) == static_cast<A2*>(&root));
|
|
|
|
}
|
|
|
|
|
|
|
|
} // t5
|
|
|
|
|
2020-10-09 01:36:33 +08:00
|
|
|
int main(int, char**)
|
2018-05-19 04:42:53 +08:00
|
|
|
{
|
|
|
|
t1::test();
|
|
|
|
t2::test();
|
|
|
|
t3::test();
|
2018-05-19 04:51:38 +08:00
|
|
|
t4::test();
|
|
|
|
t5::test();
|
2020-10-09 01:36:33 +08:00
|
|
|
|
|
|
|
return 0;
|
2018-05-19 04:42:53 +08:00
|
|
|
}
|