feat(library/type_context): use new tracing infrastructure in new type class resolution procedure

This commit is contained in:
Leonardo de Moura 2015-12-08 14:34:20 -08:00
parent 9b69ccd2f8
commit 58ab526d44
5 changed files with 46 additions and 64 deletions

View file

@ -225,6 +225,10 @@ void finalize_class_instance_resolution() {
(p : x = y) : (is_equiv (transport P p)) := ... (p : x = y) : (is_equiv (transport P p)) := ...
*/ */
static bool get_class_trace_instances(options const & o) {
return o.get_bool(name("trace", "class_instances"), false);
}
/** \brief Context for handling class-instance metavariable choice constraint */ /** \brief Context for handling class-instance metavariable choice constraint */
struct class_instance_context { struct class_instance_context {
io_state m_ios; io_state m_ios;

View file

@ -101,19 +101,21 @@ scope_trace_inc_depth::~scope_trace_inc_depth() {
g_depth--; g_depth--;
} }
scope_trace_init_bool_option::scope_trace_init_bool_option(name const & n, bool v) { void scope_trace_init_bool_option::init(name const & n, bool v) {
m_old_ios = g_ios; m_old_ios = g_ios;
if (g_env && g_ios) { if (g_env && g_ios) {
m_tmp_ios = *g_ios; m_tmp_ios.reset(new io_state(*g_ios));
options o = m_tmp_ios.get_options(); options o = m_tmp_ios->get_options();
o = o.update_if_undef(n, v); o = o.update_if_undef(n, v);
m_tmp_ios.set_options(o); m_tmp_ios->set_options(o);
g_ios = &m_tmp_ios; g_ios = m_tmp_ios.get();
} }
} }
scope_trace_init_bool_option::~scope_trace_init_bool_option() { scope_trace_init_bool_option::~scope_trace_init_bool_option() {
if (m_tmp_ios) {
g_ios = const_cast<io_state*>(m_old_ios); g_ios = const_cast<io_state*>(m_old_ios);
}
} }
io_state_stream tout() { io_state_stream tout() {

View file

@ -5,6 +5,7 @@ Released under Apache 2.0 license as described in the file LICENSE.
Author: Leonardo de Moura Author: Leonardo de Moura
*/ */
#pragma once #pragma once
#include <memory>
#include "library/io_state_stream.h" #include "library/io_state_stream.h"
namespace lean { namespace lean {
@ -13,6 +14,8 @@ void register_trace_class_alias(name const & n, name const & alias);
bool is_trace_enabled(); bool is_trace_enabled();
bool is_trace_class_enabled(name const & n); bool is_trace_class_enabled(name const & n);
#define lean_is_trace_enabled(CName) (::lean::is_trace_enabled() && ::lean::is_trace_class_enabled(CName))
class scope_trace_env { class scope_trace_env {
unsigned m_sz; unsigned m_sz;
environment const * m_old_env; environment const * m_old_env;
@ -29,19 +32,29 @@ public:
void activate(); void activate();
}; };
/* Temporarily set an option if it is not already set in the trace environment. */ #define LEAN_MERGE_(a,b) a##b
class scope_trace_init_bool_option { #define LEAN_LABEL_(a) LEAN_MERGE_(unique_name_, a)
io_state const * m_old_ios; #define LEAN_UNIQUE_NAME LEAN_LABEL_(__LINE__)
io_state m_tmp_ios;
public:
scope_trace_init_bool_option(name const & n, bool v);
~scope_trace_init_bool_option();
};
#define lean_trace_inc_depth(CName) \ #define lean_trace_inc_depth(CName) \
scope_trace_inc_depth trace_inc_depth_helper ## __LINE__; \ scope_trace_inc_depth LEAN_UNIQUE_NAME; \
if (is_trace_enabled() && is_trace_class_enabled(name(CName))) \ if (is_trace_enabled() && is_trace_class_enabled(name(CName))) \
trace_inc_depth_helper ## __LINE__.activate(); LEAN_UNIQUE_NAME.activate();
/* Temporarily set an option if it is not already set in the trace environment. */
class scope_trace_init_bool_option {
io_state const * m_old_ios{nullptr};
std::unique_ptr<io_state> m_tmp_ios;
public:
~scope_trace_init_bool_option();
void init(name const & opt, bool val);
};
#define lean_trace_init_bool(CName, Opt, Val) \
scope_trace_init_bool_option LEAN_UNIQUE_NAME; \
if (lean_is_trace_enabled(CName)) { \
LEAN_UNIQUE_NAME.init(Opt, Val); \
}
struct tdepth {}; struct tdepth {};
struct tclass { name m_cls; tclass(name const & c):m_cls(c) {} }; struct tclass { name m_cls; tclass(name const & c):m_cls(c) {} };
@ -50,8 +63,6 @@ io_state_stream tout();
io_state_stream const & operator<<(io_state_stream const & ios, tdepth const &); io_state_stream const & operator<<(io_state_stream const & ios, tdepth const &);
io_state_stream const & operator<<(io_state_stream const & ios, tclass const &); io_state_stream const & operator<<(io_state_stream const & ios, tclass const &);
#define lean_is_trace_enabled(CName) (::lean::is_trace_enabled() && ::lean::is_trace_class_enabled(CName))
#define lean_trace_plain(CName, CODE) { \ #define lean_trace_plain(CName, CODE) { \
if (lean_is_trace_enabled(CName)) { \ if (lean_is_trace_enabled(CName)) { \
CODE \ CODE \

View file

@ -13,6 +13,7 @@ Author: Leonardo de Moura
#include "kernel/for_each_fn.h" #include "kernel/for_each_fn.h"
#include "kernel/inductive/inductive.h" #include "kernel/inductive/inductive.h"
#include "library/trace.h" #include "library/trace.h"
#include "library/util.h"
#include "library/projection.h" #include "library/projection.h"
#include "library/normalize.h" #include "library/normalize.h"
#include "library/replace_visitor.h" #include "library/replace_visitor.h"
@ -23,10 +24,6 @@ Author: Leonardo de Moura
#include "library/class.h" #include "library/class.h"
#include "library/constants.h" #include "library/constants.h"
#ifndef LEAN_DEFAULT_CLASS_TRACE_INSTANCES
#define LEAN_DEFAULT_CLASS_TRACE_INSTANCES false
#endif
#ifndef LEAN_DEFAULT_CLASS_INSTANCE_MAX_DEPTH #ifndef LEAN_DEFAULT_CLASS_INSTANCE_MAX_DEPTH
#define LEAN_DEFAULT_CLASS_INSTANCE_MAX_DEPTH 32 #define LEAN_DEFAULT_CLASS_INSTANCE_MAX_DEPTH 32
#endif #endif
@ -37,14 +34,9 @@ Author: Leonardo de Moura
namespace lean { namespace lean {
static name * g_prefix = nullptr; static name * g_prefix = nullptr;
static name * g_class_trace_instances = nullptr;
static name * g_class_instance_max_depth = nullptr; static name * g_class_instance_max_depth = nullptr;
static name * g_class_trans_instances = nullptr; static name * g_class_trans_instances = nullptr;
bool get_class_trace_instances(options const & o) {
return o.get_bool(*g_class_trace_instances, LEAN_DEFAULT_CLASS_TRACE_INSTANCES);
}
unsigned get_class_instance_max_depth(options const & o) { unsigned get_class_instance_max_depth(options const & o) {
return o.get_unsigned(*g_class_instance_max_depth, LEAN_DEFAULT_CLASS_INSTANCE_MAX_DEPTH); return o.get_unsigned(*g_class_instance_max_depth, LEAN_DEFAULT_CLASS_INSTANCE_MAX_DEPTH);
} }
@ -110,7 +102,6 @@ struct type_context::ext_ctx : public extension_context {
type_context::type_context(environment const & env, io_state const & ios, tmp_local_generator * gen, type_context::type_context(environment const & env, io_state const & ios, tmp_local_generator * gen,
bool gen_owner, bool multiple_instances): bool gen_owner, bool multiple_instances):
m_env(env), m_env(env),
m_ios(ios),
m_ngen(*g_prefix), m_ngen(*g_prefix),
m_ext_ctx(new ext_ctx(*this)), m_ext_ctx(new ext_ctx(*this)),
m_local_gen(gen), m_local_gen(gen),
@ -124,7 +115,6 @@ type_context::type_context(environment const & env, io_state const & ios, tmp_lo
// TODO(Leo): use compilation options for setting config // TODO(Leo): use compilation options for setting config
m_ci_max_depth = 32; m_ci_max_depth = 32;
m_ci_trans_instances = true; m_ci_trans_instances = true;
m_ci_trace_instances = false;
update_options(ios.get_options()); update_options(ios.get_options());
} }
@ -1371,34 +1361,22 @@ bool type_context::update_options(options const & opts) {
options o = opts; options o = opts;
unsigned max_depth = get_class_instance_max_depth(o); unsigned max_depth = get_class_instance_max_depth(o);
bool trans_instances = get_class_trans_instances(o); bool trans_instances = get_class_trans_instances(o);
bool trace_instances = get_class_trace_instances(o);
if (trace_instances) {
o = o.update_if_undef(get_pp_purify_metavars_name(), false);
o = o.update_if_undef(get_pp_implicit_name(), true);
}
bool r = true; bool r = true;
if (m_ci_max_depth != max_depth || if (m_ci_max_depth != max_depth ||
m_ci_trans_instances != trans_instances || m_ci_trans_instances != trans_instances) {
m_ci_trace_instances != trace_instances) {
r = false; r = false;
} }
m_ci_max_depth = max_depth; m_ci_max_depth = max_depth;
m_ci_trans_instances = trans_instances; m_ci_trans_instances = trans_instances;
m_ci_trace_instances = trace_instances;
m_ios.set_options(o);
return r; return r;
} }
[[ noreturn ]] static void throw_class_exception(char const * msg, expr const & m) { throw_generic_exception(msg, m); } [[ noreturn ]] static void throw_class_exception(char const * msg, expr const & m) { throw_generic_exception(msg, m); }
io_state_stream type_context::diagnostic() {
return lean::diagnostic(m_env, m_ios);
}
void type_context::trace(unsigned depth, expr const & mvar, expr const & mvar_type, expr const & r) { void type_context::trace(unsigned depth, expr const & mvar, expr const & mvar_type, expr const & r) {
lean_assert(m_ci_trace_instances); auto out = tout();
auto out = diagnostic();
if (!m_ci_displayed_trace_header && m_ci_choices.size() == m_ci_choices_ini_sz + 1) { if (!m_ci_displayed_trace_header && m_ci_choices.size() == m_ci_choices_ini_sz + 1) {
out << tclass("class_instances");
if (m_pip) { if (m_pip) {
if (auto fname = m_pip->get_file_name()) { if (auto fname = m_pip->get_file_name()) {
out << fname << ":"; out << fname << ":";
@ -1409,10 +1387,7 @@ void type_context::trace(unsigned depth, expr const & mvar, expr const & mvar_ty
out << " class-instance resolution trace" << endl; out << " class-instance resolution trace" << endl;
m_ci_displayed_trace_header = true; m_ci_displayed_trace_header = true;
} }
for (unsigned i = 0; i < depth; i++) out << tclass("class_instances") << "(" << depth << ") ";
out << " ";
if (depth > 0)
out << "[" << depth << "] ";
out << mvar << " : " << instantiate_uvars_mvars(mvar_type) << " := " << r << endl; out << mvar << " : " << instantiate_uvars_mvars(mvar_type) << " := " << r << endl;
} }
@ -1446,9 +1421,7 @@ bool type_context::try_instance(ci_stack_entry const & e, expr const & inst, exp
r = mk_app(r, new_arg); r = mk_app(r, new_arg);
type = instantiate(binding_body(type), new_arg); type = instantiate(binding_body(type), new_arg);
} }
if (m_ci_trace_instances) { lean_trace_plain("class_instances", trace(e.m_depth, mk_app(mvar, locals), mvar_type, r););
trace(e.m_depth, mk_app(mvar, locals), mvar_type, r);
}
if (!is_def_eq(mvar_type, type)) { if (!is_def_eq(mvar_type, type)) {
return false; return false;
} }
@ -1504,7 +1477,7 @@ bool type_context::mk_choice_point(expr const & mvar) {
throw_class_exception("maximum class-instance resolution depth has been reached " throw_class_exception("maximum class-instance resolution depth has been reached "
"(the limit can be increased by setting option 'class.instance_max_depth') " "(the limit can be increased by setting option 'class.instance_max_depth') "
"(the class-instance resolution trace can be visualized " "(the class-instance resolution trace can be visualized "
"by setting option 'class.trace_instances')", "by setting option 'trace.class_instances')",
infer(m_ci_main_mvar)); infer(m_ci_main_mvar));
} }
// Remark: we initially tried to reject branches where mvar_type contained unassigned metavariables. // Remark: we initially tried to reject branches where mvar_type contained unassigned metavariables.
@ -1675,9 +1648,7 @@ optional<expr> type_context::ensure_no_meta(optional<expr> r) {
optional<expr> type_context::mk_class_instance_core(expr const & type) { optional<expr> type_context::mk_class_instance_core(expr const & type) {
if (auto r = check_ci_cache(type)) { if (auto r = check_ci_cache(type)) {
if (m_ci_trace_instances) { lean_trace("class_instances", tout() << "cached instance for " << type << "\n" << *r << "\n";);
diagnostic() << "cached instance for " << type << "\n" << *r << "\n";
}
return r; return r;
} }
init_search(type); init_search(type);
@ -1697,6 +1668,8 @@ void type_context::restore_choices(unsigned old_sz) {
optional<expr> type_context::mk_class_instance(expr const & type) { optional<expr> type_context::mk_class_instance(expr const & type) {
m_ci_choices.clear(); m_ci_choices.clear();
ci_choices_scope scope(*this); ci_choices_scope scope(*this);
lean_trace_init_bool("class_instances", get_pp_purify_metavars_name(), false);
lean_trace_init_bool("class_instances", get_pp_implicit_name(), true);
m_ci_displayed_trace_header = false; m_ci_displayed_trace_header = false;
auto r = mk_class_instance_core(type); auto r = mk_class_instance_core(type);
if (r) if (r)
@ -1962,15 +1935,11 @@ normalizer::normalizer(type_context & ctx, bool eta, bool nested_prop):
void initialize_type_context() { void initialize_type_context() {
g_prefix = new name(name::mk_internal_unique_name()); g_prefix = new name(name::mk_internal_unique_name());
g_class_trace_instances = new name{"class", "trace_instances"}; register_trace_class("class_instances");
g_class_instance_max_depth = new name{"class", "instance_max_depth"}; g_class_instance_max_depth = new name{"class", "instance_max_depth"};
g_class_trans_instances = new name{"class", "trans_instances"}; g_class_trans_instances = new name{"class", "trans_instances"};
register_bool_option(*g_class_trace_instances, LEAN_DEFAULT_CLASS_TRACE_INSTANCES,
"(class) display messages showing the class-instances resolution execution trace");
register_unsigned_option(*g_class_instance_max_depth, LEAN_DEFAULT_CLASS_INSTANCE_MAX_DEPTH, register_unsigned_option(*g_class_instance_max_depth, LEAN_DEFAULT_CLASS_INSTANCE_MAX_DEPTH,
"(class) max allowed depth in class-instance resolution"); "(class) max allowed depth in class-instance resolution");
register_bool_option(*g_class_trans_instances, LEAN_DEFAULT_CLASS_TRANS_INSTANCES, register_bool_option(*g_class_trans_instances, LEAN_DEFAULT_CLASS_TRANS_INSTANCES,
"(class) use automatically derived instances from the transitive closure of " "(class) use automatically derived instances from the transitive closure of "
"the structure instance graph"); "the structure instance graph");
@ -1978,7 +1947,6 @@ void initialize_type_context() {
void finalize_type_context() { void finalize_type_context() {
delete g_prefix; delete g_prefix;
delete g_class_trace_instances;
delete g_class_instance_max_depth; delete g_class_instance_max_depth;
delete g_class_trans_instances; delete g_class_trans_instances;
} }

View file

@ -13,7 +13,6 @@ Author: Leonardo de Moura
#include "library/projection.h" #include "library/projection.h"
namespace lean { namespace lean {
bool get_class_trace_instances(options const & o);
unsigned get_class_instance_max_depth(options const & o); unsigned get_class_instance_max_depth(options const & o);
bool get_class_trans_instances(options const & o); bool get_class_trans_instances(options const & o);
@ -119,7 +118,6 @@ class type_context {
struct ext_ctx; struct ext_ctx;
friend struct ext_ctx; friend struct ext_ctx;
environment m_env; environment m_env;
io_state m_ios;
name_generator m_ngen; name_generator m_ngen;
std::unique_ptr<ext_ctx> m_ext_ctx; std::unique_ptr<ext_ctx> m_ext_ctx;
tmp_local_generator * m_local_gen; tmp_local_generator * m_local_gen;
@ -271,7 +269,6 @@ class type_context {
bool m_ci_trans_instances; bool m_ci_trans_instances;
bool m_ci_trace_instances; bool m_ci_trace_instances;
io_state_stream diagnostic();
optional<name> constant_is_class(expr const & e); optional<name> constant_is_class(expr const & e);
optional<name> is_full_class(expr type); optional<name> is_full_class(expr type);
lbool is_quick_class(expr const & type, name & result); lbool is_quick_class(expr const & type, name & result);