2015-09-16 14:49:39 +00:00
|
|
|
/*
|
|
|
|
Copyright (c) 2015 Microsoft Corporation. All rights reserved.
|
|
|
|
Released under Apache 2.0 license as described in the file LICENSE.
|
|
|
|
|
|
|
|
Author: Leonardo de Moura
|
|
|
|
*/
|
|
|
|
#pragma once
|
2015-09-24 01:13:18 +00:00
|
|
|
#include "util/rb_map.h"
|
2015-09-16 14:49:39 +00:00
|
|
|
#include "kernel/expr.h"
|
2015-09-29 01:28:11 +00:00
|
|
|
#include "library/tactic/goal.h"
|
2015-09-24 01:13:18 +00:00
|
|
|
#include "library/blast/hypothesis.h"
|
2015-09-25 21:43:42 +00:00
|
|
|
#include "library/blast/branch.h"
|
2015-09-24 01:13:18 +00:00
|
|
|
|
2015-09-16 14:49:39 +00:00
|
|
|
namespace lean {
|
|
|
|
namespace blast {
|
2015-09-24 01:13:18 +00:00
|
|
|
class metavar_decl {
|
2015-10-05 04:59:37 +00:00
|
|
|
// A metavariable can be assigned to a value that contains references only to the assumptions
|
|
|
|
// that were available when the metavariable was defined.
|
|
|
|
hypothesis_idx_set m_assumptions;
|
|
|
|
expr m_type;
|
2015-09-25 21:43:42 +00:00
|
|
|
public:
|
|
|
|
metavar_decl() {}
|
2015-10-05 04:59:37 +00:00
|
|
|
metavar_decl(hypothesis_idx_set const & a, expr const & t):
|
|
|
|
m_assumptions(a), m_type(t) {}
|
2015-10-04 01:18:44 +00:00
|
|
|
/** \brief Return true iff \c h is in the context of the this metavar declaration */
|
2015-10-05 04:59:37 +00:00
|
|
|
bool contains_href(unsigned hidx) const { return m_assumptions.contains(hidx); }
|
|
|
|
bool contains_href(expr const & h) const { return contains_href(href_index(h)); }
|
2015-09-29 01:28:11 +00:00
|
|
|
expr const & get_type() const { return m_type; }
|
2015-10-04 01:18:44 +00:00
|
|
|
/** \brief Make sure the declaration context of this declaration is a subset of \c other.
|
|
|
|
\remark Return true iff the context has been modified. */
|
|
|
|
bool restrict_context_using(metavar_decl const & other);
|
2015-10-05 04:59:37 +00:00
|
|
|
hypothesis_idx_set get_assumptions() const { return m_assumptions; }
|
2015-09-24 01:13:18 +00:00
|
|
|
};
|
|
|
|
|
2015-11-09 01:29:47 +00:00
|
|
|
class proof_step_cell {
|
|
|
|
MK_LEAN_RC(); // Declare m_rc counter
|
|
|
|
void dealloc() { delete this; }
|
|
|
|
public:
|
|
|
|
virtual ~proof_step_cell() {}
|
|
|
|
/** \brief Every proof-step must provide a resolve method.
|
|
|
|
When the branch created by the proof-step is closed,
|
|
|
|
a proof pr is provided, and the proof-step can perform two operations
|
|
|
|
1- setup the next branch and return none_expr
|
|
|
|
2- finish and return a new proof */
|
|
|
|
virtual optional<expr> resolve(state & s, expr const & pr) = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
class proof_step {
|
|
|
|
proof_step_cell * m_ptr;
|
|
|
|
public:
|
|
|
|
proof_step():m_ptr(nullptr) {}
|
2015-11-09 03:18:40 +00:00
|
|
|
proof_step(proof_step_cell * c):m_ptr(c) { m_ptr->inc_ref(); }
|
2015-11-09 01:29:47 +00:00
|
|
|
proof_step(proof_step const & s):m_ptr(s.m_ptr) { if (m_ptr) m_ptr->inc_ref(); }
|
|
|
|
proof_step(proof_step && s):m_ptr(s.m_ptr) { s.m_ptr = nullptr; }
|
|
|
|
~proof_step() { if (m_ptr) m_ptr->dec_ref(); }
|
|
|
|
proof_step & operator=(proof_step const & s) { LEAN_COPY_REF(s); }
|
|
|
|
proof_step & operator=(proof_step && s) { LEAN_MOVE_REF(s); }
|
|
|
|
|
|
|
|
optional<expr> resolve(state & s, expr const & pr) {
|
|
|
|
lean_assert(m_ptr);
|
|
|
|
return m_ptr->resolve(s, pr);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2015-09-16 14:49:39 +00:00
|
|
|
class state {
|
2015-09-29 21:08:14 +00:00
|
|
|
typedef metavar_idx_map<metavar_decl> metavar_decls;
|
2015-10-02 20:11:17 +00:00
|
|
|
typedef metavar_idx_map<expr> eassignment;
|
|
|
|
typedef metavar_idx_map<level> uassignment;
|
2015-09-29 21:08:14 +00:00
|
|
|
typedef hypothesis_idx_map<metavar_idx_set> fixed_by;
|
2015-11-09 01:29:47 +00:00
|
|
|
typedef list<proof_step> proof_steps;
|
2015-10-02 22:48:01 +00:00
|
|
|
unsigned m_next_uref_index; // index of the next universe metavariable
|
2015-10-02 20:11:17 +00:00
|
|
|
uassignment m_uassignment;
|
|
|
|
unsigned m_next_mref_index; // index of the next metavariable
|
|
|
|
metavar_decls m_metavar_decls;
|
|
|
|
eassignment m_eassignment;
|
|
|
|
branch m_main;
|
2015-09-29 21:08:14 +00:00
|
|
|
// In the following mapping, each entry (h -> {m_1 ... m_n}) means that hypothesis `h` cannot be cleared
|
|
|
|
// in any branch where the metavariables m_1 ... m_n have not been replaced with the values assigned to them.
|
|
|
|
// That is, to be able to clear `h` in a branch `B`, we first need to check whether it
|
|
|
|
// is contained in this mapping or not. If it is, we should check whether any of the
|
|
|
|
// metavariables `m_1` ... `m_n` occur in `B` (this is a relatively quick check since
|
|
|
|
// `B` contains an over-approximation of all meta-variables occuring in it (i.e., m_mvar_idxs).
|
|
|
|
// If this check fails, then we should replace any assigned `m_i` with its value, if the intersection is still
|
|
|
|
// non-empty, then we cannot clear `h`.
|
2015-11-09 01:29:47 +00:00
|
|
|
fixed_by m_fixed_by;
|
|
|
|
unsigned m_depth{0};
|
|
|
|
proof_steps m_proof_steps;
|
2015-09-28 23:40:19 +00:00
|
|
|
|
2015-09-29 21:08:14 +00:00
|
|
|
void add_fixed_by(unsigned hidx, unsigned midx);
|
2015-09-28 23:40:19 +00:00
|
|
|
unsigned add_metavar_decl(metavar_decl const & decl);
|
2015-09-29 01:28:11 +00:00
|
|
|
goal to_goal(branch const &) const;
|
2015-09-28 23:40:19 +00:00
|
|
|
|
2015-09-29 01:55:24 +00:00
|
|
|
#ifdef LEAN_DEBUG
|
2015-09-29 19:42:20 +00:00
|
|
|
bool check_hypothesis(expr const & e, branch const & b, unsigned hidx, hypothesis const & h) const;
|
|
|
|
bool check_hypothesis(branch const & b, unsigned hidx, hypothesis const & h) const;
|
|
|
|
bool check_target(branch const & b) const;
|
2015-09-29 01:55:24 +00:00
|
|
|
bool check_invariant(branch const &) const;
|
|
|
|
#endif
|
2015-09-25 21:43:42 +00:00
|
|
|
public:
|
|
|
|
state();
|
2015-10-04 01:18:44 +00:00
|
|
|
|
2015-10-02 22:48:01 +00:00
|
|
|
level mk_uref();
|
|
|
|
|
2015-10-04 01:18:44 +00:00
|
|
|
bool is_uref_assigned(level const & l) const {
|
|
|
|
lean_assert(is_uref(l));
|
|
|
|
return m_uassignment.contains(uref_index(l));
|
|
|
|
}
|
|
|
|
|
|
|
|
// u := l
|
|
|
|
void assign_uref(level const & u, level const & l) {
|
|
|
|
m_uassignment.insert(uref_index(u), l);
|
|
|
|
}
|
|
|
|
|
|
|
|
level const * get_uref_assignment(level const & l) const {
|
|
|
|
lean_assert(is_uref_assigned(l));
|
|
|
|
return m_uassignment.find(uref_index(l));
|
|
|
|
}
|
|
|
|
|
|
|
|
/** \brief Instantiate any assigned uref in \c l with its assignment.
|
|
|
|
\remark This is not a const method because it may normalize the assignment. */
|
|
|
|
level instantiate_urefs(level const & l);
|
|
|
|
|
2015-09-28 23:40:19 +00:00
|
|
|
/** \brief Create a new metavariable using the given type and context.
|
|
|
|
\pre ctx must be a subset of the hypotheses in the main branch. */
|
|
|
|
expr mk_metavar(hypothesis_idx_buffer const & ctx, expr const & type);
|
2015-10-05 04:59:37 +00:00
|
|
|
expr mk_metavar(hypothesis_idx_set const & ctx, expr const & type);
|
2015-11-09 01:29:47 +00:00
|
|
|
/** \brief Create a new metavariable using the given type.
|
2015-10-05 04:59:37 +00:00
|
|
|
The context of this metavariable will be all assumption hypotheses occurring in the main branch. */
|
2015-09-28 23:40:19 +00:00
|
|
|
expr mk_metavar(expr const & type);
|
|
|
|
|
2015-10-04 01:18:44 +00:00
|
|
|
/** \brief Make sure the metavariable declaration context of mref1 is a
|
|
|
|
subset of the metavariable declaration context of mref2. */
|
|
|
|
void restrict_mref_context_using(expr const & mref1, expr const & mref2);
|
|
|
|
|
2015-10-02 22:21:28 +00:00
|
|
|
bool is_mref_assigned(expr const & e) const {
|
|
|
|
lean_assert(is_mref(e));
|
|
|
|
return m_eassignment.contains(mref_index(e));
|
|
|
|
}
|
|
|
|
|
2015-10-04 01:18:44 +00:00
|
|
|
/** \brief Return true iff \c l contains an assigned uref */
|
|
|
|
bool has_assigned_uref(level const & l) const;
|
|
|
|
bool has_assigned_uref(levels const & ls) const;
|
|
|
|
|
|
|
|
expr const * get_mref_assignment(expr const & e) const {
|
|
|
|
lean_assert(is_mref(e));
|
|
|
|
return m_eassignment.find(mref_index(e));
|
|
|
|
}
|
|
|
|
|
|
|
|
// m := e
|
|
|
|
void assign_mref(expr const & m, expr const & e) {
|
|
|
|
m_eassignment.insert(mref_index(m), e);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** \brief Return true if \c e contains an assigned mref or uref */
|
|
|
|
bool has_assigned_uref_mref(expr const & e) const;
|
|
|
|
|
|
|
|
/** \brief Instantiate any assigned mref in \c e with its assignment.
|
|
|
|
\remark This is not a const method because it may normalize the assignment. */
|
|
|
|
expr instantiate_urefs_mrefs(expr const & e);
|
|
|
|
|
2015-09-29 00:39:30 +00:00
|
|
|
/** \brief Add a new hypothesis to the main branch */
|
2015-10-05 04:27:39 +00:00
|
|
|
expr add_hypothesis(name const & n, expr const & type, expr const & value) {
|
|
|
|
return m_main.add_hypothesis(n, type, value);
|
2015-09-29 00:39:30 +00:00
|
|
|
}
|
|
|
|
|
2015-10-02 20:11:17 +00:00
|
|
|
branch & get_main_branch() { return m_main; }
|
|
|
|
branch const & get_main_branch() const { return m_main; }
|
|
|
|
|
2015-09-29 00:39:30 +00:00
|
|
|
/** \brief Add a new hypothesis to the main branch */
|
2015-10-05 04:27:39 +00:00
|
|
|
expr add_hypothesis(expr const & type, expr const & value) {
|
|
|
|
return m_main.add_hypothesis(type, value);
|
2015-09-29 00:39:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/** \brief Set target (aka conclusion, aka type of the goal, aka type of the term that must be synthesize in this branch)
|
|
|
|
of the main branch */
|
|
|
|
void set_target(expr const & type) {
|
|
|
|
return m_main.set_target(type);
|
|
|
|
}
|
|
|
|
|
2015-09-25 21:43:42 +00:00
|
|
|
metavar_decl const * get_metavar_decl(unsigned idx) const { return m_metavar_decls.find(idx); }
|
|
|
|
metavar_decl const * get_metavar_decl(expr const & e) const { return get_metavar_decl(mref_index(e)); }
|
2015-09-29 01:28:11 +00:00
|
|
|
|
|
|
|
/** \brief Convert main branch to a goal.
|
|
|
|
This is mainly used for pretty printing. However, in the future, we may use this capability
|
|
|
|
to invoke the tactic framework from the blast tactic. */
|
|
|
|
goal to_goal() const;
|
2015-09-29 01:55:24 +00:00
|
|
|
|
2015-09-29 17:04:11 +00:00
|
|
|
void display(environment const & env, io_state const & ios) const;
|
|
|
|
|
2015-10-04 01:18:44 +00:00
|
|
|
/** Auxiliary object for creating snapshots of the metavariable assignments.
|
|
|
|
\remark The snapshots are created (and restored) in constant time */
|
|
|
|
class assignment_snapshot {
|
|
|
|
state & m_state;
|
|
|
|
uassignment m_old_uassignment;
|
|
|
|
eassignment m_old_eassignment;
|
|
|
|
public:
|
|
|
|
assignment_snapshot(state & s):
|
|
|
|
m_state(s),
|
|
|
|
m_old_uassignment(s.m_uassignment),
|
|
|
|
m_old_eassignment(s.m_eassignment) {}
|
|
|
|
void restore() {
|
|
|
|
m_state.m_uassignment = m_old_uassignment;
|
|
|
|
m_state.m_eassignment = m_old_eassignment;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2015-11-09 01:29:47 +00:00
|
|
|
void push_proof_step(proof_step const & ps) {
|
|
|
|
m_depth++;
|
|
|
|
m_proof_steps = cons(ps, m_proof_steps);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool has_proof_steps() const {
|
|
|
|
return static_cast<bool>(m_proof_steps);
|
|
|
|
}
|
|
|
|
|
|
|
|
proof_step top_proof_step() const {
|
|
|
|
return head(m_proof_steps);
|
|
|
|
}
|
|
|
|
|
|
|
|
void pop_proof_step() {
|
|
|
|
lean_assert(m_proof_steps);
|
|
|
|
m_depth--;
|
|
|
|
m_proof_steps = tail(m_proof_steps);
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned get_depth() const { return m_depth; }
|
|
|
|
|
2015-09-29 01:55:24 +00:00
|
|
|
#ifdef LEAN_DEBUG
|
|
|
|
bool check_invariant() const;
|
|
|
|
#endif
|
2015-09-16 14:49:39 +00:00
|
|
|
};
|
|
|
|
}}
|