feat(frontends/lean): save transitions instead of actions in notation::parse_table

This commit is contained in:
Sebastian Ullrich 2015-09-28 22:43:04 +02:00 committed by Leonardo de Moura
parent 189a300b11
commit 8f96b725e3
4 changed files with 41 additions and 41 deletions

View file

@ -136,13 +136,13 @@ static auto parse_mixfix_notation(parser & p, mixfix_kind k, bool overload, nota
if (auto ls = get_reserved_nud_table(p.env()).find(tks)) { if (auto ls = get_reserved_nud_table(p.env()).find(tks)) {
// Remark: we are ignoring multiple actions in the reserved notation table // Remark: we are ignoring multiple actions in the reserved notation table
reserved_pt = head(ls).second; reserved_pt = head(ls).second;
reserved_action = head(ls).first; reserved_action = head(ls).first.get_action();
} }
} else { } else {
if (auto ls = get_reserved_led_table(p.env()).find(tks)) { if (auto ls = get_reserved_led_table(p.env()).find(tks)) {
// Remark: we are ignoring multiple actions in the reserved notation table // Remark: we are ignoring multiple actions in the reserved notation table
reserved_pt = head(ls).second; reserved_pt = head(ls).second;
reserved_action = head(ls).first; reserved_action = head(ls).first.get_action();
} }
} }
} }
@ -413,8 +413,8 @@ static unsigned get_default_prec(optional<parse_table> const & pt, name const &
return LEAN_DEFAULT_PRECEDENCE; return LEAN_DEFAULT_PRECEDENCE;
if (auto ls = pt->find(tk)) { if (auto ls = pt->find(tk)) {
for (auto at : ls) { for (auto at : ls) {
if (at.first.kind() == notation::action_kind::Expr) if (at.first.get_action().kind() == notation::action_kind::Expr)
return at.first.rbp(); return at.first.get_action().rbp();
} }
} }
return LEAN_DEFAULT_PRECEDENCE; return LEAN_DEFAULT_PRECEDENCE;
@ -426,7 +426,7 @@ static optional<parse_table> find_match(optional<parse_table> const & pt, transi
if (pt) { if (pt) {
if (auto ls = pt->find(new_trans.get_token())) { if (auto ls = pt->find(new_trans.get_token())) {
for (auto at : ls) { for (auto at : ls) {
if (new_trans.get_action().is_equal(at.first)) if (new_trans.get_action().is_equal(at.first.get_action()))
return optional<parse_table>(at.second); return optional<parse_table>(at.second);
} }
} }
@ -435,11 +435,11 @@ static optional<parse_table> find_match(optional<parse_table> const & pt, transi
} }
/** \brief Lift parse_table::find method to optional<parse_table> */ /** \brief Lift parse_table::find method to optional<parse_table> */
static list<pair<action, parse_table>> find_next(optional<parse_table> const & pt, name const & tk) { static list<pair<transition, parse_table>> find_next(optional<parse_table> const & pt, name const & tk) {
if (pt) if (pt)
return pt->find(tk); return pt->find(tk);
else else
return list<pair<action, parse_table>>(); return list<pair<transition, parse_table>>();
} }
static unsigned parse_binders_rbp(parser & p) { static unsigned parse_binders_rbp(parser & p) {
@ -514,7 +514,7 @@ static notation_entry parse_notation_core(parser & p, bool overload, notation_en
name tk = parse_quoted_symbol_or_token(p, new_tokens, used_default, grp); name tk = parse_quoted_symbol_or_token(p, new_tokens, used_default, grp);
if (auto at = find_next(reserved_pt, tk)) { if (auto at = find_next(reserved_pt, tk)) {
// Remark: we are ignoring multiple actions in the reserved notation table // Remark: we are ignoring multiple actions in the reserved notation table
action const & a = head(at).first; action const & a = head(at).first.get_action();
reserved_pt = head(at).second; reserved_pt = head(at).second;
switch (a.kind()) { switch (a.kind()) {
case notation::action_kind::Skip: case notation::action_kind::Skip:

View file

@ -331,9 +331,9 @@ transition replace(transition const & t, std::function<expr(expr const &)> const
} }
struct parse_table::cell { struct parse_table::cell {
bool m_nud; bool m_nud;
list<accepting> m_accept; list<accepting> m_accept;
name_map<list<pair<action, parse_table>>> m_children; name_map<list<pair<transition, parse_table>>> m_children;
MK_LEAN_RC(); // Declare m_rc counter MK_LEAN_RC(); // Declare m_rc counter
void dealloc() { delete this; } void dealloc() { delete this; }
cell(bool nud = true):m_nud(nud), m_rc(1) {} cell(bool nud = true):m_nud(nud), m_rc(1) {}
@ -347,12 +347,12 @@ parse_table::parse_table(parse_table && s):m_ptr(s.m_ptr) { s.m_ptr = nullptr; }
parse_table::~parse_table() { if (m_ptr) m_ptr->dec_ref(); } parse_table::~parse_table() { if (m_ptr) m_ptr->dec_ref(); }
parse_table & parse_table::operator=(parse_table const & s) { LEAN_COPY_REF(s); } parse_table & parse_table::operator=(parse_table const & s) { LEAN_COPY_REF(s); }
parse_table & parse_table::operator=(parse_table && s) { LEAN_MOVE_REF(s); } parse_table & parse_table::operator=(parse_table && s) { LEAN_MOVE_REF(s); }
list<pair<action, parse_table>> parse_table::find(name const & tk) const { list<pair<transition, parse_table>> parse_table::find(name const & tk) const {
list<pair<action, parse_table>> const * it = m_ptr->m_children.find(tk); list<pair<transition, parse_table>> const * it = m_ptr->m_children.find(tk);
if (it) if (it)
return *it; return *it;
else else
return list<pair<action, parse_table>>(); return list<pair<transition, parse_table>>();
} }
list<accepting> const & parse_table::is_accepting() const { list<accepting> const & parse_table::is_accepting() const {
@ -397,9 +397,9 @@ static list<accepting> insert(list<accepting> const & l, unsigned priority, list
} }
} }
static bool contains_equivalent_action(list<pair<action, parse_table>> const & l, action const & a) { static bool contains_equivalent_action(list<pair<transition, parse_table>> const & l, action const & a) {
for (auto const & p : l) { for (auto const & p : l) {
if (p.first.is_equivalent(a)) if (p.first.get_action().is_equivalent(a))
return true; return true;
} }
return false; return false;
@ -419,18 +419,18 @@ parse_table parse_table::add_core(unsigned num, transition const * ts, expr cons
r.m_ptr->m_accept = insert(new_accept, priority, postponed, a); r.m_ptr->m_accept = insert(new_accept, priority, postponed, a);
} }
} else { } else {
list<pair<action, parse_table>> const * it = r.m_ptr->m_children.find(ts->get_token()); list<pair<transition, parse_table>> const * it = r.m_ptr->m_children.find(ts->get_token());
action const & ts_act = ts->get_action(); action const & ts_act = ts->get_action();
action_kind k = ts_act.kind(); action_kind k = ts_act.kind();
if (k == action_kind::Exprs || k == action_kind::ScopedExpr) if (k == action_kind::Exprs || k == action_kind::ScopedExpr)
post_buffer.push_back(ts_act); post_buffer.push_back(ts_act);
list<pair<action, parse_table>> new_lst; list<pair<transition, parse_table>> new_lst;
if (it) { if (it) {
if (contains_equivalent_action(*it, ts_act)) { if (contains_equivalent_action(*it, ts_act)) {
buffer<pair<action, parse_table>> tmp; buffer<pair<transition, parse_table>> tmp;
to_buffer(*it, tmp); to_buffer(*it, tmp);
for (pair<action, parse_table> & p : tmp) { for (pair<transition, parse_table> & p : tmp) {
if (p.first.is_equivalent(ts_act)) { if (p.first.get_action().is_equivalent(ts_act)) {
p.second = p.second.add_core(num-1, ts+1, a, priority, overload, post_buffer); p.second = p.second.add_core(num-1, ts+1, a, priority, overload, post_buffer);
break; break;
} }
@ -438,15 +438,15 @@ parse_table parse_table::add_core(unsigned num, transition const * ts, expr cons
new_lst = to_list(tmp); new_lst = to_list(tmp);
} else { } else {
// remove incompatible actions // remove incompatible actions
new_lst = filter(*it, [&](pair<action, parse_table> const & p) { new_lst = filter(*it, [&](pair<transition, parse_table> const & p) {
return p.first.is_compatible(ts_act); return p.first.get_action().is_compatible(ts_act);
}); });
parse_table new_child = parse_table().add_core(num-1, ts+1, a, priority, overload, post_buffer); parse_table new_child = parse_table().add_core(num-1, ts+1, a, priority, overload, post_buffer);
new_lst = cons(mk_pair(ts_act, new_child), new_lst); new_lst = cons(mk_pair(*ts, new_child), new_lst);
} }
} else { } else {
parse_table new_child = parse_table().add_core(num-1, ts+1, a, priority, overload, post_buffer); parse_table new_child = parse_table().add_core(num-1, ts+1, a, priority, overload, post_buffer);
new_lst = to_list(mk_pair(ts_act, new_child)); new_lst = to_list(mk_pair(*ts, new_child));
} }
r.m_ptr->m_children.insert(ts->get_token(), new_lst); r.m_ptr->m_children.insert(ts->get_token(), new_lst);
} }
@ -527,9 +527,9 @@ void parse_table::for_each(buffer<transition> & ts,
list<accepting> const &)> const & fn) const { list<accepting> const &)> const & fn) const {
if (!is_nil(m_ptr->m_accept)) if (!is_nil(m_ptr->m_accept))
fn(ts.size(), ts.data(), m_ptr->m_accept); fn(ts.size(), ts.data(), m_ptr->m_accept);
m_ptr->m_children.for_each([&](name const & k, list<pair<action, parse_table>> const & lst) { m_ptr->m_children.for_each([&](name const & k, list<pair<transition, parse_table>> const & lst) {
for (auto const & p : lst) { for (auto const & p : lst) {
ts.push_back(transition(k, p.first)); ts.push_back(p.first);
p.second.for_each(ts, fn); p.second.for_each(ts, fn);
ts.pop_back(); ts.pop_back();
} }
@ -773,11 +773,11 @@ static int merge(lua_State * L) {
} }
static int find(lua_State * L) { static int find(lua_State * L) {
list<pair<action, parse_table>> it = to_parse_table(L, 1).find(to_name_ext(L, 2)); list<pair<transition, parse_table>> it = to_parse_table(L, 1).find(to_name_ext(L, 2));
if (it) { if (it) {
// TODO(Leo): support multiple actions // TODO(Leo): support multiple actions
auto p = head(it); auto p = head(it);
push_notation_action(L, p.first); push_notation_action(L, p.first.get_action());
push_parse_table(L, p.second); push_parse_table(L, p.second);
return 2; return 2;
} else { } else {

View file

@ -173,7 +173,7 @@ public:
return add(ts.size(), ts.begin(), a, LEAN_DEFAULT_NOTATION_PRIORITY, true); return add(ts.size(), ts.begin(), a, LEAN_DEFAULT_NOTATION_PRIORITY, true);
} }
parse_table merge(parse_table const & s, bool overload) const;\ parse_table merge(parse_table const & s, bool overload) const;\
list<pair<action, parse_table>> find(name const & tk) const; list<pair<transition, parse_table>> find(name const & tk) const;
list<accepting> const & is_accepting() const; list<accepting> const & is_accepting() const;
void for_each(std::function<void(unsigned, transition const *, list<accepting> const &)> const & fn) const; void for_each(std::function<void(unsigned, transition const *, list<accepting> const &)> const & fn) const;

View file

@ -1175,9 +1175,9 @@ void parser::process_postponed(buffer<expr> const & args, bool is_left,
} }
// Return true iff the current token is the terminator of some Exprs action, and store the matching pair in r // Return true iff the current token is the terminator of some Exprs action, and store the matching pair in r
static bool curr_is_terminator_of_exprs_action(parser const & p, list<pair<notation::action, parse_table>> const & lst, pair<notation::action, parse_table> const * & r) { static bool curr_is_terminator_of_exprs_action(parser const & p, list<pair<notation::transition, parse_table>> const & lst, pair<notation::transition, parse_table> const * & r) {
for (auto const & pr : lst) { for (auto const & pr : lst) {
notation::action const & a = pr.first; notation::action const & a = pr.first.get_action();
if (a.kind() == notation::action_kind::Exprs && if (a.kind() == notation::action_kind::Exprs &&
a.get_terminator() && a.get_terminator() &&
p.curr_is_token(*a.get_terminator())) { p.curr_is_token(*a.get_terminator())) {
@ -1189,9 +1189,9 @@ static bool curr_is_terminator_of_exprs_action(parser const & p, list<pair<notat
} }
// Return true iff \c lst contains a Skip action, and store the matching pair in r. // Return true iff \c lst contains a Skip action, and store the matching pair in r.
static bool has_skip(list<pair<notation::action, parse_table>> const & lst, pair<notation::action, parse_table> const * & r) { static bool has_skip(list<pair<notation::transition, parse_table>> const & lst, pair<notation::transition, parse_table> const * & r) {
for (auto const & p : lst) { for (auto const & p : lst) {
notation::action const & a = p.first; notation::action const & a = p.first.get_action();
if (a.kind() == notation::action_kind::Skip) { if (a.kind() == notation::action_kind::Skip) {
r = &p; r = &p;
return true; return true;
@ -1200,9 +1200,9 @@ static bool has_skip(list<pair<notation::action, parse_table>> const & lst, pair
return false; return false;
} }
static pair<notation::action, parse_table> const * get_non_skip(list<pair<notation::action, parse_table>> const & lst) { static pair<notation::transition, parse_table> const * get_non_skip(list<pair<notation::transition, parse_table>> const & lst) {
for (auto const & p : lst) { for (auto const & p : lst) {
notation::action const & a = p.first; notation::action const & a = p.first.get_action();
if (a.kind() != notation::action_kind::Skip) if (a.kind() != notation::action_kind::Skip)
return &p; return &p;
} }
@ -1234,26 +1234,26 @@ expr parser::parse_notation_core(parse_table t, expr * left, bool as_tactic) {
auto r = t.find(get_token_info().value()); auto r = t.find(get_token_info().value());
if (!r) if (!r)
break; break;
pair<notation::action, parse_table> const * curr_pair = nullptr; pair<notation::transition, parse_table> const * curr_pair = nullptr;
if (tail(r)) { if (tail(r)) {
// There is more than one possible actions. // There is more than one possible actions.
// In the current implementation, we support the following possible cases (Skip, Expr), (Skip, Exprs) amd (Skip, ScopedExpr) // In the current implementation, we support the following possible cases (Skip, Expr), (Skip, Exprs) amd (Skip, ScopedExpr)
next(); next();
if (curr_is_terminator_of_exprs_action(*this, r, curr_pair)) { if (curr_is_terminator_of_exprs_action(*this, r, curr_pair)) {
lean_assert(curr_pair->first.kind() == notation::action_kind::Exprs); lean_assert(curr_pair->first.get_action().kind() == notation::action_kind::Exprs);
} else if (has_skip(r, curr_pair) && !curr_starts_expr()) { } else if (has_skip(r, curr_pair) && !curr_starts_expr()) {
lean_assert(curr_pair->first.kind() == notation::action_kind::Skip); lean_assert(curr_pair->first.get_action().kind() == notation::action_kind::Skip);
} else { } else {
curr_pair = get_non_skip(r); curr_pair = get_non_skip(r);
} }
} else { } else {
// there is only one possible action // there is only one possible action
curr_pair = &head(r); curr_pair = &head(r);
if (curr_pair->first.kind() != notation::action_kind::Ext) if (curr_pair->first.get_action().kind() != notation::action_kind::Ext)
next(); next();
} }
lean_assert(curr_pair); lean_assert(curr_pair);
notation::action const & a = curr_pair->first; notation::action const & a = curr_pair->first.get_action();
switch (a.kind()) { switch (a.kind()) {
case notation::action_kind::Skip: case notation::action_kind::Skip:
break; break;