refactor(kernel/type_checker): simplify replace constraint_handler with closure

Signed-off-by: Leonardo de Moura <leonardo@microsoft.com>
This commit is contained in:
Leonardo de Moura 2014-06-21 12:22:24 -07:00
parent 86a56fbd2d
commit 37bee8c852
6 changed files with 33 additions and 61 deletions

View file

@ -27,8 +27,8 @@ no_constraints_allowed_exception::no_constraints_allowed_exception():exception("
exception * no_constraints_allowed_exception::clone() const { return new no_constraints_allowed_exception(); } exception * no_constraints_allowed_exception::clone() const { return new no_constraints_allowed_exception(); }
void no_constraints_allowed_exception::rethrow() const { throw *this; } void no_constraints_allowed_exception::rethrow() const { throw *this; }
void no_constraint_handler::add_cnstr(constraint const &) { add_cnstr_fn mk_no_contranint_fn() {
throw no_constraints_allowed_exception(); return add_cnstr_fn([](constraint const &) { throw no_constraints_allowed_exception(); });
} }
/** \brief Auxiliary functional object used to implement type checker. */ /** \brief Auxiliary functional object used to implement type checker. */
@ -58,7 +58,7 @@ struct type_checker::imp {
environment m_env; environment m_env;
name_generator m_gen; name_generator m_gen;
constraint_handler & m_chandler; add_cnstr_fn m_add_cnstr_fn;
std::unique_ptr<converter> m_conv; std::unique_ptr<converter> m_conv;
// In the type checker cache, we must take into account binder information. // In the type checker cache, we must take into account binder information.
// Examples: // Examples:
@ -71,8 +71,8 @@ struct type_checker::imp {
// temp flag // temp flag
level_param_names m_params; level_param_names m_params;
imp(environment const & env, name_generator const & g, constraint_handler & h, std::unique_ptr<converter> && conv, bool memoize): imp(environment const & env, name_generator const & g, add_cnstr_fn const & h, std::unique_ptr<converter> && conv, bool memoize):
m_env(env), m_gen(g), m_chandler(h), m_conv(std::move(conv)), m_conv_ctx(*this), m_tc_ctx(*this), m_env(env), m_gen(g), m_add_cnstr_fn(h), m_conv(std::move(conv)), m_conv_ctx(*this), m_tc_ctx(*this),
m_memoize(memoize) {} m_memoize(memoize) {}
optional<expr> expand_macro(expr const & m) { optional<expr> expand_macro(expr const & m) {
@ -89,9 +89,9 @@ struct type_checker::imp {
return mk_pair(instantiate(binding_body(e), local), local); return mk_pair(instantiate(binding_body(e), local), local);
} }
/** \brief Add given constraint to the constraint handler m_chandler. */ /** \brief Add given constraint using m_add_cnstr_fn. */
void add_cnstr(constraint const & c) { void add_cnstr(constraint const & c) {
m_chandler.add_cnstr(c); m_add_cnstr_fn(c);
} }
/** \brief Return true iff \c t and \c s are definitionally equal */ /** \brief Return true iff \c t and \c s are definitionally equal */
@ -403,18 +403,19 @@ struct type_checker::imp {
expr whnf(expr const & t) { return m_conv->whnf(t, m_conv_ctx); } expr whnf(expr const & t) { return m_conv->whnf(t, m_conv_ctx); }
}; };
no_constraint_handler g_no_constraint_handler; static add_cnstr_fn g_no_constraint_fn = mk_no_contranint_fn();
type_checker::type_checker(environment const & env, name_generator const & g, constraint_handler & h, std::unique_ptr<converter> && conv, bool memoize): type_checker::type_checker(environment const & env, name_generator const & g, add_cnstr_fn const & h,
std::unique_ptr<converter> && conv, bool memoize):
m_ptr(new imp(env, g, h, std::move(conv), memoize)) {} m_ptr(new imp(env, g, h, std::move(conv), memoize)) {}
type_checker::type_checker(environment const & env, name_generator const & g, std::unique_ptr<converter> && conv, bool memoize): type_checker::type_checker(environment const & env, name_generator const & g, std::unique_ptr<converter> && conv, bool memoize):
type_checker(env, g, g_no_constraint_handler, std::move(conv), memoize) {} type_checker(env, g, g_no_constraint_fn, std::move(conv), memoize) {}
static name g_tmp_prefix = name::mk_internal_unique_name(); static name g_tmp_prefix = name::mk_internal_unique_name();
type_checker::type_checker(environment const & env): type_checker::type_checker(environment const & env):
type_checker(env, name_generator(g_tmp_prefix), g_no_constraint_handler, mk_default_converter(env), true) {} type_checker(env, name_generator(g_tmp_prefix), g_no_constraint_fn, mk_default_converter(env), true) {}
type_checker::~type_checker() {} type_checker::~type_checker() {}
expr type_checker::infer(expr const & t) { return m_ptr->infer_type(t); } expr type_checker::infer(expr const & t) { return m_ptr->infer_type(t); }

View file

@ -15,17 +15,10 @@ Author: Leonardo de Moura
#include "kernel/converter.h" #include "kernel/converter.h"
namespace lean { namespace lean {
class constraint_handler { typedef std::function<void(constraint const & c)> add_cnstr_fn;
public:
virtual ~constraint_handler() {}
virtual void add_cnstr(constraint const & c) = 0;
};
/** \brief This handler always throw an exception (\c no_constraints_allowed_exception) when \c add_cnstr is invoked. */ /** \brief This handler always throw an exception (\c no_constraints_allowed_exception) when \c add_cnstr is invoked. */
class no_constraint_handler : public constraint_handler { add_cnstr_fn mk_no_contranint_fn();
public:
virtual void add_cnstr(constraint const & c);
};
/** \brief Exception used in \c no_constraint_handler. */ /** \brief Exception used in \c no_constraint_handler. */
class no_constraints_allowed_exception : public exception { class no_constraints_allowed_exception : public exception {
@ -51,9 +44,9 @@ public:
memoize: if true, then inferred types are memoized/cached memoize: if true, then inferred types are memoized/cached
*/ */
type_checker(environment const & env, name_generator const & g, constraint_handler & h, std::unique_ptr<converter> && conv, type_checker(environment const & env, name_generator const & g, add_cnstr_fn const & h, std::unique_ptr<converter> && conv,
bool memoize = true); bool memoize = true);
type_checker(environment const & env, name_generator const & g, constraint_handler & h, bool memoize = true): type_checker(environment const & env, name_generator const & g, add_cnstr_fn const & h, bool memoize = true):
type_checker(env, g, h, mk_default_converter(env), memoize) {} type_checker(env, g, h, mk_default_converter(env), memoize) {}
/** /**
\brief Similar to the previous constructor, but if a method tries to create a constraint, then an \brief Similar to the previous constructor, but if a method tries to create a constraint, then an

View file

@ -1808,37 +1808,16 @@ static void open_substitution(lua_State * L) {
SET_GLOBAL_FUN(substitution_pred, "is_substitution"); SET_GLOBAL_FUN(substitution_pred, "is_substitution");
} }
// constraint_handler // add_cnstr_fn
class lua_constraint_handler : public constraint_handler { add_cnstr_fn to_add_cnstr_fn(lua_State * L, int idx) {
luaref m_f; luaL_checktype(L, idx, LUA_TFUNCTION); // user-fun
public: luaref f(L, idx);
lua_constraint_handler(luaref const & f):m_f(f) {} return add_cnstr_fn([=](constraint const & c) {
virtual void add_cnstr(constraint const & c) { lua_State * L = f.get_state();
lua_State * L = m_f.get_state(); f.push();
m_f.push();
push_constraint(L, c); push_constraint(L, c);
pcall(L, 1, 0, 0); pcall(L, 1, 0, 0);
} });
};
DECL_UDATA(lua_constraint_handler)
int mk_constraint_handler(lua_State * L) {
luaL_checktype(L, 1, LUA_TFUNCTION); // user-fun
return push_lua_constraint_handler(L, lua_constraint_handler(luaref(L, 1)));
}
static const struct luaL_Reg lua_constraint_handler_m[] = {
{"__gc", lua_constraint_handler_gc},
{0, 0}
};
static void open_constraint_handler(lua_State * L) {
luaL_newmetatable(L, lua_constraint_handler_mt);
lua_pushvalue(L, -1);
lua_setfield(L, -2, "__index");
setfuncs(L, lua_constraint_handler_m, 0);
SET_GLOBAL_FUN(mk_constraint_handler, "constraint_handler");
SET_GLOBAL_FUN(lua_constraint_handler_pred, "is_constraint_handler");
} }
// type_checker // type_checker
@ -1857,9 +1836,9 @@ int mk_type_checker(lua_State * L) {
return push_type_checker_ref(L, std::make_shared<type_checker>(to_environment(L, 1))); return push_type_checker_ref(L, std::make_shared<type_checker>(to_environment(L, 1)));
} else if (nargs == 2) { } else if (nargs == 2) {
return push_type_checker_ref(L, std::make_shared<type_checker>(to_environment(L, 1), to_name_generator(L, 2))); return push_type_checker_ref(L, std::make_shared<type_checker>(to_environment(L, 1), to_name_generator(L, 2)));
} else if (nargs == 3 && is_lua_constraint_handler(L, 3)) { } else if (nargs == 3 && lua_isfunction(L, 3)) {
return push_type_checker_ref(L, std::make_shared<type_checker>(to_environment(L, 1), to_name_generator(L, 2), return push_type_checker_ref(L, std::make_shared<type_checker>(to_environment(L, 1), to_name_generator(L, 2),
to_lua_constraint_handler(L, 3))); to_add_cnstr_fn(L, 3)));
} else { } else {
optional<module_idx> mod_idx; bool memoize; name_set extra_opaque; optional<module_idx> mod_idx; bool memoize; name_set extra_opaque;
if (nargs == 3) { if (nargs == 3) {
@ -1871,7 +1850,7 @@ int mk_type_checker(lua_State * L) {
} else { } else {
get_type_checker_args(L, 4, mod_idx, memoize, extra_opaque); get_type_checker_args(L, 4, mod_idx, memoize, extra_opaque);
auto t = std::make_shared<type_checker>(to_environment(L, 1), to_name_generator(L, 2), auto t = std::make_shared<type_checker>(to_environment(L, 1), to_name_generator(L, 2),
to_lua_constraint_handler(L, 3), to_add_cnstr_fn(L, 3),
mk_default_converter(to_environment(L, 1), mod_idx, memoize, extra_opaque), mk_default_converter(to_environment(L, 1), mod_idx, memoize, extra_opaque),
memoize); memoize);
return push_type_checker_ref(L, t); return push_type_checker_ref(L, t);
@ -2046,7 +2025,6 @@ void open_kernel_module(lua_State * L) {
open_justification(L); open_justification(L);
open_constraint(L); open_constraint(L);
open_substitution(L); open_substitution(L);
open_constraint_handler(L);
open_type_checker(L); open_type_checker(L);
open_inductive(L); open_inductive(L);
} }

View file

@ -8,7 +8,7 @@ local b = Const("b")
print(t(b)) print(t(b))
assert(tc:whnf(t(b)) == b) assert(tc:whnf(t(b)) == b)
local cs = {} local cs = {}
local tc2 = type_checker(env, g, constraint_handler(function (c) print(c); cs[#cs+1] = c end)) local tc2 = type_checker(env, g, function (c) print(c); cs[#cs+1] = c end)
assert(tc:check(Bool) == mk_sort(mk_level_one())) assert(tc:check(Bool) == mk_sort(mk_level_one()))
print(tc:infer(t)) print(tc:infer(t))
local m = mk_metavar("m1", mk_metavar("m2", mk_sort(mk_meta_univ("u")))) local m = mk_metavar("m1", mk_metavar("m2", mk_sort(mk_meta_univ("u"))))

View file

@ -54,7 +54,7 @@ print(tc:check(Fun({{A, mk_sort(u)}, {a, A}, {b, A}, {c, A}, {d, A},
trans_u(A, b, c, d, H2, H3))))) trans_u(A, b, c, d, H2, H3)))))
local cs = {} local cs = {}
local g = name_generator("tst") local g = name_generator("tst")
local tc2 = type_checker(env, g, constraint_handler(function (c) print(c); cs[#cs+1] = c end)) local tc2 = type_checker(env, g, function (c) print(c); cs[#cs+1] = c end)
print("=================") print("=================")
local f = Const("f") local f = Const("f")
local mf_ty = mk_metavar("f_ty", Pi(A, mk_sort(u), mk_sort(mk_meta_univ("l_f")))) local mf_ty = mk_metavar("f_ty", Pi(A, mk_sort(u), mk_sort(mk_meta_univ("l_f"))))
@ -64,7 +64,7 @@ print(tc2:check(Fun({{A, mk_sort(u)}, {f, mf_ty(A)}, {a, A}},
local cs = {} local cs = {}
local tc2 = type_checker(env, g, constraint_handler(function (c) print(c); cs[#cs+1] = c end)) local tc2 = type_checker(env, g, function (c) print(c); cs[#cs+1] = c end)
local scope = {{A, mk_sort(u)}, {a, A}, {b, A}, {c, A}, {d, A}, {H1, id_u(A, b, a)}, local scope = {{A, mk_sort(u)}, {a, A}, {b, A}, {c, A}, {d, A}, {H1, id_u(A, b, a)},
{H2, id_u(A, b, c)}, {H3, id_u(A, c, d)}} {H2, id_u(A, b, c)}, {H3, id_u(A, c, d)}}
local mP = mk_metavar("P", Pi(scope, mk_metavar("P_ty", Pi(scope, mk_sort(mk_meta_univ("l_P"))))(A, a, b, c, d, H1, H2, H3)))(A, a, b, c, d, H1, H2, H3) local mP = mk_metavar("P", Pi(scope, mk_metavar("P_ty", Pi(scope, mk_sort(mk_meta_univ("l_P"))))(A, a, b, c, d, H1, H2, H3)))(A, a, b, c, d, H1, H2, H3)

View file

@ -13,7 +13,7 @@ local a = Const("a")
local m1 = mk_metavar("m1", N) local m1 = mk_metavar("m1", N)
local cs = {} local cs = {}
local ngen = name_generator("tst") local ngen = name_generator("tst")
local tc = type_checker(env, ngen, constraint_handler(function (c) print(c); cs[#cs+1] = c end)) local tc = type_checker(env, ngen, function (c) print(c); cs[#cs+1] = c end)
assert(tc:is_def_eq(f(m1), g(a))) assert(tc:is_def_eq(f(m1), g(a)))
assert(tc:is_def_eq(f(m1), a)) assert(tc:is_def_eq(f(m1), a))
assert(not tc:is_def_eq(f(a), a)) assert(not tc:is_def_eq(f(a), a))