2014-09-10 22:20:45 +00:00
|
|
|
/*
|
|
|
|
Copyright (c) 2014 Microsoft Corporation. All rights reserved.
|
|
|
|
Released under Apache 2.0 license as described in the file LICENSE.
|
|
|
|
|
|
|
|
Author: Leonardo de Moura
|
|
|
|
*/
|
2014-09-12 22:56:02 +00:00
|
|
|
#include "util/lazy_list_fn.h"
|
2014-09-25 16:59:27 +00:00
|
|
|
#include "util/flet.h"
|
2014-10-30 21:20:34 +00:00
|
|
|
#include "util/sexpr/option_declarations.h"
|
2014-09-10 22:20:45 +00:00
|
|
|
#include "kernel/instantiate.h"
|
2014-09-26 02:46:08 +00:00
|
|
|
#include "kernel/for_each_fn.h"
|
2014-09-10 22:20:45 +00:00
|
|
|
#include "kernel/abstract.h"
|
2014-10-30 21:44:58 +00:00
|
|
|
#include "kernel/error_msgs.h"
|
2014-09-10 22:20:45 +00:00
|
|
|
#include "library/unifier.h"
|
2014-09-19 20:30:08 +00:00
|
|
|
#include "library/reducible.h"
|
2014-09-11 16:58:44 +00:00
|
|
|
#include "library/metavar_closure.h"
|
2014-09-11 21:45:16 +00:00
|
|
|
#include "library/error_handling/error_handling.h"
|
2014-09-10 22:20:45 +00:00
|
|
|
#include "frontends/lean/util.h"
|
|
|
|
#include "frontends/lean/class.h"
|
|
|
|
#include "frontends/lean/local_context.h"
|
|
|
|
#include "frontends/lean/choice_iterator.h"
|
2014-10-30 21:44:58 +00:00
|
|
|
#include "frontends/lean/elaborator_exception.h"
|
2014-11-09 19:24:19 +00:00
|
|
|
#include "frontends/lean/pp_options.h"
|
2014-09-10 22:20:45 +00:00
|
|
|
|
2014-10-30 21:20:34 +00:00
|
|
|
#ifndef LEAN_DEFAULT_ELABORATOR_UNIQUE_CLASS_INSTANCES
|
|
|
|
#define LEAN_DEFAULT_ELABORATOR_UNIQUE_CLASS_INSTANCES false
|
|
|
|
#endif
|
|
|
|
|
2014-11-09 19:24:19 +00:00
|
|
|
#ifndef LEAN_DEFAULT_ELABORATOR_TRACE_INSTANCES
|
|
|
|
#define LEAN_DEFAULT_ELABORATOR_TRACE_INSTANCES false
|
|
|
|
#endif
|
|
|
|
|
2014-11-09 19:59:44 +00:00
|
|
|
#ifndef LEAN_DEFAULT_ELABORATOR_INSTANCE_MAX_DEPTH
|
|
|
|
#define LEAN_DEFAULT_ELABORATOR_INSTANCE_MAX_DEPTH 32
|
|
|
|
#endif
|
|
|
|
|
2014-09-10 22:20:45 +00:00
|
|
|
namespace lean {
|
2014-10-30 21:20:34 +00:00
|
|
|
static name * g_elaborator_unique_class_instances = nullptr;
|
2014-11-09 19:24:19 +00:00
|
|
|
static name * g_elaborator_trace_instances = nullptr;
|
2014-11-09 19:59:44 +00:00
|
|
|
static name * g_elaborator_instance_max_depth = nullptr;
|
2014-10-30 21:20:34 +00:00
|
|
|
|
|
|
|
void initialize_placeholder_elaborator() {
|
|
|
|
g_elaborator_unique_class_instances = new name{"elaborator", "unique_class_instances"};
|
2014-11-09 19:24:19 +00:00
|
|
|
g_elaborator_trace_instances = new name{"elaborator", "trace_instances"};
|
2014-11-09 19:59:44 +00:00
|
|
|
g_elaborator_instance_max_depth = new name{"elaborator", "instance_max_depth"};
|
2014-11-09 19:24:19 +00:00
|
|
|
|
2014-10-30 21:20:34 +00:00
|
|
|
register_bool_option(*g_elaborator_unique_class_instances, LEAN_DEFAULT_ELABORATOR_UNIQUE_CLASS_INSTANCES,
|
|
|
|
"(elaborator) generate an error if there is more than one solution "
|
|
|
|
"for a class-instance resolution problem");
|
2014-11-09 19:24:19 +00:00
|
|
|
|
|
|
|
register_bool_option(*g_elaborator_trace_instances, LEAN_DEFAULT_ELABORATOR_TRACE_INSTANCES,
|
|
|
|
"(elaborator) display messages showing the class-instances resolution execution trace");
|
2014-11-09 19:59:44 +00:00
|
|
|
|
|
|
|
register_unsigned_option(*g_elaborator_instance_max_depth, LEAN_DEFAULT_ELABORATOR_INSTANCE_MAX_DEPTH,
|
|
|
|
"(elaborator) max allowed depth in class-instance resolution");
|
2014-10-30 21:20:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void finalize_placeholder_elaborator() {
|
|
|
|
delete g_elaborator_unique_class_instances;
|
2014-11-09 19:24:19 +00:00
|
|
|
delete g_elaborator_trace_instances;
|
2014-11-09 19:59:44 +00:00
|
|
|
delete g_elaborator_instance_max_depth;
|
2014-10-30 21:20:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool get_elaborator_unique_class_instances(options const & o) {
|
|
|
|
return o.get_bool(*g_elaborator_unique_class_instances, LEAN_DEFAULT_ELABORATOR_UNIQUE_CLASS_INSTANCES);
|
|
|
|
}
|
|
|
|
|
2014-11-09 19:24:19 +00:00
|
|
|
bool get_elaborator_trace_instances(options const & o) {
|
|
|
|
return o.get_bool(*g_elaborator_trace_instances, LEAN_DEFAULT_ELABORATOR_TRACE_INSTANCES);
|
|
|
|
}
|
|
|
|
|
2014-11-09 19:59:44 +00:00
|
|
|
unsigned get_elaborator_instance_max_depth(options const & o) {
|
|
|
|
return o.get_unsigned(*g_elaborator_instance_max_depth, LEAN_DEFAULT_ELABORATOR_INSTANCE_MAX_DEPTH);
|
|
|
|
}
|
|
|
|
|
2014-09-10 22:20:45 +00:00
|
|
|
/** \brief Context for handling placeholder metavariable choice constraint */
|
|
|
|
struct placeholder_context {
|
2014-11-09 19:24:19 +00:00
|
|
|
io_state m_ios;
|
|
|
|
name_generator m_ngen;
|
|
|
|
type_checker_ptr m_tc;
|
2014-11-09 19:59:44 +00:00
|
|
|
expr m_main_meta;
|
2014-11-09 19:24:19 +00:00
|
|
|
bool m_relax;
|
|
|
|
bool m_use_local_instances;
|
|
|
|
bool m_trace_instances;
|
2014-11-09 19:59:44 +00:00
|
|
|
unsigned m_max_depth;
|
2014-11-09 19:47:01 +00:00
|
|
|
char const * m_fname;
|
|
|
|
optional<pos_info> m_pos;
|
2014-11-05 02:41:27 +00:00
|
|
|
placeholder_context(environment const & env, io_state const & ios,
|
2014-11-09 19:47:01 +00:00
|
|
|
name const & prefix, bool relax, bool use_local_instances):
|
2014-09-10 22:20:45 +00:00
|
|
|
m_ios(ios),
|
|
|
|
m_ngen(prefix),
|
2014-09-19 20:30:08 +00:00
|
|
|
m_tc(mk_type_checker(env, m_ngen.mk_child(), relax)),
|
2014-09-10 22:20:45 +00:00
|
|
|
m_relax(relax),
|
2014-11-09 19:47:01 +00:00
|
|
|
m_use_local_instances(use_local_instances) {
|
|
|
|
m_fname = nullptr;
|
2014-11-09 19:24:19 +00:00
|
|
|
m_trace_instances = get_elaborator_trace_instances(ios.get_options());
|
2014-11-09 19:59:44 +00:00
|
|
|
m_max_depth = get_elaborator_instance_max_depth(ios.get_options());
|
2014-11-09 19:24:19 +00:00
|
|
|
options opts = m_ios.get_options();
|
|
|
|
opts = opts.update(get_pp_purify_metavars_name(), false);
|
|
|
|
opts = opts.update(get_pp_implicit_name(), true);
|
|
|
|
m_ios.set_options(opts);
|
2014-09-10 22:20:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
environment const & env() const { return m_tc->env(); }
|
|
|
|
io_state const & ios() const { return m_ios; }
|
|
|
|
bool use_local_instances() const { return m_use_local_instances; }
|
|
|
|
type_checker & tc() const { return *m_tc; }
|
2014-11-09 19:24:19 +00:00
|
|
|
bool trace_instances() const { return m_trace_instances; }
|
2014-11-09 19:59:44 +00:00
|
|
|
void set_main_meta(expr const & meta) { m_main_meta = meta; }
|
|
|
|
expr const & get_main_meta() const { return m_main_meta; }
|
2014-11-09 19:47:01 +00:00
|
|
|
void set_pos(char const * fname, optional<pos_info> const & pos) {
|
|
|
|
m_fname = fname;
|
|
|
|
m_pos = pos;
|
|
|
|
}
|
|
|
|
optional<pos_info> const & get_pos() const { return m_pos; }
|
|
|
|
char const * get_file_name() const { return m_fname; }
|
2014-11-09 19:59:44 +00:00
|
|
|
unsigned get_max_depth() const { return m_max_depth; }
|
2014-09-10 22:20:45 +00:00
|
|
|
};
|
|
|
|
|
2014-11-05 02:41:27 +00:00
|
|
|
pair<expr, constraint> mk_placeholder_elaborator(std::shared_ptr<placeholder_context> const & C, local_context const & ctx,
|
2014-11-09 19:24:19 +00:00
|
|
|
optional<expr> const & type, tag g, unsigned depth);
|
2014-09-10 22:20:45 +00:00
|
|
|
|
|
|
|
/** \brief Whenever the elaborator finds a placeholder '_' or introduces an
|
|
|
|
implicit argument, it creates a metavariable \c ?m. It also creates a
|
|
|
|
delayed choice constraint (?m in fn).
|
|
|
|
|
|
|
|
The function \c fn produces a stream of alternative solutions for ?m.
|
|
|
|
In this case, \c fn will do the following:
|
|
|
|
1) if the elaborated type of ?m is a 'class' C, then the stream will start with
|
|
|
|
a) all local instances of class C (if elaborator.local_instances == true)
|
2014-10-07 16:44:01 +00:00
|
|
|
b) all global instances of class C
|
2014-09-10 22:20:45 +00:00
|
|
|
*/
|
|
|
|
struct placeholder_elaborator : public choice_iterator {
|
|
|
|
std::shared_ptr<placeholder_context> m_C;
|
2014-11-05 02:41:27 +00:00
|
|
|
local_context m_ctx;
|
2014-09-10 22:20:45 +00:00
|
|
|
expr m_meta;
|
|
|
|
// elaborated type of the metavariable
|
|
|
|
expr m_meta_type;
|
|
|
|
// local instances that should also be included in the
|
|
|
|
// class-instance resolution.
|
|
|
|
// This information is retrieved from the local context
|
|
|
|
list<expr> m_local_instances;
|
|
|
|
// global declaration names that are class instances.
|
|
|
|
// This information is retrieved using #get_class_instances.
|
|
|
|
list<name> m_instances;
|
|
|
|
justification m_jst;
|
2014-11-09 19:24:19 +00:00
|
|
|
unsigned m_depth;
|
2014-09-10 22:20:45 +00:00
|
|
|
|
2014-11-05 02:41:27 +00:00
|
|
|
placeholder_elaborator(std::shared_ptr<placeholder_context> const & C, local_context const & ctx,
|
2014-09-10 22:20:45 +00:00
|
|
|
expr const & meta, expr const & meta_type,
|
|
|
|
list<expr> const & local_insts, list<name> const & instances,
|
2014-11-09 19:24:19 +00:00
|
|
|
justification const & j, unsigned depth):
|
2014-11-05 02:41:27 +00:00
|
|
|
choice_iterator(), m_C(C), m_ctx(ctx), m_meta(meta), m_meta_type(meta_type),
|
2014-11-09 19:24:19 +00:00
|
|
|
m_local_instances(local_insts), m_instances(instances), m_jst(j), m_depth(depth) {
|
2014-11-09 19:59:44 +00:00
|
|
|
if (m_depth > m_C->get_max_depth()) {
|
|
|
|
throw_elaborator_exception(C->env(), "maximum class-instance resolution depth has been reached "
|
|
|
|
"(the limit can be increased by setting option 'elaborator.instance_max_depth') "
|
|
|
|
"(the class-instance resolution trace can be visualized by setting option 'elaborator.trace_instances')",
|
|
|
|
C->get_main_meta());
|
|
|
|
}
|
2014-09-10 22:20:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
constraints mk_constraints(constraint const & c, buffer<constraint> const & cs) {
|
|
|
|
return cons(c, to_list(cs.begin(), cs.end()));
|
|
|
|
}
|
|
|
|
|
2014-11-09 20:06:16 +00:00
|
|
|
void trace(expr const & t, expr const & r) {
|
2014-11-09 19:24:19 +00:00
|
|
|
if (!m_C->trace_instances())
|
|
|
|
return;
|
|
|
|
auto out = diagnostic(m_C->env(), m_C->ios());
|
2014-11-09 19:47:01 +00:00
|
|
|
if (m_depth == 0) {
|
|
|
|
if (auto fname = m_C->get_file_name()) {
|
|
|
|
out << fname << ":";
|
2014-11-09 19:24:19 +00:00
|
|
|
}
|
2014-11-09 19:47:01 +00:00
|
|
|
if (auto pos = m_C->get_pos()) {
|
|
|
|
out << pos->first << ":" << pos->second << ":";
|
|
|
|
}
|
|
|
|
out << " class-instance resolution trace" << endl;
|
2014-11-09 19:24:19 +00:00
|
|
|
}
|
2014-11-09 19:47:01 +00:00
|
|
|
for (unsigned i = 0; i < m_depth; i++)
|
|
|
|
out << " ";
|
|
|
|
if (m_depth > 0)
|
|
|
|
out << "[" << m_depth << "] ";
|
2014-11-09 20:06:16 +00:00
|
|
|
out << m_meta << " : " << t << " := " << r << endl;
|
2014-11-09 19:24:19 +00:00
|
|
|
}
|
|
|
|
|
2014-09-10 22:20:45 +00:00
|
|
|
optional<constraints> try_instance(expr const & inst, expr const & inst_type) {
|
|
|
|
type_checker & tc = m_C->tc();
|
|
|
|
name_generator & ngen = m_C->m_ngen;
|
|
|
|
tag g = inst.get_tag();
|
|
|
|
try {
|
2014-11-05 02:41:27 +00:00
|
|
|
flet<local_context> scope(m_ctx, m_ctx);
|
2014-09-10 22:20:45 +00:00
|
|
|
buffer<expr> locals;
|
|
|
|
expr meta_type = m_meta_type;
|
|
|
|
while (true) {
|
|
|
|
meta_type = tc.whnf(meta_type).first;
|
|
|
|
if (!is_pi(meta_type))
|
|
|
|
break;
|
|
|
|
expr local = mk_local(ngen.next(), binding_name(meta_type),
|
|
|
|
binding_domain(meta_type), binding_info(meta_type));
|
2014-11-05 02:41:27 +00:00
|
|
|
m_ctx.add_local(local);
|
2014-09-10 22:20:45 +00:00
|
|
|
locals.push_back(local);
|
|
|
|
meta_type = instantiate(binding_body(meta_type), local);
|
|
|
|
}
|
|
|
|
expr type = inst_type;
|
|
|
|
expr r = inst;
|
|
|
|
buffer<constraint> cs;
|
|
|
|
while (true) {
|
|
|
|
type = tc.whnf(type).first;
|
|
|
|
if (!is_pi(type))
|
|
|
|
break;
|
2014-11-09 19:24:19 +00:00
|
|
|
pair<expr, constraint> ac = mk_placeholder_elaborator(m_C, m_ctx, some_expr(binding_domain(type)), g, m_depth+1);
|
2014-09-10 22:20:45 +00:00
|
|
|
expr arg = ac.first;
|
|
|
|
cs.push_back(ac.second);
|
2014-10-15 20:17:09 +00:00
|
|
|
r = mk_app(r, arg, g);
|
2014-09-10 22:20:45 +00:00
|
|
|
type = instantiate(binding_body(type), arg);
|
|
|
|
}
|
|
|
|
r = Fun(locals, r);
|
2014-11-09 20:06:16 +00:00
|
|
|
trace(meta_type, r);
|
2014-09-10 22:20:45 +00:00
|
|
|
bool relax = m_C->m_relax;
|
|
|
|
constraint c = mk_eq_cnstr(m_meta, r, m_jst, relax);
|
|
|
|
return optional<constraints>(mk_constraints(c, cs));
|
|
|
|
} catch (exception &) {
|
|
|
|
return optional<constraints>();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
optional<constraints> try_instance(name const & inst) {
|
|
|
|
environment const & env = m_C->env();
|
|
|
|
if (auto decl = env.find(inst)) {
|
|
|
|
name_generator & ngen = m_C->m_ngen;
|
|
|
|
buffer<level> ls_buffer;
|
|
|
|
unsigned num_univ_ps = length(decl->get_univ_params());
|
|
|
|
for (unsigned i = 0; i < num_univ_ps; i++)
|
|
|
|
ls_buffer.push_back(mk_meta_univ(ngen.next()));
|
|
|
|
levels ls = to_list(ls_buffer.begin(), ls_buffer.end());
|
|
|
|
expr inst_cnst = copy_tag(m_meta, mk_constant(inst, ls));
|
|
|
|
expr inst_type = instantiate_type_univ_params(*decl, ls);
|
|
|
|
return try_instance(inst_cnst, inst_type);
|
|
|
|
} else {
|
|
|
|
return optional<constraints>();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual optional<constraints> next() {
|
|
|
|
while (!empty(m_local_instances)) {
|
|
|
|
expr inst = head(m_local_instances);
|
|
|
|
m_local_instances = tail(m_local_instances);
|
|
|
|
if (!is_local(inst))
|
|
|
|
continue;
|
|
|
|
if (auto r = try_instance(inst, mlocal_type(inst)))
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
while (!empty(m_instances)) {
|
|
|
|
name inst = head(m_instances);
|
|
|
|
m_instances = tail(m_instances);
|
|
|
|
if (auto cs = try_instance(inst))
|
|
|
|
return cs;
|
|
|
|
}
|
|
|
|
return optional<constraints>();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2014-11-09 19:24:19 +00:00
|
|
|
constraint mk_placeholder_cnstr(std::shared_ptr<placeholder_context> const & C, local_context const & ctx, expr const & m, unsigned depth) {
|
2014-09-10 22:20:45 +00:00
|
|
|
environment const & env = C->env();
|
|
|
|
justification j = mk_failed_to_synthesize_jst(env, m);
|
2014-10-07 16:44:01 +00:00
|
|
|
auto choice_fn = [=](expr const & meta, expr const & meta_type, substitution const &, name_generator const &) {
|
2014-09-10 22:20:45 +00:00
|
|
|
if (auto cls_name_it = is_ext_class(C->tc(), meta_type)) {
|
|
|
|
name cls_name = *cls_name_it;
|
2014-11-05 02:41:27 +00:00
|
|
|
list<expr> const & ctx_lst = ctx.get_data();
|
2014-09-10 22:20:45 +00:00
|
|
|
list<expr> local_insts;
|
|
|
|
if (C->use_local_instances())
|
2014-11-05 02:41:27 +00:00
|
|
|
local_insts = get_local_instances(C->tc(), ctx_lst, cls_name);
|
2014-09-10 22:20:45 +00:00
|
|
|
list<name> insts = get_class_instances(env, cls_name);
|
2014-10-07 16:44:01 +00:00
|
|
|
if (empty(local_insts) && empty(insts))
|
2014-09-10 22:20:45 +00:00
|
|
|
return lazy_list<constraints>(); // nothing to be done
|
|
|
|
// we are always strict with placeholders associated with classes
|
2014-11-09 19:24:19 +00:00
|
|
|
return choose(std::make_shared<placeholder_elaborator>(C, ctx, meta, meta_type, local_insts, insts, j, depth));
|
2014-09-10 22:20:45 +00:00
|
|
|
} else {
|
2014-09-26 02:46:08 +00:00
|
|
|
// do nothing, type is not a class...
|
|
|
|
return lazy_list<constraints>(constraints());
|
2014-09-10 22:20:45 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
bool owner = false;
|
|
|
|
bool relax = C->m_relax;
|
2014-09-11 21:45:16 +00:00
|
|
|
return mk_choice_cnstr(m, choice_fn, to_delay_factor(cnstr_group::ClassInstance),
|
|
|
|
owner, j, relax);
|
|
|
|
}
|
|
|
|
|
2014-11-05 02:41:27 +00:00
|
|
|
pair<expr, constraint> mk_placeholder_elaborator(std::shared_ptr<placeholder_context> const & C, local_context const & ctx,
|
2014-11-09 19:24:19 +00:00
|
|
|
optional<expr> const & type, tag g, unsigned depth) {
|
2014-11-05 02:41:27 +00:00
|
|
|
expr m = ctx.mk_meta(C->m_ngen, type, g);
|
2014-11-09 19:24:19 +00:00
|
|
|
constraint c = mk_placeholder_cnstr(C, ctx, m, depth);
|
2014-09-10 22:20:45 +00:00
|
|
|
return mk_pair(m, c);
|
|
|
|
}
|
|
|
|
|
2014-09-26 02:46:08 +00:00
|
|
|
/** \brief Similar to has_expr_metavar, but ignores metavariables occurring in the type
|
|
|
|
of local constants */
|
|
|
|
static bool has_expr_metavar_relaxed(expr const & e) {
|
|
|
|
if (!has_expr_metavar(e))
|
|
|
|
return false;
|
|
|
|
bool found = false;
|
|
|
|
for_each(e, [&](expr const & e, unsigned) {
|
|
|
|
if (found || !has_expr_metavar(e))
|
|
|
|
return false;
|
|
|
|
if (is_metavar(e)) {
|
|
|
|
found = true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (is_local(e))
|
|
|
|
return false; // do not visit type
|
|
|
|
return true;
|
|
|
|
});
|
|
|
|
return found;
|
|
|
|
}
|
|
|
|
|
2014-11-05 02:41:27 +00:00
|
|
|
constraint mk_placeholder_root_cnstr(std::shared_ptr<placeholder_context> const & C, local_context const & ctx,
|
|
|
|
expr const & m, bool is_strict, unifier_config const & cfg, delay_factor const & factor) {
|
2014-09-11 21:45:16 +00:00
|
|
|
environment const & env = C->env();
|
|
|
|
justification j = mk_failed_to_synthesize_jst(env, m);
|
2014-10-30 21:20:34 +00:00
|
|
|
|
2014-09-11 21:45:16 +00:00
|
|
|
auto choice_fn = [=](expr const & meta, expr const & meta_type, substitution const & s,
|
|
|
|
name_generator const & ngen) {
|
2014-09-26 02:46:08 +00:00
|
|
|
if (!is_ext_class(C->tc(), meta_type)) {
|
|
|
|
// do nothing, since type is not a class.
|
2014-09-11 21:45:16 +00:00
|
|
|
return lazy_list<constraints>(constraints());
|
|
|
|
}
|
|
|
|
pair<expr, justification> mj = update_meta(meta, s);
|
2014-10-04 17:36:10 +00:00
|
|
|
expr new_meta = mj.first;
|
|
|
|
justification new_j = mj.second;
|
2014-11-09 19:24:19 +00:00
|
|
|
unsigned depth = 0;
|
|
|
|
constraint c = mk_placeholder_cnstr(C, ctx, new_meta, depth);
|
2014-09-12 22:56:02 +00:00
|
|
|
unifier_config new_cfg(cfg);
|
|
|
|
new_cfg.m_discard = false;
|
|
|
|
new_cfg.m_use_exceptions = false;
|
2014-10-30 21:20:34 +00:00
|
|
|
|
|
|
|
auto to_cnstrs_fn = [=](substitution const & subst, constraints const & cnstrs) {
|
|
|
|
substitution new_s = subst;
|
|
|
|
// some constraints may have been postponed (example: universe level constraints)
|
|
|
|
constraints postponed = map(cnstrs,
|
|
|
|
[&](constraint const & c) {
|
|
|
|
// we erase internal justifications
|
|
|
|
return update_justification(c, mk_composite1(j, new_j));
|
|
|
|
});
|
|
|
|
metavar_closure cls(new_meta);
|
|
|
|
cls.add(meta_type);
|
|
|
|
bool relax = C->m_relax;
|
|
|
|
constraints cs = cls.mk_constraints(new_s, new_j, relax);
|
|
|
|
return append(cs, postponed);
|
|
|
|
};
|
|
|
|
|
2014-10-07 23:47:41 +00:00
|
|
|
unify_result_seq seq1 = unify(env, 1, &c, ngen, substitution(), new_cfg);
|
2014-10-30 21:20:34 +00:00
|
|
|
if (get_elaborator_unique_class_instances(C->m_ios.get_options())) {
|
|
|
|
optional<expr> solution;
|
|
|
|
substitution subst;
|
|
|
|
constraints cnstrs;
|
|
|
|
for_each(seq1, [&](pair<substitution, constraints> const & p) {
|
|
|
|
subst = p.first;
|
|
|
|
cnstrs = p.second;
|
|
|
|
expr next_solution = subst.instantiate(new_meta);
|
|
|
|
if (solution) {
|
2014-10-30 21:44:58 +00:00
|
|
|
throw_elaborator_exception(env, m, [=](formatter const & fmt) {
|
|
|
|
format r = format("ambiguous class-instance resolution, "
|
|
|
|
"there is more than one solution");
|
|
|
|
r += pp_indent_expr(fmt, *solution);
|
|
|
|
r += compose(line(), format("and"));
|
|
|
|
r += pp_indent_expr(fmt, next_solution);
|
|
|
|
return r;
|
|
|
|
});
|
2014-10-30 21:20:34 +00:00
|
|
|
} else {
|
|
|
|
solution = next_solution;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
if (!solution) {
|
|
|
|
if (is_strict)
|
|
|
|
return lazy_list<constraints>();
|
|
|
|
else
|
|
|
|
return lazy_list<constraints>(constraints());
|
|
|
|
} else {
|
2014-10-04 17:36:10 +00:00
|
|
|
// some constraints may have been postponed (example: universe level constraints)
|
2014-10-30 21:20:34 +00:00
|
|
|
return lazy_list<constraints>(to_cnstrs_fn(subst, cnstrs));
|
|
|
|
}
|
2014-09-26 02:46:08 +00:00
|
|
|
} else {
|
2014-10-30 21:20:34 +00:00
|
|
|
unify_result_seq seq2 = filter(seq1, [=](pair<substitution, constraints> const & p) {
|
|
|
|
substitution new_s = p.first;
|
|
|
|
expr result = new_s.instantiate(new_meta);
|
|
|
|
// We only keep complete solutions (modulo universe metavariables)
|
|
|
|
return !has_expr_metavar_relaxed(result);
|
|
|
|
});
|
|
|
|
lazy_list<constraints> seq3 = map2<constraints>(seq2, [=](pair<substitution, constraints> const & p) {
|
|
|
|
return to_cnstrs_fn(p.first, p.second);
|
|
|
|
});
|
|
|
|
if (is_strict) {
|
|
|
|
return seq3;
|
|
|
|
} else {
|
|
|
|
// make sure it does not fail by appending empty set of constraints
|
|
|
|
return append(seq3, lazy_list<constraints>(constraints()));
|
|
|
|
}
|
2014-09-26 02:46:08 +00:00
|
|
|
}
|
2014-09-11 21:45:16 +00:00
|
|
|
};
|
|
|
|
bool owner = false;
|
|
|
|
bool relax = C->m_relax;
|
2014-09-28 04:47:37 +00:00
|
|
|
return mk_choice_cnstr(m, choice_fn, factor, owner, j, relax);
|
2014-09-11 21:45:16 +00:00
|
|
|
}
|
|
|
|
|
2014-09-10 22:20:45 +00:00
|
|
|
/** \brief Create a metavariable, and attach choice constraint for generating
|
2014-10-07 16:44:01 +00:00
|
|
|
solutions using class-instances
|
2014-09-10 22:20:45 +00:00
|
|
|
*/
|
|
|
|
pair<expr, constraint> mk_placeholder_elaborator(
|
2014-09-25 17:11:41 +00:00
|
|
|
environment const & env, io_state const & ios, local_context const & ctx,
|
2014-10-30 19:45:41 +00:00
|
|
|
name const & prefix, optional<name> const & suffix, bool relax, bool use_local_instances,
|
2014-11-09 19:24:19 +00:00
|
|
|
bool is_strict, optional<expr> const & type, tag g, unifier_config const & cfg,
|
|
|
|
pos_info_provider const * pip) {
|
2014-11-09 19:47:01 +00:00
|
|
|
auto C = std::make_shared<placeholder_context>(env, ios, prefix, relax, use_local_instances);
|
2014-11-05 02:41:27 +00:00
|
|
|
expr m = ctx.mk_meta(C->m_ngen, suffix, type, g);
|
2014-11-09 19:59:44 +00:00
|
|
|
C->set_main_meta(m);
|
2014-11-09 19:47:01 +00:00
|
|
|
if (pip)
|
|
|
|
C->set_pos(pip->get_file_name(), pip->get_pos_info(m));
|
2014-11-05 02:41:27 +00:00
|
|
|
constraint c = mk_placeholder_root_cnstr(C, ctx, m, is_strict, cfg, delay_factor());
|
2014-09-11 21:45:16 +00:00
|
|
|
return mk_pair(m, c);
|
2014-09-10 22:20:45 +00:00
|
|
|
}
|
|
|
|
}
|