168 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			168 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			C++
		
	
	
	
//== Environment.cpp - Map from Stmt* to Locations/Values -------*- C++ -*--==//
 | 
						|
//
 | 
						|
//                     The LLVM Compiler Infrastructure
 | 
						|
//
 | 
						|
// This file is distributed under the University of Illinois Open Source
 | 
						|
// License. See LICENSE.TXT for details.
 | 
						|
//
 | 
						|
//===----------------------------------------------------------------------===//
 | 
						|
//
 | 
						|
//  This file defined the Environment and EnvironmentManager classes.
 | 
						|
//
 | 
						|
//===----------------------------------------------------------------------===//
 | 
						|
#include "clang/Analysis/PathSensitive/GRState.h"
 | 
						|
#include "clang/Analysis/Analyses/LiveVariables.h"
 | 
						|
#include "llvm/ADT/ImmutableMap.h"
 | 
						|
#include "llvm/Support/Streams.h"
 | 
						|
#include "llvm/Support/Compiler.h"
 | 
						|
 | 
						|
using namespace clang;
 | 
						|
 | 
						|
SVal Environment::GetSVal(Stmt* E, BasicValueFactory& BasicVals) const {
 | 
						|
  
 | 
						|
  for (;;) {
 | 
						|
    
 | 
						|
    switch (E->getStmtClass()) {
 | 
						|
        
 | 
						|
      case Stmt::AddrLabelExprClass:        
 | 
						|
        return Loc::MakeVal(cast<AddrLabelExpr>(E));
 | 
						|
        
 | 
						|
        // ParenExprs are no-ops.
 | 
						|
        
 | 
						|
      case Stmt::ParenExprClass:        
 | 
						|
        E = cast<ParenExpr>(E)->getSubExpr();
 | 
						|
        continue;
 | 
						|
        
 | 
						|
      case Stmt::CharacterLiteralClass: {
 | 
						|
        CharacterLiteral* C = cast<CharacterLiteral>(E);
 | 
						|
        return NonLoc::MakeVal(BasicVals, C->getValue(), C->getType());
 | 
						|
      }
 | 
						|
        
 | 
						|
      case Stmt::IntegerLiteralClass: {
 | 
						|
        return NonLoc::MakeVal(BasicVals, cast<IntegerLiteral>(E));
 | 
						|
      }
 | 
						|
        
 | 
						|
      // Casts where the source and target type are the same
 | 
						|
      // are no-ops.  We blast through these to get the descendant
 | 
						|
      // subexpression that has a value.
 | 
						|
       
 | 
						|
      case Stmt::ImplicitCastExprClass:
 | 
						|
      case Stmt::CStyleCastExprClass: {
 | 
						|
        CastExpr* C = cast<CastExpr>(E);
 | 
						|
        QualType CT = C->getType();
 | 
						|
        
 | 
						|
        if (CT->isVoidType())
 | 
						|
          return UnknownVal();
 | 
						|
        
 | 
						|
        break;
 | 
						|
      }
 | 
						|
        
 | 
						|
        // Handle all other Stmt* using a lookup.
 | 
						|
        
 | 
						|
      default:
 | 
						|
        break;
 | 
						|
    };
 | 
						|
    
 | 
						|
    break;
 | 
						|
  }
 | 
						|
  
 | 
						|
  return LookupExpr(E);
 | 
						|
}
 | 
						|
 | 
						|
SVal Environment::GetBlkExprSVal(Stmt* E, BasicValueFactory& BasicVals) const {
 | 
						|
  
 | 
						|
  while (1) {
 | 
						|
    switch (E->getStmtClass()) {
 | 
						|
      case Stmt::ParenExprClass:
 | 
						|
        E = cast<ParenExpr>(E)->getSubExpr();
 | 
						|
        continue;
 | 
						|
        
 | 
						|
      case Stmt::CharacterLiteralClass: {
 | 
						|
        CharacterLiteral* C = cast<CharacterLiteral>(E);
 | 
						|
        return NonLoc::MakeVal(BasicVals, C->getValue(), C->getType());
 | 
						|
      }
 | 
						|
        
 | 
						|
      case Stmt::IntegerLiteralClass: {
 | 
						|
        return NonLoc::MakeVal(BasicVals, cast<IntegerLiteral>(E));
 | 
						|
      }
 | 
						|
        
 | 
						|
      default:
 | 
						|
        return LookupBlkExpr(E);
 | 
						|
    }
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
Environment EnvironmentManager::BindExpr(const Environment& Env, Stmt* E,SVal V,
 | 
						|
                                         bool isBlkExpr, bool Invalidate) {  
 | 
						|
  assert (E);
 | 
						|
  
 | 
						|
  if (V.isUnknown()) {    
 | 
						|
    if (Invalidate)
 | 
						|
      return isBlkExpr ? RemoveBlkExpr(Env, E) : RemoveSubExpr(Env, E);
 | 
						|
    else
 | 
						|
      return Env;
 | 
						|
  }
 | 
						|
 | 
						|
  return isBlkExpr ? AddBlkExpr(Env, E, V) : AddSubExpr(Env, E, V);
 | 
						|
}
 | 
						|
 | 
						|
namespace {
 | 
						|
class VISIBILITY_HIDDEN MarkLiveCallback : public SymbolVisitor {
 | 
						|
  SymbolReaper &SymReaper;
 | 
						|
public:
 | 
						|
  MarkLiveCallback(SymbolReaper &symreaper) : SymReaper(symreaper) {}  
 | 
						|
  bool VisitSymbol(SymbolRef sym) { SymReaper.markLive(sym); return true; }
 | 
						|
};
 | 
						|
} // end anonymous namespace
 | 
						|
 | 
						|
// RemoveDeadBindings:
 | 
						|
//  - Remove subexpression bindings.
 | 
						|
//  - Remove dead block expression bindings.
 | 
						|
//  - Keep live block expression bindings:
 | 
						|
//   - Mark their reachable symbols live in SymbolReaper, 
 | 
						|
//     see ScanReachableSymbols.
 | 
						|
//   - Mark the region in DRoots if the binding is a loc::MemRegionVal.
 | 
						|
 | 
						|
Environment 
 | 
						|
EnvironmentManager::RemoveDeadBindings(Environment Env, Stmt* Loc,
 | 
						|
                                       SymbolReaper& SymReaper,
 | 
						|
                                       GRStateManager& StateMgr,
 | 
						|
                                       const GRState *state,
 | 
						|
                              llvm::SmallVectorImpl<const MemRegion*>& DRoots) {
 | 
						|
  
 | 
						|
  // Drop bindings for subexpressions.
 | 
						|
  Env = RemoveSubExprBindings(Env);
 | 
						|
 | 
						|
  // Iterate over the block-expr bindings.
 | 
						|
  for (Environment::beb_iterator I = Env.beb_begin(), E = Env.beb_end(); 
 | 
						|
       I != E; ++I) {
 | 
						|
    Stmt* BlkExpr = I.getKey();
 | 
						|
 | 
						|
    if (SymReaper.isLive(Loc, BlkExpr)) {
 | 
						|
      SVal X = I.getData();
 | 
						|
 | 
						|
      // If the block expr's value is a memory region, then mark that region.
 | 
						|
      if (isa<loc::MemRegionVal>(X))
 | 
						|
        DRoots.push_back(cast<loc::MemRegionVal>(X).getRegion());
 | 
						|
 | 
						|
      // Mark all symbols in the block expr's value live.
 | 
						|
      MarkLiveCallback cb(SymReaper);
 | 
						|
      StateMgr.scanReachableSymbols(X, state, cb);
 | 
						|
    } else {
 | 
						|
      // The block expr is dead.
 | 
						|
      SVal X = I.getData();
 | 
						|
 | 
						|
      // Do not misclean LogicalExpr or ConditionalOperator.  It is dead at the
 | 
						|
      // beginning of itself, but we need its UndefinedVal to determine its
 | 
						|
      // SVal.
 | 
						|
 | 
						|
      if (X.isUndef() && cast<UndefinedVal>(X).getData())
 | 
						|
        continue;
 | 
						|
 | 
						|
      Env = RemoveBlkExpr(Env, BlkExpr);
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  return Env;
 | 
						|
}
 |