2009-06-23 14:22:22 +08:00
|
|
|
//== ValueManager.cpp - Aggregate manager of symbols and SVals --*- 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 ValueManager, a class that manages symbolic values
|
|
|
|
// and SVals created for use by GRExprEngine and related classes. It
|
|
|
|
// wraps and owns SymbolManager, MemRegionManager, and BasicValueFactory.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "clang/Analysis/PathSensitive/ValueManager.h"
|
|
|
|
|
|
|
|
using namespace clang;
|
|
|
|
using namespace llvm;
|
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Utility methods for constructing SVals.
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
SVal ValueManager::makeZeroVal(QualType T) {
|
|
|
|
if (Loc::IsLocType(T))
|
2009-06-23 17:02:15 +08:00
|
|
|
return makeNull();
|
2009-06-23 14:22:22 +08:00
|
|
|
|
|
|
|
if (T->isIntegerType())
|
2009-06-23 17:02:15 +08:00
|
|
|
return makeIntVal(0, T);
|
2009-06-23 14:22:22 +08:00
|
|
|
|
|
|
|
// FIXME: Handle floats.
|
|
|
|
// FIXME: Handle structs.
|
|
|
|
return UnknownVal();
|
|
|
|
}
|
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Utility methods for constructing Non-Locs.
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
NonLoc ValueManager::makeNonLoc(const SymExpr *lhs, BinaryOperator::Opcode op,
|
|
|
|
const APSInt& v, QualType T) {
|
|
|
|
// The Environment ensures we always get a persistent APSInt in
|
|
|
|
// BasicValueFactory, so we don't need to get the APSInt from
|
|
|
|
// BasicValueFactory again.
|
|
|
|
assert(!Loc::IsLocType(T));
|
|
|
|
return nonloc::SymExprVal(SymMgr.getSymIntExpr(lhs, op, v, T));
|
|
|
|
}
|
|
|
|
|
|
|
|
NonLoc ValueManager::makeNonLoc(const SymExpr *lhs, BinaryOperator::Opcode op,
|
|
|
|
const SymExpr *rhs, QualType T) {
|
|
|
|
assert(SymMgr.getType(lhs) == SymMgr.getType(rhs));
|
|
|
|
assert(!Loc::IsLocType(T));
|
|
|
|
return nonloc::SymExprVal(SymMgr.getSymSymExpr(lhs, op, rhs, T));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-07-16 09:32:00 +08:00
|
|
|
SVal ValueManager::convertToArrayIndex(SVal V) {
|
2009-07-22 05:03:30 +08:00
|
|
|
if (V.isUnknownOrUndef())
|
|
|
|
return V;
|
|
|
|
|
2009-07-16 09:32:00 +08:00
|
|
|
// Common case: we have an appropriately sized integer.
|
|
|
|
if (nonloc::ConcreteInt* CI = dyn_cast<nonloc::ConcreteInt>(&V)) {
|
|
|
|
const llvm::APSInt& I = CI->getValue();
|
|
|
|
if (I.getBitWidth() == ArrayIndexWidth && I.isSigned())
|
|
|
|
return V;
|
|
|
|
}
|
|
|
|
|
2009-07-22 05:03:30 +08:00
|
|
|
return SVator->EvalCastNL(cast<NonLoc>(V), ArrayIndexTy);
|
2009-07-16 09:32:00 +08:00
|
|
|
}
|
|
|
|
|
2009-06-23 14:22:22 +08:00
|
|
|
SVal ValueManager::getRegionValueSymbolVal(const MemRegion* R, QualType T) {
|
2009-08-01 14:17:29 +08:00
|
|
|
|
|
|
|
if (T.isNull()) {
|
|
|
|
const TypedRegion* TR = cast<TypedRegion>(R);
|
|
|
|
T = TR->getValueType(SymMgr.getContext());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!SymbolManager::canSymbolicate(T))
|
|
|
|
return UnknownVal();
|
|
|
|
|
2009-06-23 14:22:22 +08:00
|
|
|
SymbolRef sym = SymMgr.getRegionValueSymbol(R, T);
|
2009-08-01 14:17:29 +08:00
|
|
|
|
|
|
|
// If T is of function pointer type or a block pointer type, create a
|
|
|
|
// CodeTextRegion wrapping that symbol.
|
|
|
|
if (T->isFunctionPointerType() || T->isBlockPointerType())
|
|
|
|
return loc::MemRegionVal(MemMgr.getCodeTextRegion(sym, T));
|
2009-06-23 14:22:22 +08:00
|
|
|
|
2009-08-01 14:17:29 +08:00
|
|
|
if (Loc::IsLocType(T))
|
|
|
|
return loc::MemRegionVal(MemMgr.getSymbolicRegion(sym));
|
2009-06-23 14:22:22 +08:00
|
|
|
|
2009-08-01 14:17:29 +08:00
|
|
|
return nonloc::SymbolVal(sym);
|
2009-06-23 14:22:22 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
SVal ValueManager::getConjuredSymbolVal(const Expr* E, unsigned Count) {
|
|
|
|
QualType T = E->getType();
|
2009-08-01 14:17:29 +08:00
|
|
|
|
|
|
|
if (!SymbolManager::canSymbolicate(T))
|
|
|
|
return UnknownVal();
|
|
|
|
|
2009-06-23 14:22:22 +08:00
|
|
|
SymbolRef sym = SymMgr.getConjuredSymbol(E, Count);
|
|
|
|
|
2009-07-18 14:27:01 +08:00
|
|
|
// If T is of function pointer type or a block pointer type, create a
|
|
|
|
// CodeTextRegion wrapping a symbol.
|
2009-08-01 14:17:29 +08:00
|
|
|
if (T->isFunctionPointerType() || T->isBlockPointerType())
|
2009-06-23 17:02:15 +08:00
|
|
|
return loc::MemRegionVal(MemMgr.getCodeTextRegion(sym, T));
|
2009-06-23 14:22:22 +08:00
|
|
|
|
|
|
|
if (Loc::IsLocType(T))
|
2009-06-23 17:02:15 +08:00
|
|
|
return loc::MemRegionVal(MemMgr.getSymbolicRegion(sym));
|
2009-06-23 14:22:22 +08:00
|
|
|
|
2009-08-01 14:17:29 +08:00
|
|
|
return nonloc::SymbolVal(sym);
|
2009-06-23 14:22:22 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
SVal ValueManager::getConjuredSymbolVal(const Expr* E, QualType T,
|
|
|
|
unsigned Count) {
|
2009-08-01 14:17:29 +08:00
|
|
|
|
|
|
|
if (!SymbolManager::canSymbolicate(T))
|
|
|
|
return UnknownVal();
|
2009-06-23 14:22:22 +08:00
|
|
|
|
|
|
|
SymbolRef sym = SymMgr.getConjuredSymbol(E, T, Count);
|
|
|
|
|
2009-07-18 14:27:01 +08:00
|
|
|
// If T is of function pointer type or a block pointer type, create a
|
|
|
|
// CodeTextRegion wrapping a symbol.
|
2009-08-01 14:17:29 +08:00
|
|
|
if (T->isFunctionPointerType() || T->isBlockPointerType())
|
2009-06-23 17:02:15 +08:00
|
|
|
return loc::MemRegionVal(MemMgr.getCodeTextRegion(sym, T));
|
2009-06-23 14:22:22 +08:00
|
|
|
|
|
|
|
if (Loc::IsLocType(T))
|
2009-06-23 17:02:15 +08:00
|
|
|
return loc::MemRegionVal(MemMgr.getSymbolicRegion(sym));
|
2009-06-23 14:22:22 +08:00
|
|
|
|
2009-08-01 14:17:29 +08:00
|
|
|
return nonloc::SymbolVal(sym);
|
2009-06-23 14:22:22 +08:00
|
|
|
}
|
|
|
|
|
2009-07-15 10:27:32 +08:00
|
|
|
|
|
|
|
SVal ValueManager::getDerivedRegionValueSymbolVal(SymbolRef parentSymbol,
|
|
|
|
const TypedRegion *R) {
|
|
|
|
QualType T = R->getValueType(R->getContext());
|
2009-08-01 14:17:29 +08:00
|
|
|
|
|
|
|
if (!SymbolManager::canSymbolicate(T))
|
|
|
|
return UnknownVal();
|
|
|
|
|
|
|
|
SymbolRef sym = SymMgr.getDerivedSymbol(parentSymbol, R);
|
2009-07-15 10:27:32 +08:00
|
|
|
|
|
|
|
if (Loc::IsLocType(T))
|
|
|
|
return loc::MemRegionVal(MemMgr.getSymbolicRegion(sym));
|
|
|
|
|
2009-08-01 14:17:29 +08:00
|
|
|
return nonloc::SymbolVal(sym);
|
2009-07-15 10:27:32 +08:00
|
|
|
}
|
|
|
|
|
2009-06-23 14:22:22 +08:00
|
|
|
SVal ValueManager::getFunctionPointer(const FunctionDecl* FD) {
|
|
|
|
CodeTextRegion* R
|
|
|
|
= MemMgr.getCodeTextRegion(FD, Context.getPointerType(FD->getType()));
|
|
|
|
return loc::MemRegionVal(R);
|
|
|
|
}
|