2016-05-18 22:30:21 +00:00
|
|
|
#include "config.h" // IWYU pragma: keep
|
|
|
|
|
2019-10-13 22:50:48 +00:00
|
|
|
#include "parse_productions.h"
|
|
|
|
|
2015-07-25 15:14:25 +00:00
|
|
|
#include <stdio.h>
|
2016-04-21 06:00:54 +00:00
|
|
|
|
|
|
|
#include "common.h"
|
2019-05-27 22:56:53 +00:00
|
|
|
#include "flog.h"
|
2016-05-02 22:48:57 +00:00
|
|
|
#include "parse_constants.h"
|
2018-01-07 21:34:04 +00:00
|
|
|
#include "parse_grammar.h"
|
2016-05-02 22:48:57 +00:00
|
|
|
#include "parse_tree.h"
|
2013-07-25 22:24:22 +00:00
|
|
|
|
|
|
|
using namespace parse_productions;
|
2018-01-07 23:53:36 +00:00
|
|
|
using namespace grammar;
|
2013-07-25 22:24:22 +00:00
|
|
|
|
2019-11-19 02:34:50 +00:00
|
|
|
#define NO_PRODUCTION nullptr
|
2013-07-28 22:19:38 +00:00
|
|
|
|
2016-05-02 22:48:57 +00:00
|
|
|
// Herein are encoded the productions for our LL2 fish grammar.
|
|
|
|
//
|
2019-10-23 01:11:13 +00:00
|
|
|
// Each symbol (e.g. symbol_job_list) has a corresponding function (e.g. resolve_job_list). The
|
|
|
|
// function accepts two tokens, representing the first and second lookahead, and returns a
|
2016-05-02 22:48:57 +00:00
|
|
|
// production representing the rule, or NULL on error. There is also a tag value which is returned
|
|
|
|
// by reference; the tag is a sort of node annotation.
|
|
|
|
//
|
|
|
|
// Productions are generally a static const array, and we return a pointer to the array (yes,
|
|
|
|
// really).
|
|
|
|
|
2018-01-07 23:53:36 +00:00
|
|
|
#define RESOLVE(SYM) \
|
|
|
|
const production_element_t *SYM::resolve( \
|
2016-05-02 22:48:57 +00:00
|
|
|
const parse_token_t &token1, const parse_token_t &token2, parse_node_tag_t *out_tag)
|
|
|
|
|
|
|
|
/// A job_list is a list of jobs, separated by semicolons or newlines.
|
|
|
|
RESOLVE(job_list) {
|
2016-10-09 21:41:56 +00:00
|
|
|
UNUSED(token2);
|
|
|
|
UNUSED(out_tag);
|
2016-11-10 05:37:49 +00:00
|
|
|
|
2016-05-02 22:48:57 +00:00
|
|
|
switch (token1.type) {
|
|
|
|
case parse_token_type_string: {
|
|
|
|
// Some keywords are special.
|
|
|
|
switch (token1.keyword) {
|
2013-07-27 06:59:12 +00:00
|
|
|
case parse_keyword_end:
|
|
|
|
case parse_keyword_else:
|
2016-05-02 22:48:57 +00:00
|
|
|
case parse_keyword_case: {
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<empty>(); // end this job list
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
|
|
|
default: {
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<normal>(); // normal string
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
2013-07-27 06:59:12 +00:00
|
|
|
}
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
2013-07-27 06:59:12 +00:00
|
|
|
case parse_token_type_pipe:
|
|
|
|
case parse_token_type_redirection:
|
2016-05-02 22:48:57 +00:00
|
|
|
case parse_token_type_background: {
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<normal>();
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
|
|
|
case parse_token_type_end: {
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<empty_line>();
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
|
|
|
case parse_token_type_terminate: {
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<empty>(); // no more commands, just transition to empty
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
2019-05-05 10:09:25 +00:00
|
|
|
default: {
|
|
|
|
return NO_PRODUCTION;
|
|
|
|
}
|
2013-07-27 06:59:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-03 02:09:16 +00:00
|
|
|
// A job decorator is AND or OR
|
|
|
|
RESOLVE(job_decorator) {
|
2019-09-08 17:42:24 +00:00
|
|
|
// If it's followed by --help, it's not a decoration.
|
|
|
|
if (token2.is_help_argument) {
|
2019-12-20 04:41:53 +00:00
|
|
|
*out_tag = parse_job_decoration_none;
|
2019-09-08 17:42:24 +00:00
|
|
|
return production_for<empty>();
|
|
|
|
}
|
2018-03-03 02:09:16 +00:00
|
|
|
|
|
|
|
switch (token1.keyword) {
|
|
|
|
case parse_keyword_and: {
|
2019-12-20 04:41:53 +00:00
|
|
|
*out_tag = parse_job_decoration_and;
|
2018-03-03 02:09:16 +00:00
|
|
|
return production_for<ands>();
|
|
|
|
}
|
|
|
|
case parse_keyword_or: {
|
2019-12-20 04:41:53 +00:00
|
|
|
*out_tag = parse_job_decoration_or;
|
2018-03-03 02:09:16 +00:00
|
|
|
return production_for<ors>();
|
|
|
|
}
|
|
|
|
default: {
|
2019-12-20 04:41:53 +00:00
|
|
|
*out_tag = parse_job_decoration_none;
|
2018-03-03 02:09:16 +00:00
|
|
|
return production_for<empty>();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-02 02:30:48 +00:00
|
|
|
RESOLVE(job_conjunction_continuation) {
|
2018-03-01 21:39:39 +00:00
|
|
|
UNUSED(token2);
|
|
|
|
UNUSED(out_tag);
|
|
|
|
switch (token1.type) {
|
|
|
|
case parse_token_type_andand:
|
2019-12-20 04:41:53 +00:00
|
|
|
*out_tag = parse_job_decoration_and;
|
2018-03-01 21:39:39 +00:00
|
|
|
return production_for<andands>();
|
|
|
|
case parse_token_type_oror:
|
2019-12-20 04:41:53 +00:00
|
|
|
*out_tag = parse_job_decoration_or;
|
2018-03-01 21:39:39 +00:00
|
|
|
return production_for<orors>();
|
|
|
|
default:
|
|
|
|
return production_for<empty>();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-02 22:48:57 +00:00
|
|
|
RESOLVE(job_continuation) {
|
2016-10-09 21:41:56 +00:00
|
|
|
UNUSED(token2);
|
2016-10-09 21:38:26 +00:00
|
|
|
UNUSED(out_tag);
|
2016-11-10 05:37:49 +00:00
|
|
|
|
2016-05-02 22:48:57 +00:00
|
|
|
switch (token1.type) {
|
|
|
|
case parse_token_type_pipe: {
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<piped>(); // pipe, continuation
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
|
|
|
default: {
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<empty>(); // not a pipe, no job continuation
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
2013-07-27 06:59:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-02 22:48:57 +00:00
|
|
|
// A statement is a normal command, or an if / while / and etc.
|
|
|
|
RESOLVE(statement) {
|
2016-10-09 21:38:26 +00:00
|
|
|
UNUSED(out_tag);
|
2016-11-10 05:37:49 +00:00
|
|
|
|
2016-05-02 22:48:57 +00:00
|
|
|
// The only block-like builtin that takes any parameters is 'function' So go to decorated
|
|
|
|
// statements if the subsequent token looks like '--'. The logic here is subtle:
|
|
|
|
//
|
|
|
|
// If we are 'begin', then we expect to be invoked with no arguments.
|
|
|
|
// If we are 'function', then we are a non-block if we are invoked with -h or --help
|
|
|
|
// If we are anything else, we require an argument, so do the same thing if the subsequent token
|
|
|
|
// is a statement terminator.
|
|
|
|
if (token1.type == parse_token_type_string) {
|
|
|
|
// If we are a function, then look for help arguments. Otherwise, if the next token looks
|
|
|
|
// like an option (starts with a dash), then parse it as a decorated statement.
|
|
|
|
if (token1.keyword == parse_keyword_function && token2.is_help_argument) {
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<decorated>();
|
2016-05-02 22:48:57 +00:00
|
|
|
} else if (token1.keyword != parse_keyword_function && token2.has_dash_prefix) {
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<decorated>();
|
2013-10-09 22:57:10 +00:00
|
|
|
}
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2016-05-02 22:48:57 +00:00
|
|
|
// Likewise if the next token doesn't look like an argument at all. This corresponds to e.g.
|
|
|
|
// a "naked if".
|
|
|
|
bool naked_invocation_invokes_help =
|
|
|
|
(token1.keyword != parse_keyword_begin && token1.keyword != parse_keyword_end);
|
|
|
|
if (naked_invocation_invokes_help &&
|
|
|
|
(token2.type == parse_token_type_end || token2.type == parse_token_type_terminate)) {
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<decorated>();
|
2013-10-12 09:46:49 +00:00
|
|
|
}
|
2013-10-09 22:57:10 +00:00
|
|
|
}
|
|
|
|
|
2016-05-02 22:48:57 +00:00
|
|
|
switch (token1.type) {
|
|
|
|
case parse_token_type_string: {
|
|
|
|
switch (token1.keyword) {
|
2018-03-05 00:06:32 +00:00
|
|
|
case parse_keyword_not:
|
|
|
|
case parse_keyword_exclam: {
|
2018-03-03 02:09:16 +00:00
|
|
|
return production_for<nots>();
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
2013-07-27 06:59:12 +00:00
|
|
|
case parse_keyword_for:
|
|
|
|
case parse_keyword_while:
|
|
|
|
case parse_keyword_function:
|
2016-05-02 22:48:57 +00:00
|
|
|
case parse_keyword_begin: {
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<block>();
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
|
|
|
case parse_keyword_if: {
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<ifs>();
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
|
|
|
case parse_keyword_else: {
|
2013-07-27 06:59:12 +00:00
|
|
|
return NO_PRODUCTION;
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
|
|
|
case parse_keyword_switch: {
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<switchs>();
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
|
|
|
case parse_keyword_end: {
|
2013-07-27 06:59:12 +00:00
|
|
|
return NO_PRODUCTION;
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
|
|
|
// All other keywords fall through to decorated statement.
|
2019-05-05 10:09:25 +00:00
|
|
|
default: {
|
|
|
|
return production_for<decorated>();
|
|
|
|
}
|
2013-07-27 06:59:12 +00:00
|
|
|
}
|
|
|
|
break;
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
2013-07-27 06:59:12 +00:00
|
|
|
case parse_token_type_pipe:
|
|
|
|
case parse_token_type_redirection:
|
|
|
|
case parse_token_type_background:
|
2016-05-02 22:48:57 +00:00
|
|
|
case parse_token_type_terminate: {
|
2013-07-27 06:59:12 +00:00
|
|
|
return NO_PRODUCTION;
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
2019-05-05 10:09:25 +00:00
|
|
|
default: {
|
|
|
|
return NO_PRODUCTION;
|
|
|
|
}
|
2013-07-27 06:59:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-02 22:48:57 +00:00
|
|
|
RESOLVE(else_clause) {
|
2016-10-09 21:41:56 +00:00
|
|
|
UNUSED(token2);
|
|
|
|
UNUSED(out_tag);
|
2016-11-10 05:37:49 +00:00
|
|
|
|
2016-05-02 22:48:57 +00:00
|
|
|
switch (token1.keyword) {
|
|
|
|
case parse_keyword_else: {
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<else_cont>();
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
2019-05-05 10:09:25 +00:00
|
|
|
default: {
|
|
|
|
return production_for<empty>();
|
|
|
|
}
|
2013-07-27 06:59:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-02 22:48:57 +00:00
|
|
|
RESOLVE(else_continuation) {
|
2016-10-09 21:41:56 +00:00
|
|
|
UNUSED(token2);
|
|
|
|
UNUSED(out_tag);
|
2015-12-15 22:59:03 +00:00
|
|
|
|
2016-05-02 22:48:57 +00:00
|
|
|
switch (token1.keyword) {
|
|
|
|
case parse_keyword_if: {
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<else_if>();
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
2019-05-05 10:09:25 +00:00
|
|
|
default: {
|
|
|
|
return production_for<else_only>();
|
|
|
|
}
|
2013-07-27 06:59:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-02 22:48:57 +00:00
|
|
|
RESOLVE(case_item_list) {
|
2016-10-09 21:41:56 +00:00
|
|
|
UNUSED(token2);
|
|
|
|
UNUSED(out_tag);
|
2016-11-10 05:37:49 +00:00
|
|
|
|
2016-05-02 22:48:57 +00:00
|
|
|
if (token1.keyword == parse_keyword_case)
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<case_items>();
|
2016-05-02 22:48:57 +00:00
|
|
|
else if (token1.type == parse_token_type_end)
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<blank_line>();
|
2016-05-02 22:48:57 +00:00
|
|
|
else
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<empty>();
|
2013-07-27 06:59:12 +00:00
|
|
|
}
|
|
|
|
|
2018-03-05 00:06:32 +00:00
|
|
|
RESOLVE(not_statement) {
|
|
|
|
UNUSED(token2);
|
|
|
|
UNUSED(out_tag);
|
|
|
|
switch (token1.keyword) {
|
|
|
|
case parse_keyword_not:
|
|
|
|
return production_for<nots>();
|
|
|
|
case parse_keyword_exclam:
|
|
|
|
return production_for<exclams>();
|
|
|
|
default:
|
|
|
|
return NO_PRODUCTION;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-02 22:48:57 +00:00
|
|
|
RESOLVE(andor_job_list) {
|
2016-10-09 21:41:56 +00:00
|
|
|
UNUSED(out_tag);
|
2016-05-02 22:48:57 +00:00
|
|
|
|
|
|
|
if (token1.type == parse_token_type_end) {
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<empty_line>();
|
2016-05-02 22:48:57 +00:00
|
|
|
} else if (token1.keyword == parse_keyword_and || token1.keyword == parse_keyword_or) {
|
|
|
|
// Check that the argument to and/or is a string that's not help. Otherwise it's either 'and
|
|
|
|
// --help' or a naked 'and', and not part of this list.
|
|
|
|
if (token2.type == parse_token_type_string && !token2.is_help_argument) {
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<andor_job>();
|
2015-12-19 22:45:45 +00:00
|
|
|
}
|
|
|
|
}
|
2016-05-02 22:48:57 +00:00
|
|
|
// All other cases end the list.
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<empty>();
|
2015-12-19 22:45:45 +00:00
|
|
|
}
|
|
|
|
|
2016-05-02 22:48:57 +00:00
|
|
|
RESOLVE(argument_list) {
|
2016-10-09 21:41:56 +00:00
|
|
|
UNUSED(token2);
|
|
|
|
UNUSED(out_tag);
|
2016-05-02 22:48:57 +00:00
|
|
|
switch (token1.type) {
|
|
|
|
case parse_token_type_string: {
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<arg>();
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
2019-05-05 10:09:25 +00:00
|
|
|
default: {
|
|
|
|
return production_for<empty>();
|
|
|
|
}
|
2013-07-27 06:59:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-02 22:48:57 +00:00
|
|
|
RESOLVE(freestanding_argument_list) {
|
2016-10-09 21:41:56 +00:00
|
|
|
UNUSED(token2);
|
|
|
|
UNUSED(out_tag);
|
2015-12-15 22:59:03 +00:00
|
|
|
|
2016-05-02 22:48:57 +00:00
|
|
|
switch (token1.type) {
|
|
|
|
case parse_token_type_string: {
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<arg>();
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
|
|
|
case parse_token_type_end: {
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<semicolon>();
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
2019-05-05 10:09:25 +00:00
|
|
|
default: {
|
|
|
|
return production_for<empty>();
|
|
|
|
}
|
2014-03-27 18:17:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-02 22:48:57 +00:00
|
|
|
RESOLVE(block_header) {
|
2016-10-09 21:41:56 +00:00
|
|
|
UNUSED(token2);
|
|
|
|
UNUSED(out_tag);
|
2015-12-15 22:59:03 +00:00
|
|
|
|
2016-05-02 22:48:57 +00:00
|
|
|
switch (token1.keyword) {
|
|
|
|
case parse_keyword_for: {
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<forh>();
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
|
|
|
case parse_keyword_while: {
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<whileh>();
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
|
|
|
case parse_keyword_function: {
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<funch>();
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
|
|
|
case parse_keyword_begin: {
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<beginh>();
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
2019-05-05 10:09:25 +00:00
|
|
|
default: {
|
|
|
|
return NO_PRODUCTION;
|
|
|
|
}
|
2013-07-27 06:59:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
Support FOO=bar syntax for passing variables to individual commands
This adds initial support for statements with prefixed variable assignments.
Statments like this are supported:
a=1 b=$a echo $b # outputs 1
Just like in other shells, the left-hand side of each assignment must
be a valid variable identifier (no quoting/escaping). Array indexing
(PATH[1]=/bin ls $PATH) is *not* yet supported, but can be added fairly
easily.
The right hand side may be any valid string token, like a command
substitution, or a brace expansion.
Since `a=* foo` is equivalent to `begin set -lx a *; foo; end`,
the assignment, like `set`, uses nullglob behavior, e.g. below command
can safely be used to check if a directory is empty.
x=/nothing/{,.}* test (count $x) -eq 0
Generic file completion is done after the equal sign, so for example
pressing tab after something like `HOME=/` completes files in the
root directory
Subcommand completion works, so something like
`GIT_DIR=repo.git and command git ` correctly calls git completions
(but the git completion does not use the variable as of now).
The variable assignment is highlighted like an argument.
Closes #6048
2019-10-23 01:13:29 +00:00
|
|
|
RESOLVE(variable_assignments) {
|
|
|
|
UNUSED(token2);
|
|
|
|
UNUSED(out_tag);
|
|
|
|
if (token1.may_be_variable_assignment) {
|
|
|
|
assert(token1.type == parse_token_type_string);
|
|
|
|
return production_for<var>();
|
|
|
|
}
|
|
|
|
return production_for<empty>();
|
|
|
|
}
|
|
|
|
|
2016-05-02 22:48:57 +00:00
|
|
|
RESOLVE(decorated_statement) {
|
2019-09-08 18:09:32 +00:00
|
|
|
// and/or are typically parsed in job_conjunction at the beginning of a job
|
|
|
|
// However they may be reached here through e.g. true && and false.
|
|
|
|
// Refuse to parse them as a command except for --help. See #6089.
|
|
|
|
if ((token1.keyword == parse_keyword_and || token1.keyword == parse_keyword_or) &&
|
|
|
|
!token2.is_help_argument) {
|
|
|
|
return NO_PRODUCTION;
|
|
|
|
}
|
|
|
|
|
2016-05-02 22:48:57 +00:00
|
|
|
// If this is e.g. 'command --help' then the command is 'command' and not a decoration. If the
|
|
|
|
// second token is not a string, then this is a naked 'command' and we should execute it as
|
|
|
|
// undecorated.
|
|
|
|
if (token2.type != parse_token_type_string || token2.has_dash_prefix) {
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<plains>();
|
2013-10-12 09:46:49 +00:00
|
|
|
}
|
2014-01-15 09:40:40 +00:00
|
|
|
|
2016-05-02 22:48:57 +00:00
|
|
|
switch (token1.keyword) {
|
|
|
|
case parse_keyword_command: {
|
2015-12-15 22:59:03 +00:00
|
|
|
*out_tag = parse_statement_decoration_command;
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<cmds>();
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
|
|
|
case parse_keyword_builtin: {
|
2015-12-15 22:59:03 +00:00
|
|
|
*out_tag = parse_statement_decoration_builtin;
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<builtins>();
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
|
|
|
case parse_keyword_exec: {
|
2015-12-15 22:59:03 +00:00
|
|
|
*out_tag = parse_statement_decoration_exec;
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<execs>();
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
|
|
|
default: {
|
|
|
|
*out_tag = parse_statement_decoration_none;
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<plains>();
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
2013-07-27 06:59:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-02 22:48:57 +00:00
|
|
|
RESOLVE(arguments_or_redirections_list) {
|
2016-10-09 21:41:56 +00:00
|
|
|
UNUSED(token2);
|
|
|
|
UNUSED(out_tag);
|
2016-11-10 05:37:49 +00:00
|
|
|
|
2016-05-02 22:48:57 +00:00
|
|
|
switch (token1.type) {
|
2013-07-27 06:59:12 +00:00
|
|
|
case parse_token_type_string:
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<arg>();
|
2018-01-22 21:18:34 +00:00
|
|
|
case parse_token_type_redirection:
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<redir>();
|
2018-01-22 21:18:34 +00:00
|
|
|
default:
|
|
|
|
return production_for<empty>();
|
2013-07-28 22:19:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-18 22:37:44 +00:00
|
|
|
RESOLVE(optional_newlines) {
|
|
|
|
UNUSED(token2);
|
2018-04-01 00:06:13 +00:00
|
|
|
UNUSED(out_tag);
|
2018-02-18 22:37:44 +00:00
|
|
|
if (token1.is_newline) return production_for<newlines>();
|
|
|
|
return production_for<empty>();
|
|
|
|
}
|
|
|
|
|
2016-05-02 22:48:57 +00:00
|
|
|
RESOLVE(optional_background) {
|
2016-10-09 21:41:56 +00:00
|
|
|
UNUSED(token2);
|
2016-11-10 05:37:49 +00:00
|
|
|
|
2016-05-02 22:48:57 +00:00
|
|
|
switch (token1.type) {
|
|
|
|
case parse_token_type_background: {
|
2015-12-15 22:59:03 +00:00
|
|
|
*out_tag = parse_background;
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<background>();
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
|
|
|
default: {
|
2015-12-15 22:59:03 +00:00
|
|
|
*out_tag = parse_no_background;
|
2018-01-07 23:53:36 +00:00
|
|
|
return production_for<empty>();
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
2013-07-28 22:19:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-21 10:45:07 +00:00
|
|
|
RESOLVE(optional_time) {
|
2020-02-23 22:42:31 +00:00
|
|
|
if (token1.keyword == parse_keyword_time && !token2.is_help_argument) {
|
|
|
|
*out_tag = parse_optional_time_time;
|
|
|
|
return production_for<time>();
|
2019-12-21 10:45:07 +00:00
|
|
|
}
|
2020-02-23 22:42:31 +00:00
|
|
|
*out_tag = parse_optional_time_no_time;
|
|
|
|
return production_for<empty>();
|
2019-12-21 10:45:07 +00:00
|
|
|
}
|
|
|
|
|
2016-11-10 05:37:49 +00:00
|
|
|
const production_element_t *parse_productions::production_for_token(parse_token_type_t node_type,
|
|
|
|
const parse_token_t &input1,
|
|
|
|
const parse_token_t &input2,
|
|
|
|
parse_node_tag_t *out_tag) {
|
2018-10-02 16:19:56 +00:00
|
|
|
// this is **extremely** chatty
|
2020-01-19 12:47:37 +00:00
|
|
|
FLOGF(parse_productions_chatty, L"Resolving production for %ls with input token <%ls>",
|
2016-10-30 02:01:19 +00:00
|
|
|
token_type_description(node_type), input1.describe().c_str());
|
2013-08-11 07:35:00 +00:00
|
|
|
|
2016-05-02 22:48:57 +00:00
|
|
|
// Fetch the function to resolve the list of productions.
|
2016-11-10 05:37:49 +00:00
|
|
|
const production_element_t *(*resolver)(const parse_token_t &input1, //!OCLINT(unused param)
|
|
|
|
const parse_token_t &input2, //!OCLINT(unused param)
|
|
|
|
parse_node_tag_t *out_tag) = //!OCLINT(unused param)
|
2019-11-19 02:34:50 +00:00
|
|
|
nullptr;
|
2016-05-02 22:48:57 +00:00
|
|
|
switch (node_type) {
|
2018-02-18 22:37:44 +00:00
|
|
|
// Handle all of our grammar elements
|
|
|
|
#define ELEM(SYM) \
|
|
|
|
case (symbol_##SYM): \
|
|
|
|
resolver = SYM::resolve; \
|
|
|
|
break;
|
|
|
|
#include "parse_grammar_elements.inc"
|
2013-08-11 07:35:00 +00:00
|
|
|
|
2018-02-18 22:37:44 +00:00
|
|
|
// Everything else is an error.
|
2013-07-28 22:19:38 +00:00
|
|
|
case parse_token_type_string:
|
|
|
|
case parse_token_type_pipe:
|
|
|
|
case parse_token_type_redirection:
|
|
|
|
case parse_token_type_background:
|
2018-03-01 21:39:39 +00:00
|
|
|
case parse_token_type_andand:
|
|
|
|
case parse_token_type_oror:
|
2013-07-28 22:19:38 +00:00
|
|
|
case parse_token_type_end:
|
2016-05-02 22:48:57 +00:00
|
|
|
case parse_token_type_terminate: {
|
2019-05-30 11:04:40 +00:00
|
|
|
FLOGF(error, L"Terminal token type %ls passed to %s", token_type_description(node_type),
|
2019-06-04 03:30:48 +00:00
|
|
|
__FUNCTION__);
|
2013-07-28 22:19:38 +00:00
|
|
|
PARSER_DIE();
|
|
|
|
break;
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
2013-08-08 22:06:46 +00:00
|
|
|
case parse_special_type_parse_error:
|
|
|
|
case parse_special_type_tokenizer_error:
|
2016-05-02 22:48:57 +00:00
|
|
|
case parse_special_type_comment: {
|
2019-05-30 11:04:40 +00:00
|
|
|
FLOGF(error, L"Special type %ls passed to %s\n", token_type_description(node_type),
|
2019-06-04 03:30:48 +00:00
|
|
|
__FUNCTION__);
|
2013-08-08 22:06:46 +00:00
|
|
|
PARSER_DIE();
|
|
|
|
break;
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
|
|
|
case token_type_invalid: {
|
2019-05-30 11:04:40 +00:00
|
|
|
FLOGF(error, L"token_type_invalid passed to %s", __FUNCTION__);
|
2013-07-28 22:19:38 +00:00
|
|
|
PARSER_DIE();
|
|
|
|
break;
|
2016-05-02 22:48:57 +00:00
|
|
|
}
|
2013-07-28 22:19:38 +00:00
|
|
|
}
|
2019-11-19 02:34:50 +00:00
|
|
|
PARSE_ASSERT(resolver != nullptr);
|
2013-08-11 07:35:00 +00:00
|
|
|
|
2016-11-10 05:37:49 +00:00
|
|
|
const production_element_t *result = resolver(input1, input2, out_tag);
|
2019-11-19 02:34:50 +00:00
|
|
|
if (result == nullptr) {
|
2020-01-19 12:47:37 +00:00
|
|
|
FLOGF(parse_productions, L"Node type '%ls' has no production for input '%ls' (in %s)",
|
2016-10-30 02:01:19 +00:00
|
|
|
token_type_description(node_type), input1.describe().c_str(), __FUNCTION__);
|
2013-07-28 22:19:38 +00:00
|
|
|
}
|
2015-12-15 22:59:03 +00:00
|
|
|
|
2013-07-28 22:19:38 +00:00
|
|
|
return result;
|
|
|
|
}
|