371 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			371 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
	
//===--- PathDiagnostic.cpp - Path-Specific Diagnostic Handling -*- 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 the PathDiagnostic-related interfaces.
 | 
						|
//
 | 
						|
//===----------------------------------------------------------------------===//
 | 
						|
 | 
						|
#include "clang/StaticAnalyzer/Core/BugReporter/PathDiagnostic.h"
 | 
						|
#include "clang/StaticAnalyzer/Core/PathSensitive/ExplodedGraph.h"
 | 
						|
#include "clang/AST/Expr.h"
 | 
						|
#include "clang/AST/Decl.h"
 | 
						|
#include "clang/AST/DeclObjC.h"
 | 
						|
#include "clang/AST/ParentMap.h"
 | 
						|
#include "clang/AST/StmtCXX.h"
 | 
						|
#include "llvm/ADT/SmallString.h"
 | 
						|
 | 
						|
using namespace clang;
 | 
						|
using namespace ento;
 | 
						|
 | 
						|
bool PathDiagnosticMacroPiece::containsEvent() const {
 | 
						|
  for (const_iterator I = begin(), E = end(); I!=E; ++I) {
 | 
						|
    if (isa<PathDiagnosticEventPiece>(*I))
 | 
						|
      return true;
 | 
						|
 | 
						|
    if (PathDiagnosticMacroPiece *MP = dyn_cast<PathDiagnosticMacroPiece>(*I))
 | 
						|
      if (MP->containsEvent())
 | 
						|
        return true;
 | 
						|
  }
 | 
						|
 | 
						|
  return false;
 | 
						|
}
 | 
						|
 | 
						|
static StringRef StripTrailingDots(StringRef s) {
 | 
						|
  for (StringRef::size_type i = s.size(); i != 0; --i)
 | 
						|
    if (s[i - 1] != '.')
 | 
						|
      return s.substr(0, i);
 | 
						|
  return "";
 | 
						|
}
 | 
						|
 | 
						|
PathDiagnosticPiece::PathDiagnosticPiece(StringRef s,
 | 
						|
                                         Kind k, DisplayHint hint)
 | 
						|
  : str(StripTrailingDots(s)), kind(k), Hint(hint) {}
 | 
						|
 | 
						|
PathDiagnosticPiece::PathDiagnosticPiece(Kind k, DisplayHint hint)
 | 
						|
  : kind(k), Hint(hint) {}
 | 
						|
 | 
						|
PathDiagnosticPiece::~PathDiagnosticPiece() {}
 | 
						|
PathDiagnosticEventPiece::~PathDiagnosticEventPiece() {}
 | 
						|
PathDiagnosticControlFlowPiece::~PathDiagnosticControlFlowPiece() {}
 | 
						|
 | 
						|
PathDiagnosticMacroPiece::~PathDiagnosticMacroPiece() {
 | 
						|
  for (iterator I = begin(), E = end(); I != E; ++I) delete *I;
 | 
						|
}
 | 
						|
 | 
						|
PathDiagnostic::PathDiagnostic() : Size(0) {}
 | 
						|
 | 
						|
PathDiagnostic::~PathDiagnostic() {
 | 
						|
  for (iterator I = begin(), E = end(); I != E; ++I) delete &*I;
 | 
						|
}
 | 
						|
 | 
						|
void PathDiagnostic::resetPath(bool deletePieces) {
 | 
						|
  Size = 0;
 | 
						|
 | 
						|
  if (deletePieces)
 | 
						|
    for (iterator I=begin(), E=end(); I!=E; ++I)
 | 
						|
      delete &*I;
 | 
						|
 | 
						|
  path.clear();
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
PathDiagnostic::PathDiagnostic(StringRef bugtype, StringRef desc,
 | 
						|
                               StringRef category)
 | 
						|
  : Size(0),
 | 
						|
    BugType(StripTrailingDots(bugtype)),
 | 
						|
    Desc(StripTrailingDots(desc)),
 | 
						|
    Category(StripTrailingDots(category)) {}
 | 
						|
 | 
						|
void PathDiagnosticClient::HandlePathDiagnostic(const PathDiagnostic *D) {
 | 
						|
  // For now this simply forwards to HandlePathDiagnosticImpl.  In the future
 | 
						|
  // we can use this indirection to control for multi-threaded access to
 | 
						|
  // the PathDiagnosticClient from multiple bug reporters.
 | 
						|
  HandlePathDiagnosticImpl(D);
 | 
						|
}
 | 
						|
 | 
						|
//===----------------------------------------------------------------------===//
 | 
						|
// PathDiagnosticLocation methods.
 | 
						|
//===----------------------------------------------------------------------===//
 | 
						|
 | 
						|
static SourceLocation getValidSourceLocation(const Stmt* S,
 | 
						|
                                             LocationOrAnalysisContext LAC) {
 | 
						|
  SourceLocation L = S->getLocStart();
 | 
						|
  assert(!LAC.isNull() && "A valid LocationContext or AnalysisContext should "
 | 
						|
                          "be passed to PathDiagnosticLocation upon creation.");
 | 
						|
 | 
						|
  // S might be a temporary statement that does not have a location in the
 | 
						|
  // source code, so find an enclosing statement and use it's location.
 | 
						|
  if (!L.isValid()) {
 | 
						|
 | 
						|
    ParentMap *PM = 0;
 | 
						|
    if (LAC.is<const LocationContext*>())
 | 
						|
      PM = &LAC.get<const LocationContext*>()->getParentMap();
 | 
						|
    else
 | 
						|
      PM = &LAC.get<AnalysisContext*>()->getParentMap();
 | 
						|
 | 
						|
    while (!L.isValid()) {
 | 
						|
      S = PM->getParent(S);
 | 
						|
      L = S->getLocStart();
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  return L;
 | 
						|
}
 | 
						|
 | 
						|
PathDiagnosticLocation
 | 
						|
  PathDiagnosticLocation::createBegin(const Decl *D,
 | 
						|
                                      const SourceManager &SM) {
 | 
						|
  return PathDiagnosticLocation(D->getLocStart(), SM, SingleLocK);
 | 
						|
}
 | 
						|
 | 
						|
PathDiagnosticLocation
 | 
						|
  PathDiagnosticLocation::createBegin(const Stmt *S,
 | 
						|
                                      const SourceManager &SM,
 | 
						|
                                      LocationOrAnalysisContext LAC) {
 | 
						|
  return PathDiagnosticLocation(getValidSourceLocation(S, LAC),
 | 
						|
                                SM, SingleLocK);
 | 
						|
}
 | 
						|
 | 
						|
PathDiagnosticLocation
 | 
						|
  PathDiagnosticLocation::createOperatorLoc(const BinaryOperator *BO,
 | 
						|
                                            const SourceManager &SM) {
 | 
						|
  return PathDiagnosticLocation(BO->getOperatorLoc(), SM, SingleLocK);
 | 
						|
}
 | 
						|
 | 
						|
PathDiagnosticLocation
 | 
						|
  PathDiagnosticLocation::createMemberLoc(const MemberExpr *ME,
 | 
						|
                                          const SourceManager &SM) {
 | 
						|
  return PathDiagnosticLocation(ME->getMemberLoc(), SM, SingleLocK);
 | 
						|
}
 | 
						|
 | 
						|
PathDiagnosticLocation
 | 
						|
  PathDiagnosticLocation::createBeginBrace(const CompoundStmt *CS,
 | 
						|
                                           const SourceManager &SM) {
 | 
						|
  SourceLocation L = CS->getLBracLoc();
 | 
						|
  return PathDiagnosticLocation(L, SM, SingleLocK);
 | 
						|
}
 | 
						|
 | 
						|
PathDiagnosticLocation
 | 
						|
  PathDiagnosticLocation::createEndBrace(const CompoundStmt *CS,
 | 
						|
                                         const SourceManager &SM) {
 | 
						|
  SourceLocation L = CS->getRBracLoc();
 | 
						|
  return PathDiagnosticLocation(L, SM, SingleLocK);
 | 
						|
}
 | 
						|
 | 
						|
PathDiagnosticLocation
 | 
						|
  PathDiagnosticLocation::createDeclBegin(const LocationContext *LC,
 | 
						|
                                          const SourceManager &SM) {
 | 
						|
  // FIXME: Should handle CXXTryStmt if analyser starts supporting C++.
 | 
						|
  if (const CompoundStmt *CS =
 | 
						|
        dyn_cast_or_null<CompoundStmt>(LC->getDecl()->getBody()))
 | 
						|
    if (!CS->body_empty()) {
 | 
						|
      SourceLocation Loc = (*CS->body_begin())->getLocStart();
 | 
						|
      return PathDiagnosticLocation(Loc, SM, SingleLocK);
 | 
						|
    }
 | 
						|
 | 
						|
  return PathDiagnosticLocation();
 | 
						|
}
 | 
						|
 | 
						|
PathDiagnosticLocation
 | 
						|
  PathDiagnosticLocation::createDeclEnd(const LocationContext *LC,
 | 
						|
                                        const SourceManager &SM) {
 | 
						|
  SourceLocation L = LC->getDecl()->getBodyRBrace();
 | 
						|
  return PathDiagnosticLocation(L, SM, SingleLocK);
 | 
						|
}
 | 
						|
 | 
						|
PathDiagnosticLocation
 | 
						|
  PathDiagnosticLocation::create(const ProgramPoint& P,
 | 
						|
                                 const SourceManager &SMng) {
 | 
						|
 | 
						|
  const Stmt* S = 0;
 | 
						|
  if (const BlockEdge *BE = dyn_cast<BlockEdge>(&P)) {
 | 
						|
    const CFGBlock *BSrc = BE->getSrc();
 | 
						|
    S = BSrc->getTerminatorCondition();
 | 
						|
  }
 | 
						|
  else if (const PostStmt *PS = dyn_cast<PostStmt>(&P)) {
 | 
						|
    S = PS->getStmt();
 | 
						|
  }
 | 
						|
 | 
						|
  return PathDiagnosticLocation(S, SMng, P.getLocationContext());
 | 
						|
 | 
						|
  if (!S)
 | 
						|
    return PathDiagnosticLocation();
 | 
						|
}
 | 
						|
 | 
						|
PathDiagnosticLocation
 | 
						|
  PathDiagnosticLocation::createEndOfPath(const ExplodedNode* N,
 | 
						|
                                          const SourceManager &SM) {
 | 
						|
  assert(N && "Cannot create a location with a null node.");
 | 
						|
 | 
						|
  const ExplodedNode *NI = N;
 | 
						|
 | 
						|
  while (NI) {
 | 
						|
    ProgramPoint P = NI->getLocation();
 | 
						|
    const LocationContext *LC = P.getLocationContext();
 | 
						|
    if (const StmtPoint *PS = dyn_cast<StmtPoint>(&P))
 | 
						|
      return PathDiagnosticLocation(PS->getStmt(), SM, LC);
 | 
						|
    else if (const BlockEdge *BE = dyn_cast<BlockEdge>(&P)) {
 | 
						|
      const Stmt *Term = BE->getSrc()->getTerminator();
 | 
						|
      assert(Term);
 | 
						|
      return PathDiagnosticLocation(Term, SM, LC);
 | 
						|
    }
 | 
						|
    NI = NI->succ_empty() ? 0 : *(NI->succ_begin());
 | 
						|
  }
 | 
						|
 | 
						|
  return createDeclEnd(N->getLocationContext(), SM);
 | 
						|
}
 | 
						|
 | 
						|
PathDiagnosticLocation PathDiagnosticLocation::createSingleLocation(
 | 
						|
                                           const PathDiagnosticLocation &PDL) {
 | 
						|
  FullSourceLoc L = PDL.asLocation();
 | 
						|
  return PathDiagnosticLocation(L, L.getManager(), SingleLocK);
 | 
						|
}
 | 
						|
 | 
						|
FullSourceLoc
 | 
						|
  PathDiagnosticLocation::genLocation(SourceLocation L, LocationOrAnalysisContext LAC) const {
 | 
						|
  assert(isValid());
 | 
						|
  // Note that we want a 'switch' here so that the compiler can warn us in
 | 
						|
  // case we add more cases.
 | 
						|
  switch (K) {
 | 
						|
    case SingleLocK:
 | 
						|
    case RangeK:
 | 
						|
      break;
 | 
						|
    case StmtK:
 | 
						|
      return FullSourceLoc(getValidSourceLocation(S, LAC),
 | 
						|
                           const_cast<SourceManager&>(*SM));
 | 
						|
    case DeclK:
 | 
						|
      return FullSourceLoc(D->getLocation(), const_cast<SourceManager&>(*SM));
 | 
						|
  }
 | 
						|
 | 
						|
  return FullSourceLoc(L, const_cast<SourceManager&>(*SM));
 | 
						|
}
 | 
						|
 | 
						|
PathDiagnosticRange
 | 
						|
  PathDiagnosticLocation::genRange(SourceLocation L, LocationOrAnalysisContext LAC) const {
 | 
						|
  assert(isValid());
 | 
						|
  // Note that we want a 'switch' here so that the compiler can warn us in
 | 
						|
  // case we add more cases.
 | 
						|
  switch (K) {
 | 
						|
    case SingleLocK:
 | 
						|
      return PathDiagnosticRange(SourceRange(L,L), true);
 | 
						|
    case RangeK:
 | 
						|
      break;
 | 
						|
    case StmtK: {
 | 
						|
      const Stmt *S = asStmt();
 | 
						|
      switch (S->getStmtClass()) {
 | 
						|
        default:
 | 
						|
          break;
 | 
						|
        case Stmt::DeclStmtClass: {
 | 
						|
          const DeclStmt *DS = cast<DeclStmt>(S);
 | 
						|
          if (DS->isSingleDecl()) {
 | 
						|
            // Should always be the case, but we'll be defensive.
 | 
						|
            return SourceRange(DS->getLocStart(),
 | 
						|
                               DS->getSingleDecl()->getLocation());
 | 
						|
          }
 | 
						|
          break;
 | 
						|
        }
 | 
						|
          // FIXME: Provide better range information for different
 | 
						|
          //  terminators.
 | 
						|
        case Stmt::IfStmtClass:
 | 
						|
        case Stmt::WhileStmtClass:
 | 
						|
        case Stmt::DoStmtClass:
 | 
						|
        case Stmt::ForStmtClass:
 | 
						|
        case Stmt::ChooseExprClass:
 | 
						|
        case Stmt::IndirectGotoStmtClass:
 | 
						|
        case Stmt::SwitchStmtClass:
 | 
						|
        case Stmt::BinaryConditionalOperatorClass:
 | 
						|
        case Stmt::ConditionalOperatorClass:
 | 
						|
        case Stmt::ObjCForCollectionStmtClass: {
 | 
						|
          SourceLocation L = getValidSourceLocation(S, LAC);
 | 
						|
          return SourceRange(L, L);
 | 
						|
        }
 | 
						|
      }
 | 
						|
 | 
						|
      return S->getSourceRange();
 | 
						|
    }
 | 
						|
    case DeclK:
 | 
						|
      if (const ObjCMethodDecl *MD = dyn_cast<ObjCMethodDecl>(D))
 | 
						|
        return MD->getSourceRange();
 | 
						|
      if (const FunctionDecl *FD = dyn_cast<FunctionDecl>(D)) {
 | 
						|
        if (Stmt *Body = FD->getBody())
 | 
						|
          return Body->getSourceRange();
 | 
						|
      }
 | 
						|
      else {
 | 
						|
        SourceLocation L = D->getLocation();
 | 
						|
        return PathDiagnosticRange(SourceRange(L, L), true);
 | 
						|
      }
 | 
						|
  }
 | 
						|
 | 
						|
  return SourceRange(L,L);
 | 
						|
}
 | 
						|
 | 
						|
void PathDiagnosticLocation::flatten() {
 | 
						|
  if (K == StmtK) {
 | 
						|
    K = RangeK;
 | 
						|
    S = 0;
 | 
						|
    D = 0;
 | 
						|
  }
 | 
						|
  else if (K == DeclK) {
 | 
						|
    K = SingleLocK;
 | 
						|
    S = 0;
 | 
						|
    D = 0;
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
//===----------------------------------------------------------------------===//
 | 
						|
// FoldingSet profiling methods.
 | 
						|
//===----------------------------------------------------------------------===//
 | 
						|
 | 
						|
void PathDiagnosticLocation::Profile(llvm::FoldingSetNodeID &ID) const {
 | 
						|
  ID.AddInteger(Range.getBegin().getRawEncoding());
 | 
						|
  ID.AddInteger(Range.getEnd().getRawEncoding());
 | 
						|
  ID.AddInteger(Loc.getRawEncoding());
 | 
						|
  return;
 | 
						|
}
 | 
						|
 | 
						|
void PathDiagnosticPiece::Profile(llvm::FoldingSetNodeID &ID) const {
 | 
						|
  ID.AddInteger((unsigned) getKind());
 | 
						|
  ID.AddString(str);
 | 
						|
  // FIXME: Add profiling support for code hints.
 | 
						|
  ID.AddInteger((unsigned) getDisplayHint());
 | 
						|
  for (range_iterator I = ranges_begin(), E = ranges_end(); I != E; ++I) {
 | 
						|
    ID.AddInteger(I->getBegin().getRawEncoding());
 | 
						|
    ID.AddInteger(I->getEnd().getRawEncoding());
 | 
						|
  }  
 | 
						|
}
 | 
						|
 | 
						|
void PathDiagnosticSpotPiece::Profile(llvm::FoldingSetNodeID &ID) const {
 | 
						|
  PathDiagnosticPiece::Profile(ID);
 | 
						|
  ID.Add(Pos);
 | 
						|
}
 | 
						|
 | 
						|
void PathDiagnosticControlFlowPiece::Profile(llvm::FoldingSetNodeID &ID) const {
 | 
						|
  PathDiagnosticPiece::Profile(ID);
 | 
						|
  for (const_iterator I = begin(), E = end(); I != E; ++I)
 | 
						|
    ID.Add(*I);
 | 
						|
}
 | 
						|
 | 
						|
void PathDiagnosticMacroPiece::Profile(llvm::FoldingSetNodeID &ID) const {
 | 
						|
  PathDiagnosticSpotPiece::Profile(ID);
 | 
						|
  for (const_iterator I = begin(), E = end(); I != E; ++I)
 | 
						|
    ID.Add(**I);
 | 
						|
}
 | 
						|
 | 
						|
void PathDiagnostic::Profile(llvm::FoldingSetNodeID &ID) const {
 | 
						|
  ID.AddInteger(Size);
 | 
						|
  ID.AddString(BugType);
 | 
						|
  ID.AddString(Desc);
 | 
						|
  ID.AddString(Category);
 | 
						|
  for (const_iterator I = begin(), E = end(); I != E; ++I)
 | 
						|
    ID.Add(*I);
 | 
						|
  
 | 
						|
  for (meta_iterator I = meta_begin(), E = meta_end(); I != E; ++I)
 | 
						|
    ID.AddString(*I);
 | 
						|
}
 |