2013-10-01 00:58:46 +00:00
|
|
|
/*
|
|
|
|
Copyright (c) 2013 Microsoft Corporation. All rights reserved.
|
|
|
|
Released under Apache 2.0 license as described in the file LICENSE.
|
|
|
|
|
|
|
|
Author: Leonardo de Moura
|
|
|
|
*/
|
2013-10-27 18:02:29 +00:00
|
|
|
#include <vector>
|
2013-10-14 18:50:51 +00:00
|
|
|
#include "util/buffer.h"
|
2014-02-21 01:19:03 +00:00
|
|
|
#include "util/int64.h"
|
2013-10-23 20:42:14 +00:00
|
|
|
#include "kernel/justification.h"
|
2014-01-04 01:13:10 +00:00
|
|
|
#include "kernel/metavar.h"
|
2013-10-01 00:58:46 +00:00
|
|
|
|
|
|
|
namespace lean {
|
2014-02-21 01:19:03 +00:00
|
|
|
format to_pos(optional<expr> const & e, pos_info_provider const * p) {
|
2013-10-16 00:32:02 +00:00
|
|
|
if (!p || !e)
|
2014-02-21 01:19:03 +00:00
|
|
|
return format();
|
2013-12-08 07:21:07 +00:00
|
|
|
format f = p->pp(*e);
|
2013-10-16 00:32:02 +00:00
|
|
|
if (!f)
|
2014-02-21 01:19:03 +00:00
|
|
|
return format();
|
|
|
|
return f + space();
|
2013-10-16 00:32:02 +00:00
|
|
|
}
|
|
|
|
|
2014-02-21 01:19:03 +00:00
|
|
|
typedef uint64 approx_set;
|
|
|
|
static approx_set mk_empty_set() { return 0; }
|
|
|
|
static approx_set mk_union(approx_set s1, approx_set s2) { return s1 | s2; }
|
|
|
|
static approx_set mk_intersection(approx_set s1, approx_set s2) { return s1 & s2; }
|
|
|
|
static approx_set mk_singleton(unsigned i) { return static_cast<uint64>(1) << (i % 64); }
|
|
|
|
static approx_set may_contain(approx_set s, unsigned i) { return mk_intersection(s, mk_singleton(i)) != 0ull; }
|
2013-10-14 18:50:51 +00:00
|
|
|
|
2014-02-21 01:19:03 +00:00
|
|
|
enum class justification_kind { Asserted, Composite, ExtComposite, Assumption, ExtAssumption };
|
|
|
|
|
|
|
|
approx_set get_approx_assumption_set(justification const & j);
|
|
|
|
|
|
|
|
struct justification_cell {
|
|
|
|
MK_LEAN_RC();
|
|
|
|
justification_kind m_kind;
|
|
|
|
void dealloc();
|
|
|
|
justification_cell(justification_kind k):m_rc(0), m_kind(k) {}
|
|
|
|
bool is_asserted() const { return m_kind == justification_kind::Asserted; }
|
|
|
|
bool is_assumption() const { return m_kind == justification_kind::Assumption || m_kind == justification_kind::ExtAssumption; }
|
|
|
|
bool is_composite() const { return m_kind == justification_kind::Composite || m_kind == justification_kind::ExtComposite; }
|
|
|
|
bool is_ext_assumption() const { return m_kind == justification_kind::ExtAssumption; }
|
|
|
|
bool is_ext_composite() const { return m_kind == justification_kind::ExtComposite; }
|
|
|
|
};
|
|
|
|
|
|
|
|
struct asserted_cell : public justification_cell {
|
|
|
|
pp_jst_fn m_fn;
|
|
|
|
optional<expr> m_expr;
|
|
|
|
asserted_cell(pp_jst_fn const & fn, optional<expr> const & e):
|
|
|
|
justification_cell(justification_kind::Asserted),
|
|
|
|
m_fn(fn), m_expr(e) {}
|
|
|
|
};
|
|
|
|
|
|
|
|
struct composite_cell : public justification_cell {
|
|
|
|
approx_set m_assumption_set; // approximated set of assumptions contained in child1 and child2
|
|
|
|
justification m_child[2];
|
|
|
|
composite_cell(justification_kind k, justification const & j1, justification const & j2):
|
|
|
|
justification_cell(k) {
|
|
|
|
m_child[0] = j1;
|
|
|
|
m_child[1] = j2;
|
|
|
|
m_assumption_set = mk_union(get_approx_assumption_set(j1), get_approx_assumption_set(j2));
|
|
|
|
}
|
|
|
|
composite_cell(justification const & j1, justification const & j2):
|
|
|
|
composite_cell(justification_kind::Composite, j1, j2) {}
|
|
|
|
};
|
|
|
|
|
|
|
|
struct ext_composite_cell : public composite_cell {
|
|
|
|
pp_jst_fn m_fn;
|
|
|
|
optional<expr> m_expr;
|
|
|
|
ext_composite_cell(justification const & j1, justification const & j2, pp_jst_fn const & fn, optional<expr> const & e):
|
|
|
|
composite_cell(justification_kind::ExtComposite, j1, j2),
|
|
|
|
m_fn(fn),
|
|
|
|
m_expr(e) {}
|
|
|
|
};
|
|
|
|
|
|
|
|
struct assumption_cell : public justification_cell {
|
|
|
|
unsigned m_idx;
|
|
|
|
assumption_cell(justification_kind k, unsigned idx):
|
|
|
|
justification_cell(k), m_idx(idx) {}
|
|
|
|
assumption_cell(unsigned idx):
|
|
|
|
assumption_cell(justification_kind::Assumption, idx) {}
|
|
|
|
};
|
|
|
|
|
|
|
|
struct ext_assumption_cell : public assumption_cell {
|
|
|
|
pp_jst_fn m_fn;
|
|
|
|
optional<expr> m_expr;
|
|
|
|
ext_assumption_cell(unsigned idx, pp_jst_fn const & fn, optional<expr> const & e):
|
|
|
|
assumption_cell(justification_kind::ExtAssumption, idx),
|
|
|
|
m_fn(fn),
|
|
|
|
m_expr(e) {}
|
|
|
|
};
|
|
|
|
|
|
|
|
asserted_cell * to_asserted(justification_cell * j) { lean_assert(j && j->is_asserted()); return static_cast<asserted_cell*>(j); }
|
|
|
|
assumption_cell * to_assumption(justification_cell * j) { lean_assert(j && j->is_assumption()); return static_cast<assumption_cell*>(j); }
|
|
|
|
ext_assumption_cell * to_ext_assumption(justification_cell * j) { lean_assert(j && j->is_ext_assumption()); return static_cast<ext_assumption_cell*>(j); }
|
|
|
|
composite_cell * to_composite(justification_cell * j) { lean_assert(j && j->is_composite()); return static_cast<composite_cell*>(j); }
|
|
|
|
ext_composite_cell * to_ext_composite(justification_cell * j) { lean_assert(j && j->is_composite()); return static_cast<ext_composite_cell*>(j); }
|
|
|
|
|
|
|
|
approx_set get_approx_assumption_set(justification const & j) {
|
|
|
|
justification_cell * it = j.raw();
|
|
|
|
if (!it)
|
|
|
|
return mk_empty_set();
|
|
|
|
switch (it->m_kind) {
|
|
|
|
case justification_kind::Asserted:
|
|
|
|
return mk_empty_set();
|
|
|
|
case justification_kind::Assumption: case justification_kind::ExtAssumption:
|
|
|
|
return mk_singleton(to_assumption(it)->m_idx);
|
|
|
|
case justification_kind::Composite: case justification_kind::ExtComposite:
|
|
|
|
return to_composite(it)->m_assumption_set;
|
|
|
|
}
|
|
|
|
lean_unreachable(); // LCOV_EXCL_LINE
|
2013-10-26 21:21:29 +00:00
|
|
|
}
|
|
|
|
|
2014-02-21 01:19:03 +00:00
|
|
|
void justification_cell::dealloc() {
|
|
|
|
switch (m_kind) {
|
|
|
|
case justification_kind::Asserted: delete to_asserted(this); break;
|
|
|
|
case justification_kind::Assumption: delete to_assumption(this); break;
|
|
|
|
case justification_kind::ExtAssumption: delete to_ext_assumption(this); break;
|
|
|
|
case justification_kind::Composite: delete to_composite(this); break;
|
|
|
|
case justification_kind::ExtComposite: delete to_ext_composite(this); break;
|
|
|
|
}
|
|
|
|
}
|
2014-01-04 01:13:10 +00:00
|
|
|
|
2014-02-21 01:19:03 +00:00
|
|
|
bool depends_on(justification const & j, unsigned i) {
|
|
|
|
if (!may_contain(get_approx_assumption_set(j), i))
|
|
|
|
return false;
|
2013-10-23 20:42:14 +00:00
|
|
|
buffer<justification_cell *> todo;
|
2014-02-21 01:19:03 +00:00
|
|
|
todo.push_back(j.raw());
|
2013-10-21 17:27:18 +00:00
|
|
|
while (!todo.empty()) {
|
2013-10-23 20:42:14 +00:00
|
|
|
justification_cell * curr = todo.back();
|
2013-10-14 18:50:51 +00:00
|
|
|
todo.pop_back();
|
2014-02-21 01:19:03 +00:00
|
|
|
switch (curr->m_kind) {
|
|
|
|
case justification_kind::Asserted:
|
|
|
|
break;
|
|
|
|
case justification_kind::Assumption: case justification_kind::ExtAssumption:
|
|
|
|
if (to_assumption(curr)->m_idx == i)
|
|
|
|
return true;
|
|
|
|
break;
|
|
|
|
case justification_kind::Composite: case justification_kind::ExtComposite:
|
2014-05-02 19:15:01 +00:00
|
|
|
for (unsigned k = 0; k < 2; k++) {
|
|
|
|
justification c = to_composite(curr)->m_child[k];
|
2014-02-21 01:19:03 +00:00
|
|
|
if (may_contain(get_approx_assumption_set(c), i))
|
|
|
|
todo.push_back(c.raw());
|
2013-10-14 18:50:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2014-02-21 01:19:03 +00:00
|
|
|
|
|
|
|
justification const & composite_child1(justification const & j) {
|
|
|
|
lean_assert(j.is_composite());
|
|
|
|
return to_composite(j.raw())->m_child[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
justification const & composite_child2(justification const & j) {
|
|
|
|
lean_assert(j.is_composite());
|
2014-02-23 00:12:06 +00:00
|
|
|
return to_composite(j.raw())->m_child[1];
|
2014-02-21 01:19:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
unsigned assumption_idx(justification const & j) {
|
|
|
|
lean_assert(j.is_assumption());
|
|
|
|
return to_assumption(j.raw())->m_idx;
|
|
|
|
}
|
|
|
|
|
|
|
|
justification::justification():m_ptr(nullptr) {}
|
|
|
|
justification::justification(justification_cell * ptr):m_ptr(ptr) { if (m_ptr) m_ptr->inc_ref(); }
|
|
|
|
justification::justification(justification const & s):m_ptr(s.m_ptr) { if (m_ptr) m_ptr->inc_ref(); }
|
|
|
|
justification::justification(justification && s):m_ptr(s.m_ptr) { s.m_ptr = nullptr; }
|
|
|
|
justification::~justification() { if (m_ptr) m_ptr->dec_ref(); }
|
|
|
|
bool justification::is_none() const { return m_ptr == nullptr; }
|
|
|
|
bool justification::is_asserted() const { return m_ptr && m_ptr->is_asserted(); }
|
|
|
|
bool justification::is_assumption() const { return m_ptr && m_ptr->is_assumption(); }
|
|
|
|
bool justification::is_composite() const { return m_ptr && m_ptr->is_composite(); }
|
|
|
|
justification & justification::operator=(justification const & s) { LEAN_COPY_REF(s); }
|
|
|
|
justification & justification::operator=(justification && s) { LEAN_MOVE_REF(s); }
|
|
|
|
optional<expr> justification::get_main_expr() const {
|
|
|
|
justification_cell * it = m_ptr;
|
|
|
|
while (true) {
|
|
|
|
if (!it)
|
|
|
|
return none_expr();
|
|
|
|
switch (it->m_kind) {
|
|
|
|
case justification_kind::Asserted:
|
|
|
|
return to_asserted(it)->m_expr;
|
|
|
|
case justification_kind::ExtAssumption:
|
|
|
|
return to_ext_assumption(it)->m_expr;
|
|
|
|
case justification_kind::ExtComposite:
|
|
|
|
return to_ext_composite(it)->m_expr;
|
|
|
|
case justification_kind::Assumption:
|
|
|
|
return none_expr();
|
|
|
|
case justification_kind::Composite:
|
|
|
|
it = to_composite(it)->m_child[0].raw();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
format justification::pp(formatter const & fmt, options const & opts, pos_info_provider const * p, substitution const & s) {
|
|
|
|
justification_cell * it = m_ptr;
|
|
|
|
while (true) {
|
|
|
|
if (!it)
|
|
|
|
return format();
|
|
|
|
switch (it->m_kind) {
|
|
|
|
case justification_kind::Asserted:
|
|
|
|
return to_asserted(it)->m_fn(fmt, opts, p, s);
|
|
|
|
case justification_kind::ExtAssumption:
|
|
|
|
return to_ext_assumption(it)->m_fn(fmt, opts, p, s);
|
|
|
|
case justification_kind::ExtComposite:
|
|
|
|
return to_ext_composite(it)->m_fn(fmt, opts, p, s);
|
|
|
|
case justification_kind::Assumption:
|
|
|
|
return format(format("Assumption "), format(to_assumption(it)->m_idx));
|
|
|
|
case justification_kind::Composite:
|
|
|
|
it = to_composite(it)->m_child[0].raw();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-02-25 00:30:56 +00:00
|
|
|
justification mk_composite(justification const & j1, justification const & j2, optional<expr> const & s, pp_jst_fn const & fn) {
|
2014-02-22 19:47:22 +00:00
|
|
|
if (j1.is_none())
|
|
|
|
return j2;
|
|
|
|
if (j2.is_none())
|
|
|
|
return j1;
|
2014-02-21 01:19:03 +00:00
|
|
|
return justification(new ext_composite_cell(j1, j2, fn, s));
|
|
|
|
}
|
|
|
|
justification mk_composite1(justification const & j1, justification const & j2) {
|
2014-02-22 19:47:22 +00:00
|
|
|
if (j1.is_none())
|
|
|
|
return j2;
|
|
|
|
if (j2.is_none())
|
|
|
|
return j1;
|
2014-02-21 01:19:03 +00:00
|
|
|
return justification(new composite_cell(j1, j2));
|
|
|
|
}
|
2014-02-25 00:30:56 +00:00
|
|
|
justification mk_assumption_justification(unsigned idx, optional<expr> const & s, pp_jst_fn const & fn) {
|
2014-02-21 01:19:03 +00:00
|
|
|
return justification(new ext_assumption_cell(idx, fn, s));
|
|
|
|
}
|
|
|
|
justification mk_assumption_justification(unsigned idx) {
|
|
|
|
return justification(new assumption_cell(idx));
|
|
|
|
}
|
2014-02-25 00:30:56 +00:00
|
|
|
justification mk_justification(optional<expr> const & s, pp_jst_fn const & fn) {
|
2014-02-21 01:19:03 +00:00
|
|
|
return justification(new asserted_cell(fn, s));
|
|
|
|
}
|
2014-02-25 00:30:56 +00:00
|
|
|
justification mk_justification(optional<expr> const & s, pp_jst_sfn const & fn) {
|
|
|
|
return mk_justification(s, [=](formatter const & fmt, options const & opts, pos_info_provider const * p, substitution const & subst) {
|
|
|
|
return compose(to_pos(s, p), fn(fmt, opts, subst)); });
|
2014-02-21 01:19:03 +00:00
|
|
|
}
|
2014-05-15 18:16:48 +00:00
|
|
|
justification mk_justification(char const * msg, optional<expr> const & s) {
|
|
|
|
std::string _msg(msg);
|
|
|
|
return mk_justification(s, [=](formatter const &, options const &, pos_info_provider const *, substitution const &) {
|
|
|
|
return format(_msg);
|
|
|
|
});
|
|
|
|
}
|
2014-02-23 00:12:06 +00:00
|
|
|
std::ostream & operator<<(std::ostream & out, justification const & j) {
|
|
|
|
if (j.is_none()) {
|
|
|
|
out << "none";
|
|
|
|
} else if (j.is_asserted()) {
|
|
|
|
out << "asserted";
|
|
|
|
} else if (j.is_assumption()) {
|
|
|
|
out << "(assumption " << assumption_idx(j) << ")";
|
|
|
|
} else if (j.is_composite()) {
|
|
|
|
out << "(join " << composite_child1(j) << " " << composite_child2(j) << ")";
|
|
|
|
} else {
|
|
|
|
out << "unexpected";
|
|
|
|
}
|
|
|
|
return out;
|
|
|
|
}
|
2013-10-01 00:58:46 +00:00
|
|
|
}
|