lean2/src/kernel/justification.cpp

383 lines
14 KiB
C++
Raw Normal View History

/*
Copyright (c) 2013 Microsoft Corporation. All rights reserved.
Released under Apache 2.0 license as described in the file LICENSE.
Author: Leonardo de Moura
*/
#include <string>
#include <vector>
#include <unordered_set>
#include "util/buffer.h"
#include "util/int64.h"
#include "util/memory_pool.h"
#include "kernel/justification.h"
#include "kernel/metavar.h"
#ifndef LEAN_DEFAULT_DEPENDS_ON_CACHE_CAPACITY
#define LEAN_DEFAULT_DEPENDS_ON_CACHE_CAPACITY 1024*8
#endif
namespace lean {
format to_pos(optional<expr> const & e, pos_info_provider const * p) {
if (!p || !e)
return format();
format f = p->pp(*e);
if (!f)
return format();
return f + space();
}
format pp_previous_error_header(formatter const &, pos_info_provider const * pos_prov, optional<expr> const & ref, bool is_main) {
if (!is_main) {
format r = line();
r += to_pos(ref, pos_prov);
r += format("previous error additional information");
r += line();
return r;
} else {
return format();
}
}
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; }
enum class justification_kind { Asserted, Composite, Wrapper, Assumption };
approx_set get_approx_assumption_set(justification const & j);
LEAN_THREAD_VALUE(unsigned, g_hash_alloc_jst_counter, 0);
struct justification_cell {
MK_LEAN_RC();
justification_kind m_kind;
unsigned m_hash_alloc;
void dealloc();
justification_cell(justification_kind k):m_rc(0), m_kind(k) {
m_hash_alloc = g_hash_alloc_jst_counter;
g_hash_alloc_jst_counter++;
}
bool is_asserted() const { return m_kind == justification_kind::Asserted; }
bool is_assumption() const { return m_kind == justification_kind::Assumption; }
bool is_composite() const { return m_kind == justification_kind::Composite; }
bool is_wrapper() const { return m_kind == justification_kind::Wrapper; }
};
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 wrapper_cell : public justification_cell {
pp_jst_fn m_fn;
optional<expr> m_expr;
justification m_jst;
wrapper_cell(justification const & j, pp_jst_fn const & fn, optional<expr> const & e):
justification_cell(justification_kind::Wrapper),
m_fn(fn), m_expr(e), m_jst(j) {}
};
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) {}
};
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); }
composite_cell * to_composite(justification_cell * j) { lean_assert(j && j->is_composite()); return static_cast<composite_cell*>(j); }
wrapper_cell * to_wrapper(justification_cell * j) { lean_assert(j && j->is_wrapper()); return static_cast<wrapper_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:
return mk_singleton(to_assumption(it)->m_idx);
case justification_kind::Composite:
return to_composite(it)->m_assumption_set;
case justification_kind::Wrapper:
return get_approx_assumption_set(to_wrapper(it)->m_jst);
}
lean_unreachable(); // LCOV_EXCL_LINE
}
DEF_THREAD_MEMORY_POOL(get_asserted_allocator, sizeof(asserted_cell));
DEF_THREAD_MEMORY_POOL(get_composite_allocator, sizeof(composite_cell));
DEF_THREAD_MEMORY_POOL(get_assumption_allocator, sizeof(assumption_cell));
DEF_THREAD_MEMORY_POOL(get_wrapper_allocator, sizeof(wrapper_cell));
void justification_cell::dealloc() {
switch (m_kind) {
case justification_kind::Asserted:
to_asserted(this)->~asserted_cell();
get_asserted_allocator().recycle(this);
break;
case justification_kind::Assumption:
to_assumption(this)->~assumption_cell();
get_assumption_allocator().recycle(this);
break;
case justification_kind::Composite:
to_composite(this)->~composite_cell();
get_composite_allocator().recycle(this);
break;
case justification_kind::Wrapper:
to_wrapper(this)->~wrapper_cell();
get_wrapper_allocator().recycle(this);
break;
}
}
struct depends_on_cache {
unsigned m_capacity;
std::vector<justification_cell const *> m_cache;
std::vector<unsigned> m_used;
depends_on_cache(unsigned c):m_capacity(c), m_cache(c, nullptr) {}
bool visited(justification_cell const * j) {
unsigned i = j->m_hash_alloc % m_capacity;
if (m_cache[i]) {
if (m_cache[i] == j) {
return true;
} else {
m_cache[i] = j;
return false;
}
} else {
m_cache[i] = j;
m_used.push_back(i);
return false;
}
}
void clear() {
for (unsigned i : m_used) {
lean_assert(m_cache[i]);
m_cache[i] = nullptr;
}
m_used.clear();
}
};
MK_THREAD_LOCAL_GET(depends_on_cache, get_depends_on_cache, LEAN_DEFAULT_DEPENDS_ON_CACHE_CAPACITY);
struct depends_on_cache_ref {
depends_on_cache & m_cache;
depends_on_cache_ref():m_cache(get_depends_on_cache()) {}
~depends_on_cache_ref() { m_cache.clear(); }
depends_on_cache * operator->() { return &m_cache; }
};
bool depends_on(justification const & j, unsigned i) {
if (!may_contain(get_approx_assumption_set(j), i))
return false;
depends_on_cache_ref cache;
buffer<justification_cell *> todo;
todo.push_back(j.raw());
while (!todo.empty()) {
justification_cell * curr = todo.back();
todo.pop_back();
switch (curr->m_kind) {
case justification_kind::Asserted:
break;
case justification_kind::Assumption:
if (to_assumption(curr)->m_idx == i)
return true;
break;
case justification_kind::Composite:
for (unsigned k = 0; k < 2; k++) {
justification c = to_composite(curr)->m_child[k];
if (!cache->visited(c.raw()) && may_contain(get_approx_assumption_set(c), i))
todo.push_back(c.raw());
}
break;
case justification_kind::Wrapper: {
justification c = to_wrapper(curr)->m_jst;
if (!cache->visited(c.raw()) && may_contain(get_approx_assumption_set(c), i))
todo.push_back(c.raw());
break;
}}
}
return false;
}
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());
return to_composite(j.raw())->m_child[1];
}
justification const & wrapper_child(justification const & j) {
lean_assert(j.is_wrapper());
return to_wrapper(j.raw())->m_jst;
}
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(); }
bool justification::is_wrapper() const { return m_ptr && m_ptr->is_wrapper(); }
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::Assumption:
return none_expr();
case justification_kind::Composite:
it = to_composite(it)->m_child[0].raw();
break;
case justification_kind::Wrapper:
return to_wrapper(it)->m_expr;
}
}
}
struct jst_hash_fn {
unsigned operator()(justification const & j) const { return j.raw()->m_hash_alloc; }
};
struct jst_ptr_eq {
bool operator()(justification const & j1, justification const & j2) const { return j1.raw() == j2.raw(); }
};
class justification_set {
std::unordered_set<justification, jst_hash_fn, jst_ptr_eq> m_set;
public:
bool contains(justification const & j) const { return m_set.find(j) != m_set.end(); }
void insert(justification const & j) { m_set.insert(j); }
};
format justification::pp_core(formatter const & fmt, pos_info_provider const * p, substitution const & s,
justification_set & set, bool is_main) const {
if (set.contains(*this))
return format();
set.insert(*this);
justification_cell * it = m_ptr;
if (!it)
return format();
switch (it->m_kind) {
case justification_kind::Asserted:
return to_asserted(it)->m_fn(fmt, p, s, is_main);
case justification_kind::Wrapper:
return to_wrapper(it)->m_fn(fmt, p, s, is_main);
case justification_kind::Assumption:
if (is_main)
return format(format("Assumption "), format(to_assumption(it)->m_idx));
else
return format();
case justification_kind::Composite:
return
to_composite(it)->m_child[0].pp_core(fmt, p, s, set, is_main) +
to_composite(it)->m_child[1].pp_core(fmt, p, s, set, false);
}
}
format justification::pp(formatter const & fmt, pos_info_provider const * p, substitution const & s) const {
justification_set set;
return pp_core(fmt, p, s, set, true);
}
justification mk_wrapper(justification const & j, optional<expr> const & s, pp_jst_fn const & fn) {
return justification(new (get_wrapper_allocator().allocate()) wrapper_cell(j, fn, s));
}
justification mk_composite(justification const & j1, justification const & j2, optional<expr> const & s, pp_jst_fn const & fn) {
return mk_wrapper(mk_composite1(j1, j2), s, fn);
}
justification mk_composite1(justification const & j1, justification const & j2) {
if (j1.is_none() || j1.raw() == j2.raw())
return j2;
if (j2.is_none())
return j1;
return justification(new (get_composite_allocator().allocate()) composite_cell(j1, j2));
}
justification mk_assumption_justification(unsigned idx, optional<expr> const & s, pp_jst_fn const & fn) {
return mk_wrapper(mk_assumption_justification(idx), s, fn);
}
justification mk_assumption_justification(unsigned idx) {
return justification(new (get_assumption_allocator().allocate()) assumption_cell(idx));
}
justification mk_justification(optional<expr> const & s, pp_jst_fn const & fn) {
return justification(new (get_asserted_allocator().allocate()) asserted_cell(fn, s));
}
justification mk_justification(optional<expr> const & s, pp_jst_sfn const & fn) {
return mk_justification(s, [=](formatter const & fmt, pos_info_provider const *, substitution const & subst, bool is_main) {
// Remark: we are not using to_pos(s, p) anymore because we don't try to display complicated error messages anymore.
// return compose(to_pos(s, p), fn(fmt, subst));
if (is_main)
return fn(fmt, subst);
else
return format();
});
}
justification mk_justification(char const * msg, optional<expr> const & s) {
std::string _msg(msg);
return mk_justification(s, [=](formatter const &, pos_info_provider const *, substitution const &, bool is_main) {
if (is_main)
return format(_msg);
else
return format();
});
}
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 if (j.is_wrapper()) {
out << "(wrapper " << wrapper_child(j) << ")";
} else {
out << "unexpected";
}
return out;
}
}