aboutsummaryrefslogtreecommitdiff
path: root/include/clang/Analysis/PathSensitive/Checker.h
blob: f70b6129c4f3bc7228cfb3f620234fb12bb5e714 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
//== Checker.h - Abstract interface for checkers -----------------*- C++ -*--=//
//
//                     The LLVM Compiler Infrastructure
//
// This file is distributed under the University of Illinois Open Source
// License. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
//
//  This file defines Checker and CheckerVisitor, classes used for creating
//  domain-specific checks.
//
//===----------------------------------------------------------------------===//

#ifndef LLVM_CLANG_ANALYSIS_CHECKER
#define LLVM_CLANG_ANALYSIS_CHECKER
#include "clang/Analysis/Support/SaveAndRestore.h"
#include "clang/Analysis/PathSensitive/GRCoreEngine.h"
#include "clang/Analysis/PathSensitive/GRState.h"
#include "clang/AST/ExprCXX.h"
#include "clang/AST/ExprObjC.h"
#include "clang/AST/StmtCXX.h"
#include "clang/AST/StmtObjC.h"

//===----------------------------------------------------------------------===//
// Checker interface.
//===----------------------------------------------------------------------===//

namespace clang {
  class GRExprEngine;

class CheckerContext {
  ExplodedNodeSet<GRState> &Dst;
  GRStmtNodeBuilder<GRState> &B;
  GRExprEngine &Eng;
  ExplodedNode<GRState> *Pred;
  SaveAndRestore<bool> OldSink;
  SaveAndRestore<const void*> OldTag;
  SaveAndRestore<ProgramPoint::Kind> OldPointKind;
  SaveOr OldHasGen;

public:
  CheckerContext(ExplodedNodeSet<GRState> &dst,
                 GRStmtNodeBuilder<GRState> &builder,
                 GRExprEngine &eng,
                 ExplodedNode<GRState> *pred,
                 const void *tag, bool preVisit)
    : Dst(dst), B(builder), Eng(eng), Pred(pred), 
      OldSink(B.BuildSinks), OldTag(B.Tag),
      OldPointKind(B.PointKind), OldHasGen(B.HasGeneratedNode) {
        assert(Dst.empty());
        B.Tag = tag;
        if (preVisit)
          B.PointKind = ProgramPoint::PreStmtKind;        
      }
  
  ~CheckerContext() {
    if (!B.BuildSinks && Dst.empty() && !B.HasGeneratedNode)
      Dst.Add(Pred);
  }
  
  ConstraintManager &getConstraintManager() {
      return Eng.getConstraintManager();
  }
  ExplodedNodeSet<GRState> &getNodeSet() { return Dst; }
  GRStmtNodeBuilder<GRState> &getNodeBuilder() { return B; }
  ExplodedNode<GRState> *&getPredecessor() { return Pred; }
  const GRState *getState() { return B.GetState(Pred); }
  
  ExplodedNode<GRState> *generateNode(const Stmt* S,
                                      const GRState *state) {
    return B.generateNode(S, state, Pred);
  }
  
  void addTransition(ExplodedNode<GRState> *node) {
    Dst.Add(node);
  }
  
  void EmitReport(BugReport *R) {
    Eng.getBugReporter().EmitReport(R);
  }
};
  
class Checker {
private:
  friend class GRExprEngine;

  void GR_Visit(ExplodedNodeSet<GRState> &Dst,
                GRStmtNodeBuilder<GRState> &Builder,
                GRExprEngine &Eng,
                const Stmt *stmt,
                  ExplodedNode<GRState> *Pred, bool isPrevisit) {
    CheckerContext C(Dst, Builder, Eng, Pred, getTag(), isPrevisit);    
    assert(isPrevisit && "Only previsit supported for now.");
    _PreVisit(C, stmt);
  }
  
public:
  virtual ~Checker() {}
  virtual void _PreVisit(CheckerContext &C, const Stmt *stmt) = 0;
  virtual const void *getTag() = 0;
};

} // end clang namespace

#endif