blob: 8ad962875b060dee6ea9205cfa4421aa5053d43d [file] [log] [blame]
Jordan Rose42ee04d2012-06-29 00:33:10 +00001//== TraversalChecker.cpp -------------------------------------- -*- 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//
Jordan Rose8889cf02012-07-10 23:56:23 +000010// These checkers print various aspects of the ExprEngine's traversal of the CFG
11// as it builds the ExplodedGraph.
Jordan Rose42ee04d2012-06-29 00:33:10 +000012//
13//===----------------------------------------------------------------------===//
14#include "ClangSACheckers.h"
15#include "clang/AST/ParentMap.h"
16#include "clang/AST/StmtObjC.h"
17#include "clang/StaticAnalyzer/Core/Checker.h"
18#include "clang/StaticAnalyzer/Core/CheckerManager.h"
Jordan Rose4f7df9b2012-07-26 21:39:41 +000019#include "clang/StaticAnalyzer/Core/PathSensitive/CallEvent.h"
Jordan Rose42ee04d2012-06-29 00:33:10 +000020#include "clang/StaticAnalyzer/Core/PathSensitive/CheckerContext.h"
Benjamin Kramer444a1302012-12-01 17:12:56 +000021#include "llvm/Support/raw_ostream.h"
Jordan Rose42ee04d2012-06-29 00:33:10 +000022
23using namespace clang;
24using namespace ento;
25
26namespace {
27class TraversalDumper : public Checker< check::BranchCondition,
Devin Coughlin8d922aa2016-02-19 01:35:10 +000028 check::BeginFunction,
Anna Zaks3fdcc0b2013-01-03 00:25:29 +000029 check::EndFunction > {
Jordan Rose42ee04d2012-06-29 00:33:10 +000030public:
31 void checkBranchCondition(const Stmt *Condition, CheckerContext &C) const;
Devin Coughlin8d922aa2016-02-19 01:35:10 +000032 void checkBeginFunction(CheckerContext &C) const;
Anna Zaks3fdcc0b2013-01-03 00:25:29 +000033 void checkEndFunction(CheckerContext &C) const;
Jordan Rose42ee04d2012-06-29 00:33:10 +000034};
35}
36
37void TraversalDumper::checkBranchCondition(const Stmt *Condition,
38 CheckerContext &C) const {
39 // Special-case Objective-C's for-in loop, which uses the entire loop as its
40 // condition. We just print the collection expression.
41 const Stmt *Parent = dyn_cast<ObjCForCollectionStmt>(Condition);
42 if (!Parent) {
43 const ParentMap &Parents = C.getLocationContext()->getParentMap();
44 Parent = Parents.getParent(Condition);
45 }
46
47 // It is mildly evil to print directly to llvm::outs() rather than emitting
48 // warnings, but this ensures things do not get filtered out by the rest of
49 // the static analyzer machinery.
50 SourceLocation Loc = Parent->getLocStart();
51 llvm::outs() << C.getSourceManager().getSpellingLineNumber(Loc) << " "
52 << Parent->getStmtClassName() << "\n";
53}
54
Devin Coughlin8d922aa2016-02-19 01:35:10 +000055void TraversalDumper::checkBeginFunction(CheckerContext &C) const {
56 llvm::outs() << "--BEGIN FUNCTION--\n";
57}
58
Anna Zaks3fdcc0b2013-01-03 00:25:29 +000059void TraversalDumper::checkEndFunction(CheckerContext &C) const {
60 llvm::outs() << "--END FUNCTION--\n";
Jordan Rose42ee04d2012-06-29 00:33:10 +000061}
62
63void ento::registerTraversalDumper(CheckerManager &mgr) {
64 mgr.registerChecker<TraversalDumper>();
65}
Jordan Rose8889cf02012-07-10 23:56:23 +000066
67//------------------------------------------------------------------------------
68
69namespace {
Jordan Rosee4bdb102013-03-21 18:16:59 +000070class CallDumper : public Checker< check::PreCall,
71 check::PostCall > {
Jordan Rose8889cf02012-07-10 23:56:23 +000072public:
73 void checkPreCall(const CallEvent &Call, CheckerContext &C) const;
Jordan Rosee4bdb102013-03-21 18:16:59 +000074 void checkPostCall(const CallEvent &Call, CheckerContext &C) const;
Jordan Rose8889cf02012-07-10 23:56:23 +000075};
76}
77
78void CallDumper::checkPreCall(const CallEvent &Call, CheckerContext &C) const {
79 unsigned Indentation = 0;
80 for (const LocationContext *LC = C.getLocationContext()->getParent();
Craig Topper0dbb7832014-05-27 02:45:47 +000081 LC != nullptr; LC = LC->getParent())
Jordan Rose8889cf02012-07-10 23:56:23 +000082 ++Indentation;
83
84 // It is mildly evil to print directly to llvm::outs() rather than emitting
85 // warnings, but this ensures things do not get filtered out by the rest of
86 // the static analyzer machinery.
87 llvm::outs().indent(Indentation);
88 Call.dump(llvm::outs());
89}
90
Jordan Rosee4bdb102013-03-21 18:16:59 +000091void CallDumper::checkPostCall(const CallEvent &Call, CheckerContext &C) const {
92 const Expr *CallE = Call.getOriginExpr();
93 if (!CallE)
94 return;
95
96 unsigned Indentation = 0;
97 for (const LocationContext *LC = C.getLocationContext()->getParent();
Craig Topper0dbb7832014-05-27 02:45:47 +000098 LC != nullptr; LC = LC->getParent())
Jordan Rosee4bdb102013-03-21 18:16:59 +000099 ++Indentation;
100
101 // It is mildly evil to print directly to llvm::outs() rather than emitting
102 // warnings, but this ensures things do not get filtered out by the rest of
103 // the static analyzer machinery.
104 llvm::outs().indent(Indentation);
105 if (Call.getResultType()->isVoidType())
106 llvm::outs() << "Returning void\n";
107 else
108 llvm::outs() << "Returning " << C.getSVal(CallE) << "\n";
109}
110
Jordan Rose8889cf02012-07-10 23:56:23 +0000111void ento::registerCallDumper(CheckerManager &mgr) {
112 mgr.registerChecker<CallDumper>();
113}