/********************* */ /*! \file smt2toredlog.cpp ** \verbatim ** Top contributors (to current version): ** Dejan Jovanovic, Andrew Reynolds, Aina Niemetz ** This file is part of the CVC4 project. ** Copyright (c) 2009-2019 by the authors listed in the file AUTHORS ** in the top-level source directory) and their institutional affiliations. ** All rights reserved. See the file COPYING in the top-level source ** directory for licensing information.\endverbatim ** ** \brief [[ Add one-line brief description here ]] ** ** [[ Add lengthier description here ]] ** \todo document this file **/ #include #include #include #include #include #include #include #include using namespace std; using namespace CVC4; using namespace CVC4::parser; using namespace CVC4::options; void translate_to_redlog( string input, string command, const vector& info_tags, const vector& info_data, const map& variables, const vector& assertions); int main(int argc, char* argv[]) { // Get the filename string input(argv[1]); // Get the redlog command string command(argv[2]); // Create the expression manager Options options; options.setInputLanguage(language::input::LANG_SMTLIB_V2); std::unique_ptr solver = std::unique_ptr(new api::Solver(&options)); // Create the parser ParserBuilder parserBuilder(solver.get(), input, options); Parser* parser = parserBuilder.build(); // Variables and assertions std::map variables; vector info_tags; vector info_data; vector assertions; Command* cmd; while ((cmd = parser->nextCommand())) { SetInfoCommand* setinfo = dynamic_cast(cmd); if (setinfo) { info_tags.push_back(setinfo->getFlag()); info_data.push_back(setinfo->getSExpr().getValue()); delete cmd; continue; } DeclareFunctionCommand* declare = dynamic_cast(cmd); if (declare) { string name = declare->getSymbol(); Expr var = parser->getVariable(name).getExpr(); unsigned n = variables.size(); variables[var] = n; delete cmd; continue; } AssertCommand* assert = dynamic_cast(cmd); if (assert) { assertions.push_back(solver->getSmtEngine()->simplify(assert->getExpr())); delete cmd; continue; } delete cmd; } // Do the translation translate_to_redlog(input, command, info_tags, info_data, variables, assertions); // Get rid of the parser delete parser; } void translate_to_redlog_term(const map& variables, const Expr& term) { bool first; unsigned n = term.getNumChildren(); if (n == 0) { if (term.getKind() == kind::CONST_RATIONAL) { cout << term.getConst(); } else { assert(variables.find(term) != variables.end()); cout << "x" << variables.find(term)->second; } } else { switch (term.getKind()) { case kind::PLUS: cout << "("; first = true; for (unsigned i = 0; i < n; ++ i) { if (!first) { cout << " + "; } first = false; translate_to_redlog_term(variables, term[i]); } cout << ")"; break; case kind::MULT: cout << "("; first = true; for (unsigned i = 0; i < n; ++ i) { if (!first) { cout << " * "; } first = false; translate_to_redlog_term(variables, term[i]); } cout << ")"; break; case kind::MINUS: cout << "("; translate_to_redlog_term(variables, term[0]); cout << " - "; translate_to_redlog_term(variables, term[1]); cout << ")"; break; case kind::DIVISION: cout << "("; translate_to_redlog_term(variables, term[0]); cout << " / "; translate_to_redlog_term(variables, term[1]); cout << ")"; break; case kind::UMINUS: cout << "(-("; translate_to_redlog_term(variables, term[0]); cout << "))"; break; default: assert(false); break; } } } void translate_to_redlog(const map& variables, const Expr& assertion) { bool first; unsigned n = assertion.getNumChildren(); if (n == 0) { if (assertion.isConst()) { if (assertion.getConst()) { cout << "(1 > 0)"; } else { cout << "(1 < 0)"; } } else { assert(false); } } else { std::string op; bool binary = false; bool theory = false; switch (assertion.getKind()) { case kind::NOT: cout << "(not "; translate_to_redlog(variables, assertion[0]); cout << ")"; break; case kind::OR: first = true; cout << "("; for (unsigned i = 0; i < n; ++ i) { if (!first) { cout << " or "; } first = false; translate_to_redlog(variables, assertion[i]); } cout << ")"; break; case kind::AND: first = true; cout << "("; for (unsigned i = 0; i < n; ++ i) { if (!first) { cout << " and "; } first = false; translate_to_redlog(variables, assertion[i]); } cout << ")"; break; case kind::IMPLIES: cout << "("; translate_to_redlog(variables, assertion[0]); cout << " impl "; translate_to_redlog(variables, assertion[1]); cout << ")"; break; case kind::EQUAL: if( assertion[0].getType().isBoolean() ){ cout << "("; translate_to_redlog(variables, assertion[0]); cout << " equiv "; translate_to_redlog(variables, assertion[1]); cout << ")"; }else{ op = "="; theory = true; } break; case kind::LT: op = "<"; theory = true; break; case kind::LEQ: op = "<="; theory = true; break; case kind::GT: op = ">"; theory = true; break; case kind::GEQ: op = ">="; theory = true; break; default: assert(false); break; } if (binary) { cout << "("; translate_to_redlog(variables, assertion[0]); cout << " " << op << " "; translate_to_redlog(variables, assertion[1]); cout << ")"; } if (theory) { cout << "("; translate_to_redlog_term(variables, assertion[0]); cout << " " << op << " "; translate_to_redlog_term(variables, assertion[1]); cout << ")"; } } } void translate_to_redlog( string input, string command, const vector& info_tags, const vector& info_data, const std::map& variables, const vector& assertions) { bool first; // Dump out the information cout << "load redlog;" << endl; cout << "rlset ofsf;" << endl; // Dump the variables cout << "phi := ex({"; first = true; for (unsigned i = 0; i < variables.size(); ++ i) { if (!first) { cout << ","; } cout << " x" << i; if (first) { first = false; } } cout << "},"; // The assertions first = true; for (unsigned i = 0; i < assertions.size(); ++ i) { if (first == false) { cout << " and "; } first = false; translate_to_redlog(variables, assertions[i]); } cout << ");" << endl; cout << "result := " << command << " phi;" << endl; cout << "result;" << endl; cout << "quit;" << endl; }