// 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 }