feat(kernel/type_checker): expose app_delayed_justification

Signed-off-by: Leonardo de Moura <leonardo@microsoft.com>
This commit is contained in:
Leonardo de Moura 2014-06-24 14:54:37 -07:00
parent 2c0f596665
commit 609aeae390
2 changed files with 30 additions and 26 deletions

View file

@ -233,32 +233,21 @@ void type_checker::check_level(level const & l, expr const & s) {
throw_kernel_exception(m_env, sstream() << "invalid reference to undefined universe level parameter '" << *n2 << "'", s); throw_kernel_exception(m_env, sstream() << "invalid reference to undefined universe level parameter '" << *n2 << "'", s);
} }
/** app_delayed_justification::app_delayed_justification(environment const & env, expr const & e, expr const & f_type, expr const & a_type):
\brief Create a justification for a application type mismatch, m_env(env), m_e(e), m_fn_type(f_type), m_arg_type(a_type) {}
\c e is the application, \c fn_type and \c arg_type are the function and argument type.
*/
struct app_delayed_jst : public delayed_justification {
environment const & m_env;
expr const & m_e;
expr const & m_fn_type;
expr const & m_arg_type;
optional<justification> m_jst;
app_delayed_jst(environment const & env, expr const & e, expr const & f_type, expr const & a_type):
m_env(env), m_e(e), m_fn_type(f_type), m_arg_type(a_type) {}
virtual justification get() { justification app_delayed_justification::get() {
if (!m_jst) { if (!m_jst) {
m_jst = mk_justification(app_arg(m_e), m_jst = mk_justification(app_arg(m_e),
[=](formatter const & fmt, options const & o, substitution const & subst) { [=](formatter const & fmt, options const & o, substitution const & subst) {
return pp_app_type_mismatch(fmt, m_env, o, return pp_app_type_mismatch(fmt, m_env, o,
subst.instantiate_metavars_wo_jst(m_e), subst.instantiate_metavars_wo_jst(m_e),
subst.instantiate_metavars_wo_jst(binding_domain(m_fn_type)), subst.instantiate_metavars_wo_jst(binding_domain(m_fn_type)),
subst.instantiate_metavars_wo_jst(m_arg_type)); subst.instantiate_metavars_wo_jst(m_arg_type));
}); });
}
return *m_jst;
} }
}; return *m_jst;
}
/** /**
\brief Return type of expression \c e, if \c infer_only is false, then it also check whether \c e is type correct or not. \brief Return type of expression \c e, if \c infer_only is false, then it also check whether \c e is type correct or not.
@ -343,7 +332,7 @@ expr type_checker::infer_type_core(expr const & e, bool infer_only) {
expr f_type = ensure_pi_core(infer_type_core(app_fn(e), infer_only), app_fn(e)); expr f_type = ensure_pi_core(infer_type_core(app_fn(e), infer_only), app_fn(e));
if (!infer_only) { if (!infer_only) {
expr a_type = infer_type_core(app_arg(e), infer_only); expr a_type = infer_type_core(app_arg(e), infer_only);
app_delayed_jst jst(m_env, e, f_type, a_type); app_delayed_justification jst(m_env, e, f_type, a_type);
if (!is_def_eq(a_type, binding_domain(f_type), jst)) { if (!is_def_eq(a_type, binding_domain(f_type), jst)) {
environment env = m_env; environment env = m_env;
throw_kernel_exception(m_env, app_arg(e), throw_kernel_exception(m_env, app_arg(e),

View file

@ -81,7 +81,6 @@ class type_checker {
void check_level(level const & l, expr const & s); void check_level(level const & l, expr const & s);
expr infer_type_core(expr const & e, bool infer_only); expr infer_type_core(expr const & e, bool infer_only);
expr infer_type(expr const & e); expr infer_type(expr const & e);
bool is_def_eq(expr const & t, expr const & s, delayed_justification & jst);
extension_context & get_extension() { return m_tc_ctx; } extension_context & get_extension() { return m_tc_ctx; }
public: public:
class scope { class scope {
@ -140,6 +139,7 @@ public:
/** \brief Return true iff t is definitionally equal to s. */ /** \brief Return true iff t is definitionally equal to s. */
bool is_def_eq(expr const & t, expr const & s); bool is_def_eq(expr const & t, expr const & s);
bool is_def_eq(expr const & t, expr const & s, justification const & j); bool is_def_eq(expr const & t, expr const & s, justification const & j);
bool is_def_eq(expr const & t, expr const & s, delayed_justification & jst);
/** \brief Return true iff t is a proposition. */ /** \brief Return true iff t is a proposition. */
bool is_prop(expr const & t); bool is_prop(expr const & t);
/** \brief Return the weak head normal form of \c t. */ /** \brief Return the weak head normal form of \c t. */
@ -173,4 +173,19 @@ public:
certified_declaration check(environment const & env, declaration const & d, certified_declaration check(environment const & env, declaration const & d,
name_generator const & g, name_set const & extra_opaque = name_set(), bool memoize = true); name_generator const & g, name_set const & extra_opaque = name_set(), bool memoize = true);
certified_declaration check(environment const & env, declaration const & d, name_set const & extra_opaque = name_set(), bool memoize = true); certified_declaration check(environment const & env, declaration const & d, name_set const & extra_opaque = name_set(), bool memoize = true);
/**
\brief Create a justification for a application type mismatch,
\c e is the application, \c fn_type and \c arg_type are the function and argument type.
*/
class app_delayed_justification : public delayed_justification {
environment const & m_env;
expr const & m_e;
expr const & m_fn_type;
expr const & m_arg_type;
optional<justification> m_jst;
public:
app_delayed_justification(environment const & env, expr const & e, expr const & f_type, expr const & a_type);
virtual justification get();
};
} }