annotate gcc/testsuite/g++.dg/cpp2a/constexpr-dynamic4.C @ 145:1830386684a0

gcc-9.2.0
author anatofuz
date Thu, 13 Feb 2020 11:34:05 +0900
parents
children
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
145
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
1 // PR c++/88337 - Implement P1327R1: Allow dynamic_cast/typeid in constexpr.
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
2 // { dg-do compile { target c++2a } }
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
3 // { dg-additional-options "-fdelete-null-pointer-checks" }
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
4
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
5 // From clang's constant-expression-cxx2a.cpp.
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
6
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
7 struct A2 { virtual void a2(); };
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
8 struct A : A2 { virtual void a(); };
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
9 struct B : A {};
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
10 struct C2 { virtual void c2(); };
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
11 struct C : A, C2 { A *c = dynamic_cast<A*>(static_cast<C2*>(this)); };
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
12 struct D { virtual void d(); };
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
13 struct E { virtual void e(); };
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
14 struct F : B, C, D, private E { void *f = dynamic_cast<void*>(static_cast<D*>(this)); };
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
15 struct Padding { virtual void padding(); };
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
16 struct G : Padding, F {};
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
17
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
18 constexpr G g;
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
19
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
20 // During construction of C, A is unambiguous subobject of dynamic type C.
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
21 static_assert(g.c == (C*)&g);
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
22 // ... but in the complete object, the same is not true, so the runtime fails.
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
23 static_assert(dynamic_cast<const A*>(static_cast<const C2*>(&g)) == nullptr);
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
24
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
25 // dynamic_cast<void*> produces a pointer to the object of the dynamic type.
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
26 static_assert(g.f == (void*)(F*)&g);
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
27 static_assert(dynamic_cast<const void*>(static_cast<const D*>(&g)) == &g);
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
28
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
29 constexpr int d_a = (dynamic_cast<const A&>(static_cast<const D&>(g)), 0); // { dg-error "reference .dynamic_cast. failed" }
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
30 // { dg-message ".A. is an ambiguous base class of dynamic type .G." "" { target *-*-* } .-1 }
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
31
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
32 // Can navigate from A2 to its A...
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
33 static_assert(&dynamic_cast<A&>((A2&)(B&)g) == &(A&)(B&)g);
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
34 // ... and from B to its A ...
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
35 static_assert(&dynamic_cast<A&>((B&)g) == &(A&)(B&)g);
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
36 // ... but not from D.
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
37 static_assert(&dynamic_cast<A&>((D&)g) == &(A&)(B&)g); // { dg-error "non-constant condition for static assertion|reference .dynamic_cast. failed" }
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
38 // { dg-message ".A. is an ambiguous base class of dynamic type .G." "" { target *-*-* } .-1 }
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
39
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
40 // Can cast from A2 to sibling class D.
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
41 static_assert(&dynamic_cast<D&>((A2&)(B&)g) == &(D&)g);
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
42
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
43 // Cannot cast from private base E to derived class F.
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
44 constexpr int e_f = (dynamic_cast<F&>((E&)g), 0); // { dg-error "reference .dynamic_cast. failed" }
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
45 // { dg-message "static type .const E. of its operand is a non-public base class of dynamic type .G." "" { target *-*-* } .-1 }
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
46
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
47 // Cannot cast from B to private sibling E.
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
48 constexpr int b_e = (dynamic_cast<E&>((B&)g), 0); // { dg-error "reference .dynamic_cast. failed" }
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
49 // { dg-message "dynamic type .G. of its operand does not have an unambiguous public base class .E." "" { target *-*-* } .-1 }
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
50
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
51 struct Unrelated { virtual void unrelated(); };
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
52
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
53 constexpr int b_unrelated = (dynamic_cast<Unrelated&>((B&)g), 0); // { dg-error "reference .dynamic_cast. failed" }
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
54 // { dg-message "dynamic type .G. of its operand does not have an unambiguous public base class .Unrelated." "" { target *-*-* } .-1 }
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
55 constexpr int e_unrelated = (dynamic_cast<Unrelated&>((E&)g), 0); // { dg-error "reference .dynamic_cast. failed" }
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
56 // { dg-message "static type .const E. of its operand is a non-public base class of dynamic type .G." "" { target *-*-* } .-1 }