189 lines
6.2 KiB
C++
189 lines
6.2 KiB
C++
/*
|
|
Copyright (c) 2016 Microsoft Corporation. All rights reserved.
|
|
Released under Apache 2.0 license as described in the file LICENSE.
|
|
|
|
Author: Leonardo de Moura
|
|
*/
|
|
#include <string>
|
|
#include "kernel/inductive/inductive.h"
|
|
#include "library/util.h"
|
|
#include "library/module.h"
|
|
#include "library/trace.h"
|
|
#include "library/normalize.h"
|
|
#include "library/vm/vm.h"
|
|
#include "library/compiler/util.h"
|
|
#include "library/compiler/compiler_step_visitor.h"
|
|
|
|
namespace lean {
|
|
struct inline_ext : public environment_extension {
|
|
name_set m_set;
|
|
};
|
|
|
|
struct inline_ext_reg {
|
|
unsigned m_ext_id;
|
|
inline_ext_reg() { m_ext_id = environment::register_extension(std::make_shared<inline_ext>()); }
|
|
};
|
|
|
|
static inline_ext_reg * g_ext = nullptr;
|
|
static inline_ext const & get_extension(environment const & env) {
|
|
return static_cast<inline_ext const &>(env.get_extension(g_ext->m_ext_id));
|
|
}
|
|
static environment update(environment const & env, inline_ext const & ext) {
|
|
return env.update(g_ext->m_ext_id, std::make_shared<inline_ext>(ext));
|
|
}
|
|
|
|
static std::string * g_inline_key = nullptr;
|
|
|
|
environment add_inline(environment const & env, name const & n) {
|
|
inline_ext ext = get_extension(env);
|
|
ext.m_set.insert(n);
|
|
environment new_env = update(env, ext);
|
|
return module::add(new_env, *g_inline_key, [=](environment const &, serializer & s) { s << n; });
|
|
}
|
|
|
|
static void inline_reader(deserializer & d, shared_environment & senv,
|
|
std::function<void(asynch_update_fn const &)> &,
|
|
std::function<void(delayed_update_fn const &)> &) {
|
|
name n;
|
|
d >> n;
|
|
senv.update([=](environment const & env) -> environment {
|
|
inline_ext ext = get_extension(env);
|
|
ext.m_set.insert(n);
|
|
return update(env, ext);
|
|
});
|
|
}
|
|
|
|
bool is_inline(environment const & env, name const & n) {
|
|
return get_extension(env).m_set.contains(n);
|
|
}
|
|
|
|
void initialize_inliner() {
|
|
g_ext = new inline_ext_reg();
|
|
g_inline_key = new std::string("inline");
|
|
register_module_object_reader(*g_inline_key, inline_reader);
|
|
}
|
|
|
|
void finalize_inliner() {
|
|
delete g_inline_key;
|
|
delete g_ext;
|
|
}
|
|
|
|
class inline_simple_definitions_fn : public compiler_step_visitor {
|
|
/* Return true iff v is of the form (g y_1 ... y_n) where
|
|
y_i is a constant or a variable.
|
|
Moreover, y_i's variables are pairwise distinct. */
|
|
bool is_simple_application(expr const & v) {
|
|
buffer<expr> ys;
|
|
buffer<bool> bitmap;
|
|
expr const & g = get_app_args(v, ys);
|
|
if (!is_constant(g) && !is_var(g))
|
|
return false;
|
|
for (expr const & y : ys) {
|
|
if (!is_var(y) && !is_constant(y))
|
|
return false;
|
|
if (is_var(y)) {
|
|
unsigned vidx = var_idx(y);
|
|
if (vidx >= bitmap.size())
|
|
bitmap.resize(vidx+1, false);
|
|
if (bitmap[vidx]) {
|
|
/* y_i's are not pairwise distinct */
|
|
return false;
|
|
}
|
|
bitmap[vidx] = true;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
expr default_visit_app(expr const & e) {
|
|
return compiler_step_visitor::visit_app(e);
|
|
}
|
|
|
|
bool is_nonrecursive_recursor(name const & n) {
|
|
if (auto I_name = inductive::is_elim_rule(env(), n)) {
|
|
return !is_recursive_datatype(env(), *I_name);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/* Try to reduce cases_on (and nonrecursive recursor) application
|
|
if major became a constructor */
|
|
expr visit_cases_on_app(expr const & e_0) {
|
|
expr e = default_visit_app(e_0);
|
|
buffer<expr> args;
|
|
expr const & fn = get_app_args(e, args);
|
|
lean_assert(is_constant(fn));
|
|
bool is_cases_on = is_cases_on_recursor(env(), const_name(fn));
|
|
name const & rec_name = const_name(fn);
|
|
name const & I_name = rec_name.get_prefix();
|
|
unsigned nparams = *inductive::get_num_params(env(), I_name);
|
|
unsigned nindices = *inductive::get_num_indices(env(), I_name);
|
|
unsigned major_idx;
|
|
if (is_cases_on) {
|
|
major_idx = nparams + 1 + nindices;
|
|
} else {
|
|
major_idx = *inductive::get_elim_major_idx(env(), rec_name);
|
|
}
|
|
expr major = beta_reduce(args[major_idx]);
|
|
if (is_constructor_app(env(), major)) {
|
|
/* Major premise became a constructor. So, we should reduce. */
|
|
expr new_e = e;
|
|
if (is_cases_on) {
|
|
/* unfold cases_on */
|
|
if (auto r = unfold_term(env(), new_e))
|
|
new_e = *r;
|
|
else
|
|
return e;
|
|
}
|
|
/* reduce */
|
|
if (auto r = ctx().norm_ext(new_e))
|
|
return compiler_step_visitor::visit(beta_reduce(*r));
|
|
}
|
|
return e;
|
|
}
|
|
|
|
virtual expr visit_app(expr const & e) override {
|
|
expr const & fn = get_app_fn(e);
|
|
if (!is_constant(fn))
|
|
return default_visit_app(e);
|
|
name const & n = const_name(fn);
|
|
if (is_vm_builtin_function(n))
|
|
return default_visit_app(e);
|
|
if (is_cases_on_recursor(env(), n) || is_nonrecursive_recursor(n))
|
|
return visit_cases_on_app(e);
|
|
unsigned nargs = get_app_num_args(e);
|
|
declaration d = env().get(n);
|
|
if (!d.is_definition() || d.is_theorem())
|
|
return default_visit_app(e);
|
|
expr v = d.get_value();
|
|
unsigned arity = 0;
|
|
while (is_lambda(v)) {
|
|
arity++;
|
|
v = binding_body(v);
|
|
}
|
|
|
|
if (arity > nargs) {
|
|
// not fully applied
|
|
return default_visit_app(e);
|
|
}
|
|
|
|
if (is_inline(env(), n) || is_simple_application(v)) {
|
|
if (auto r = unfold_term(env(), e))
|
|
return visit(*r);
|
|
}
|
|
|
|
if (auto r = ctx().reduce_projection(e)) {
|
|
return visit(*r);
|
|
}
|
|
|
|
return default_visit_app(e);
|
|
}
|
|
|
|
public:
|
|
inline_simple_definitions_fn(environment const & env):compiler_step_visitor(env) {}
|
|
};
|
|
|
|
expr inline_simple_definitions(environment const & env, expr const & e) {
|
|
return inline_simple_definitions_fn(env)(e);
|
|
}
|
|
}
|