feat(library/definitional/equations): add 'equations_result' macro used to wrap multiple functions being defined by recursive equations
This is only useful when compiling "mutually recursive" functions.
This commit is contained in:
parent
3325d791de
commit
322cdb8a98
2 changed files with 97 additions and 31 deletions
|
@ -4,6 +4,7 @@ Released under Apache 2.0 license as described in the file LICENSE.
|
||||||
|
|
||||||
Author: Leonardo de Moura
|
Author: Leonardo de Moura
|
||||||
*/
|
*/
|
||||||
|
#include <algorithm>
|
||||||
#include <string>
|
#include <string>
|
||||||
#include "util/sstream.h"
|
#include "util/sstream.h"
|
||||||
#include "util/list_fn.h"
|
#include "util/list_fn.h"
|
||||||
|
@ -28,9 +29,11 @@ static name * g_equations_name = nullptr;
|
||||||
static name * g_equation_name = nullptr;
|
static name * g_equation_name = nullptr;
|
||||||
static name * g_decreasing_name = nullptr;
|
static name * g_decreasing_name = nullptr;
|
||||||
static name * g_inaccessible_name = nullptr;
|
static name * g_inaccessible_name = nullptr;
|
||||||
|
static name * g_equations_result_name = nullptr;
|
||||||
static std::string * g_equations_opcode = nullptr;
|
static std::string * g_equations_opcode = nullptr;
|
||||||
static std::string * g_equation_opcode = nullptr;
|
static std::string * g_equation_opcode = nullptr;
|
||||||
static std::string * g_decreasing_opcode = nullptr;
|
static std::string * g_decreasing_opcode = nullptr;
|
||||||
|
static std::string * g_equations_result_opcode = nullptr;
|
||||||
|
|
||||||
[[ noreturn ]] static void throw_eqs_ex() { throw exception("unexpected occurrence of 'equations' expression"); }
|
[[ noreturn ]] static void throw_eqs_ex() { throw exception("unexpected occurrence of 'equations' expression"); }
|
||||||
|
|
||||||
|
@ -169,16 +172,43 @@ expr update_equations(expr const & eqns, buffer<expr> const & new_eqs) {
|
||||||
expr mk_inaccessible(expr const & e) { return mk_annotation(*g_inaccessible_name, e); }
|
expr mk_inaccessible(expr const & e) { return mk_annotation(*g_inaccessible_name, e); }
|
||||||
bool is_inaccessible(expr const & e) { return is_annotation(e, *g_inaccessible_name); }
|
bool is_inaccessible(expr const & e) { return is_annotation(e, *g_inaccessible_name); }
|
||||||
|
|
||||||
|
// Auxiliary macro used to store the result of a set of equations defining a mutually recursive
|
||||||
|
// definition.
|
||||||
|
class equations_result_macro_cell : public macro_definition_cell {
|
||||||
|
public:
|
||||||
|
virtual name get_name() const { return *g_equations_result_name; }
|
||||||
|
virtual pair<expr, constraint_seq> get_type(expr const & m, extension_context & ctx) const {
|
||||||
|
return ctx.infer_type(macro_arg(m, 0));
|
||||||
|
}
|
||||||
|
virtual optional<expr> expand(expr const & m, extension_context &) const {
|
||||||
|
return some_expr(macro_arg(m, 0));
|
||||||
|
}
|
||||||
|
virtual void write(serializer & s) const { s << *g_equations_result_opcode; }
|
||||||
|
};
|
||||||
|
|
||||||
|
static macro_definition * g_equations_result = nullptr;
|
||||||
|
|
||||||
|
static expr mk_equations_result(unsigned n, expr const * rs) {
|
||||||
|
return mk_macro(*g_equations_result, n, rs);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool is_equations_result(expr const & e) { return is_macro(e) && macro_def(e) == *g_equations_result; }
|
||||||
|
unsigned get_equations_result_size(expr const & e) { return macro_num_args(e); }
|
||||||
|
expr const & get_equations_result(expr const & e, unsigned i) { return macro_arg(e, i); }
|
||||||
|
|
||||||
void initialize_equations() {
|
void initialize_equations() {
|
||||||
g_equations_name = new name("equations");
|
g_equations_name = new name("equations");
|
||||||
g_equation_name = new name("equation");
|
g_equation_name = new name("equation");
|
||||||
g_decreasing_name = new name("decreasing");
|
g_decreasing_name = new name("decreasing");
|
||||||
g_inaccessible_name = new name("innaccessible");
|
g_inaccessible_name = new name("innaccessible");
|
||||||
|
g_equations_result_name = new name("equations_result");
|
||||||
g_equation = new macro_definition(new equation_macro_cell());
|
g_equation = new macro_definition(new equation_macro_cell());
|
||||||
g_decreasing = new macro_definition(new decreasing_macro_cell());
|
g_decreasing = new macro_definition(new decreasing_macro_cell());
|
||||||
|
g_equations_result = new macro_definition(new equations_result_macro_cell());
|
||||||
g_equations_opcode = new std::string("Eqns");
|
g_equations_opcode = new std::string("Eqns");
|
||||||
g_equation_opcode = new std::string("Eqn");
|
g_equation_opcode = new std::string("Eqn");
|
||||||
g_decreasing_opcode = new std::string("Decr");
|
g_decreasing_opcode = new std::string("Decr");
|
||||||
|
g_equations_result_opcode = new std::string("EqnR");
|
||||||
register_annotation(*g_inaccessible_name);
|
register_annotation(*g_inaccessible_name);
|
||||||
register_macro_deserializer(*g_equations_opcode,
|
register_macro_deserializer(*g_equations_opcode,
|
||||||
[](deserializer & d, unsigned num, expr const * args) {
|
[](deserializer & d, unsigned num, expr const * args) {
|
||||||
|
@ -206,14 +236,21 @@ void initialize_equations() {
|
||||||
throw corrupted_stream_exception();
|
throw corrupted_stream_exception();
|
||||||
return mk_decreasing(args[0], args[1]);
|
return mk_decreasing(args[0], args[1]);
|
||||||
});
|
});
|
||||||
|
register_macro_deserializer(*g_equations_result_opcode,
|
||||||
|
[](deserializer &, unsigned num, expr const * args) {
|
||||||
|
return mk_equations_result(num, args);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
void finalize_equations() {
|
void finalize_equations() {
|
||||||
|
delete g_equations_result_opcode;
|
||||||
delete g_equation_opcode;
|
delete g_equation_opcode;
|
||||||
delete g_equations_opcode;
|
delete g_equations_opcode;
|
||||||
delete g_decreasing_opcode;
|
delete g_decreasing_opcode;
|
||||||
|
delete g_equations_result;
|
||||||
delete g_equation;
|
delete g_equation;
|
||||||
delete g_decreasing;
|
delete g_decreasing;
|
||||||
|
delete g_equations_result_name;
|
||||||
delete g_equations_name;
|
delete g_equations_name;
|
||||||
delete g_equation_name;
|
delete g_equation_name;
|
||||||
delete g_decreasing_name;
|
delete g_decreasing_name;
|
||||||
|
@ -1303,21 +1340,8 @@ class equation_compiler_fn {
|
||||||
lean_assert(check_program(prgs[0]));
|
lean_assert(check_program(prgs[0]));
|
||||||
}
|
}
|
||||||
|
|
||||||
expr compile_brec_on(buffer<program> & prgs) {
|
expr compile_brec_on_core(buffer<program> const & prgs,
|
||||||
lean_assert(!prgs.empty());
|
buffer<unsigned> const & arg_pos, buffer<unsigned> const & rec_arg_pos) {
|
||||||
buffer<unsigned> rec_arg_pos;
|
|
||||||
if (!find_rec_args(prgs, rec_arg_pos)) {
|
|
||||||
throw_error(sstream() << "invalid recursive equations, "
|
|
||||||
<< "failed to find recursive arguments that are structurally smaller "
|
|
||||||
<< "(possible solution: use well-founded recursion)");
|
|
||||||
}
|
|
||||||
// Remark: move_params updates argument positions.
|
|
||||||
// Thus, we copy rec_arg_pos to arg_pos.
|
|
||||||
// We use rec_arg_pos when invoking elim_rec_apps_fn
|
|
||||||
buffer<unsigned> arg_pos;
|
|
||||||
arg_pos.append(rec_arg_pos);
|
|
||||||
move_params(prgs, arg_pos);
|
|
||||||
|
|
||||||
// Return the recursive argument of the i-th program
|
// Return the recursive argument of the i-th program
|
||||||
auto get_rec_arg = [&](unsigned i) -> expr {
|
auto get_rec_arg = [&](unsigned i) -> expr {
|
||||||
program const & pi = prgs[i];
|
program const & pi = prgs[i];
|
||||||
|
@ -1513,6 +1537,40 @@ class equation_compiler_fn {
|
||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
expr compile_brec_on(buffer<program> & prgs) {
|
||||||
|
lean_assert(!prgs.empty());
|
||||||
|
buffer<unsigned> rec_arg_pos;
|
||||||
|
if (!find_rec_args(prgs, rec_arg_pos)) {
|
||||||
|
throw_error(sstream() << "invalid recursive equations, "
|
||||||
|
<< "failed to find recursive arguments that are structurally smaller "
|
||||||
|
<< "(possible solution: use well-founded recursion)");
|
||||||
|
}
|
||||||
|
// Remark: move_params updates argument positions.
|
||||||
|
// Thus, we copy rec_arg_pos to arg_pos.
|
||||||
|
// We use rec_arg_pos when invoking elim_rec_apps_fn
|
||||||
|
buffer<unsigned> arg_pos;
|
||||||
|
arg_pos.append(rec_arg_pos);
|
||||||
|
move_params(prgs, arg_pos);
|
||||||
|
buffer<expr> rs;
|
||||||
|
for (unsigned i = 0; i < prgs.size(); i++) {
|
||||||
|
// Remark: this loop is very hackish.
|
||||||
|
// We are "compiling" the code prgs.size() times!
|
||||||
|
// This is wasteful. We should rewrite this.
|
||||||
|
std::swap(prgs[0], prgs[i]);
|
||||||
|
std::swap(arg_pos[0], arg_pos[i]);
|
||||||
|
std::swap(rec_arg_pos[0], rec_arg_pos[i]);
|
||||||
|
rs.push_back(compile_brec_on_core(prgs, arg_pos, rec_arg_pos));
|
||||||
|
std::swap(prgs[0], prgs[i]);
|
||||||
|
std::swap(arg_pos[0], arg_pos[i]);
|
||||||
|
std::swap(rec_arg_pos[0], rec_arg_pos[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (rs.size() > 1)
|
||||||
|
return mk_equations_result(rs.size(), rs.data());
|
||||||
|
else
|
||||||
|
return rs[0];
|
||||||
|
}
|
||||||
|
|
||||||
expr compile_wf(buffer<program> & /* prgs */) {
|
expr compile_wf(buffer<program> & /* prgs */) {
|
||||||
// TODO(Leo)
|
// TODO(Leo)
|
||||||
return expr();
|
return expr();
|
||||||
|
|
|
@ -40,6 +40,14 @@ bool is_inaccessible(expr const & e);
|
||||||
expr compile_equations(type_checker & tc, io_state const & ios, expr const & eqns,
|
expr compile_equations(type_checker & tc, io_state const & ios, expr const & eqns,
|
||||||
expr const & meta, expr const & meta_type, bool relax);
|
expr const & meta, expr const & meta_type, bool relax);
|
||||||
|
|
||||||
|
/** \brief Return true if \c e is an auxiliary macro used to store the result of mutually recursive declarations.
|
||||||
|
For example, if a set of recursive equations is defining \c n mutually recursive functions, we wrap
|
||||||
|
the \c n resulting functions with an \c equations_result macro.
|
||||||
|
*/
|
||||||
|
bool is_equations_result(expr const & e);
|
||||||
|
unsigned get_equations_result_size(expr const & e);
|
||||||
|
expr const & get_equations_result(expr const & e, unsigned i);
|
||||||
|
|
||||||
void initialize_equations();
|
void initialize_equations();
|
||||||
void finalize_equations();
|
void finalize_equations();
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue