|
@ -4,31 +4,6 @@ |
|
|
|
|
|
|
|
|
namespace storm { |
|
|
namespace storm { |
|
|
namespace solver { |
|
|
namespace solver { |
|
|
#ifdef STORM_HAVE_MSAT
|
|
|
|
|
|
MathSatSmtSolver::MathSatModelReference::Z3ModelReference(z3::model &m, storm::adapters::Z3ExpressionAdapter &adapter) : m_model(m), m_adapter(adapter) { |
|
|
|
|
|
|
|
|
|
|
|
} |
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
bool MathSatSmtSolver::Z3ModelReference::getBooleanValue(std::string const& name) const { |
|
|
|
|
|
#ifdef STORM_HAVE_MSAT
|
|
|
|
|
|
z3::expr z3Expr = this->m_adapter.translateExpression(storm::expressions::Expression::createBooleanVariable(name)); |
|
|
|
|
|
z3::expr z3ExprValuation = m_model.eval(z3Expr, true); |
|
|
|
|
|
return this->m_adapter.translateExpression(z3ExprValuation).evaluateAsBool(); |
|
|
|
|
|
#else
|
|
|
|
|
|
LOG_THROW(false, storm::exceptions::NotImplementedException, "StoRM is compiled without MathSat support."); |
|
|
|
|
|
#endif
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
int_fast64_t MathSatSmtSolver::Z3ModelReference::getIntegerValue(std::string const& name) const { |
|
|
|
|
|
#ifdef STORM_HAVE_MSAT
|
|
|
|
|
|
z3::expr z3Expr = this->m_adapter.translateExpression(storm::expressions::Expression::createIntegerVariable(name)); |
|
|
|
|
|
z3::expr z3ExprValuation = m_model.eval(z3Expr, true); |
|
|
|
|
|
return this->m_adapter.translateExpression(z3ExprValuation).evaluateAsInt(); |
|
|
|
|
|
#else
|
|
|
|
|
|
LOG_THROW(false, storm::exceptions::NotImplementedException, "StoRM is compiled without MathSat support."); |
|
|
|
|
|
#endif
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
MathSatSmtSolver::MathSatSmtSolver(Options options) |
|
|
MathSatSmtSolver::MathSatSmtSolver(Options options) |
|
|
#ifdef STORM_HAVE_MSAT
|
|
|
#ifdef STORM_HAVE_MSAT
|
|
@ -40,12 +15,11 @@ namespace storm { |
|
|
m_cfg = msat_create_config(); |
|
|
m_cfg = msat_create_config(); |
|
|
|
|
|
|
|
|
if (static_cast<int>(options)& static_cast<int>(Options::InterpolantComputation)) { |
|
|
if (static_cast<int>(options)& static_cast<int>(Options::InterpolantComputation)) { |
|
|
msat_res = msat_set_option(m_cfg, "interpolation", "true"); |
|
|
|
|
|
if (msat_res != 0) { |
|
|
|
|
|
LOG4CPLUS_WARN(logger, "MathSAT returned an error!"); |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
msat_set_option(m_cfg, "interpolation", "true"); |
|
|
} |
|
|
} |
|
|
m_env = msat_create_env(m_cfg); |
|
|
m_env = msat_create_env(m_cfg); |
|
|
|
|
|
|
|
|
|
|
|
m_adapter = new storm::adapters::MathSatExpressionAdapter(m_env, variableToDeclMap); |
|
|
#endif
|
|
|
#endif
|
|
|
} |
|
|
} |
|
|
MathSatSmtSolver::~MathSatSmtSolver() { |
|
|
MathSatSmtSolver::~MathSatSmtSolver() { |
|
@ -94,7 +68,7 @@ namespace storm { |
|
|
void MathSatSmtSolver::assertExpression(storm::expressions::Expression const& e) |
|
|
void MathSatSmtSolver::assertExpression(storm::expressions::Expression const& e) |
|
|
{ |
|
|
{ |
|
|
#ifdef STORM_HAVE_MSAT
|
|
|
#ifdef STORM_HAVE_MSAT
|
|
|
msat_assert_formula(m_env, m_adapter.translateExpression(e, true)); |
|
|
|
|
|
|
|
|
msat_assert_formula(m_env, m_adapter->translateExpression(e, true)); |
|
|
#else
|
|
|
#else
|
|
|
LOG_THROW(false, storm::exceptions::NotImplementedException, "StoRM is compiled without MathSat support."); |
|
|
LOG_THROW(false, storm::exceptions::NotImplementedException, "StoRM is compiled without MathSat support."); |
|
|
#endif
|
|
|
#endif
|
|
@ -129,7 +103,7 @@ namespace storm { |
|
|
mathSatAssumptions.reserve(assumptions.size()); |
|
|
mathSatAssumptions.reserve(assumptions.size()); |
|
|
|
|
|
|
|
|
for (storm::expressions::Expression assumption : assumptions) { |
|
|
for (storm::expressions::Expression assumption : assumptions) { |
|
|
mathSatAssumptions.push_back(this->m_adapter.translateExpression(assumption)); |
|
|
|
|
|
|
|
|
mathSatAssumptions.push_back(this->m_adapter->translateExpression(assumption)); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
switch (msat_solve_with_assumptions(m_env, mathSatAssumptions.data(), mathSatAssumptions.size())) { |
|
|
switch (msat_solve_with_assumptions(m_env, mathSatAssumptions.data(), mathSatAssumptions.size())) { |
|
@ -157,7 +131,7 @@ namespace storm { |
|
|
mathSatAssumptions.reserve(assumptions.size()); |
|
|
mathSatAssumptions.reserve(assumptions.size()); |
|
|
|
|
|
|
|
|
for (storm::expressions::Expression assumption : assumptions) { |
|
|
for (storm::expressions::Expression assumption : assumptions) { |
|
|
mathSatAssumptions.push_back(this->m_adapter.translateExpression(assumption)); |
|
|
|
|
|
|
|
|
mathSatAssumptions.push_back(this->m_adapter->translateExpression(assumption)); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
switch (msat_solve_with_assumptions(m_env, mathSatAssumptions.data(), mathSatAssumptions.size())) { |
|
|
switch (msat_solve_with_assumptions(m_env, mathSatAssumptions.data(), mathSatAssumptions.size())) { |
|
@ -199,7 +173,7 @@ namespace storm { |
|
|
msat_term t, v; |
|
|
msat_term t, v; |
|
|
msat_model_iterator_next(model, &t, &v); |
|
|
msat_model_iterator_next(model, &t, &v); |
|
|
|
|
|
|
|
|
storm::expressions::Expression var_i_interp = this->m_adapter.translateTerm(v); |
|
|
|
|
|
|
|
|
storm::expressions::Expression var_i_interp = this->m_adapter->translateTerm(v); |
|
|
char* name = msat_decl_get_name(msat_term_get_decl(t)); |
|
|
char* name = msat_decl_get_name(msat_term_get_decl(t)); |
|
|
|
|
|
|
|
|
switch (var_i_interp.getReturnType()) { |
|
|
switch (var_i_interp.getReturnType()) { |
|
@ -244,18 +218,27 @@ namespace storm { |
|
|
|
|
|
|
|
|
#ifdef STORM_HAVE_MSAT
|
|
|
#ifdef STORM_HAVE_MSAT
|
|
|
struct AllsatValuationsCallbackUserData { |
|
|
struct AllsatValuationsCallbackUserData { |
|
|
msat_env env; |
|
|
|
|
|
|
|
|
msat_env &env; |
|
|
storm::adapters::MathSatExpressionAdapter &adapter; |
|
|
storm::adapters::MathSatExpressionAdapter &adapter; |
|
|
storm::expressions::SimpleValuation& valuation; |
|
|
|
|
|
uint_fast64_t n; |
|
|
|
|
|
|
|
|
std::function<bool(storm::expressions::SimpleValuation&)> &callback; |
|
|
}; |
|
|
}; |
|
|
|
|
|
|
|
|
int allsatValuationsCallback(msat_term *model, int size, void *user_data) { |
|
|
int allsatValuationsCallback(msat_term *model, int size, void *user_data) { |
|
|
AllsatValuationsCallbackUserData* user = reinterpret_cast<AllsatValuationsCallbackUserData*>(user_data); |
|
|
AllsatValuationsCallbackUserData* user = reinterpret_cast<AllsatValuationsCallbackUserData*>(user_data); |
|
|
++n; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
storm::expressions::SimpleValuation valuation; |
|
|
|
|
|
|
|
|
for (int i = 0; i < size; ++i) { |
|
|
for (int i = 0; i < size; ++i) { |
|
|
///
|
|
|
|
|
|
|
|
|
bool currentTermValue = true; |
|
|
|
|
|
msat_term currentTerm = model[i]; |
|
|
|
|
|
if (msat_term_is_not(user->env, currentTerm)) { |
|
|
|
|
|
currentTerm = msat_term_get_arg(currentTerm, 0); |
|
|
|
|
|
currentTermValue = false; |
|
|
|
|
|
} |
|
|
|
|
|
char* name = msat_decl_get_name(msat_term_get_decl(currentTerm)); |
|
|
|
|
|
std::string name_str(name); |
|
|
|
|
|
valuation.addBooleanIdentifier(name_str, currentTermValue); |
|
|
|
|
|
msat_free(name); |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
#endif
|
|
|
#endif
|
|
@ -264,13 +247,21 @@ namespace storm { |
|
|
uint_fast64_t MathSatSmtSolver::allSat(std::vector<storm::expressions::Expression> const& important, std::function<bool(storm::expressions::SimpleValuation&)> callback) |
|
|
uint_fast64_t MathSatSmtSolver::allSat(std::vector<storm::expressions::Expression> const& important, std::function<bool(storm::expressions::SimpleValuation&)> callback) |
|
|
{ |
|
|
{ |
|
|
#ifdef STORM_HAVE_MSAT
|
|
|
#ifdef STORM_HAVE_MSAT
|
|
|
|
|
|
std::vector<msat_term> msatImportant; |
|
|
|
|
|
msatImportant.reserve(important.size()); |
|
|
|
|
|
|
|
|
for (storm::expressions::Expression e : important) { |
|
|
for (storm::expressions::Expression e : important) { |
|
|
if (!e.isVariable()) { |
|
|
if (!e.isVariable()) { |
|
|
throw storm::exceptions::InvalidArgumentException() << "The important expressions for AllSat must be atoms, i.e. variable expressions."; |
|
|
throw storm::exceptions::InvalidArgumentException() << "The important expressions for AllSat must be atoms, i.e. variable expressions."; |
|
|
} |
|
|
} |
|
|
|
|
|
msatImportant.push_back(m_adapter->translateExpression(e)); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
AllsatValuationsCallbackUserData allSatUserData; |
|
|
|
|
|
allSatUserData.adapter = m_adapter; |
|
|
|
|
|
allSatUserData.env = m_env; |
|
|
|
|
|
allSatUserData.callback = callback; |
|
|
|
|
|
int numModels = msat_all_sat(m_env, msatImportant.data(), msatImportant.size(), &allsatValuationsCallback, &allSatUserData); |
|
|
|
|
|
|
|
|
return numModels; |
|
|
return numModels; |
|
|
#else
|
|
|
#else
|
|
@ -281,13 +272,7 @@ namespace storm { |
|
|
uint_fast64_t MathSatSmtSolver::allSat(std::function<bool(SmtSolver::ModelReference&)> callback, std::vector<storm::expressions::Expression> const& important) |
|
|
uint_fast64_t MathSatSmtSolver::allSat(std::function<bool(SmtSolver::ModelReference&)> callback, std::vector<storm::expressions::Expression> const& important) |
|
|
{ |
|
|
{ |
|
|
#ifdef STORM_HAVE_MSAT
|
|
|
#ifdef STORM_HAVE_MSAT
|
|
|
for (storm::expressions::Expression e : important) { |
|
|
|
|
|
if (!e.isVariable()) { |
|
|
|
|
|
throw storm::exceptions::InvalidArgumentException() << "The important expressions for AllSat must be atoms, i.e. variable expressions."; |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
return numModels; |
|
|
|
|
|
|
|
|
LOG_THROW(false, storm::exceptions::NotImplementedException, "Not Implemented."); |
|
|
#else
|
|
|
#else
|
|
|
LOG_THROW(false, storm::exceptions::NotImplementedException, "StoRM is compiled without MathSat support."); |
|
|
LOG_THROW(false, storm::exceptions::NotImplementedException, "StoRM is compiled without MathSat support."); |
|
|
#endif
|
|
|
#endif
|
|
@ -309,12 +294,55 @@ namespace storm { |
|
|
unsatAssumptions.reserve(numUnsatAssumpations); |
|
|
unsatAssumptions.reserve(numUnsatAssumpations); |
|
|
|
|
|
|
|
|
for (unsigned int i = 0; i < numUnsatAssumpations; ++i) { |
|
|
for (unsigned int i = 0; i < numUnsatAssumpations; ++i) { |
|
|
unsatAssumptions.push_back(this->m_adapter.translateTerm(msatUnsatAssumptions[i])); |
|
|
|
|
|
|
|
|
unsatAssumptions.push_back(this->m_adapter->translateTerm(msatUnsatAssumptions[i])); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
return unsatAssumptions; |
|
|
return unsatAssumptions; |
|
|
#else
|
|
|
#else
|
|
|
LOG_THROW(false, storm::exceptions::NotImplementedException, "StoRM is compiled without MathSat support."); |
|
|
LOG_THROW(false, storm::exceptions::NotImplementedException, "StoRM is compiled without MathSat support."); |
|
|
|
|
|
#endif
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
void MathSatSmtSolver::setInterpolationGroup(uint_fast64_t group) { |
|
|
|
|
|
#ifdef STORM_HAVE_MSAT
|
|
|
|
|
|
auto groupIter = this->interpolationGroups.find(group); |
|
|
|
|
|
if( groupIter == this->interpolationGroups.end() ) { |
|
|
|
|
|
int newGroup = msat_create_itp_group(m_env); |
|
|
|
|
|
auto insertResult = this->interpolationGroups.insert(std::make_pair(group, newGroup)); |
|
|
|
|
|
if (!insertResult.second) { |
|
|
|
|
|
throw storm::exceptions::InvalidStateException() << "Internal error in MathSAT wrapper: Unable to insert newly created interpolation group."; |
|
|
|
|
|
} |
|
|
|
|
|
groupIter = insertResult.first; |
|
|
|
|
|
} |
|
|
|
|
|
msat_set_itp_group(m_env, groupIter->second); |
|
|
|
|
|
#else
|
|
|
|
|
|
LOG_THROW(false, storm::exceptions::NotImplementedException, "StoRM is compiled without MathSat support."); |
|
|
|
|
|
#endif
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
storm::expressions::Expression MathSatSmtSolver::getInterpolant(std::vector<uint_fast64_t> groupsA) { |
|
|
|
|
|
#ifdef STORM_HAVE_MSAT
|
|
|
|
|
|
if (lastResult != SmtSolver::CheckResult::UNSAT) { |
|
|
|
|
|
throw storm::exceptions::InvalidStateException() << "getInterpolant was called but last state is not unsat."; |
|
|
|
|
|
} |
|
|
|
|
|
if (lastCheckAssumptions) { |
|
|
|
|
|
throw storm::exceptions::InvalidStateException() << "getInterpolant was called but last check had assumptions."; |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
std::vector<int> msatInterpolationGroupsA; |
|
|
|
|
|
msatInterpolationGroupsA.reserve(groupsA.size()); |
|
|
|
|
|
for (auto groupOfA : groupsA) { |
|
|
|
|
|
auto groupIter = this->interpolationGroups.find(groupOfA); |
|
|
|
|
|
if (groupIter == this->interpolationGroups.end()) { |
|
|
|
|
|
throw storm::exceptions::InvalidArgumentException() << "Requested interpolant for non existing interpolation group " << groupOfA; |
|
|
|
|
|
} |
|
|
|
|
|
msatInterpolationGroupsA.push_back(groupIter->second); |
|
|
|
|
|
} |
|
|
|
|
|
msat_term interpolant = msat_get_interpolant(m_env, msatInterpolationGroupsA.data(), msatInterpolationGroupsA.size()); |
|
|
|
|
|
|
|
|
|
|
|
return this->m_adapter->translateTerm(interpolant); |
|
|
|
|
|
#else
|
|
|
|
|
|
LOG_THROW(false, storm::exceptions::NotImplementedException, "StoRM is compiled without MathSat support."); |
|
|
#endif
|
|
|
#endif
|
|
|
} |
|
|
} |
|
|
} |
|
|
} |