//===-- Solver.cpp --------------------------------------------------------===// // // The KLEE Symbolic Virtual Machine // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #include "klee/Solver.h" #include "klee/SolverImpl.h" #include "SolverStats.h" #include "STPBuilder.h" #include "MetaSMTBuilder.h" #include "klee/Constraints.h" #include "klee/Expr.h" #include "klee/TimerStatIncrementer.h" #include "klee/util/Assignment.h" #include "klee/util/ExprPPrinter.h" #include "klee/util/ExprUtil.h" #include "klee/Internal/Support/Timer.h" #include "klee/CommandLine.h" #define vc_bvBoolExtract IAMTHESPAWNOFSATAN #include #include #include #include #include #include #include #include #include #include #include "llvm/Support/CommandLine.h" #include "llvm/Support/ErrorHandling.h" llvm::cl::opt IgnoreSolverFailures("ignore-solver-failures", llvm::cl::init(false), llvm::cl::desc("Ignore any solver failures (default=off)")); using namespace klee; #ifdef SUPPORT_METASMT #include #include #include #include #include #include #include using namespace metaSMT; using namespace metaSMT::solver; #endif /* SUPPORT_METASMT */ /***/ SolverImpl::~SolverImpl() { } bool SolverImpl::computeValidity(const Query& query, Solver::Validity &result) { bool isTrue, isFalse; if (!computeTruth(query, isTrue)) return false; if (isTrue) { result = Solver::True; } else { if (!computeTruth(query.negateExpr(), isFalse)) return false; result = isFalse ? Solver::False : Solver::Unknown; } return true; } const char* SolverImpl::getOperationStatusString(SolverRunStatus statusCode) { switch (statusCode) { case SOLVER_RUN_STATUS_SUCCESS_SOLVABLE: return "OPERATION SUCCESSFUL, QUERY IS SOLVABLE"; case SOLVER_RUN_STATUS_SUCCESS_UNSOLVABLE: return "OPERATION SUCCESSFUL, QUERY IS UNSOLVABLE"; case SOLVER_RUN_STATUS_FAILURE: return "OPERATION FAILED"; case SOLVER_RUN_STATUS_TIMEOUT: return "SOLVER TIMEOUT"; case SOLVER_RUN_STATUS_FORK_FAILED: return "FORK FAILED"; case SOLVER_RUN_STATUS_INTERRUPTED: return "SOLVER PROCESS INTERRUPTED"; case SOLVER_RUN_STATUS_UNEXPECTED_EXIT_CODE: return "UNEXPECTED SOLVER PROCESS EXIT CODE"; case SOLVER_RUN_STATUS_WAITPID_FAILED: return "WAITPID FAILED FOR SOLVER PROCESS"; default: return "UNRECOGNIZED OPERATION STATUS"; } } const char *Solver::validity_to_str(Validity v) { switch (v) { default: return "Unknown"; case True: return "True"; case False: return "False"; } } Solver::~Solver() { delete impl; } char *Solver::getConstraintLog(const Query& query) { return impl->getConstraintLog(query); } void Solver::setCoreSolverTimeout(double timeout) { impl->setCoreSolverTimeout(timeout); } bool Solver::evaluate(const Query& query, Validity &result) { assert(query.expr->getWidth() == Expr::Bool && "Invalid expression type!"); // Maintain invariants implementations expect. if (ConstantExpr *CE = dyn_cast(query.expr)) { result = CE->isTrue() ? True : False; return true; } return impl->computeValidity(query, result); } bool Solver::mustBeTrue(const Query& query, bool &result) { assert(query.expr->getWidth() == Expr::Bool && "Invalid expression type!"); // Maintain invariants implementations expect. if (ConstantExpr *CE = dyn_cast(query.expr)) { result = CE->isTrue() ? true : false; return true; } return impl->computeTruth(query, result); } bool Solver::mustBeFalse(const Query& query, bool &result) { return mustBeTrue(query.negateExpr(), result); } bool Solver::mayBeTrue(const Query& query, bool &result) { bool res; if (!mustBeFalse(query, res)) return false; result = !res; return true; } bool Solver::mayBeFalse(const Query& query, bool &result) { bool res; if (!mustBeTrue(query, res)) return false; result = !res; return true; } bool Solver::getValue(const Query& query, ref &result) { // Maintain invariants implementation expect. if (ConstantExpr *CE = dyn_cast(query.expr)) { result = CE; return true; } // FIXME: Push ConstantExpr requirement down. ref tmp; if (!impl->computeValue(query, tmp)) return false; result = cast(tmp); return true; } bool Solver::getInitialValues(const Query& query, const std::vector &objects, std::vector< std::vector > &values) { bool hasSolution; bool success = impl->computeInitialValues(query, objects, values, hasSolution); // FIXME: Propogate this out. if (!hasSolution) return false; return success; } std::pair< ref, ref > Solver::getRange(const Query& query) { ref e = query.expr; Expr::Width width = e->getWidth(); uint64_t min, max; if (width==1) { Solver::Validity result; if (!evaluate(query, result)) assert(0 && "computeValidity failed"); switch (result) { case Solver::True: min = max = 1; break; case Solver::False: min = max = 0; break; default: min = 0, max = 1; break; } } else if (ConstantExpr *CE = dyn_cast(e)) { min = max = CE->getZExtValue(); } else { // binary search for # of useful bits uint64_t lo=0, hi=width, mid, bits=0; while (lo &result); bool computeInitialValues(const Query&, const std::vector &objects, std::vector< std::vector > &values, bool &hasSolution); SolverRunStatus getOperationStatusCode(); char *getConstraintLog(const Query&); void setCoreSolverTimeout(double timeout); }; bool ValidatingSolver::computeTruth(const Query& query, bool &isValid) { bool answer; if (!solver->impl->computeTruth(query, isValid)) return false; if (!oracle->impl->computeTruth(query, answer)) return false; if (isValid != answer) assert(0 && "invalid solver result (computeTruth)"); return true; } bool ValidatingSolver::computeValidity(const Query& query, Solver::Validity &result) { Solver::Validity answer; if (!solver->impl->computeValidity(query, result)) return false; if (!oracle->impl->computeValidity(query, answer)) return false; if (result != answer) assert(0 && "invalid solver result (computeValidity)"); return true; } bool ValidatingSolver::computeValue(const Query& query, ref &result) { bool answer; if (!solver->impl->computeValue(query, result)) return false; // We don't want to compare, but just make sure this is a legal // solution. if (!oracle->impl->computeTruth(query.withExpr(NeExpr::create(query.expr, result)), answer)) return false; if (answer) assert(0 && "invalid solver result (computeValue)"); return true; } bool ValidatingSolver::computeInitialValues(const Query& query, const std::vector &objects, std::vector< std::vector > &values, bool &hasSolution) { bool answer; if (!solver->impl->computeInitialValues(query, objects, values, hasSolution)) return false; if (hasSolution) { // Assert the bindings as constraints, and verify that the // conjunction of the actual constraints is satisfiable. std::vector< ref > bindings; for (unsigned i = 0; i != values.size(); ++i) { const Array *array = objects[i]; assert(array); for (unsigned j=0; jsize; j++) { unsigned char value = values[i][j]; bindings.push_back(EqExpr::create(ReadExpr::create(UpdateList(array, 0), ConstantExpr::alloc(j, array->getDomain())), ConstantExpr::alloc(value, array->getRange()))); } } ConstraintManager tmp(bindings); ref constraints = Expr::createIsZero(query.expr); for (ConstraintManager::const_iterator it = query.constraints.begin(), ie = query.constraints.end(); it != ie; ++it) constraints = AndExpr::create(constraints, *it); if (!oracle->impl->computeTruth(Query(tmp, constraints), answer)) return false; if (!answer) assert(0 && "invalid solver result (computeInitialValues)"); } else { if (!oracle->impl->computeTruth(query, answer)) return false; if (!answer) assert(0 && "invalid solver result (computeInitialValues)"); } return true; } SolverImpl::SolverRunStatus ValidatingSolver::getOperationStatusCode() { return solver->impl->getOperationStatusCode(); } char *ValidatingSolver::getConstraintLog(const Query& query) { return solver->impl->getConstraintLog(query); } void ValidatingSolver::setCoreSolverTimeout(double timeout) { solver->impl->setCoreSolverTimeout(timeout); } Solver *klee::createValidatingSolver(Solver *s, Solver *oracle) { return new Solver(new ValidatingSolver(s, oracle)); } /***/ class DummySolverImpl : public SolverImpl { public: DummySolverImpl() {} bool computeValidity(const Query&, Solver::Validity &result) { ++stats::queries; // FIXME: We should have stats::queriesFail; return false; } bool computeTruth(const Query&, bool &isValid) { ++stats::queries; // FIXME: We should have stats::queriesFail; return false; } bool computeValue(const Query&, ref &result) { ++stats::queries; ++stats::queryCounterexamples; return false; } bool computeInitialValues(const Query&, const std::vector &objects, std::vector< std::vector > &values, bool &hasSolution) { ++stats::queries; ++stats::queryCounterexamples; return false; } SolverRunStatus getOperationStatusCode() { return SOLVER_RUN_STATUS_FAILURE; } }; Solver *klee::createDummySolver() { return new Solver(new DummySolverImpl()); } /***/ class STPSolverImpl : public SolverImpl { private: VC vc; STPBuilder *builder; double timeout; bool useForkedSTP; SolverRunStatus runStatusCode; public: STPSolverImpl(bool _useForkedSTP, bool _optimizeDivides = true); ~STPSolverImpl(); char *getConstraintLog(const Query&); void setCoreSolverTimeout(double _timeout) { timeout = _timeout; } bool computeTruth(const Query&, bool &isValid); bool computeValue(const Query&, ref &result); bool computeInitialValues(const Query&, const std::vector &objects, std::vector< std::vector > &values, bool &hasSolution); SolverRunStatus getOperationStatusCode(); }; static unsigned char *shared_memory_ptr; static int shared_memory_id = 0; // Darwin by default has a very small limit on the maximum amount of shared // memory, which will quickly be exhausted by KLEE running its tests in // parallel. For now, we work around this by just requesting a smaller size -- // in practice users hitting this limit on counterexample sizes probably already // are hitting more serious scalability issues. #ifdef __APPLE__ static const unsigned shared_memory_size = 1<<16; #else static const unsigned shared_memory_size = 1<<20; #endif static void stp_error_handler(const char* err_msg) { fprintf(stderr, "error: STP Error: %s\n", err_msg); abort(); } STPSolverImpl::STPSolverImpl(bool _useForkedSTP, bool _optimizeDivides) : vc(vc_createValidityChecker()), builder(new STPBuilder(vc, _optimizeDivides)), timeout(0.0), useForkedSTP(_useForkedSTP), runStatusCode(SOLVER_RUN_STATUS_FAILURE) { assert(vc && "unable to create validity checker"); assert(builder && "unable to create STPBuilder"); // In newer versions of STP, a memory management mechanism has been // introduced that automatically invalidates certain C interface // pointers at vc_Destroy time. This caused double-free errors // due to the ExprHandle destructor also attempting to invalidate // the pointers using vc_DeleteExpr. By setting EXPRDELETE to 0 // we restore the old behaviour. vc_setInterfaceFlags(vc, EXPRDELETE, 0); vc_registerErrorHandler(::stp_error_handler); if (useForkedSTP) { assert(shared_memory_id == 0 && "shared memory id already allocated"); shared_memory_id = shmget(IPC_PRIVATE, shared_memory_size, IPC_CREAT | 0700); if (shared_memory_id < 0) llvm::report_fatal_error("unable to allocate shared memory region"); shared_memory_ptr = (unsigned char*) shmat(shared_memory_id, NULL, 0); if (shared_memory_ptr == (void*)-1) llvm::report_fatal_error("unable to attach shared memory region"); shmctl(shared_memory_id, IPC_RMID, NULL); } } STPSolverImpl::~STPSolverImpl() { // Detach the memory region. shmdt(shared_memory_ptr); shared_memory_ptr = 0; shared_memory_id = 0; delete builder; vc_Destroy(vc); } /***/ STPSolver::STPSolver(bool useForkedSTP, bool optimizeDivides) : Solver(new STPSolverImpl(useForkedSTP, optimizeDivides)) { } char *STPSolver::getConstraintLog(const Query &query) { return impl->getConstraintLog(query); } void STPSolver::setCoreSolverTimeout(double timeout) { impl->setCoreSolverTimeout(timeout); } /***/ char *STPSolverImpl::getConstraintLog(const Query &query) { vc_push(vc); for (std::vector< ref >::const_iterator it = query.constraints.begin(), ie = query.constraints.end(); it != ie; ++it) vc_assertFormula(vc, builder->construct(*it)); assert(query.expr == ConstantExpr::alloc(0, Expr::Bool) && "Unexpected expression in query!"); char *buffer; unsigned long length; vc_printQueryStateToBuffer(vc, builder->getFalse(), &buffer, &length, false); vc_pop(vc); return buffer; } bool STPSolverImpl::computeTruth(const Query& query, bool &isValid) { std::vector objects; std::vector< std::vector > values; bool hasSolution; if (!computeInitialValues(query, objects, values, hasSolution)) return false; isValid = !hasSolution; return true; } bool STPSolverImpl::computeValue(const Query& query, ref &result) { std::vector objects; std::vector< std::vector > values; bool hasSolution; // Find the object used in the expression, and compute an assignment // for them. findSymbolicObjects(query.expr, objects); if (!computeInitialValues(query.withFalse(), objects, values, hasSolution)) return false; assert(hasSolution && "state has invalid constraint set"); // Evaluate the expression with the computed assignment. Assignment a(objects, values); result = a.evaluate(query.expr); return true; } static SolverImpl::SolverRunStatus runAndGetCex(::VC vc, STPBuilder *builder, ::VCExpr q, const std::vector &objects, std::vector< std::vector > &values, bool &hasSolution) { // XXX I want to be able to timeout here, safely hasSolution = !vc_query(vc, q); if (hasSolution) { values.reserve(objects.size()); for (std::vector::const_iterator it = objects.begin(), ie = objects.end(); it != ie; ++it) { const Array *array = *it; std::vector data; data.reserve(array->size); for (unsigned offset = 0; offset < array->size; offset++) { ExprHandle counter = vc_getCounterExample(vc, builder->getInitialRead(array, offset)); unsigned char val = getBVUnsigned(counter); data.push_back(val); } values.push_back(data); } } if (true == hasSolution) { return SolverImpl::SOLVER_RUN_STATUS_SUCCESS_SOLVABLE; } else { return SolverImpl::SOLVER_RUN_STATUS_SUCCESS_UNSOLVABLE; } } static void stpTimeoutHandler(int x) { _exit(52); } static SolverImpl::SolverRunStatus runAndGetCexForked(::VC vc, STPBuilder *builder, ::VCExpr q, const std::vector &objects, std::vector< std::vector > &values, bool &hasSolution, double timeout) { unsigned char *pos = shared_memory_ptr; unsigned sum = 0; for (std::vector::const_iterator it = objects.begin(), ie = objects.end(); it != ie; ++it) sum += (*it)->size; if (sum >= shared_memory_size) llvm::report_fatal_error("not enough shared memory for counterexample"); fflush(stdout); fflush(stderr); int pid = fork(); if (pid==-1) { fprintf(stderr, "ERROR: fork failed (for STP)"); if (!IgnoreSolverFailures) exit(1); return SolverImpl::SOLVER_RUN_STATUS_FORK_FAILED; } if (pid == 0) { if (timeout) { ::alarm(0); /* Turn off alarm so we can safely set signal handler */ ::signal(SIGALRM, stpTimeoutHandler); ::alarm(std::max(1, (int)timeout)); } unsigned res = vc_query(vc, q); if (!res) { for (std::vector::const_iterator it = objects.begin(), ie = objects.end(); it != ie; ++it) { const Array *array = *it; for (unsigned offset = 0; offset < array->size; offset++) { ExprHandle counter = vc_getCounterExample(vc, builder->getInitialRead(array, offset)); *pos++ = getBVUnsigned(counter); } } } _exit(res); } else { int status; pid_t res; do { res = waitpid(pid, &status, 0); } while (res < 0 && errno == EINTR); if (res < 0) { fprintf(stderr, "ERROR: waitpid() for STP failed"); if (!IgnoreSolverFailures) exit(1); return SolverImpl::SOLVER_RUN_STATUS_WAITPID_FAILED; } // From timed_run.py: It appears that linux at least will on // "occasion" return a status when the process was terminated by a // signal, so test signal first. if (WIFSIGNALED(status) || !WIFEXITED(status)) { fprintf(stderr, "ERROR: STP did not return successfully. Most likely you forgot to run 'ulimit -s unlimited'\n"); if (!IgnoreSolverFailures) { exit(1); } return SolverImpl::SOLVER_RUN_STATUS_INTERRUPTED; } int exitcode = WEXITSTATUS(status); if (exitcode==0) { hasSolution = true; } else if (exitcode==1) { hasSolution = false; } else if (exitcode==52) { fprintf(stderr, "error: STP timed out"); // mark that a timeout occurred return SolverImpl::SOLVER_RUN_STATUS_TIMEOUT; } else { fprintf(stderr, "error: STP did not return a recognized code"); if (!IgnoreSolverFailures) exit(1); return SolverImpl::SOLVER_RUN_STATUS_UNEXPECTED_EXIT_CODE; } if (hasSolution) { values = std::vector< std::vector >(objects.size()); unsigned i=0; for (std::vector::const_iterator it = objects.begin(), ie = objects.end(); it != ie; ++it) { const Array *array = *it; std::vector &data = values[i++]; data.insert(data.begin(), pos, pos + array->size); pos += array->size; } } if (true == hasSolution) { return SolverImpl::SOLVER_RUN_STATUS_SUCCESS_SOLVABLE; } else { return SolverImpl::SOLVER_RUN_STATUS_SUCCESS_UNSOLVABLE; } } } bool STPSolverImpl::computeInitialValues(const Query &query, const std::vector &objects, std::vector< std::vector > &values, bool &hasSolution) { runStatusCode = SOLVER_RUN_STATUS_FAILURE; TimerStatIncrementer t(stats::queryTime); vc_push(vc); for (ConstraintManager::const_iterator it = query.constraints.begin(), ie = query.constraints.end(); it != ie; ++it) vc_assertFormula(vc, builder->construct(*it)); ++stats::queries; ++stats::queryCounterexamples; ExprHandle stp_e = builder->construct(query.expr); if (0) { char *buf; unsigned long len; vc_printQueryStateToBuffer(vc, stp_e, &buf, &len, false); fprintf(stderr, "note: STP query: %.*s\n", (unsigned) len, buf); } bool success; if (useForkedSTP) { runStatusCode = runAndGetCexForked(vc, builder, stp_e, objects, values, hasSolution, timeout); success = ((SOLVER_RUN_STATUS_SUCCESS_SOLVABLE == runStatusCode) || (SOLVER_RUN_STATUS_SUCCESS_UNSOLVABLE == runStatusCode)); } else { runStatusCode = runAndGetCex(vc, builder, stp_e, objects, values, hasSolution); success = true; } if (success) { if (hasSolution) ++stats::queriesInvalid; else ++stats::queriesValid; } vc_pop(vc); return success; } SolverImpl::SolverRunStatus STPSolverImpl::getOperationStatusCode() { return runStatusCode; } #ifdef SUPPORT_METASMT // ------------------------------------- MetaSMTSolverImpl class declaration ------------------------------ template class MetaSMTSolverImpl : public SolverImpl { private: SolverContext _meta_solver; MetaSMTSolver *_solver; MetaSMTBuilder *_builder; double _timeout; bool _useForked; SolverRunStatus _runStatusCode; public: MetaSMTSolverImpl(MetaSMTSolver *solver, bool useForked, bool optimizeDivides); virtual ~MetaSMTSolverImpl(); char *getConstraintLog(const Query&); void setCoreSolverTimeout(double timeout) { _timeout = timeout; } bool computeTruth(const Query&, bool &isValid); bool computeValue(const Query&, ref &result); bool computeInitialValues(const Query &query, const std::vector &objects, std::vector< std::vector > &values, bool &hasSolution); SolverImpl::SolverRunStatus runAndGetCex(ref query_expr, const std::vector &objects, std::vector< std::vector > &values, bool &hasSolution); SolverImpl::SolverRunStatus runAndGetCexForked(const Query &query, const std::vector &objects, std::vector< std::vector > &values, bool &hasSolution, double timeout); SolverRunStatus getOperationStatusCode(); SolverContext& get_meta_solver() { return(_meta_solver); }; }; // ------------------------------------- MetaSMTSolver methods -------------------------------------------- template MetaSMTSolver::MetaSMTSolver(bool useForked, bool optimizeDivides) : Solver(new MetaSMTSolverImpl(this, useForked, optimizeDivides)) { } template MetaSMTSolver::~MetaSMTSolver() { } template char *MetaSMTSolver::getConstraintLog(const Query& query) { return(impl->getConstraintLog(query)); } template void MetaSMTSolver::setCoreSolverTimeout(double timeout) { impl->setCoreSolverTimeout(timeout); } // ------------------------------------- MetaSMTSolverImpl methods ---------------------------------------- template MetaSMTSolverImpl::MetaSMTSolverImpl(MetaSMTSolver *solver, bool useForked, bool optimizeDivides) : _solver(solver), _builder(new MetaSMTBuilder(_meta_solver, optimizeDivides)), _timeout(0.0), _useForked(useForked) { assert(_solver && "unable to create MetaSMTSolver"); assert(_builder && "unable to create MetaSMTBuilder"); if (_useForked) { shared_memory_id = shmget(IPC_PRIVATE, shared_memory_size, IPC_CREAT | 0700); assert(shared_memory_id >= 0 && "shmget failed"); shared_memory_ptr = (unsigned char*) shmat(shared_memory_id, NULL, 0); assert(shared_memory_ptr != (void*) -1 && "shmat failed"); shmctl(shared_memory_id, IPC_RMID, NULL); } } template MetaSMTSolverImpl::~MetaSMTSolverImpl() { } template char *MetaSMTSolverImpl::getConstraintLog(const Query&) { const char* msg = "Not supported"; char *buf = new char[strlen(msg) + 1]; strcpy(buf, msg); return(buf); } template bool MetaSMTSolverImpl::computeTruth(const Query& query, bool &isValid) { bool success = false; std::vector objects; std::vector< std::vector > values; bool hasSolution; if (computeInitialValues(query, objects, values, hasSolution)) { // query.expr is valid iff !query.expr is not satisfiable isValid = !hasSolution; success = true; } return(success); } template bool MetaSMTSolverImpl::computeValue(const Query& query, ref &result) { bool success = false; std::vector objects; std::vector< std::vector > values; bool hasSolution; // Find the object used in the expression, and compute an assignment for them. findSymbolicObjects(query.expr, objects); if (computeInitialValues(query.withFalse(), objects, values, hasSolution)) { assert(hasSolution && "state has invalid constraint set"); // Evaluate the expression with the computed assignment. Assignment a(objects, values); result = a.evaluate(query.expr); success = true; } return(success); } template bool MetaSMTSolverImpl::computeInitialValues(const Query &query, const std::vector &objects, std::vector< std::vector > &values, bool &hasSolution) { _runStatusCode = SOLVER_RUN_STATUS_FAILURE; TimerStatIncrementer t(stats::queryTime); assert(_builder); /* * FIXME push() and pop() work for Z3 but not for Boolector. * If using Z3, use push() and pop() and assert constraints. * If using Boolector, assume constrainsts instead of asserting them. */ //push(_meta_solver); if (!_useForked) { for (ConstraintManager::const_iterator it = query.constraints.begin(), ie = query.constraints.end(); it != ie; ++it) { //assertion(_meta_solver, _builder->construct(*it)); assumption(_meta_solver, _builder->construct(*it)); } } ++stats::queries; ++stats::queryCounterexamples; bool success = true; if (_useForked) { _runStatusCode = runAndGetCexForked(query, objects, values, hasSolution, _timeout); success = ((SOLVER_RUN_STATUS_SUCCESS_SOLVABLE == _runStatusCode) || (SOLVER_RUN_STATUS_SUCCESS_UNSOLVABLE == _runStatusCode)); } else { _runStatusCode = runAndGetCex(query.expr, objects, values, hasSolution); success = true; } if (success) { if (hasSolution) { ++stats::queriesInvalid; } else { ++stats::queriesValid; } } //pop(_meta_solver); return(success); } template SolverImpl::SolverRunStatus MetaSMTSolverImpl::runAndGetCex(ref query_expr, const std::vector &objects, std::vector< std::vector > &values, bool &hasSolution) { // assume the negation of the query assumption(_meta_solver, _builder->construct(Expr::createIsZero(query_expr))); hasSolution = solve(_meta_solver); if (hasSolution) { values.reserve(objects.size()); for (std::vector::const_iterator it = objects.begin(), ie = objects.end(); it != ie; ++it) { const Array *array = *it; assert(array); typename SolverContext::result_type array_exp = _builder->getInitialArray(array); std::vector data; data.reserve(array->size); for (unsigned offset = 0; offset < array->size; offset++) { typename SolverContext::result_type elem_exp = evaluate( _meta_solver, metaSMT::logic::Array::select(array_exp, bvuint(offset, array->getDomain()))); unsigned char elem_value = metaSMT::read_value(_meta_solver, elem_exp); data.push_back(elem_value); } values.push_back(data); } } if (true == hasSolution) { return(SolverImpl::SOLVER_RUN_STATUS_SUCCESS_SOLVABLE); } else { return(SolverImpl::SOLVER_RUN_STATUS_SUCCESS_UNSOLVABLE); } } static void metaSMTTimeoutHandler(int x) { _exit(52); } template SolverImpl::SolverRunStatus MetaSMTSolverImpl::runAndGetCexForked(const Query &query, const std::vector &objects, std::vector< std::vector > &values, bool &hasSolution, double timeout) { unsigned char *pos = shared_memory_ptr; unsigned sum = 0; for (std::vector::const_iterator it = objects.begin(), ie = objects.end(); it != ie; ++it) { sum += (*it)->size; } // sum += sizeof(uint64_t); sum += sizeof(stats::queryConstructs); assert(sum < shared_memory_size && "not enough shared memory for counterexample"); fflush(stdout); fflush(stderr); int pid = fork(); if (pid == -1) { fprintf(stderr, "error: fork failed (for metaSMT)"); return(SolverImpl::SOLVER_RUN_STATUS_FORK_FAILED); } if (pid == 0) { if (timeout) { ::alarm(0); /* Turn off alarm so we can safely set signal handler */ ::signal(SIGALRM, metaSMTTimeoutHandler); ::alarm(std::max(1, (int) timeout)); } for (ConstraintManager::const_iterator it = query.constraints.begin(), ie = query.constraints.end(); it != ie; ++it) { assertion(_meta_solver, _builder->construct(*it)); //assumption(_meta_solver, _builder->construct(*it)); } std::vector< std::vector > aux_arr_exprs; if (UseMetaSMT == METASMT_BACKEND_BOOLECTOR) { for (std::vector::const_iterator it = objects.begin(), ie = objects.end(); it != ie; ++it) { std::vector aux_arr; const Array *array = *it; assert(array); typename SolverContext::result_type array_exp = _builder->getInitialArray(array); for (unsigned offset = 0; offset < array->size; offset++) { typename SolverContext::result_type elem_exp = evaluate( _meta_solver, metaSMT::logic::Array::select(array_exp, bvuint(offset, array->getDomain()))); aux_arr.push_back(elem_exp); } aux_arr_exprs.push_back(aux_arr); } assert(aux_arr_exprs.size() == objects.size()); } // assume the negation of the query // can be also asserted instead of assumed as we are in a child process assumption(_meta_solver, _builder->construct(Expr::createIsZero(query.expr))); unsigned res = solve(_meta_solver); if (res) { if (UseMetaSMT != METASMT_BACKEND_BOOLECTOR) { for (std::vector::const_iterator it = objects.begin(), ie = objects.end(); it != ie; ++it) { const Array *array = *it; assert(array); typename SolverContext::result_type array_exp = _builder->getInitialArray(array); for (unsigned offset = 0; offset < array->size; offset++) { typename SolverContext::result_type elem_exp = evaluate( _meta_solver, metaSMT::logic::Array::select(array_exp, bvuint(offset, array->getDomain()))); unsigned char elem_value = metaSMT::read_value(_meta_solver, elem_exp); *pos++ = elem_value; } } } else { typename std::vector< std::vector >::const_iterator eit = aux_arr_exprs.begin(), eie = aux_arr_exprs.end(); for (std::vector::const_iterator it = objects.begin(), ie = objects.end(); it != ie, eit != eie; ++it, ++eit) { const Array *array = *it; const std::vector& arr_exp = *eit; assert(array); assert(array->size == arr_exp.size()); for (unsigned offset = 0; offset < array->size; offset++) { unsigned char elem_value = metaSMT::read_value(_meta_solver, arr_exp[offset]); *pos++ = elem_value; } } } } assert((uint64_t*) pos); *((uint64_t*) pos) = stats::queryConstructs; _exit(!res); } else { int status; pid_t res; do { res = waitpid(pid, &status, 0); } while (res < 0 && errno == EINTR); if (res < 0) { fprintf(stderr, "error: waitpid() for metaSMT failed"); return(SolverImpl::SOLVER_RUN_STATUS_WAITPID_FAILED); } // From timed_run.py: It appears that linux at least will on // "occasion" return a status when the process was terminated by a // signal, so test signal first. if (WIFSIGNALED(status) || !WIFEXITED(status)) { fprintf(stderr, "error: metaSMT did not return successfully (status = %d) \n", WTERMSIG(status)); return(SolverImpl::SOLVER_RUN_STATUS_INTERRUPTED); } int exitcode = WEXITSTATUS(status); if (exitcode == 0) { hasSolution = true; } else if (exitcode == 1) { hasSolution = false; } else if (exitcode == 52) { fprintf(stderr, "error: metaSMT timed out"); return(SolverImpl::SOLVER_RUN_STATUS_TIMEOUT); } else { fprintf(stderr, "error: metaSMT did not return a recognized code"); return(SolverImpl::SOLVER_RUN_STATUS_UNEXPECTED_EXIT_CODE); } if (hasSolution) { values = std::vector< std::vector >(objects.size()); unsigned i = 0; for (std::vector::const_iterator it = objects.begin(), ie = objects.end(); it != ie; ++it) { const Array *array = *it; assert(array); std::vector &data = values[i++]; data.insert(data.begin(), pos, pos + array->size); pos += array->size; } } stats::queryConstructs += (*((uint64_t*) pos) - stats::queryConstructs); if (true == hasSolution) { return SolverImpl::SOLVER_RUN_STATUS_SUCCESS_SOLVABLE; } else { return SolverImpl::SOLVER_RUN_STATUS_SUCCESS_UNSOLVABLE; } } } template SolverImpl::SolverRunStatus MetaSMTSolverImpl::getOperationStatusCode() { return _runStatusCode; } template class MetaSMTSolver< DirectSolver_Context < Boolector> >; template class MetaSMTSolver< DirectSolver_Context < Z3_Backend> >; template class MetaSMTSolver< DirectSolver_Context < STP_Backend> >; #endif /* SUPPORT_METASMT */ /// void Query::dump() const { llvm::errs() << "Constraints [\n"; for (ConstraintManager::const_iterator i = constraints.begin(); i != constraints.end(); i++) { (*i)->dump(); } llvm::errs() << "]\n"; llvm::errs() << "Query [\n"; expr->dump(); llvm::errs() << "]\n"; }