Remove header_node_for_block_statement

This commit is contained in:
ridiculousfish 2018-01-15 18:41:14 -08:00
parent d7c28c9316
commit c1b60fa8e1
3 changed files with 35 additions and 54 deletions

View file

@ -1377,18 +1377,6 @@ bool parse_node_tree_t::statement_is_in_pipeline(const parse_node_t &node,
return result; return result;
} }
const parse_node_t *parse_node_tree_t::header_node_for_block_statement(
const parse_node_t &node) const {
const parse_node_t *result = NULL;
if (node.type == symbol_block_statement) {
const parse_node_t *block_header = this->get_child(node, 0, symbol_block_header);
if (block_header != NULL) {
result = this->get_child(*block_header, 0);
}
}
return result;
}
parse_node_tree_t::parse_node_list_t parse_node_tree_t::comment_nodes_for_node( parse_node_tree_t::parse_node_list_t parse_node_tree_t::comment_nodes_for_node(
const parse_node_t &parent) const { const parse_node_t &parent) const {
parse_node_list_t result; parse_node_list_t result;

View file

@ -200,10 +200,12 @@ class parse_node_tree_t : public std::vector<parse_node_t> {
/// only the second or additional commands are. /// only the second or additional commands are.
bool statement_is_in_pipeline(const parse_node_t &node, bool include_first) const; bool statement_is_in_pipeline(const parse_node_t &node, bool include_first) const;
/// If the given node is a block statement, returns the header node (for_header, while_header, /// Given a node, return all of its comment nodes.
/// begin_header, or function_header). Otherwise returns NULL. parse_node_list_t comment_nodes_for_node(const parse_node_t &node) const;
const parse_node_t *header_node_for_block_statement(const parse_node_t &node) const;
private:
template <typename Type>
friend class tnode_t;
/// Given a node list (e.g. of type symbol_job_list) and a node type (e.g. symbol_job), return /// Given a node list (e.g. of type symbol_job_list) and a node type (e.g. symbol_job), return
/// the next element of the given type in that list, and the tail (by reference). Returns NULL /// the next element of the given type in that list, and the tail (by reference). Returns NULL
/// if we've exhausted the list. /// if we've exhausted the list.
@ -211,10 +213,6 @@ class parse_node_tree_t : public std::vector<parse_node_t> {
parse_token_type_t item_type, parse_token_type_t item_type,
const parse_node_t **list_tail) const; const parse_node_t **list_tail) const;
/// Given a node, return all of its comment nodes.
parse_node_list_t comment_nodes_for_node(const parse_node_t &node) const;
private:
// Finds the last node of a given type underneath a given node, or NULL if it could not be // Finds the last node of a given type underneath a given node, or NULL if it could not be
// found. If parent is NULL, this finds the last node in the tree of that type. // found. If parent is NULL, this finds the last node in the tree of that type.
const parse_node_t *find_last_node_of_type(parse_token_type_t type, const parse_node_t *find_last_node_of_type(parse_token_type_t type,
@ -266,6 +264,13 @@ class tnode_t {
assert((!n || n->type == Type::token) && "node has wrong type"); assert((!n || n->type == Type::token) && "node has wrong type");
} }
// Try to create a tnode from the given tree and parse node.
// Returns an empty node if the parse node is null, or has the wrong type.
static tnode_t try_create(const parse_node_tree_t *tree, const parse_node_t *node) {
assert(tree && "tree cannot be null");
return tnode_t(tree, node && node->type == Type::token ? node : nullptr);
}
/// Temporary conversion to parse_node_t to assist in migration. /// Temporary conversion to parse_node_t to assist in migration.
/* implicit */ operator const parse_node_t &() const { /* implicit */ operator const parse_node_t &() const {
assert(nodeptr && "Empty tnode_t"); assert(nodeptr && "Empty tnode_t");

View file

@ -1183,6 +1183,7 @@ parser_test_error_bits_t parse_util_detect_errors(const wcstring &buff_src,
if (maybe_t<wcstring> mcommand = if (maybe_t<wcstring> mcommand =
command_for_plain_statement({&node_tree, &node}, buff_src)) { command_for_plain_statement({&node_tree, &node}, buff_src)) {
using namespace grammar;
wcstring command = std::move(*mcommand); wcstring command = std::move(*mcommand);
// Check that we can expand the command. // Check that we can expand the command.
if (!expand_one(command, if (!expand_one(command,
@ -1202,17 +1203,16 @@ parser_test_error_bits_t parse_util_detect_errors(const wcstring &buff_src,
// Check that we don't return from outside a function. But we allow it if it's // Check that we don't return from outside a function. But we allow it if it's
// 'return --help'. // 'return --help'.
if (!errored && command == L"return") { if (!errored && command == L"return") {
const parse_node_t *ancestor = &node;
bool found_function = false; bool found_function = false;
while (ancestor != NULL) { for (const parse_node_t *ancestor = &node; ancestor != nullptr;
const parse_node_t *possible_function_header = ancestor = node_tree.get_parent(*ancestor)) {
node_tree.header_node_for_block_statement(*ancestor); auto fh = tnode_t<block_statement>::try_create(&node_tree, ancestor)
if (possible_function_header != NULL && .child<0>()
possible_function_header->type == symbol_function_header) { .try_get_child<function_header, 0>();
if (fh) {
found_function = true; found_function = true;
break; break;
} }
ancestor = node_tree.get_parent(*ancestor);
} }
if (!found_function && !first_argument_is_help(node_tree, node, buff_src)) { if (!found_function && !first_argument_is_help(node_tree, node, buff_src)) {
errored = append_syntax_error(&parse_errors, node.source_start, errored = append_syntax_error(&parse_errors, node.source_start,
@ -1227,35 +1227,23 @@ parser_test_error_bits_t parse_util_detect_errors(const wcstring &buff_src,
// This is a little funny because we can't tell if it's a 'for' or 'while' // This is a little funny because we can't tell if it's a 'for' or 'while'
// loop from the ancestor alone; we need the header. That is, we hit a // loop from the ancestor alone; we need the header. That is, we hit a
// block_statement, and have to check its header. // block_statement, and have to check its header.
bool found_loop = false, end_search = false; bool found_loop = false;
const parse_node_t *ancestor = &node; for (const parse_node_t *ancestor = &node; ancestor != nullptr;
while (ancestor != NULL && !end_search) { ancestor = node_tree.get_parent(*ancestor)) {
const parse_node_t *loop_or_function_header = tnode_t<block_header> bh =
node_tree.header_node_for_block_statement(*ancestor); tnode_t<block_statement>::try_create(&node_tree, ancestor)
if (loop_or_function_header != NULL) { .child<0>();
switch (loop_or_function_header->type) { if (bh.try_get_child<while_header, 0>() ||
case symbol_while_header: bh.try_get_child<for_header, 0>()) {
case symbol_for_header: {
// This is a loop header, so we can break or continue. // This is a loop header, so we can break or continue.
found_loop = true; found_loop = true;
end_search = true;
break; break;
} } else if (bh.try_get_child<function_header, 0>()) {
case symbol_function_header: {
// This is a function header, so we cannot break or // This is a function header, so we cannot break or
// continue. We stop our search here. // continue. We stop our search here.
found_loop = false; found_loop = false;
end_search = true;
break; break;
} }
default: {
// Most likely begin / end style block, which makes no
// difference.
break;
}
}
}
ancestor = node_tree.get_parent(*ancestor);
} }
if (!found_loop && !first_argument_is_help(node_tree, node, buff_src)) { if (!found_loop && !first_argument_is_help(node_tree, node, buff_src)) {