AnalysisContext.h revision b1b5daf30d2597e066936772bd206500232d7d65
1326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek//=== AnalysisContext.h - Analysis context for Path Sens analysis --*- C++ -*-//
2326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek//
3326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek//                     The LLVM Compiler Infrastructure
4326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek//
5326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek// This file is distributed under the University of Illinois Open Source
6326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek// License. See LICENSE.TXT for details.
7326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek//
8326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek//===----------------------------------------------------------------------===//
9326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek//
10326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek// This file defines AnalysisContext, a class that manages the analysis context
11326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek// data for path sensitive analysis.
12326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek//
13326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek//===----------------------------------------------------------------------===//
14326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
15326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek#ifndef LLVM_CLANG_ANALYSIS_ANALYSISCONTEXT_H
16326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek#define LLVM_CLANG_ANALYSIS_ANALYSISCONTEXT_H
17326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
18326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek#include "clang/AST/Decl.h"
19892697dd2287caf7c29aaaa82909b0e90b8b63feTed Kremenek#include "clang/AST/Expr.h"
20b8ad5ee345fa1fdd1fa9253f2d01f69becc88a04Ted Kremenek#include "clang/Analysis/CFG.h"
21326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek#include "llvm/ADT/OwningPtr.h"
22bc5cb8a5fe2b88f917d47ceb58b53696a121e57eTed Kremenek#include "llvm/ADT/IntrusiveRefCntPtr.h"
23326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek#include "llvm/ADT/FoldingSet.h"
24326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek#include "llvm/ADT/PointerUnion.h"
25326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek#include "llvm/ADT/DenseMap.h"
26326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek#include "llvm/Support/Allocator.h"
27326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
28326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremeneknamespace clang {
29326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
30326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenekclass Decl;
31326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenekclass Stmt;
32af13d5b25b360e698cc1cf1055ad7d14e008e505Ted Kremenekclass CFGReverseBlockReachabilityAnalysis;
33283a358aecb75e30fcd486f2206f6c03c5e7f11dTed Kremenekclass CFGStmtMap;
34326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenekclass LiveVariables;
35a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenekclass ManagedAnalysis;
36326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenekclass ParentMap;
37db34ab70961ca4b24b600eb47053d7af304659f5Tom Careclass PseudoConstantAnalysis;
38326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenekclass ImplicitParamDecl;
39326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenekclass LocationContextManager;
40326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenekclass StackFrameContext;
41b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenekclass AnalysisContextManager;
42b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenekclass LocationContext;
43b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek
44c6238d2786cfd961b94580b3d3675a1b3ff0721cZhongxing Xunamespace idx { class TranslationUnit; }
45c6238d2786cfd961b94580b3d3675a1b3ff0721cZhongxing Xu
46a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek/// The base class of a hierarchy of objects representing analyses tied
47a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek/// to AnalysisContext.
48a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenekclass ManagedAnalysis {
49a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenekprotected:
50a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek  ManagedAnalysis() {}
51a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenekpublic:
52a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek  virtual ~ManagedAnalysis();
53a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek
54a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek  // Subclasses need to implement:
55a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek  //
56a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek  //  static const void *getTag();
57a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek  //
58a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek  // Which returns a fixed pointer address to distinguish classes of
59a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek  // analysis objects.  They also need to implement:
60a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek  //
61a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek  //  static [Derived*] create(AnalysisContext &Ctx);
62a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek  //
63a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek  // which creates the analysis object given an AnalysisContext.
64a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek};
65a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek
66b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek
67326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek/// AnalysisContext contains the context data for the function or method under
68326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek/// analysis.
69326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenekclass AnalysisContext {
70b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek  /// Backpoint to the AnalysisManager object that created this AnalysisContext.
71b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek  /// This may be null.
72b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek  AnalysisContextManager *Manager;
73b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek
74326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  const Decl *D;
75326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
76c6238d2786cfd961b94580b3d3675a1b3ff0721cZhongxing Xu  // TranslationUnit is NULL if we don't have multiple translation units.
772ce43c8f43254a9edea53a20dc0e69195bc82ae0Zhongxing Xu  idx::TranslationUnit *TU;
78c6238d2786cfd961b94580b3d3675a1b3ff0721cZhongxing Xu
79b8ad5ee345fa1fdd1fa9253f2d01f69becc88a04Ted Kremenek  llvm::OwningPtr<CFG> cfg, completeCFG;
80b8ad5ee345fa1fdd1fa9253f2d01f69becc88a04Ted Kremenek  llvm::OwningPtr<CFGStmtMap> cfgStmtMap;
81b8ad5ee345fa1fdd1fa9253f2d01f69becc88a04Ted Kremenek
82b8ad5ee345fa1fdd1fa9253f2d01f69becc88a04Ted Kremenek  CFG::BuildOptions cfgBuildOptions;
83b8ad5ee345fa1fdd1fa9253f2d01f69becc88a04Ted Kremenek  CFG::BuildOptions::ForcedBlkExprs *forcedBlkExprs;
84b8ad5ee345fa1fdd1fa9253f2d01f69becc88a04Ted Kremenek
85ad5a894df1841698c824381b414630799adc26caTed Kremenek  bool builtCFG, builtCompleteCFG;
86b8ad5ee345fa1fdd1fa9253f2d01f69becc88a04Ted Kremenek
87b8ad5ee345fa1fdd1fa9253f2d01f69becc88a04Ted Kremenek  llvm::OwningPtr<LiveVariables> liveness;
88b8ad5ee345fa1fdd1fa9253f2d01f69becc88a04Ted Kremenek  llvm::OwningPtr<LiveVariables> relaxedLiveness;
89b8ad5ee345fa1fdd1fa9253f2d01f69becc88a04Ted Kremenek  llvm::OwningPtr<ParentMap> PM;
90b8ad5ee345fa1fdd1fa9253f2d01f69becc88a04Ted Kremenek  llvm::OwningPtr<PseudoConstantAnalysis> PCA;
91af13d5b25b360e698cc1cf1055ad7d14e008e505Ted Kremenek  llvm::OwningPtr<CFGReverseBlockReachabilityAnalysis> CFA;
92b8ad5ee345fa1fdd1fa9253f2d01f69becc88a04Ted Kremenek
93326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  llvm::BumpPtrAllocator A;
94b8ad5ee345fa1fdd1fa9253f2d01f69becc88a04Ted Kremenek
95b8ad5ee345fa1fdd1fa9253f2d01f69becc88a04Ted Kremenek  // FIXME: remove.
96b8ad5ee345fa1fdd1fa9253f2d01f69becc88a04Ted Kremenek  llvm::DenseMap<const BlockDecl*,void*> *ReferencedBlockVars;
97b8ad5ee345fa1fdd1fa9253f2d01f69becc88a04Ted Kremenek
98a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek  void *ManagedAnalyses;
99a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek
100326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenekpublic:
101b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek  AnalysisContext(AnalysisContextManager *Mgr,
102b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek                  const Decl *D,
103b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek                  idx::TranslationUnit *TU);
104bc5cb8a5fe2b88f917d47ceb58b53696a121e57eTed Kremenek
105b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek  AnalysisContext(AnalysisContextManager *Mgr,
106b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek                  const Decl *D,
107b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek                  idx::TranslationUnit *TU,
108b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek                  const CFG::BuildOptions &BuildOptions);
109326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
110326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  ~AnalysisContext();
111326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
112326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  ASTContext &getASTContext() { return D->getASTContext(); }
113c6238d2786cfd961b94580b3d3675a1b3ff0721cZhongxing Xu  const Decl *getDecl() const { return D; }
114c6238d2786cfd961b94580b3d3675a1b3ff0721cZhongxing Xu
1152ce43c8f43254a9edea53a20dc0e69195bc82ae0Zhongxing Xu  idx::TranslationUnit *getTranslationUnit() const { return TU; }
116c6238d2786cfd961b94580b3d3675a1b3ff0721cZhongxing Xu
11774fb1a493cf5d2dd0fb51a4eadf74e85e10a3457Ted Kremenek  /// Return the build options used to construct the CFG.
11874fb1a493cf5d2dd0fb51a4eadf74e85e10a3457Ted Kremenek  CFG::BuildOptions &getCFGBuildOptions() {
11974fb1a493cf5d2dd0fb51a4eadf74e85e10a3457Ted Kremenek    return cfgBuildOptions;
12074fb1a493cf5d2dd0fb51a4eadf74e85e10a3457Ted Kremenek  }
12174fb1a493cf5d2dd0fb51a4eadf74e85e10a3457Ted Kremenek
12274fb1a493cf5d2dd0fb51a4eadf74e85e10a3457Ted Kremenek  const CFG::BuildOptions &getCFGBuildOptions() const {
12374fb1a493cf5d2dd0fb51a4eadf74e85e10a3457Ted Kremenek    return cfgBuildOptions;
12474fb1a493cf5d2dd0fb51a4eadf74e85e10a3457Ted Kremenek  }
12574fb1a493cf5d2dd0fb51a4eadf74e85e10a3457Ted Kremenek
126326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  /// getAddEHEdges - Return true iff we are adding exceptional edges from
127326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  /// callExprs.  If this is false, then try/catch statements and blocks
128326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  /// reachable from them can appear to be dead in the CFG, analysis passes must
129326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  /// cope with that.
130b8ad5ee345fa1fdd1fa9253f2d01f69becc88a04Ted Kremenek  bool getAddEHEdges() const { return cfgBuildOptions.AddEHEdges; }
131b8ad5ee345fa1fdd1fa9253f2d01f69becc88a04Ted Kremenek  bool getUseUnoptimizedCFG() const {
132bc5cb8a5fe2b88f917d47ceb58b53696a121e57eTed Kremenek      return !cfgBuildOptions.PruneTriviallyFalseEdges;
133b8ad5ee345fa1fdd1fa9253f2d01f69becc88a04Ted Kremenek  }
134b8ad5ee345fa1fdd1fa9253f2d01f69becc88a04Ted Kremenek  bool getAddImplicitDtors() const { return cfgBuildOptions.AddImplicitDtors; }
135b8ad5ee345fa1fdd1fa9253f2d01f69becc88a04Ted Kremenek  bool getAddInitializers() const { return cfgBuildOptions.AddInitializers; }
1369b823e8e1ccb8a2cb49923bad22a80ca96f41f92Ted Kremenek
1370d28d360b5559abda755e50b855ba5e59727d9cdTed Kremenek  void registerForcedBlockExpression(const Stmt *stmt);
1380d28d360b5559abda755e50b855ba5e59727d9cdTed Kremenek  const CFGBlock *getBlockForRegisteredExpression(const Stmt *stmt);
1390d28d360b5559abda755e50b855ba5e59727d9cdTed Kremenek
140a2d7e6511a8767dc67381c210601b895a8ebae39Anna Zaks  Stmt *getBody() const;
141326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  CFG *getCFG();
142283a358aecb75e30fcd486f2206f6c03c5e7f11dTed Kremenek
143283a358aecb75e30fcd486f2206f6c03c5e7f11dTed Kremenek  CFGStmtMap *getCFGStmtMap();
14404eeba43040969c05cfcb563195ef5b199297b62Anders Carlsson
145af13d5b25b360e698cc1cf1055ad7d14e008e505Ted Kremenek  CFGReverseBlockReachabilityAnalysis *getCFGReachablityAnalysis();
14642461eecee98fff3671b3c14ce10f1a9e18cc95cTed Kremenek
147ad5a894df1841698c824381b414630799adc26caTed Kremenek  /// Return a version of the CFG without any edges pruned.
148ad5a894df1841698c824381b414630799adc26caTed Kremenek  CFG *getUnoptimizedCFG();
149ad5a894df1841698c824381b414630799adc26caTed Kremenek
15004eeba43040969c05cfcb563195ef5b199297b62Anders Carlsson  void dumpCFG();
15104eeba43040969c05cfcb563195ef5b199297b62Anders Carlsson
1525d98994c7749312a43ce6adf45537979a98e7afdChandler Carruth  /// \brief Returns true if we have built a CFG for this analysis context.
1535d98994c7749312a43ce6adf45537979a98e7afdChandler Carruth  /// Note that this doesn't correspond to whether or not a valid CFG exists, it
1545d98994c7749312a43ce6adf45537979a98e7afdChandler Carruth  /// corresponds to whether we *attempted* to build one.
1555d98994c7749312a43ce6adf45537979a98e7afdChandler Carruth  bool isCFGBuilt() const { return builtCFG; }
1565d98994c7749312a43ce6adf45537979a98e7afdChandler Carruth
157326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  ParentMap &getParentMap();
158db34ab70961ca4b24b600eb47053d7af304659f5Tom Care  PseudoConstantAnalysis *getPseudoConstantAnalysis();
159326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
160326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  typedef const VarDecl * const * referenced_decls_iterator;
161326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
162326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  std::pair<referenced_decls_iterator, referenced_decls_iterator>
163326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek    getReferencedBlockVars(const BlockDecl *BD);
164d064fdc4b7b64ca55b40b70490c79d6f569df78eTed Kremenek
165326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  /// Return the ImplicitParamDecl* associated with 'self' if this
166326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  /// AnalysisContext wraps an ObjCMethodDecl.  Returns NULL otherwise.
167326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  const ImplicitParamDecl *getSelfDecl() const;
168a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek
169b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek  const StackFrameContext *getStackFrame(LocationContext const *Parent,
170b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek                                         const Stmt *S,
171b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek                                         const CFGBlock *Blk,
172b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek                                         unsigned Idx);
173b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek
174a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek  /// Return the specified analysis object, lazily running the analysis if
175a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek  /// necessary.  Return NULL if the analysis could not run.
176a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek  template <typename T>
177a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek  T *getAnalysis() {
178a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek    const void *tag = T::getTag();
179a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek    ManagedAnalysis *&data = getAnalysisImpl(tag);
180a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek    if (!data) {
181a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek      data = T::create(*this);
182a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek    }
183a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek    return static_cast<T*>(data);
184a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek  }
185a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenekprivate:
186a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek  ManagedAnalysis *&getAnalysisImpl(const void* tag);
187bc5cb8a5fe2b88f917d47ceb58b53696a121e57eTed Kremenek
188b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek  LocationContextManager &getLocationContextManager();
189326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek};
190326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
191326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenekclass LocationContext : public llvm::FoldingSetNode {
192326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenekpublic:
193326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  enum ContextKind { StackFrame, Scope, Block };
194326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
195326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenekprivate:
196326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  ContextKind Kind;
197a02d893f15d4663bdba3bd92ade10070bf0510e4Zhongxing Xu
198a02d893f15d4663bdba3bd92ade10070bf0510e4Zhongxing Xu  // AnalysisContext can't be const since some methods may modify its member.
199326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  AnalysisContext *Ctx;
200a02d893f15d4663bdba3bd92ade10070bf0510e4Zhongxing Xu
201326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  const LocationContext *Parent;
202326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
203326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenekprotected:
204326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  LocationContext(ContextKind k, AnalysisContext *ctx,
205326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek                  const LocationContext *parent)
206326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek    : Kind(k), Ctx(ctx), Parent(parent) {}
207326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
208326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenekpublic:
209326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  virtual ~LocationContext();
210d064fdc4b7b64ca55b40b70490c79d6f569df78eTed Kremenek
211326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  ContextKind getKind() const { return Kind; }
212326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
213326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  AnalysisContext *getAnalysisContext() const { return Ctx; }
214326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
2152ce43c8f43254a9edea53a20dc0e69195bc82ae0Zhongxing Xu  idx::TranslationUnit *getTranslationUnit() const {
216c6238d2786cfd961b94580b3d3675a1b3ff0721cZhongxing Xu    return Ctx->getTranslationUnit();
217c6238d2786cfd961b94580b3d3675a1b3ff0721cZhongxing Xu  }
218c6238d2786cfd961b94580b3d3675a1b3ff0721cZhongxing Xu
219326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  const LocationContext *getParent() const { return Parent; }
220326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
2218ddf7cead8a67342a4584a203e0bf736b7efedbeZhongxing Xu  bool isParentOf(const LocationContext *LC) const;
2228ddf7cead8a67342a4584a203e0bf736b7efedbeZhongxing Xu
223326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  const Decl *getDecl() const { return getAnalysisContext()->getDecl(); }
224326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
225326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  CFG *getCFG() const { return getAnalysisContext()->getCFG(); }
226326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
227a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek  template <typename T>
228a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek  T *getAnalysis() const {
229a5937bbfd19e61d651a58b0f0ffeef68457902a5Ted Kremenek    return getAnalysisContext()->getAnalysis<T>();
230326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  }
231326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
232d064fdc4b7b64ca55b40b70490c79d6f569df78eTed Kremenek  ParentMap &getParentMap() const {
233326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek    return getAnalysisContext()->getParentMap();
234326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  }
235326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
236326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  const ImplicitParamDecl *getSelfDecl() const {
237326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek    return Ctx->getSelfDecl();
238326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  }
239d064fdc4b7b64ca55b40b70490c79d6f569df78eTed Kremenek
240326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  const StackFrameContext *getCurrentStackFrame() const;
241326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  const StackFrameContext *
242326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek    getStackFrameForDeclContext(const DeclContext *DC) const;
243326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
244326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  virtual void Profile(llvm::FoldingSetNodeID &ID) = 0;
245326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
246326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  static bool classof(const LocationContext*) { return true; }
247326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
248326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenekpublic:
249326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  static void ProfileCommon(llvm::FoldingSetNodeID &ID,
250326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek                            ContextKind ck,
251326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek                            AnalysisContext *ctx,
252326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek                            const LocationContext *parent,
2539c378f705405d37f49795d5e915989de774fe11fTed Kremenek                            const void *data);
254326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek};
255326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
256326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenekclass StackFrameContext : public LocationContext {
25726c9cb55cb96643c0759c08d037c16c309864087Zhongxing Xu  // The callsite where this stack frame is established.
258892697dd2287caf7c29aaaa82909b0e90b8b63feTed Kremenek  const Stmt *CallSite;
259326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
260326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  // The parent block of the callsite.
261326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  const CFGBlock *Block;
262326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
263326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  // The index of the callsite in the CFGBlock.
264326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  unsigned Index;
265326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
266326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  friend class LocationContextManager;
267326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  StackFrameContext(AnalysisContext *ctx, const LocationContext *parent,
268892697dd2287caf7c29aaaa82909b0e90b8b63feTed Kremenek                    const Stmt *s, const CFGBlock *blk,
269d706434b0231c76fd9acf30060646a7aa8f69aefZhongxing Xu                    unsigned idx)
270892697dd2287caf7c29aaaa82909b0e90b8b63feTed Kremenek    : LocationContext(StackFrame, ctx, parent), CallSite(s),
271d706434b0231c76fd9acf30060646a7aa8f69aefZhongxing Xu      Block(blk), Index(idx) {}
272326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
273326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenekpublic:
274326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  ~StackFrameContext() {}
275326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
276892697dd2287caf7c29aaaa82909b0e90b8b63feTed Kremenek  const Stmt *getCallSite() const { return CallSite; }
277d706434b0231c76fd9acf30060646a7aa8f69aefZhongxing Xu
278326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  const CFGBlock *getCallSiteBlock() const { return Block; }
279326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
280326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  unsigned getIndex() const { return Index; }
281326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
282326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  void Profile(llvm::FoldingSetNodeID &ID);
283d064fdc4b7b64ca55b40b70490c79d6f569df78eTed Kremenek
284326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  static void Profile(llvm::FoldingSetNodeID &ID, AnalysisContext *ctx,
285d064fdc4b7b64ca55b40b70490c79d6f569df78eTed Kremenek                      const LocationContext *parent, const Stmt *s,
286892697dd2287caf7c29aaaa82909b0e90b8b63feTed Kremenek                      const CFGBlock *blk, unsigned idx) {
287326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek    ProfileCommon(ID, StackFrame, ctx, parent, s);
288326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek    ID.AddPointer(blk);
289326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek    ID.AddInteger(idx);
290326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  }
291326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
2929c378f705405d37f49795d5e915989de774fe11fTed Kremenek  static bool classof(const LocationContext *Ctx) {
293326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek    return Ctx->getKind() == StackFrame;
294326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  }
295326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek};
296326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
297326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenekclass ScopeContext : public LocationContext {
298326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  const Stmt *Enter;
299d064fdc4b7b64ca55b40b70490c79d6f569df78eTed Kremenek
300326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  friend class LocationContextManager;
301326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  ScopeContext(AnalysisContext *ctx, const LocationContext *parent,
302326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek               const Stmt *s)
303326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek    : LocationContext(Scope, ctx, parent), Enter(s) {}
304326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
305326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenekpublic:
306326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  ~ScopeContext() {}
307326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
308326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  void Profile(llvm::FoldingSetNodeID &ID);
309326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
310326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  static void Profile(llvm::FoldingSetNodeID &ID, AnalysisContext *ctx,
311326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek                      const LocationContext *parent, const Stmt *s) {
312326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek    ProfileCommon(ID, Scope, ctx, parent, s);
313326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  }
314326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
3159c378f705405d37f49795d5e915989de774fe11fTed Kremenek  static bool classof(const LocationContext *Ctx) {
316326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek    return Ctx->getKind() == Scope;
317326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  }
318326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek};
319326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
320326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenekclass BlockInvocationContext : public LocationContext {
321326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  // FIXME: Add back context-sensivity (we don't want libAnalysis to know
322326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  //  about MemRegion).
323326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  const BlockDecl *BD;
324326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
325326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  friend class LocationContextManager;
326326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
327326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  BlockInvocationContext(AnalysisContext *ctx, const LocationContext *parent,
328326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek                         const BlockDecl *bd)
329326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek    : LocationContext(Block, ctx, parent), BD(bd) {}
330326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
331326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenekpublic:
332326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  ~BlockInvocationContext() {}
333326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
334326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  const BlockDecl *getBlockDecl() const { return BD; }
335326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
336326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  void Profile(llvm::FoldingSetNodeID &ID);
337326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
338326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  static void Profile(llvm::FoldingSetNodeID &ID, AnalysisContext *ctx,
339326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek                      const LocationContext *parent, const BlockDecl *bd) {
340326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek    ProfileCommon(ID, Block, ctx, parent, bd);
341326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  }
342d064fdc4b7b64ca55b40b70490c79d6f569df78eTed Kremenek
3439c378f705405d37f49795d5e915989de774fe11fTed Kremenek  static bool classof(const LocationContext *Ctx) {
344326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek    return Ctx->getKind() == Block;
345326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  }
346326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek};
347326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
348326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenekclass LocationContextManager {
349326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  llvm::FoldingSet<LocationContext> Contexts;
350326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenekpublic:
351326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  ~LocationContextManager();
352d064fdc4b7b64ca55b40b70490c79d6f569df78eTed Kremenek
353326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  const StackFrameContext *getStackFrame(AnalysisContext *ctx,
354326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek                                         const LocationContext *parent,
355892697dd2287caf7c29aaaa82909b0e90b8b63feTed Kremenek                                         const Stmt *s,
356d706434b0231c76fd9acf30060646a7aa8f69aefZhongxing Xu                                         const CFGBlock *blk, unsigned idx);
357326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
358326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  const ScopeContext *getScope(AnalysisContext *ctx,
359326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek                               const LocationContext *parent,
360326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek                               const Stmt *s);
361d064fdc4b7b64ca55b40b70490c79d6f569df78eTed Kremenek
362326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  /// Discard all previously created LocationContext objects.
363326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  void clear();
364326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenekprivate:
365326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  template <typename LOC, typename DATA>
366326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek  const LOC *getLocationContext(AnalysisContext *ctx,
367326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek                                const LocationContext *parent,
368326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek                                const DATA *d);
369326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek};
370326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek
371b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenekclass AnalysisContextManager {
372b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek  typedef llvm::DenseMap<const Decl*, AnalysisContext*> ContextMap;
373b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek
374b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek  ContextMap Contexts;
375b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek  LocationContextManager LocContexts;
376b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek  CFG::BuildOptions cfgBuildOptions;
377b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek
378b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenekpublic:
379b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek  AnalysisContextManager(bool useUnoptimizedCFG = false,
380b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek                         bool addImplicitDtors = false,
381b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek                         bool addInitializers = false);
382b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek
383b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek  ~AnalysisContextManager();
384b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek
385b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek  AnalysisContext *getContext(const Decl *D, idx::TranslationUnit *TU = 0);
386b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek
387b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek  bool getUseUnoptimizedCFG() const {
388b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek    return !cfgBuildOptions.PruneTriviallyFalseEdges;
389b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek  }
390b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek
391b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek  CFG::BuildOptions &getCFGBuildOptions() {
392b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek    return cfgBuildOptions;
393b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek  }
394b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek
395b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek  const StackFrameContext *getStackFrame(AnalysisContext *Ctx,
396b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek                                         LocationContext const *Parent,
397b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek                                         const Stmt *S,
398b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek                                         const CFGBlock *Blk,
399b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek                                         unsigned Idx) {
400b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek    return LocContexts.getStackFrame(Ctx, Parent, S, Blk, Idx);
401b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek  }
402b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek
403b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek  // Get the top level stack frame.
404b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek  const StackFrameContext *getStackFrame(Decl const *D,
405b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek                                         idx::TranslationUnit *TU) {
406b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek    return LocContexts.getStackFrame(getContext(D, TU), 0, 0, 0, 0);
407b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek  }
408b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek
409b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek  // Get a stack frame with parent.
410b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek  StackFrameContext const *getStackFrame(const Decl *D,
411b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek                                         LocationContext const *Parent,
412b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek                                         const Stmt *S,
413b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek                                         const CFGBlock *Blk,
414b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek                                         unsigned Idx) {
415b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek    return LocContexts.getStackFrame(getContext(D), Parent, S, Blk, Idx);
416b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek  }
417b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek
418b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek
419b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek  /// Discard all previously created AnalysisContexts.
420b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek  void clear();
421b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek
422b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenekprivate:
423b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek  friend class AnalysisContext;
424b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek
425b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek  LocationContextManager &getLocationContextManager() {
426b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek    return LocContexts;
427b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek  }
428b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek};
429b1b5daf30d2597e066936772bd206500232d7d65Ted Kremenek
430326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek} // end clang namespace
431326be568e2cb04285c84e6e26a3e6b3822607361Ted Kremenek#endif
432