]> CyberLeo.Net >> Repos - FreeBSD/releng/9.1.git/blob - contrib/llvm/tools/clang/include/clang/StaticAnalyzer/Core/BugReporter/BugReporterVisitor.h
Copy stable/9 to releng/9.1 as part of the 9.1-RELEASE release process.
[FreeBSD/releng/9.1.git] / contrib / llvm / tools / clang / include / clang / StaticAnalyzer / Core / BugReporter / BugReporterVisitor.h
1 //===---  BugReporterVisitor.h - Generate PathDiagnostics -------*- 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 declares BugReporterVisitors, which are used to generate enhanced
11 //  diagnostic traces.
12 //
13 //===----------------------------------------------------------------------===//
14
15 #ifndef LLVM_CLANG_GR_BUGREPORTERVISITOR
16 #define LLVM_CLANG_GR_BUGREPORTERVISITOR
17
18 #include "clang/StaticAnalyzer/Core/PathSensitive/SVals.h"
19 #include "llvm/ADT/FoldingSet.h"
20
21 namespace clang {
22
23 namespace ento {
24
25 class BugReport;
26 class BugReporterContext;
27 class ExplodedNode;
28 class MemRegion;
29 class PathDiagnosticPiece;
30
31 /// \brief BugReporterVisitors are used to add custom diagnostics along a path.
32 ///
33 /// Custom visitors should subclass the BugReporterVisitorImpl class for a
34 /// default implementation of the clone() method.
35 /// (Warning: if you have a deep subclass of BugReporterVisitorImpl, the
36 /// default implementation of clone() will NOT do the right thing, and you
37 /// will have to provide your own implementation.)
38 class BugReporterVisitor : public llvm::FoldingSetNode {
39 public:
40   virtual ~BugReporterVisitor();
41
42   /// \brief Returns a copy of this BugReporter.
43   ///
44   /// Custom BugReporterVisitors should not override this method directly.
45   /// Instead, they should inherit from BugReporterVisitorImpl and provide
46   /// a protected or public copy constructor.
47   ///
48   /// (Warning: if you have a deep subclass of BugReporterVisitorImpl, the
49   /// default implementation of clone() will NOT do the right thing, and you
50   /// will have to provide your own implementation.)
51   virtual BugReporterVisitor *clone() const = 0;
52
53   /// \brief Return a diagnostic piece which should be associated with the
54   /// given node.
55   ///
56   /// The last parameter can be used to register a new visitor with the given
57   /// BugReport while processing a node.
58   virtual PathDiagnosticPiece *VisitNode(const ExplodedNode *N,
59                                          const ExplodedNode *PrevN,
60                                          BugReporterContext &BRC,
61                                          BugReport &BR) = 0;
62
63   /// \brief Provide custom definition for the final diagnostic piece on the
64   /// path - the piece, which is displayed before the path is expanded.
65   ///
66   /// If returns NULL the default implementation will be used.
67   /// Also note that at most one visitor of a BugReport should generate a
68   /// non-NULL end of path diagnostic piece.
69   virtual PathDiagnosticPiece *getEndPath(BugReporterContext &BRC,
70                                           const ExplodedNode *N,
71                                           BugReport &BR);
72
73   virtual void Profile(llvm::FoldingSetNodeID &ID) const = 0;
74
75   /// \brief Generates the default final diagnostic piece.
76   static PathDiagnosticPiece *getDefaultEndPath(BugReporterContext &BRC,
77                                                 const ExplodedNode *N,
78                                                 BugReport &BR);
79
80 };
81
82 /// This class provides a convenience implementation for clone() using the
83 /// Curiously-Recurring Template Pattern. If you are implementing a custom
84 /// BugReporterVisitor, subclass BugReporterVisitorImpl and provide a public
85 /// or protected copy constructor.
86 ///
87 /// (Warning: if you have a deep subclass of BugReporterVisitorImpl, the
88 /// default implementation of clone() will NOT do the right thing, and you
89 /// will have to provide your own implementation.)
90 template <class DERIVED>
91 class BugReporterVisitorImpl : public BugReporterVisitor {
92   virtual BugReporterVisitor *clone() const {
93     return new DERIVED(*static_cast<const DERIVED *>(this));
94   }
95 };
96
97 class FindLastStoreBRVisitor
98   : public BugReporterVisitorImpl<FindLastStoreBRVisitor>
99 {
100   const MemRegion *R;
101   SVal V;
102   bool satisfied;
103   const ExplodedNode *StoreSite;
104
105 public:
106   /// \brief Convenience method to create a visitor given only the MemRegion.
107   /// Returns NULL if the visitor cannot be created. For example, when the
108   /// corresponding value is unknown.
109   static BugReporterVisitor *createVisitorObject(const ExplodedNode *N,
110                                                  const MemRegion *R);
111
112   /// Creates a visitor for every VarDecl inside a Stmt and registers it with
113   /// the BugReport.
114   static void registerStatementVarDecls(BugReport &BR, const Stmt *S);
115
116   FindLastStoreBRVisitor(SVal v, const MemRegion *r)
117   : R(r), V(v), satisfied(false), StoreSite(0) {
118     assert (!V.isUnknown() && "Cannot track unknown value.");
119
120     // TODO: Does it make sense to allow undef values here?
121     // (If not, also see UndefCapturedBlockVarChecker)?
122   }
123
124   void Profile(llvm::FoldingSetNodeID &ID) const;
125
126   PathDiagnosticPiece *VisitNode(const ExplodedNode *N,
127                                  const ExplodedNode *PrevN,
128                                  BugReporterContext &BRC,
129                                  BugReport &BR);
130 };
131
132 class TrackConstraintBRVisitor
133   : public BugReporterVisitorImpl<TrackConstraintBRVisitor>
134 {
135   DefinedSVal Constraint;
136   const bool Assumption;
137   bool isSatisfied;
138
139 public:
140   TrackConstraintBRVisitor(DefinedSVal constraint, bool assumption)
141   : Constraint(constraint), Assumption(assumption), isSatisfied(false) {}
142
143   void Profile(llvm::FoldingSetNodeID &ID) const;
144
145   PathDiagnosticPiece *VisitNode(const ExplodedNode *N,
146                                  const ExplodedNode *PrevN,
147                                  BugReporterContext &BRC,
148                                  BugReport &BR);
149 };
150
151 class NilReceiverBRVisitor
152   : public BugReporterVisitorImpl<NilReceiverBRVisitor>
153 {
154 public:
155   void Profile(llvm::FoldingSetNodeID &ID) const {
156     static int x = 0;
157     ID.AddPointer(&x);
158   }
159
160   PathDiagnosticPiece *VisitNode(const ExplodedNode *N,
161                                  const ExplodedNode *PrevN,
162                                  BugReporterContext &BRC,
163                                  BugReport &BR);
164 };
165
166 /// Visitor that tries to report interesting diagnostics from conditions.
167 class ConditionBRVisitor : public BugReporterVisitorImpl<ConditionBRVisitor> {
168 public:
169   void Profile(llvm::FoldingSetNodeID &ID) const {
170     static int x = 0;
171     ID.AddPointer(&x);
172   }
173
174   
175   virtual PathDiagnosticPiece *VisitNode(const ExplodedNode *N,
176                                          const ExplodedNode *Prev,
177                                          BugReporterContext &BRC,
178                                          BugReport &BR);
179
180   PathDiagnosticPiece *VisitNodeImpl(const ExplodedNode *N,
181                                      const ExplodedNode *Prev,
182                                      BugReporterContext &BRC,
183                                      BugReport &BR);
184   
185   PathDiagnosticPiece *VisitTerminator(const Stmt *Term,
186                                        const ExplodedNode *N,
187                                        const CFGBlock *srcBlk,
188                                        const CFGBlock *dstBlk,
189                                        BugReport &R,
190                                        BugReporterContext &BRC);
191
192   PathDiagnosticPiece *VisitTrueTest(const Expr *Cond,
193                                      bool tookTrue,
194                                      BugReporterContext &BRC,
195                                      BugReport &R,
196                                      const ExplodedNode *N);
197
198   PathDiagnosticPiece *VisitTrueTest(const Expr *Cond,
199                                      const DeclRefExpr *DR,
200                                      const bool tookTrue,
201                                      BugReporterContext &BRC,
202                                      BugReport &R,
203                                      const ExplodedNode *N);
204
205   PathDiagnosticPiece *VisitTrueTest(const Expr *Cond,
206                                      const BinaryOperator *BExpr,
207                                      const bool tookTrue,
208                                      BugReporterContext &BRC,
209                                      BugReport &R,
210                                      const ExplodedNode *N);
211   
212   PathDiagnosticPiece *VisitConditionVariable(StringRef LhsString,
213                                               const Expr *CondVarExpr,
214                                               const bool tookTrue,
215                                               BugReporterContext &BRC,
216                                               BugReport &R,
217                                               const ExplodedNode *N);
218
219   bool patternMatch(const Expr *Ex,
220                     llvm::raw_ostream &Out,
221                     BugReporterContext &BRC,
222                     BugReport &R,
223                     const ExplodedNode *N,
224                     llvm::Optional<bool> &prunable);
225 };
226   
227 namespace bugreporter {
228
229 BugReporterVisitor *getTrackNullOrUndefValueVisitor(const ExplodedNode *N,
230                                                     const Stmt *S,
231                                                     BugReport *R);
232
233 const Stmt *GetDerefExpr(const ExplodedNode *N);
234 const Stmt *GetDenomExpr(const ExplodedNode *N);
235 const Stmt *GetCalleeExpr(const ExplodedNode *N);
236 const Stmt *GetRetValExpr(const ExplodedNode *N);
237
238 } // end namespace clang
239 } // end namespace ento
240 } // end namespace bugreporter
241
242
243 #endif //LLVM_CLANG_GR__BUGREPORTERVISITOR