mirror of
https://github.com/fish-shell/fish-shell
synced 2024-12-26 12:53:13 +00:00
Stuff
This commit is contained in:
parent
54f683fc6b
commit
d54346b205
4 changed files with 632 additions and 60 deletions
614
expression.cpp
614
expression.cpp
|
@ -1,9 +1,607 @@
|
||||||
//
|
|
||||||
// expression.cpp
|
|
||||||
// fish
|
|
||||||
//
|
|
||||||
// Created by Peter Ammon on 5/25/13.
|
|
||||||
//
|
|
||||||
//
|
|
||||||
|
|
||||||
#include "expression.h"
|
#include "expression.h"
|
||||||
|
#include <stack>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
/* Fish grammar:
|
||||||
|
|
||||||
|
# A statement_list is a list of statements, separated by semicolons or newlines
|
||||||
|
|
||||||
|
statement_list = <empty> |
|
||||||
|
statement statement_list
|
||||||
|
|
||||||
|
# A statement is a normal job, or an if / while / and etc.
|
||||||
|
|
||||||
|
statement = boolean_statement | block_statement | decorated_statement
|
||||||
|
|
||||||
|
# A block is a conditional, loop, or begin/end
|
||||||
|
|
||||||
|
block_statement = block_header statement_list END arguments_or_redirections_list
|
||||||
|
block_header = if_header | for_header | while_header | function_header | begin_header
|
||||||
|
if_header = IF statement
|
||||||
|
for_header = FOR var_name IN arguments_or_redirections_list STATEMENT_TERMINATOR
|
||||||
|
while_header = WHILE statement
|
||||||
|
begin_header = BEGIN STATEMENT_TERMINATOR
|
||||||
|
function_header = FUNCTION arguments_or_redirections_list STATEMENT_TERMINATOR
|
||||||
|
|
||||||
|
# A boolean statement is AND or OR or NOT
|
||||||
|
|
||||||
|
boolean_statement = AND statement | OR statement | NOT statement
|
||||||
|
|
||||||
|
# A decorated_statement is a command with a list of arguments_or_redirections, possibly with "builtin" or "command"
|
||||||
|
|
||||||
|
decorated_statement = COMMAND plain_statement | BUILTIN plain_statement | plain_statement
|
||||||
|
plain_statement = command arguments_or_redirections_list terminator
|
||||||
|
|
||||||
|
arguments_or_redirections_list = <empty> |
|
||||||
|
argument_or_redirection arguments_or_redirections_list
|
||||||
|
argument_or_redirection = redirection | <TOK_STRING>
|
||||||
|
redirection = <TOK_REDIRECTION>
|
||||||
|
|
||||||
|
terminator = <TOK_END> | <TOK_BACKGROUND>
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define PARSE_ASSERT(a) assert(a)
|
||||||
|
|
||||||
|
#define PARSER_DIE() assert(0)
|
||||||
|
|
||||||
|
#if 1
|
||||||
|
class parse_command_t;
|
||||||
|
|
||||||
|
enum parse_token_type_t
|
||||||
|
{
|
||||||
|
token_type_invalid,
|
||||||
|
|
||||||
|
// Non-terminal tokens
|
||||||
|
symbol_statement_list,
|
||||||
|
symbol_statement,
|
||||||
|
symbol_block_statement,
|
||||||
|
symbol_block_header,
|
||||||
|
symbol_if_header,
|
||||||
|
symbol_for_header,
|
||||||
|
symbol_while_header,
|
||||||
|
symbol_begin_header,
|
||||||
|
symbol_function_header,
|
||||||
|
symbol_boolean_statement,
|
||||||
|
symbol_decorated_statement,
|
||||||
|
symbol_plain_statement,
|
||||||
|
symbol_arguments_or_redirections_list,
|
||||||
|
symbol_argument_or_redirection,
|
||||||
|
|
||||||
|
|
||||||
|
// Terminal types
|
||||||
|
parse_token_type_string,
|
||||||
|
parse_token_type_pipe,
|
||||||
|
parse_token_type_redirection,
|
||||||
|
parse_token_background,
|
||||||
|
parse_token_type_end,
|
||||||
|
parse_token_type_terminate,
|
||||||
|
};
|
||||||
|
|
||||||
|
enum parse_keyword_t
|
||||||
|
{
|
||||||
|
parse_keyword_none,
|
||||||
|
parse_keyword_if,
|
||||||
|
parse_keyword_else,
|
||||||
|
parse_keyword_for,
|
||||||
|
parse_keyword_in,
|
||||||
|
parse_keyword_while,
|
||||||
|
parse_keyword_begin,
|
||||||
|
parse_keyword_function,
|
||||||
|
parse_keyword_switch,
|
||||||
|
parse_keyword_end,
|
||||||
|
parse_keyword_and,
|
||||||
|
parse_keyword_or,
|
||||||
|
parse_keyword_not,
|
||||||
|
parse_keyword_command,
|
||||||
|
parse_keyword_builtin
|
||||||
|
};
|
||||||
|
|
||||||
|
struct parse_stack_element_t
|
||||||
|
{
|
||||||
|
enum parse_token_type_t type;
|
||||||
|
enum parse_keyword_t keyword;
|
||||||
|
|
||||||
|
// Construct a token type, with no keyword
|
||||||
|
parse_stack_element_t(enum parse_token_type_t t) : type(t), keyword(parse_keyword_none)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
// Construct a string type from a keyword
|
||||||
|
parse_stack_element_t(enum parse_keyword_t k) : type(parse_token_type_string), keyword(k)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
struct parse_token_t
|
||||||
|
{
|
||||||
|
enum parse_token_type_t type; // The type of the token as represnted by the parser
|
||||||
|
enum token_type tokenizer_type; // The type of the token as represented by the tokenizer
|
||||||
|
enum parse_keyword_t keyword; // Any keyword represented by this parser
|
||||||
|
size_t source_start;
|
||||||
|
size_t source_end;
|
||||||
|
};
|
||||||
|
|
||||||
|
// Convert from tokenizer_t's token type to our token
|
||||||
|
static parse_token_t parse_token_from_tokenizer_token(enum token_type tokenizer_token_type)
|
||||||
|
{
|
||||||
|
parse_token_t result = {};
|
||||||
|
result.tokenizer_type = tokenizer_token_type;
|
||||||
|
switch (tokenizer_token_type)
|
||||||
|
{
|
||||||
|
case TOK_STRING:
|
||||||
|
result.type = parse_token_type_string;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case TOK_PIPE:
|
||||||
|
result.type = parse_token_type_pipe;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case TOK_END:
|
||||||
|
result.type = parse_token_type_end;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case TOK_BACKGROUND:
|
||||||
|
result.type = parse_token_background;
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
fprintf(stderr, "Bad token type %d passed to %s\n", (int)tokenizer_token_type, __FUNCTION__);
|
||||||
|
assert(0);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Root of a parse tree */
|
||||||
|
class parse_statement_list_t;
|
||||||
|
class parse_tree_t
|
||||||
|
{
|
||||||
|
friend class parse_ll_t;
|
||||||
|
|
||||||
|
parse_statement_list_t *root;
|
||||||
|
};
|
||||||
|
|
||||||
|
/** Base class for nodes of a parse tree */
|
||||||
|
class parse_node_base_t
|
||||||
|
{
|
||||||
|
/* Backreference to the tree */
|
||||||
|
parse_tree_t * const tree;
|
||||||
|
|
||||||
|
/* Type of the node */
|
||||||
|
const enum parse_token_type_t type;
|
||||||
|
|
||||||
|
/* Start in the source code */
|
||||||
|
const unsigned int source_start;
|
||||||
|
|
||||||
|
/* Length of our range in the source code */
|
||||||
|
const unsigned int source_length;
|
||||||
|
|
||||||
|
public:
|
||||||
|
parse_node_base_t(parse_tree_t *tr, parse_token_type_t ty) : tree(tr), type(ty), source_start(0), source_length(0)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual ~parse_node_base_t()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
class parse_statement_t;
|
||||||
|
class parse_statement_list_t : public parse_node_base_t
|
||||||
|
{
|
||||||
|
std::vector<parse_statement_t *> statements; //deleted by destructor
|
||||||
|
public:
|
||||||
|
parse_statement_list_t(parse_tree_t *t) : parse_node_base_t(t, symbol_statement_list)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
class parse_statement_t : public parse_node_base_t
|
||||||
|
{
|
||||||
|
// abstract class
|
||||||
|
|
||||||
|
public:
|
||||||
|
parse_statement_t(parse_tree_t *t, parse_token_type_t ty) : parse_node_base_t(t, ty)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
class parse_boolean_statement_t : public parse_statement_t
|
||||||
|
{
|
||||||
|
enum {
|
||||||
|
boolean_and,
|
||||||
|
boolean_or,
|
||||||
|
boolean_not
|
||||||
|
};
|
||||||
|
parse_statement_t *subject;
|
||||||
|
|
||||||
|
parse_boolean_statement_t(parse_tree_t *t) : subject(NULL), parse_statement_t(t, symbol_boolean_statement)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
class parse_plain_statement_t;
|
||||||
|
class parse_decorated_statement_t : public parse_statement_t
|
||||||
|
{
|
||||||
|
enum {
|
||||||
|
decoration_command,
|
||||||
|
decoration_builtin
|
||||||
|
} decoration;
|
||||||
|
|
||||||
|
parse_plain_statement_t *subject;
|
||||||
|
|
||||||
|
parse_decorated_statement_t(parse_tree_t *t) : subject(NULL), parse_statement_t(t, symbol_decorated_statement)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
class parse_plain_statement_t : public parse_statement_t
|
||||||
|
{
|
||||||
|
wcstring_list_t arguments;
|
||||||
|
wcstring_list_t redirections;
|
||||||
|
|
||||||
|
parse_plain_statement_t(parse_tree_t *t) : parse_statement_t(t, symbol_plain_statement)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
class parse_block_statement_t : public parse_statement_t
|
||||||
|
{
|
||||||
|
// abstract class
|
||||||
|
parse_block_statement_t(parse_tree_t *t, parse_token_type_t ty) : parse_statement_t(t, ty)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
class parse_ll_t
|
||||||
|
{
|
||||||
|
friend class parse_t;
|
||||||
|
|
||||||
|
std::stack<parse_stack_element_t> symbol_stack; // LL parser stack
|
||||||
|
std::stack<parse_node_base_t *> node_stack; // stack of nodes we are constructing; owned by the tree (not by us!)
|
||||||
|
parse_tree_t *tree; //tree we are constructing
|
||||||
|
|
||||||
|
// Constructor
|
||||||
|
parse_ll_t()
|
||||||
|
{
|
||||||
|
this->tree = new parse_tree_t();
|
||||||
|
tree->root = new parse_statement_list_t(this->tree);;
|
||||||
|
|
||||||
|
symbol_stack.push(symbol_statement_list); // goal token
|
||||||
|
node_stack.push(tree->root); //outermost node
|
||||||
|
}
|
||||||
|
|
||||||
|
// implementation of certain parser constructions
|
||||||
|
void accept_token(parse_token_t token);
|
||||||
|
void accept_token_statement_list(parse_token_t token);
|
||||||
|
void accept_token_statement(parse_token_t token);
|
||||||
|
void accept_token_block_header(parse_token_t token);
|
||||||
|
void accept_token_boolean_statement(parse_token_t token);
|
||||||
|
void accept_token_decorated_statement(parse_token_t token);
|
||||||
|
void accept_token_arguments_or_redirections_list(parse_token_t token);
|
||||||
|
void accept_token_argument_or_redirection(parse_token_t token);
|
||||||
|
|
||||||
|
void token_unhandled(parse_token_t token, const char *function);
|
||||||
|
|
||||||
|
void parse_error(const wchar_t *expected, parse_token_t token);
|
||||||
|
|
||||||
|
parse_token_type_t stack_top_type() const
|
||||||
|
{
|
||||||
|
return symbol_stack.top().type;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Pop from the top of the symbol stack, then push. Note that these are pushed in reverse order, so the first argument will be on the top of the stack
|
||||||
|
inline void symbol_stack_pop_push(parse_stack_element_t tok1 = token_type_invalid, parse_stack_element_t tok2 = token_type_invalid, parse_stack_element_t tok3 = token_type_invalid, parse_stack_element_t tok4 = token_type_invalid, parse_stack_element_t tok5 = token_type_invalid)
|
||||||
|
{
|
||||||
|
symbol_stack.pop();
|
||||||
|
if (tok5.type != token_type_invalid) symbol_stack.push(tok5);
|
||||||
|
if (tok4.type != token_type_invalid) symbol_stack.push(tok4);
|
||||||
|
if (tok3.type != token_type_invalid) symbol_stack.push(tok3);
|
||||||
|
if (tok2.type != token_type_invalid) symbol_stack.push(tok2);
|
||||||
|
if (tok1.type != token_type_invalid) symbol_stack.push(tok1);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
void parse_ll_t::token_unhandled(parse_token_t token, const char *function)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "Unhandled token with type %d in function %s\n", (int)token.type, function);
|
||||||
|
PARSER_DIE();
|
||||||
|
}
|
||||||
|
|
||||||
|
void parse_ll_t::parse_error(const wchar_t *expected, parse_token_t token)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "Expected a %ls, instead got a token of type %d\n", expected, (int)token.type);
|
||||||
|
}
|
||||||
|
|
||||||
|
void parse_ll_t::accept_token_statement_list(parse_token_t token)
|
||||||
|
{
|
||||||
|
PARSE_ASSERT(symbol_stack.top().type == symbol_statement_list);
|
||||||
|
switch (token.type)
|
||||||
|
{
|
||||||
|
case parse_token_type_string:
|
||||||
|
case parse_token_type_pipe:
|
||||||
|
case parse_token_type_redirection:
|
||||||
|
case parse_token_background:
|
||||||
|
case parse_token_type_end:
|
||||||
|
symbol_stack_pop_push(symbol_statement, symbol_statement_list);
|
||||||
|
|
||||||
|
break;
|
||||||
|
|
||||||
|
case parse_token_type_terminate:
|
||||||
|
// no more commands, just transition to empty
|
||||||
|
symbol_stack_pop_push();
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
token_unhandled(token, __FUNCTION__);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void parse_ll_t::accept_token_statement(parse_token_t token)
|
||||||
|
{
|
||||||
|
PARSE_ASSERT(stack_top_type() == symbol_statement);
|
||||||
|
switch (token.type)
|
||||||
|
{
|
||||||
|
case parse_token_type_string:
|
||||||
|
switch (token.keyword)
|
||||||
|
{
|
||||||
|
case parse_keyword_and:
|
||||||
|
case parse_keyword_or:
|
||||||
|
case parse_keyword_not:
|
||||||
|
symbol_stack_pop_push(symbol_boolean_statement);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case parse_keyword_if:
|
||||||
|
case parse_keyword_else:
|
||||||
|
case parse_keyword_for:
|
||||||
|
case parse_keyword_in:
|
||||||
|
case parse_keyword_while:
|
||||||
|
case parse_keyword_begin:
|
||||||
|
case parse_keyword_function:
|
||||||
|
case parse_keyword_switch:
|
||||||
|
symbol_stack_pop_push(symbol_block_statement);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case parse_keyword_end:
|
||||||
|
// TODO
|
||||||
|
break;
|
||||||
|
|
||||||
|
case parse_keyword_none:
|
||||||
|
case parse_keyword_command:
|
||||||
|
case parse_keyword_builtin:
|
||||||
|
symbol_stack_pop_push(symbol_decorated_statement);
|
||||||
|
break;
|
||||||
|
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case parse_token_type_pipe:
|
||||||
|
case parse_token_type_redirection:
|
||||||
|
case parse_token_background:
|
||||||
|
case parse_token_type_end:
|
||||||
|
case parse_token_type_terminate:
|
||||||
|
parse_error(L"command", token);
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
token_unhandled(token, __FUNCTION__);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void parse_ll_t::accept_token_block_header(parse_token_t token)
|
||||||
|
{
|
||||||
|
PARSE_ASSERT(stack_top_type() == symbol_block_header);
|
||||||
|
switch (token.type)
|
||||||
|
{
|
||||||
|
case parse_token_type_string:
|
||||||
|
switch (token.keyword)
|
||||||
|
{
|
||||||
|
case parse_keyword_if:
|
||||||
|
symbol_stack_pop_push(symbol_if_header);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case parse_keyword_else:
|
||||||
|
//todo
|
||||||
|
break;
|
||||||
|
|
||||||
|
case parse_keyword_for:
|
||||||
|
symbol_stack_pop_push(symbol_for_header);
|
||||||
|
break;
|
||||||
|
|
||||||
|
|
||||||
|
case parse_keyword_while:
|
||||||
|
symbol_stack_pop_push(symbol_while_header);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case parse_keyword_begin:
|
||||||
|
symbol_stack_pop_push(symbol_begin_header);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case parse_keyword_function:
|
||||||
|
symbol_stack_pop_push(symbol_function_header);
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
token_unhandled(token, __FUNCTION__);
|
||||||
|
break;
|
||||||
|
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
token_unhandled(token, __FUNCTION__);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void parse_ll_t::accept_token_boolean_statement(parse_token_t token)
|
||||||
|
{
|
||||||
|
PARSE_ASSERT(stack_top_type() == symbol_boolean_statement);
|
||||||
|
switch (token.type)
|
||||||
|
{
|
||||||
|
case parse_token_type_string:
|
||||||
|
switch (token.keyword)
|
||||||
|
{
|
||||||
|
case parse_keyword_and:
|
||||||
|
symbol_stack_pop_push(parse_keyword_and, symbol_statement);
|
||||||
|
break;
|
||||||
|
case parse_keyword_or:
|
||||||
|
symbol_stack_pop_push(parse_keyword_or, symbol_statement);
|
||||||
|
break;
|
||||||
|
case parse_keyword_not:
|
||||||
|
symbol_stack_pop_push(parse_keyword_not, symbol_statement);
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
token_unhandled(token, __FUNCTION__);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
default:
|
||||||
|
token_unhandled(token, __FUNCTION__);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void parse_ll_t::accept_token_decorated_statement(parse_token_t token)
|
||||||
|
{
|
||||||
|
PARSE_ASSERT(stack_top_type() == symbol_decorated_statement);
|
||||||
|
switch (token.type)
|
||||||
|
{
|
||||||
|
case parse_token_type_string:
|
||||||
|
switch (token.keyword)
|
||||||
|
{
|
||||||
|
case parse_keyword_command:
|
||||||
|
symbol_stack_pop_push(parse_keyword_command, symbol_statement);
|
||||||
|
break;
|
||||||
|
case parse_keyword_builtin:
|
||||||
|
symbol_stack_pop_push(parse_keyword_builtin, symbol_statement);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
symbol_stack_pop_push(symbol_plain_statement);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
default:
|
||||||
|
token_unhandled(token, __FUNCTION__);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void parse_ll_t::accept_token_arguments_or_redirections_list(parse_token_t token)
|
||||||
|
{
|
||||||
|
PARSE_ASSERT(stack_top_type() == symbol_arguments_or_redirections_list);
|
||||||
|
switch (token.type)
|
||||||
|
{
|
||||||
|
case parse_token_type_string:
|
||||||
|
case parse_token_type_redirection:
|
||||||
|
symbol_stack_pop_push(symbol_argument_or_redirection, symbol_arguments_or_redirections_list);
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
// Some other token, end of list
|
||||||
|
symbol_stack_pop_push();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void parse_ll_t::accept_token_argument_or_redirection(parse_token_t token)
|
||||||
|
{
|
||||||
|
PARSE_ASSERT(stack_top_type() == symbol_argument_or_redirection);
|
||||||
|
switch (token.type)
|
||||||
|
{
|
||||||
|
case parse_token_type_string:
|
||||||
|
symbol_stack_pop_push();
|
||||||
|
// Got an argument
|
||||||
|
break;
|
||||||
|
|
||||||
|
case parse_token_type_redirection:
|
||||||
|
symbol_stack_pop_push();
|
||||||
|
// Got a redirection
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
token_unhandled(token, __FUNCTION__);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void parse_ll_t::accept_token(parse_token_t token)
|
||||||
|
{
|
||||||
|
assert(! symbol_stack.empty());
|
||||||
|
switch (stack_top_type())
|
||||||
|
{
|
||||||
|
case symbol_statement_list:
|
||||||
|
accept_token_statement_list(token);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case symbol_statement:
|
||||||
|
accept_token_statement(token);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case symbol_block_statement:
|
||||||
|
symbol_stack_pop_push(symbol_block_header, symbol_statement_list, parse_keyword_end, symbol_arguments_or_redirections_list);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case symbol_block_header:
|
||||||
|
accept_token_block_header(token);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case symbol_if_header:
|
||||||
|
break;
|
||||||
|
|
||||||
|
case symbol_for_header:
|
||||||
|
symbol_stack_pop_push(parse_keyword_for, parse_token_type_string, parse_keyword_in, symbol_arguments_or_redirections_list, parse_token_type_end);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case symbol_while_header:
|
||||||
|
symbol_stack_pop_push(parse_keyword_while, symbol_statement);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case symbol_begin_header:
|
||||||
|
symbol_stack_pop_push(parse_keyword_begin, parse_token_type_end);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case symbol_function_header:
|
||||||
|
symbol_stack_pop_push(parse_keyword_function, symbol_arguments_or_redirections_list, parse_token_type_end);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case symbol_boolean_statement:
|
||||||
|
accept_token_boolean_statement(token);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case symbol_decorated_statement:
|
||||||
|
accept_token_decorated_statement(token);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case symbol_plain_statement:
|
||||||
|
symbol_stack_pop_push(parse_token_type_string, symbol_arguments_or_redirections_list, parse_token_type_end);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case symbol_arguments_or_redirections_list:
|
||||||
|
accept_token_arguments_or_redirections_list(token);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case symbol_argument_or_redirection:
|
||||||
|
accept_token_argument_or_redirection(token);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
class parse_sr_t
|
||||||
|
{
|
||||||
|
friend class parse_t;
|
||||||
|
|
||||||
|
std::vector<parse_node_base_t *> node_stack;
|
||||||
|
void accept_token(parse_token_t token);
|
||||||
|
};
|
||||||
|
|
||||||
|
parse_t::parse_t() : parser(new parse_sr_t())
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
75
expression.h
75
expression.h
|
@ -12,13 +12,23 @@
|
||||||
#include "config.h"
|
#include "config.h"
|
||||||
#include "util.h"
|
#include "util.h"
|
||||||
#include "common.h"
|
#include "common.h"
|
||||||
|
#include "tokenizer.h"
|
||||||
|
|
||||||
|
|
||||||
|
class parse_ll_t;
|
||||||
|
class parse_sr_t;
|
||||||
|
class parse_t
|
||||||
|
{
|
||||||
|
parse_sr_t * const parser;
|
||||||
|
parse_t();
|
||||||
|
};
|
||||||
|
|
||||||
/* Fish grammar:
|
/* Fish grammar:
|
||||||
|
|
||||||
# A statement_list is a list of statements, separated by semicolons or newlines
|
# A statement_list is a list of statements, separated by semicolons or newlines
|
||||||
|
|
||||||
statement_list = <empty> | statement | statement statement_list
|
statement_list = <empty> |
|
||||||
|
statement statement_list
|
||||||
|
|
||||||
# A statement is a normal job, or an if / while / and etc.
|
# A statement is a normal job, or an if / while / and etc.
|
||||||
|
|
||||||
|
@ -43,61 +53,24 @@
|
||||||
decorated_statement = COMMAND plain_statement | BUILTIN plain_statement | plain_statement
|
decorated_statement = COMMAND plain_statement | BUILTIN plain_statement | plain_statement
|
||||||
plain_statement = command arguments_or_redirections_list terminator
|
plain_statement = command arguments_or_redirections_list terminator
|
||||||
|
|
||||||
arguments_or_redirections_list = <empty> | argument_or_redirection
|
arguments_or_redirections_list = <empty> |
|
||||||
argument_or_redirection = redirection | <STRING>
|
argument_or_redirection arguments_or_redirections_list
|
||||||
redirection = REDIRECTION
|
argument_or_redirection = redirection | <TOK_STRING>
|
||||||
|
redirection = <TOK_REDIRECTION>
|
||||||
|
|
||||||
|
terminator = <TOK_END> | <TOK_BACKGROUND>
|
||||||
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|
||||||
class parse_command_t;
|
/* fish Shift-Reduce grammar:
|
||||||
|
|
||||||
/** Root of a parse tree */
|
|
||||||
class parse_tree_t
|
IF <- if_statement
|
||||||
{
|
FOR <- for_statement
|
||||||
/** Literal source code */
|
|
||||||
wcstring source;
|
|
||||||
|
|
||||||
/** Initial node */
|
|
||||||
parse_command_list_t *child;
|
|
||||||
};
|
|
||||||
|
|
||||||
/** Base class for nodes of a parse tree */
|
*/
|
||||||
class parse_node_base_t
|
|
||||||
{
|
|
||||||
/* Backreference to the tree */
|
|
||||||
parse_tree_t * const tree;
|
|
||||||
|
|
||||||
/* Start in the source code */
|
|
||||||
const unsigned int source_start;
|
|
||||||
|
|
||||||
/* Length of our range in the source code */
|
|
||||||
const unsigned int source_length;
|
|
||||||
};
|
|
||||||
|
|
||||||
class parse_statement_list_t : public parse_node_base_t
|
|
||||||
{
|
|
||||||
std::vector<parse_statement_t *> statements;
|
|
||||||
};
|
|
||||||
|
|
||||||
class parse_statement_t : public parse_node_base_t
|
|
||||||
{
|
|
||||||
|
|
||||||
};
|
|
||||||
|
|
||||||
class parse_boolean_statement_t : public parse_statement_t
|
|
||||||
{
|
|
||||||
|
|
||||||
};
|
|
||||||
|
|
||||||
class parse_plain_statement_t : public parse_statement_t
|
|
||||||
{
|
|
||||||
|
|
||||||
};
|
|
||||||
|
|
||||||
class parse_block_statement_t : public parse_statement_t
|
|
||||||
{
|
|
||||||
|
|
||||||
};
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -1311,6 +1311,7 @@ void parser_t::parse_job_argument_list(process_t *p,
|
||||||
case TOK_BACKGROUND:
|
case TOK_BACKGROUND:
|
||||||
{
|
{
|
||||||
job_set_flag(j, JOB_FOREGROUND, 0);
|
job_set_flag(j, JOB_FOREGROUND, 0);
|
||||||
|
// PCA note fall through, this is deliberate. The background modifier & terminates a command
|
||||||
}
|
}
|
||||||
|
|
||||||
case TOK_END:
|
case TOK_END:
|
||||||
|
|
|
@ -22,7 +22,7 @@ enum token_type
|
||||||
TOK_INVALID,/**< Invalid token */
|
TOK_INVALID,/**< Invalid token */
|
||||||
TOK_STRING,/**< String token */
|
TOK_STRING,/**< String token */
|
||||||
TOK_PIPE,/**< Pipe token */
|
TOK_PIPE,/**< Pipe token */
|
||||||
TOK_END,/**< End token */
|
TOK_END,/**< End token (semicolon or newline, not literal end) */
|
||||||
TOK_REDIRECT_OUT, /**< redirection token */
|
TOK_REDIRECT_OUT, /**< redirection token */
|
||||||
TOK_REDIRECT_APPEND,/**< redirection append token */
|
TOK_REDIRECT_APPEND,/**< redirection append token */
|
||||||
TOK_REDIRECT_IN,/**< input redirection token */
|
TOK_REDIRECT_IN,/**< input redirection token */
|
||||||
|
|
Loading…
Reference in a new issue