//===-- TimingSolver.cpp --------------------------------------------------===// // // The KLEE Symbolic Virtual Machine // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #include "TimingSolver.h" #include "klee/Config/Version.h" #include "klee/ExecutionState.h" #include "klee/Solver.h" #include "klee/Statistics.h" #include "CoreStats.h" #if LLVM_VERSION_CODE < LLVM_VERSION(2, 9) #include "llvm/System/Process.h" #else #include "llvm/Support/Process.h" #endif using namespace klee; using namespace llvm; /***/ bool TimingSolver::evaluate(const ExecutionState& state, ref expr, Solver::Validity &result) { // Fast path, to avoid timer and OS overhead. if (ConstantExpr *CE = dyn_cast(expr)) { result = CE->isTrue() ? Solver::True : Solver::False; return true; } sys::TimeValue now(0,0),user(0,0),delta(0,0),sys(0,0); sys::Process::GetTimeUsage(now,user,sys); if (simplifyExprs) expr = state.constraints.simplifyExpr(expr); bool success = solver->evaluate(Query(state.constraints, expr), result); sys::Process::GetTimeUsage(delta,user,sys); delta -= now; stats::solverTime += delta.usec(); state.queryCost += delta.usec()/1000000.; return success; } bool TimingSolver::mustBeTrue(const ExecutionState& state, ref expr, bool &result) { // Fast path, to avoid timer and OS overhead. if (ConstantExpr *CE = dyn_cast(expr)) { result = CE->isTrue() ? true : false; return true; } sys::TimeValue now(0,0),user(0,0),delta(0,0),sys(0,0); sys::Process::GetTimeUsage(now,user,sys); if (simplifyExprs) expr = state.constraints.simplifyExpr(expr); bool success = solver->mustBeTrue(Query(state.constraints, expr), result); sys::Process::GetTimeUsage(delta,user,sys); delta -= now; stats::solverTime += delta.usec(); state.queryCost += delta.usec()/1000000.; return success; } bool TimingSolver::mustBeFalse(const ExecutionState& state, ref expr, bool &result) { return mustBeTrue(state, Expr::createIsZero(expr), result); } bool TimingSolver::mayBeTrue(const ExecutionState& state, ref expr, bool &result) { bool res; if (!mustBeFalse(state, expr, res)) return false; result = !res; return true; } bool TimingSolver::mayBeFalse(const ExecutionState& state, ref expr, bool &result) { bool res; if (!mustBeTrue(state, expr, res)) return false; result = !res; return true; } bool TimingSolver::getValue(const ExecutionState& state, ref expr, ref &result) { // Fast path, to avoid timer and OS overhead. if (ConstantExpr *CE = dyn_cast(expr)) { result = CE; return true; } sys::TimeValue now(0,0),user(0,0),delta(0,0),sys(0,0); sys::Process::GetTimeUsage(now,user,sys); if (simplifyExprs) expr = state.constraints.simplifyExpr(expr); bool success = solver->getValue(Query(state.constraints, expr), result); sys::Process::GetTimeUsage(delta,user,sys); delta -= now; stats::solverTime += delta.usec(); state.queryCost += delta.usec()/1000000.; return success; } bool TimingSolver::getInitialValues(const ExecutionState& state, const std::vector &objects, std::vector< std::vector > &result) { if (objects.empty()) return true; sys::TimeValue now(0,0),user(0,0),delta(0,0),sys(0,0); sys::Process::GetTimeUsage(now,user,sys); bool success = solver->getInitialValues(Query(state.constraints, ConstantExpr::alloc(0, Expr::Bool)), objects, result); sys::Process::GetTimeUsage(delta,user,sys); delta -= now; stats::solverTime += delta.usec(); state.queryCost += delta.usec()/1000000.; return success; } std::pair< ref, ref > TimingSolver::getRange(const ExecutionState& state, ref expr) { return solver->getRange(Query(state.constraints, expr)); }