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