Hans Wennborg | c9bd88e | 2014-01-14 19:35:09 +0000 | [diff] [blame] | 1 | // RUN: %clang_cc1 -triple %itanium_abi_triple -fsyntax-only %s |
Hans Wennborg | 3d79154 | 2014-02-24 15:58:24 +0000 | [diff] [blame^] | 2 | // RUN: %clang_cc1 -triple %ms_abi_triple -verify %s |
Reid Kleckner | 23f4c4b | 2013-06-21 12:45:15 +0000 | [diff] [blame] | 3 | |
| 4 | namespace Test1 { |
Peter Collingbourne | b289fe6 | 2013-05-20 14:12:25 +0000 | [diff] [blame] | 5 | |
| 6 | // Should be accepted under the Itanium ABI (first RUN line) but rejected |
| 7 | // under the Microsoft ABI (second RUN line), as Microsoft ABI requires |
Hans Wennborg | 3d79154 | 2014-02-24 15:58:24 +0000 | [diff] [blame^] | 8 | // operator delete() lookups to be done when vtables are marked used. |
Peter Collingbourne | b289fe6 | 2013-05-20 14:12:25 +0000 | [diff] [blame] | 9 | |
| 10 | struct A { |
| 11 | void operator delete(void *); // expected-note {{member found by ambiguous name lookup}} |
| 12 | }; |
| 13 | |
| 14 | struct B { |
| 15 | void operator delete(void *); // expected-note {{member found by ambiguous name lookup}} |
| 16 | }; |
| 17 | |
| 18 | struct C : A, B { |
| 19 | ~C(); |
| 20 | }; |
| 21 | |
| 22 | struct VC : A, B { |
| 23 | virtual ~VC(); // expected-error {{member 'operator delete' found in multiple base classes of different types}} |
| 24 | }; |
Reid Kleckner | 23f4c4b | 2013-06-21 12:45:15 +0000 | [diff] [blame] | 25 | |
Hans Wennborg | 3d79154 | 2014-02-24 15:58:24 +0000 | [diff] [blame^] | 26 | void f(VC vc) { |
| 27 | // This marks VC's vtable used. |
| 28 | } |
| 29 | |
Reid Kleckner | 23f4c4b | 2013-06-21 12:45:15 +0000 | [diff] [blame] | 30 | } |
| 31 | |
| 32 | namespace Test2 { |
| 33 | |
| 34 | // In the MSVC ABI, functions must destroy their aggregate arguments. foo |
| 35 | // requires a dtor for B, but we can't implicitly define it because ~A is |
| 36 | // private. bar should be able to call A's private dtor without error, even |
| 37 | // though MSVC rejects bar. |
Reid Kleckner | 23f4c4b | 2013-06-21 12:45:15 +0000 | [diff] [blame] | 38 | class A { |
| 39 | private: |
Hans Wennborg | 0f3c10c | 2014-01-13 17:23:24 +0000 | [diff] [blame] | 40 | ~A(); // expected-note {{declared private here}} |
Reid Kleckner | 23f4c4b | 2013-06-21 12:45:15 +0000 | [diff] [blame] | 41 | int a; |
| 42 | }; |
| 43 | |
| 44 | struct B : public A { // expected-error {{base class 'Test2::A' has private destructor}} |
| 45 | int b; |
| 46 | }; |
| 47 | |
| 48 | struct C { |
| 49 | ~C(); |
| 50 | int c; |
| 51 | }; |
| 52 | |
| 53 | struct D { |
| 54 | // D has a non-trivial implicit dtor that destroys C. |
| 55 | C o; |
| 56 | }; |
| 57 | |
| 58 | void foo(B b) { } // expected-note {{implicit destructor for 'Test2::B' first required here}} |
Hans Wennborg | 0f3c10c | 2014-01-13 17:23:24 +0000 | [diff] [blame] | 59 | void bar(A a) { } // no error; MSVC rejects this, but we skip the direct access check. |
Reid Kleckner | 23f4c4b | 2013-06-21 12:45:15 +0000 | [diff] [blame] | 60 | void baz(D d) { } // no error |
| 61 | |
| 62 | } |
| 63 | |
| 64 | #ifdef MSVC_ABI |
| 65 | namespace Test3 { |
| 66 | |
| 67 | class A { |
| 68 | A(); |
Hans Wennborg | 0f3c10c | 2014-01-13 17:23:24 +0000 | [diff] [blame] | 69 | ~A(); // expected-note {{implicitly declared private here}} |
Reid Kleckner | 23f4c4b | 2013-06-21 12:45:15 +0000 | [diff] [blame] | 70 | friend void bar(A); |
| 71 | int a; |
| 72 | }; |
| 73 | |
| 74 | void bar(A a) { } |
Hans Wennborg | 0f3c10c | 2014-01-13 17:23:24 +0000 | [diff] [blame] | 75 | void baz(A a) { } // no error; MSVC rejects this, but the standard allows it. |
Reid Kleckner | 23f4c4b | 2013-06-21 12:45:15 +0000 | [diff] [blame] | 76 | |
| 77 | // MSVC accepts foo() but we reject it for consistency with Itanium. MSVC also |
| 78 | // rejects this if A has a copy ctor or if we call A's ctor. |
| 79 | void foo(A *a) { |
| 80 | bar(*a); // expected-error {{temporary of type 'Test3::A' has private destructor}} |
| 81 | } |
| 82 | } |
| 83 | #endif |
| 84 | |
| 85 | namespace Test4 { |
| 86 | // Don't try to access the dtor of an incomplete on a function declaration. |
| 87 | class A; |
| 88 | void foo(A a); |
| 89 | } |