133 lines
		
	
	
		
			2.9 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			133 lines
		
	
	
		
			2.9 KiB
		
	
	
	
		
			C++
		
	
	
	
| // 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) {}
 | |
| }
 |