/* >, < 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 } %code { std::unique_ptr phase2_parser::make() { return std::make_unique(); } } %left PIPE_PIPE AMP_AMP. %left PIPE. %token_type {std::string} %default_type {command_ptr} %type start {void} %type opt_command_list {void} %type command_list {void} %type opt_command_sep {void} /* these are put into a queue for immmediate execution */ start ::= opt_command_list. opt_command_list ::= . opt_command_list ::= command_list. command_list ::= command_list opt_command(C) sep . { if (C) command_queue.emplace_back(std::move(C)); } command_list ::= opt_command(C) sep. { if (C) command_queue.emplace_back(std::move(C)); } /* command_list ::= opt_command_sep. command_list ::= command_list opt_command_sep. opt_command_sep ::= opt_command(C) sep. { if (C) command_queue.emplace_back(std::move(C)); } */ /* compound_list is identical to command_list, but it is not executed immediately. */ %type opt_compound_list { command_ptr_vector } %type compound_list { command_ptr_vector } //%type opt_paren_list { command_ptr_vector } //%type paren_list { command_ptr_vector } opt_compound_list ::= . opt_compound_list(RV) ::= compound_list(L). { RV = std::move(L); } compound_list(RV) ::= compound_list(L) opt_command(C) sep . { RV = std::move(L); if (C) RV.emplace_back(std::move(C)); } compound_list(RV) ::= opt_command(C) sep. { if (C) RV.emplace_back(std::move(C)); } /* opt_paren_list ::= . opt_paren_list(RV) ::= paren_list(L). { RV = std::move(L); } paren_list(RV) ::= opt_command(C). { if (C) RV.emplace_back(std::move(C)); } paren_list(RV) ::= paren_list(L) sep opt_command(C). { RV = std::move(L); if (C) RV.emplace_back(std::move(C)); } */ sep ::= SEMI. sep ::= NL. %type opt_command { command_ptr } opt_command(RV) ::= command(C). { RV = std::move(C); } opt_command ::= . %type command { command_ptr } /* nb -- ||, &&, | -- both sides are optional. This does not. */ command(RV) ::= command(L) PIPE_PIPE opt_nl command(R). { RV = std::make_unique(std::move(L), std::move(R)); } command(RV) ::= command(L) AMP_AMP opt_nl command(R). { RV = std::make_unique(std::move(L), std::move(R)); } /* command(RV) ::= command PIPE opt_nl command. { RV = std::make_unique(std::move(L), std::move(R)); } */ command(RV) ::= term(T). { RV = std::move(T); } term(RV) ::= COMMAND(C). { RV = std::make_unique(std::move(C)); } term(RV) ::= EVALUATE(C). { RV = std::make_unique(std::move(C)); } term(RV) ::= if_command(C). { RV = std::move(C); } term(RV) ::= begin_command(C). { RV = std::move(C); } term(RV) ::= paren_command(C). { RV = std::move(C); } /* * fall back to an end error. w/o fallback, it will cause a parse conflict. */ /* %fallback ERROR END RPAREN ELSE ELSE_IF. term(RV) ::= ERROR(C). { RV = std::make_unique(@C, std::move(C)); } */ /* opt_compound_list requires a sep after every item -- not ok for paren command! */ /* paren_command(RV) ::= LPAREN(T) opt_paren_list(L) RPAREN(E). { RV = std::make_unique(@T, std::move(L), std::move(T), std::move(E)); } */ /* compound list ends with a separator. paren command does not need the final separator */ paren_command(RV) ::= LPAREN(T) opt_command(C) RPAREN(E). { command_ptr_vector L; if (C) L.emplace_back(std::move(C)); RV = std::make_unique(@T, std::move(L), std::move(T), std::move(E)); } paren_command(RV) ::= LPAREN(T) compound_list(L) RPAREN(E). { RV = std::make_unique(@T, std::move(L), std::move(T), std::move(E)); } paren_command(RV) ::= LPAREN(T) compound_list(L) command(C) RPAREN(E). { if (C) L.emplace_back(std::move(C)); RV = std::make_unique(@T, std::move(L), std::move(T), std::move(E)); } begin_command(RV) ::= BEGIN(T) sep opt_compound_list(L) END(E). { RV = std::make_unique(@T, std::move(L), std::move(T), std::move(E)); } if_command(RV) ::= IF(I) sep opt_compound_list(L) END(E). { if_command::clause_vector_type v; v.emplace_back(std::make_unique(IF, std::move(L), std::move(I))); RV = std::make_unique( std::move(v), std::move(E) ); } if_command(RV) ::= IF(I) sep opt_compound_list(L) else_command(EC) END(E). { if_command::clause_vector_type v; v.emplace_back(std::make_unique(IF, std::move(L), std::move(I))); for(auto &c : EC) { v.emplace_back(std::move(c)); } RV = std::make_unique( std::move(v), std::move(E)); } %token_class else ELSE_IF ELSE. %type else_command { if_command::clause_vector_type } else_command(RV) ::= else(E) sep opt_compound_list(L). { RV.emplace_back(std::make_unique(@E, std::move(L), std::move(E))); } /* else_command(RV) ::= ELSE_IF(E) sep opt_compound_list(L). { RV.emplace_back(std::make_unique(@E, std::move(L), std::move(E))); } */ else_command(RV) ::= else_command(EC) else(E) sep opt_compound_list(L). { RV = std::move(EC); RV.emplace_back(std::make_unique(@E, std::move(L), std::move(E))); } /* else_command(RV) ::= else_command(EC) ELSE_IF(E) sep opt_compound_list(L). { RV = std::move(EC); RV.emplace_back(std::make_unique(std::move(E), std::move(L))); } */ opt_nl ::= . opt_nl ::= nl. nl ::= NL. nl ::= nl NL.