1 //== MemRegion.h - 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 #ifndef LLVM_CLANG_GR_MEMREGION_H
17 #define LLVM_CLANG_GR_MEMREGION_H
18
19 #include "clang/AST/ASTContext.h"
20 #include "clang/AST/CharUnits.h"
21 #include "clang/AST/Decl.h"
22 #include "clang/AST/ExprObjC.h"
23 #include "clang/Basic/LLVM.h"
24 #include "clang/StaticAnalyzer/Core/PathSensitive/SVals.h"
25 #include "llvm/ADT/FoldingSet.h"
26 #include "llvm/Support/ErrorHandling.h"
27 #include <string>
28
29 namespace llvm {
30 class BumpPtrAllocator;
31 }
32
33 namespace clang {
34
35 class LocationContext;
36 class StackFrameContext;
37
38 namespace ento {
39
40 class MemRegionManager;
41 class MemSpaceRegion;
42 class SValBuilder;
43 class VarRegion;
44 class CodeTextRegion;
45
46 /// Represent a region's offset within the top level base region.
47 class RegionOffset {
48 /// The base region.
49 const MemRegion *R;
50
51 /// The bit offset within the base region. It shouldn't be negative.
52 int64_t Offset;
53
54 public:
55 // We're using a const instead of an enumeration due to the size required;
56 // Visual Studio will only create enumerations of size int, not long long.
57 static const int64_t Symbolic = INT64_MAX;
58
RegionOffset()59 RegionOffset() : R(0) {}
RegionOffset(const MemRegion * r,int64_t off)60 RegionOffset(const MemRegion *r, int64_t off) : R(r), Offset(off) {}
61
getRegion()62 const MemRegion *getRegion() const { return R; }
63
hasSymbolicOffset()64 bool hasSymbolicOffset() const { return Offset == Symbolic; }
65
getOffset()66 int64_t getOffset() const {
67 assert(!hasSymbolicOffset());
68 return Offset;
69 }
70
isValid()71 bool isValid() const { return R; }
72 };
73
74 //===----------------------------------------------------------------------===//
75 // Base region classes.
76 //===----------------------------------------------------------------------===//
77
78 /// MemRegion - The root abstract class for all memory regions.
79 class MemRegion : public llvm::FoldingSetNode {
80 friend class MemRegionManager;
81 public:
82 enum Kind {
83 // Memory spaces.
84 GenericMemSpaceRegionKind,
85 StackLocalsSpaceRegionKind,
86 StackArgumentsSpaceRegionKind,
87 HeapSpaceRegionKind,
88 UnknownSpaceRegionKind,
89 StaticGlobalSpaceRegionKind,
90 GlobalInternalSpaceRegionKind,
91 GlobalSystemSpaceRegionKind,
92 GlobalImmutableSpaceRegionKind,
93 BEG_NON_STATIC_GLOBAL_MEMSPACES = GlobalInternalSpaceRegionKind,
94 END_NON_STATIC_GLOBAL_MEMSPACES = GlobalImmutableSpaceRegionKind,
95 BEG_GLOBAL_MEMSPACES = StaticGlobalSpaceRegionKind,
96 END_GLOBAL_MEMSPACES = GlobalImmutableSpaceRegionKind,
97 BEG_MEMSPACES = GenericMemSpaceRegionKind,
98 END_MEMSPACES = GlobalImmutableSpaceRegionKind,
99 // Untyped regions.
100 SymbolicRegionKind,
101 AllocaRegionKind,
102 // Typed regions.
103 BEG_TYPED_REGIONS,
104 FunctionTextRegionKind = BEG_TYPED_REGIONS,
105 BlockTextRegionKind,
106 BlockDataRegionKind,
107 BEG_TYPED_VALUE_REGIONS,
108 CompoundLiteralRegionKind = BEG_TYPED_VALUE_REGIONS,
109 CXXThisRegionKind,
110 StringRegionKind,
111 ObjCStringRegionKind,
112 ElementRegionKind,
113 // Decl Regions.
114 BEG_DECL_REGIONS,
115 VarRegionKind = BEG_DECL_REGIONS,
116 FieldRegionKind,
117 ObjCIvarRegionKind,
118 END_DECL_REGIONS = ObjCIvarRegionKind,
119 CXXTempObjectRegionKind,
120 CXXBaseObjectRegionKind,
121 END_TYPED_VALUE_REGIONS = CXXBaseObjectRegionKind,
122 END_TYPED_REGIONS = CXXBaseObjectRegionKind
123 };
124
125 private:
126 const Kind kind;
127
128 protected:
MemRegion(Kind k)129 MemRegion(Kind k) : kind(k) {}
130 virtual ~MemRegion();
131
132 public:
133 ASTContext &getContext() const;
134
135 virtual void Profile(llvm::FoldingSetNodeID& ID) const = 0;
136
137 virtual MemRegionManager* getMemRegionManager() const = 0;
138
139 const MemSpaceRegion *getMemorySpace() const;
140
141 const MemRegion *getBaseRegion() const;
142
143 /// Check if the region is a subregion of the given region.
144 virtual bool isSubRegionOf(const MemRegion *R) const;
145
146 const MemRegion *StripCasts(bool StripBaseCasts = true) const;
147
148 bool hasGlobalsOrParametersStorage() const;
149
150 bool hasStackStorage() const;
151
152 bool hasStackNonParametersStorage() const;
153
154 bool hasStackParametersStorage() const;
155
156 /// Compute the offset within the top level memory object.
157 RegionOffset getAsOffset() const;
158
159 /// \brief Get a string representation of a region for debug use.
160 std::string getString() const;
161
162 virtual void dumpToStream(raw_ostream &os) const;
163
164 void dump() const;
165
166 /// \brief Returns true if this region can be printed in a user-friendly way.
167 virtual bool canPrintPretty() const;
168
169 /// \brief Print the region for use in diagnostics.
170 virtual void printPretty(raw_ostream &os) const;
171
getKind()172 Kind getKind() const { return kind; }
173
174 template<typename RegionTy> const RegionTy* getAs() const;
175
isBoundable()176 virtual bool isBoundable() const { return false; }
177 };
178
179 /// MemSpaceRegion - A memory region that represents a "memory space";
180 /// for example, the set of global variables, the stack frame, etc.
181 class MemSpaceRegion : public MemRegion {
182 protected:
183 friend class MemRegionManager;
184
185 MemRegionManager *Mgr;
186
187 MemSpaceRegion(MemRegionManager *mgr, Kind k = GenericMemSpaceRegionKind)
MemRegion(k)188 : MemRegion(k), Mgr(mgr) {
189 assert(classof(this));
190 }
191
getMemRegionManager()192 MemRegionManager* getMemRegionManager() const { return Mgr; }
193
194 public:
isBoundable()195 bool isBoundable() const { return false; }
196
197 void Profile(llvm::FoldingSetNodeID &ID) const;
198
classof(const MemRegion * R)199 static bool classof(const MemRegion *R) {
200 Kind k = R->getKind();
201 return k >= BEG_MEMSPACES && k <= END_MEMSPACES;
202 }
203 };
204
205 class GlobalsSpaceRegion : public MemSpaceRegion {
206 virtual void anchor();
207 protected:
GlobalsSpaceRegion(MemRegionManager * mgr,Kind k)208 GlobalsSpaceRegion(MemRegionManager *mgr, Kind k)
209 : MemSpaceRegion(mgr, k) {}
210 public:
classof(const MemRegion * R)211 static bool classof(const MemRegion *R) {
212 Kind k = R->getKind();
213 return k >= BEG_GLOBAL_MEMSPACES && k <= END_GLOBAL_MEMSPACES;
214 }
215 };
216
217 /// \brief The region of the static variables within the current CodeTextRegion
218 /// scope.
219 ///
220 /// Currently, only the static locals are placed there, so we know that these
221 /// variables do not get invalidated by calls to other functions.
222 class StaticGlobalSpaceRegion : public GlobalsSpaceRegion {
223 friend class MemRegionManager;
224
225 const CodeTextRegion *CR;
226
StaticGlobalSpaceRegion(MemRegionManager * mgr,const CodeTextRegion * cr)227 StaticGlobalSpaceRegion(MemRegionManager *mgr, const CodeTextRegion *cr)
228 : GlobalsSpaceRegion(mgr, StaticGlobalSpaceRegionKind), CR(cr) {}
229
230 public:
231 void Profile(llvm::FoldingSetNodeID &ID) const;
232
233 void dumpToStream(raw_ostream &os) const;
234
getCodeRegion()235 const CodeTextRegion *getCodeRegion() const { return CR; }
236
classof(const MemRegion * R)237 static bool classof(const MemRegion *R) {
238 return R->getKind() == StaticGlobalSpaceRegionKind;
239 }
240 };
241
242 /// \brief The region for all the non-static global variables.
243 ///
244 /// This class is further split into subclasses for efficient implementation of
245 /// invalidating a set of related global values as is done in
246 /// RegionStoreManager::invalidateRegions (instead of finding all the dependent
247 /// globals, we invalidate the whole parent region).
248 class NonStaticGlobalSpaceRegion : public GlobalsSpaceRegion {
249 friend class MemRegionManager;
250
251 protected:
NonStaticGlobalSpaceRegion(MemRegionManager * mgr,Kind k)252 NonStaticGlobalSpaceRegion(MemRegionManager *mgr, Kind k)
253 : GlobalsSpaceRegion(mgr, k) {}
254
255 public:
256
classof(const MemRegion * R)257 static bool classof(const MemRegion *R) {
258 Kind k = R->getKind();
259 return k >= BEG_NON_STATIC_GLOBAL_MEMSPACES &&
260 k <= END_NON_STATIC_GLOBAL_MEMSPACES;
261 }
262 };
263
264 /// \brief The region containing globals which are defined in system/external
265 /// headers and are considered modifiable by system calls (ex: errno).
266 class GlobalSystemSpaceRegion : public NonStaticGlobalSpaceRegion {
267 friend class MemRegionManager;
268
GlobalSystemSpaceRegion(MemRegionManager * mgr)269 GlobalSystemSpaceRegion(MemRegionManager *mgr)
270 : NonStaticGlobalSpaceRegion(mgr, GlobalSystemSpaceRegionKind) {}
271
272 public:
273
274 void dumpToStream(raw_ostream &os) const;
275
classof(const MemRegion * R)276 static bool classof(const MemRegion *R) {
277 return R->getKind() == GlobalSystemSpaceRegionKind;
278 }
279 };
280
281 /// \brief The region containing globals which are considered not to be modified
282 /// or point to data which could be modified as a result of a function call
283 /// (system or internal). Ex: Const global scalars would be modeled as part of
284 /// this region. This region also includes most system globals since they have
285 /// low chance of being modified.
286 class GlobalImmutableSpaceRegion : public NonStaticGlobalSpaceRegion {
287 friend class MemRegionManager;
288
GlobalImmutableSpaceRegion(MemRegionManager * mgr)289 GlobalImmutableSpaceRegion(MemRegionManager *mgr)
290 : NonStaticGlobalSpaceRegion(mgr, GlobalImmutableSpaceRegionKind) {}
291
292 public:
293
294 void dumpToStream(raw_ostream &os) const;
295
classof(const MemRegion * R)296 static bool classof(const MemRegion *R) {
297 return R->getKind() == GlobalImmutableSpaceRegionKind;
298 }
299 };
300
301 /// \brief The region containing globals which can be modified by calls to
302 /// "internally" defined functions - (for now just) functions other then system
303 /// calls.
304 class GlobalInternalSpaceRegion : public NonStaticGlobalSpaceRegion {
305 friend class MemRegionManager;
306
GlobalInternalSpaceRegion(MemRegionManager * mgr)307 GlobalInternalSpaceRegion(MemRegionManager *mgr)
308 : NonStaticGlobalSpaceRegion(mgr, GlobalInternalSpaceRegionKind) {}
309
310 public:
311
312 void dumpToStream(raw_ostream &os) const;
313
classof(const MemRegion * R)314 static bool classof(const MemRegion *R) {
315 return R->getKind() == GlobalInternalSpaceRegionKind;
316 }
317 };
318
319 class HeapSpaceRegion : public MemSpaceRegion {
320 virtual void anchor();
321 friend class MemRegionManager;
322
HeapSpaceRegion(MemRegionManager * mgr)323 HeapSpaceRegion(MemRegionManager *mgr)
324 : MemSpaceRegion(mgr, HeapSpaceRegionKind) {}
325 public:
326
327 void dumpToStream(raw_ostream &os) const;
328
classof(const MemRegion * R)329 static bool classof(const MemRegion *R) {
330 return R->getKind() == HeapSpaceRegionKind;
331 }
332 };
333
334 class UnknownSpaceRegion : public MemSpaceRegion {
335 virtual void anchor();
336 friend class MemRegionManager;
UnknownSpaceRegion(MemRegionManager * mgr)337 UnknownSpaceRegion(MemRegionManager *mgr)
338 : MemSpaceRegion(mgr, UnknownSpaceRegionKind) {}
339 public:
340
341 void dumpToStream(raw_ostream &os) const;
342
classof(const MemRegion * R)343 static bool classof(const MemRegion *R) {
344 return R->getKind() == UnknownSpaceRegionKind;
345 }
346 };
347
348 class StackSpaceRegion : public MemSpaceRegion {
349 private:
350 const StackFrameContext *SFC;
351
352 protected:
StackSpaceRegion(MemRegionManager * mgr,Kind k,const StackFrameContext * sfc)353 StackSpaceRegion(MemRegionManager *mgr, Kind k, const StackFrameContext *sfc)
354 : MemSpaceRegion(mgr, k), SFC(sfc) {
355 assert(classof(this));
356 }
357
358 public:
getStackFrame()359 const StackFrameContext *getStackFrame() const { return SFC; }
360
361 void Profile(llvm::FoldingSetNodeID &ID) const;
362
classof(const MemRegion * R)363 static bool classof(const MemRegion *R) {
364 Kind k = R->getKind();
365 return k >= StackLocalsSpaceRegionKind &&
366 k <= StackArgumentsSpaceRegionKind;
367 }
368 };
369
370 class StackLocalsSpaceRegion : public StackSpaceRegion {
371 virtual void anchor();
372 friend class MemRegionManager;
StackLocalsSpaceRegion(MemRegionManager * mgr,const StackFrameContext * sfc)373 StackLocalsSpaceRegion(MemRegionManager *mgr, const StackFrameContext *sfc)
374 : StackSpaceRegion(mgr, StackLocalsSpaceRegionKind, sfc) {}
375 public:
376
377 void dumpToStream(raw_ostream &os) const;
378
classof(const MemRegion * R)379 static bool classof(const MemRegion *R) {
380 return R->getKind() == StackLocalsSpaceRegionKind;
381 }
382 };
383
384 class StackArgumentsSpaceRegion : public StackSpaceRegion {
385 private:
386 virtual void anchor();
387 friend class MemRegionManager;
StackArgumentsSpaceRegion(MemRegionManager * mgr,const StackFrameContext * sfc)388 StackArgumentsSpaceRegion(MemRegionManager *mgr, const StackFrameContext *sfc)
389 : StackSpaceRegion(mgr, StackArgumentsSpaceRegionKind, sfc) {}
390 public:
391
392 void dumpToStream(raw_ostream &os) const;
393
classof(const MemRegion * R)394 static bool classof(const MemRegion *R) {
395 return R->getKind() == StackArgumentsSpaceRegionKind;
396 }
397 };
398
399
400 /// SubRegion - A region that subsets another larger region. Most regions
401 /// are subclasses of SubRegion.
402 class SubRegion : public MemRegion {
403 private:
404 virtual void anchor();
405 protected:
406 const MemRegion* superRegion;
SubRegion(const MemRegion * sReg,Kind k)407 SubRegion(const MemRegion* sReg, Kind k) : MemRegion(k), superRegion(sReg) {}
408 public:
getSuperRegion()409 const MemRegion* getSuperRegion() const {
410 return superRegion;
411 }
412
413 /// getExtent - Returns the size of the region in bytes.
getExtent(SValBuilder & svalBuilder)414 virtual DefinedOrUnknownSVal getExtent(SValBuilder &svalBuilder) const {
415 return UnknownVal();
416 }
417
418 MemRegionManager* getMemRegionManager() const;
419
420 virtual bool isSubRegionOf(const MemRegion* R) const;
421
classof(const MemRegion * R)422 static bool classof(const MemRegion* R) {
423 return R->getKind() > END_MEMSPACES;
424 }
425 };
426
427 //===----------------------------------------------------------------------===//
428 // MemRegion subclasses.
429 //===----------------------------------------------------------------------===//
430
431 /// AllocaRegion - A region that represents an untyped blob of bytes created
432 /// by a call to 'alloca'.
433 class AllocaRegion : public SubRegion {
434 friend class MemRegionManager;
435 protected:
436 unsigned Cnt; // Block counter. Used to distinguish different pieces of
437 // memory allocated by alloca at the same call site.
438 const Expr *Ex;
439
AllocaRegion(const Expr * ex,unsigned cnt,const MemRegion * superRegion)440 AllocaRegion(const Expr *ex, unsigned cnt, const MemRegion *superRegion)
441 : SubRegion(superRegion, AllocaRegionKind), Cnt(cnt), Ex(ex) {}
442
443 public:
444
getExpr()445 const Expr *getExpr() const { return Ex; }
446
isBoundable()447 bool isBoundable() const { return true; }
448
449 DefinedOrUnknownSVal getExtent(SValBuilder &svalBuilder) const;
450
451 void Profile(llvm::FoldingSetNodeID& ID) const;
452
453 static void ProfileRegion(llvm::FoldingSetNodeID& ID, const Expr *Ex,
454 unsigned Cnt, const MemRegion *superRegion);
455
456 void dumpToStream(raw_ostream &os) const;
457
classof(const MemRegion * R)458 static bool classof(const MemRegion* R) {
459 return R->getKind() == AllocaRegionKind;
460 }
461 };
462
463 /// TypedRegion - An abstract class representing regions that are typed.
464 class TypedRegion : public SubRegion {
465 public:
466 virtual void anchor();
467 protected:
TypedRegion(const MemRegion * sReg,Kind k)468 TypedRegion(const MemRegion* sReg, Kind k) : SubRegion(sReg, k) {}
469
470 public:
471 virtual QualType getLocationType() const = 0;
472
getDesugaredLocationType(ASTContext & Context)473 QualType getDesugaredLocationType(ASTContext &Context) const {
474 return getLocationType().getDesugaredType(Context);
475 }
476
isBoundable()477 bool isBoundable() const { return true; }
478
classof(const MemRegion * R)479 static bool classof(const MemRegion* R) {
480 unsigned k = R->getKind();
481 return k >= BEG_TYPED_REGIONS && k <= END_TYPED_REGIONS;
482 }
483 };
484
485 /// TypedValueRegion - An abstract class representing regions having a typed value.
486 class TypedValueRegion : public TypedRegion {
487 public:
488 virtual void anchor();
489 protected:
TypedValueRegion(const MemRegion * sReg,Kind k)490 TypedValueRegion(const MemRegion* sReg, Kind k) : TypedRegion(sReg, k) {}
491
492 public:
493 virtual QualType getValueType() const = 0;
494
getLocationType()495 virtual QualType getLocationType() const {
496 // FIXME: We can possibly optimize this later to cache this value.
497 QualType T = getValueType();
498 ASTContext &ctx = getContext();
499 if (T->getAs<ObjCObjectType>())
500 return ctx.getObjCObjectPointerType(T);
501 return ctx.getPointerType(getValueType());
502 }
503
getDesugaredValueType(ASTContext & Context)504 QualType getDesugaredValueType(ASTContext &Context) const {
505 QualType T = getValueType();
506 return T.getTypePtrOrNull() ? T.getDesugaredType(Context) : T;
507 }
508
509 DefinedOrUnknownSVal getExtent(SValBuilder &svalBuilder) const;
510
classof(const MemRegion * R)511 static bool classof(const MemRegion* R) {
512 unsigned k = R->getKind();
513 return k >= BEG_TYPED_VALUE_REGIONS && k <= END_TYPED_VALUE_REGIONS;
514 }
515 };
516
517
518 class CodeTextRegion : public TypedRegion {
519 public:
520 virtual void anchor();
521 protected:
CodeTextRegion(const MemRegion * sreg,Kind k)522 CodeTextRegion(const MemRegion *sreg, Kind k) : TypedRegion(sreg, k) {}
523 public:
isBoundable()524 bool isBoundable() const { return false; }
525
classof(const MemRegion * R)526 static bool classof(const MemRegion* R) {
527 Kind k = R->getKind();
528 return k >= FunctionTextRegionKind && k <= BlockTextRegionKind;
529 }
530 };
531
532 /// FunctionTextRegion - A region that represents code texts of function.
533 class FunctionTextRegion : public CodeTextRegion {
534 const NamedDecl *FD;
535 public:
FunctionTextRegion(const NamedDecl * fd,const MemRegion * sreg)536 FunctionTextRegion(const NamedDecl *fd, const MemRegion* sreg)
537 : CodeTextRegion(sreg, FunctionTextRegionKind), FD(fd) {
538 assert(isa<ObjCMethodDecl>(fd) || isa<FunctionDecl>(fd));
539 }
540
getLocationType()541 QualType getLocationType() const {
542 const ASTContext &Ctx = getContext();
543 if (const FunctionDecl *D = dyn_cast<FunctionDecl>(FD)) {
544 return Ctx.getPointerType(D->getType());
545 }
546
547 assert(isa<ObjCMethodDecl>(FD));
548 assert(false && "Getting the type of ObjCMethod is not supported yet");
549
550 // TODO: We might want to return a different type here (ex: id (*ty)(...))
551 // depending on how it is used.
552 return QualType();
553 }
554
getDecl()555 const NamedDecl *getDecl() const {
556 return FD;
557 }
558
559 virtual void dumpToStream(raw_ostream &os) const;
560
561 void Profile(llvm::FoldingSetNodeID& ID) const;
562
563 static void ProfileRegion(llvm::FoldingSetNodeID& ID, const NamedDecl *FD,
564 const MemRegion*);
565
classof(const MemRegion * R)566 static bool classof(const MemRegion* R) {
567 return R->getKind() == FunctionTextRegionKind;
568 }
569 };
570
571
572 /// BlockTextRegion - A region that represents code texts of blocks (closures).
573 /// Blocks are represented with two kinds of regions. BlockTextRegions
574 /// represent the "code", while BlockDataRegions represent instances of blocks,
575 /// which correspond to "code+data". The distinction is important, because
576 /// like a closure a block captures the values of externally referenced
577 /// variables.
578 class BlockTextRegion : public CodeTextRegion {
579 friend class MemRegionManager;
580
581 const BlockDecl *BD;
582 AnalysisDeclContext *AC;
583 CanQualType locTy;
584
BlockTextRegion(const BlockDecl * bd,CanQualType lTy,AnalysisDeclContext * ac,const MemRegion * sreg)585 BlockTextRegion(const BlockDecl *bd, CanQualType lTy,
586 AnalysisDeclContext *ac, const MemRegion* sreg)
587 : CodeTextRegion(sreg, BlockTextRegionKind), BD(bd), AC(ac), locTy(lTy) {}
588
589 public:
getLocationType()590 QualType getLocationType() const {
591 return locTy;
592 }
593
getDecl()594 const BlockDecl *getDecl() const {
595 return BD;
596 }
597
getAnalysisDeclContext()598 AnalysisDeclContext *getAnalysisDeclContext() const { return AC; }
599
600 virtual void dumpToStream(raw_ostream &os) const;
601
602 void Profile(llvm::FoldingSetNodeID& ID) const;
603
604 static void ProfileRegion(llvm::FoldingSetNodeID& ID, const BlockDecl *BD,
605 CanQualType, const AnalysisDeclContext*,
606 const MemRegion*);
607
classof(const MemRegion * R)608 static bool classof(const MemRegion* R) {
609 return R->getKind() == BlockTextRegionKind;
610 }
611 };
612
613 /// BlockDataRegion - A region that represents a block instance.
614 /// Blocks are represented with two kinds of regions. BlockTextRegions
615 /// represent the "code", while BlockDataRegions represent instances of blocks,
616 /// which correspond to "code+data". The distinction is important, because
617 /// like a closure a block captures the values of externally referenced
618 /// variables.
619 class BlockDataRegion : public TypedRegion {
620 friend class MemRegionManager;
621 const BlockTextRegion *BC;
622 const LocationContext *LC; // Can be null */
623 void *ReferencedVars;
624 void *OriginalVars;
625
BlockDataRegion(const BlockTextRegion * bc,const LocationContext * lc,const MemRegion * sreg)626 BlockDataRegion(const BlockTextRegion *bc, const LocationContext *lc,
627 const MemRegion *sreg)
628 : TypedRegion(sreg, BlockDataRegionKind), BC(bc), LC(lc),
629 ReferencedVars(0), OriginalVars(0) {}
630
631 public:
getCodeRegion()632 const BlockTextRegion *getCodeRegion() const { return BC; }
633
getDecl()634 const BlockDecl *getDecl() const { return BC->getDecl(); }
635
getLocationType()636 QualType getLocationType() const { return BC->getLocationType(); }
637
638 class referenced_vars_iterator {
639 const MemRegion * const *R;
640 const MemRegion * const *OriginalR;
641 public:
referenced_vars_iterator(const MemRegion * const * r,const MemRegion * const * originalR)642 explicit referenced_vars_iterator(const MemRegion * const *r,
643 const MemRegion * const *originalR)
644 : R(r), OriginalR(originalR) {}
645
getCapturedRegion()646 const VarRegion *getCapturedRegion() const {
647 return cast<VarRegion>(*R);
648 }
getOriginalRegion()649 const VarRegion *getOriginalRegion() const {
650 return cast<VarRegion>(*OriginalR);
651 }
652
653 bool operator==(const referenced_vars_iterator &I) const {
654 assert((R == 0) == (I.R == 0));
655 return I.R == R;
656 }
657 bool operator!=(const referenced_vars_iterator &I) const {
658 assert((R == 0) == (I.R == 0));
659 return I.R != R;
660 }
661 referenced_vars_iterator &operator++() {
662 ++R;
663 ++OriginalR;
664 return *this;
665 }
666 };
667
668 /// Return the original region for a captured region, if
669 /// one exists.
670 const VarRegion *getOriginalRegion(const VarRegion *VR) const;
671
672 referenced_vars_iterator referenced_vars_begin() const;
673 referenced_vars_iterator referenced_vars_end() const;
674
675 virtual void dumpToStream(raw_ostream &os) const;
676
677 void Profile(llvm::FoldingSetNodeID& ID) const;
678
679 static void ProfileRegion(llvm::FoldingSetNodeID&, const BlockTextRegion *,
680 const LocationContext *, const MemRegion *);
681
classof(const MemRegion * R)682 static bool classof(const MemRegion* R) {
683 return R->getKind() == BlockDataRegionKind;
684 }
685 private:
686 void LazyInitializeReferencedVars();
687 std::pair<const VarRegion *, const VarRegion *>
688 getCaptureRegions(const VarDecl *VD);
689 };
690
691 /// SymbolicRegion - A special, "non-concrete" region. Unlike other region
692 /// clases, SymbolicRegion represents a region that serves as an alias for
693 /// either a real region, a NULL pointer, etc. It essentially is used to
694 /// map the concept of symbolic values into the domain of regions. Symbolic
695 /// regions do not need to be typed.
696 class SymbolicRegion : public SubRegion {
697 protected:
698 const SymbolRef sym;
699
700 public:
SymbolicRegion(const SymbolRef s,const MemRegion * sreg)701 SymbolicRegion(const SymbolRef s, const MemRegion* sreg)
702 : SubRegion(sreg, SymbolicRegionKind), sym(s) {}
703
getSymbol()704 SymbolRef getSymbol() const {
705 return sym;
706 }
707
isBoundable()708 bool isBoundable() const { return true; }
709
710 DefinedOrUnknownSVal getExtent(SValBuilder &svalBuilder) const;
711
712 void Profile(llvm::FoldingSetNodeID& ID) const;
713
714 static void ProfileRegion(llvm::FoldingSetNodeID& ID,
715 SymbolRef sym,
716 const MemRegion* superRegion);
717
718 void dumpToStream(raw_ostream &os) const;
719
classof(const MemRegion * R)720 static bool classof(const MemRegion* R) {
721 return R->getKind() == SymbolicRegionKind;
722 }
723 };
724
725 /// StringRegion - Region associated with a StringLiteral.
726 class StringRegion : public TypedValueRegion {
727 friend class MemRegionManager;
728 const StringLiteral* Str;
729 protected:
730
StringRegion(const StringLiteral * str,const MemRegion * sreg)731 StringRegion(const StringLiteral* str, const MemRegion* sreg)
732 : TypedValueRegion(sreg, StringRegionKind), Str(str) {}
733
734 static void ProfileRegion(llvm::FoldingSetNodeID& ID,
735 const StringLiteral* Str,
736 const MemRegion* superRegion);
737
738 public:
739
getStringLiteral()740 const StringLiteral* getStringLiteral() const { return Str; }
741
getValueType()742 QualType getValueType() const {
743 return Str->getType();
744 }
745
746 DefinedOrUnknownSVal getExtent(SValBuilder &svalBuilder) const;
747
isBoundable()748 bool isBoundable() const { return false; }
749
Profile(llvm::FoldingSetNodeID & ID)750 void Profile(llvm::FoldingSetNodeID& ID) const {
751 ProfileRegion(ID, Str, superRegion);
752 }
753
754 void dumpToStream(raw_ostream &os) const;
755
classof(const MemRegion * R)756 static bool classof(const MemRegion* R) {
757 return R->getKind() == StringRegionKind;
758 }
759 };
760
761 /// The region associated with an ObjCStringLiteral.
762 class ObjCStringRegion : public TypedValueRegion {
763 friend class MemRegionManager;
764 const ObjCStringLiteral* Str;
765 protected:
766
ObjCStringRegion(const ObjCStringLiteral * str,const MemRegion * sreg)767 ObjCStringRegion(const ObjCStringLiteral* str, const MemRegion* sreg)
768 : TypedValueRegion(sreg, ObjCStringRegionKind), Str(str) {}
769
770 static void ProfileRegion(llvm::FoldingSetNodeID& ID,
771 const ObjCStringLiteral* Str,
772 const MemRegion* superRegion);
773
774 public:
775
getObjCStringLiteral()776 const ObjCStringLiteral* getObjCStringLiteral() const { return Str; }
777
getValueType()778 QualType getValueType() const {
779 return Str->getType();
780 }
781
isBoundable()782 bool isBoundable() const { return false; }
783
Profile(llvm::FoldingSetNodeID & ID)784 void Profile(llvm::FoldingSetNodeID& ID) const {
785 ProfileRegion(ID, Str, superRegion);
786 }
787
788 void dumpToStream(raw_ostream &os) const;
789
classof(const MemRegion * R)790 static bool classof(const MemRegion* R) {
791 return R->getKind() == ObjCStringRegionKind;
792 }
793 };
794
795 /// CompoundLiteralRegion - A memory region representing a compound literal.
796 /// Compound literals are essentially temporaries that are stack allocated
797 /// or in the global constant pool.
798 class CompoundLiteralRegion : public TypedValueRegion {
799 private:
800 friend class MemRegionManager;
801 const CompoundLiteralExpr *CL;
802
CompoundLiteralRegion(const CompoundLiteralExpr * cl,const MemRegion * sReg)803 CompoundLiteralRegion(const CompoundLiteralExpr *cl, const MemRegion* sReg)
804 : TypedValueRegion(sReg, CompoundLiteralRegionKind), CL(cl) {}
805
806 static void ProfileRegion(llvm::FoldingSetNodeID& ID,
807 const CompoundLiteralExpr *CL,
808 const MemRegion* superRegion);
809 public:
getValueType()810 QualType getValueType() const {
811 return CL->getType();
812 }
813
isBoundable()814 bool isBoundable() const { return !CL->isFileScope(); }
815
816 void Profile(llvm::FoldingSetNodeID& ID) const;
817
818 void dumpToStream(raw_ostream &os) const;
819
getLiteralExpr()820 const CompoundLiteralExpr *getLiteralExpr() const { return CL; }
821
classof(const MemRegion * R)822 static bool classof(const MemRegion* R) {
823 return R->getKind() == CompoundLiteralRegionKind;
824 }
825 };
826
827 class DeclRegion : public TypedValueRegion {
828 protected:
829 const Decl *D;
830
DeclRegion(const Decl * d,const MemRegion * sReg,Kind k)831 DeclRegion(const Decl *d, const MemRegion* sReg, Kind k)
832 : TypedValueRegion(sReg, k), D(d) {}
833
834 static void ProfileRegion(llvm::FoldingSetNodeID& ID, const Decl *D,
835 const MemRegion* superRegion, Kind k);
836
837 public:
getDecl()838 const Decl *getDecl() const { return D; }
839 void Profile(llvm::FoldingSetNodeID& ID) const;
840
classof(const MemRegion * R)841 static bool classof(const MemRegion* R) {
842 unsigned k = R->getKind();
843 return k >= BEG_DECL_REGIONS && k <= END_DECL_REGIONS;
844 }
845 };
846
847 class VarRegion : public DeclRegion {
848 friend class MemRegionManager;
849
850 // Constructors and private methods.
VarRegion(const VarDecl * vd,const MemRegion * sReg)851 VarRegion(const VarDecl *vd, const MemRegion* sReg)
852 : DeclRegion(vd, sReg, VarRegionKind) {}
853
ProfileRegion(llvm::FoldingSetNodeID & ID,const VarDecl * VD,const MemRegion * superRegion)854 static void ProfileRegion(llvm::FoldingSetNodeID& ID, const VarDecl *VD,
855 const MemRegion *superRegion) {
856 DeclRegion::ProfileRegion(ID, VD, superRegion, VarRegionKind);
857 }
858
859 void Profile(llvm::FoldingSetNodeID& ID) const;
860
861 public:
getDecl()862 const VarDecl *getDecl() const { return cast<VarDecl>(D); }
863
864 const StackFrameContext *getStackFrame() const;
865
getValueType()866 QualType getValueType() const {
867 // FIXME: We can cache this if needed.
868 return getDecl()->getType();
869 }
870
871 void dumpToStream(raw_ostream &os) const;
872
classof(const MemRegion * R)873 static bool classof(const MemRegion* R) {
874 return R->getKind() == VarRegionKind;
875 }
876
877 bool canPrintPretty() const;
878 void printPretty(raw_ostream &os) const;
879 };
880
881 /// CXXThisRegion - Represents the region for the implicit 'this' parameter
882 /// in a call to a C++ method. This region doesn't represent the object
883 /// referred to by 'this', but rather 'this' itself.
884 class CXXThisRegion : public TypedValueRegion {
885 friend class MemRegionManager;
CXXThisRegion(const PointerType * thisPointerTy,const MemRegion * sReg)886 CXXThisRegion(const PointerType *thisPointerTy,
887 const MemRegion *sReg)
888 : TypedValueRegion(sReg, CXXThisRegionKind), ThisPointerTy(thisPointerTy) {}
889
890 static void ProfileRegion(llvm::FoldingSetNodeID &ID,
891 const PointerType *PT,
892 const MemRegion *sReg);
893
894 void Profile(llvm::FoldingSetNodeID &ID) const;
895
896 public:
getValueType()897 QualType getValueType() const {
898 return QualType(ThisPointerTy, 0);
899 }
900
901 void dumpToStream(raw_ostream &os) const;
902
classof(const MemRegion * R)903 static bool classof(const MemRegion* R) {
904 return R->getKind() == CXXThisRegionKind;
905 }
906
907 private:
908 const PointerType *ThisPointerTy;
909 };
910
911 class FieldRegion : public DeclRegion {
912 friend class MemRegionManager;
913
FieldRegion(const FieldDecl * fd,const MemRegion * sReg)914 FieldRegion(const FieldDecl *fd, const MemRegion* sReg)
915 : DeclRegion(fd, sReg, FieldRegionKind) {}
916
917 public:
getDecl()918 const FieldDecl *getDecl() const { return cast<FieldDecl>(D); }
919
getValueType()920 QualType getValueType() const {
921 // FIXME: We can cache this if needed.
922 return getDecl()->getType();
923 }
924
925 DefinedOrUnknownSVal getExtent(SValBuilder &svalBuilder) const;
926
ProfileRegion(llvm::FoldingSetNodeID & ID,const FieldDecl * FD,const MemRegion * superRegion)927 static void ProfileRegion(llvm::FoldingSetNodeID& ID, const FieldDecl *FD,
928 const MemRegion* superRegion) {
929 DeclRegion::ProfileRegion(ID, FD, superRegion, FieldRegionKind);
930 }
931
classof(const MemRegion * R)932 static bool classof(const MemRegion* R) {
933 return R->getKind() == FieldRegionKind;
934 }
935
936 void dumpToStream(raw_ostream &os) const;
937
938 bool canPrintPretty() const;
939 void printPretty(raw_ostream &os) const;
940 };
941
942 class ObjCIvarRegion : public DeclRegion {
943
944 friend class MemRegionManager;
945
946 ObjCIvarRegion(const ObjCIvarDecl *ivd, const MemRegion* sReg);
947
948 static void ProfileRegion(llvm::FoldingSetNodeID& ID, const ObjCIvarDecl *ivd,
949 const MemRegion* superRegion);
950
951 public:
952 const ObjCIvarDecl *getDecl() const;
953 QualType getValueType() const;
954
955 bool canPrintPretty() const;
956 void printPretty(raw_ostream &os) const;
957
958 void dumpToStream(raw_ostream &os) const;
959
classof(const MemRegion * R)960 static bool classof(const MemRegion* R) {
961 return R->getKind() == ObjCIvarRegionKind;
962 }
963 };
964 //===----------------------------------------------------------------------===//
965 // Auxiliary data classes for use with MemRegions.
966 //===----------------------------------------------------------------------===//
967
968 class ElementRegion;
969
970 class RegionRawOffset {
971 private:
972 friend class ElementRegion;
973
974 const MemRegion *Region;
975 CharUnits Offset;
976
977 RegionRawOffset(const MemRegion* reg, CharUnits offset = CharUnits::Zero())
Region(reg)978 : Region(reg), Offset(offset) {}
979
980 public:
981 // FIXME: Eventually support symbolic offsets.
getOffset()982 CharUnits getOffset() const { return Offset; }
getRegion()983 const MemRegion *getRegion() const { return Region; }
984
985 void dumpToStream(raw_ostream &os) const;
986 void dump() const;
987 };
988
989 /// \brief ElementRegin is used to represent both array elements and casts.
990 class ElementRegion : public TypedValueRegion {
991 friend class MemRegionManager;
992
993 QualType ElementType;
994 NonLoc Index;
995
ElementRegion(QualType elementType,NonLoc Idx,const MemRegion * sReg)996 ElementRegion(QualType elementType, NonLoc Idx, const MemRegion* sReg)
997 : TypedValueRegion(sReg, ElementRegionKind),
998 ElementType(elementType), Index(Idx) {
999 assert((!Idx.getAs<nonloc::ConcreteInt>() ||
1000 Idx.castAs<nonloc::ConcreteInt>().getValue().isSigned()) &&
1001 "The index must be signed");
1002 }
1003
1004 static void ProfileRegion(llvm::FoldingSetNodeID& ID, QualType elementType,
1005 SVal Idx, const MemRegion* superRegion);
1006
1007 public:
1008
getIndex()1009 NonLoc getIndex() const { return Index; }
1010
getValueType()1011 QualType getValueType() const {
1012 return ElementType;
1013 }
1014
getElementType()1015 QualType getElementType() const {
1016 return ElementType;
1017 }
1018 /// Compute the offset within the array. The array might also be a subobject.
1019 RegionRawOffset getAsArrayOffset() const;
1020
1021 void dumpToStream(raw_ostream &os) const;
1022
1023 void Profile(llvm::FoldingSetNodeID& ID) const;
1024
classof(const MemRegion * R)1025 static bool classof(const MemRegion* R) {
1026 return R->getKind() == ElementRegionKind;
1027 }
1028 };
1029
1030 // C++ temporary object associated with an expression.
1031 class CXXTempObjectRegion : public TypedValueRegion {
1032 friend class MemRegionManager;
1033
1034 Expr const *Ex;
1035
CXXTempObjectRegion(Expr const * E,MemRegion const * sReg)1036 CXXTempObjectRegion(Expr const *E, MemRegion const *sReg)
1037 : TypedValueRegion(sReg, CXXTempObjectRegionKind), Ex(E) {}
1038
1039 static void ProfileRegion(llvm::FoldingSetNodeID &ID,
1040 Expr const *E, const MemRegion *sReg);
1041
1042 public:
getExpr()1043 const Expr *getExpr() const { return Ex; }
1044
getValueType()1045 QualType getValueType() const {
1046 return Ex->getType();
1047 }
1048
1049 void dumpToStream(raw_ostream &os) const;
1050
1051 void Profile(llvm::FoldingSetNodeID &ID) const;
1052
classof(const MemRegion * R)1053 static bool classof(const MemRegion* R) {
1054 return R->getKind() == CXXTempObjectRegionKind;
1055 }
1056 };
1057
1058 // CXXBaseObjectRegion represents a base object within a C++ object. It is
1059 // identified by the base class declaration and the region of its parent object.
1060 class CXXBaseObjectRegion : public TypedValueRegion {
1061 friend class MemRegionManager;
1062
1063 llvm::PointerIntPair<const CXXRecordDecl *, 1, bool> Data;
1064
CXXBaseObjectRegion(const CXXRecordDecl * RD,bool IsVirtual,const MemRegion * SReg)1065 CXXBaseObjectRegion(const CXXRecordDecl *RD, bool IsVirtual,
1066 const MemRegion *SReg)
1067 : TypedValueRegion(SReg, CXXBaseObjectRegionKind), Data(RD, IsVirtual) {}
1068
1069 static void ProfileRegion(llvm::FoldingSetNodeID &ID, const CXXRecordDecl *RD,
1070 bool IsVirtual, const MemRegion *SReg);
1071
1072 public:
getDecl()1073 const CXXRecordDecl *getDecl() const { return Data.getPointer(); }
isVirtual()1074 bool isVirtual() const { return Data.getInt(); }
1075
1076 QualType getValueType() const;
1077
1078 void dumpToStream(raw_ostream &os) const;
1079
1080 void Profile(llvm::FoldingSetNodeID &ID) const;
1081
classof(const MemRegion * region)1082 static bool classof(const MemRegion *region) {
1083 return region->getKind() == CXXBaseObjectRegionKind;
1084 }
1085 };
1086
1087 template<typename RegionTy>
getAs()1088 const RegionTy* MemRegion::getAs() const {
1089 if (const RegionTy* RT = dyn_cast<RegionTy>(this))
1090 return RT;
1091
1092 return NULL;
1093 }
1094
1095 //===----------------------------------------------------------------------===//
1096 // MemRegionManager - Factory object for creating regions.
1097 //===----------------------------------------------------------------------===//
1098
1099 class MemRegionManager {
1100 ASTContext &C;
1101 llvm::BumpPtrAllocator& A;
1102 llvm::FoldingSet<MemRegion> Regions;
1103
1104 GlobalInternalSpaceRegion *InternalGlobals;
1105 GlobalSystemSpaceRegion *SystemGlobals;
1106 GlobalImmutableSpaceRegion *ImmutableGlobals;
1107
1108
1109 llvm::DenseMap<const StackFrameContext *, StackLocalsSpaceRegion *>
1110 StackLocalsSpaceRegions;
1111 llvm::DenseMap<const StackFrameContext *, StackArgumentsSpaceRegion *>
1112 StackArgumentsSpaceRegions;
1113 llvm::DenseMap<const CodeTextRegion *, StaticGlobalSpaceRegion *>
1114 StaticsGlobalSpaceRegions;
1115
1116 HeapSpaceRegion *heap;
1117 UnknownSpaceRegion *unknown;
1118 MemSpaceRegion *code;
1119
1120 public:
MemRegionManager(ASTContext & c,llvm::BumpPtrAllocator & a)1121 MemRegionManager(ASTContext &c, llvm::BumpPtrAllocator& a)
1122 : C(c), A(a), InternalGlobals(0), SystemGlobals(0), ImmutableGlobals(0),
1123 heap(0), unknown(0), code(0) {}
1124
1125 ~MemRegionManager();
1126
getContext()1127 ASTContext &getContext() { return C; }
1128
getAllocator()1129 llvm::BumpPtrAllocator &getAllocator() { return A; }
1130
1131 /// getStackLocalsRegion - Retrieve the memory region associated with the
1132 /// specified stack frame.
1133 const StackLocalsSpaceRegion *
1134 getStackLocalsRegion(const StackFrameContext *STC);
1135
1136 /// getStackArgumentsRegion - Retrieve the memory region associated with
1137 /// function/method arguments of the specified stack frame.
1138 const StackArgumentsSpaceRegion *
1139 getStackArgumentsRegion(const StackFrameContext *STC);
1140
1141 /// getGlobalsRegion - Retrieve the memory region associated with
1142 /// global variables.
1143 const GlobalsSpaceRegion *getGlobalsRegion(
1144 MemRegion::Kind K = MemRegion::GlobalInternalSpaceRegionKind,
1145 const CodeTextRegion *R = 0);
1146
1147 /// getHeapRegion - Retrieve the memory region associated with the
1148 /// generic "heap".
1149 const HeapSpaceRegion *getHeapRegion();
1150
1151 /// getUnknownRegion - Retrieve the memory region associated with unknown
1152 /// memory space.
1153 const MemSpaceRegion *getUnknownRegion();
1154
1155 const MemSpaceRegion *getCodeRegion();
1156
1157 /// getAllocaRegion - Retrieve a region associated with a call to alloca().
1158 const AllocaRegion *getAllocaRegion(const Expr *Ex, unsigned Cnt,
1159 const LocationContext *LC);
1160
1161 /// getCompoundLiteralRegion - Retrieve the region associated with a
1162 /// given CompoundLiteral.
1163 const CompoundLiteralRegion*
1164 getCompoundLiteralRegion(const CompoundLiteralExpr *CL,
1165 const LocationContext *LC);
1166
1167 /// getCXXThisRegion - Retrieve the [artificial] region associated with the
1168 /// parameter 'this'.
1169 const CXXThisRegion *getCXXThisRegion(QualType thisPointerTy,
1170 const LocationContext *LC);
1171
1172 /// \brief Retrieve or create a "symbolic" memory region.
1173 const SymbolicRegion* getSymbolicRegion(SymbolRef Sym);
1174
1175 /// \brief Return a unique symbolic region belonging to heap memory space.
1176 const SymbolicRegion *getSymbolicHeapRegion(SymbolRef sym);
1177
1178 const StringRegion *getStringRegion(const StringLiteral* Str);
1179
1180 const ObjCStringRegion *getObjCStringRegion(const ObjCStringLiteral *Str);
1181
1182 /// getVarRegion - Retrieve or create the memory region associated with
1183 /// a specified VarDecl and LocationContext.
1184 const VarRegion* getVarRegion(const VarDecl *D, const LocationContext *LC);
1185
1186 /// getVarRegion - Retrieve or create the memory region associated with
1187 /// a specified VarDecl and super region.
1188 const VarRegion* getVarRegion(const VarDecl *D, const MemRegion *superR);
1189
1190 /// getElementRegion - Retrieve the memory region associated with the
1191 /// associated element type, index, and super region.
1192 const ElementRegion *getElementRegion(QualType elementType, NonLoc Idx,
1193 const MemRegion *superRegion,
1194 ASTContext &Ctx);
1195
getElementRegionWithSuper(const ElementRegion * ER,const MemRegion * superRegion)1196 const ElementRegion *getElementRegionWithSuper(const ElementRegion *ER,
1197 const MemRegion *superRegion) {
1198 return getElementRegion(ER->getElementType(), ER->getIndex(),
1199 superRegion, ER->getContext());
1200 }
1201
1202 /// getFieldRegion - Retrieve or create the memory region associated with
1203 /// a specified FieldDecl. 'superRegion' corresponds to the containing
1204 /// memory region (which typically represents the memory representing
1205 /// a structure or class).
1206 const FieldRegion *getFieldRegion(const FieldDecl *fd,
1207 const MemRegion* superRegion);
1208
getFieldRegionWithSuper(const FieldRegion * FR,const MemRegion * superRegion)1209 const FieldRegion *getFieldRegionWithSuper(const FieldRegion *FR,
1210 const MemRegion *superRegion) {
1211 return getFieldRegion(FR->getDecl(), superRegion);
1212 }
1213
1214 /// getObjCIvarRegion - Retrieve or create the memory region associated with
1215 /// a specified Objective-c instance variable. 'superRegion' corresponds
1216 /// to the containing region (which typically represents the Objective-C
1217 /// object).
1218 const ObjCIvarRegion *getObjCIvarRegion(const ObjCIvarDecl *ivd,
1219 const MemRegion* superRegion);
1220
1221 const CXXTempObjectRegion *getCXXTempObjectRegion(Expr const *Ex,
1222 LocationContext const *LC);
1223
1224 /// Create a CXXBaseObjectRegion with the given base class for region
1225 /// \p Super.
1226 ///
1227 /// The type of \p Super is assumed be a class deriving from \p BaseClass.
1228 const CXXBaseObjectRegion *
1229 getCXXBaseObjectRegion(const CXXRecordDecl *BaseClass, const MemRegion *Super,
1230 bool IsVirtual);
1231
1232 /// Create a CXXBaseObjectRegion with the same CXXRecordDecl but a different
1233 /// super region.
1234 const CXXBaseObjectRegion *
getCXXBaseObjectRegionWithSuper(const CXXBaseObjectRegion * baseReg,const MemRegion * superRegion)1235 getCXXBaseObjectRegionWithSuper(const CXXBaseObjectRegion *baseReg,
1236 const MemRegion *superRegion) {
1237 return getCXXBaseObjectRegion(baseReg->getDecl(), superRegion,
1238 baseReg->isVirtual());
1239 }
1240
1241 const FunctionTextRegion *getFunctionTextRegion(const NamedDecl *FD);
1242 const BlockTextRegion *getBlockTextRegion(const BlockDecl *BD,
1243 CanQualType locTy,
1244 AnalysisDeclContext *AC);
1245
1246 /// getBlockDataRegion - Get the memory region associated with an instance
1247 /// of a block. Unlike many other MemRegions, the LocationContext*
1248 /// argument is allowed to be NULL for cases where we have no known
1249 /// context.
1250 const BlockDataRegion *getBlockDataRegion(const BlockTextRegion *bc,
1251 const LocationContext *lc = NULL);
1252
1253 private:
1254 template <typename RegionTy, typename A1>
1255 RegionTy* getRegion(const A1 a1);
1256
1257 template <typename RegionTy, typename A1>
1258 RegionTy* getSubRegion(const A1 a1, const MemRegion* superRegion);
1259
1260 template <typename RegionTy, typename A1, typename A2>
1261 RegionTy* getRegion(const A1 a1, const A2 a2);
1262
1263 template <typename RegionTy, typename A1, typename A2>
1264 RegionTy* getSubRegion(const A1 a1, const A2 a2,
1265 const MemRegion* superRegion);
1266
1267 template <typename RegionTy, typename A1, typename A2, typename A3>
1268 RegionTy* getSubRegion(const A1 a1, const A2 a2, const A3 a3,
1269 const MemRegion* superRegion);
1270
1271 template <typename REG>
1272 const REG* LazyAllocate(REG*& region);
1273
1274 template <typename REG, typename ARG>
1275 const REG* LazyAllocate(REG*& region, ARG a);
1276 };
1277
1278 //===----------------------------------------------------------------------===//
1279 // Out-of-line member definitions.
1280 //===----------------------------------------------------------------------===//
1281
getContext()1282 inline ASTContext &MemRegion::getContext() const {
1283 return getMemRegionManager()->getContext();
1284 }
1285
1286 } // end GR namespace
1287
1288 } // end clang namespace
1289
1290 //===----------------------------------------------------------------------===//
1291 // Pretty-printing regions.
1292 //===----------------------------------------------------------------------===//
1293
1294 namespace llvm {
1295 static inline raw_ostream &operator<<(raw_ostream &os,
1296 const clang::ento::MemRegion* R) {
1297 R->dumpToStream(os);
1298 return os;
1299 }
1300 } // end llvm namespace
1301
1302 #endif
1303