2014-06-09 20:18:10 +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
|
|
|
|
*/
|
|
|
|
#pragma once
|
2014-06-18 03:39:42 +00:00
|
|
|
#include <string>
|
2014-06-09 20:18:10 +00:00
|
|
|
#include <utility>
|
|
|
|
#include "util/buffer.h"
|
2014-06-09 22:24:33 +00:00
|
|
|
#include "util/lua.h"
|
2014-06-09 20:18:10 +00:00
|
|
|
#include "kernel/expr.h"
|
2014-10-18 17:55:39 +00:00
|
|
|
#include "library/head_map.h"
|
2014-06-10 16:11:45 +00:00
|
|
|
#include "frontends/lean/token_table.h"
|
2014-06-16 19:28:58 +00:00
|
|
|
#include "frontends/lean/parser_pos_provider.h"
|
2014-06-09 20:18:10 +00:00
|
|
|
|
2015-06-30 23:58:08 +00:00
|
|
|
#ifndef LEAN_DEFAULT_NOTATION_PRIORITY
|
|
|
|
#define LEAN_DEFAULT_NOTATION_PRIORITY 1000
|
|
|
|
#endif
|
|
|
|
|
2014-06-09 20:18:10 +00:00
|
|
|
namespace lean {
|
2014-06-11 00:02:06 +00:00
|
|
|
class parser;
|
2014-12-02 20:04:18 +00:00
|
|
|
class io_state_stream;
|
2014-06-09 20:18:10 +00:00
|
|
|
namespace notation {
|
2014-06-16 19:28:58 +00:00
|
|
|
typedef std::function<expr(parser &, unsigned, expr const *, pos_info const &)> parse_fn;
|
2014-06-09 20:18:10 +00:00
|
|
|
|
2014-06-18 03:39:42 +00:00
|
|
|
enum class action_kind { Skip, Expr, Exprs, Binder, Binders, ScopedExpr, Ext, LuaExt };
|
2014-06-09 20:18:10 +00:00
|
|
|
struct action_cell;
|
|
|
|
|
|
|
|
/**
|
|
|
|
\brief Action that must be performed after a token is consumed.
|
|
|
|
The possible actions are:
|
|
|
|
|
|
|
|
- Skip: do nothing, just consume the token.
|
|
|
|
- Expr: parse a Lean expression using a given right-binding-power (rbp).
|
|
|
|
- Exprs: parse a sequence of expressions separated by a given token.
|
|
|
|
The expressions are parsed using the given right-binding-power (rbp).
|
|
|
|
The multiple expressions are combined into a single expression by folding them
|
|
|
|
using an expression (rec) with two free variables and an initial element, fold left and right are supported.
|
|
|
|
- Binder: parse a single Lean binder.
|
|
|
|
- Binders: parse a sequence of Lean binders.
|
|
|
|
- ScopedExpr: parse an expression using the previous binder(s).
|
|
|
|
The resulting expression is computed using an expression (rec) with one free variable.
|
|
|
|
- Ext: invokes an external procedure parse_fn. This is used to define notation that does
|
|
|
|
complicated notation.
|
|
|
|
|
|
|
|
The parse_table datastructure applies "left factoring" when actions are "compatible".
|
|
|
|
|
|
|
|
\remark Ext actions are never compatible with anything.
|
|
|
|
*/
|
|
|
|
class action {
|
|
|
|
action_cell * m_ptr;
|
|
|
|
explicit action(action_cell * ptr);
|
|
|
|
static action g_skip_action;
|
|
|
|
static action g_binder_action;
|
|
|
|
static action g_binders_action;
|
|
|
|
public:
|
|
|
|
action();
|
|
|
|
action(action const & s);
|
|
|
|
action(action && s);
|
|
|
|
~action();
|
|
|
|
|
|
|
|
action & operator=(action const & s);
|
|
|
|
action & operator=(action && s);
|
|
|
|
|
2014-09-23 17:00:36 +00:00
|
|
|
friend void initialize_parse_table();
|
2014-06-09 20:18:10 +00:00
|
|
|
friend action mk_expr_action(unsigned rbp);
|
2014-11-24 00:42:53 +00:00
|
|
|
friend action mk_binders_action(unsigned rbp);
|
|
|
|
friend action mk_binder_action(unsigned rbp);
|
2014-11-09 22:08:33 +00:00
|
|
|
friend action mk_exprs_action(name const & sep, expr const & rec, optional<expr> const & ini,
|
2014-09-23 17:00:36 +00:00
|
|
|
optional<name> const & terminator, bool right, unsigned rbp);
|
2014-06-10 16:39:01 +00:00
|
|
|
friend action mk_scoped_expr_action(expr const & rec, unsigned rbp, bool lambda);
|
2014-10-28 00:12:25 +00:00
|
|
|
friend action mk_ext_action_core(parse_fn const & fn);
|
2014-06-12 16:08:38 +00:00
|
|
|
friend action mk_ext_action(parse_fn const & fn);
|
2014-06-18 03:39:42 +00:00
|
|
|
friend action mk_ext_lua_action(char const * lua_fn);
|
2014-06-09 20:18:10 +00:00
|
|
|
|
|
|
|
action_kind kind() const;
|
|
|
|
unsigned rbp() const;
|
|
|
|
name const & get_sep() const;
|
|
|
|
expr const & get_rec() const;
|
2014-11-09 22:08:33 +00:00
|
|
|
optional<expr> const & get_initial() const;
|
2014-07-30 22:04:44 +00:00
|
|
|
optional<name> const & get_terminator() const;
|
2014-06-09 20:18:10 +00:00
|
|
|
bool is_fold_right() const;
|
2014-06-10 16:39:01 +00:00
|
|
|
bool use_lambda_abstraction() const;
|
2014-06-09 20:18:10 +00:00
|
|
|
parse_fn const & get_parse_fn() const;
|
2014-06-18 03:39:42 +00:00
|
|
|
std::string const & get_lua_fn() const;
|
2014-06-09 20:18:10 +00:00
|
|
|
|
2014-07-30 19:37:35 +00:00
|
|
|
bool is_equal(action const & a) const;
|
2015-08-17 01:21:29 +00:00
|
|
|
bool is_equivalent(action const & a) const;
|
2015-08-17 15:41:30 +00:00
|
|
|
// We say two actions are compatible if the parser can decide which one to choose by looking at the next token.
|
|
|
|
bool is_compatible(action const & a) const;
|
2015-08-17 01:21:29 +00:00
|
|
|
|
2014-12-02 20:04:18 +00:00
|
|
|
void display(io_state_stream & out) const;
|
2014-10-18 17:55:39 +00:00
|
|
|
|
|
|
|
/** \brief Return true iff the action is not Ext or LuaExt */
|
|
|
|
bool is_simple() const;
|
2014-06-09 20:18:10 +00:00
|
|
|
};
|
2014-10-18 18:49:27 +00:00
|
|
|
inline bool operator==(action const & a1, action const & a2) { return a1.is_equal(a2); }
|
|
|
|
inline bool operator!=(action const & a1, action const & a2) { return !a1.is_equal(a2); }
|
2014-06-09 20:18:10 +00:00
|
|
|
|
|
|
|
action mk_skip_action();
|
|
|
|
action mk_expr_action(unsigned rbp = 0);
|
2014-11-09 22:08:33 +00:00
|
|
|
action mk_exprs_action(name const & sep, expr const & rec, optional<expr> const & ini, optional<name> const & terminator, bool right,
|
2014-10-03 20:47:49 +00:00
|
|
|
unsigned rbp = 0);
|
2014-11-24 00:42:53 +00:00
|
|
|
action mk_binder_action(unsigned rbp = 0);
|
|
|
|
action mk_binders_action(unsigned rbp = 0);
|
2014-06-10 16:39:01 +00:00
|
|
|
action mk_scoped_expr_action(expr const & rec, unsigned rbp = 0, bool lambda = true);
|
2014-10-28 00:12:25 +00:00
|
|
|
action mk_ext_action_core(parse_fn const & fn);
|
2014-06-12 16:08:38 +00:00
|
|
|
action mk_ext_action(parse_fn const & fn);
|
2014-06-18 03:39:42 +00:00
|
|
|
action mk_ext_lua_action(char const * lua_fn);
|
2014-06-09 20:18:10 +00:00
|
|
|
|
2014-07-19 23:23:55 +00:00
|
|
|
/** \brief Apply \c f to expressions embedded in the given action */
|
|
|
|
action replace(action const & a, std::function<expr(expr const &)> const & f);
|
|
|
|
|
2014-06-09 20:18:10 +00:00
|
|
|
class transition {
|
|
|
|
name m_token;
|
|
|
|
action m_action;
|
|
|
|
public:
|
|
|
|
transition(name const & t, action const & a):
|
|
|
|
m_token(t), m_action(a) {}
|
|
|
|
name const & get_token() const { return m_token; }
|
|
|
|
action const & get_action() const { return m_action; }
|
2014-10-18 17:55:39 +00:00
|
|
|
bool is_simple() const { return m_action.is_simple(); }
|
|
|
|
bool is_safe_ascii() const { return m_token.is_safe_ascii(); }
|
2014-06-09 20:18:10 +00:00
|
|
|
};
|
2014-10-18 18:49:27 +00:00
|
|
|
inline bool operator==(transition const & t1, transition const & t2) {
|
|
|
|
return t1.get_token() == t2.get_token() && t1.get_action() == t2.get_action();
|
|
|
|
}
|
|
|
|
inline bool operator!=(transition const & t1, transition const & t2) {
|
|
|
|
return !(t1 == t2);
|
|
|
|
}
|
2014-10-18 17:55:39 +00:00
|
|
|
|
2014-07-19 23:23:55 +00:00
|
|
|
/** \brief Apply \c f to expressions embedded in the given transition */
|
|
|
|
transition replace(transition const & t, std::function<expr(expr const &)> const & f);
|
|
|
|
|
2015-08-17 01:21:29 +00:00
|
|
|
class accepting {
|
|
|
|
unsigned m_prio;
|
|
|
|
list<action> m_postponed; // exprs and scoped_expr actions
|
|
|
|
expr m_expr; // resulting expression
|
|
|
|
public:
|
|
|
|
accepting(unsigned prio, list<action> const & post, expr const & e):
|
|
|
|
m_prio(prio), m_postponed(post), m_expr(e) {}
|
|
|
|
unsigned get_prio() const { return m_prio; }
|
|
|
|
list<action> const & get_postponed() const { return m_postponed; }
|
|
|
|
expr const & get_expr() const { return m_expr; }
|
|
|
|
};
|
|
|
|
|
|
|
|
void display(io_state_stream & out, unsigned num, transition const * ts, list<accepting> const & es, bool nud,
|
2014-12-02 20:04:18 +00:00
|
|
|
optional<token_table> const & tt);
|
|
|
|
|
2014-06-09 20:18:10 +00:00
|
|
|
/**
|
|
|
|
\brief Parse table "transition rules" for implementing a Pratt's parser.
|
|
|
|
The table supports "left factoring" in the methods \c add and \c merge.
|
|
|
|
*/
|
|
|
|
class parse_table {
|
|
|
|
struct cell;
|
|
|
|
cell * m_ptr;
|
|
|
|
explicit parse_table(cell * c);
|
2015-08-17 01:21:29 +00:00
|
|
|
parse_table add_core(unsigned num, transition const * ts, expr const & a, unsigned priority, bool overload, buffer<action> & postponed) const;
|
2015-06-30 23:58:08 +00:00
|
|
|
void for_each(buffer<transition> & ts, std::function<void(unsigned, transition const *,
|
2015-08-17 01:21:29 +00:00
|
|
|
list<accepting> const &)> const & fn) const;
|
2014-06-09 20:18:10 +00:00
|
|
|
public:
|
2014-06-09 22:24:33 +00:00
|
|
|
parse_table(bool nud = true);
|
2014-06-09 20:18:10 +00:00
|
|
|
parse_table(parse_table const & s);
|
|
|
|
parse_table(parse_table && s);
|
|
|
|
~parse_table();
|
|
|
|
parse_table & operator=(parse_table const & n);
|
|
|
|
parse_table & operator=(parse_table&& n);
|
|
|
|
|
2014-06-09 22:24:33 +00:00
|
|
|
bool is_nud() const;
|
2015-06-30 23:58:08 +00:00
|
|
|
parse_table add(unsigned num, transition const * ts, expr const & a, unsigned priority, bool overload) const;
|
2014-06-12 03:56:10 +00:00
|
|
|
parse_table add(std::initializer_list<transition> const & ts, expr const & a) const {
|
2015-06-30 23:58:08 +00:00
|
|
|
return add(ts.size(), ts.begin(), a, LEAN_DEFAULT_NOTATION_PRIORITY, true);
|
2014-06-12 03:56:10 +00:00
|
|
|
}
|
2015-08-17 01:21:29 +00:00
|
|
|
parse_table merge(parse_table const & s, bool overload) const;\
|
2015-09-28 20:43:04 +00:00
|
|
|
list<pair<transition, parse_table>> find(name const & tk) const;
|
2015-08-17 01:21:29 +00:00
|
|
|
list<accepting> const & is_accepting() const;
|
|
|
|
void for_each(std::function<void(unsigned, transition const *, list<accepting> const &)> const & fn) const;
|
2014-07-07 16:31:42 +00:00
|
|
|
|
2014-12-02 20:04:18 +00:00
|
|
|
void display(io_state_stream & out, optional<token_table> const & tk) const;
|
2014-06-09 20:18:10 +00:00
|
|
|
};
|
2014-10-18 17:55:39 +00:00
|
|
|
|
|
|
|
/** \brief Given a notation definition, return the "head symbol" of
|
|
|
|
every instance of the given notation.
|
|
|
|
|
|
|
|
\remark The result is none if the notation uses actions implemented in C++ or Lua.
|
|
|
|
The result is none if the denotation is a variable.
|
|
|
|
*/
|
|
|
|
optional<head_index> get_head_index(unsigned num, transition const * ts, expr const & a);
|
|
|
|
|
2014-09-23 17:00:36 +00:00
|
|
|
void initialize_parse_table();
|
|
|
|
void finalize_parse_table();
|
2014-06-09 20:18:10 +00:00
|
|
|
}
|
2014-06-09 22:54:56 +00:00
|
|
|
typedef notation::parse_table parse_table;
|
2014-06-09 22:24:33 +00:00
|
|
|
void open_parse_table(lua_State * L);
|
2014-09-23 17:00:36 +00:00
|
|
|
inline void initialize_parse_table() { notation::initialize_parse_table(); }
|
|
|
|
inline void finalize_parse_table() { notation::finalize_parse_table(); }
|
2014-06-09 20:18:10 +00:00
|
|
|
}
|