refactor(frontends/lean/scanner): use the parser configuration in the environment

Signed-off-by: Leonardo de Moura <leonardo@microsoft.com>
This commit is contained in:
Leonardo de Moura 2014-06-10 10:59:12 -07:00
parent e2adb101d5
commit d3e3301208
4 changed files with 78 additions and 45 deletions

View file

@ -12,6 +12,9 @@ struct parser_ext : public environment_extension {
token_table m_tokens;
parse_table m_nud;
parse_table m_led;
parser_ext() {
m_tokens = mk_default_token_table();
}
};
struct parser_ext_reg {

View file

@ -8,6 +8,7 @@ Author: Leonardo de Moura
#include <string>
#include "util/exception.h"
#include "frontends/lean/scanner.h"
#include "frontends/lean/parser_config.h"
namespace lean {
void scanner::next() {
@ -211,7 +212,7 @@ auto scanner::read_key_cmd_id() -> token_kind {
static char const * error_msg = "unexpected token";
char c = curr();
unsigned num_cs = 1; // number of characters read
token_table const * it = find(m_tokens, c);
token_table const * it = find(*m_tokens, c);
token_info const * info = nullptr;
unsigned key_size = 0;
if (it) {
@ -278,7 +279,8 @@ static name g_begin_script_tk("(*");
static name g_begin_comment_tk("--");
static name g_begin_comment_block_tk("(--");
auto scanner::scan() -> token_kind {
auto scanner::scan(environment const & env) -> token_kind {
m_tokens = &get_token_table(env);
while (true) {
char c = curr();
m_pos = m_spos;
@ -318,8 +320,8 @@ auto scanner::scan() -> token_kind {
}
}
scanner::scanner(token_table const & tks, std::istream & strm, char const * strm_name):
m_tokens(tks), m_stream(strm), m_spos(0), m_sline(1), m_curr(0), m_pos(0), m_line(1),
scanner::scanner(std::istream & strm, char const * strm_name):
m_tokens(nullptr), m_stream(strm), m_spos(0), m_sline(1), m_curr(0), m_pos(0), m_line(1),
m_token_info(nullptr) {
m_stream_name = strm_name ? strm_name : "[unknown]";
next();

View file

@ -8,8 +8,10 @@ Author: Leonardo de Moura
#include <iostream>
#include "util/name.h"
#include "util/numerics/mpq.h"
#include "kernel/environment.h"
#include "frontends/lean/token_table.h"
namespace lean {
/**
\brief Scanner. The behavior of the scanner is controlled using a token set.
@ -22,7 +24,7 @@ class scanner {
public:
enum class token_kind {Keyword, CommandKeyword, ScriptBlock, Identifier, Numeral, Decimal, String, Eof};
protected:
token_table m_tokens;
token_table const * m_tokens;
std::istream & m_stream;
std::string m_stream_name;
@ -60,11 +62,11 @@ protected:
token_kind read_key_cmd_id();
public:
scanner(token_table const & tks, std::istream & strm, char const * strm_name = nullptr);
scanner(std::istream & strm, char const * strm_name = nullptr);
int get_line() const { return m_line; }
int get_pos() const { return m_pos; }
token_kind scan();
token_kind scan(environment const & env);
mpq const & get_num_val() const { return m_num_val; }
name const & get_name_val() const { return m_name_val; }

View file

@ -9,15 +9,16 @@ Author: Leonardo de Moura
#include "util/escaped.h"
#include "util/exception.h"
#include "frontends/lean/scanner.h"
#include "frontends/lean/parser_config.h"
using namespace lean;
#define tk scanner::token_kind
static void scan(char const * str, token_table set = mk_default_token_table()) {
static void scan(char const * str, environment const & env = environment()) {
std::istringstream in(str);
scanner s(set, in, "[string]");
scanner s(in, "[string]");
while (true) {
tk k = s.scan();
tk k = s.scan(env);
if (k == tk::Eof)
break;
if (k == tk::Identifier)
@ -35,9 +36,9 @@ static void scan(char const * str, token_table set = mk_default_token_table()) {
std::cout << "\n";
}
static void scan_success(char const * str, token_table set = mk_default_token_table()) {
static void scan_success(char const * str, environment const & env = environment()) {
try {
scan(str, set);
scan(str, env);
} catch (exception & ex) {
std::cout << "ERROR: " << ex.what() << "\n";
lean_unreachable();
@ -54,13 +55,13 @@ static void scan_error(char const * str) {
}
static void check(char const * str, std::initializer_list<tk> const & l,
token_table set = mk_default_token_table()) {
environment const & env = environment()) {
try {
auto it = l.begin();
std::istringstream in(str);
scanner s(set, in, "[string]");
scanner s(in, "[string]");
while (true) {
tk k = s.scan();
tk k = s.scan(env);
if (k == tk::Eof) {
lean_assert(it == l.end());
return;
@ -75,37 +76,40 @@ static void check(char const * str, std::initializer_list<tk> const & l,
}
}
static void check_name(char const * str, name const & expected, token_table set = mk_default_token_table()) {
static void check_name(char const * str, name const & expected, environment const & env = environment()) {
std::istringstream in(str);
scanner s(set, in, "[string]");
tk k = s.scan();
scanner s(in, "[string]");
tk k = s.scan(env);
lean_assert(k == tk::Identifier);
lean_assert(s.get_name_val() == expected);
lean_assert(s.scan() == tk::Eof);
lean_assert(s.scan(env) == tk::Eof);
}
static void check_keyword(char const * str, name const & expected, token_table set = mk_default_token_table()) {
static void check_keyword(char const * str, name const & expected, environment const & env = environment()) {
std::istringstream in(str);
scanner s(set, in, "[string]");
tk k = s.scan();
scanner s(in, "[string]");
tk k = s.scan(env);
lean_assert(k == tk::Keyword);
lean_assert(s.get_token_info().value() == expected);
lean_assert(s.scan() == tk::Eof);
lean_assert(s.scan(env) == tk::Eof);
}
static void tst1() {
token_table s = mk_default_token_table();
environment env;
token_table s = get_token_table(env);
s = add_token(s, "+", "plus");
s = add_token(s, "=", "eq");
env = update_token_table(env, s);
scan_success("a..a");
check("a..a", {tk::Identifier, tk::Keyword, tk::Keyword, tk::Identifier});
check("Type.{0}", {tk::Keyword, tk::Keyword, tk::Numeral, tk::Keyword});
s = add_token(s, "ab+cde", "weird1");
s = add_token(s, "+cd", "weird2");
scan_success("ab+cd", s);
check("ab+cd", {tk::Identifier, tk::Keyword}, s);
scan_success("ab+cde", s);
check("ab+cde", {tk::Keyword}, s);
env = update_token_table(env, s);
scan_success("ab+cd", env);
check("ab+cd", {tk::Identifier, tk::Keyword}, env);
scan_success("ab+cde", env);
check("ab+cde", {tk::Keyword}, env);
scan_success("Type.{0}");
scan_success("0.a a");
scan_success("0.");
@ -115,11 +119,11 @@ static void tst1() {
scan_success("..");
scan_success("....");
scan_success("....\n..");
scan_success("a", s);
scan_success("a", env);
scan_success("a. b.c..");
scan_success(".. ..");
scan_success("....\n..");
scan_success("fun(x: forall A : Type, A -> A), x+1 = 2.0 λvalue.foo. . . a", s);
scan_success("fun(x: forall A : Type, A -> A), x+1 = 2.0 λvalue.foo. . . a", env);
}
static void tst2() {
@ -133,11 +137,14 @@ static void tst2() {
check_name("x.bla", name({"x", "bla"}));
scan_error("+++");
environment env;
token_table s = mk_default_token_table();
s = add_token(s, "+++", "tplus");
check_keyword("+++", "tplus", s);
env = update_token_table(env, s);
check_keyword("+++", "tplus", env);
s = add_token(s, "+", "plus");
check("x+y", {tk::Identifier, tk::Keyword, tk::Identifier}, s);
env = update_token_table(env, s);
check("x+y", {tk::Identifier, tk::Keyword, tk::Identifier}, env);
check("-- testing", {});
check("(-- testing --)", {});
check("(-- (-- testing\n --) --)", {});
@ -148,7 +155,8 @@ static void tst2() {
check("int->int", {tk::Identifier, tk::Keyword, tk::Identifier});
check_keyword("->", "->");
s = add_token(s, "-+->", "arrow");
check("Int -+-> Int", {tk::Identifier, tk::Keyword, tk::Identifier}, s);
env = update_token_table(env, s);
check("Int -+-> Int", {tk::Identifier, tk::Keyword, tk::Identifier}, env);
check("x := 10", {tk::Identifier, tk::Keyword, tk::Numeral});
check("{x}", {tk::Keyword, tk::Identifier, tk::Keyword});
check("\u03BB \u2200 \u2192", {tk::Keyword, tk::Keyword, tk::Keyword});
@ -174,10 +182,28 @@ static void tst3() {
scan("{ } . forall exists let in \u2200 := _");
}
static void tst4(unsigned N) {
std::string big;
for (unsigned i = 0; i < N; i++)
big += "aaa ";
std::istringstream in(big);
environment env;
scanner s(in, "[string]");
unsigned i = 0;
while (true) {
tk k = s.scan(env);
if (k == tk::Eof)
break;
i++;
}
std::cout << i << "\n";
}
int main() {
save_stack_info();
tst1();
tst2();
tst3();
tst4(100000);
return has_violations() ? 1 : 0;
}