mirror of
https://github.com/intel/llvm.git
synced 2026-02-04 03:26:06 +08:00
Also adds a getKnownValue() method to SValuator, which gets the integer value of an SVal if it is known to only have one possible value. There are more places in the code that could be using this, but in general we want to be dealing entirely in SVals, so its usefulness is limited. The only visible functionality change is that extents are now honored for any DeclRegion, such as fields and Objective-C ivars, rather than just variables. This shows up in bounds-checking and cast-size-checking. llvm-svn: 107577
85 lines
2.5 KiB
C++
85 lines
2.5 KiB
C++
//=== BuiltinFunctionChecker.cpp --------------------------------*- C++ -*-===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This checker evaluates clang builtin functions.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "GRExprEngineInternalChecks.h"
|
|
#include "clang/Checker/PathSensitive/Checker.h"
|
|
#include "clang/Basic/Builtins.h"
|
|
|
|
using namespace clang;
|
|
|
|
namespace {
|
|
|
|
class BuiltinFunctionChecker : public Checker {
|
|
public:
|
|
static void *getTag() { static int tag = 0; return &tag; }
|
|
virtual bool EvalCallExpr(CheckerContext &C, const CallExpr *CE);
|
|
};
|
|
|
|
}
|
|
|
|
void clang::RegisterBuiltinFunctionChecker(GRExprEngine &Eng) {
|
|
Eng.registerCheck(new BuiltinFunctionChecker());
|
|
}
|
|
|
|
bool BuiltinFunctionChecker::EvalCallExpr(CheckerContext &C,const CallExpr *CE){
|
|
const GRState *state = C.getState();
|
|
const Expr *Callee = CE->getCallee();
|
|
SVal L = state->getSVal(Callee);
|
|
const FunctionDecl *FD = L.getAsFunctionDecl();
|
|
|
|
if (!FD)
|
|
return false;
|
|
|
|
unsigned id = FD->getBuiltinID();
|
|
|
|
if (!id)
|
|
return false;
|
|
|
|
switch (id) {
|
|
case Builtin::BI__builtin_expect: {
|
|
// For __builtin_expect, just return the value of the subexpression.
|
|
assert (CE->arg_begin() != CE->arg_end());
|
|
SVal X = state->getSVal(*(CE->arg_begin()));
|
|
C.GenerateNode(state->BindExpr(CE, X));
|
|
return true;
|
|
}
|
|
|
|
case Builtin::BI__builtin_alloca: {
|
|
// FIXME: Refactor into StoreManager itself?
|
|
MemRegionManager& RM = C.getStoreManager().getRegionManager();
|
|
const AllocaRegion* R =
|
|
RM.getAllocaRegion(CE, C.getNodeBuilder().getCurrentBlockCount(),
|
|
C.getPredecessor()->getLocationContext());
|
|
|
|
// Set the extent of the region in bytes. This enables us to use the
|
|
// SVal of the argument directly. If we save the extent in bits, we
|
|
// cannot represent values like symbol*8.
|
|
DefinedOrUnknownSVal Size =
|
|
cast<DefinedOrUnknownSVal>(state->getSVal(*(CE->arg_begin())));
|
|
|
|
ValueManager& ValMgr = C.getValueManager();
|
|
DefinedOrUnknownSVal Extent = R->getExtent(ValMgr);
|
|
|
|
SValuator& SVator = ValMgr.getSValuator();
|
|
DefinedOrUnknownSVal ExtentMatchesSizeArg =
|
|
SVator.EvalEQ(state, Extent, Size);
|
|
state = state->Assume(ExtentMatchesSizeArg, true);
|
|
|
|
C.GenerateNode(state->BindExpr(CE, loc::MemRegionVal(R)));
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|