Run DCE after a LoopFlatten test to reduce spurious output [nfc]
[llvm-project.git] / clang / test / Analysis / DeleteWithNonVirtualDtor.cpp
blob2809f3758eee1be18cfa8c3adfd05085585873c4
1 // RUN: %clang_cc1 -analyze -analyzer-checker=alpha.cplusplus.DeleteWithNonVirtualDtor -std=c++11 -verify -analyzer-output=text %s
3 struct Virtual {
4 virtual ~Virtual() {}
5 };
7 struct VDerived : public Virtual {};
9 struct NonVirtual {
10 ~NonVirtual() {}
13 struct NVDerived : public NonVirtual {};
14 struct NVDoubleDerived : public NVDerived {};
16 struct Base {
17 virtual void destroy() = 0;
20 class PrivateDtor final : public Base {
21 public:
22 void destroy() { delete this; }
23 private:
24 ~PrivateDtor() {}
27 struct ImplicitNV {
28 virtual void f();
31 struct ImplicitNVDerived : public ImplicitNV {};
33 NVDerived *get();
35 NonVirtual *create() {
36 NonVirtual *x = new NVDerived(); // expected-note{{Casting from 'NVDerived' to 'NonVirtual' here}}
37 return x;
40 void sink(NonVirtual *x) {
41 delete x; // expected-warning{{Destruction of a polymorphic object with no virtual destructor}}
42 // expected-note@-1{{Destruction of a polymorphic object with no virtual destructor}}
45 void sinkCast(NonVirtual *y) {
46 delete reinterpret_cast<NVDerived*>(y);
49 void sinkParamCast(NVDerived *z) {
50 delete z;
53 void singleDerived() {
54 NonVirtual *sd;
55 sd = new NVDerived(); // expected-note{{Casting from 'NVDerived' to 'NonVirtual' here}}
56 delete sd; // expected-warning{{Destruction of a polymorphic object with no virtual destructor}}
57 // expected-note@-1{{Destruction of a polymorphic object with no virtual destructor}}
60 void singleDerivedArr() {
61 NonVirtual *sda = new NVDerived[5]; // expected-note{{Casting from 'NVDerived' to 'NonVirtual' here}}
62 delete[] sda; // expected-warning{{Destruction of a polymorphic object with no virtual destructor}}
63 // expected-note@-1{{Destruction of a polymorphic object with no virtual destructor}}
66 void doubleDerived() {
67 NonVirtual *dd = new NVDoubleDerived(); // expected-note{{Casting from 'NVDoubleDerived' to 'NonVirtual' here}}
68 delete (dd); // expected-warning{{Destruction of a polymorphic object with no virtual destructor}}
69 // expected-note@-1{{Destruction of a polymorphic object with no virtual destructor}}
72 void assignThroughFunction() {
73 NonVirtual *atf = get(); // expected-note{{Casting from 'NVDerived' to 'NonVirtual' here}}
74 delete atf; // expected-warning{{Destruction of a polymorphic object with no virtual destructor}}
75 // expected-note@-1{{Destruction of a polymorphic object with no virtual destructor}}
78 void assignThroughFunction2() {
79 NonVirtual *atf2;
80 atf2 = get(); // expected-note{{Casting from 'NVDerived' to 'NonVirtual' here}}
81 delete atf2; // expected-warning{{Destruction of a polymorphic object with no virtual destructor}}
82 // expected-note@-1{{Destruction of a polymorphic object with no virtual destructor}}
85 void createThroughFunction() {
86 NonVirtual *ctf = create(); // expected-note{{Calling 'create'}}
87 // expected-note@-1{{Returning from 'create'}}
88 delete ctf; // expected-warning {{Destruction of a polymorphic object with no virtual destructor}}
89 // expected-note@-1{{Destruction of a polymorphic object with no virtual destructor}}
92 void deleteThroughFunction() {
93 NonVirtual *dtf = new NVDerived(); // expected-note{{Casting from 'NVDerived' to 'NonVirtual' here}}
94 sink(dtf); // expected-note{{Calling 'sink'}}
97 void singleCastCStyle() {
98 NVDerived *sccs = new NVDerived();
99 NonVirtual *sccs2 = (NonVirtual*)sccs; // expected-note{{Casting from 'NVDerived' to 'NonVirtual' here}}
100 delete sccs2; // expected-warning{{Destruction of a polymorphic object with no virtual destructor}}
101 // expected-note@-1{{Destruction of a polymorphic object with no virtual destructor}}
104 void doubleCastCStyle() {
105 NonVirtual *dccs = new NVDerived(); // expected-note{{Casting from 'NVDerived' to 'NonVirtual' here}}
106 NVDerived *dccs2 = (NVDerived*)dccs; // expected-note{{Casting from 'NonVirtual' to 'NVDerived' here}}
107 dccs = (NonVirtual*)dccs2; // expected-note{{Casting from 'NVDerived' to 'NonVirtual' here}}
108 delete dccs; // expected-warning{{Destruction of a polymorphic object with no virtual destructor}}
109 // expected-note@-1{{Destruction of a polymorphic object with no virtual destructor}}
112 void singleCast() {
113 NVDerived *sc = new NVDerived();
114 NonVirtual *sc2 = reinterpret_cast<NonVirtual*>(sc); // expected-note{{Casting from 'NVDerived' to 'NonVirtual' here}}
115 delete sc2; // expected-warning{{Destruction of a polymorphic object with no virtual destructor}}
116 // expected-note@-1{{Destruction of a polymorphic object with no virtual destructor}}
119 void doubleCast() {
120 NonVirtual *dd = new NVDerived(); // expected-note {{Casting from 'NVDerived' to 'NonVirtual' here}}
121 NVDerived *dd2 = reinterpret_cast<NVDerived*>(dd); // expected-note {{Casting from 'NonVirtual' to 'NVDerived' here}}
122 dd = reinterpret_cast<NonVirtual*>(dd2); // expected-note {{Casting from 'NVDerived' to 'NonVirtual' here}}
123 delete dd; // expected-warning {{Destruction of a polymorphic object with no virtual destructor}}
124 // expected-note@-1{{Destruction of a polymorphic object with no virtual destructor}}
127 void implicitNV() {
128 ImplicitNV *invd = new ImplicitNVDerived(); // expected-note{{Casting from 'ImplicitNVDerived' to 'ImplicitNV' here}}
129 delete invd; // expected-warning{{Destruction of a polymorphic object with no virtual destructor}}
130 // expected-note@-1{{Destruction of a polymorphic object with no virtual destructor}}
133 void doubleDecl() {
134 ImplicitNV *dd1, *dd2;
135 dd1 = new ImplicitNVDerived(); // expected-note{{Casting from 'ImplicitNVDerived' to 'ImplicitNV' here}}
136 delete dd1; // expected-warning{{Destruction of a polymorphic object with no virtual destructor}}
137 // expected-note@-1{{Destruction of a polymorphic object with no virtual destructor}}
140 void virtualBase() {
141 Virtual *vb = new VDerived();
142 delete vb; // no-warning
145 void notDerived() {
146 NonVirtual *nd = new NonVirtual();
147 delete nd; // no-warning
150 void notDerivedArr() {
151 NonVirtual *nda = new NonVirtual[3];
152 delete[] nda; // no-warning
155 void cast() {
156 NonVirtual *c = new NVDerived();
157 delete reinterpret_cast<NVDerived*>(c); // no-warning
160 void deleteThroughFunction2() {
161 NonVirtual *dtf2 = new NVDerived();
162 sinkCast(dtf2); // no-warning
165 void deleteThroughFunction3() {
166 NVDerived *dtf3;
167 dtf3 = new NVDerived();
168 sinkParamCast(dtf3); // no-warning
171 void stackVar() {
172 NonVirtual sv2;
173 delete &sv2; // no-warning
176 // Deleting a polymorphic object with a non-virtual dtor
177 // is not a problem if it is referenced by its precise type.
179 void preciseType() {
180 NVDerived *pt = new NVDerived();
181 delete pt; // no-warning
184 void privateDtor() {
185 Base *pd = new PrivateDtor();
186 pd->destroy(); // no-warning