//===-- 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 "klee/TimerStatIncrementer.h" #include "CoreStats.h" 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; } TimerStatIncrementer timer(stats::solverTime); if (simplifyExprs) expr = state.constraints.simplifyExpr(expr); bool success = solver->evaluate(Query(state.constraints, expr), result); state.queryCost += timer.check() / 1e6; 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; } TimerStatIncrementer timer(stats::solverTime); if (simplifyExprs) expr = state.constraints.simplifyExpr(expr); bool success = solver->mustBeTrue(Query(state.constraints, expr), result); state.queryCost += timer.check() / 1e6; 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; } TimerStatIncrementer timer(stats::solverTime); if (simplifyExprs) expr = state.constraints.simplifyExpr(expr); bool success = solver->getValue(Query(state.constraints, expr), result); state.queryCost += timer.check() / 1e6; return success; } bool TimingSolver::getInitialValues(const ExecutionState& state, const std::vector &objects, std::vector< std::vector > &result) { if (objects.empty()) return true; TimerStatIncrementer timer(stats::solverTime); bool success = solver->getInitialValues(Query(state.constraints, ConstantExpr::alloc(0, Expr::Bool)), objects, result); state.queryCost += timer.check() / 1e6; return success; } std::pair< ref, ref > TimingSolver::getRange(const ExecutionState& state, ref expr) { return solver->getRange(Query(state.constraints, expr)); }