fix(kernel/type_checker): check for undefined global/local levels

Signed-off-by: Leonardo de Moura <leonardo@microsoft.com>
This commit is contained in:
Leonardo de Moura 2014-05-12 14:23:04 -07:00
parent b6d6df161f
commit 89d6a1e691

View file

@ -298,6 +298,13 @@ struct type_checker::imp {
}); });
} }
void check_level(level const & l) {
if (auto n1 = get_undef_global(l, m_env))
throw_kernel_exception(m_env, sstream() << "invalid reference to undefined global universe level '" << *n1 << "'");
if (auto n2 = get_undef_param(l, m_params))
throw_kernel_exception(m_env, sstream() << "invalid reference to undefined universe level parameter '" << *n2 << "'");
}
/** /**
\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.
@ -321,6 +328,8 @@ struct type_checker::imp {
case expr_kind::Var: case expr_kind::Var:
throw_kernel_exception(m_env, "type checker does not support free variables, replace them with local constants before invoking it"); throw_kernel_exception(m_env, "type checker does not support free variables, replace them with local constants before invoking it");
case expr_kind::Sort: case expr_kind::Sort:
if (!infer_only)
check_level(sort_level(e));
r = mk_sort(mk_succ(sort_level(e))); r = mk_sort(mk_succ(sort_level(e)));
break; break;
case expr_kind::Constant: { case expr_kind::Constant: {
@ -330,6 +339,10 @@ struct type_checker::imp {
if (length(ps) != length(ls)) if (length(ps) != length(ls))
throw_kernel_exception(m_env, sstream() << "incorrect number of universe levels parameters for '" << const_name(e) << "', #" throw_kernel_exception(m_env, sstream() << "incorrect number of universe levels parameters for '" << const_name(e) << "', #"
<< length(ps) << " expected, #" << length(ls) << " provided"); << length(ps) << " expected, #" << length(ls) << " provided");
if (!infer_only) {
for (level const & l : ls)
check_level(l);
}
r = instantiate_params(d.get_type(), ps, ls); r = instantiate_params(d.get_type(), ps, ls);
break; break;
} }
@ -463,13 +476,13 @@ certified_definition check(environment const & env, definition const & d, name_g
check_name(env, d.get_name()); check_name(env, d.get_name());
type_checker checker1(env, g, mk_default_converter(env, optional<module_idx>(), memoize, extra_opaque)); type_checker checker1(env, g, mk_default_converter(env, optional<module_idx>(), memoize, extra_opaque));
checker1.check(d.get_type()); checker1.check(d.get_type(), d.get_params());
if (d.is_definition()) { if (d.is_definition()) {
optional<module_idx> midx; optional<module_idx> midx;
if (d.is_opaque()) if (d.is_opaque())
midx = optional<module_idx>(d.get_module_idx()); midx = optional<module_idx>(d.get_module_idx());
type_checker checker2(env, g, mk_default_converter(env, midx, memoize, extra_opaque)); type_checker checker2(env, g, mk_default_converter(env, midx, memoize, extra_opaque));
expr val_type = checker2.check(d.get_value()); expr val_type = checker2.check(d.get_value(), d.get_params());
if (!checker2.is_def_eq(val_type, d.get_type())) { if (!checker2.is_def_eq(val_type, d.get_type())) {
throw_kernel_exception(env, d.get_value(), throw_kernel_exception(env, d.get_value(),
[=](formatter const & fmt, options const & o) { [=](formatter const & fmt, options const & o) {