lean4-htt/src/library/compiler/inliner.cpp
Leonardo de Moura 8ed89c6ac3 chore(library): remove normalize.cpp
The command `#reduce` was also temporarily removed.
2018-09-04 10:51:14 -07:00

175 lines
6 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/attribute_manager.h"
#include "library/vm/vm.h"
#include "library/compiler/util.h"
#include "library/compiler/compiler_step_visitor.h"
namespace lean {
bool is_inline(environment const & env, name const & n) {
// declaration is marked with [inline]
if (has_attribute(env, "inline", n)) {
return true;
}
// decl._main is an auxiliary declaration, check decl instead
if (n.is_string() && n.get_string().data()[0] == '_') {
return is_inline(env, n.get_prefix());
}
return false;
}
void initialize_inliner() {
register_system_attribute(basic_attribute::with_check(
"inline", "mark definition to always be inlined",
[](environment const & env, name const & d, bool) -> void {
auto decl = env.get(d);
if (!decl.is_definition())
throw exception("invalid 'inline' use, only definitions can be marked as inline");
}));
}
void finalize_inliner() {
}
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_bvar(y) && !is_constant(y))
return false;
if (is_bvar(y)) {
nat const & n_vidx = bvar_idx(y);
if (!n_vidx.is_small())
return false;
unsigned vidx = n_vidx.get_small_value();
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) {
expr new_e = compiler_step_visitor::visit_app(e);
if (new_e != e)
return visit(new_e);
else
return new_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) {
buffer<expr> args;
expr const & fn = get_app_args(e, args);
lean_assert(is_constant(fn));
for (expr & arg : args)
arg = visit(arg);
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);
}
if (major_idx >= args.size())
return mk_app(fn, args);
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 = mk_app(fn, args);
if (is_cases_on) {
/* unfold cases_on */
if (auto r = unfold_term(env(), new_e))
new_e = *r;
else
return new_e;
}
/* reduce */
if (auto r = ctx().reduce_recursor(new_e))
return visit(beta_reduce(*r));
}
return mk_app(fn, args);
}
optional<expr> reduce_projection(expr const & e) {
/* When trying to reduce a projection, we should only unfold reducible constants. */
type_context_old::transparency_scope _(ctx(), transparency_mode::Instances);
return ctx().reduce_projection(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);
constant_info info = env().get(n);
if (!info.is_definition())
return default_visit_app(e);
expr v = info.get_value();
unsigned arity = 0;
while (is_lambda(v)) {
arity++;
v = binding_body(v);
}
if (is_inline(env(), n) || (is_simple_application(v) && arity <= nargs)) {
if (auto r = unfold_term(env(), e))
return visit(expr(*r));
}
if (arity <= nargs) {
if (auto r = reduce_projection(e)) {
return visit(*r);
}
}
return default_visit_app(e);
}
public:
inline_simple_definitions_fn(environment const & env, abstract_context_cache & cache):
compiler_step_visitor(env, cache) {}
};
expr inline_simple_definitions(environment const & env, abstract_context_cache & cache, expr const & e) {
return inline_simple_definitions_fn(env, cache)(e);
}
}