1 | // RUN: %clang_cc1 -verify -std=c++11 %s |
2 | // expected-no-diagnostics |
3 | template <typename T> struct OwnPtr { |
4 | T *p; |
5 | ~OwnPtr() { |
6 | static_assert(sizeof(T) > 0, "incomplete T"); |
7 | delete p; |
8 | } |
9 | }; |
10 | |
11 | namespace use_vtable_for_vcall { |
12 | struct Incomplete; |
13 | struct A { |
14 | virtual ~A() {} |
15 | virtual void m() {} |
16 | }; |
17 | struct B : A { |
18 | B(); |
19 | virtual void m() { } |
20 | virtual void m2() { static_cast<A *>(this)->m(); } |
21 | OwnPtr<Incomplete> m_sqlError; |
22 | }; |
23 | |
24 | void f() { |
25 | // Since B's constructor is declared out of line, nothing in this file |
26 | // references a vtable, so the destructor doesn't get built. |
27 | A *b = new B(); |
28 | b->m(); |
29 | delete b; |
30 | } |
31 | } |
32 | |
33 | namespace dont_mark_qualified_vcall { |
34 | struct Incomplete; |
35 | struct A { |
36 | virtual ~A() {} |
37 | virtual void m() {} |
38 | }; |
39 | struct B : A { |
40 | B(); |
41 | // Previously we would mark B's vtable referenced to devirtualize this call to |
42 | // A::m, even though it's not a virtual call. |
43 | virtual void m() { A::m(); } |
44 | OwnPtr<Incomplete> m_sqlError; |
45 | }; |
46 | |
47 | B *f() { |
48 | return new B(); |
49 | } |
50 | } |
51 | |