• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 //===--- CheckerManager.h - Static Analyzer Checker Manager -----*- 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 // Defines the Static Analyzer Checker Manager.
11 //
12 //===----------------------------------------------------------------------===//
13 
14 #ifndef LLVM_CLANG_SA_CORE_CHECKERMANAGER_H
15 #define LLVM_CLANG_SA_CORE_CHECKERMANAGER_H
16 
17 #include "clang/Analysis/ProgramPoint.h"
18 #include "clang/Basic/LangOptions.h"
19 #include "clang/StaticAnalyzer/Core/PathSensitive/Store.h"
20 #include "clang/StaticAnalyzer/Core/AnalyzerOptions.h"
21 #include "llvm/ADT/DenseMap.h"
22 #include "llvm/ADT/FoldingSet.h"
23 #include "llvm/ADT/SmallVector.h"
24 #include <vector>
25 
26 namespace clang {
27   class Decl;
28   class Stmt;
29   class CallExpr;
30 
31 namespace ento {
32   class CheckerBase;
33   class ExprEngine;
34   class AnalysisManager;
35   class BugReporter;
36   class CheckerContext;
37   class SimpleCall;
38   class ObjCMethodCall;
39   class SVal;
40   class ExplodedNode;
41   class ExplodedNodeSet;
42   class ExplodedGraph;
43   class ProgramState;
44   class NodeBuilder;
45   struct NodeBuilderContext;
46   class MemRegion;
47   class SymbolReaper;
48 
49 template <typename T> class CheckerFn;
50 
51 template <typename RET, typename P1, typename P2, typename P3, typename P4,
52           typename P5>
53 class CheckerFn<RET(P1, P2, P3, P4, P5)> {
54   typedef RET (*Func)(void *, P1, P2, P3, P4, P5);
55   Func Fn;
56 public:
57   CheckerBase *Checker;
CheckerFn(CheckerBase * checker,Func fn)58   CheckerFn(CheckerBase *checker, Func fn) : Fn(fn), Checker(checker) { }
operator()59   RET operator()(P1 p1, P2 p2, P3 p3, P4 p4, P5 p5) const {
60     return Fn(Checker, p1, p2, p3, p4, p5);
61   }
62 };
63 
64 template <typename RET, typename P1, typename P2, typename P3, typename P4>
65 class CheckerFn<RET(P1, P2, P3, P4)> {
66   typedef RET (*Func)(void *, P1, P2, P3, P4);
67   Func Fn;
68 public:
69   CheckerBase *Checker;
CheckerFn(CheckerBase * checker,Func fn)70   CheckerFn(CheckerBase *checker, Func fn) : Fn(fn), Checker(checker) { }
operator()71   RET operator()(P1 p1, P2 p2, P3 p3, P4 p4) const {
72     return Fn(Checker, p1, p2, p3, p4);
73   }
74 };
75 
76 template <typename RET, typename P1, typename P2, typename P3>
77 class CheckerFn<RET(P1, P2, P3)> {
78   typedef RET (*Func)(void *, P1, P2, P3);
79   Func Fn;
80 public:
81   CheckerBase *Checker;
CheckerFn(CheckerBase * checker,Func fn)82   CheckerFn(CheckerBase *checker, Func fn) : Fn(fn), Checker(checker) { }
operator()83   RET operator()(P1 p1, P2 p2, P3 p3) const { return Fn(Checker, p1, p2, p3); }
84 };
85 
86 template <typename RET, typename P1, typename P2>
87 class CheckerFn<RET(P1, P2)> {
88   typedef RET (*Func)(void *, P1, P2);
89   Func Fn;
90 public:
91   CheckerBase *Checker;
CheckerFn(CheckerBase * checker,Func fn)92   CheckerFn(CheckerBase *checker, Func fn) : Fn(fn), Checker(checker) { }
operator()93   RET operator()(P1 p1, P2 p2) const { return Fn(Checker, p1, p2); }
94 };
95 
96 template <typename RET, typename P1>
97 class CheckerFn<RET(P1)> {
98   typedef RET (*Func)(void *, P1);
99   Func Fn;
100 public:
101   CheckerBase *Checker;
CheckerFn(CheckerBase * checker,Func fn)102   CheckerFn(CheckerBase *checker, Func fn) : Fn(fn), Checker(checker) { }
operator()103   RET operator()(P1 p1) const { return Fn(Checker, p1); }
104 };
105 
106 template <typename RET>
107 class CheckerFn<RET()> {
108   typedef RET (*Func)(void *);
109   Func Fn;
110 public:
111   CheckerBase *Checker;
CheckerFn(CheckerBase * checker,Func fn)112   CheckerFn(CheckerBase *checker, Func fn) : Fn(fn), Checker(checker) { }
operator()113   RET operator()() const { return Fn(Checker); }
114 };
115 
116 /// \brief Describes the different reasons a pointer escapes
117 /// during analysis.
118 enum PointerEscapeKind {
119   /// A pointer escapes due to binding its value to a location
120   /// that the analyzer cannot track.
121   PSK_EscapeOnBind,
122 
123   /// The pointer has been passed to a function call directly.
124   PSK_DirectEscapeOnCall,
125 
126   /// The pointer has been passed to a function indirectly.
127   /// For example, the pointer is accessible through an
128   /// argument to a function.
129   PSK_IndirectEscapeOnCall,
130 
131   /// The reason for pointer escape is unknown. For example,
132   /// a region containing this pointer is invalidated.
133   PSK_EscapeOther
134 };
135 
136 class CheckerManager {
137   const LangOptions LangOpts;
138   AnalyzerOptionsRef AOptions;
139 public:
CheckerManager(const LangOptions & langOpts,AnalyzerOptionsRef AOptions)140   CheckerManager(const LangOptions &langOpts,
141                  AnalyzerOptionsRef AOptions)
142     : LangOpts(langOpts),
143       AOptions(AOptions) {}
144 
145   ~CheckerManager();
146 
147   bool hasPathSensitiveCheckers() const;
148 
149   void finishedCheckerRegistration();
150 
getLangOpts()151   const LangOptions &getLangOpts() const { return LangOpts; }
getAnalyzerOptions()152   AnalyzerOptions &getAnalyzerOptions() { return *AOptions; }
153 
154   typedef CheckerBase *CheckerRef;
155   typedef const void *CheckerTag;
156   typedef CheckerFn<void ()> CheckerDtor;
157 
158 //===----------------------------------------------------------------------===//
159 // registerChecker
160 //===----------------------------------------------------------------------===//
161 
162   /// \brief Used to register checkers.
163   ///
164   /// \returns a pointer to the checker object.
165   template <typename CHECKER>
registerChecker()166   CHECKER *registerChecker() {
167     CheckerTag tag = getTag<CHECKER>();
168     CheckerRef &ref = CheckerTags[tag];
169     if (ref)
170       return static_cast<CHECKER *>(ref); // already registered.
171 
172     CHECKER *checker = new CHECKER();
173     CheckerDtors.push_back(CheckerDtor(checker, destruct<CHECKER>));
174     CHECKER::_register(checker, *this);
175     ref = checker;
176     return checker;
177   }
178 
179   template <typename CHECKER>
registerChecker(AnalyzerOptions & AOpts)180   CHECKER *registerChecker(AnalyzerOptions &AOpts) {
181     CheckerTag tag = getTag<CHECKER>();
182     CheckerRef &ref = CheckerTags[tag];
183     if (ref)
184       return static_cast<CHECKER *>(ref); // already registered.
185 
186     CHECKER *checker = new CHECKER(AOpts);
187     CheckerDtors.push_back(CheckerDtor(checker, destruct<CHECKER>));
188     CHECKER::_register(checker, *this);
189     ref = checker;
190     return checker;
191   }
192 
193 //===----------------------------------------------------------------------===//
194 // Functions for running checkers for AST traversing..
195 //===----------------------------------------------------------------------===//
196 
197   /// \brief Run checkers handling Decls.
198   void runCheckersOnASTDecl(const Decl *D, AnalysisManager& mgr,
199                             BugReporter &BR);
200 
201   /// \brief Run checkers handling Decls containing a Stmt body.
202   void runCheckersOnASTBody(const Decl *D, AnalysisManager& mgr,
203                             BugReporter &BR);
204 
205 //===----------------------------------------------------------------------===//
206 // Functions for running checkers for path-sensitive checking.
207 //===----------------------------------------------------------------------===//
208 
209   /// \brief Run checkers for pre-visiting Stmts.
210   ///
211   /// The notification is performed for every explored CFGElement, which does
212   /// not include the control flow statements such as IfStmt.
213   ///
214   /// \sa runCheckersForBranchCondition, runCheckersForPostStmt
runCheckersForPreStmt(ExplodedNodeSet & Dst,const ExplodedNodeSet & Src,const Stmt * S,ExprEngine & Eng)215   void runCheckersForPreStmt(ExplodedNodeSet &Dst,
216                              const ExplodedNodeSet &Src,
217                              const Stmt *S,
218                              ExprEngine &Eng) {
219     runCheckersForStmt(/*isPreVisit=*/true, Dst, Src, S, Eng);
220   }
221 
222   /// \brief Run checkers for post-visiting Stmts.
223   ///
224   /// The notification is performed for every explored CFGElement, which does
225   /// not include the control flow statements such as IfStmt.
226   ///
227   /// \sa runCheckersForBranchCondition, runCheckersForPreStmt
228   void runCheckersForPostStmt(ExplodedNodeSet &Dst,
229                               const ExplodedNodeSet &Src,
230                               const Stmt *S,
231                               ExprEngine &Eng,
232                               bool wasInlined = false) {
233     runCheckersForStmt(/*isPreVisit=*/false, Dst, Src, S, Eng, wasInlined);
234   }
235 
236   /// \brief Run checkers for visiting Stmts.
237   void runCheckersForStmt(bool isPreVisit,
238                           ExplodedNodeSet &Dst, const ExplodedNodeSet &Src,
239                           const Stmt *S, ExprEngine &Eng,
240                           bool wasInlined = false);
241 
242   /// \brief Run checkers for pre-visiting obj-c messages.
runCheckersForPreObjCMessage(ExplodedNodeSet & Dst,const ExplodedNodeSet & Src,const ObjCMethodCall & msg,ExprEngine & Eng)243   void runCheckersForPreObjCMessage(ExplodedNodeSet &Dst,
244                                     const ExplodedNodeSet &Src,
245                                     const ObjCMethodCall &msg,
246                                     ExprEngine &Eng) {
247     runCheckersForObjCMessage(/*isPreVisit=*/true, Dst, Src, msg, Eng);
248   }
249 
250   /// \brief Run checkers for post-visiting obj-c messages.
251   void runCheckersForPostObjCMessage(ExplodedNodeSet &Dst,
252                                      const ExplodedNodeSet &Src,
253                                      const ObjCMethodCall &msg,
254                                      ExprEngine &Eng,
255                                      bool wasInlined = false) {
256     runCheckersForObjCMessage(/*isPreVisit=*/false, Dst, Src, msg, Eng,
257                               wasInlined);
258   }
259 
260   /// \brief Run checkers for visiting obj-c messages.
261   void runCheckersForObjCMessage(bool isPreVisit,
262                                  ExplodedNodeSet &Dst,
263                                  const ExplodedNodeSet &Src,
264                                  const ObjCMethodCall &msg, ExprEngine &Eng,
265                                  bool wasInlined = false);
266 
267   /// \brief Run checkers for pre-visiting obj-c messages.
runCheckersForPreCall(ExplodedNodeSet & Dst,const ExplodedNodeSet & Src,const CallEvent & Call,ExprEngine & Eng)268   void runCheckersForPreCall(ExplodedNodeSet &Dst, const ExplodedNodeSet &Src,
269                              const CallEvent &Call, ExprEngine &Eng) {
270     runCheckersForCallEvent(/*isPreVisit=*/true, Dst, Src, Call, Eng);
271   }
272 
273   /// \brief Run checkers for post-visiting obj-c messages.
274   void runCheckersForPostCall(ExplodedNodeSet &Dst, const ExplodedNodeSet &Src,
275                               const CallEvent &Call, ExprEngine &Eng,
276                               bool wasInlined = false) {
277     runCheckersForCallEvent(/*isPreVisit=*/false, Dst, Src, Call, Eng,
278                             wasInlined);
279   }
280 
281   /// \brief Run checkers for visiting obj-c messages.
282   void runCheckersForCallEvent(bool isPreVisit, ExplodedNodeSet &Dst,
283                                const ExplodedNodeSet &Src,
284                                const CallEvent &Call, ExprEngine &Eng,
285                                bool wasInlined = false);
286 
287   /// \brief Run checkers for load/store of a location.
288   void runCheckersForLocation(ExplodedNodeSet &Dst,
289                               const ExplodedNodeSet &Src,
290                               SVal location,
291                               bool isLoad,
292                               const Stmt *NodeEx,
293                               const Stmt *BoundEx,
294                               ExprEngine &Eng);
295 
296   /// \brief Run checkers for binding of a value to a location.
297   void runCheckersForBind(ExplodedNodeSet &Dst,
298                           const ExplodedNodeSet &Src,
299                           SVal location, SVal val,
300                           const Stmt *S, ExprEngine &Eng,
301                           const ProgramPoint &PP);
302 
303   /// \brief Run checkers for end of analysis.
304   void runCheckersForEndAnalysis(ExplodedGraph &G, BugReporter &BR,
305                                  ExprEngine &Eng);
306 
307   /// \brief Run checkers on end of function.
308   void runCheckersForEndFunction(NodeBuilderContext &BC,
309                                  ExplodedNodeSet &Dst,
310                                  ExplodedNode *Pred,
311                                  ExprEngine &Eng);
312 
313   /// \brief Run checkers for branch condition.
314   void runCheckersForBranchCondition(const Stmt *condition,
315                                      ExplodedNodeSet &Dst, ExplodedNode *Pred,
316                                      ExprEngine &Eng);
317 
318   /// \brief Run checkers for live symbols.
319   ///
320   /// Allows modifying SymbolReaper object. For example, checkers can explicitly
321   /// register symbols of interest as live. These symbols will not be marked
322   /// dead and removed.
323   void runCheckersForLiveSymbols(ProgramStateRef state,
324                                  SymbolReaper &SymReaper);
325 
326   /// \brief Run checkers for dead symbols.
327   ///
328   /// Notifies checkers when symbols become dead. For example, this allows
329   /// checkers to aggressively clean up/reduce the checker state and produce
330   /// precise diagnostics.
331   void runCheckersForDeadSymbols(ExplodedNodeSet &Dst,
332                                  const ExplodedNodeSet &Src,
333                                  SymbolReaper &SymReaper, const Stmt *S,
334                                  ExprEngine &Eng,
335                                  ProgramPoint::Kind K);
336 
337   /// \brief True if at least one checker wants to check region changes.
338   bool wantsRegionChangeUpdate(ProgramStateRef state);
339 
340   /// \brief Run checkers for region changes.
341   ///
342   /// This corresponds to the check::RegionChanges callback.
343   /// \param state The current program state.
344   /// \param invalidated A set of all symbols potentially touched by the change.
345   /// \param ExplicitRegions The regions explicitly requested for invalidation.
346   ///   For example, in the case of a function call, these would be arguments.
347   /// \param Regions The transitive closure of accessible regions,
348   ///   i.e. all regions that may have been touched by this change.
349   /// \param Call The call expression wrapper if the regions are invalidated
350   ///   by a call.
351   ProgramStateRef
352   runCheckersForRegionChanges(ProgramStateRef state,
353                               const InvalidatedSymbols *invalidated,
354                               ArrayRef<const MemRegion *> ExplicitRegions,
355                               ArrayRef<const MemRegion *> Regions,
356                               const CallEvent *Call);
357 
358   /// \brief Run checkers when pointers escape.
359   ///
360   /// This notifies the checkers about pointer escape, which occurs whenever
361   /// the analyzer cannot track the symbol any more. For example, as a
362   /// result of assigning a pointer into a global or when it's passed to a
363   /// function call the analyzer cannot model.
364   ///
365   /// \param State The state at the point of escape.
366   /// \param Escaped The list of escaped symbols.
367   /// \param Call The corresponding CallEvent, if the symbols escape as
368   ///        parameters to the given call.
369   /// \param IsConst Specifies if the pointer is const.
370   /// \returns Checkers can modify the state by returning a new one.
371   ProgramStateRef
372   runCheckersForPointerEscape(ProgramStateRef State,
373                               const InvalidatedSymbols &Escaped,
374                               const CallEvent *Call,
375                               PointerEscapeKind Kind,
376                               bool IsConst = false);
377 
378   /// \brief Run checkers for handling assumptions on symbolic values.
379   ProgramStateRef runCheckersForEvalAssume(ProgramStateRef state,
380                                            SVal Cond, bool Assumption);
381 
382   /// \brief Run checkers for evaluating a call.
383   ///
384   /// Warning: Currently, the CallEvent MUST come from a CallExpr!
385   void runCheckersForEvalCall(ExplodedNodeSet &Dst,
386                               const ExplodedNodeSet &Src,
387                               const CallEvent &CE, ExprEngine &Eng);
388 
389   /// \brief Run checkers for the entire Translation Unit.
390   void runCheckersOnEndOfTranslationUnit(const TranslationUnitDecl *TU,
391                                          AnalysisManager &mgr,
392                                          BugReporter &BR);
393 
394   /// \brief Run checkers for debug-printing a ProgramState.
395   ///
396   /// Unlike most other callbacks, any checker can simply implement the virtual
397   /// method CheckerBase::printState if it has custom data to print.
398   /// \param Out The output stream
399   /// \param State The state being printed
400   /// \param NL The preferred representation of a newline.
401   /// \param Sep The preferred separator between different kinds of data.
402   void runCheckersForPrintState(raw_ostream &Out, ProgramStateRef State,
403                                 const char *NL, const char *Sep);
404 
405 //===----------------------------------------------------------------------===//
406 // Internal registration functions for AST traversing.
407 //===----------------------------------------------------------------------===//
408 
409   // Functions used by the registration mechanism, checkers should not touch
410   // these directly.
411 
412   typedef CheckerFn<void (const Decl *, AnalysisManager&, BugReporter &)>
413       CheckDeclFunc;
414 
415   typedef bool (*HandlesDeclFunc)(const Decl *D);
416   void _registerForDecl(CheckDeclFunc checkfn, HandlesDeclFunc isForDeclFn);
417 
418   void _registerForBody(CheckDeclFunc checkfn);
419 
420 //===----------------------------------------------------------------------===//
421 // Internal registration functions for path-sensitive checking.
422 //===----------------------------------------------------------------------===//
423 
424   typedef CheckerFn<void (const Stmt *, CheckerContext &)> CheckStmtFunc;
425 
426   typedef CheckerFn<void (const ObjCMethodCall &, CheckerContext &)>
427       CheckObjCMessageFunc;
428 
429   typedef CheckerFn<void (const CallEvent &, CheckerContext &)>
430       CheckCallFunc;
431 
432   typedef CheckerFn<void (const SVal &location, bool isLoad,
433                           const Stmt *S,
434                           CheckerContext &)>
435       CheckLocationFunc;
436 
437   typedef CheckerFn<void (const SVal &location, const SVal &val,
438                           const Stmt *S, CheckerContext &)>
439       CheckBindFunc;
440 
441   typedef CheckerFn<void (ExplodedGraph &, BugReporter &, ExprEngine &)>
442       CheckEndAnalysisFunc;
443 
444   typedef CheckerFn<void (CheckerContext &)>
445       CheckEndFunctionFunc;
446 
447   typedef CheckerFn<void (const Stmt *, CheckerContext &)>
448       CheckBranchConditionFunc;
449 
450   typedef CheckerFn<void (SymbolReaper &, CheckerContext &)>
451       CheckDeadSymbolsFunc;
452 
453   typedef CheckerFn<void (ProgramStateRef,SymbolReaper &)> CheckLiveSymbolsFunc;
454 
455   typedef CheckerFn<ProgramStateRef (ProgramStateRef,
456                                 const InvalidatedSymbols *symbols,
457                                 ArrayRef<const MemRegion *> ExplicitRegions,
458                                 ArrayRef<const MemRegion *> Regions,
459                                 const CallEvent *Call)>
460       CheckRegionChangesFunc;
461 
462   typedef CheckerFn<bool (ProgramStateRef)> WantsRegionChangeUpdateFunc;
463 
464   typedef CheckerFn<ProgramStateRef (ProgramStateRef,
465                                      const InvalidatedSymbols &Escaped,
466                                      const CallEvent *Call,
467                                      PointerEscapeKind Kind,
468                                      bool IsConst)>
469       CheckPointerEscapeFunc;
470 
471   typedef CheckerFn<ProgramStateRef (ProgramStateRef,
472                                           const SVal &cond, bool assumption)>
473       EvalAssumeFunc;
474 
475   typedef CheckerFn<bool (const CallExpr *, CheckerContext &)>
476       EvalCallFunc;
477 
478   typedef CheckerFn<void (const TranslationUnitDecl *,
479                           AnalysisManager&, BugReporter &)>
480       CheckEndOfTranslationUnit;
481 
482   typedef bool (*HandlesStmtFunc)(const Stmt *D);
483   void _registerForPreStmt(CheckStmtFunc checkfn,
484                            HandlesStmtFunc isForStmtFn);
485   void _registerForPostStmt(CheckStmtFunc checkfn,
486                             HandlesStmtFunc isForStmtFn);
487 
488   void _registerForPreObjCMessage(CheckObjCMessageFunc checkfn);
489   void _registerForPostObjCMessage(CheckObjCMessageFunc checkfn);
490 
491   void _registerForPreCall(CheckCallFunc checkfn);
492   void _registerForPostCall(CheckCallFunc checkfn);
493 
494   void _registerForLocation(CheckLocationFunc checkfn);
495 
496   void _registerForBind(CheckBindFunc checkfn);
497 
498   void _registerForEndAnalysis(CheckEndAnalysisFunc checkfn);
499 
500   void _registerForEndFunction(CheckEndFunctionFunc checkfn);
501 
502   void _registerForBranchCondition(CheckBranchConditionFunc checkfn);
503 
504   void _registerForLiveSymbols(CheckLiveSymbolsFunc checkfn);
505 
506   void _registerForDeadSymbols(CheckDeadSymbolsFunc checkfn);
507 
508   void _registerForRegionChanges(CheckRegionChangesFunc checkfn,
509                                  WantsRegionChangeUpdateFunc wantUpdateFn);
510 
511   void _registerForPointerEscape(CheckPointerEscapeFunc checkfn);
512 
513   void _registerForConstPointerEscape(CheckPointerEscapeFunc checkfn);
514 
515   void _registerForEvalAssume(EvalAssumeFunc checkfn);
516 
517   void _registerForEvalCall(EvalCallFunc checkfn);
518 
519   void _registerForEndOfTranslationUnit(CheckEndOfTranslationUnit checkfn);
520 
521 //===----------------------------------------------------------------------===//
522 // Internal registration functions for events.
523 //===----------------------------------------------------------------------===//
524 
525   typedef void *EventTag;
526   typedef CheckerFn<void (const void *event)> CheckEventFunc;
527 
528   template <typename EVENT>
_registerListenerForEvent(CheckEventFunc checkfn)529   void _registerListenerForEvent(CheckEventFunc checkfn) {
530     EventInfo &info = Events[getTag<EVENT>()];
531     info.Checkers.push_back(checkfn);
532   }
533 
534   template <typename EVENT>
_registerDispatcherForEvent()535   void _registerDispatcherForEvent() {
536     EventInfo &info = Events[getTag<EVENT>()];
537     info.HasDispatcher = true;
538   }
539 
540   template <typename EVENT>
_dispatchEvent(const EVENT & event)541   void _dispatchEvent(const EVENT &event) const {
542     EventsTy::const_iterator I = Events.find(getTag<EVENT>());
543     if (I == Events.end())
544       return;
545     const EventInfo &info = I->second;
546     for (unsigned i = 0, e = info.Checkers.size(); i != e; ++i)
547       info.Checkers[i](&event);
548   }
549 
550 //===----------------------------------------------------------------------===//
551 // Implementation details.
552 //===----------------------------------------------------------------------===//
553 
554 private:
555   template <typename CHECKER>
destruct(void * obj)556   static void destruct(void *obj) { delete static_cast<CHECKER *>(obj); }
557 
558   template <typename T>
getTag()559   static void *getTag() { static int tag; return &tag; }
560 
561   llvm::DenseMap<CheckerTag, CheckerRef> CheckerTags;
562 
563   std::vector<CheckerDtor> CheckerDtors;
564 
565   struct DeclCheckerInfo {
566     CheckDeclFunc CheckFn;
567     HandlesDeclFunc IsForDeclFn;
568   };
569   std::vector<DeclCheckerInfo> DeclCheckers;
570 
571   std::vector<CheckDeclFunc> BodyCheckers;
572 
573   typedef SmallVector<CheckDeclFunc, 4> CachedDeclCheckers;
574   typedef llvm::DenseMap<unsigned, CachedDeclCheckers> CachedDeclCheckersMapTy;
575   CachedDeclCheckersMapTy CachedDeclCheckersMap;
576 
577   struct StmtCheckerInfo {
578     CheckStmtFunc CheckFn;
579     HandlesStmtFunc IsForStmtFn;
580     bool IsPreVisit;
581   };
582   std::vector<StmtCheckerInfo> StmtCheckers;
583 
584   struct CachedStmtCheckersKey {
585     unsigned StmtKind;
586     bool IsPreVisit;
587 
CachedStmtCheckersKeyCachedStmtCheckersKey588     CachedStmtCheckersKey() : StmtKind(0), IsPreVisit(0) { }
CachedStmtCheckersKeyCachedStmtCheckersKey589     CachedStmtCheckersKey(unsigned stmtKind, bool isPreVisit)
590       : StmtKind(stmtKind), IsPreVisit(isPreVisit) { }
591 
getSentinelCachedStmtCheckersKey592     static CachedStmtCheckersKey getSentinel() {
593       return CachedStmtCheckersKey(~0U, 0);
594     }
getHashValueCachedStmtCheckersKey595     unsigned getHashValue() const {
596       llvm::FoldingSetNodeID ID;
597       ID.AddInteger(StmtKind);
598       ID.AddBoolean(IsPreVisit);
599       return ID.ComputeHash();
600     }
601     bool operator==(const CachedStmtCheckersKey &RHS) const {
602       return StmtKind == RHS.StmtKind && IsPreVisit == RHS.IsPreVisit;
603     }
604   };
605   friend struct llvm::DenseMapInfo<CachedStmtCheckersKey>;
606 
607   typedef SmallVector<CheckStmtFunc, 4> CachedStmtCheckers;
608   typedef llvm::DenseMap<CachedStmtCheckersKey, CachedStmtCheckers>
609       CachedStmtCheckersMapTy;
610   CachedStmtCheckersMapTy CachedStmtCheckersMap;
611 
612   CachedStmtCheckers *getCachedStmtCheckersFor(const Stmt *S, bool isPreVisit);
613 
614   std::vector<CheckObjCMessageFunc> PreObjCMessageCheckers;
615   std::vector<CheckObjCMessageFunc> PostObjCMessageCheckers;
616 
617   std::vector<CheckCallFunc> PreCallCheckers;
618   std::vector<CheckCallFunc> PostCallCheckers;
619 
620   std::vector<CheckLocationFunc> LocationCheckers;
621 
622   std::vector<CheckBindFunc> BindCheckers;
623 
624   std::vector<CheckEndAnalysisFunc> EndAnalysisCheckers;
625 
626   std::vector<CheckEndFunctionFunc> EndFunctionCheckers;
627 
628   std::vector<CheckBranchConditionFunc> BranchConditionCheckers;
629 
630   std::vector<CheckLiveSymbolsFunc> LiveSymbolsCheckers;
631 
632   std::vector<CheckDeadSymbolsFunc> DeadSymbolsCheckers;
633 
634   struct RegionChangesCheckerInfo {
635     CheckRegionChangesFunc CheckFn;
636     WantsRegionChangeUpdateFunc WantUpdateFn;
637   };
638   std::vector<RegionChangesCheckerInfo> RegionChangesCheckers;
639 
640   std::vector<CheckPointerEscapeFunc> PointerEscapeCheckers;
641 
642   std::vector<EvalAssumeFunc> EvalAssumeCheckers;
643 
644   std::vector<EvalCallFunc> EvalCallCheckers;
645 
646   std::vector<CheckEndOfTranslationUnit> EndOfTranslationUnitCheckers;
647 
648   struct EventInfo {
649     SmallVector<CheckEventFunc, 4> Checkers;
650     bool HasDispatcher;
651     EventInfo() : HasDispatcher(false) { }
652   };
653 
654   typedef llvm::DenseMap<EventTag, EventInfo> EventsTy;
655   EventsTy Events;
656 };
657 
658 } // end ento namespace
659 
660 } // end clang namespace
661 
662 namespace llvm {
663   /// Define DenseMapInfo so that CachedStmtCheckersKey can be used as key
664   /// in DenseMap and DenseSets.
665   template <>
666   struct DenseMapInfo<clang::ento::CheckerManager::CachedStmtCheckersKey> {
667     static inline clang::ento::CheckerManager::CachedStmtCheckersKey
668         getEmptyKey() {
669       return clang::ento::CheckerManager::CachedStmtCheckersKey();
670     }
671     static inline clang::ento::CheckerManager::CachedStmtCheckersKey
672         getTombstoneKey() {
673       return clang::ento::CheckerManager::CachedStmtCheckersKey::getSentinel();
674     }
675 
676     static unsigned
677         getHashValue(clang::ento::CheckerManager::CachedStmtCheckersKey S) {
678       return S.getHashValue();
679     }
680 
681     static bool isEqual(clang::ento::CheckerManager::CachedStmtCheckersKey LHS,
682                        clang::ento::CheckerManager::CachedStmtCheckersKey RHS) {
683       return LHS == RHS;
684     }
685   };
686 } // end namespace llvm
687 
688 #endif
689