blob: 9f066f44692fb010e3844b254484009a8c8c9fe5 [file] [log] [blame]
Ted Kremenek9e240492008-10-04 05:50:14 +00001//== MemRegion.cpp - Abstract memory regions for static analysis --*- C++ -*--//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10// This file defines MemRegion and its subclasses. MemRegion defines a
11// partially-typed abstraction of memory useful for path-sensitive dataflow
12// analyses.
13//
14//===----------------------------------------------------------------------===//
15
16#include "llvm/Support/raw_ostream.h"
17#include "clang/Analysis/PathSensitive/MemRegion.h"
18
19using namespace clang;
20
21
22MemRegion::~MemRegion() {}
23
Zhongxing Xu7e5d6ed2009-01-08 13:17:14 +000024bool SubRegion::isSubRegionOf(const MemRegion* R) const {
25 const MemRegion* r = getSuperRegion();
26 while (r != 0) {
27 if (r == R)
28 return true;
29 if (const SubRegion* sr = dyn_cast<SubRegion>(r))
30 r = sr->getSuperRegion();
31 else
32 break;
33 }
34 return false;
35}
36
Ted Kremenek9e240492008-10-04 05:50:14 +000037void MemSpaceRegion::Profile(llvm::FoldingSetNodeID& ID) const {
38 ID.AddInteger((unsigned)getKind());
39}
40
Zhongxing Xue9f4e542008-10-25 14:13:41 +000041void StringRegion::ProfileRegion(llvm::FoldingSetNodeID& ID,
42 const StringLiteral* Str,
43 const MemRegion* superRegion) {
44 ID.AddInteger((unsigned) StringRegionKind);
45 ID.AddPointer(Str);
46 ID.AddPointer(superRegion);
47}
48
Ted Kremenek7090ae12008-11-02 00:34:33 +000049void AllocaRegion::ProfileRegion(llvm::FoldingSetNodeID& ID,
50 const Expr* Ex, unsigned cnt) {
51 ID.AddInteger((unsigned) AllocaRegionKind);
52 ID.AddPointer(Ex);
53 ID.AddInteger(cnt);
54}
55
56void AllocaRegion::Profile(llvm::FoldingSetNodeID& ID) const {
57 ProfileRegion(ID, Ex, Cnt);
58}
59
Ted Kremenek0312c0e2009-03-01 05:44:08 +000060void TypedViewRegion::ProfileRegion(llvm::FoldingSetNodeID& ID, QualType T,
Zhongxing Xudc0a25d2008-11-16 04:07:26 +000061 const MemRegion* superRegion) {
Ted Kremenek0312c0e2009-03-01 05:44:08 +000062 ID.AddInteger((unsigned) TypedViewRegionKind);
Zhongxing Xudc0a25d2008-11-16 04:07:26 +000063 ID.Add(T);
64 ID.AddPointer(superRegion);
65}
66
Ted Kremenek329d6fd2008-10-27 20:57:58 +000067void CompoundLiteralRegion::Profile(llvm::FoldingSetNodeID& ID) const {
68 CompoundLiteralRegion::ProfileRegion(ID, CL, superRegion);
69}
70
71void CompoundLiteralRegion::ProfileRegion(llvm::FoldingSetNodeID& ID,
72 const CompoundLiteralExpr* CL,
73 const MemRegion* superRegion) {
74 ID.AddInteger((unsigned) CompoundLiteralRegionKind);
75 ID.AddPointer(CL);
76 ID.AddPointer(superRegion);
77}
78
Ted Kremenek9e240492008-10-04 05:50:14 +000079void DeclRegion::ProfileRegion(llvm::FoldingSetNodeID& ID, const Decl* D,
80 const MemRegion* superRegion, Kind k) {
81 ID.AddInteger((unsigned) k);
82 ID.AddPointer(D);
83 ID.AddPointer(superRegion);
84}
85
86void DeclRegion::Profile(llvm::FoldingSetNodeID& ID) const {
87 DeclRegion::ProfileRegion(ID, D, superRegion, getKind());
88}
89
Ted Kremenek2dabd432008-12-05 02:27:51 +000090void SymbolicRegion::ProfileRegion(llvm::FoldingSetNodeID& ID, SymbolRef sym) {
Ted Kremenek993f1c72008-10-17 20:28:54 +000091 ID.AddInteger((unsigned) MemRegion::SymbolicRegionKind);
Ted Kremenek6d0e2d22008-12-05 02:39:38 +000092 ID.Add(sym);
Ted Kremenek993f1c72008-10-17 20:28:54 +000093}
94
95void SymbolicRegion::Profile(llvm::FoldingSetNodeID& ID) const {
96 SymbolicRegion::ProfileRegion(ID, sym);
97}
98
Ted Kremenekf936f452009-05-04 06:18:28 +000099void ElementRegion::ProfileRegion(llvm::FoldingSetNodeID& ID,
100 QualType ElementType, SVal Idx,
Zhongxing Xu511191c2008-10-21 05:27:10 +0000101 const MemRegion* superRegion) {
102 ID.AddInteger(MemRegion::ElementRegionKind);
Ted Kremenekf936f452009-05-04 06:18:28 +0000103 ID.Add(ElementType);
Zhongxing Xu511191c2008-10-21 05:27:10 +0000104 ID.AddPointer(superRegion);
105 Idx.Profile(ID);
106}
107
108void ElementRegion::Profile(llvm::FoldingSetNodeID& ID) const {
Ted Kremenekf936f452009-05-04 06:18:28 +0000109 ElementRegion::ProfileRegion(ID, ElementType, Index, superRegion);
Zhongxing Xu511191c2008-10-21 05:27:10 +0000110}
Zhongxing Xu27b57062008-10-27 13:17:02 +0000111
Zhongxing Xuec13d922009-04-10 08:45:10 +0000112void CodeTextRegion::ProfileRegion(llvm::FoldingSetNodeID& ID, const void* data,
113 QualType t) {
114 ID.AddInteger(MemRegion::CodeTextRegionKind);
115 ID.AddPointer(data);
116 ID.Add(t);
117}
118
119void CodeTextRegion::Profile(llvm::FoldingSetNodeID& ID) const {
120 CodeTextRegion::ProfileRegion(ID, Data, LocationType);
121}
122
Zhongxing Xu026c6632009-02-05 06:57:29 +0000123//===----------------------------------------------------------------------===//
Ted Kremenek9e240492008-10-04 05:50:14 +0000124// Region pretty-printing.
125//===----------------------------------------------------------------------===//
126
127std::string MemRegion::getString() const {
128 std::string s;
129 llvm::raw_string_ostream os(s);
130 print(os);
131 return os.str();
132}
133
134void MemRegion::print(llvm::raw_ostream& os) const {
135 os << "<Unknown Region>";
136}
137
Ted Kremenek7090ae12008-11-02 00:34:33 +0000138void AllocaRegion::print(llvm::raw_ostream& os) const {
139 os << "alloca{" << (void*) Ex << ',' << Cnt << '}';
140}
141
Ted Kremenek72e03202009-04-21 19:56:58 +0000142void CodeTextRegion::print(llvm::raw_ostream& os) const {
143 os << "code{";
144 if (isDeclared())
Ted Kremenek82539b02009-04-29 15:37:24 +0000145 os << getDecl()->getDeclName().getAsString();
Ted Kremenek72e03202009-04-21 19:56:58 +0000146 else
147 os << '$' << getSymbol();
148
149 os << '}';
150}
151
Ted Kremenek5639a3e2009-04-21 18:09:22 +0000152void CompoundLiteralRegion::print(llvm::raw_ostream& os) const {
153 // FIXME: More elaborate pretty-printing.
154 os << "{ " << (void*) CL << " }";
155}
156
157void ElementRegion::print(llvm::raw_ostream& os) const {
158 superRegion->print(os);
159 os << '['; Index.print(os); os << ']';
160}
161
162void FieldRegion::print(llvm::raw_ostream& os) const {
163 superRegion->print(os);
164 os << "->" << getDecl()->getNameAsString();
165}
166
167void StringRegion::print(llvm::raw_ostream& os) const {
168 Str->printPretty(os);
169}
170
171void SymbolicRegion::print(llvm::raw_ostream& os) const {
172 os << "SymRegion-" << sym;
173}
174
Ted Kremenek0312c0e2009-03-01 05:44:08 +0000175void TypedViewRegion::print(llvm::raw_ostream& os) const {
Ted Kremenekb2dea732009-03-11 21:57:34 +0000176 os << "typed_view{" << LValueType.getAsString() << ',';
Ted Kremenek500d2ee2008-12-17 19:25:50 +0000177 getSuperRegion()->print(os);
178 os << '}';
179}
180
Ted Kremenek9e240492008-10-04 05:50:14 +0000181void VarRegion::print(llvm::raw_ostream& os) const {
Chris Lattnerd9d22dd2008-11-24 05:29:24 +0000182 os << cast<VarDecl>(D)->getNameAsString();
Ted Kremenek9e240492008-10-04 05:50:14 +0000183}
184
185//===----------------------------------------------------------------------===//
186// MemRegionManager methods.
187//===----------------------------------------------------------------------===//
188
189MemSpaceRegion* MemRegionManager::LazyAllocate(MemSpaceRegion*& region) {
190
191 if (!region) {
192 region = (MemSpaceRegion*) A.Allocate<MemSpaceRegion>();
193 new (region) MemSpaceRegion();
194 }
195
196 return region;
197}
198
199MemSpaceRegion* MemRegionManager::getStackRegion() {
200 return LazyAllocate(stack);
201}
202
203MemSpaceRegion* MemRegionManager::getGlobalsRegion() {
204 return LazyAllocate(globals);
205}
206
207MemSpaceRegion* MemRegionManager::getHeapRegion() {
208 return LazyAllocate(heap);
209}
210
Zhongxing Xu17892752008-10-08 02:50:44 +0000211MemSpaceRegion* MemRegionManager::getUnknownRegion() {
212 return LazyAllocate(unknown);
213}
214
Zhongxing Xuec13d922009-04-10 08:45:10 +0000215MemSpaceRegion* MemRegionManager::getCodeRegion() {
216 return LazyAllocate(code);
217}
218
Zhongxing Xu4193eca2008-12-20 06:32:12 +0000219bool MemRegionManager::onStack(const MemRegion* R) {
220 while (const SubRegion* SR = dyn_cast<SubRegion>(R))
221 R = SR->getSuperRegion();
222
223 return (R != 0) && (R == stack);
224}
225
226bool MemRegionManager::onHeap(const MemRegion* R) {
227 while (const SubRegion* SR = dyn_cast<SubRegion>(R))
228 R = SR->getSuperRegion();
229
230 return (R != 0) && (R == heap);
231}
232
Zhongxing Xue9f4e542008-10-25 14:13:41 +0000233StringRegion* MemRegionManager::getStringRegion(const StringLiteral* Str) {
234 llvm::FoldingSetNodeID ID;
235 MemSpaceRegion* GlobalsR = getGlobalsRegion();
236
237 StringRegion::ProfileRegion(ID, Str, GlobalsR);
238
239 void* InsertPos;
240 MemRegion* data = Regions.FindNodeOrInsertPos(ID, InsertPos);
241 StringRegion* R = cast_or_null<StringRegion>(data);
242
243 if (!R) {
244 R = (StringRegion*) A.Allocate<StringRegion>();
245 new (R) StringRegion(Str, GlobalsR);
246 Regions.InsertNode(R, InsertPos);
247 }
248
249 return R;
250}
251
Ted Kremenek9a1f03a2008-10-27 21:01:26 +0000252VarRegion* MemRegionManager::getVarRegion(const VarDecl* d) {
253
254 const MemRegion* superRegion = d->hasLocalStorage() ? getStackRegion()
255 : getGlobalsRegion();
256
Ted Kremenek9e240492008-10-04 05:50:14 +0000257 llvm::FoldingSetNodeID ID;
258 DeclRegion::ProfileRegion(ID, d, superRegion, MemRegion::VarRegionKind);
259
260 void* InsertPos;
261 MemRegion* data = Regions.FindNodeOrInsertPos(ID, InsertPos);
262 VarRegion* R = cast_or_null<VarRegion>(data);
263
264 if (!R) {
265 R = (VarRegion*) A.Allocate<VarRegion>();
266 new (R) VarRegion(d, superRegion);
267 Regions.InsertNode(R, InsertPos);
268 }
269
270 return R;
271}
272
Ted Kremenek329d6fd2008-10-27 20:57:58 +0000273CompoundLiteralRegion*
274MemRegionManager::getCompoundLiteralRegion(const CompoundLiteralExpr* CL) {
275 // Is this compound literal allocated on the stack or is part of the
276 // global constant pool?
277 const MemRegion* superRegion = CL->isFileScope() ?
278 getGlobalsRegion() : getStackRegion();
279
280 // Profile the compound literal.
281 llvm::FoldingSetNodeID ID;
282 CompoundLiteralRegion::ProfileRegion(ID, CL, superRegion);
283
284 void* InsertPos;
285 MemRegion* data = Regions.FindNodeOrInsertPos(ID, InsertPos);
286 CompoundLiteralRegion* R = cast_or_null<CompoundLiteralRegion>(data);
287
288 if (!R) {
289 R = (CompoundLiteralRegion*) A.Allocate<CompoundLiteralRegion>();
290 new (R) CompoundLiteralRegion(CL, superRegion);
291 Regions.InsertNode(R, InsertPos);
292 }
293
294 return R;
295}
296
Ted Kremenekabb042f2008-12-13 19:24:37 +0000297ElementRegion*
Ted Kremenekf936f452009-05-04 06:18:28 +0000298MemRegionManager::getElementRegion(QualType elementType, SVal Idx,
Zhongxing Xu4c44e242009-05-06 08:15:46 +0000299 const MemRegion* superRegion){
Ted Kremenekabb042f2008-12-13 19:24:37 +0000300
Zhongxing Xu511191c2008-10-21 05:27:10 +0000301 llvm::FoldingSetNodeID ID;
Ted Kremenekf936f452009-05-04 06:18:28 +0000302 ElementRegion::ProfileRegion(ID, elementType, Idx, superRegion);
Zhongxing Xu511191c2008-10-21 05:27:10 +0000303
304 void* InsertPos;
305 MemRegion* data = Regions.FindNodeOrInsertPos(ID, InsertPos);
306 ElementRegion* R = cast_or_null<ElementRegion>(data);
307
308 if (!R) {
309 R = (ElementRegion*) A.Allocate<ElementRegion>();
Ted Kremenekf936f452009-05-04 06:18:28 +0000310 new (R) ElementRegion(elementType, Idx, superRegion);
Zhongxing Xu511191c2008-10-21 05:27:10 +0000311 Regions.InsertNode(R, InsertPos);
312 }
313
314 return R;
315}
316
Zhongxing Xuec13d922009-04-10 08:45:10 +0000317CodeTextRegion* MemRegionManager::getCodeTextRegion(const FunctionDecl* fd,
318 QualType t) {
319 llvm::FoldingSetNodeID ID;
320 CodeTextRegion::ProfileRegion(ID, fd, t);
321 void* InsertPos;
322 MemRegion* data = Regions.FindNodeOrInsertPos(ID, InsertPos);
323 CodeTextRegion* R = cast_or_null<CodeTextRegion>(data);
324
325 if (!R) {
326 R = (CodeTextRegion*) A.Allocate<CodeTextRegion>();
327 new (R) CodeTextRegion(fd, t, getCodeRegion());
328 Regions.InsertNode(R, InsertPos);
329 }
330
331 return R;
332}
333
334CodeTextRegion* MemRegionManager::getCodeTextRegion(SymbolRef sym, QualType t) {
335 llvm::FoldingSetNodeID ID;
336 CodeTextRegion::ProfileRegion(ID, sym, t);
337 void* InsertPos;
338 MemRegion* data = Regions.FindNodeOrInsertPos(ID, InsertPos);
339 CodeTextRegion* R = cast_or_null<CodeTextRegion>(data);
340
341 if (!R) {
342 R = (CodeTextRegion*) A.Allocate<CodeTextRegion>();
343 new (R) CodeTextRegion(sym, t, getCodeRegion());
344 Regions.InsertNode(R, InsertPos);
345 }
346
347 return R;
348}
349
Ted Kremenek993f1c72008-10-17 20:28:54 +0000350/// getSymbolicRegion - Retrieve or create a "symbolic" memory region.
Ted Kremeneke0e4ebf2009-03-26 03:35:11 +0000351SymbolicRegion* MemRegionManager::getSymbolicRegion(SymbolRef sym) {
Ted Kremenek993f1c72008-10-17 20:28:54 +0000352 llvm::FoldingSetNodeID ID;
353 SymbolicRegion::ProfileRegion(ID, sym);
Ted Kremenek993f1c72008-10-17 20:28:54 +0000354 void* InsertPos;
355 MemRegion* data = Regions.FindNodeOrInsertPos(ID, InsertPos);
356 SymbolicRegion* R = cast_or_null<SymbolicRegion>(data);
357
358 if (!R) {
359 R = (SymbolicRegion*) A.Allocate<SymbolicRegion>();
Zhongxing Xu026c6632009-02-05 06:57:29 +0000360 // SymbolicRegion's storage class is usually unknown.
Ted Kremeneke0e4ebf2009-03-26 03:35:11 +0000361 new (R) SymbolicRegion(sym, getUnknownRegion());
Ted Kremenek993f1c72008-10-17 20:28:54 +0000362 Regions.InsertNode(R, InsertPos);
363 }
364
365 return R;
366}
367
Ted Kremenek9e240492008-10-04 05:50:14 +0000368FieldRegion* MemRegionManager::getFieldRegion(const FieldDecl* d,
Ted Kremenek993f1c72008-10-17 20:28:54 +0000369 const MemRegion* superRegion) {
Ted Kremenek9e240492008-10-04 05:50:14 +0000370 llvm::FoldingSetNodeID ID;
371 DeclRegion::ProfileRegion(ID, d, superRegion, MemRegion::FieldRegionKind);
372
373 void* InsertPos;
374 MemRegion* data = Regions.FindNodeOrInsertPos(ID, InsertPos);
375 FieldRegion* R = cast_or_null<FieldRegion>(data);
376
377 if (!R) {
378 R = (FieldRegion*) A.Allocate<FieldRegion>();
379 new (R) FieldRegion(d, superRegion);
380 Regions.InsertNode(R, InsertPos);
381 }
382
383 return R;
384}
385
Ted Kremenek993f1c72008-10-17 20:28:54 +0000386ObjCIvarRegion*
387MemRegionManager::getObjCIvarRegion(const ObjCIvarDecl* d,
388 const MemRegion* superRegion) {
Ted Kremenek9e240492008-10-04 05:50:14 +0000389 llvm::FoldingSetNodeID ID;
390 DeclRegion::ProfileRegion(ID, d, superRegion, MemRegion::ObjCIvarRegionKind);
391
392 void* InsertPos;
393 MemRegion* data = Regions.FindNodeOrInsertPos(ID, InsertPos);
394 ObjCIvarRegion* R = cast_or_null<ObjCIvarRegion>(data);
395
396 if (!R) {
Zhongxing Xu722c2882008-10-06 03:03:33 +0000397 R = (ObjCIvarRegion*) A.Allocate<ObjCIvarRegion>();
398 new (R) ObjCIvarRegion(d, superRegion);
Ted Kremenek9e240492008-10-04 05:50:14 +0000399 Regions.InsertNode(R, InsertPos);
400 }
401
402 return R;
403}
404
Ted Kremeneka7f1b9e2008-10-24 20:30:08 +0000405ObjCObjectRegion*
406MemRegionManager::getObjCObjectRegion(const ObjCInterfaceDecl* d,
407 const MemRegion* superRegion) {
408 llvm::FoldingSetNodeID ID;
409 DeclRegion::ProfileRegion(ID, d, superRegion,
410 MemRegion::ObjCObjectRegionKind);
411
412 void* InsertPos;
413 MemRegion* data = Regions.FindNodeOrInsertPos(ID, InsertPos);
414 ObjCObjectRegion* R = cast_or_null<ObjCObjectRegion>(data);
415
416 if (!R) {
417 R = (ObjCObjectRegion*) A.Allocate<ObjCObjectRegion>();
418 new (R) ObjCObjectRegion(d, superRegion);
419 Regions.InsertNode(R, InsertPos);
420 }
421
422 return R;
423}
424
Ted Kremenek0312c0e2009-03-01 05:44:08 +0000425TypedViewRegion*
426MemRegionManager::getTypedViewRegion(QualType t, const MemRegion* superRegion) {
Zhongxing Xudc0a25d2008-11-16 04:07:26 +0000427 llvm::FoldingSetNodeID ID;
Ted Kremenek0312c0e2009-03-01 05:44:08 +0000428 TypedViewRegion::ProfileRegion(ID, t, superRegion);
Zhongxing Xudc0a25d2008-11-16 04:07:26 +0000429
430 void* InsertPos;
431 MemRegion* data = Regions.FindNodeOrInsertPos(ID, InsertPos);
Ted Kremenek0312c0e2009-03-01 05:44:08 +0000432 TypedViewRegion* R = cast_or_null<TypedViewRegion>(data);
Zhongxing Xudc0a25d2008-11-16 04:07:26 +0000433
434 if (!R) {
Ted Kremenek0312c0e2009-03-01 05:44:08 +0000435 R = (TypedViewRegion*) A.Allocate<TypedViewRegion>();
436 new (R) TypedViewRegion(t, superRegion);
Zhongxing Xudc0a25d2008-11-16 04:07:26 +0000437 Regions.InsertNode(R, InsertPos);
438 }
439
440 return R;
441}
Ted Kremeneka7f1b9e2008-10-24 20:30:08 +0000442
Ted Kremenek7090ae12008-11-02 00:34:33 +0000443AllocaRegion* MemRegionManager::getAllocaRegion(const Expr* E, unsigned cnt) {
444 llvm::FoldingSetNodeID ID;
445 AllocaRegion::ProfileRegion(ID, E, cnt);
446
447 void* InsertPos;
448 MemRegion* data = Regions.FindNodeOrInsertPos(ID, InsertPos);
449 AllocaRegion* R = cast_or_null<AllocaRegion>(data);
450
451 if (!R) {
452 R = (AllocaRegion*) A.Allocate<AllocaRegion>();
453 new (R) AllocaRegion(E, cnt, getStackRegion());
454 Regions.InsertNode(R, InsertPos);
455 }
456
457 return R;
458}
459
Ted Kremenek9e240492008-10-04 05:50:14 +0000460bool MemRegionManager::hasStackStorage(const MemRegion* R) {
Ted Kremenek993f1c72008-10-17 20:28:54 +0000461
462 // Only subregions can have stack storage.
Ted Kremenek7090ae12008-11-02 00:34:33 +0000463 const SubRegion* SR = dyn_cast<SubRegion>(R);
464
Ted Kremenek993f1c72008-10-17 20:28:54 +0000465 if (!SR)
466 return false;
Ted Kremenek7090ae12008-11-02 00:34:33 +0000467
Ted Kremenek9e240492008-10-04 05:50:14 +0000468 MemSpaceRegion* S = getStackRegion();
469
Ted Kremenek993f1c72008-10-17 20:28:54 +0000470 while (SR) {
471 R = SR->getSuperRegion();
472 if (R == S)
473 return true;
474
475 SR = dyn_cast<SubRegion>(R);
Ted Kremenek9e240492008-10-04 05:50:14 +0000476 }
Ted Kremenek1670e402009-04-11 00:11:10 +0000477
Ted Kremenek9e240492008-10-04 05:50:14 +0000478 return false;
479}
Ted Kremenek1670e402009-04-11 00:11:10 +0000480
481
482//===----------------------------------------------------------------------===//
483// View handling.
484//===----------------------------------------------------------------------===//
485
486const MemRegion *TypedViewRegion::removeViews() const {
487 const SubRegion *SR = this;
488 const MemRegion *R = SR;
489 while (SR && isa<TypedViewRegion>(SR)) {
490 R = SR->getSuperRegion();
491 SR = dyn_cast<SubRegion>(R);
492 }
493 return R;
494}