Allow Boolean expressions (aka propositions) to be used as types.

Signed-off-by: Leonardo de Moura <leonardo@microsoft.com>
This commit is contained in:
Leonardo de Moura 2013-08-04 18:26:01 -07:00
parent 4b5d60f2b2
commit 5f77a2367f
5 changed files with 69 additions and 33 deletions

View file

@ -32,5 +32,8 @@ inline context extend(context const & c, name const & n, expr const & t, expr co
inline context extend(context const & c, name const & n, expr const & t) { inline context extend(context const & c, name const & n, expr const & t) {
return context(context_entry(n, t), c); return context(context_entry(n, t), c);
} }
inline bool empty(context const & c) {
return is_nil(c);
}
std::ostream & operator<<(std::ostream & out, context const & c); std::ostream & operator<<(std::ostream & out, context const & c);
} }

View file

@ -7,6 +7,7 @@ Author: Leonardo de Moura
#pragma once #pragma once
#include <iostream> #include <iostream>
#include <memory> #include <memory>
#include "expr.h"
#include "level.h" #include "level.h"
namespace lean { namespace lean {
@ -67,5 +68,22 @@ public:
\pre has_parent() \pre has_parent()
*/ */
environment parent() const; environment parent() const;
/**
\brief Add a new definition n : t := v.
It throws an exception if v does not have type t.
If opaque == true, then definition is not used by normalizer.
*/
void add_definition(name const & n, expr const & t, expr const & v, bool opaque = false);
/**
\brief Add a new definition n : infer_type(v) := v.
If opaque == true, then definition is not used by normalizer.
*/
void add_definition(name const & n, expr const & v, bool opaque = false);
/**
*/
void add_fact(name const & n, expr const & t);
}; };
} }

View file

@ -14,13 +14,27 @@ Author: Leonardo de Moura
#include "trace.h" #include "trace.h"
namespace lean { namespace lean {
bool is_convertible_core(expr const & expected, expr const & given, environment const & env) {
if (expected == given)
return true;
if (is_type(expected) && is_type(given)) {
if (env.is_ge(ty_level(expected), ty_level(given)))
return true;
}
return false;
}
bool is_convertible(expr const & expected, expr const & given, environment const & env, context const & ctx) {
if (is_convertible_core(expected, given, env))
return true;
expr e_n = normalize(expected, env, ctx);
expr g_n = normalize(given, env, ctx);
return is_convertible_core(e_n, g_n, env);
}
class infer_type_fn { class infer_type_fn {
environment const & m_env; environment const & m_env;
expr normalize(expr const & e, context const & ctx) {
return ::lean::normalize(e, m_env, ctx);
}
expr lookup(context const & c, unsigned i) { expr lookup(context const & c, unsigned i) {
context const & def_c = ::lean::lookup(c, i); context const & def_c = ::lean::lookup(c, i);
lean_assert(length(c) >= length(def_c)); lean_assert(length(c) >= length(def_c));
@ -30,22 +44,30 @@ class infer_type_fn {
level infer_universe(expr const & t, context const & ctx) { level infer_universe(expr const & t, context const & ctx) {
lean_trace("type_check", tout << "infer universe\n" << t << "\n";); lean_trace("type_check", tout << "infer universe\n" << t << "\n";);
expr u = normalize(infer_type(t, ctx), ctx); expr u = normalize(infer_type(t, ctx), m_env, ctx);
if (is_type(u)) if (is_type(u))
return ty_level(u); return ty_level(u);
if (is_bool_type(u))
return level();
std::ostringstream buffer; std::ostringstream buffer;
buffer << "type expected, in context:\n" << ctx << "\ngiven:\n" << t; buffer << "type expected";
if (!empty(ctx))
buffer << ", in context:\n" << ctx;
buffer << "\ngiven:\n" << t;
throw exception(buffer.str()); throw exception(buffer.str());
} }
expr check_pi(expr const & e, context const & ctx) { expr check_pi(expr const & e, context const & ctx) {
if (is_pi(e)) if (is_pi(e))
return e; return e;
expr r = normalize(e, ctx); expr r = normalize(e, m_env, ctx);
if (is_pi(r)) if (is_pi(r))
return r; return r;
std::ostringstream buffer; std::ostringstream buffer;
buffer << "function expected, in context:\n" << ctx << "\ngiven:\n" << e; buffer << "function expected";
if (!empty(ctx))
buffer << ", in context:\n" << ctx;
buffer << "\ngiven:\n" << e;
throw exception(buffer.str()); throw exception(buffer.str());
} }
@ -53,30 +75,6 @@ class infer_type_fn {
return check_pi(infer_type(e, ctx), ctx); return check_pi(infer_type(e, ctx), ctx);
} }
bool check_type_core(expr const & expected, expr const & given) {
if (expected == given)
return true;
if (is_type(expected) && is_type(given)) {
if (m_env.is_ge(ty_level(expected), ty_level(given)))
return true;
}
return false;
}
void check_type(expr const & e, unsigned i, expr const & expected, expr const & given, context const & ctx) {
if (check_type_core(expected, given))
return;
expr e_n = normalize(expected, ctx);
expr g_n = normalize(given, ctx);
if (check_type_core(e_n, g_n))
return;
std::ostringstream buffer;
buffer << "type mismatch at argument " << i << " of\n" << e
<< "\nexpected type:\n" << expected
<< "\ngiven type:\n" << given << "\nin context:\n" << ctx;;
throw exception(buffer.str());
}
expr infer_type(expr const & e, context const & ctx) { expr infer_type(expr const & e, context const & ctx) {
lean_trace("type_check", tout << "infer type\n" << e << "\n" << ctx << "\n";); lean_trace("type_check", tout << "infer type\n" << e << "\n" << ctx << "\n";);
switch (e.kind()) { switch (e.kind()) {
@ -93,7 +91,14 @@ class infer_type_fn {
while (true) { while (true) {
expr const & c = arg(e, i); expr const & c = arg(e, i);
expr c_t = infer_type(c, ctx); expr c_t = infer_type(c, ctx);
check_type(e, i, abst_domain(f_t), c_t, ctx); if (!is_convertible(abst_domain(f_t), c_t, m_env, ctx)) {
std::ostringstream buffer;
buffer << "type mismatch at argument " << i << " of\n" << e
<< "\nexpected type:\n" << abst_domain(f_t)
<< "\ngiven type:\n" << c_t;
if (!empty(ctx))
buffer << "\nin context:\n" << ctx;
}
f_t = instantiate(abst_body(f_t), c); f_t = instantiate(abst_body(f_t), c);
i++; i++;
if (i == num) if (i == num)

View file

@ -11,4 +11,5 @@ Author: Leonardo de Moura
namespace lean { namespace lean {
expr infer_type(expr const & e, environment const & env, context const & ctx = context()); expr infer_type(expr const & e, environment const & env, context const & ctx = context());
bool is_convertible(expr const & t1, expr const & t2, environment const & env, context const & ctx = context());
} }

View file

@ -9,6 +9,7 @@ Author: Leonardo de Moura
#include "abstract.h" #include "abstract.h"
#include "exception.h" #include "exception.h"
#include "builtin.h" #include "builtin.h"
#include "arith.h"
#include "trace.h" #include "trace.h"
#include "test.h" #include "test.h"
using namespace lean; using namespace lean;
@ -73,11 +74,19 @@ static void tst3() {
std::cout << infer_type(t, env) << "\n"; std::cout << infer_type(t, env) << "\n";
} }
static void tst4() {
environment env;
expr a = eq(int_value(1), int_value(2));
expr pr = lambda("x", a, var(0));
std::cout << infer_type(pr, env) << "\n";
}
int main() { int main() {
continue_on_violation(true); continue_on_violation(true);
enable_trace("type_check"); enable_trace("type_check");
tst1(); tst1();
tst2(); tst2();
tst3(); tst3();
tst4();
return has_violations() ? 1 : 0; return has_violations() ? 1 : 0;
} }