// RUN: %clang_cc1 -fexceptions -fcxx-exceptions -std=c++11 -triple %itanium_abi_triple -fsyntax-only %s // RUN: %clang_cc1 -fexceptions -fcxx-exceptions -std=c++11 -triple %ms_abi_triple -verify %s namespace Test1 { // Should be accepted under the Itanium ABI (first RUN line) but rejected // under the Microsoft ABI (second RUN line), as Microsoft ABI requires // operator delete() lookups to be done when vtables are marked used. struct A { void operator delete(void *); // expected-note {{member found by ambiguous name lookup}} }; struct B { void operator delete(void *); // expected-note {{member found by ambiguous name lookup}} }; struct C : A, B { ~C(); }; struct VC : A, B { virtual ~VC(); // expected-error {{member 'operator delete' found in multiple base classes of different types}} }; void f() { // This marks VC's vtable used. VC vc; } } namespace Test2 { // In the MSVC ABI, functions must destroy their aggregate arguments. foo // requires a dtor for B, but we can't implicitly define it because ~A is // private. bar should be able to call A's private dtor without error, even // though MSVC rejects bar. class A { private: ~A(); int a; }; struct B : public A { // expected-note {{destructor of 'B' is implicitly deleted because base class 'Test2::A' has an inaccessible destructor}} int b; }; struct C { ~C(); int c; }; struct D { // D has a non-trivial implicit dtor that destroys C. C o; }; void foo(B b) { } // expected-error {{attempt to use a deleted function}} void bar(A a) { } // no error; MSVC rejects this, but we skip the direct access check. void baz(D d) { } // no error } #ifdef MSVC_ABI namespace Test3 { class A { A(); ~A(); // expected-note {{implicitly declared private here}} friend void bar(A); int a; }; void bar(A a) { } void baz(A a) { } // no error; MSVC rejects this, but the standard allows it. // MSVC accepts foo() but we reject it for consistency with Itanium. MSVC also // rejects this if A has a copy ctor or if we call A's ctor. void foo(A *a) { bar(*a); // expected-error {{temporary of type 'Test3::A' has private destructor}} } } #endif namespace Test4 { // Don't try to access the dtor of an incomplete on a function declaration. class A; void foo(A a); } #ifdef MSVC_ABI namespace Test5 { // Do the operator delete access control check from the context of the dtor. class A { protected: void operator delete(void *); }; class B : public A { virtual ~B(); }; B *test() { // Previously, marking the vtable used here would do the operator delete // lookup from this context, which doesn't have access. return new B; } } #endif namespace Test6 { class A { protected: void operator delete(void *); }; class B : public A { virtual ~B(); public: virtual void m_fn1(); }; void fn1(B *b) { b->m_fn1(); } } namespace Test7 { class A { protected: void operator delete(void *); }; struct B : public A { virtual ~B(); }; void fn1(B b) {} }