Add interrupt to parser. Add elaborator to parser. Add placeholder support in the scanner.
Signed-off-by: Leonardo de Moura <leonardo@microsoft.com>
This commit is contained in:
parent
02b72acc2f
commit
ece6e6ca6a
9 changed files with 257 additions and 81 deletions
|
@ -18,6 +18,7 @@ Author: Leonardo de Moura
|
|||
#include "expr_maps.h"
|
||||
#include "sstream.h"
|
||||
#include "kernel_exception.h"
|
||||
#include "elaborator.h"
|
||||
#include "lean_frontend.h"
|
||||
#include "lean_parser.h"
|
||||
#include "lean_scanner.h"
|
||||
|
@ -31,7 +32,27 @@ Author: Leonardo de Moura
|
|||
#include <readline/history.h>
|
||||
#endif
|
||||
|
||||
#ifndef LEAN_DEFAULT_PARSER_SHOW_ERRORS
|
||||
#define LEAN_DEFAULT_PARSER_SHOW_ERRORS true
|
||||
#endif
|
||||
|
||||
#ifndef LEAN_DEFAULT_PARSER_VERBOSE
|
||||
#define LEAN_DEFAULT_PARSER_VERBOSE true
|
||||
#endif
|
||||
|
||||
namespace lean {
|
||||
// ==========================================
|
||||
// Parser configuration options
|
||||
static name g_parser_verbose {"lean", "parser", "verbose"};
|
||||
static name g_parser_show_errors {"lean", "parser", "show_errors"};
|
||||
|
||||
RegisterBoolOption(g_parser_verbose, LEAN_DEFAULT_PARSER_VERBOSE, "(lean parser) disable/enable parser verbose messages");
|
||||
RegisterBoolOption(g_parser_show_errors, LEAN_DEFAULT_PARSER_SHOW_ERRORS, "(lean parser) display error messages in the regular output channel");
|
||||
|
||||
bool get_parser_verbose(options const & opts) { return opts.get_bool(g_parser_verbose, LEAN_DEFAULT_PARSER_VERBOSE); }
|
||||
bool get_parser_show_errors(options const & opts) { return opts.get_bool(g_parser_show_errors, LEAN_DEFAULT_PARSER_SHOW_ERRORS); }
|
||||
// ==========================================
|
||||
|
||||
// ==========================================
|
||||
// Builtin commands
|
||||
static name g_definition_kwd("Definition");
|
||||
|
@ -72,19 +93,13 @@ static unsigned g_level_plus_prec = 10;
|
|||
static unsigned g_level_cup_prec = 5;
|
||||
// ==========================================
|
||||
|
||||
// ==========================================
|
||||
// Auxiliary constant used to mark applications of overloaded operators.
|
||||
static name g_overload_name(name(name(name(0u), "parser"), "overload"));
|
||||
static expr g_overload = mk_constant(g_overload_name);
|
||||
// ==========================================
|
||||
|
||||
// A name that can't be created by the user.
|
||||
// It is used as placeholder for parsing A -> B expressions which
|
||||
// are syntax sugar for (Pi (_ : A), B)
|
||||
static name g_unused(name(0u), "parser");
|
||||
|
||||
/**
|
||||
\brief Functional object for parsing
|
||||
\brief Actual implementation for the parser functional object
|
||||
|
||||
\remark It is an instance of a Pratt parser
|
||||
(http://en.wikipedia.org/wiki/Pratt_parser) described in the paper
|
||||
|
@ -92,13 +107,14 @@ static name g_unused(name(0u), "parser");
|
|||
and it is easy to support user-defined infix/prefix/postfix/mixfix
|
||||
operators.
|
||||
*/
|
||||
class parser_fn {
|
||||
class parser::imp {
|
||||
typedef scoped_map<name, unsigned, name_hash, name_eq> local_decls;
|
||||
typedef std::unordered_map<name, expr, name_hash, name_eq> builtins;
|
||||
typedef std::pair<unsigned, unsigned> pos_info;
|
||||
typedef expr_map<pos_info> expr_pos_info;
|
||||
frontend m_frontend;
|
||||
scanner m_scanner;
|
||||
elaborator m_elaborator;
|
||||
scanner::token m_curr;
|
||||
bool m_use_exceptions;
|
||||
bool m_interactive;
|
||||
|
@ -108,6 +124,12 @@ class parser_fn {
|
|||
builtins m_builtins;
|
||||
expr_pos_info m_expr_pos_info;
|
||||
pos_info m_last_cmd_pos;
|
||||
// Reference to temporary parser used to process import command.
|
||||
// We need this reference to be able to interrupt it.
|
||||
interruptable_ptr<parser> m_import_parser;
|
||||
|
||||
bool m_verbose;
|
||||
bool m_show_errors;
|
||||
|
||||
/** \brief Exception used to track parsing erros, it does not leak outside of this class. */
|
||||
struct parser_error : public exception {
|
||||
|
@ -121,10 +143,10 @@ class parser_fn {
|
|||
local declarations.
|
||||
*/
|
||||
struct mk_scope {
|
||||
parser_fn & m_fn;
|
||||
imp & m_fn;
|
||||
local_decls::mk_scope m_scope;
|
||||
unsigned m_old_num_local_decls;
|
||||
mk_scope(parser_fn & fn):m_fn(fn), m_scope(fn.m_local_decls), m_old_num_local_decls(fn.m_num_local_decls) {}
|
||||
mk_scope(imp & fn):m_fn(fn), m_scope(fn.m_local_decls), m_old_num_local_decls(fn.m_num_local_decls) {}
|
||||
~mk_scope() { m_fn.m_num_local_decls = m_old_num_local_decls; }
|
||||
};
|
||||
|
||||
|
@ -149,11 +171,6 @@ class parser_fn {
|
|||
scanner::token curr() const { return m_curr; }
|
||||
/** \brief Read the next token if the current one is not End-of-file. */
|
||||
void next() { if (m_curr != scanner::token::Eof) scan(); }
|
||||
/** \brief Keep consume tokens until we find a Command or End-of-file. */
|
||||
void sync() {
|
||||
while (curr() != scanner::token::CommandId && curr() != scanner::token::Eof)
|
||||
next();
|
||||
}
|
||||
|
||||
/** \brief Return the name associated with the current token. */
|
||||
name const & curr_name() const { return m_scanner.get_name_val(); }
|
||||
|
@ -364,7 +381,7 @@ class parser_fn {
|
|||
expr r = *it;
|
||||
++it;
|
||||
for (; it != fs.end(); ++it)
|
||||
r = mk_app(g_overload, *it, r);
|
||||
r = mk_app(mk_overload_marker(), *it, r);
|
||||
return r;
|
||||
}
|
||||
|
||||
|
@ -767,6 +784,19 @@ class parser_fn {
|
|||
}
|
||||
}
|
||||
|
||||
/** \brief Create a fresh metavariable. */
|
||||
expr mk_metavar() {
|
||||
// TODO
|
||||
return expr();
|
||||
}
|
||||
|
||||
/** \brief Parse \c _ a hole that must be filled by the elaborator. */
|
||||
expr parse_placeholder() {
|
||||
auto p = pos();
|
||||
next();
|
||||
return save(mk_metavar(), p);
|
||||
}
|
||||
|
||||
/**
|
||||
\brief Auxiliary method used when processing the beginning of an expression.
|
||||
*/
|
||||
|
@ -782,6 +812,7 @@ class parser_fn {
|
|||
case scanner::token::IntVal: return parse_int();
|
||||
case scanner::token::DecimalVal: return parse_decimal();
|
||||
case scanner::token::StringVal: return parse_string();
|
||||
case scanner::token::Placeholder: return parse_placeholder();
|
||||
case scanner::token::Type: return parse_type();
|
||||
default:
|
||||
throw parser_error("invalid expression, unexpected token", pos());
|
||||
|
@ -800,6 +831,7 @@ class parser_fn {
|
|||
case scanner::token::IntVal: return mk_app(left, parse_int());
|
||||
case scanner::token::DecimalVal: return mk_app(left, parse_decimal());
|
||||
case scanner::token::StringVal: return mk_app(left, parse_string());
|
||||
case scanner::token::Placeholder: return mk_app(left, parse_placeholder());
|
||||
case scanner::token::Type: return mk_app(left, parse_type());
|
||||
default: return left;
|
||||
}
|
||||
|
@ -873,9 +905,11 @@ class parser_fn {
|
|||
}
|
||||
if (is_definition) {
|
||||
m_frontend.add_definition(id, type, val);
|
||||
if (m_verbose)
|
||||
regular(m_frontend) << " Defined: " << id << endl;
|
||||
} else {
|
||||
m_frontend.add_theorem(id, type, val);
|
||||
if (m_verbose)
|
||||
regular(m_frontend) << " Proved: " << id << endl;
|
||||
}
|
||||
}
|
||||
|
@ -909,6 +943,7 @@ class parser_fn {
|
|||
check_colon_next("invalid variable declaration, ':' expected");
|
||||
expr type = elaborate(parse_expr());
|
||||
m_frontend.add_var(id, type);
|
||||
if (m_verbose)
|
||||
regular(m_frontend) << " Assumed: " << id << endl;
|
||||
}
|
||||
|
||||
|
@ -919,6 +954,7 @@ class parser_fn {
|
|||
check_colon_next("invalid axiom, ':' expected");
|
||||
expr type = elaborate(parse_expr());
|
||||
m_frontend.add_axiom(id, type);
|
||||
if (m_verbose)
|
||||
regular(m_frontend) << " Assumed: " << id << endl;
|
||||
}
|
||||
|
||||
|
@ -1080,6 +1116,8 @@ class parser_fn {
|
|||
default:
|
||||
throw parser_error("invalid option value, 'true', 'false', string, integer or decimal value expected", pos());
|
||||
}
|
||||
updt_options();
|
||||
if (m_verbose)
|
||||
regular(m_frontend) << " Set option: " << id << endl;
|
||||
next();
|
||||
}
|
||||
|
@ -1090,7 +1128,17 @@ class parser_fn {
|
|||
std::ifstream in(fname);
|
||||
if (!in.is_open())
|
||||
throw parser_error(sstream() << "invalid import command, failed to open file '" << fname << "'", m_last_cmd_pos);
|
||||
::lean::parse_commands(m_frontend, in, m_use_exceptions, false);
|
||||
try {
|
||||
if (m_verbose)
|
||||
regular(m_frontend) << "Importing file '" << fname << "'" << endl;
|
||||
parser import_parser(m_frontend, in, true /* use exceptions */, false /* not interactive */);
|
||||
scoped_set_interruptable_ptr<parser> set(m_import_parser, &import_parser);
|
||||
import_parser();
|
||||
} catch (interrupted &) {
|
||||
throw;
|
||||
} catch (exception &) {
|
||||
throw parser_error(sstream() << "failed to import file '" << fname << "'", m_last_cmd_pos);
|
||||
}
|
||||
}
|
||||
|
||||
void parse_help() {
|
||||
|
@ -1132,7 +1180,9 @@ class parser_fn {
|
|||
|
||||
/** \brief Parse a Lean command. */
|
||||
void parse_command() {
|
||||
m_frontend.reset_interrupt();
|
||||
if (m_interactive && !m_use_exceptions)
|
||||
reset_interrupt();
|
||||
m_elaborator.clear();
|
||||
m_expr_pos_info.clear();
|
||||
m_last_cmd_pos = pos();
|
||||
name const & cmd_id = curr_name();
|
||||
|
@ -1186,12 +1236,26 @@ class parser_fn {
|
|||
sync();
|
||||
}
|
||||
|
||||
void updt_options() {
|
||||
m_verbose = get_parser_verbose(m_frontend.get_state().get_options());
|
||||
m_show_errors = get_parser_show_errors(m_frontend.get_state().get_options());
|
||||
}
|
||||
|
||||
/** \brief Keep consuming tokens until we find a Command or End-of-file. */
|
||||
void sync() {
|
||||
show_prompt();
|
||||
while (curr() != scanner::token::CommandId && curr() != scanner::token::Eof)
|
||||
next();
|
||||
}
|
||||
|
||||
public:
|
||||
parser_fn(frontend & fe, std::istream & in, bool use_exceptions, bool interactive):
|
||||
imp(frontend & fe, std::istream & in, bool use_exceptions, bool interactive):
|
||||
m_frontend(fe),
|
||||
m_scanner(in),
|
||||
m_elaborator(fe),
|
||||
m_use_exceptions(use_exceptions),
|
||||
m_interactive(interactive) {
|
||||
updt_options();
|
||||
m_found_errors = false;
|
||||
m_num_local_decls = 0;
|
||||
m_scanner.set_command_keywords(g_command_keywords);
|
||||
|
@ -1223,32 +1287,29 @@ public:
|
|||
}
|
||||
} catch (parser_error & ex) {
|
||||
m_found_errors = true;
|
||||
if (m_show_errors)
|
||||
display_error(ex.what(), ex.m_pos.first, ex.m_pos.second);
|
||||
if (m_use_exceptions) {
|
||||
throw parser_exception(ex.what(), ex.m_pos.first, ex.m_pos.second);
|
||||
} else {
|
||||
display_error(ex.what(), ex.m_pos.first, ex.m_pos.second);
|
||||
}
|
||||
} catch (kernel_exception & ex) {
|
||||
m_found_errors = true;
|
||||
if (m_use_exceptions) {
|
||||
throw;
|
||||
} else {
|
||||
if (m_show_errors)
|
||||
display_error(ex);
|
||||
}
|
||||
} catch (interrupted & ex) {
|
||||
if (m_use_exceptions) {
|
||||
if (m_use_exceptions)
|
||||
throw;
|
||||
} else {
|
||||
} catch (interrupted & ex) {
|
||||
if (m_verbose)
|
||||
regular(m_frontend) << "\n!!!Interrupted!!!" << endl;
|
||||
sync();
|
||||
}
|
||||
if (m_use_exceptions)
|
||||
throw;
|
||||
} catch (exception & ex) {
|
||||
m_found_errors = true;
|
||||
if (m_use_exceptions) {
|
||||
throw;
|
||||
} else {
|
||||
if (m_show_errors)
|
||||
display_error(ex.what());
|
||||
}
|
||||
if (m_use_exceptions)
|
||||
throw;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -1261,12 +1322,45 @@ public:
|
|||
throw parser_exception(ex.what(), ex.m_pos.first, ex.m_pos.second);
|
||||
}
|
||||
}
|
||||
|
||||
void set_interrupt(bool flag) {
|
||||
m_frontend.set_interrupt(flag);
|
||||
m_elaborator.set_interrupt(flag);
|
||||
m_import_parser.set_interrupt(flag);
|
||||
}
|
||||
|
||||
void reset_interrupt() {
|
||||
set_interrupt(false);
|
||||
}
|
||||
};
|
||||
bool parse_commands(frontend & fe, std::istream & in, bool use_exceptions, bool interactive) {
|
||||
parser_fn::show_prompt(interactive, fe);
|
||||
return parser_fn(fe, in, use_exceptions, interactive).parse_commands();
|
||||
|
||||
parser::parser(frontend fe, std::istream & in, bool use_exceptions, bool interactive) {
|
||||
parser::imp::show_prompt(interactive, fe);
|
||||
m_ptr.reset(new imp(fe, in, use_exceptions, interactive));
|
||||
}
|
||||
bool parse_commands_from_stdin(frontend & fe) {
|
||||
|
||||
parser::~parser() {
|
||||
}
|
||||
|
||||
bool parser::operator()() {
|
||||
return m_ptr->parse_commands();
|
||||
}
|
||||
|
||||
void parser::set_interrupt(bool flag) {
|
||||
m_ptr->set_interrupt(flag);
|
||||
}
|
||||
|
||||
expr parser::parse_expr() {
|
||||
return m_ptr->parse_expr_main();
|
||||
}
|
||||
|
||||
shell::shell(frontend & fe):m_frontend(fe) {
|
||||
}
|
||||
|
||||
shell::~shell() {
|
||||
}
|
||||
|
||||
bool shell::operator()() {
|
||||
#ifdef LEAN_USE_READLINE
|
||||
bool errors = false;
|
||||
char * input;
|
||||
|
@ -1276,15 +1370,22 @@ bool parse_commands_from_stdin(frontend & fe) {
|
|||
return errors;
|
||||
add_history(input);
|
||||
std::istringstream strm(input);
|
||||
if (!parse_commands(fe, strm, false, false))
|
||||
{
|
||||
parser p(m_frontend, strm, false, false);
|
||||
scoped_set_interruptable_ptr<parser> set(m_parser, &p);
|
||||
if (!p())
|
||||
errors = true;
|
||||
}
|
||||
free(input);
|
||||
}
|
||||
#else
|
||||
return parse_commands(fe, std::cin, false, true) ? 0 : 1;
|
||||
parser p(m_frontend, std::cin, false, true);
|
||||
scoped_set_interruptable_ptr<parser> set(m_parser, &p);
|
||||
return p();
|
||||
#endif
|
||||
}
|
||||
expr parse_expr(frontend const & fe, std::istream & in) {
|
||||
return parser_fn(const_cast<frontend&>(fe), in, true, false).parse_expr_main();
|
||||
|
||||
void shell::set_interrupt(bool flag) {
|
||||
m_parser.set_interrupt(flag);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -7,9 +7,47 @@ Author: Leonardo de Moura
|
|||
#pragma once
|
||||
#include <iostream>
|
||||
#include "lean_frontend.h"
|
||||
#include "interruptable_ptr.h"
|
||||
|
||||
namespace lean {
|
||||
bool parse_commands(frontend & fe, std::istream & in, bool use_exceptions = true, bool interactive = false);
|
||||
bool parse_commands_from_stdin(frontend & fe);
|
||||
expr parse_expr(frontend const & fe, std::istream & in);
|
||||
/** \brief Functional object for parsing commands and expressions */
|
||||
class parser {
|
||||
class imp;
|
||||
std::unique_ptr<imp> m_ptr;
|
||||
public:
|
||||
parser(frontend fe, std::istream & in, bool use_exceptions = true, bool interactive = false);
|
||||
~parser();
|
||||
|
||||
/** \brief Parse a sequence of commands */
|
||||
bool operator()();
|
||||
|
||||
/** \brief Parse a single expression */
|
||||
expr parse_expr();
|
||||
|
||||
void set_interrupt(bool flag);
|
||||
void interrupt() { set_interrupt(true); }
|
||||
void reset_interrupt() { set_interrupt(false); }
|
||||
};
|
||||
|
||||
/** \brief Implements the Read Eval Print loop */
|
||||
class shell {
|
||||
frontend m_frontend;
|
||||
interruptable_ptr<parser> m_parser;
|
||||
public:
|
||||
shell(frontend & fe);
|
||||
~shell();
|
||||
|
||||
bool operator()();
|
||||
|
||||
void set_interrupt(bool flag);
|
||||
void interrupt() { set_interrupt(true); }
|
||||
void reset_interrupt() { set_interrupt(false); }
|
||||
};
|
||||
|
||||
inline bool parse_commands(frontend & fe, std::istream & in, bool use_exceptions = true, bool interactive = false) {
|
||||
return parser(fe, in, use_exceptions, interactive)();
|
||||
}
|
||||
inline expr parse_expr(frontend const & fe, std::istream & in) {
|
||||
return parser(fe, in).parse_expr();
|
||||
}
|
||||
}
|
||||
|
|
|
@ -27,6 +27,7 @@ static name g_forall_name("forall");
|
|||
static name g_forall_unicode("\u2200");
|
||||
static name g_exists_name("exists");
|
||||
static name g_exists_unicode("\u2203");
|
||||
static name g_placeholder_name("_");
|
||||
|
||||
static char g_normalized[256];
|
||||
|
||||
|
@ -215,6 +216,8 @@ scanner::token scanner::read_a_symbol() {
|
|||
return token::Let;
|
||||
else if (m_name_val == g_in_name)
|
||||
return token::In;
|
||||
else if (m_name_val == g_placeholder_name)
|
||||
return token::Placeholder;
|
||||
else
|
||||
return is_command(m_name_val) ? token::CommandId : token::Id;
|
||||
}
|
||||
|
@ -392,6 +395,7 @@ std::ostream & operator<<(std::ostream & out, scanner::token const & t) {
|
|||
case scanner::token::Eq: out << "="; break;
|
||||
case scanner::token::Assign: out << ":="; break;
|
||||
case scanner::token::Type: out << "Type"; break;
|
||||
case scanner::token::Placeholder: out << "_"; break;
|
||||
case scanner::token::Eof: out << "EOF"; break;
|
||||
}
|
||||
return out;
|
||||
|
|
|
@ -19,7 +19,7 @@ class scanner {
|
|||
public:
|
||||
enum class token {
|
||||
LeftParen, RightParen, LeftCurlyBracket, RightCurlyBracket, Colon, Comma, Period, Lambda, Pi, Arrow,
|
||||
Let, In, Forall, Exists, Id, CommandId, IntVal, DecimalVal, StringVal, Eq, Assign, Type, Eof
|
||||
Let, In, Forall, Exists, Id, CommandId, IntVal, DecimalVal, StringVal, Eq, Assign, Type, Placeholder, Eof
|
||||
};
|
||||
protected:
|
||||
int m_spos; // position in the current line of the stream
|
||||
|
|
|
@ -12,6 +12,16 @@ Author: Leonardo de Moura
|
|||
#include "exception.h"
|
||||
|
||||
namespace lean {
|
||||
static name g_overload_name(name(name(name(0u), "library"), "overload"));
|
||||
static expr g_overload = mk_constant(g_overload_name);
|
||||
|
||||
bool is_overload_marker(expr const & e) {
|
||||
return e == g_overload;
|
||||
}
|
||||
|
||||
expr mk_overload_marker() {
|
||||
return g_overload;
|
||||
}
|
||||
|
||||
expr elaborator::lookup(context const & c, unsigned i) {
|
||||
auto p = lookup_ext(c, i);
|
||||
|
@ -21,7 +31,7 @@ expr elaborator::lookup(context const & c, unsigned i) {
|
|||
return lift_free_vars(def.get_domain(), length(c) - length(def_c));
|
||||
}
|
||||
|
||||
elaborator::elaborator(environment & env):
|
||||
elaborator::elaborator(environment const & env):
|
||||
m_env(env), m_metaenv(env) {
|
||||
}
|
||||
|
||||
|
|
|
@ -14,7 +14,7 @@ namespace lean {
|
|||
the value of implicit arguments.
|
||||
*/
|
||||
class elaborator {
|
||||
environment & m_env;
|
||||
environment const & m_env;
|
||||
metavar_env m_metaenv;
|
||||
|
||||
expr lookup(context const & c, unsigned i);
|
||||
|
@ -24,8 +24,20 @@ class elaborator {
|
|||
expr process(expr const & e, context const & ctx);
|
||||
|
||||
public:
|
||||
elaborator(environment & env);
|
||||
elaborator(environment const & env);
|
||||
metavar_env & menv() { return m_metaenv; }
|
||||
|
||||
expr operator()(expr const & e);
|
||||
|
||||
void clear() { m_metaenv.clear(); }
|
||||
|
||||
void set_interrupt(bool flag) { m_metaenv.set_interrupt(flag); }
|
||||
void interrupt() { set_interrupt(true); }
|
||||
void reset_interrupt() { set_interrupt(false); }
|
||||
};
|
||||
|
||||
/** \brief Return true iff \c e is a special constant used to mark application of overloads. */
|
||||
bool is_overload_marker(expr const & e);
|
||||
/** \brief Return the overload marker */
|
||||
expr mk_overload_marker();
|
||||
}
|
||||
|
|
|
@ -445,6 +445,10 @@ void metavar_env::set_interrupt(bool flag) {
|
|||
m_interrupted = flag;
|
||||
}
|
||||
|
||||
void metavar_env::clear() {
|
||||
m_cells.clear();
|
||||
}
|
||||
|
||||
void metavar_env::display(std::ostream & out) const {
|
||||
for (unsigned i = 0; i < m_cells.size(); i++) {
|
||||
out << "?" << i << " --> ";
|
||||
|
|
|
@ -146,6 +146,11 @@ public:
|
|||
*/
|
||||
context const & get_context(expr const & m);
|
||||
|
||||
/**
|
||||
\brief Clear/Reset the state.
|
||||
*/
|
||||
void clear();
|
||||
|
||||
void set_interrupt(bool flag);
|
||||
|
||||
void display(std::ostream & out) const;
|
||||
|
|
|
@ -7,19 +7,20 @@
|
|||
#include "lean_parser.h"
|
||||
using namespace lean;
|
||||
|
||||
static interruptable_ptr<frontend> g_lean_frontend;
|
||||
static interruptable_ptr<shell> g_lean_shell;
|
||||
|
||||
static void on_ctrl_c(int) {
|
||||
g_lean_frontend.set_interrupt(true);
|
||||
g_lean_shell.set_interrupt(true);
|
||||
}
|
||||
|
||||
bool lean_shell() {
|
||||
std::cout << "Lean (version " << LEAN_VERSION_MAJOR << "." << LEAN_VERSION_MINOR << ")\n";
|
||||
std::cout << "Type Ctrl-D to exit or 'Help.' for help."<< std::endl;
|
||||
frontend f;
|
||||
scoped_set_interruptable_ptr<frontend> set(g_lean_frontend, &f);
|
||||
shell sh(f);
|
||||
scoped_set_interruptable_ptr<shell> set(g_lean_shell, &sh);
|
||||
signal(SIGINT, on_ctrl_c);
|
||||
return parse_commands_from_stdin(f);
|
||||
return sh();
|
||||
}
|
||||
|
||||
int main(int argc, char ** argv) {
|
||||
|
@ -30,7 +31,8 @@ int main(int argc, char ** argv) {
|
|||
frontend f;
|
||||
for (int i = 1; i < argc; i++) {
|
||||
std::ifstream in(argv[i]);
|
||||
if (!parse_commands(f, in))
|
||||
parser p(f, in);
|
||||
if (!p())
|
||||
ok = false;
|
||||
}
|
||||
return ok ? 0 : 1;
|
||||
|
|
Loading…
Reference in a new issue