feat(frontends/lean/token_set): add token_set Lua API
Signed-off-by: Leonardo de Moura <leonardo@microsoft.com>
This commit is contained in:
parent
25640faaeb
commit
00e0cc15ba
4 changed files with 145 additions and 4 deletions
|
@ -7,10 +7,12 @@ Author: Leonardo de Moura
|
||||||
#include <sstream>
|
#include <sstream>
|
||||||
#include "util/lua.h"
|
#include "util/lua.h"
|
||||||
#include "util/script_state.h"
|
#include "util/script_state.h"
|
||||||
|
#include "frontends/lean/token_set.h"
|
||||||
#include "frontends/lean/parse_table.h"
|
#include "frontends/lean/parse_table.h"
|
||||||
|
|
||||||
namespace lean {
|
namespace lean {
|
||||||
void open_frontend_lean(lua_State * L) {
|
void open_frontend_lean(lua_State * L) {
|
||||||
|
open_token_set(L);
|
||||||
open_parse_table(L);
|
open_parse_table(L);
|
||||||
}
|
}
|
||||||
void register_frontend_lean_module() {
|
void register_frontend_lean_module() {
|
||||||
|
|
|
@ -20,15 +20,21 @@ token_set add_token(token_set const & s, char const * token, unsigned prec) {
|
||||||
token_set add_token(token_set const & s, char const * token, char const * val, unsigned prec) {
|
token_set add_token(token_set const & s, char const * token, char const * val, unsigned prec) {
|
||||||
return insert(s, token, token_info(val, prec));
|
return insert(s, token, token_info(val, prec));
|
||||||
}
|
}
|
||||||
token_set merge(token_set const & s1, token_set const & s2) {
|
|
||||||
return merge(s1, s2);
|
|
||||||
}
|
|
||||||
token_set const * find(token_set const & s, char c) {
|
token_set const * find(token_set const & s, char c) {
|
||||||
return s.find(c);
|
return s.find(c);
|
||||||
}
|
}
|
||||||
token_info const * value_of(token_set const & s) {
|
token_info const * value_of(token_set const & s) {
|
||||||
return s.value();
|
return s.value();
|
||||||
}
|
}
|
||||||
|
void for_each(token_set const & s, std::function<void(char const *, token_info const &)> const & fn) {
|
||||||
|
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);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
static char const * g_lambda_unicode = "\u03BB";
|
static char const * g_lambda_unicode = "\u03BB";
|
||||||
static char const * g_pi_unicode = "\u03A0";
|
static char const * g_pi_unicode = "\u03A0";
|
||||||
static char const * g_forall_unicode = "\u2200";
|
static char const * g_forall_unicode = "\u2200";
|
||||||
|
@ -82,5 +88,90 @@ public:
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
static init_token_set_fn g_init;
|
static init_token_set_fn g_init;
|
||||||
|
token_set mk_token_set() { return token_set(); }
|
||||||
token_set mk_default_token_set() { return g_init.m_token_set; }
|
token_set mk_default_token_set() { return g_init.m_token_set; }
|
||||||
|
|
||||||
|
DECL_UDATA(token_set)
|
||||||
|
static int mk_token_set(lua_State * L) { return push_token_set(L, mk_token_set()); }
|
||||||
|
static int mk_default_token_set(lua_State * L) { return push_token_set(L, mk_default_token_set()); }
|
||||||
|
static int add_command_token(lua_State * L) {
|
||||||
|
int nargs = lua_gettop(L);
|
||||||
|
if (nargs == 2)
|
||||||
|
return push_token_set(L, add_command_token(to_token_set(L, 1), lua_tostring(L, 2)));
|
||||||
|
else
|
||||||
|
return push_token_set(L, add_command_token(to_token_set(L, 1), lua_tostring(L, 2), lua_tostring(L, 3)));
|
||||||
|
}
|
||||||
|
static int add_token(lua_State * L) {
|
||||||
|
int nargs = lua_gettop(L);
|
||||||
|
if (nargs == 3)
|
||||||
|
return push_token_set(L, add_token(to_token_set(L, 1), lua_tostring(L, 2), lua_tonumber(L, 3)));
|
||||||
|
else
|
||||||
|
return push_token_set(L, add_token(to_token_set(L, 1), lua_tostring(L, 2), lua_tostring(L, 3), lua_tonumber(L, 4)));
|
||||||
|
}
|
||||||
|
static int merge(lua_State * L) {
|
||||||
|
return push_token_set(L, merge(to_token_set(L, 1), to_token_set(L, 2)));
|
||||||
|
}
|
||||||
|
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];
|
||||||
|
}
|
||||||
|
auto it = to_token_set(L, 1).find(k);
|
||||||
|
if (it)
|
||||||
|
return push_token_set(L, *it);
|
||||||
|
else
|
||||||
|
return push_nil(L);
|
||||||
|
}
|
||||||
|
static int value_of(lua_State * L) {
|
||||||
|
auto it = value_of(to_token_set(L, 1));
|
||||||
|
if (it) {
|
||||||
|
push_boolean(L, it->is_command());
|
||||||
|
push_name(L, it->value());
|
||||||
|
push_integer(L, it->precedence());
|
||||||
|
return 3;
|
||||||
|
} else {
|
||||||
|
push_nil(L);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
static int for_each(lua_State * L) {
|
||||||
|
token_set const & t = to_token_set(L, 1);
|
||||||
|
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());
|
||||||
|
lua_pushinteger(L, info.precedence());
|
||||||
|
pcall(L, 4, 0, 0);
|
||||||
|
});
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static const struct luaL_Reg token_set_m[] = {
|
||||||
|
{"__gc", token_set_gc},
|
||||||
|
{"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}
|
||||||
|
};
|
||||||
|
|
||||||
|
void open_token_set(lua_State * L) {
|
||||||
|
luaL_newmetatable(L, token_set_mt);
|
||||||
|
lua_pushvalue(L, -1);
|
||||||
|
lua_setfield(L, -2, "__index");
|
||||||
|
setfuncs(L, token_set_m, 0);
|
||||||
|
|
||||||
|
SET_GLOBAL_FUN(token_set_pred, "is_token_set");
|
||||||
|
SET_GLOBAL_FUN(mk_default_token_set, "default_token_set");
|
||||||
|
SET_GLOBAL_FUN(mk_token_set, "token_set");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -9,6 +9,7 @@ Author: Leonardo de Moura
|
||||||
#include <string>
|
#include <string>
|
||||||
#include "util/trie.h"
|
#include "util/trie.h"
|
||||||
#include "util/name.h"
|
#include "util/name.h"
|
||||||
|
#include "util/lua.h"
|
||||||
|
|
||||||
namespace lean {
|
namespace lean {
|
||||||
class token_info {
|
class token_info {
|
||||||
|
@ -25,12 +26,14 @@ public:
|
||||||
};
|
};
|
||||||
|
|
||||||
typedef ctrie<token_info> token_set;
|
typedef ctrie<token_info> token_set;
|
||||||
|
token_set mk_token_set();
|
||||||
token_set mk_default_token_set();
|
token_set mk_default_token_set();
|
||||||
token_set add_command_token(token_set const & s, char const * token);
|
token_set add_command_token(token_set const & s, char const * token);
|
||||||
token_set add_command_token(token_set const & s, char const * token, char const * val);
|
token_set add_command_token(token_set const & s, char const * token, char const * val);
|
||||||
token_set add_token(token_set const & s, char const * token, unsigned prec = 0);
|
token_set add_token(token_set const & s, char const * token, unsigned prec = 0);
|
||||||
token_set add_token(token_set const & s, char const * token, char const * val, unsigned prec = 0);
|
token_set add_token(token_set const & s, char const * token, char const * val, unsigned prec = 0);
|
||||||
token_set merge(token_set const & s1, token_set const & s2);
|
void for_each(token_set const & s, std::function<void(char const *, token_info const&)> const & fn);
|
||||||
token_set const * find(token_set const & s, char c);
|
token_set const * find(token_set const & s, char c);
|
||||||
token_info const * value_of(token_set const & s);
|
token_info const * value_of(token_set const & s);
|
||||||
|
void open_token_set(lua_State * L);
|
||||||
}
|
}
|
||||||
|
|
45
tests/lua/token_set.lua
Normal file
45
tests/lua/token_set.lua
Normal file
|
@ -0,0 +1,45 @@
|
||||||
|
function display_token_set(s)
|
||||||
|
s:for_each(function(k, cmd, val, prec)
|
||||||
|
io.write(k)
|
||||||
|
if cmd then
|
||||||
|
io.write(" [command]")
|
||||||
|
end
|
||||||
|
print(" => " .. tostring(val) .. " " .. tostring(prec))
|
||||||
|
end)
|
||||||
|
end
|
||||||
|
|
||||||
|
function token_set_size(s)
|
||||||
|
local r = 0
|
||||||
|
s:for_each(function() r = r + 1 end)
|
||||||
|
return r
|
||||||
|
end
|
||||||
|
|
||||||
|
local s = token_set()
|
||||||
|
assert(is_token_set(s))
|
||||||
|
assert(token_set_size(s) == 0)
|
||||||
|
s = s:add_command_token("test", "tst1")
|
||||||
|
s = s:add_command_token("tast", "tst2")
|
||||||
|
s = s:add_command_token("tests", "tst3")
|
||||||
|
s = s:add_command_token("fests", "tst4")
|
||||||
|
s = s:add_command_token("tes", "tst5")
|
||||||
|
s = s:add_token("++", "++", 65)
|
||||||
|
s = s:add_token("++-", "plusminus")
|
||||||
|
assert(token_set_size(s) == 7)
|
||||||
|
display_token_set(s)
|
||||||
|
|
||||||
|
|
||||||
|
print("========")
|
||||||
|
local s2 = default_token_set()
|
||||||
|
display_token_set(s2)
|
||||||
|
assert(token_set_size(s2) > 0)
|
||||||
|
local sz1 = token_set_size(s)
|
||||||
|
local sz2 = token_set_size(s2)
|
||||||
|
s2 = s2:merge(s)
|
||||||
|
assert(token_set_size(s2) == sz1 + sz2)
|
||||||
|
s2 = s2:find("t"):find("e")
|
||||||
|
print("========")
|
||||||
|
display_token_set(s2)
|
||||||
|
assert(token_set_size(s2) == 3)
|
||||||
|
s2 = s2:find("s")
|
||||||
|
local cmd, val, prec = s2:value_of()
|
||||||
|
assert(val == name("tst5"))
|
Loading…
Reference in a new issue