Fix CHCSmtLib2Interface

This commit is contained in:
Leo Alt 2021-05-18 19:12:06 +02:00
parent a3634934d1
commit daea5f886d
6 changed files with 77 additions and 25 deletions

View File

@ -53,8 +53,7 @@ void CHCSmtLib2Interface::reset()
m_accumulatedOutput.clear(); m_accumulatedOutput.clear();
m_variables.clear(); m_variables.clear();
m_unhandledQueries.clear(); m_unhandledQueries.clear();
if (m_queryTimeout) m_sortNames.clear();
write("(set-option :timeout " + to_string(*m_queryTimeout) + ")");
} }
void CHCSmtLib2Interface::registerRelation(Expression const& _expr) void CHCSmtLib2Interface::registerRelation(Expression const& _expr)
@ -64,27 +63,25 @@ void CHCSmtLib2Interface::registerRelation(Expression const& _expr)
if (!m_variables.count(_expr.name)) if (!m_variables.count(_expr.name))
{ {
auto fSort = dynamic_pointer_cast<FunctionSort>(_expr.sort); auto fSort = dynamic_pointer_cast<FunctionSort>(_expr.sort);
string domain = m_smtlib2->toSmtLibSort(fSort->domain); string domain = toSmtLibSort(fSort->domain);
// Relations are predicates which have implicit codomain Bool. // Relations are predicates which have implicit codomain Bool.
m_variables.insert(_expr.name); m_variables.insert(_expr.name);
write( write(
"(declare-rel |" + "(declare-fun |" +
_expr.name + _expr.name +
"| " + "| " +
domain + domain +
")" " Bool)"
); );
} }
} }
void CHCSmtLib2Interface::addRule(Expression const& _expr, std::string const& _name) void CHCSmtLib2Interface::addRule(Expression const& _expr, std::string const& /*_name*/)
{ {
write( write(
"(rule (! " + "(assert\n(forall " + forall() + "\n" +
m_smtlib2->toSExpr(_expr) + m_smtlib2->toSExpr(_expr) +
" :named " + "))\n\n"
_name +
"))"
); );
} }
@ -92,22 +89,29 @@ pair<CheckResult, CHCSolverInterface::CexGraph> CHCSmtLib2Interface::query(Expre
{ {
string accumulated{}; string accumulated{};
swap(m_accumulatedOutput, accumulated); swap(m_accumulatedOutput, accumulated);
for (auto const& var: m_smtlib2->variables()) solAssert(m_smtlib2, "");
declareVariable(var.first, var.second); writeHeader();
for (auto const& [type, decl]: m_smtlib2->userSorts())
write(decl);
m_accumulatedOutput += accumulated; m_accumulatedOutput += accumulated;
string queryRule = "(assert\n(forall " + forall() + "\n" +
"(=> " + _block.name + " false)"
"))";
string response = querySolver( string response = querySolver(
m_accumulatedOutput + m_accumulatedOutput +
"\n(query " + _block.name + " :print-certificate true)" queryRule +
"\n(check-sat)"
); );
swap(m_accumulatedOutput, accumulated);
CheckResult result; CheckResult result;
// TODO proper parsing // TODO proper parsing
if (boost::starts_with(response, "sat\n")) if (boost::starts_with(response, "sat"))
result = CheckResult::SATISFIABLE;
else if (boost::starts_with(response, "unsat\n"))
result = CheckResult::UNSATISFIABLE; result = CheckResult::UNSATISFIABLE;
else if (boost::starts_with(response, "unknown\n")) else if (boost::starts_with(response, "unsat"))
result = CheckResult::SATISFIABLE;
else if (boost::starts_with(response, "unknown"))
result = CheckResult::UNKNOWN; result = CheckResult::UNKNOWN;
else else
result = CheckResult::ERROR; result = CheckResult::ERROR;
@ -124,10 +128,46 @@ void CHCSmtLib2Interface::declareVariable(string const& _name, SortPointer const
else if (!m_variables.count(_name)) else if (!m_variables.count(_name))
{ {
m_variables.insert(_name); m_variables.insert(_name);
write("(declare-var |" + _name + "| " + m_smtlib2->toSmtLibSort(*_sort) + ')'); write("(declare-var |" + _name + "| " + toSmtLibSort(*_sort) + ')');
} }
} }
string CHCSmtLib2Interface::toSmtLibSort(Sort const& _sort)
{
if (!m_sortNames.count(&_sort))
m_sortNames[&_sort] = m_smtlib2->toSmtLibSort(_sort);
return m_sortNames.at(&_sort);
}
string CHCSmtLib2Interface::toSmtLibSort(vector<SortPointer> const& _sorts)
{
string ssort("(");
for (auto const& sort: _sorts)
ssort += toSmtLibSort(*sort) + " ";
ssort += ")";
return ssort;
}
void CHCSmtLib2Interface::writeHeader()
{
if (m_queryTimeout)
write("(set-option :timeout " + to_string(*m_queryTimeout) + ")");
write("(set-logic HORN)\n");
}
string CHCSmtLib2Interface::forall()
{
string vars("(");
for (auto const& [name, sort]: m_smtlib2->variables())
{
solAssert(sort, "");
if (sort->kind != Kind::Function)
vars += " (" + name + " " + toSmtLibSort(*sort) + ")";
}
vars += ")";
return vars;
}
void CHCSmtLib2Interface::declareFunction(string const& _name, SortPointer const& _sort) void CHCSmtLib2Interface::declareFunction(string const& _name, SortPointer const& _sort)
{ {
smtAssert(_sort, ""); smtAssert(_sort, "");
@ -137,8 +177,8 @@ void CHCSmtLib2Interface::declareFunction(string const& _name, SortPointer const
{ {
auto fSort = dynamic_pointer_cast<FunctionSort>(_sort); auto fSort = dynamic_pointer_cast<FunctionSort>(_sort);
smtAssert(fSort->codomain, ""); smtAssert(fSort->codomain, "");
string domain = m_smtlib2->toSmtLibSort(fSort->domain); string domain = toSmtLibSort(fSort->domain);
string codomain = m_smtlib2->toSmtLibSort(*fSort->codomain); string codomain = toSmtLibSort(*fSort->codomain);
m_variables.insert(_name); m_variables.insert(_name);
write( write(
"(declare-fun |" + "(declare-fun |" +

View File

@ -53,6 +53,12 @@ public:
SMTLib2Interface* smtlib2Interface() const { return m_smtlib2.get(); } SMTLib2Interface* smtlib2Interface() const { return m_smtlib2.get(); }
private: private:
std::string toSmtLibSort(Sort const& _sort);
std::string toSmtLibSort(std::vector<SortPointer> const& _sort);
void writeHeader();
std::string forall();
void declareFunction(std::string const& _name, SortPointer const& _sort); void declareFunction(std::string const& _name, SortPointer const& _sort);
void write(std::string _data); void write(std::string _data);
@ -70,6 +76,8 @@ private:
std::vector<std::string> m_unhandledQueries; std::vector<std::string> m_unhandledQueries;
frontend::ReadCallback::Callback m_smtCallback; frontend::ReadCallback::Callback m_smtCallback;
std::map<Sort const*, std::string> m_sortNames;
}; };
} }

View File

@ -23,6 +23,8 @@
#include <boost/algorithm/string/join.hpp> #include <boost/algorithm/string/join.hpp>
#include <boost/algorithm/string/predicate.hpp> #include <boost/algorithm/string/predicate.hpp>
#include <range/v3/algorithm/find_if.hpp>
#include <array> #include <array>
#include <fstream> #include <fstream>
#include <iostream> #include <iostream>
@ -231,14 +233,15 @@ string SMTLib2Interface::toSmtLibSort(Sort const& _sort)
{ {
auto const& tupleSort = dynamic_cast<TupleSort const&>(_sort); auto const& tupleSort = dynamic_cast<TupleSort const&>(_sort);
string tupleName = "|" + tupleSort.name + "|"; string tupleName = "|" + tupleSort.name + "|";
if (!m_userSorts.count(tupleName)) auto isName = [&](auto entry) { return entry.first == tupleName; };
if (ranges::find_if(m_userSorts, isName) == m_userSorts.end())
{ {
m_userSorts.insert(tupleName);
string decl("(declare-datatypes ((" + tupleName + " 0)) (((" + tupleName); string decl("(declare-datatypes ((" + tupleName + " 0)) (((" + tupleName);
smtAssert(tupleSort.members.size() == tupleSort.components.size(), ""); smtAssert(tupleSort.members.size() == tupleSort.components.size(), "");
for (unsigned i = 0; i < tupleSort.members.size(); ++i) for (unsigned i = 0; i < tupleSort.members.size(); ++i)
decl += " (|" + tupleSort.members.at(i) + "| " + toSmtLibSort(*tupleSort.components.at(i)) + ")"; decl += " (|" + tupleSort.members.at(i) + "| " + toSmtLibSort(*tupleSort.components.at(i)) + ")";
decl += "))))"; decl += "))))";
m_userSorts.emplace_back(tupleName, decl);
write(decl); write(decl);
} }

View File

@ -66,6 +66,8 @@ public:
std::map<std::string, SortPointer> variables() { return m_variables; } std::map<std::string, SortPointer> variables() { return m_variables; }
std::vector<std::pair<std::string, std::string>>const& userSorts() const { return m_userSorts; }
private: private:
void declareFunction(std::string const& _name, SortPointer const& _sort); void declareFunction(std::string const& _name, SortPointer const& _sort);
@ -79,7 +81,7 @@ private:
std::vector<std::string> m_accumulatedOutput; std::vector<std::string> m_accumulatedOutput;
std::map<std::string, SortPointer> m_variables; std::map<std::string, SortPointer> m_variables;
std::set<std::string> m_userSorts; std::vector<std::pair<std::string, std::string>> m_userSorts;
std::map<util::h256, std::string> m_queryResponses; std::map<util::h256, std::string> m_queryResponses;
std::vector<std::string> m_unhandledQueries; std::vector<std::string> m_unhandledQueries;

View File

@ -946,7 +946,6 @@ void CHC::resetSourceAnalysis()
if (!usesZ3) if (!usesZ3)
{ {
auto smtlib2Interface = dynamic_cast<CHCSmtLib2Interface*>(m_interface.get()); auto smtlib2Interface = dynamic_cast<CHCSmtLib2Interface*>(m_interface.get());
smtlib2Interface->reset();
solAssert(smtlib2Interface, ""); solAssert(smtlib2Interface, "");
m_context.setSolver(smtlib2Interface->smtlib2Interface()); m_context.setSolver(smtlib2Interface->smtlib2Interface());
} }

View File

@ -118,7 +118,7 @@ SortPointer smtSort(frontend::Type const& _type)
else else
tupleName = arrayType->baseType()->toString(true); tupleName = arrayType->baseType()->toString(true);
tupleName += "[]"; tupleName += "_array";
} }
else else
tupleName = _type.toString(true); tupleName = _type.toString(true);