fix(library/tactic): add relax_main_opaque flag to proof_state objects, closes #274
This commit is contained in:
parent
a2ef835809
commit
0c185fc4ab
8 changed files with 54 additions and 34 deletions
|
@ -108,7 +108,10 @@ expr elaborator::mk_local(name const & n, expr const & t, binder_info const & bi
|
||||||
|
|
||||||
void elaborator::register_meta(expr const & meta) {
|
void elaborator::register_meta(expr const & meta) {
|
||||||
lean_assert(is_meta(meta));
|
lean_assert(is_meta(meta));
|
||||||
m_mvar2meta.insert(mlocal_name(get_app_fn(meta)), meta);
|
name const & n = mlocal_name(get_app_fn(meta));
|
||||||
|
m_mvar2meta.insert(n, meta);
|
||||||
|
if (m_relax_main_opaque)
|
||||||
|
m_relaxed_mvars.insert(n);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** \brief Convert the metavariable to the metavariable application that captures
|
/** \brief Convert the metavariable to the metavariable application that captures
|
||||||
|
@ -1058,7 +1061,8 @@ void elaborator::solve_unassigned_mvar(substitution & subst, expr mvar, name_set
|
||||||
expr type = m_tc[m_relax_main_opaque]->infer(*meta).first;
|
expr type = m_tc[m_relax_main_opaque]->infer(*meta).first;
|
||||||
// first solve unassigned metavariables in type
|
// first solve unassigned metavariables in type
|
||||||
type = solve_unassigned_mvars(subst, type, visited);
|
type = solve_unassigned_mvars(subst, type, visited);
|
||||||
proof_state ps = to_proof_state(*meta, type, subst, m_ngen.mk_child());
|
bool relax_main_opaque = m_relaxed_mvars.contains(mlocal_name(mvar));
|
||||||
|
proof_state ps = to_proof_state(*meta, type, subst, m_ngen.mk_child(), relax_main_opaque);
|
||||||
if (auto pre_tac = get_pre_tactic_for(subst, mvar, visited)) {
|
if (auto pre_tac = get_pre_tactic_for(subst, mvar, visited)) {
|
||||||
if (is_begin_end_annotation(*pre_tac)) {
|
if (is_begin_end_annotation(*pre_tac)) {
|
||||||
try_using_begin_end(subst, mvar, ps, *pre_tac);
|
try_using_begin_end(subst, mvar, ps, *pre_tac);
|
||||||
|
@ -1109,7 +1113,8 @@ void elaborator::display_unassigned_mvars(expr const & e, substitution const & s
|
||||||
expr meta = tmp_s.instantiate(*it);
|
expr meta = tmp_s.instantiate(*it);
|
||||||
expr meta_type = tmp_s.instantiate(type_checker(env()).infer(meta).first);
|
expr meta_type = tmp_s.instantiate(type_checker(env()).infer(meta).first);
|
||||||
goal g(meta, meta_type);
|
goal g(meta, meta_type);
|
||||||
proof_state ps(goals(g), s, m_ngen, constraints());
|
bool relax = true;
|
||||||
|
proof_state ps(goals(g), s, m_ngen, constraints(), relax);
|
||||||
display_unsolved_proof_state(e, ps, "don't know how to synthesize placeholder");
|
display_unsolved_proof_state(e, ps, "don't know how to synthesize placeholder");
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
|
|
|
@ -35,6 +35,8 @@ class elaborator : public coercion_info_manager {
|
||||||
local_context m_context; // current local context: a list of local constants
|
local_context m_context; // current local context: a list of local constants
|
||||||
local_context m_full_context; // superset of m_context, it also contains non-contextual locals.
|
local_context m_full_context; // superset of m_context, it also contains non-contextual locals.
|
||||||
mvar2meta m_mvar2meta;
|
mvar2meta m_mvar2meta;
|
||||||
|
// Set of metavariable that where created when m_relax_main_opaque flag is set to true.
|
||||||
|
name_set m_relaxed_mvars;
|
||||||
cache m_cache;
|
cache m_cache;
|
||||||
// The following vector contains sorts that we should check
|
// The following vector contains sorts that we should check
|
||||||
// whether the computed universe is too specific or not.
|
// whether the computed universe is too specific or not.
|
||||||
|
|
|
@ -322,7 +322,8 @@ justification mk_failed_to_synthesize_jst(environment const & env, expr const &
|
||||||
substitution tmp(subst);
|
substitution tmp(subst);
|
||||||
expr new_m = instantiate_meta(m, tmp);
|
expr new_m = instantiate_meta(m, tmp);
|
||||||
expr new_type = type_checker(env).infer(new_m).first;
|
expr new_type = type_checker(env).infer(new_m).first;
|
||||||
proof_state ps = to_proof_state(new_m, new_type, name_generator("dontcare"));
|
bool relax_main_opaque = true; // this value doesn't really matter for pretty printing
|
||||||
|
proof_state ps = to_proof_state(new_m, new_type, name_generator("dontcare"), relax_main_opaque);
|
||||||
return format("failed to synthesize placeholder") + line() + ps.pp(fmt);
|
return format("failed to synthesize placeholder") + line() + ps.pp(fmt);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
|
@ -55,12 +55,12 @@ enum subgoals_action_kind { IgnoreSubgoals, AddRevSubgoals, AddSubgoals };
|
||||||
|
|
||||||
static proof_state_seq apply_tactic_core(environment const & env, io_state const & ios, proof_state const & s,
|
static proof_state_seq apply_tactic_core(environment const & env, io_state const & ios, proof_state const & s,
|
||||||
expr const & _e, buffer<constraint> & cs,
|
expr const & _e, buffer<constraint> & cs,
|
||||||
bool add_meta, subgoals_action_kind subgoals_action, bool relax_main_opaque) {
|
bool add_meta, subgoals_action_kind subgoals_action) {
|
||||||
goals const & gs = s.get_goals();
|
goals const & gs = s.get_goals();
|
||||||
if (empty(gs))
|
if (empty(gs))
|
||||||
return proof_state_seq();
|
return proof_state_seq();
|
||||||
name_generator ngen = s.get_ngen();
|
name_generator ngen = s.get_ngen();
|
||||||
std::shared_ptr<type_checker> tc(mk_type_checker(env, ngen.mk_child(), relax_main_opaque));
|
std::shared_ptr<type_checker> tc(mk_type_checker(env, ngen.mk_child(), s.relax_main_opaque()));
|
||||||
goal g = head(gs);
|
goal g = head(gs);
|
||||||
goals tail_gs = tail(gs);
|
goals tail_gs = tail(gs);
|
||||||
expr t = g.get_type();
|
expr t = g.get_type();
|
||||||
|
@ -86,7 +86,7 @@ static proof_state_seq apply_tactic_core(environment const & env, io_state const
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
metavar_closure cls(t);
|
metavar_closure cls(t);
|
||||||
cls.mk_constraints(s.get_subst(), justification(), relax_main_opaque);
|
cls.mk_constraints(s.get_subst(), justification(), s.relax_main_opaque());
|
||||||
pair<bool, constraint_seq> dcs = tc->is_def_eq(t, e_t);
|
pair<bool, constraint_seq> dcs = tc->is_def_eq(t, e_t);
|
||||||
if (!dcs.first)
|
if (!dcs.first)
|
||||||
return proof_state_seq();
|
return proof_state_seq();
|
||||||
|
@ -123,17 +123,17 @@ static proof_state_seq apply_tactic_core(environment const & env, io_state const
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return proof_state(new_gs, new_subst, new_ngen, postponed);
|
return proof_state(s, new_gs, new_subst, new_ngen, postponed);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
static proof_state_seq apply_tactic_core(environment const & env, io_state const & ios, proof_state const & s, expr const & e,
|
static proof_state_seq apply_tactic_core(environment const & env, io_state const & ios, proof_state const & s, expr const & e,
|
||||||
bool add_meta, subgoals_action_kind subgoals_action, bool relax_main_opaque) {
|
bool add_meta, subgoals_action_kind subgoals_action) {
|
||||||
buffer<constraint> cs;
|
buffer<constraint> cs;
|
||||||
return apply_tactic_core(env, ios, s, e, cs, add_meta, subgoals_action, relax_main_opaque);
|
return apply_tactic_core(env, ios, s, e, cs, add_meta, subgoals_action);
|
||||||
}
|
}
|
||||||
|
|
||||||
tactic eassumption_tactic(bool relax_main_opaque) {
|
tactic eassumption_tactic() {
|
||||||
return tactic([=](environment const & env, io_state const & ios, proof_state const & s) {
|
return tactic([=](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))
|
||||||
|
@ -143,13 +143,13 @@ tactic eassumption_tactic(bool relax_main_opaque) {
|
||||||
buffer<expr> hs;
|
buffer<expr> hs;
|
||||||
get_app_args(g.get_meta(), hs);
|
get_app_args(g.get_meta(), hs);
|
||||||
for (expr const & h : hs) {
|
for (expr const & h : hs) {
|
||||||
r = append(r, apply_tactic_core(env, ios, s, h, false, IgnoreSubgoals, relax_main_opaque));
|
r = append(r, apply_tactic_core(env, ios, s, h, false, IgnoreSubgoals));
|
||||||
}
|
}
|
||||||
return r;
|
return r;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
tactic apply_tactic(elaborate_fn const & elab, expr const & e, bool rev, bool relax_main_opaque) {
|
tactic apply_tactic(elaborate_fn const & elab, expr const & e, bool rev) {
|
||||||
return tactic([=](environment const & env, io_state const & ios, proof_state const & s) {
|
return tactic([=](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))
|
||||||
|
@ -162,8 +162,8 @@ tactic apply_tactic(elaborate_fn const & elab, expr const & e, bool rev, bool re
|
||||||
new_e = ecs.first;
|
new_e = ecs.first;
|
||||||
to_buffer(ecs.second, cs);
|
to_buffer(ecs.second, cs);
|
||||||
to_buffer(s.get_postponed(), cs);
|
to_buffer(s.get_postponed(), cs);
|
||||||
proof_state new_s(s.get_goals(), s.get_subst(), ngen, constraints());
|
proof_state new_s(s, ngen, constraints());
|
||||||
return apply_tactic_core(env, ios, new_s, new_e, cs, true, rev ? AddRevSubgoals : AddSubgoals, relax_main_opaque);
|
return apply_tactic_core(env, ios, new_s, new_e, cs, true, rev ? AddRevSubgoals : AddSubgoals);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -8,8 +8,8 @@ Author: Leonardo de Moura
|
||||||
#include "util/lua.h"
|
#include "util/lua.h"
|
||||||
#include "library/tactic/elaborate.h"
|
#include "library/tactic/elaborate.h"
|
||||||
namespace lean {
|
namespace lean {
|
||||||
tactic apply_tactic(elaborate_fn const & fn, expr const & e, bool rev_goals = false, bool relax_main_opaque = true);
|
tactic apply_tactic(elaborate_fn const & fn, expr const & e, bool rev_goals = false);
|
||||||
tactic eassumption_tactic(bool relax_main_opaque = true);
|
tactic eassumption_tactic();
|
||||||
void open_apply_tactic(lua_State * L);
|
void open_apply_tactic(lua_State * L);
|
||||||
void initialize_apply_tactic();
|
void initialize_apply_tactic();
|
||||||
void finalize_apply_tactic();
|
void finalize_apply_tactic();
|
||||||
|
|
|
@ -22,7 +22,7 @@ bool solve_constraints(environment const & env, io_state const & ios, proof_stat
|
||||||
if (auto p = rseq.pull()) {
|
if (auto p = rseq.pull()) {
|
||||||
substitution new_subst = p->first.first;
|
substitution new_subst = p->first.first;
|
||||||
constraints new_postponed = p->first.second;
|
constraints new_postponed = p->first.second;
|
||||||
ps = proof_state(ps.get_goals(), new_subst, ngen, new_postponed);
|
ps = proof_state(ps, ps.get_goals(), new_subst, ngen, new_postponed);
|
||||||
return true;
|
return true;
|
||||||
} else {
|
} else {
|
||||||
return false;
|
return false;
|
||||||
|
@ -62,7 +62,7 @@ optional<expr> elaborate_with_respect_to(environment const & env, io_state const
|
||||||
substitution new_subst = p->first.first;
|
substitution new_subst = p->first.first;
|
||||||
constraints new_postponed = p->first.second;
|
constraints new_postponed = p->first.second;
|
||||||
new_e = new_subst.instantiate(new_e);
|
new_e = new_subst.instantiate(new_e);
|
||||||
s = proof_state(gs, new_subst, ngen, new_postponed);
|
s = proof_state(s, gs, new_subst, ngen, new_postponed);
|
||||||
return some_expr(new_e);
|
return some_expr(new_e);
|
||||||
} else {
|
} else {
|
||||||
return none_expr();
|
return none_expr();
|
||||||
|
|
|
@ -30,9 +30,9 @@ bool get_proof_state_goal_names(options const & opts) {
|
||||||
return opts.get_bool(*g_proof_state_goal_names, LEAN_PROOF_STATE_GOAL_NAMES);
|
return opts.get_bool(*g_proof_state_goal_names, LEAN_PROOF_STATE_GOAL_NAMES);
|
||||||
}
|
}
|
||||||
|
|
||||||
proof_state::proof_state(goals const & gs, substitution const & s,
|
proof_state::proof_state(goals const & gs, substitution const & s, name_generator const & ngen,
|
||||||
name_generator const & ngen, constraints const & postponed):
|
constraints const & postponed, bool relax_main_opaque):
|
||||||
m_goals(gs), m_subst(s), m_ngen(ngen), m_postponed(postponed) {
|
m_goals(gs), m_subst(s), m_ngen(ngen), m_postponed(postponed), m_relax_main_opaque(relax_main_opaque) {
|
||||||
if (std::any_of(gs.begin(), gs.end(),
|
if (std::any_of(gs.begin(), gs.end(),
|
||||||
[&](goal const & g) { return s.is_assigned(g.get_mvar()); })) {
|
[&](goal const & g) { return s.is_assigned(g.get_mvar()); })) {
|
||||||
m_goals = filter(gs, [&](goal const & g) { return !s.is_assigned(g.get_mvar()); });
|
m_goals = filter(gs, [&](goal const & g) { return !s.is_assigned(g.get_mvar()); });
|
||||||
|
@ -77,12 +77,13 @@ io_state_stream const & operator<<(io_state_stream const & out, proof_state cons
|
||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
|
|
||||||
proof_state to_proof_state(expr const & meta, expr const & type, substitution const & subst, name_generator ngen) {
|
proof_state to_proof_state(expr const & meta, expr const & type, substitution const & subst, name_generator ngen,
|
||||||
return proof_state(goals(goal(meta, type)), subst, ngen, constraints());
|
bool relax_main_opaque) {
|
||||||
|
return proof_state(goals(goal(meta, type)), subst, ngen, constraints(), relax_main_opaque);
|
||||||
}
|
}
|
||||||
|
|
||||||
proof_state to_proof_state(expr const & meta, expr const & type, name_generator ngen) {
|
proof_state to_proof_state(expr const & meta, expr const & type, name_generator ngen, bool relax_main_opaque) {
|
||||||
return to_proof_state(meta, type, substitution(), ngen);
|
return to_proof_state(meta, type, substitution(), ngen, relax_main_opaque);
|
||||||
}
|
}
|
||||||
|
|
||||||
DECL_UDATA(goals)
|
DECL_UDATA(goals)
|
||||||
|
@ -165,8 +166,9 @@ static int mk_proof_state(lua_State * L) {
|
||||||
} else if (nargs == 3 && is_proof_state(L, 1)) {
|
} else if (nargs == 3 && is_proof_state(L, 1)) {
|
||||||
return push_proof_state(L, proof_state(to_proof_state(L, 1), to_goals(L, 2), to_substitution(L, 3)));
|
return push_proof_state(L, proof_state(to_proof_state(L, 1), to_goals(L, 2), to_substitution(L, 3)));
|
||||||
} else if (nargs == 3) {
|
} else if (nargs == 3) {
|
||||||
|
bool relax_main_opaque = true;
|
||||||
return push_proof_state(L, proof_state(to_goals(L, 1), to_substitution(L, 2), to_name_generator(L, 3),
|
return push_proof_state(L, proof_state(to_goals(L, 1), to_substitution(L, 2), to_name_generator(L, 3),
|
||||||
constraints()));
|
constraints(), relax_main_opaque));
|
||||||
} else {
|
} else {
|
||||||
throw exception("proof_state invalid number of arguments");
|
throw exception("proof_state invalid number of arguments");
|
||||||
}
|
}
|
||||||
|
@ -175,10 +177,11 @@ static int mk_proof_state(lua_State * L) {
|
||||||
static name * g_tmp_prefix = nullptr;
|
static name * g_tmp_prefix = nullptr;
|
||||||
static int to_proof_state(lua_State * L) {
|
static int to_proof_state(lua_State * L) {
|
||||||
int nargs = lua_gettop(L);
|
int nargs = lua_gettop(L);
|
||||||
|
bool relax_main_opaque = true;
|
||||||
if (nargs == 2)
|
if (nargs == 2)
|
||||||
return push_proof_state(L, to_proof_state(to_expr(L, 1), to_expr(L, 2), name_generator(*g_tmp_prefix)));
|
return push_proof_state(L, to_proof_state(to_expr(L, 1), to_expr(L, 2), name_generator(*g_tmp_prefix), relax_main_opaque));
|
||||||
else
|
else
|
||||||
return push_proof_state(L, to_proof_state(to_expr(L, 1), to_expr(L, 2), to_name_generator(L, 3)));
|
return push_proof_state(L, to_proof_state(to_expr(L, 1), to_expr(L, 2), to_name_generator(L, 3), relax_main_opaque));
|
||||||
}
|
}
|
||||||
|
|
||||||
static int proof_state_tostring(lua_State * L) {
|
static int proof_state_tostring(lua_State * L) {
|
||||||
|
|
|
@ -20,23 +20,31 @@ class proof_state {
|
||||||
substitution m_subst;
|
substitution m_subst;
|
||||||
name_generator m_ngen;
|
name_generator m_ngen;
|
||||||
constraints m_postponed;
|
constraints m_postponed;
|
||||||
|
bool m_relax_main_opaque;
|
||||||
public:
|
public:
|
||||||
proof_state(goals const & gs, substitution const & s, name_generator const & ngen, constraints const & postponed);
|
proof_state(goals const & gs, substitution const & s, name_generator const & ngen, constraints const & postponed,
|
||||||
|
bool relax_main_opaque);
|
||||||
|
proof_state(proof_state const & s, goals const & gs, substitution const & subst, name_generator const & ngen,
|
||||||
|
constraints const & postponed):
|
||||||
|
proof_state(gs, subst, ngen, postponed, s.relax_main_opaque()) {}
|
||||||
proof_state(proof_state const & s, goals const & gs, substitution const & subst, name_generator const & ngen):
|
proof_state(proof_state const & s, goals const & gs, substitution const & subst, name_generator const & ngen):
|
||||||
proof_state(gs, subst, ngen, s.m_postponed) {}
|
proof_state(gs, subst, ngen, s.m_postponed, s.m_relax_main_opaque) {}
|
||||||
proof_state(proof_state const & s, goals const & gs, substitution const & subst):
|
proof_state(proof_state const & s, goals const & gs, substitution const & subst):
|
||||||
proof_state(s, gs, subst, s.m_ngen) {}
|
proof_state(s, gs, subst, s.m_ngen) {}
|
||||||
proof_state(proof_state const & s, goals const & gs, name_generator const & ngen):
|
proof_state(proof_state const & s, goals const & gs, name_generator const & ngen):
|
||||||
proof_state(s, gs, s.m_subst, ngen) {}
|
proof_state(s, gs, s.m_subst, ngen) {}
|
||||||
proof_state(proof_state const & s, goals const & gs):
|
proof_state(proof_state const & s, goals const & gs):
|
||||||
proof_state(s, gs, s.m_subst) {}
|
proof_state(s, gs, s.m_subst) {}
|
||||||
|
proof_state(proof_state const & s, name_generator const & ngen, constraints const & postponed):
|
||||||
|
proof_state(s.m_goals, s.m_subst, ngen, postponed, s.m_relax_main_opaque) {}
|
||||||
proof_state(proof_state const & s, name_generator const & ngen):
|
proof_state(proof_state const & s, name_generator const & ngen):
|
||||||
proof_state(s.m_goals, s.m_subst, ngen, s.m_postponed) {}
|
proof_state(s, ngen, s.m_postponed) {}
|
||||||
|
|
||||||
goals const & get_goals() const { return m_goals; }
|
goals const & get_goals() const { return m_goals; }
|
||||||
substitution const & get_subst() const { return m_subst; }
|
substitution const & get_subst() const { return m_subst; }
|
||||||
name_generator const & get_ngen() const { return m_ngen; }
|
name_generator const & get_ngen() const { return m_ngen; }
|
||||||
constraints const & get_postponed() const { return m_postponed; }
|
constraints const & get_postponed() const { return m_postponed; }
|
||||||
|
bool relax_main_opaque() const { return m_relax_main_opaque; }
|
||||||
|
|
||||||
/** \brief Return true iff this state does not have any goals left */
|
/** \brief Return true iff this state does not have any goals left */
|
||||||
bool is_final_state() const { return empty(m_goals); }
|
bool is_final_state() const { return empty(m_goals); }
|
||||||
|
@ -46,8 +54,9 @@ public:
|
||||||
inline optional<proof_state> some_proof_state(proof_state const & s) { return some(s); }
|
inline optional<proof_state> some_proof_state(proof_state const & s) { return some(s); }
|
||||||
inline optional<proof_state> none_proof_state() { return optional<proof_state> (); }
|
inline optional<proof_state> none_proof_state() { return optional<proof_state> (); }
|
||||||
|
|
||||||
proof_state to_proof_state(expr const & meta, expr const & type, substitution const & subst, name_generator ngen);
|
proof_state to_proof_state(expr const & meta, expr const & type, substitution const & subst, name_generator ngen,
|
||||||
proof_state to_proof_state(expr const & meta, expr const & type, name_generator ngen);
|
bool relax_main_opaque);
|
||||||
|
proof_state to_proof_state(expr const & meta, expr const & type, name_generator ngen, bool relax_main_opaque);
|
||||||
|
|
||||||
goals map_goals(proof_state const & s, std::function<optional<goal>(goal const & g)> f);
|
goals map_goals(proof_state const & s, std::function<optional<goal>(goal const & g)> f);
|
||||||
io_state_stream const & operator<<(io_state_stream const & out, proof_state const & s);
|
io_state_stream const & operator<<(io_state_stream const & out, proof_state const & s);
|
||||||
|
|
Loading…
Reference in a new issue