feat(library/simplifier): add 'simp.funext' and 'simp.propext' options

This commit is contained in:
Leonardo de Moura 2015-07-21 18:23:10 -07:00
parent 0c0f07332e
commit e74c6eef3d
4 changed files with 77 additions and 17 deletions

View file

@ -30,6 +30,7 @@ name const * g_eq_subst = nullptr;
name const * g_exists_elim = nullptr; name const * g_exists_elim = nullptr;
name const * g_false = nullptr; name const * g_false = nullptr;
name const * g_false_rec = nullptr; name const * g_false_rec = nullptr;
name const * g_funext = nullptr;
name const * g_heq = nullptr; name const * g_heq = nullptr;
name const * g_heq_refl = nullptr; name const * g_heq_refl = nullptr;
name const * g_heq_to_eq = nullptr; name const * g_heq_to_eq = nullptr;
@ -171,6 +172,7 @@ void initialize_constants() {
g_exists_elim = new name{"exists", "elim"}; g_exists_elim = new name{"exists", "elim"};
g_false = new name{"false"}; g_false = new name{"false"};
g_false_rec = new name{"false", "rec"}; g_false_rec = new name{"false", "rec"};
g_funext = new name{"funext"};
g_heq = new name{"heq"}; g_heq = new name{"heq"};
g_heq_refl = new name{"heq", "refl"}; g_heq_refl = new name{"heq", "refl"};
g_heq_to_eq = new name{"heq", "to_eq"}; g_heq_to_eq = new name{"heq", "to_eq"};
@ -313,6 +315,7 @@ void finalize_constants() {
delete g_exists_elim; delete g_exists_elim;
delete g_false; delete g_false;
delete g_false_rec; delete g_false_rec;
delete g_funext;
delete g_heq; delete g_heq;
delete g_heq_refl; delete g_heq_refl;
delete g_heq_to_eq; delete g_heq_to_eq;
@ -454,6 +457,7 @@ name const & get_eq_subst_name() { return *g_eq_subst; }
name const & get_exists_elim_name() { return *g_exists_elim; } name const & get_exists_elim_name() { return *g_exists_elim; }
name const & get_false_name() { return *g_false; } name const & get_false_name() { return *g_false; }
name const & get_false_rec_name() { return *g_false_rec; } name const & get_false_rec_name() { return *g_false_rec; }
name const & get_funext_name() { return *g_funext; }
name const & get_heq_name() { return *g_heq; } name const & get_heq_name() { return *g_heq; }
name const & get_heq_refl_name() { return *g_heq_refl; } name const & get_heq_refl_name() { return *g_heq_refl; }
name const & get_heq_to_eq_name() { return *g_heq_to_eq; } name const & get_heq_to_eq_name() { return *g_heq_to_eq; }

View file

@ -32,6 +32,7 @@ name const & get_eq_subst_name();
name const & get_exists_elim_name(); name const & get_exists_elim_name();
name const & get_false_name(); name const & get_false_name();
name const & get_false_rec_name(); name const & get_false_rec_name();
name const & get_funext_name();
name const & get_heq_name(); name const & get_heq_name();
name const & get_heq_refl_name(); name const & get_heq_refl_name();
name const & get_heq_to_eq_name(); name const & get_heq_to_eq_name();

View file

@ -25,6 +25,7 @@ eq.subst
exists.elim exists.elim
false false
false.rec false.rec
funext
heq heq
heq.refl heq.refl
heq.to_eq heq.to_eq

View file

@ -8,7 +8,6 @@ Author: Leonardo de Moura
#include "library/constants.h" #include "library/constants.h"
#include "library/util.h" #include "library/util.h"
#include "library/tactic/expr_to_tactic.h" #include "library/tactic/expr_to_tactic.h"
#include "library/tactic/relation_tactics.h"
#include "library/simplifier/simp_tactic.h" #include "library/simplifier/simp_tactic.h"
#ifndef LEAN_DEFAULT_SIMP_SINGLE_PASS #ifndef LEAN_DEFAULT_SIMP_SINGLE_PASS
@ -43,6 +42,14 @@ Author: Leonardo de Moura
#define LEAN_DEFAULT_SIMP_ASSUMPTIONS false #define LEAN_DEFAULT_SIMP_ASSUMPTIONS false
#endif #endif
#ifndef LEAN_DEFAULT_SIMP_FUNEXT
#define LEAN_DEFAULT_SIMP_FUNEXT true
#endif
#ifndef LEAN_DEFAULT_SIMP_PROPEXT
#define LEAN_DEFAULT_SIMP_PROPEXT true
#endif
namespace lean { namespace lean {
name const * g_simp_single_pass = nullptr; name const * g_simp_single_pass = nullptr;
name const * g_simp_bottom_up = nullptr; name const * g_simp_bottom_up = nullptr;
@ -52,6 +59,8 @@ name const * g_simp_memoize = nullptr;
name const * g_simp_max_steps = nullptr; name const * g_simp_max_steps = nullptr;
name const * g_simp_trace = nullptr; name const * g_simp_trace = nullptr;
name const * g_simp_assumptions = nullptr; name const * g_simp_assumptions = nullptr;
name const * g_simp_funext = nullptr;
name const * g_simp_propext = nullptr;
bool get_simp_single_pass(options const & opts) { bool get_simp_single_pass(options const & opts) {
return opts.get_bool(*g_simp_single_pass, LEAN_DEFAULT_SIMP_SINGLE_PASS); return opts.get_bool(*g_simp_single_pass, LEAN_DEFAULT_SIMP_SINGLE_PASS);
@ -85,6 +94,14 @@ bool get_simp_assumptions(options const & opts) {
return opts.get_bool(*g_simp_assumptions, LEAN_DEFAULT_SIMP_ASSUMPTIONS); return opts.get_bool(*g_simp_assumptions, LEAN_DEFAULT_SIMP_ASSUMPTIONS);
} }
bool get_simp_funext(options const & opts) {
return opts.get_bool(*g_simp_funext, LEAN_DEFAULT_SIMP_FUNEXT);
}
bool get_simp_propext(options const & opts) {
return opts.get_bool(*g_simp_propext, LEAN_DEFAULT_SIMP_PROPEXT);
}
expr const * g_simp_tactic = nullptr; expr const * g_simp_tactic = nullptr;
expr mk_simp_tactic_expr(buffer<expr> const & ls, buffer<name> const & ns, expr mk_simp_tactic_expr(buffer<expr> const & ls, buffer<name> const & ns,
@ -111,6 +128,10 @@ class simp_tactic_fn {
name_generator m_ngen; name_generator m_ngen;
optional<tactic> m_tactic; optional<tactic> m_tactic;
// transient state
unsigned m_steps;
goal m_g;
// configuration options // configuration options
bool m_single_pass; bool m_single_pass;
bool m_bottom_up; bool m_bottom_up;
@ -120,8 +141,11 @@ class simp_tactic_fn {
unsigned m_max_steps; unsigned m_max_steps;
bool m_trace; bool m_trace;
bool m_assumptions; bool m_assumptions;
bool m_funext;
bool m_propext;
bool m_standard;
void set_options(options const & o) { void set_options(environment const & env, options const & o) {
m_single_pass = get_simp_single_pass(o); m_single_pass = get_simp_single_pass(o);
m_bottom_up = get_simp_bottom_up(o); m_bottom_up = get_simp_bottom_up(o);
m_beta_eta = get_simp_beta_eta(o); m_beta_eta = get_simp_beta_eta(o);
@ -130,28 +154,43 @@ class simp_tactic_fn {
m_max_steps = get_simp_max_steps(o); m_max_steps = get_simp_max_steps(o);
m_trace = get_simp_trace(o); m_trace = get_simp_trace(o);
m_assumptions = get_simp_assumptions(o); m_assumptions = get_simp_assumptions(o);
if (is_standard(env)) {
m_funext = get_simp_funext(o) && env.find(get_funext_name());
m_propext = get_simp_propext(o) && env.find(get_propext_name());
m_standard = true;
} else {
// TODO(Leo): add support for function extensionality in HoTT mode
m_funext = false;
m_propext = false;
m_standard = false;
}
} }
public: public:
simp_tactic_fn(environment const & env, io_state const & ios, name_generator && ngen, simp_tactic_fn(environment const & env, io_state const & ios, name_generator && ngen,
buffer<expr> const & /* ls */, buffer<name> const & /* ns */, buffer<name> const & /* ex */, buffer<expr> const & /* ls */, buffer<name> const & /* ns */, buffer<name> const & /* ex */,
optional<tactic> const & tac):m_env(env), m_ios(ios), m_ngen(ngen), m_tactic(tac) { optional<tactic> const & tac):m_env(env), m_ios(ios), m_ngen(ngen), m_tactic(tac) {
set_options(m_ios.get_options()); set_options(env, m_ios.get_options());
} }
pair<goal, substitution> operator()(goal const & g, location const & /* loc */, substitution const & s) { enum res_kind { Simplified, Solved, DidNothing };
std::tuple<res_kind, goal, substitution> operator()(goal const & g, location const & /* loc */, substitution const & s) {
m_g = g;
m_steps = 0;
// TODO(Leo) // TODO(Leo)
return mk_pair(g, s); return std::make_tuple(DidNothing, g, s);
} }
}; };
tactic mk_simp_tactic(elaborate_fn const & elab, buffer<expr> const & ls, buffer<name> const & ns, tactic mk_simp_tactic(elaborate_fn const & elab, buffer<expr> const & ls, buffer<name> const & ns,
buffer<name> const & ex, optional<tactic> tac, location const & loc) { buffer<name> const & ex, optional<tactic> tac, location const & loc) {
return tactic([=](environment const & env, io_state const & ios, proof_state const & s) { return tactic01([=](environment const & env, io_state const & ios, proof_state const & s) {
goals const & gs = s.get_goals(); goals const & gs = s.get_goals();
if (empty(gs)) { if (empty(gs)) {
throw_no_goal_if_enabled(s); throw_no_goal_if_enabled(s);
return proof_state_seq(); return none_proof_state();
} }
goal const & g = head(gs); goal const & g = head(gs);
name_generator new_ngen = s.get_ngen(); name_generator new_ngen = s.get_ngen();
@ -166,16 +205,21 @@ tactic mk_simp_tactic(elaborate_fn const & elab, buffer<expr> const & ls, buffer
new_ls.push_back(new_l); new_ls.push_back(new_l);
} }
simp_tactic_fn simp(env, ios, new_ngen.mk_child(), new_ls, ns, ex, tac); simp_tactic_fn simp(env, ios, new_ngen.mk_child(), new_ls, ns, ex, tac);
pair<goal, substitution> r = simp(g, loc, new_subst); goal new_g; simp_tactic_fn::res_kind k;
goal new_g = r.first; std::tie(k, new_g, new_subst) = simp(g, loc, new_subst);
new_subst = r.second; switch (k) {
proof_state new_s(s, cons(new_g, tail(gs)), new_subst, new_ngen); case simp_tactic_fn::Simplified: {
proof_state new_s(s, cons(new_g, tail(gs)), new_subst, new_ngen);
bool fail_if_metavars = true; return some_proof_state(new_s);
tactic post_tac = try_tactic(refl_tactic(elab, fail_if_metavars)); }
// TODO(Leo): remove now_tactic case simp_tactic_fn::Solved: {
post_tac = then(post_tac, now_tactic()); proof_state new_s(s, tail(gs), new_subst, new_ngen);
return post_tac(env, ios, new_s); return some_proof_state(new_s);
}
case simp_tactic_fn::DidNothing:
return none_proof_state();
}
lean_unreachable();
}); });
} }
@ -243,6 +287,14 @@ void initialize_simp_tactic() {
g_simp_assumptions = new name{"simp", "assumptions"}; g_simp_assumptions = new name{"simp", "assumptions"};
register_bool_option(*g_simp_assumptions, LEAN_DEFAULT_SIMP_ASSUMPTIONS, register_bool_option(*g_simp_assumptions, LEAN_DEFAULT_SIMP_ASSUMPTIONS,
"(simp tactic) if true assumptions/hypotheses are automatically used as rewriting rules"); "(simp tactic) if true assumptions/hypotheses are automatically used as rewriting rules");
g_simp_funext = new name{"simp", "funext"};
register_bool_option(*g_simp_funext, LEAN_DEFAULT_SIMP_FUNEXT,
"(simp tactic) avoid function extensionality even if theorem/axiom is in the environment");
g_simp_propext = new name{"simp", "propext"};
register_bool_option(*g_simp_funext, LEAN_DEFAULT_SIMP_PROPEXT,
"(simp tactic) avoid proposition extensionality even if axiom is in the environment, this option is ignored in HoTT mode");
} }
void finalize_simp_tactic() { void finalize_simp_tactic() {
@ -255,5 +307,7 @@ void finalize_simp_tactic() {
delete g_simp_max_steps; delete g_simp_max_steps;
delete g_simp_trace; delete g_simp_trace;
delete g_simp_assumptions; delete g_simp_assumptions;
delete g_simp_funext;
delete g_simp_propext;
} }
} }