mpw-shell/phase2-parser.lemon

201 lines
4.8 KiB
Plaintext
Raw Normal View History

2016-01-30 12:44:42 -05:00
/*
>, < redirection is handled later, after environment expansion.
(also, redirection can be in the middle of a command.)
*/
%include {
#include "phase2.h"
#include "command.h"
#define LEMON_SUPER phase2_parser
2016-02-04 21:57:17 -05:00
#include "phase2-parser.h"
2016-01-30 12:44:42 -05:00
}
%code {
std::unique_ptr<phase2_parser> phase2_parser::make() {
return std::make_unique<yypParser>();
}
2016-02-04 21:57:17 -05:00
bool phase2_parser::continuation() const {
yypParser *self = (yypParser *)this;
for (const auto &e : *self) {
if (e.major == BEGIN) return true;
if (e.major == LPAREN) return true;
if (e.major == IF) return true;
if (e.major == AMP_AMP) return true;
if (e.major == PIPE_PIPE) return true;
if (e.major == LOOP) return true;
2016-06-16 00:04:29 -04:00
if (e.major == FOR) return true;
2016-07-26 14:20:11 -04:00
if (e.major == PIPE) return true;
if (e.major == PIPE_PIPE) return true;
if (e.major == AMP_AMP) return true;
2016-02-04 21:57:17 -05:00
}
return false;
}
2016-01-30 12:44:42 -05:00
}
%left PIPE_PIPE AMP_AMP.
%left PIPE.
%token_type {std::string}
%default_type {command_ptr}
%type start {void}
%type command_list {void}
/* these are put into a queue for immmediate execution */
2016-02-04 21:45:04 -05:00
start ::= command_list.
2016-02-01 20:38:29 -05:00
2016-01-30 12:44:42 -05:00
2016-02-04 21:45:04 -05:00
command_list ::= .
command_list ::= command_list sep .
command_list ::= command_list command(C) sep . {
2016-01-30 12:44:42 -05:00
if (C) command_queue.emplace_back(std::move(C));
}
2016-02-01 20:38:29 -05:00
2016-01-30 12:44:42 -05:00
/*
compound_list is identical to command_list, but it is not executed immediately.
*/
%type compound_list { command_ptr_vector }
2016-02-01 20:38:29 -05:00
2016-02-04 21:45:04 -05:00
compound_list ::= .
2016-08-05 10:34:05 -04:00
compound_list(L) ::= compound_list(L) sep.
2016-02-01 20:38:29 -05:00
2016-08-05 10:34:05 -04:00
compound_list(L) ::= compound_list(L) command(C) sep . {
if (C) L.emplace_back(std::move(C));
2016-02-01 20:38:29 -05:00
}
2016-02-04 21:45:04 -05:00
2016-01-30 12:44:42 -05:00
sep ::= SEMI.
sep ::= NL.
%type command { command_ptr }
2016-02-01 20:38:29 -05:00
/* nb -- ||, &&, | -- both sides are optional. This does not. */
2016-01-30 12:44:42 -05:00
command(RV) ::= command(L) PIPE_PIPE opt_nl command(R). {
RV = std::make_unique<or_command>(std::move(L), std::move(R));
}
2016-02-01 20:38:29 -05:00
command(RV) ::= command(L) AMP_AMP opt_nl command(R). {
2016-01-30 12:44:42 -05:00
RV = std::make_unique<and_command>(std::move(L), std::move(R));
}
2016-07-26 14:20:11 -04:00
command(RV) ::= command(L) PIPE opt_nl command(R). {
2016-01-30 12:44:42 -05:00
RV = std::make_unique<pipe_command>(std::move(L), std::move(R));
}
2016-08-05 10:34:05 -04:00
command(C) ::= term(C).
2016-01-30 12:44:42 -05:00
term(RV) ::= COMMAND(C). { RV = std::make_unique<simple_command>(std::move(C)); }
2016-02-01 20:38:29 -05:00
term(RV) ::= EVALUATE(C). { RV = std::make_unique<evaluate_command>(std::move(C)); }
term(RV) ::= BREAK(C). { RV = std::make_unique<break_command>(std::move(C)); }
term(RV) ::= CONTINUE(C). { RV = std::make_unique<continue_command>(std::move(C)); }
2016-08-05 10:34:05 -04:00
term(C) ::= if_command(C).
term(C) ::= begin_command(C).
term(C) ::= paren_command(C).
term(C) ::= loop_command(C).
term(C) ::= for_command(C).
2016-01-30 12:44:42 -05:00
2016-07-23 11:40:40 -04:00
/* lexer error (mismatched quotes, etc) */
term(RV) ::= ERROR(C). {
RV = std::make_unique<error_command>(@C, std::move(C));
}
2016-02-01 20:38:29 -05:00
/*
* fall back to an end error. w/o fallback, it will cause a parse conflict.
*/
/*
%fallback ERROR END RPAREN ELSE ELSE_IF.
2016-07-23 11:40:40 -04:00
2016-02-01 20:38:29 -05:00
*/
2016-07-19 12:37:25 -04:00
/*
term(RV) ::= error RPAREN.
term(RV) ::= error END.
term(RV) ::= LPAREN error RPAREN.
term(RV) ::= BEGIN error END.
term(RV) ::= IF error END.
term(RV) ::= LOOP error END.
term(RV) ::= FOR error END.
*/
2016-02-01 20:38:29 -05:00
/* compound list ends with a separator. paren command does not need the final separator */
2016-02-04 21:45:04 -05:00
%type paren_list { command_ptr_vector }
2016-08-05 10:34:05 -04:00
paren_list(L) ::= compound_list(L) .
2016-01-30 12:44:42 -05:00
2016-08-05 10:34:05 -04:00
paren_list(L) ::= compound_list(L) command(C) . {
L.emplace_back(std::move(C));
2016-02-01 20:38:29 -05:00
}
2016-02-04 21:45:04 -05:00
paren_command(RV) ::= LPAREN(T) paren_list(L) RPAREN(E). {
2016-02-01 20:38:29 -05:00
RV = std::make_unique<begin_command>(@T, std::move(L), std::move(T), std::move(E));
}
2016-02-04 21:45:04 -05:00
begin_command(RV) ::= BEGIN(T) sep compound_list(L) END(E). {
2016-02-01 20:38:29 -05:00
RV = std::make_unique<begin_command>(@T, std::move(L), std::move(T), std::move(E));
2016-01-30 12:44:42 -05:00
}
loop_command(RV) ::= LOOP(T) sep compound_list(L) END(E). {
RV = std::make_unique<loop_command>(@T, std::move(L), std::move(T), std::move(E));
}
2016-06-16 00:04:29 -04:00
for_command(RV) ::= FOR(T) sep compound_list(L) END(E). {
RV = std::make_unique<for_command>(@T, std::move(L), std::move(T), std::move(E));
}
2016-02-04 21:45:04 -05:00
if_command(RV) ::= IF(I) sep compound_list(L) END(E). {
2016-01-30 12:44:42 -05:00
if_command::clause_vector_type v;
v.emplace_back(std::make_unique<if_else_clause>(IF, std::move(L), std::move(I)));
RV = std::make_unique<if_command>(
std::move(v),
std::move(E)
);
}
2016-02-04 21:45:04 -05:00
if_command(RV) ::= IF(I) sep compound_list(L) else_command(EC) END(E). {
2016-01-30 12:44:42 -05:00
if_command::clause_vector_type v;
v.emplace_back(std::make_unique<if_else_clause>(IF, std::move(L), std::move(I)));
for(auto &c : EC) { v.emplace_back(std::move(c)); }
RV = std::make_unique<if_command>(
std::move(v), std::move(E));
}
%token_class else ELSE_IF ELSE.
%type else_command { if_command::clause_vector_type }
2016-02-04 21:45:04 -05:00
else_command(RV) ::= else(E) sep compound_list(L). {
2016-01-30 12:44:42 -05:00
RV.emplace_back(std::make_unique<if_else_clause>(@E, std::move(L), std::move(E)));
}
2016-02-04 21:45:04 -05:00
2016-08-05 10:34:05 -04:00
else_command(EC) ::= else_command(EC) else(E) sep compound_list(L). {
EC.emplace_back(std::make_unique<if_else_clause>(@E, std::move(L), std::move(E)));
2016-01-30 12:44:42 -05:00
}
opt_nl ::= .
2016-02-04 21:45:04 -05:00
opt_nl ::= opt_nl NL .