forked from OSchip/llvm-project
				
			
		
			
				
	
	
		
			188 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			188 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			C++
		
	
	
	
| // RUN: %clang_cc1 -analyze -analyzer-checker=alpha.cplusplus.DeleteWithNonVirtualDtor -std=c++11 -verify -analyzer-output=text %s
 | |
| 
 | |
| struct Virtual {
 | |
|   virtual ~Virtual() {}
 | |
| };
 | |
| 
 | |
| struct VDerived : public Virtual {};
 | |
| 
 | |
| struct NonVirtual {
 | |
|   ~NonVirtual() {}
 | |
| };
 | |
| 
 | |
| struct NVDerived : public NonVirtual {};
 | |
| struct NVDoubleDerived : public NVDerived {};
 | |
| 
 | |
| struct Base {
 | |
|   virtual void destroy() = 0;
 | |
| };
 | |
| 
 | |
| class PrivateDtor final : public Base {
 | |
| public:
 | |
|   void destroy() { delete this; }
 | |
| private:
 | |
|   ~PrivateDtor() {}
 | |
| };
 | |
| 
 | |
| struct ImplicitNV {
 | |
|   virtual void f();
 | |
| };
 | |
| 
 | |
| struct ImplicitNVDerived : public ImplicitNV {};
 | |
| 
 | |
| NVDerived *get();
 | |
| 
 | |
| NonVirtual *create() {
 | |
|   NonVirtual *x = new NVDerived(); // expected-note{{Conversion from derived to base happened here}}
 | |
|   return x;
 | |
| }
 | |
| 
 | |
| void sink(NonVirtual *x) {
 | |
|   delete x; // expected-warning{{Destruction of a polymorphic object with no virtual destructor}}
 | |
|   // expected-note@-1{{Destruction of a polymorphic object with no virtual destructor}}
 | |
| }
 | |
| 
 | |
| void sinkCast(NonVirtual *y) {
 | |
|   delete reinterpret_cast<NVDerived*>(y);
 | |
| }
 | |
| 
 | |
| void sinkParamCast(NVDerived *z) {
 | |
|   delete z;
 | |
| }
 | |
| 
 | |
| void singleDerived() {
 | |
|   NonVirtual *sd;
 | |
|   sd = new NVDerived(); // expected-note{{Conversion from derived to base happened here}}
 | |
|   delete sd; // expected-warning{{Destruction of a polymorphic object with no virtual destructor}}
 | |
|   // expected-note@-1{{Destruction of a polymorphic object with no virtual destructor}}
 | |
| }
 | |
| 
 | |
| void singleDerivedArr() {
 | |
|   NonVirtual *sda = new NVDerived[5]; // expected-note{{Conversion from derived to base happened here}}
 | |
|   delete[] sda; // expected-warning{{Destruction of a polymorphic object with no virtual destructor}}
 | |
|   // expected-note@-1{{Destruction of a polymorphic object with no virtual destructor}}
 | |
| }
 | |
| 
 | |
| void doubleDerived() {
 | |
|   NonVirtual *dd = new NVDoubleDerived(); // expected-note{{Conversion from derived to base happened here}}
 | |
|   delete (dd); // expected-warning{{Destruction of a polymorphic object with no virtual destructor}}
 | |
|   // expected-note@-1{{Destruction of a polymorphic object with no virtual destructor}}
 | |
| }
 | |
| 
 | |
| void assignThroughFunction() {
 | |
|   NonVirtual *atf = get(); // expected-note{{Conversion from derived to base happened here}}
 | |
|   delete atf; // expected-warning{{Destruction of a polymorphic object with no virtual destructor}}
 | |
|   // expected-note@-1{{Destruction of a polymorphic object with no virtual destructor}}
 | |
| }
 | |
| 
 | |
| void assignThroughFunction2() {
 | |
|   NonVirtual *atf2;
 | |
|   atf2 = get(); // expected-note{{Conversion from derived to base happened here}}
 | |
|   delete atf2; // expected-warning{{Destruction of a polymorphic object with no virtual destructor}}
 | |
|   // expected-note@-1{{Destruction of a polymorphic object with no virtual destructor}}
 | |
| }
 | |
| 
 | |
| void createThroughFunction() {
 | |
|   NonVirtual *ctf = create(); // expected-note{{Calling 'create'}}
 | |
|   // expected-note@-1{{Returning from 'create'}}
 | |
|   delete ctf; // expected-warning {{Destruction of a polymorphic object with no virtual destructor}}
 | |
|   // expected-note@-1{{Destruction of a polymorphic object with no virtual destructor}}
 | |
| }
 | |
| 
 | |
| void deleteThroughFunction() {
 | |
|   NonVirtual *dtf = new NVDerived(); // expected-note{{Conversion from derived to base happened here}}
 | |
|   sink(dtf); // expected-note{{Calling 'sink'}}
 | |
| }
 | |
| 
 | |
| void singleCastCStyle() {
 | |
|   NVDerived *sccs = new NVDerived();
 | |
|   NonVirtual *sccs2 = (NonVirtual*)sccs; // expected-note{{Conversion from derived to base happened here}}
 | |
|   delete sccs2; // expected-warning{{Destruction of a polymorphic object with no virtual destructor}}
 | |
|   // expected-note@-1{{Destruction of a polymorphic object with no virtual destructor}}
 | |
| }
 | |
| 
 | |
| void doubleCastCStyle() {
 | |
|   NonVirtual *dccs = new NVDerived();
 | |
|   NVDerived *dccs2 = (NVDerived*)dccs;
 | |
|   dccs = (NonVirtual*)dccs2; // expected-note{{Conversion from derived to base happened here}}
 | |
|   delete dccs; // expected-warning{{Destruction of a polymorphic object with no virtual destructor}}
 | |
|   // expected-note@-1{{Destruction of a polymorphic object with no virtual destructor}}
 | |
| }
 | |
| 
 | |
| void singleCast() {
 | |
|   NVDerived *sc = new NVDerived();
 | |
|   NonVirtual *sc2 = reinterpret_cast<NonVirtual*>(sc); // expected-note{{Conversion from derived to base happened here}}
 | |
|   delete sc2; // expected-warning{{Destruction of a polymorphic object with no virtual destructor}}
 | |
|   // expected-note@-1{{Destruction of a polymorphic object with no virtual destructor}}
 | |
| }
 | |
| 
 | |
| void doubleCast() {
 | |
|   NonVirtual *dd = new NVDerived();
 | |
|   NVDerived *dd2 = reinterpret_cast<NVDerived*>(dd);
 | |
|   dd = reinterpret_cast<NonVirtual*>(dd2); // expected-note {{Conversion from derived to base happened here}}
 | |
|   delete dd; // expected-warning {{Destruction of a polymorphic object with no virtual destructor}}
 | |
|   // expected-note@-1{{Destruction of a polymorphic object with no virtual destructor}}
 | |
| }
 | |
| 
 | |
| void implicitNV() {
 | |
|   ImplicitNV *invd = new ImplicitNVDerived(); // expected-note{{Conversion from derived to base happened here}}
 | |
|   delete invd; // expected-warning{{Destruction of a polymorphic object with no virtual destructor}}
 | |
|   // expected-note@-1{{Destruction of a polymorphic object with no virtual destructor}}
 | |
| }
 | |
| 
 | |
| void doubleDecl() {
 | |
|   ImplicitNV *dd1, *dd2;
 | |
|   dd1 = new ImplicitNVDerived(); // expected-note{{Conversion from derived to base happened here}}
 | |
|   delete dd1; // expected-warning{{Destruction of a polymorphic object with no virtual destructor}}
 | |
|   // expected-note@-1{{Destruction of a polymorphic object with no virtual destructor}}
 | |
| }
 | |
| 
 | |
| void virtualBase() {
 | |
|   Virtual *vb = new VDerived();
 | |
|   delete vb; // no-warning
 | |
| }
 | |
| 
 | |
| void notDerived() {
 | |
|   NonVirtual *nd = new NonVirtual();
 | |
|   delete nd; // no-warning
 | |
| }
 | |
| 
 | |
| void notDerivedArr() {
 | |
|   NonVirtual *nda = new NonVirtual[3];
 | |
|   delete[] nda; // no-warning
 | |
| }
 | |
| 
 | |
| void cast() {
 | |
|   NonVirtual *c = new NVDerived();
 | |
|   delete reinterpret_cast<NVDerived*>(c); // no-warning
 | |
| }
 | |
| 
 | |
| void deleteThroughFunction2() {
 | |
|   NonVirtual *dtf2 = new NVDerived();
 | |
|   sinkCast(dtf2); // no-warning
 | |
| }
 | |
| 
 | |
| void deleteThroughFunction3() {
 | |
|   NVDerived *dtf3;
 | |
|   dtf3 = new NVDerived();
 | |
|   sinkParamCast(dtf3); // no-warning
 | |
| }
 | |
| 
 | |
| void stackVar() {
 | |
|   NonVirtual sv2;
 | |
|   delete &sv2; // no-warning
 | |
| }
 | |
| 
 | |
| // Deleting a polymorphic object with a non-virtual dtor
 | |
| // is not a problem if it is referenced by its precise type.
 | |
| 
 | |
| void preciseType() {
 | |
|   NVDerived *pt = new NVDerived();
 | |
|   delete pt; // no-warning
 | |
| }
 | |
| 
 | |
| void privateDtor() {
 | |
|   Base *pd = new PrivateDtor();
 | |
|   pd->destroy(); // no-warning
 | |
| }
 |