2014-06-03 09:34:12 +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-06-14 00:30:35 +00:00
|
|
|
#include <limits>
|
2014-06-05 20:10:50 +00:00
|
|
|
#include <utility>
|
2014-08-19 23:28:58 +00:00
|
|
|
#include "util/pair.h"
|
2014-06-10 16:11:45 +00:00
|
|
|
#include "frontends/lean/token_table.h"
|
2014-06-03 09:34:12 +00:00
|
|
|
|
|
|
|
namespace lean {
|
2014-12-04 23:11:23 +00:00
|
|
|
static unsigned g_arrow_prec = 25;
|
2014-12-12 01:31:12 +00:00
|
|
|
static unsigned g_decreasing_prec = 100;
|
2014-12-04 23:11:23 +00:00
|
|
|
static unsigned g_max_prec = 1024;
|
2015-01-20 02:40:33 +00:00
|
|
|
static unsigned g_Max_prec = 1024*1024;
|
2014-12-04 23:11:23 +00:00
|
|
|
static unsigned g_plus_prec = 65;
|
|
|
|
static unsigned g_cup_prec = 60;
|
2014-06-25 19:50:47 +00:00
|
|
|
unsigned get_max_prec() { return g_max_prec; }
|
2015-01-20 02:40:33 +00:00
|
|
|
unsigned get_Max_prec() { return g_Max_prec; }
|
2014-06-13 22:13:32 +00:00
|
|
|
unsigned get_arrow_prec() { return g_arrow_prec; }
|
2014-12-04 23:11:23 +00:00
|
|
|
unsigned get_decreasing_prec() { return g_decreasing_prec; }
|
2015-04-28 20:43:05 +00:00
|
|
|
static token_table update(token_table const & s, char const * token, char const * val,
|
|
|
|
optional<unsigned> expr_prec, optional<unsigned> tac_prec) {
|
|
|
|
lean_assert(expr_prec || tac_prec);
|
|
|
|
token_info info(token, val, 0, 0);
|
|
|
|
if (token_info const * old_info = find(s, token)) {
|
|
|
|
info = info.update_expr_precedence(old_info->expr_precedence());
|
|
|
|
info = info.update_tactic_precedence(old_info->tactic_precedence());
|
|
|
|
}
|
|
|
|
if (expr_prec)
|
|
|
|
info = info.update_expr_precedence(*expr_prec);
|
|
|
|
if (tac_prec)
|
|
|
|
info = info.update_tactic_precedence(*tac_prec);
|
|
|
|
return insert(s, token, info);
|
|
|
|
}
|
2014-06-10 16:11:45 +00:00
|
|
|
token_table add_command_token(token_table const & s, char const * token) {
|
2014-06-05 19:40:31 +00:00
|
|
|
return insert(s, token, token_info(token));
|
2014-06-03 09:34:12 +00:00
|
|
|
}
|
2014-06-10 16:11:45 +00:00
|
|
|
token_table add_command_token(token_table const & s, char const * token, char const * val) {
|
2014-06-18 16:14:50 +00:00
|
|
|
return insert(s, token, token_info(token, val));
|
2014-06-05 20:05:40 +00:00
|
|
|
}
|
2014-06-10 16:11:45 +00:00
|
|
|
token_table add_token(token_table const & s, char const * token, unsigned prec) {
|
2015-04-28 20:43:05 +00:00
|
|
|
return update(s, token, token, optional<unsigned>(prec), optional<unsigned>());
|
2014-06-05 19:40:31 +00:00
|
|
|
}
|
2014-06-10 16:11:45 +00:00
|
|
|
token_table add_token(token_table const & s, char const * token, char const * val, unsigned prec) {
|
2015-04-28 20:43:05 +00:00
|
|
|
return update(s, token, val, optional<unsigned>(prec), optional<unsigned>());
|
|
|
|
}
|
|
|
|
token_table add_tactic_token(token_table const & s, char const * token, unsigned prec) {
|
|
|
|
return update(s, token, token, optional<unsigned>(), optional<unsigned>(prec));
|
|
|
|
}
|
|
|
|
token_table add_tactic_token(token_table const & s, char const * token, char const * val, unsigned prec) {
|
|
|
|
return update(s, token, val, optional<unsigned>(), optional<unsigned>(prec));
|
2014-06-05 19:40:31 +00:00
|
|
|
}
|
2014-06-10 16:11:45 +00:00
|
|
|
token_table const * find(token_table const & s, char c) {
|
2014-06-05 19:40:31 +00:00
|
|
|
return s.find(c);
|
|
|
|
}
|
2014-06-10 16:11:45 +00:00
|
|
|
token_info const * value_of(token_table const & s) {
|
2014-06-05 19:40:31 +00:00
|
|
|
return s.value();
|
2014-06-03 09:34:12 +00:00
|
|
|
}
|
2015-04-28 20:43:05 +00:00
|
|
|
optional<unsigned> get_expr_precedence(token_table const & s, char const * token) {
|
2014-06-15 05:13:25 +00:00
|
|
|
auto it = find(s, token);
|
2015-04-28 20:43:05 +00:00
|
|
|
return it ? optional<unsigned>(it->expr_precedence()) : optional<unsigned>();
|
|
|
|
}
|
|
|
|
optional<unsigned> get_tactic_precedence(token_table const & s, char const * token) {
|
|
|
|
auto it = find(s, token);
|
|
|
|
return it ? optional<unsigned>(it->tactic_precedence()) : optional<unsigned>();
|
2014-06-15 05:13:25 +00:00
|
|
|
}
|
2014-09-06 03:59:14 +00:00
|
|
|
bool is_token(token_table const & s, char const * token) {
|
|
|
|
return static_cast<bool>(find(s, token));
|
|
|
|
}
|
2014-06-10 16:11:45 +00:00
|
|
|
void for_each(token_table const & s, std::function<void(char const *, token_info const &)> const & fn) {
|
2014-06-09 23:49:22 +00:00
|
|
|
s.for_each([&](unsigned num, char const * keys, token_info const & info) {
|
|
|
|
buffer<char> str;
|
|
|
|
str.append(num, keys);
|
|
|
|
str.push_back(0);
|
|
|
|
fn(str.data(), info);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2014-12-04 23:11:23 +00:00
|
|
|
static char const * g_lambda_unicode = "\u03BB";
|
|
|
|
static char const * g_pi_unicode = "\u03A0";
|
|
|
|
static char const * g_forall_unicode = "\u2200";
|
|
|
|
static char const * g_arrow_unicode = "\u2192";
|
|
|
|
static char const * g_cup = "\u2294";
|
|
|
|
static char const * g_qed_unicode = "∎";
|
|
|
|
static char const * g_decreasing_unicode = "↓";
|
2014-06-12 16:08:38 +00:00
|
|
|
|
2014-09-23 17:00:36 +00:00
|
|
|
void init_token_table(token_table & t) {
|
2014-08-19 23:28:58 +00:00
|
|
|
pair<char const *, unsigned> builtin[] =
|
2015-02-25 22:30:42 +00:00
|
|
|
{{"fun", 0}, {"Pi", 0}, {"let", 0}, {"in", 0}, {"at", 0}, {"have", 0}, {"assert", 0}, {"show", 0}, {"obtain", 0},
|
2015-05-06 01:47:25 +00:00
|
|
|
{"if", 0}, {"then", 0}, {"else", 0}, {"by", 0}, {"by+", 0}, {"hiding", 0}, {"replacing", 0}, {"renaming", 0},
|
2014-06-16 17:41:08 +00:00
|
|
|
{"from", 0}, {"(", g_max_prec}, {")", 0}, {"{", g_max_prec}, {"}", 0}, {"_", g_max_prec},
|
2014-09-28 19:20:42 +00:00
|
|
|
{"[", g_max_prec}, {"]", 0}, {"⦃", g_max_prec}, {"⦄", 0}, {".{", 0}, {"Type", g_max_prec},
|
2015-02-05 18:15:58 +00:00
|
|
|
{"{|", g_max_prec}, {"|}", 0}, {"⊢", 0}, {"⟨", g_max_prec}, {"⟩", 0}, {"^", 0}, {"↑", 0}, {"▸", 0},
|
2015-04-28 20:43:05 +00:00
|
|
|
{"using", 0}, {"|", 0}, {"!", g_max_prec}, {"?", 0}, {"with", 0}, {"...", 0}, {",", 0},
|
2015-04-25 00:21:08 +00:00
|
|
|
{".", 0}, {":", 0}, {"::", 0}, {"calc", 0}, {"rewrite", 0}, {"esimp", 0}, {"fold", 0}, {"unfold", 0},
|
2015-04-30 18:57:40 +00:00
|
|
|
{"generalize", 0}, {"as", 0},
|
2015-03-26 01:22:20 +00:00
|
|
|
{":=", 0}, {"--", 0}, {"#", 0},
|
2015-05-06 01:47:25 +00:00
|
|
|
{"(*", 0}, {"/-", 0}, {"begin", g_max_prec}, {"begin+", g_max_prec}, {"proof", g_max_prec}, {"qed", 0}, {"@", g_max_prec},
|
2014-12-04 23:11:23 +00:00
|
|
|
{"sorry", g_max_prec}, {"+", g_plus_prec}, {g_cup, g_cup_prec}, {"->", g_arrow_prec},
|
2015-01-10 18:11:13 +00:00
|
|
|
{"?(", g_max_prec}, {"⌞", g_max_prec}, {"⌟", 0}, {"match", 0},
|
2015-01-26 19:31:12 +00:00
|
|
|
{"<d", g_decreasing_prec}, {"renaming", 0}, {"extends", 0}, {nullptr, 0}};
|
2014-06-05 20:05:40 +00:00
|
|
|
|
2014-09-28 19:20:42 +00:00
|
|
|
char const * commands[] =
|
2015-05-09 03:54:16 +00:00
|
|
|
{"theorem", "axiom", "axioms", "variable", "protected", "private", "reveal",
|
2015-04-20 00:45:58 +00:00
|
|
|
"definition", "example", "coercion", "abbreviation",
|
2014-10-02 23:20:52 +00:00
|
|
|
"variables", "parameter", "parameters", "constant", "constants", "[persistent]", "[visible]", "[instance]",
|
2015-03-05 22:28:10 +00:00
|
|
|
"[none]", "[class]", "[coercion]", "[reducible]", "[irreducible]", "[semireducible]", "[quasireducible]",
|
2015-05-02 22:15:35 +00:00
|
|
|
"[parsing-only]", "[multiple-instances]", "[symm]", "[trans]", "[refl]", "[subst]",
|
2015-05-09 18:49:55 +00:00
|
|
|
"evaluate", "check", "eval", "[wf]", "[whnf]", "[priority", "[unfold-f]",
|
2015-05-07 18:56:42 +00:00
|
|
|
"[constructor]", "[unfold-c", "print",
|
2015-03-06 21:03:30 +00:00
|
|
|
"end", "namespace", "section", "prelude", "help",
|
2015-01-26 19:31:12 +00:00
|
|
|
"import", "inductive", "record", "structure", "module", "universe", "universes", "local",
|
2015-04-22 18:32:02 +00:00
|
|
|
"precedence", "reserve", "infixl", "infixr", "infix", "postfix", "prefix", "notation",
|
2015-04-28 00:46:13 +00:00
|
|
|
"tactic_infixl", "tactic_infixr", "tactic_infix", "tactic_postfix", "tactic_prefix", "tactic_notation",
|
2014-10-31 06:24:09 +00:00
|
|
|
"exit", "set_option", "open", "export", "calc_subst", "calc_refl", "calc_trans", "calc_symm", "tactic_hint",
|
2015-04-20 00:45:58 +00:00
|
|
|
"add_begin_end_tactic", "set_begin_end_tactic", "instance", "class",
|
|
|
|
"multiple_instances", "find_decl", "attribute", "persistent",
|
2015-04-23 22:27:56 +00:00
|
|
|
"include", "omit", "migrate", "init_quotient", "init_hits", "#erase_cache", "#projections", "#telescope_eq", nullptr};
|
2014-06-05 20:05:40 +00:00
|
|
|
|
2014-08-19 23:28:58 +00:00
|
|
|
pair<char const *, char const *> aliases[] =
|
2014-06-11 17:56:04 +00:00
|
|
|
{{g_lambda_unicode, "fun"}, {"forall", "Pi"}, {g_forall_unicode, "Pi"}, {g_pi_unicode, "Pi"},
|
2014-09-12 01:11:58 +00:00
|
|
|
{g_qed_unicode, "qed"}, {nullptr, nullptr}};
|
2014-06-05 20:05:40 +00:00
|
|
|
|
2014-08-19 23:28:58 +00:00
|
|
|
pair<char const *, char const *> cmd_aliases[] =
|
2015-02-12 02:46:03 +00:00
|
|
|
{{"lemma", "theorem"}, {"premise", "variable"}, {"premises", "variables"},
|
|
|
|
{"corollary", "theorem"}, {"hypothesis", "parameter"}, {"conjecture", "parameter"},
|
2014-11-05 19:40:08 +00:00
|
|
|
{"record", "structure"}, {nullptr, nullptr}};
|
2014-06-05 20:05:40 +00:00
|
|
|
|
2014-06-11 17:56:04 +00:00
|
|
|
auto it = builtin;
|
2014-06-14 00:30:35 +00:00
|
|
|
while (it->first) {
|
|
|
|
t = add_token(t, it->first, it->second);
|
2014-06-11 17:56:04 +00:00
|
|
|
it++;
|
|
|
|
}
|
2014-06-05 20:05:40 +00:00
|
|
|
|
2014-06-14 00:30:35 +00:00
|
|
|
auto it2 = commands;
|
|
|
|
while (*it2) {
|
|
|
|
t = add_command_token(t, *it2);
|
|
|
|
++it2;
|
2014-06-11 17:56:04 +00:00
|
|
|
}
|
2014-06-05 20:05:40 +00:00
|
|
|
|
2014-06-14 00:30:35 +00:00
|
|
|
auto it3 = aliases;
|
|
|
|
while (it3->first) {
|
2014-10-21 01:40:55 +00:00
|
|
|
t = add_token(t, it3->first, it3->second, 0);
|
2014-06-14 00:30:35 +00:00
|
|
|
it3++;
|
2014-06-11 17:56:04 +00:00
|
|
|
}
|
2014-06-13 22:13:32 +00:00
|
|
|
t = add_token(t, g_arrow_unicode, "->", get_arrow_prec());
|
2014-12-04 23:11:23 +00:00
|
|
|
t = add_token(t, g_decreasing_unicode, "<d", get_decreasing_prec());
|
2014-06-05 20:05:40 +00:00
|
|
|
|
2014-06-14 00:30:35 +00:00
|
|
|
auto it4 = cmd_aliases;
|
|
|
|
while (it4->first) {
|
|
|
|
t = add_command_token(t, it4->first, it4->second);
|
|
|
|
++it4;
|
2014-06-03 09:34:12 +00:00
|
|
|
}
|
2014-06-11 17:56:04 +00:00
|
|
|
}
|
|
|
|
|
2014-09-23 17:00:36 +00:00
|
|
|
static token_table * g_default_token_table = nullptr;
|
|
|
|
|
2014-06-11 17:56:04 +00:00
|
|
|
token_table mk_default_token_table() {
|
2014-09-23 17:00:36 +00:00
|
|
|
return *g_default_token_table;
|
|
|
|
}
|
|
|
|
|
|
|
|
void initialize_token_table() {
|
|
|
|
g_default_token_table = new token_table();
|
|
|
|
init_token_table(*g_default_token_table);
|
2014-06-11 17:56:04 +00:00
|
|
|
}
|
|
|
|
|
2014-09-23 17:00:36 +00:00
|
|
|
void finalize_token_table() {
|
|
|
|
delete g_default_token_table;
|
|
|
|
}
|
2014-06-11 17:56:04 +00:00
|
|
|
|
2014-06-10 16:11:45 +00:00
|
|
|
token_table mk_token_table() { return token_table(); }
|
2014-06-09 23:49:22 +00:00
|
|
|
|
2014-06-10 16:11:45 +00:00
|
|
|
DECL_UDATA(token_table)
|
|
|
|
static int mk_token_table(lua_State * L) { return push_token_table(L, mk_token_table()); }
|
|
|
|
static int mk_default_token_table(lua_State * L) { return push_token_table(L, mk_default_token_table()); }
|
2014-06-09 23:49:22 +00:00
|
|
|
static int add_command_token(lua_State * L) {
|
|
|
|
int nargs = lua_gettop(L);
|
|
|
|
if (nargs == 2)
|
2014-06-10 16:11:45 +00:00
|
|
|
return push_token_table(L, add_command_token(to_token_table(L, 1), lua_tostring(L, 2)));
|
2014-06-09 23:49:22 +00:00
|
|
|
else
|
2014-06-10 16:11:45 +00:00
|
|
|
return push_token_table(L, add_command_token(to_token_table(L, 1), lua_tostring(L, 2), lua_tostring(L, 3)));
|
2014-06-09 23:49:22 +00:00
|
|
|
}
|
|
|
|
static int add_token(lua_State * L) {
|
|
|
|
int nargs = lua_gettop(L);
|
|
|
|
if (nargs == 3)
|
2014-06-10 16:11:45 +00:00
|
|
|
return push_token_table(L, add_token(to_token_table(L, 1), lua_tostring(L, 2), lua_tonumber(L, 3)));
|
2014-06-09 23:49:22 +00:00
|
|
|
else
|
2014-06-10 16:11:45 +00:00
|
|
|
return push_token_table(L, add_token(to_token_table(L, 1), lua_tostring(L, 2), lua_tostring(L, 3), lua_tonumber(L, 4)));
|
2014-06-09 23:49:22 +00:00
|
|
|
}
|
|
|
|
static int merge(lua_State * L) {
|
2014-06-10 16:11:45 +00:00
|
|
|
return push_token_table(L, merge(to_token_table(L, 1), to_token_table(L, 2)));
|
2014-06-09 23:49:22 +00:00
|
|
|
}
|
|
|
|
static int find(lua_State * L) {
|
|
|
|
char k;
|
|
|
|
if (lua_isnumber(L, 2)) {
|
|
|
|
k = lua_tonumber(L, 2);
|
|
|
|
} else {
|
|
|
|
char const * str = lua_tostring(L, 2);
|
|
|
|
if (strlen(str) != 1)
|
|
|
|
throw exception("arg #2 must be a string of length 1");
|
|
|
|
k = str[0];
|
|
|
|
}
|
2014-06-10 16:11:45 +00:00
|
|
|
auto it = to_token_table(L, 1).find(k);
|
2014-06-09 23:49:22 +00:00
|
|
|
if (it)
|
2014-06-10 16:11:45 +00:00
|
|
|
return push_token_table(L, *it);
|
2014-06-09 23:49:22 +00:00
|
|
|
else
|
|
|
|
return push_nil(L);
|
|
|
|
}
|
|
|
|
static int value_of(lua_State * L) {
|
2014-06-10 16:11:45 +00:00
|
|
|
auto it = value_of(to_token_table(L, 1));
|
2014-06-09 23:49:22 +00:00
|
|
|
if (it) {
|
|
|
|
push_boolean(L, it->is_command());
|
|
|
|
push_name(L, it->value());
|
2015-04-28 20:43:05 +00:00
|
|
|
push_integer(L, it->expr_precedence());
|
2014-06-09 23:49:22 +00:00
|
|
|
return 3;
|
|
|
|
} else {
|
|
|
|
push_nil(L);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
static int for_each(lua_State * L) {
|
2014-06-10 16:11:45 +00:00
|
|
|
token_table const & t = to_token_table(L, 1);
|
2014-06-09 23:49:22 +00:00
|
|
|
luaL_checktype(L, 2, LUA_TFUNCTION); // user-fun
|
|
|
|
for_each(t, [&](char const * k, token_info const & info) {
|
|
|
|
lua_pushvalue(L, 2);
|
|
|
|
lua_pushstring(L, k);
|
|
|
|
lua_pushboolean(L, info.is_command());
|
|
|
|
push_name(L, info.value());
|
2015-04-28 20:43:05 +00:00
|
|
|
lua_pushinteger(L, info.expr_precedence());
|
2014-06-09 23:49:22 +00:00
|
|
|
pcall(L, 4, 0, 0);
|
|
|
|
});
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-06-10 16:11:45 +00:00
|
|
|
static const struct luaL_Reg token_table_m[] = {
|
|
|
|
{"__gc", token_table_gc},
|
2014-06-09 23:49:22 +00:00
|
|
|
{"add_command_token", safe_function<add_command_token>},
|
|
|
|
{"add_token", safe_function<add_token>},
|
|
|
|
{"merge", safe_function<merge>},
|
|
|
|
{"find", safe_function<find>},
|
|
|
|
{"value_of", safe_function<value_of>},
|
|
|
|
{"for_each", safe_function<for_each>},
|
|
|
|
{0, 0}
|
|
|
|
};
|
|
|
|
|
2014-06-10 16:11:45 +00:00
|
|
|
void open_token_table(lua_State * L) {
|
|
|
|
luaL_newmetatable(L, token_table_mt);
|
2014-06-09 23:49:22 +00:00
|
|
|
lua_pushvalue(L, -1);
|
|
|
|
lua_setfield(L, -2, "__index");
|
2014-06-10 16:11:45 +00:00
|
|
|
setfuncs(L, token_table_m, 0);
|
2014-06-09 23:49:22 +00:00
|
|
|
|
2014-06-10 16:11:45 +00:00
|
|
|
SET_GLOBAL_FUN(token_table_pred, "is_token_table");
|
|
|
|
SET_GLOBAL_FUN(mk_default_token_table, "default_token_table");
|
|
|
|
SET_GLOBAL_FUN(mk_token_table, "token_table");
|
2014-06-09 23:49:22 +00:00
|
|
|
}
|
2014-06-03 09:34:12 +00:00
|
|
|
}
|