diff --git a/builtin.cpp b/builtin.cpp index 3e09a1fce..17b0caaa8 100644 --- a/builtin.cpp +++ b/builtin.cpp @@ -1038,14 +1038,14 @@ static int builtin_emit(parser_t &parser, wchar_t **argv) static int builtin_generic(parser_t &parser, wchar_t **argv) { int argc=builtin_count_args(argv); - + /* Hackish - if we have no arguments other than the command, we are a "naked invocation" and we just print help */ if (argc == 1) { builtin_print_help(parser, argv[0], stdout_buffer); return STATUS_BUILTIN_ERROR; } - + woptind=0; static const struct woption @@ -1754,16 +1754,16 @@ static int builtin_pwd(parser_t &parser, wchar_t **argv) int define_function(parser_t &parser, const wcstring_list_t &c_args, const wcstring &contents, wcstring *out_err) { assert(out_err != NULL); - + /* wgetopt expects 'function' as the first argument. Make a new wcstring_list with that property. */ wcstring_list_t args; args.push_back(L"function"); args.insert(args.end(), c_args.begin(), c_args.end()); - + /* Hackish const_cast matches the one in builtin_run */ const null_terminated_array_t argv_array(args); wchar_t **argv = const_cast(argv_array.get()); - + int argc = builtin_count_args(argv); int res=STATUS_BUILTIN_OK; wchar_t *desc=0; @@ -1806,9 +1806,9 @@ int define_function(parser_t &parser, const wcstring_list_t &c_args, const wcstr case 0: if (long_options[opt_index].flag != 0) break; - - - + + + append_format(*out_err, BUILTIN_ERR_UNKNOWN, argv[0], @@ -1876,7 +1876,7 @@ int define_function(parser_t &parser, const wcstring_list_t &c_args, const wcstr if (is_subshell) { size_t block_idx = 0; - + /* Find the outermost substitution block */ for (block_idx = 0; ; block_idx++) { @@ -1884,7 +1884,7 @@ int define_function(parser_t &parser, const wcstring_list_t &c_args, const wcstr if (b == NULL || b->type() == SUBST) break; } - + /* Go one step beyond that, to get to the caller */ const block_t *caller_block = parser.block_at_index(block_idx + 1); if (caller_block != NULL && caller_block->job != NULL) @@ -2046,9 +2046,9 @@ int define_function(parser_t &parser, const wcstring_list_t &c_args, const wcstr event_t &e = d.events.at(i); e.function_name = d.name; } - + d.definition = contents.c_str(); - + // TODO: fix def_offset inside function_add function_add(d, parser); } @@ -2068,7 +2068,7 @@ static int builtin_function(parser_t &parser, wchar_t **argv) builtin_print_help(parser, argv[0], stdout_buffer); return STATUS_BUILTIN_OK; } - + int argc = builtin_count_args(argv); int res=STATUS_BUILTIN_OK; wchar_t *desc=0; @@ -2181,7 +2181,7 @@ static int builtin_function(parser_t &parser, wchar_t **argv) if (is_subshell) { size_t block_idx = 0; - + /* Find the outermost substitution block */ for (block_idx = 0; ; block_idx++) { @@ -2189,7 +2189,7 @@ static int builtin_function(parser_t &parser, wchar_t **argv) if (b == NULL || b->type() == SUBST) break; } - + /* Go one step beyond that, to get to the caller */ const block_t *caller_block = parser.block_at_index(block_idx + 1); if (caller_block != NULL && caller_block->job != NULL) @@ -3965,7 +3965,7 @@ static int builtin_break_continue(parser_t &parser, wchar_t **argv) { parser.block_at_index(block_idx)->skip = true; } - + /* Skip the loop itself */ block_t *loop_block = parser.block_at_index(loop_idx); loop_block->skip = true; @@ -4043,7 +4043,7 @@ static int builtin_return(parser_t &parser, wchar_t **argv) builtin_print_help(parser, argv[0], stderr_buffer); return STATUS_BUILTIN_ERROR; } - + /* Skip everything up to (and then including) the function block */ for (size_t i=0; i < function_block_idx; i++) { @@ -4063,8 +4063,8 @@ static int builtin_switch(parser_t &parser, wchar_t **argv) { int res=STATUS_BUILTIN_OK; int argc = builtin_count_args(argv); - - /* Hackish - if we have no arguments other than the command, we are a "naked invocation" and we just print help */ + + /* Hackish - if we have no arguments other than the command, we are a "naked invocation" and we just print help */ if (argc == 1) { builtin_print_help(parser, argv[0], stdout_buffer); @@ -4307,7 +4307,7 @@ int builtin_parse(parser_t &parser, wchar_t **argv) stdout_buffer.append(errors.at(i).describe(src)); stdout_buffer.push_back(L'\n'); } - + stdout_buffer.append(L"(Reparsed with continue after error)\n"); parse_tree.clear(); errors.clear(); diff --git a/builtin_set.cpp b/builtin_set.cpp index f7622260a..633416497 100644 --- a/builtin_set.cpp +++ b/builtin_set.cpp @@ -712,7 +712,7 @@ static int builtin_set(parser_t &parser, wchar_t **argv) retcode = 1; break; } - + size_t idx_count = indexes.size(); size_t val_count = argc-woptind-1; diff --git a/common.cpp b/common.cpp index 518625600..d20040b87 100644 --- a/common.cpp +++ b/common.cpp @@ -1130,7 +1130,7 @@ static size_t read_unquoted_escape(const wchar_t *input, wcstring *result, bool switch (c) { - /* A null character after a backslash is an error */ + /* A null character after a backslash is an error */ case L'\0': { /* Adjust in_pos to only include the backslash */ @@ -1177,11 +1177,11 @@ static size_t read_unquoted_escape(const wchar_t *input, wcstring *result, bool { chars=8; max_val = WCHAR_MAX; - + // Don't exceed the largest Unicode code point - see #1107 if (0x10FFFF < max_val) max_val = (wchar_t)0x10FFFF; - + break; } diff --git a/complete.cpp b/complete.cpp index dab7dcb26..e5b775768 100644 --- a/complete.cpp +++ b/complete.cpp @@ -472,7 +472,7 @@ void completion_autoload_t::command_removed(const wcstring &cmd) void append_completion(std::vector &completions, const wcstring &comp, const wcstring &desc, complete_flags_t flags, string_fuzzy_match_t match) { /* If we just constructed the completion and used push_back, we would get two string copies. Try to avoid that by making a stubby completion in the vector first, and then copying our string in. Note that completion_t's constructor will munge 'flags' so it's important that we pass those to the constructor. - + Nasty hack for #1241 - since the constructor needs the completion string to resolve AUTO_SPACE, and we aren't providing it with the completion, we have to do the resolution ourselves. We should get this resolving out of the constructor. */ const wcstring empty; @@ -1811,56 +1811,56 @@ void complete(const wcstring &cmd_with_subcmds, std::vector &comps parse_util_cmdsubst_extent(cmd_with_subcmds.c_str(), cmd_with_subcmds.size(), &cmdsubst_begin, &cmdsubst_end); assert(cmdsubst_begin != NULL && cmdsubst_end != NULL && cmdsubst_end >= cmdsubst_begin); const wcstring cmd = wcstring(cmdsubst_begin, cmdsubst_end - cmdsubst_begin); - + /* Make our completer */ completer_t completer(cmd, flags); - + wcstring current_command; const size_t pos = cmd.size(); bool done=false; bool use_command = 1; bool use_function = 1; bool use_builtin = 1; - + // debug( 1, L"Complete '%ls'", cmd ); - + const wchar_t *cmd_cstr = cmd.c_str(); const wchar_t *tok_begin = NULL, *prev_begin = NULL, *prev_end = NULL; parse_util_token_extent(cmd_cstr, cmd.size(), &tok_begin, NULL, &prev_begin, &prev_end); - + /** If we are completing a variable name or a tilde expansion user name, we do that and return. No need for any other completions. */ - + const wcstring current_token = tok_begin; - + if (!done) { done = completer.try_complete_variable(current_token) || completer.try_complete_user(current_token); } - + if (!done) { //const size_t prev_token_len = (prev_begin ? prev_end - prev_begin : 0); //const wcstring prev_token(prev_begin, prev_token_len); - + parse_node_tree_t tree; parse_tree_from_string(cmd, parse_flag_continue_after_error | parse_flag_accept_incomplete_tokens, &tree, NULL); - + /* Find the plain statement that contains the position */ const parse_node_t *plain_statement = tree.find_node_matching_source_location(symbol_plain_statement, pos, NULL); if (plain_statement != NULL) { assert(plain_statement->has_source() && plain_statement->type == symbol_plain_statement); - + /* Get the command node */ const parse_node_t *cmd_node = tree.get_child(*plain_statement, 0, parse_token_type_string); - + /* Get the actual command string */ if (cmd_node != NULL) current_command = cmd_node->get_source(cmd); - + /* Check the decoration */ switch (tree.decoration_for_plain_statement(*plain_statement)) { @@ -1869,20 +1869,20 @@ void complete(const wcstring &cmd_with_subcmds, std::vector &comps use_function = true; use_builtin = true; break; - + case parse_statement_decoration_command: use_command = true; use_function = false; use_builtin = false; break; - + case parse_statement_decoration_builtin: use_command = false; use_function = false; use_builtin = true; break; } - + if (cmd_node && cmd_node->location_in_or_at_end_of_source_range(pos)) { /* Complete command filename */ @@ -1892,7 +1892,7 @@ void complete(const wcstring &cmd_with_subcmds, std::vector &comps { /* Get all the arguments */ const parse_node_tree_t::parse_node_list_t all_arguments = tree.find_nodes(*plain_statement, symbol_argument); - + /* See whether we are in an argument. We may also be in a redirection, or nothing at all. */ size_t matching_arg_index = -1; for (size_t i=0; i < all_arguments.size(); i++) @@ -1904,17 +1904,17 @@ void complete(const wcstring &cmd_with_subcmds, std::vector &comps break; } } - + bool had_ddash = false; wcstring current_argument, previous_argument; if (matching_arg_index != (size_t)(-1)) { /* Get the current argument and the previous argument, if we have one */ current_argument = all_arguments.at(matching_arg_index)->get_source(cmd); - + if (matching_arg_index > 0) previous_argument = all_arguments.at(matching_arg_index - 1)->get_source(cmd); - + /* Check to see if we have a preceding double-dash */ for (size_t i=0; i < matching_arg_index; i++) { @@ -1925,34 +1925,34 @@ void complete(const wcstring &cmd_with_subcmds, std::vector &comps } } } - + bool do_file = false; - + wcstring current_command_unescape, previous_argument_unescape, current_argument_unescape; if (unescape_string(current_command, ¤t_command_unescape, UNESCAPE_DEFAULT) && - unescape_string(previous_argument, &previous_argument_unescape, UNESCAPE_DEFAULT) && - unescape_string(current_argument, ¤t_argument_unescape, UNESCAPE_INCOMPLETE)) + unescape_string(previous_argument, &previous_argument_unescape, UNESCAPE_DEFAULT) && + unescape_string(current_argument, ¤t_argument_unescape, UNESCAPE_INCOMPLETE)) { do_file = completer.complete_param(current_command_unescape, previous_argument_unescape, current_argument_unescape, !had_ddash); } - + /* If we have found no command specific completions at all, fall back to using file completions. */ if (completer.empty()) do_file = true; - + /* And if we're autosuggesting, and the token is empty, don't do file suggestions */ if ((flags & COMPLETION_REQUEST_AUTOSUGGESTION) && current_argument_unescape.empty()) do_file = false; - + /* This function wants the unescaped string */ completer.complete_param_expand(current_token, do_file); } } } - + comps = completer.get_completions(); } diff --git a/env_universal.cpp b/env_universal.cpp index ed7396f6a..72e9cc0e6 100644 --- a/env_universal.cpp +++ b/env_universal.cpp @@ -132,12 +132,12 @@ static int try_get_socket_once(void) if (connect(s, (struct sockaddr *)&local, sizeof local) == -1) { close(s); - + /* If it fails on first try, it's probably no serious error, but fishd hasn't been launched yet. This happens (at least) on the first concurrent session. */ if (get_socket_count > 1) wperror(L"connect"); - + return -1; } @@ -438,8 +438,8 @@ void env_universal_set(const wcstring &name, const wcstring &value, bool exportv else { message_t *msg = create_message(exportv?SET_EXPORT:SET, - name.c_str(), - value.c_str()); + name.c_str(), + value.c_str()); if (!msg) { diff --git a/event.cpp b/event.cpp index a6fab6205..65d337f02 100644 --- a/event.cpp +++ b/event.cpp @@ -146,13 +146,13 @@ static int event_is_blocked(const event_t &e) { const block_t *block; parser_t &parser = parser_t::principal_parser(); - + size_t idx = 0; while ((block = parser.block_at_index(idx++))) { if (event_block_list_blocks_type(block->event_blocks, e.type)) return true; - + } return event_block_list_blocks_type(parser.global_event_blocks, e.type); } diff --git a/exec.cpp b/exec.cpp index dba7b534e..832c88847 100644 --- a/exec.cpp +++ b/exec.cpp @@ -582,7 +582,7 @@ static void exec_no_exec(parser_t &parser, const job_t *job) /* With the new parser, commands aren't responsible for pushing / popping blocks, so there's nothing to do */ return; } - + /* Hack hack hack. If this is an 'end' job, then trigger a pop. If this is a job that would create a block, trigger a push. See #624 */ const process_t *p = job->first_process; if (p && p->type == INTERNAL_BUILTIN) @@ -988,7 +988,7 @@ void exec_job(parser_t &parser, job_t *j) } break; } - + case INTERNAL_BUILTIN: { int builtin_stdin=0; @@ -1133,20 +1133,20 @@ void exec_job(parser_t &parser, job_t *j) } break; } - + case EXTERNAL: /* External commands are handled in the next switch statement below */ break; - + case INTERNAL_EXEC: /* We should have handled exec up above */ assert(0 && "INTERNAL_EXEC process found in pipeline, where it should never be. Aborting."); break; - + case INTERNAL_BUFFER: /* Internal buffers are handled in the next switch statement below */ break; - + } if (exec_error) @@ -1507,7 +1507,7 @@ void exec_job(parser_t &parser, job_t *j) break; } - + case INTERNAL_EXEC: { /* We should have handled exec up above */ @@ -1581,7 +1581,7 @@ static int exec_subshell_internal(const wcstring &cmd, wcstring_list_t *lst, boo int prev_subshell = is_subshell; const int prev_status = proc_get_last_status(); char sep=0; - + //fprintf(stderr, "subcmd %ls\n", cmd.c_str()); const env_var_t ifs = env_get_string(L"IFS"); diff --git a/expand.cpp b/expand.cpp index 6216da222..cd7194a3b 100644 --- a/expand.cpp +++ b/expand.cpp @@ -793,7 +793,7 @@ static int expand_pid(const wcstring &instr_with_sep, append_completion(out, instr_with_sep); return 1; } - + /* expand_string calls us with internal separators in instr...sigh */ wcstring instr = instr_with_sep; remove_internal_separator(instr, false); @@ -1624,7 +1624,7 @@ wcstring replace_home_directory_with_tilde(const wcstring &str) { home_directory.push_back(L'/'); } - + // Now check if the home_directory prefixes the string if (string_prefixes_string(home_directory, result)) { diff --git a/fish_tests.cpp b/fish_tests.cpp index 0144d3f8e..6c536bca3 100644 --- a/fish_tests.cpp +++ b/fish_tests.cpp @@ -149,7 +149,11 @@ static void err(const wchar_t *blah, ...) /* Test sane escapes */ static void test_unescape_sane() { - const struct test_t {const wchar_t * input; const wchar_t * expected;} tests[] = + const struct test_t + { + const wchar_t * input; + const wchar_t * expected; + } tests[] = { {L"abcd", L"abcd"}, {L"'abcd'", L"abcd"}, @@ -439,7 +443,7 @@ static void test_tok() } } } - + /* Test redirection_type_for_string */ if (redirection_type_for_string(L"<") != TOK_REDIRECT_IN) err(L"redirection_type_for_string failed on line %ld", (long)__LINE__); if (redirection_type_for_string(L"^") != TOK_REDIRECT_OUT) err(L"redirection_type_for_string failed on line %ld", (long)__LINE__); @@ -623,45 +627,45 @@ static void test_parser() { err(L"'break' command outside of loop block context undetected"); } - + if (parse_util_detect_errors(L"break --help")) { err(L"'break --help' incorrectly marked as error"); } - + if (! parse_util_detect_errors(L"while false ; function foo ; break ; end ; end ")) { err(L"'break' command inside function allowed to break from loop outside it"); } - + if (!parse_util_detect_errors(L"exec ls|less") || !parse_util_detect_errors(L"echo|return")) { err(L"Invalid pipe command undetected"); } - + if (parse_util_detect_errors(L"for i in foo ; switch $i ; case blah ; break; end; end ")) { err(L"'break' command inside switch falsely reported as error"); } - + if (parse_util_detect_errors(L"or cat | cat") || parse_util_detect_errors(L"and cat | cat")) { err(L"boolean command at beginning of pipeline falsely reported as error"); } - + if (! parse_util_detect_errors(L"cat | and cat")) { err(L"'and' command in pipeline not reported as error"); } - + if (! parse_util_detect_errors(L"cat | exec") || ! parse_util_detect_errors(L"exec | cat")) { err(L"'exec' command in pipeline not reported as error"); } - + say(L"Testing basic evaluation"); #if 0 @@ -675,7 +679,7 @@ static void test_parser() { err(L"Invalid block mode when evaluating undetected"); } - + /* Ensure that we don't crash on infinite self recursion and mutual recursion. These must use the principal parser because we cannot yet execute jobs on other parsers (!) */ say(L"Testing recursion detection"); parser_t::principal_parser().eval(L"function recursive ; recursive ; end ; recursive; ", io_chain_t(), TOP); @@ -717,34 +721,34 @@ static void test_1_cancellation(const wchar_t *src) static void test_cancellation() { say(L"Testing Ctrl-C cancellation. If this hangs, that's a bug!"); - + /* Enable fish's signal handling here. We need to make this interactive for fish to install its signal handlers */ proc_push_interactive(1); signal_set_handlers(); - + /* This tests that we can correctly ctrl-C out of certain loop constructs, and that nothing gets printed if we do */ - + /* Here the command substitution is an infinite loop. echo never even gets its argument, so when we cancel we expect no output */ test_1_cancellation(L"echo (while true ; echo blah ; end)"); - + fprintf(stderr, "."); - + /* Nasty infinite loop that doesn't actually execute anything */ test_1_cancellation(L"echo (while true ; end) (while true ; end) (while true ; end)"); fprintf(stderr, "."); - + test_1_cancellation(L"while true ; end"); fprintf(stderr, "."); - + test_1_cancellation(L"for i in (while true ; end) ; end"); fprintf(stderr, "."); fprintf(stderr, "\n"); - + /* Restore signal handling */ proc_pop_interactive(); signal_reset_handlers(); - + /* Ensure that we don't think we should cancel */ reader_reset_interrupted(); } @@ -752,27 +756,28 @@ static void test_cancellation() static void test_indents() { say(L"Testing indents"); - + // Here are the components of our source and the indents we expect those to be - struct indent_component_t { + struct indent_component_t + { const wchar_t *txt; int indent; }; - + const indent_component_t components1[] = { {L"if foo", 0}, {L"end", 0}, {NULL, -1} }; - + const indent_component_t components2[] = { {L"if foo", 0}, {L"", 1}, //trailing newline! {NULL, -1} }; - + const indent_component_t components3[] = { {L"if foo", 0}, @@ -780,7 +785,7 @@ static void test_indents() {L"end", 0}, //trailing newline! {NULL, -1} }; - + const indent_component_t components4[] = { {L"if foo", 0}, @@ -790,7 +795,7 @@ static void test_indents() {L"", 0}, {NULL, -1} }; - + const indent_component_t components5[] = { {L"if foo", 0}, @@ -798,7 +803,7 @@ static void test_indents() {L"", 2}, {NULL, -1} }; - + const indent_component_t components6[] = { {L"begin", 0}, @@ -806,7 +811,7 @@ static void test_indents() {L"", 1}, {NULL, -1} }; - + const indent_component_t components7[] = { {L"begin; end", 0}, @@ -814,7 +819,7 @@ static void test_indents() {L"", 0}, {NULL, -1} }; - + const indent_component_t components8[] = { {L"if foo", 0}, @@ -831,7 +836,7 @@ static void test_indents() {L"", 1}, {NULL, -1} }; - + const indent_component_t components10[] = { {L"switch foo", 0}, @@ -841,7 +846,7 @@ static void test_indents() {L"", 2}, {NULL, -1} }; - + const indent_component_t components11[] = { {L"switch foo", 0}, @@ -850,7 +855,7 @@ static void test_indents() }; - + const indent_component_t *tests[] = {components1, components2, components3, components4, components5, components6, components7, components8, components9, components10, components11}; for (size_t which = 0; which < sizeof tests / sizeof *tests; which++) { @@ -861,7 +866,7 @@ static void test_indents() { component_count++; } - + // Generate the expected indents wcstring text; std::vector expected_indents; @@ -876,10 +881,10 @@ static void test_indents() expected_indents.resize(text.size(), components[i].indent); } assert(expected_indents.size() == text.size()); - + // Compute the indents std::vector indents = parse_util_compute_indents(text); - + if (expected_indents.size() != indents.size()) { err(L"Indent vector has wrong size! Expected %lu, actual %lu", expected_indents.size(), indents.size()); @@ -990,7 +995,7 @@ static int expand_test(const wchar_t *in, int flags, ...) size_t i=0; int res=1; wchar_t *arg; - + if (expand_string(in, output, flags)) { @@ -1468,22 +1473,22 @@ static void test_complete(void) assert(completions.size() == 2); assert(completions.at(0).completion == L"$Foo1"); assert(completions.at(1).completion == L"$Bar1"); - + completions.clear(); complete(L"echo (/bin/mkdi", completions, COMPLETION_REQUEST_DEFAULT); assert(completions.size() == 1); assert(completions.at(0).completion == L"r"); - + completions.clear(); complete(L"echo (ls /bin/mkdi", completions, COMPLETION_REQUEST_DEFAULT); assert(completions.size() == 1); assert(completions.at(0).completion == L"r"); - + completions.clear(); complete(L"echo (command ls /bin/mkdi", completions, COMPLETION_REQUEST_DEFAULT); assert(completions.size() == 1); assert(completions.at(0).completion == L"r"); - + /* Add a function and test completing it in various ways */ struct function_data_t func_data; func_data.name = L"scuttlebutt"; @@ -1505,7 +1510,7 @@ static void test_complete(void) completions.clear(); complete(L"echo (builtin scuttlebut", completions, COMPLETION_REQUEST_DEFAULT); assert(completions.size() == 0); - + complete_set_variable_names(NULL); } @@ -2267,13 +2272,13 @@ static void test_new_parser_correctness(void) static inline bool string_for_permutation(const wcstring *fuzzes, size_t fuzz_count, size_t len, size_t permutation, wcstring *out_str) { out_str->clear(); - + size_t remaining_permutation = permutation; for (size_t i=0; i < len; i++) { size_t idx = remaining_permutation % fuzz_count; remaining_permutation /= fuzz_count; - + out_str->append(fuzzes[idx]); out_str->push_back(L' '); } @@ -2307,11 +2312,11 @@ static void test_new_parser_fuzzing(void) L"&", L";", }; - + /* Generate a list of strings of all keyword / token combinations. */ wcstring src; src.reserve(128); - + parse_node_tree_t node_tree; parse_error_list_t errors; @@ -2331,7 +2336,7 @@ static void test_new_parser_fuzzing(void) } if (log_it) fprintf(stderr, "done (%lu)\n", permutation); - + } double end = timef(); if (log_it) @@ -2344,7 +2349,7 @@ static bool test_1_parse_ll2(const wcstring &src, wcstring *out_cmd, wcstring *o out_cmd->clear(); out_joined_args->clear(); *out_deco = parse_statement_decoration_none; - + bool result = false; parse_node_tree_t tree; if (parse_tree_from_string(src, parse_flag_none, &tree, NULL)) @@ -2357,13 +2362,13 @@ static bool test_1_parse_ll2(const wcstring &src, wcstring *out_cmd, wcstring *o return false; } const parse_node_t &stmt = *stmt_nodes.at(0); - + /* Return its decoration */ *out_deco = tree.decoration_for_plain_statement(stmt); - + /* Return its command */ tree.command_for_plain_statement(stmt, src, out_cmd); - + /* Return arguments separated by spaces */ const parse_node_tree_t::parse_node_list_t arg_nodes = tree.find_nodes(stmt, symbol_argument); for (size_t i=0; i < arg_nodes.size(); i++) @@ -2380,7 +2385,7 @@ static bool test_1_parse_ll2(const wcstring &src, wcstring *out_cmd, wcstring *o static void test_new_parser_ll2(void) { say(L"Testing parser two-token lookahead"); - + const struct { wcstring src; @@ -2402,7 +2407,7 @@ static void test_new_parser_ll2(void) {L"function", L"function", L"", parse_statement_decoration_none}, {L"function --help", L"function", L"--help", parse_statement_decoration_none} }; - + for (size_t i=0; i < sizeof tests / sizeof *tests; i++) { wcstring cmd, args; @@ -2417,7 +2422,7 @@ static void test_new_parser_ll2(void) if (deco != tests[i].deco) err(L"When parsing '%ls', expected decoration %d but got %d on line %ld", tests[i].src.c_str(), (int)tests[i].deco, (int)deco, (long)__LINE__); } - + /* Verify that 'function -h' and 'function --help' are plain statements but 'function --foo' is not (#1240) */ const struct { @@ -2438,7 +2443,7 @@ static void test_new_parser_ll2(void) { err(L"Failed to parse '%ls'", tests2[i].src.c_str()); } - + const parse_node_tree_t::parse_node_list_t node_list = tree.find_nodes(tree.at(0), tests2[i].type); if (node_list.size() == 0) { @@ -2455,7 +2460,7 @@ static void test_new_parser_ad_hoc() { /* Very ad-hoc tests for issues encountered */ say(L"Testing new parser ad hoc tests"); - + /* Ensure that 'case' terminates a job list */ const wcstring src = L"switch foo ; case bar; case baz; end"; parse_node_tree_t parse_tree; @@ -2464,7 +2469,7 @@ static void test_new_parser_ad_hoc() { err(L"Parsing failed"); } - + /* Expect three case_item_lists: one for each case, and a terminal one. The bug was that we'd try to run a command 'case' */ const parse_node_t &root = parse_tree.at(0); const parse_node_tree_t::parse_node_list_t node_list = parse_tree.find_nodes(root, symbol_case_item_list); @@ -2487,25 +2492,25 @@ static void test_new_parser_errors(void) {L"echo 'abc", parse_error_tokenizer_unterminated_quote}, {L"'", parse_error_tokenizer_unterminated_quote}, {L"echo (abc", parse_error_tokenizer_unterminated_subshell}, - + {L"end", parse_error_unbalancing_end}, {L"echo hi ; end", parse_error_unbalancing_end}, - + {L"else", parse_error_unbalancing_else}, {L"if true ; end ; else", parse_error_unbalancing_else}, - + {L"case", parse_error_unbalancing_case}, {L"if true ; case ; end", parse_error_unbalancing_case}, - + {L"foo || bar", parse_error_double_pipe}, {L"foo && bar", parse_error_double_background}, }; - + for (size_t i = 0; i < sizeof tests / sizeof *tests; i++) { const wcstring src = tests[i].src; parse_error_code_t expected_code = tests[i].code; - + parse_error_list_t errors; parse_node_tree_t parse_tree; bool success = parse_tree_from_string(src, parse_flag_none, &parse_tree, &errors); @@ -2513,7 +2518,7 @@ static void test_new_parser_errors(void) { err(L"Source '%ls' was expected to fail to parse, but succeeded", src.c_str()); } - + if (errors.size() != 1) { err(L"Source '%ls' was expected to produce 1 error, but instead produced %lu errors", src.c_str(), errors.size()); @@ -2526,9 +2531,9 @@ static void test_new_parser_errors(void) err(L"\t\t%ls", errors.at(i).describe(src).c_str()); } } - + } - + } static void test_highlighting(void) @@ -2537,13 +2542,14 @@ static void test_highlighting(void) if (system("mkdir -p /tmp/fish_highlight_test/")) err(L"mkdir failed"); if (system("touch /tmp/fish_highlight_test/foo")) err(L"touch failed"); if (system("touch /tmp/fish_highlight_test/bar")) err(L"touch failed"); - + // Here are the components of our source and the colors we expect those to be - struct highlight_component_t { + struct highlight_component_t + { const wchar_t *txt; int color; }; - + const highlight_component_t components1[] = { {L"echo", highlight_spec_command}, @@ -2551,7 +2557,7 @@ static void test_highlighting(void) {L"&", highlight_spec_statement_terminator}, {NULL, -1} }; - + const highlight_component_t components2[] = { {L"command", highlight_spec_command}, @@ -2561,7 +2567,7 @@ static void test_highlighting(void) {L"&", highlight_spec_statement_terminator}, {NULL, -1} }; - + const highlight_component_t components3[] = { {L"if command ls", highlight_spec_command}, @@ -2574,7 +2580,7 @@ static void test_highlighting(void) {L"end", highlight_spec_command}, {NULL, -1} }; - + /* Verify that cd shows errors for non-directories */ const highlight_component_t components4[] = { @@ -2582,14 +2588,14 @@ static void test_highlighting(void) {L"/tmp/fish_highlight_test", highlight_spec_param | highlight_modifier_valid_path}, {NULL, -1} }; - + const highlight_component_t components5[] = { {L"cd", highlight_spec_command}, {L"/tmp/fish_highlight_test/foo", highlight_spec_error}, {NULL, -1} }; - + const highlight_component_t components6[] = { {L"cd", highlight_spec_command}, @@ -2598,7 +2604,7 @@ static void test_highlighting(void) {L"definitely_not_a_directory", highlight_spec_error}, {NULL, -1} }; - + // Command substitutions const highlight_component_t components7[] = { @@ -2610,57 +2616,57 @@ static void test_highlighting(void) {L")", highlight_spec_operator}, {NULL, -1} }; - + // Redirections substitutions const highlight_component_t components8[] = { {L"echo", highlight_spec_command}, {L"param1", highlight_spec_param}, - + /* Input redirection */ {L"<", highlight_spec_redirection}, {L"/bin/echo", highlight_spec_redirection}, - + /* Output redirection to a valid fd */ {L"1>&2", highlight_spec_redirection}, - + /* Output redirection to an invalid fd */ {L"2>&", highlight_spec_redirection}, {L"LOL", highlight_spec_error}, /* Just a param, not a redirection */ {L"/tmp/blah", highlight_spec_param}, - + /* Input redirection from directory */ {L"<", highlight_spec_redirection}, {L"/tmp/", highlight_spec_error}, - + /* Output redirection to an invalid path */ {L"3>", highlight_spec_redirection}, {L"/not/a/valid/path/nope", highlight_spec_error}, - + /* Output redirection to directory */ {L"3>", highlight_spec_redirection}, {L"/tmp/nope/", highlight_spec_error}, - + /* Redirections to overflow fd */ {L"99999999999999999999>&2", highlight_spec_error}, {L"2>&", highlight_spec_redirection}, {L"99999999999999999999", highlight_spec_error}, - + /* Output redirection containing a command substitution */ {L"4>", highlight_spec_redirection}, {L"(", highlight_spec_operator}, {L"echo", highlight_spec_command}, {L"/tmp/somewhere", highlight_spec_param}, {L")", highlight_spec_operator}, - + /* Just another param */ {L"param2", highlight_spec_param}, {NULL, -1} }; - + const highlight_component_t components9[] = { {L"end", highlight_spec_error}, @@ -2677,7 +2683,7 @@ static void test_highlighting(void) {NULL, -1} }; - + const highlight_component_t *tests[] = {components1, components2, components3, components4, components5, components6, components7, components8, components9, components10}; for (size_t which = 0; which < sizeof tests / sizeof *tests; which++) { @@ -2688,7 +2694,7 @@ static void test_highlighting(void) { component_count++; } - + // Generate the text wcstring text; std::vector expected_colors; @@ -2703,10 +2709,10 @@ static void test_highlighting(void) expected_colors.resize(text.size(), components[i].color); } assert(expected_colors.size() == text.size()); - + std::vector colors(text.size()); highlight_shell(text, colors, 20, NULL, env_vars_snapshot_t()); - + if (expected_colors.size() != colors.size()) { err(L"Color vector has wrong size! Expected %lu, actual %lu", expected_colors.size(), colors.size()); @@ -2717,7 +2723,7 @@ static void test_highlighting(void) // Hackish space handling. We don't care about the colors in spaces. if (text.at(i) == L' ') continue; - + if (expected_colors.at(i) != colors.at(i)) { const wcstring spaces(i, L' '); @@ -2725,7 +2731,7 @@ static void test_highlighting(void) } } } - + system("rm -Rf /tmp/fish_highlight_test"); } @@ -2750,7 +2756,7 @@ int main(int argc, char **argv) builtin_init(); reader_init(); env_init(); - + /* Set default signal handlers, so we can ctrl-C out of this */ signal_reset_handlers(); diff --git a/function.cpp b/function.cpp index eadcca7c6..8421a4daa 100644 --- a/function.cpp +++ b/function.cpp @@ -194,7 +194,7 @@ void function_add(const function_data_t &data, const parser_t &parser) { def_offset = parser.line_number_of_character_at_offset(parser.current_block()->tok_pos); } - + const function_map_t::value_type new_pair(data.name, function_info_t(data, filename, def_offset, is_autoload)); loaded_functions.insert(new_pair); diff --git a/highlight.cpp b/highlight.cpp index 35db0dfc1..0bce91ce6 100644 --- a/highlight.cpp +++ b/highlight.cpp @@ -341,7 +341,7 @@ bool plain_statement_get_expanded_command(const wcstring &src, const parse_node_ { assert(plain_statement.type == symbol_plain_statement); bool result = false; - + /* Get the command */ wcstring cmd; if (tree.command_for_plain_statement(plain_statement, src, &cmd)) @@ -709,11 +709,11 @@ static bool has_expand_reserved(const wcstring &str) static bool autosuggest_parse_command(const wcstring &buff, wcstring *out_expanded_command, parse_node_t *out_last_arg) { bool result = false; - + /* Parse the buffer */ parse_node_tree_t parse_tree; parse_tree_from_string(buff, parse_flag_continue_after_error | parse_flag_accept_incomplete_tokens, &parse_tree, NULL); - + /* Find the last statement */ const parse_node_t *last_statement = parse_tree.find_last_node_of_type(symbol_plain_statement, NULL); if (last_statement != NULL) @@ -722,7 +722,7 @@ static bool autosuggest_parse_command(const wcstring &buff, wcstring *out_expand { /* We got it */ result = true; - + /* Find the last argument. If we don't get one, return an invalid node. */ const parse_node_t *last_arg = parse_tree.find_last_node_of_type(symbol_argument, last_statement); if (last_arg != NULL) @@ -739,7 +739,7 @@ bool autosuggest_suggest_special(const wcstring &str, const wcstring &working_di { if (str.empty()) return false; - + ASSERT_IS_BACKGROUND_THREAD(); /* Parse the string */ @@ -1672,43 +1672,43 @@ class highlighter_t { /* The string we're highlighting. Note this is a reference memmber variable (to avoid copying)! We must not outlive this! */ const wcstring &buff; - + /* Cursor position */ const size_t cursor_pos; - + /* Environment variables. Again, a reference member variable! */ const env_vars_snapshot_t &vars; - + /* Working directory */ const wcstring working_directory; - + /* The resulting colors */ typedef std::vector color_array_t; color_array_t color_array; - + /* The parse tree of the buff */ parse_node_tree_t parse_tree; - + /* Color an argument */ void color_argument(const parse_node_t &node); - + /* Color a redirection */ void color_redirection(const parse_node_t &node); /* Color the arguments of the given node */ void color_arguments(const parse_node_t &list_node); - + /* Color the redirections of the given node */ void color_redirections(const parse_node_t &list_node); - + /* Color all the children of the command with the given type */ void color_children(const parse_node_t &parent, parse_token_type_t type, int color); - + /* Colors the source range of a node with a given color */ void color_node(const parse_node_t &node, int color); - - public: - + +public: + /* Constructor */ highlighter_t(const wcstring &str, size_t pos, const env_vars_snapshot_t &ev, const wcstring &wd) : buff(str), cursor_pos(pos), vars(ev), working_directory(wd), color_array(str.size()) { @@ -1716,7 +1716,7 @@ class highlighter_t this->parse_tree.clear(); parse_tree_from_string(buff, parse_flag_continue_after_error | parse_flag_include_comments, &this->parse_tree, NULL); } - + /* Perform highlighting, returning an array of colors */ const color_array_t &highlight(); }; @@ -1740,16 +1740,16 @@ void highlighter_t::color_argument(const parse_node_t &node) { if (! node.has_source()) return; - + const wcstring arg_str = node.get_source(this->buff); - + /* Get an iterator to the colors associated with the argument */ const size_t arg_start = node.source_start; const color_array_t::iterator arg_colors = color_array.begin() + arg_start; /* Color this argument without concern for command substitutions */ color_argument_internal(arg_str, arg_colors); - + /* Now do command substitutions */ size_t cmdsub_cursor = 0, cmdsub_start = 0, cmdsub_end = 0; wcstring cmdsub_contents; @@ -1758,16 +1758,16 @@ void highlighter_t::color_argument(const parse_node_t &node) /* The cmdsub_start is the open paren. cmdsub_end is either the close paren or the end of the string. cmdsub_contents extends from one past cmdsub_start to cmdsub_end */ assert(cmdsub_end > cmdsub_start); assert(cmdsub_end - cmdsub_start - 1 == cmdsub_contents.size()); - + /* Found a command substitution. Compute the position of the start and end of the cmdsub contents, within our overall src. */ const size_t arg_subcmd_start = arg_start + cmdsub_start, arg_subcmd_end = arg_start + cmdsub_end; - + /* Highlight the parens. The open paren must exist; the closed paren may not if it was incomplete. */ assert(cmdsub_start < arg_str.size()); this->color_array.at(arg_subcmd_start) = highlight_spec_operator; if (arg_subcmd_end < this->buff.size()) this->color_array.at(arg_subcmd_end) = highlight_spec_operator; - + /* Compute the cursor's position within the cmdsub. We must be past the open paren (hence >) but can be at the end of the string or closed paren (hence <=) */ size_t cursor_subpos = CURSOR_POSITION_INVALID; if (cursor_pos != CURSOR_POSITION_INVALID && cursor_pos > arg_subcmd_start && cursor_pos <= arg_subcmd_end) @@ -1775,11 +1775,11 @@ void highlighter_t::color_argument(const parse_node_t &node) /* The -1 because the cmdsub_contents does not include the open paren */ cursor_subpos = cursor_pos - arg_subcmd_start - 1; } - + /* Highlight it recursively. */ highlighter_t cmdsub_highlighter(cmdsub_contents, cursor_subpos, this->vars, this->working_directory); const color_array_t &subcolors = cmdsub_highlighter.highlight(); - + /* Copy out the subcolors back into our array */ assert(subcolors.size() == cmdsub_contents.size()); std::copy(subcolors.begin(), subcolors.end(), this->color_array.begin() + arg_subcmd_start + 1); @@ -1801,7 +1801,7 @@ static bool node_is_potential_path(const wcstring &src, const parse_node_t &node /* Big hack: is_potential_path expects a tilde, but unescape_string gives us HOME_DIRECTORY. Put it back. */ if (! token.empty() && token.at(0) == HOME_DIRECTORY) token.at(0) = L'~'; - + const wcstring_list_t working_directory_list(1, working_directory); result = is_potential_path(token, working_directory_list, PATH_EXPAND_TILDE); } @@ -1822,7 +1822,7 @@ void highlighter_t::color_arguments(const parse_node_t &list_node) cmd_is_cd = (cmd_str == L"cd"); } } - + /* Find all the arguments of this list */ const parse_node_tree_t::parse_node_list_t nodes = this->parse_tree.find_nodes(list_node, symbol_argument); @@ -1831,7 +1831,7 @@ void highlighter_t::color_arguments(const parse_node_t &list_node) const parse_node_t *child = nodes.at(i); assert(child != NULL && child->type == symbol_argument); this->color_argument(*child); - + if (cmd_is_cd) { /* Mark this as an error if it's not 'help' and not a valid cd path */ @@ -1853,18 +1853,18 @@ void highlighter_t::color_redirection(const parse_node_t &redirection_node) assert(redirection_node.type == symbol_redirection); if (! redirection_node.has_source()) return; - + const parse_node_t *redirection_primitive = this->parse_tree.get_child(redirection_node, 0, parse_token_type_redirection); //like 2> const parse_node_t *redirection_target = this->parse_tree.get_child(redirection_node, 1, parse_token_type_string); //like &1 or file path - + if (redirection_primitive != NULL) { wcstring target; const enum token_type redirect_type = this->parse_tree.type_for_redirection(redirection_node, this->buff, NULL, &target); - + /* We may get a TOK_NONE redirection type, e.g. if the redirection is invalid */ this->color_node(*redirection_primitive, redirect_type == TOK_NONE ? highlight_spec_error : highlight_spec_redirection); - + /* Check if the argument contains a command substitution. If so, highlight it as a param even though it's a command redirection, and don't try to do any other validation. */ if (parse_util_locate_cmdsubst(target.c_str(), NULL, NULL, true) != 0) { @@ -1893,12 +1893,12 @@ void highlighter_t::color_redirection(const parse_node_t &redirection_node) const wchar_t *target_cstr = target.c_str(); wchar_t *end = NULL; int fd = fish_wcstoi(target_cstr, &end, 10); - + /* The iswdigit check ensures there's no leading whitespace, the *end check ensures the entire string was consumed, and the numeric checks ensure the fd is at least zero and there was no overflow */ target_is_valid = (iswdigit(target_cstr[0]) && *end == L'\0' && fd >= 0 && fd < INT_MAX); } break; - + case TOK_REDIRECT_IN: { /* Input redirections must have a readable non-directory */ @@ -1906,7 +1906,7 @@ void highlighter_t::color_redirection(const parse_node_t &redirection_node) target_is_valid = ! waccess(target_path, R_OK) && ! wstat(target_path, &buf) && ! S_ISDIR(buf.st_mode); } break; - + case TOK_REDIRECT_OUT: case TOK_REDIRECT_APPEND: case TOK_REDIRECT_NOCLOB: @@ -1914,13 +1914,13 @@ void highlighter_t::color_redirection(const parse_node_t &redirection_node) /* Test whether the file exists, and whether it's writable (possibly after creating it). access() returns failure if the file does not exist. */ bool file_exists = false, file_is_writable = false; int err = 0; - + struct stat buf = {}; if (wstat(target_path, &buf) < 0) { err = errno; } - + if (string_suffixes_string(L"/", target)) { /* Redirections to things that are directories is definitely not allowed */ @@ -1937,11 +1937,11 @@ void highlighter_t::color_redirection(const parse_node_t &redirection_node) { /* File does not exist. Check if its parent directory is writable. */ wcstring parent = wdirname(target_path); - + /* Ensure that the parent ends with the path separator. This will ensure that we get an error if the parent directory is not really a directory. */ if (! string_suffixes_string(L"/", parent)) parent.push_back(L'/'); - + /* Now the file is considered writable if the parent directory is writable */ file_exists = false; file_is_writable = (0 == waccess(parent, W_OK)); @@ -1952,19 +1952,19 @@ void highlighter_t::color_redirection(const parse_node_t &redirection_node) file_exists = false; file_is_writable = false; } - + /* NOCLOB means that we must not overwrite files that exist */ - target_is_valid = file_is_writable && ! (file_exists && redirect_type == TOK_REDIRECT_NOCLOB); + target_is_valid = file_is_writable && !(file_exists && redirect_type == TOK_REDIRECT_NOCLOB); } break; - + default: /* We should not get here, since the node was marked as a redirection, but treat it as an error for paranoia */ target_is_valid = false; break; } } - + if (redirection_target != NULL) { this->color_node(*redirection_target, target_is_valid ? highlight_spec_redirection : highlight_spec_error); @@ -2017,30 +2017,30 @@ static bool command_is_valid(const wcstring &cmd, enum parse_statement_decoratio command_ok = false; implicit_cd_ok = false; } - + /* Check them */ bool is_valid = false; - + /* Builtins */ if (! is_valid && builtin_ok) is_valid = builtin_exists(cmd); - + /* Functions */ if (! is_valid && function_ok) is_valid = function_exists_no_autoload(cmd, vars); - + /* Abbreviations */ if (! is_valid && abbreviation_ok) is_valid = expand_abbreviation(cmd, NULL); - + /* Regular commands */ if (! is_valid && command_ok) is_valid = path_get_path(cmd, NULL, vars); - + /* Implicit cd */ if (! is_valid && implicit_cd_ok) is_valid = path_can_be_implicit_cd(cmd, NULL, working_directory.c_str(), vars); - + /* Return what we got */ return is_valid; } @@ -2048,16 +2048,16 @@ static bool command_is_valid(const wcstring &cmd, enum parse_statement_decoratio const highlighter_t::color_array_t & highlighter_t::highlight() { ASSERT_IS_BACKGROUND_THREAD(); - + const size_t length = buff.size(); assert(this->buff.size() == this->color_array.size()); - + if (length == 0) return color_array; /* Start out at zero */ std::fill(this->color_array.begin(), this->color_array.end(), 0); - + /* Parse the buffer */ parse_node_tree_t parse_tree; parse_tree_from_string(buff, parse_flag_continue_after_error | parse_flag_include_comments, &parse_tree, NULL); @@ -2111,7 +2111,7 @@ const highlighter_t::color_array_t & highlighter_t::highlight() { bool is_valid_cmd = false; wcstring cmd(buff, cmd_node->source_start, cmd_node->source_length); - + /* Try expanding it. If we cannot, it's an error. */ bool expanded = expand_one(cmd, EXPAND_SKIP_CMDSUBST | EXPAND_SKIP_VARIABLES | EXPAND_SKIP_JOBS); if (expanded && ! has_expand_reserved(cmd)) @@ -2149,7 +2149,7 @@ const highlighter_t::color_array_t & highlighter_t::highlight() break; } } - + if (this->cursor_pos <= this->buff.size()) { /* If the cursor is over an argument, and that argument is a valid path, underline it */ @@ -2160,7 +2160,7 @@ const highlighter_t::color_array_t & highlighter_t::highlight() /* Must be an argument with source */ if (node.type != symbol_argument || ! node.has_source()) continue; - + /* See if this node contains the cursor. We check <= source_length so that, when backspacing (and the cursor is just beyond the last token), we may still underline it */ if (this->cursor_pos >= node.source_start && this->cursor_pos - node.source_start <= node.source_length) { @@ -2180,7 +2180,7 @@ const highlighter_t::color_array_t & highlighter_t::highlight() } } } - + return color_array; } @@ -2188,7 +2188,7 @@ void highlight_shell_new_parser(const wcstring &buff, std::vectorforbidden_function.empty()) { return NULL; } const wcstring &forbidden_function_name = parser->forbidden_function.back(); - + /* Get the first job in the job list. */ const parse_node_t *first_job = tree.next_node_in_node_list(job_list, symbol_job, NULL); if (first_job == NULL) @@ -88,10 +88,10 @@ const parse_node_t *parse_execution_context_t::infinite_recursive_statement_in_j /* Here's the statement node we find that's infinite recursive */ const parse_node_t *infinite_recursive_statement = NULL; - + /* Get the list of statements */ const parse_node_tree_t::parse_node_list_t statements = tree.specific_statements_for_job(*first_job); - + /* Find all the decorated statements. We are interested in statements with no decoration (i.e. not command, not builtin) whose command expands to the forbidden function */ for (size_t i=0; i < statements.size(); i++) { @@ -101,19 +101,19 @@ const parse_node_t *parse_execution_context_t::infinite_recursive_statement_in_j { continue; } - + const parse_node_t &plain_statement = tree.find_child(statement, symbol_plain_statement); if (tree.decoration_for_plain_statement(plain_statement) != parse_statement_decoration_none) { /* This statement has a decoration like 'builtin' or 'command', and therefore is not infinite recursion. In particular this is what enables 'wrapper functions' */ continue; } - + /* Ok, this is an undecorated plain statement. Get and expand its command */ wcstring cmd; tree.command_for_plain_statement(plain_statement, src, &cmd); expand_one(cmd, EXPAND_SKIP_CMDSUBST | EXPAND_SKIP_VARIABLES); - + if (cmd == forbidden_function_name) { /* This is it */ @@ -125,7 +125,7 @@ const parse_node_t *parse_execution_context_t::infinite_recursive_statement_in_j break; } } - + assert(infinite_recursive_statement == NULL || infinite_recursive_statement->type == symbol_decorated_statement); return infinite_recursive_statement; } @@ -134,10 +134,10 @@ enum process_type_t parse_execution_context_t::process_type_for_command(const pa { assert(plain_statement.type == symbol_plain_statement); enum process_type_t process_type = EXTERNAL; - + /* Determine the process type, which depends on the statement decoration (command, builtin, etc) */ enum parse_statement_decoration_t decoration = tree.decoration_for_plain_statement(plain_statement); - + /* Do the "exec hack" */ if (decoration != parse_statement_decoration_command && cmd == L"exec") { @@ -203,7 +203,7 @@ parse_execution_context_t::execution_cancellation_reason_t parse_execution_conte bool parse_execution_context_t::job_is_simple_block(const parse_node_t &job_node) const { assert(job_node.type == symbol_job); - + /* Must have one statement */ const parse_node_t &statement = *get_child(job_node, 0, symbol_statement); const parse_node_t &specific_statement = *get_child(statement, 0); @@ -213,7 +213,7 @@ bool parse_execution_context_t::job_is_simple_block(const parse_node_t &job_node return false; } - + /* Must be no pipes */ const parse_node_t &continuation = *get_child(job_node, 1, symbol_job_continuation); if (continuation.child_count > 0) @@ -221,7 +221,7 @@ bool parse_execution_context_t::job_is_simple_block(const parse_node_t &job_node /* Multiple statements in this job, so there's pipes involved */ return false; } - + /* Check for arguments and redirections. All of the above types have an arguments / redirections list. It must be empty. */ const parse_node_t &args_and_redirections = tree.find_child(specific_statement, symbol_arguments_or_redirections_list); if (args_and_redirections.child_count > 0) @@ -229,7 +229,7 @@ bool parse_execution_context_t::job_is_simple_block(const parse_node_t &job_node /* Non-empty, we have an argument or redirection */ return false; } - + /* Ok, we are a simple block! */ return true; } @@ -237,12 +237,12 @@ bool parse_execution_context_t::job_is_simple_block(const parse_node_t &job_node parse_execution_result_t parse_execution_context_t::run_if_statement(const parse_node_t &statement) { assert(statement.type == symbol_if_statement); - + /* Push an if block */ if_block_t *ib = new if_block_t(); ib->node_offset = this->get_offset(statement); parser->push_block(ib); - + parse_execution_result_t result = parse_execution_success; /* We have a sequence of if clauses, with a final else, resulting in a single job list that we execute */ @@ -256,10 +256,10 @@ parse_execution_result_t parse_execution_context_t::run_if_statement(const parse result = parse_execution_cancelled; break; } - + assert(if_clause != NULL && else_clause != NULL); const parse_node_t &condition = *get_child(*if_clause, 1, symbol_job); - + /* Check the condition. We treat parse_execution_errored here as failure, in accordance with historic behavior */ parse_execution_result_t cond_ret = run_1_job(condition, ib); bool take_branch = (cond_ret == parse_execution_success) && proc_get_last_status() == EXIT_SUCCESS; @@ -296,7 +296,7 @@ parse_execution_result_t parse_execution_context_t::run_if_statement(const parse } } } - + /* Execute any job list we got */ if (job_list_to_execute != NULL) { @@ -305,7 +305,7 @@ parse_execution_result_t parse_execution_context_t::run_if_statement(const parse /* Done */ parser->pop_block(ib); - + return result; } @@ -313,19 +313,19 @@ parse_execution_result_t parse_execution_context_t::run_begin_statement(const pa { assert(header.type == symbol_begin_header); assert(contents.type == symbol_job_list); - + /* Basic begin/end block. Push a scope block. */ scope_block_t *sb = new scope_block_t(BEGIN); parser->push_block(sb); - + /* Run the job list */ parse_execution_result_t ret = run_job_list(contents, sb); - + /* Pop the block */ parser->pop_block(sb); - + return ret; - } +} /* Define a function */ parse_execution_result_t parse_execution_context_t::run_function_statement(const parse_node_t &header, const parse_node_t &contents) @@ -333,24 +333,24 @@ parse_execution_result_t parse_execution_context_t::run_function_statement(const assert(header.type == symbol_function_header); assert(contents.type == symbol_job_list); parse_execution_result_t result = parse_execution_success; - + /* Get arguments */ const parse_node_t *unmatched_wildcard = NULL; wcstring_list_t argument_list = this->determine_arguments(header, &unmatched_wildcard); - + if (unmatched_wildcard != NULL) { report_unmatched_wildcard_error(*unmatched_wildcard); result = parse_execution_errored; } - + if (result == parse_execution_success) { const wcstring contents_str = get_source(contents); wcstring error_str; int err = define_function(*parser, argument_list, contents_str, &error_str); proc_set_last_status(err); - + if (! error_str.empty()) { this->report_error(header, L"%ls", error_str.c_str()); @@ -358,42 +358,42 @@ parse_execution_result_t parse_execution_context_t::run_function_statement(const } } return result; - + } parse_execution_result_t parse_execution_context_t::run_block_statement(const parse_node_t &statement) { assert(statement.type == symbol_block_statement); - + const parse_node_t &block_header = *get_child(statement, 0, symbol_block_header); //block header const parse_node_t &header = *get_child(block_header, 0); //specific header type (e.g. for loop) const parse_node_t &contents = *get_child(statement, 2, symbol_job_list); //block contents - + parse_execution_result_t ret = parse_execution_success; switch (header.type) { case symbol_for_header: ret = run_for_statement(header, contents); break; - + case symbol_while_header: ret = run_while_statement(header, contents); break; - + case symbol_function_header: ret = run_function_statement(header, contents); break; - + case symbol_begin_header: ret = run_begin_statement(header, contents); break; - + default: fprintf(stderr, "Unexpected block header: %ls\n", header.describe().c_str()); PARSER_DIE(); break; } - + return ret; } @@ -401,11 +401,11 @@ parse_execution_result_t parse_execution_context_t::run_for_statement(const pars { assert(header.type == symbol_for_header); assert(block_contents.type == symbol_job_list); - + /* Get the variable name: `for var_name in ...` */ const parse_node_t &var_name_node = *get_child(header, 1, parse_token_type_string); const wcstring for_var_name = get_source(var_name_node); - + /* Get the contents to iterate over. */ const parse_node_t *unmatched_wildcard = NULL; wcstring_list_t argument_list = this->determine_arguments(header, &unmatched_wildcard); @@ -413,9 +413,9 @@ parse_execution_result_t parse_execution_context_t::run_for_statement(const pars { return report_unmatched_wildcard_error(*unmatched_wildcard); } - + parse_execution_result_t ret = parse_execution_success; - + for_block_t *fb = new for_block_t(for_var_name); parser->push_block(fb); @@ -431,16 +431,16 @@ parse_execution_result_t parse_execution_context_t::run_for_statement(const pars ret = parse_execution_cancelled; break; } - + const wcstring &for_variable = fb->variable; const wcstring &val = fb->sequence.back(); env_set(for_variable, val.c_str(), ENV_LOCAL); fb->sequence.pop_back(); fb->loop_status = LOOP_NORMAL; fb->skip = 0; - + this->run_job_list(block_contents, fb); - + if (this->cancellation_reason(fb) == execution_cancellation_loop_control) { /* Handle break or continue */ @@ -457,9 +457,9 @@ parse_execution_result_t parse_execution_context_t::run_for_statement(const pars } } } - + parser->pop_block(fb); - + return ret; } @@ -469,13 +469,13 @@ parse_execution_result_t parse_execution_context_t::run_switch_statement(const p assert(statement.type == symbol_switch_statement); parse_execution_result_t ret = parse_execution_success; const parse_node_t *matching_case_item = NULL; - + parse_execution_result_t result = parse_execution_success; - + /* Get the switch variable */ const parse_node_t &switch_value_node = *get_child(statement, 1, parse_token_type_string); const wcstring switch_value = get_source(switch_value_node); - + /* Expand it */ std::vector switch_values_expanded; int expand_ret = expand_string(switch_value, switch_values_expanded, EXPAND_NO_DESCRIPTIONS); @@ -484,11 +484,11 @@ parse_execution_result_t parse_execution_context_t::run_switch_statement(const p case EXPAND_ERROR: { result = report_error(switch_value_node, - _(L"Could not expand string '%ls'"), - switch_value.c_str()); + _(L"Could not expand string '%ls'"), + switch_value.c_str()); break; } - + case EXPAND_WILDCARD_NO_MATCH: { /* Store the node that failed to expand */ @@ -496,30 +496,30 @@ parse_execution_result_t parse_execution_context_t::run_switch_statement(const p ret = parse_execution_errored; break; } - + case EXPAND_WILDCARD_MATCH: case EXPAND_OK: { break; } } - + if (result == parse_execution_success && switch_values_expanded.size() != 1) { - result = report_error(switch_value_node, - _(L"switch: Expected exactly one argument, got %lu\n"), - switch_values_expanded.size()); + result = report_error(switch_value_node, + _(L"switch: Expected exactly one argument, got %lu\n"), + switch_values_expanded.size()); } const wcstring &switch_value_expanded = switch_values_expanded.at(0).completion; - + switch_block_t *sb = new switch_block_t(switch_value_expanded); parser->push_block(sb); - + if (result == parse_execution_success) { /* Expand case statements */ const parse_node_t *case_item_list = get_child(statement, 3, symbol_case_item_list); - + /* Loop while we don't have a match but do have more of the list */ while (matching_case_item == NULL && case_item_list != NULL) { @@ -528,7 +528,7 @@ parse_execution_result_t parse_execution_context_t::run_switch_statement(const p result = parse_execution_cancelled; break; } - + /* Get the next item and the remainder of the list */ const parse_node_t *case_item = tree.next_node_in_node_list(*case_item_list, symbol_case_item, &case_item_list); if (case_item == NULL) @@ -536,22 +536,22 @@ parse_execution_result_t parse_execution_context_t::run_switch_statement(const p /* No more items */ break; } - + /* Pull out the argument list */ const parse_node_t &arg_list = *get_child(*case_item, 1, symbol_argument_list); - + /* Expand arguments. We explicitly ignore unmatched_wildcard. That is, a case item list may have a wildcard that fails to expand to anything. */ const wcstring_list_t case_args = this->determine_arguments(arg_list, NULL); - + for (size_t i=0; i < case_args.size(); i++) { const wcstring &arg = case_args.at(i); - + /* Unescape wildcards so they can be expanded again */ wchar_t *unescaped_arg = parse_util_unescape_wildcards(arg.c_str()); bool match = wildcard_match(switch_value_expanded, unescaped_arg); free(unescaped_arg); - + /* If this matched, we're done */ if (match) { @@ -561,7 +561,7 @@ parse_execution_result_t parse_execution_context_t::run_switch_statement(const p } } } - + if (result == parse_execution_success && matching_case_item != NULL) { /* Success, evaluate the job list */ @@ -570,7 +570,7 @@ parse_execution_result_t parse_execution_context_t::run_switch_statement(const p } parser->pop_block(sb); - + return result; } @@ -578,41 +578,41 @@ parse_execution_result_t parse_execution_context_t::run_while_statement(const pa { assert(header.type == symbol_while_header); assert(block_contents.type == symbol_job_list); - + /* Push a while block */ while_block_t *wb = new while_block_t(); wb->status = WHILE_TEST_FIRST; wb->node_offset = this->get_offset(header); parser->push_block(wb); - + parse_execution_result_t ret = parse_execution_success; - + /* The condition and contents of the while loop, as a job and job list respectively */ const parse_node_t &while_condition = *get_child(header, 1, symbol_job); - + /* Run while the condition is true */ for (;;) { /* Check the condition */ parse_execution_result_t cond_result = this->run_1_job(while_condition, wb); - + /* We only continue on successful execution and EXIT_SUCCESS */ if (cond_result != parse_execution_success || proc_get_last_status() != EXIT_SUCCESS) { break; } - + /* Check cancellation */ if (this->should_cancel_execution(wb)) { ret = parse_execution_cancelled; break; } - - + + /* The block ought to go inside the loop (see #1212) */ this->run_job_list(block_contents, wb); - + if (this->cancellation_reason(wb) == execution_cancellation_loop_control) { /* Handle break or continue */ @@ -629,10 +629,10 @@ parse_execution_result_t parse_execution_context_t::run_while_statement(const pa } } } - + /* Done */ parser->pop_block(wb); - + return ret; } @@ -646,20 +646,20 @@ parse_execution_result_t parse_execution_context_t::report_error(const parse_nod error.source_start = node.source_start; error.source_length = node.source_length; error.code = parse_error_syntax; //hackish - + va_list va; va_start(va, fmt); error.text = vformat_string(fmt, va); va_end(va); - + /* Get a backtrace */ wcstring backtrace_and_desc; const parse_error_list_t error_list = parse_error_list_t(1, error); parser->get_backtrace(src, error_list, &backtrace_and_desc); - + fprintf(stderr, "%ls", backtrace_and_desc.c_str()); } - + return parse_execution_errored; } @@ -682,29 +682,29 @@ parse_execution_result_t parse_execution_context_t::report_unmatched_wildcard_er void parse_execution_context_t::handle_command_not_found(const wcstring &cmd_str, const parse_node_t &statement_node, int err_code) { assert(statement_node.type == symbol_plain_statement); - + /* We couldn't find the specified command. This is a non-fatal error. We want to set the exit status to 127, which is the standard number used by other shells like bash and zsh. */ - + const wchar_t * const cmd = cmd_str.c_str(); const wchar_t * const equals_ptr = wcschr(cmd, L'='); if (equals_ptr != NULL) { /* Try to figure out if this is a pure variable assignment (foo=bar), or if this appears to be running a command (foo=bar ruby...) */ - + const wcstring name_str = wcstring(cmd, equals_ptr - cmd); //variable name, up to the = const wcstring val_str = wcstring(equals_ptr + 1); //variable value, past the = - - + + const parse_node_tree_t::parse_node_list_t args = tree.find_nodes(statement_node, symbol_argument, 1); - + if (! args.empty()) { const wcstring argument = get_source(*args.at(0)); - + wcstring ellipsis_str = wcstring(1, ellipsis_char); if (ellipsis_str == L"$") ellipsis_str = L"..."; - + /* Looks like a command */ this->report_error(statement_node, _(L"Unknown command '%ls'. Did you mean to run %ls with a modified environment? Try 'env %ls=%ls %ls%ls'. See the help section on the set command by typing 'help set'."), @@ -726,7 +726,7 @@ void parse_execution_context_t::handle_command_not_found(const wcstring &cmd_str } else if (cmd[0]==L'$' || cmd[0] == VARIABLE_EXPAND || cmd[0] == VARIABLE_EXPAND_SINGLE) { - + const env_var_t val_wstr = env_get_string(cmd+1); const wchar_t *val = val_wstr.missing() ? NULL : val_wstr.c_str(); if (val) @@ -766,15 +766,15 @@ void parse_execution_context_t::handle_command_not_found(const wcstring &cmd_str command not found handler that can make better error messages */ - + wcstring_list_t event_args; event_args.push_back(cmd_str); event_fire_generic(L"fish_command_not_found", &event_args); - + /* Here we want to report an error (so it shows a backtrace), but with no text */ this->report_error(statement_node, L""); } - + /* Set the last proc status appropriately */ proc_set_last_status(err_code==ENOENT?STATUS_UNKNOWN_COMMAND:STATUS_NOT_EXECUTABLE); } @@ -785,15 +785,15 @@ parse_execution_result_t parse_execution_context_t::populate_plain_process(job_t assert(job != NULL); assert(proc != NULL); assert(statement.type == symbol_plain_statement); - + /* We may decide that a command should be an implicit cd */ bool use_implicit_cd = false; - + /* Get the command. We expect to always get it here. */ wcstring cmd; bool got_cmd = tree.command_for_plain_statement(statement, src, &cmd); assert(got_cmd); - + /* Expand it as a command. Return an error on failure. */ bool expanded = expand_one(cmd, EXPAND_SKIP_CMDSUBST | EXPAND_SKIP_VARIABLES); if (! expanded) @@ -801,26 +801,26 @@ parse_execution_result_t parse_execution_context_t::populate_plain_process(job_t report_error(statement, ILLEGAL_CMD_ERR_MSG, cmd.c_str()); return parse_execution_errored; } - + /* Determine the process type */ enum process_type_t process_type = process_type_for_command(statement, cmd); - + /* Check for stack overflow */ if (process_type == INTERNAL_FUNCTION && parser->forbidden_function.size() > FISH_MAX_STACK_DEPTH) { this->report_error(statement, CALL_STACK_LIMIT_EXCEEDED_ERR_MSG); return parse_execution_errored; } - + wcstring path_to_external_command; if (process_type == EXTERNAL) { /* Determine the actual command. This may be an implicit cd. */ bool has_command = path_get_path(cmd, &path_to_external_command); - + /* If there was no command, then we care about the value of errno after checking for it, to distinguish between e.g. no file vs permissions problem */ const int no_cmd_err_code = errno; - + /* If the specified command does not exist, and is undecorated, try using an implicit cd. */ if (! has_command && tree.decoration_for_plain_statement(statement) == parse_statement_decoration_none) { @@ -833,7 +833,7 @@ parse_execution_result_t parse_execution_context_t::populate_plain_process(job_t use_implicit_cd = path_can_be_implicit_cd(cmd, &implicit_cd_path); } } - + if (! has_command && ! use_implicit_cd) { /* No command */ @@ -841,7 +841,7 @@ parse_execution_result_t parse_execution_context_t::populate_plain_process(job_t return parse_execution_errored; } } - + /* The argument list and set of IO redirections that we will construct for the process */ wcstring_list_t argument_list; io_chain_t process_io_chain; @@ -851,7 +851,7 @@ parse_execution_result_t parse_execution_context_t::populate_plain_process(job_t argument_list.push_back(L"cd"); argument_list.push_back(cmd); path_to_external_command.clear(); - + /* If we have defined a wrapper around cd, use it, otherwise use the cd builtin */ process_type = function_exists(L"cd") ? INTERNAL_FUNCTION : INTERNAL_BUILTIN; } @@ -861,7 +861,7 @@ parse_execution_result_t parse_execution_context_t::populate_plain_process(job_t const parse_node_t *unmatched_wildcard = NULL; argument_list = this->determine_arguments(statement, &unmatched_wildcard); argument_list.insert(argument_list.begin(), cmd); - + /* If we were not able to expand any wildcards, here is the first one that failed */ if (unmatched_wildcard != NULL) { @@ -869,18 +869,18 @@ parse_execution_result_t parse_execution_context_t::populate_plain_process(job_t report_unmatched_wildcard_error(*unmatched_wildcard); return parse_execution_errored; } - + /* The set of IO redirections that we construct for the process */ if (! this->determine_io_chain(statement, &process_io_chain)) { return parse_execution_errored; } - + /* Determine the process type */ process_type = process_type_for_command(statement, cmd); } - + /* Populate the process */ proc->type = process_type; proc->set_argv(argument_list); @@ -893,24 +893,24 @@ parse_execution_result_t parse_execution_context_t::populate_plain_process(job_t wcstring_list_t parse_execution_context_t::determine_arguments(const parse_node_t &parent, const parse_node_t **out_unmatched_wildcard_node) { wcstring_list_t argument_list; - + /* Whether we failed to match any wildcards, and succeeded in matching any wildcards */ bool unmatched_wildcard = false, matched_wildcard = false; - + /* First node that failed to expand as a wildcard (if any) */ const parse_node_t *unmatched_wildcard_node = NULL; - + /* Get all argument nodes underneath the statement */ const parse_node_tree_t::parse_node_list_t argument_nodes = tree.find_nodes(parent, symbol_argument); argument_list.reserve(argument_nodes.size()); for (size_t i=0; i < argument_nodes.size(); i++) { const parse_node_t &arg_node = *argument_nodes.at(i); - + /* Expect all arguments to have source */ assert(arg_node.has_source()); const wcstring arg_str = arg_node.get_source(src); - + /* Expand this string */ std::vector arg_expanded; int expand_ret = expand_string(arg_str, arg_expanded, EXPAND_NO_DESCRIPTIONS); @@ -919,11 +919,11 @@ wcstring_list_t parse_execution_context_t::determine_arguments(const parse_node_ case EXPAND_ERROR: { this->report_error(arg_node, - _(L"Could not expand string '%ls'"), - arg_str.c_str()); + _(L"Could not expand string '%ls'"), + arg_str.c_str()); break; } - + case EXPAND_WILDCARD_NO_MATCH: { /* Store the node that failed to expand */ @@ -934,32 +934,32 @@ wcstring_list_t parse_execution_context_t::determine_arguments(const parse_node_ } break; } - + case EXPAND_WILDCARD_MATCH: { matched_wildcard = true; break; } - + case EXPAND_OK: { break; } } - + /* Now copy over any expanded arguments */ for (size_t i=0; i < arg_expanded.size(); i++) { argument_list.push_back(arg_expanded.at(i).completion); } } - + /* Return if we had a wildcard problem */ if (out_unmatched_wildcard_node != NULL && unmatched_wildcard && ! matched_wildcard) { *out_unmatched_wildcard_node = unmatched_wildcard_node; } - + return argument_list; } @@ -967,31 +967,31 @@ bool parse_execution_context_t::determine_io_chain(const parse_node_t &statement { io_chain_t result; bool errored = false; - + /* We are called with a statement of varying types. We require that the statement have an arguments_or_redirections_list child. */ const parse_node_t &args_and_redirections_list = tree.find_child(statement_node, symbol_arguments_or_redirections_list); - + /* Get all redirection nodes underneath the statement */ const parse_node_tree_t::parse_node_list_t redirect_nodes = tree.find_nodes(args_and_redirections_list, symbol_redirection); for (size_t i=0; i < redirect_nodes.size(); i++) { const parse_node_t &redirect_node = *redirect_nodes.at(i); - + int source_fd = -1; /* source fd */ wcstring target; /* file path or target fd */ enum token_type redirect_type = tree.type_for_redirection(redirect_node, src, &source_fd, &target); - + /* PCA: I can't justify this EXPAND_SKIP_VARIABLES flag. It was like this when I got here. */ bool target_expanded = expand_one(target, no_exec ? EXPAND_SKIP_VARIABLES : 0); if (! target_expanded || target.empty()) { /* Should improve this error message */ errored = report_error(redirect_node, - _(L"Invalid redirection target: %ls"), - target.c_str()); + _(L"Invalid redirection target: %ls"), + target.c_str()); } - - + + /* Generate the actual IO redirection */ shared_ptr new_io; assert(redirect_type != TOK_NONE); @@ -1011,8 +1011,8 @@ bool parse_execution_context_t::determine_io_chain(const parse_node_t &statement if (old_fd < 0 || errno || *end) { errored = report_error(redirect_node, - _(L"Requested redirection to '%ls', which is not a valid file descriptor"), - target.c_str()); + _(L"Requested redirection to '%ls', which is not a valid file descriptor"), + target.c_str()); } else { @@ -1021,7 +1021,7 @@ bool parse_execution_context_t::determine_io_chain(const parse_node_t &statement } break; } - + case TOK_REDIRECT_OUT: case TOK_REDIRECT_APPEND: case TOK_REDIRECT_IN: @@ -1032,7 +1032,7 @@ bool parse_execution_context_t::determine_io_chain(const parse_node_t &statement new_io.reset(new_io_file); break; } - + default: { // Should be unreachable @@ -1041,14 +1041,14 @@ bool parse_execution_context_t::determine_io_chain(const parse_node_t &statement break; } } - + /* Append the new_io if we got one */ if (new_io.get() != NULL) { result.push_back(new_io); } } - + if (out_chain && ! errored) { std::swap(*out_chain, result); @@ -1063,12 +1063,12 @@ parse_execution_result_t parse_execution_context_t::populate_boolean_process(job assert(bool_statement.type == symbol_boolean_statement); switch (bool_statement.production_idx) { - // These magic numbers correspond to productions for boolean_statement + // These magic numbers correspond to productions for boolean_statement case 0: // AND. Skip if the last job failed. skip_job = (proc_get_last_status() != 0); break; - + case 1: // OR. Skip if the last job succeeded. skip_job = (proc_get_last_status() == 0); @@ -1078,7 +1078,7 @@ parse_execution_result_t parse_execution_context_t::populate_boolean_process(job // NOT. Negate it. job_set_flag(job, JOB_NEGATE, !job_get_flag(job, JOB_NEGATE)); break; - + default: { fprintf(stderr, "Unexpected production in boolean statement\n"); @@ -1086,7 +1086,7 @@ parse_execution_result_t parse_execution_context_t::populate_boolean_process(job break; } } - + if (skip_job) { return parse_execution_skipped; @@ -1102,7 +1102,7 @@ parse_execution_result_t parse_execution_context_t::populate_block_process(job_t { /* We handle block statements by creating INTERNAL_BLOCK_NODE, that will bounce back to us when it's time to execute them */ assert(statement_node.type == symbol_block_statement || statement_node.type == symbol_if_statement || statement_node.type == symbol_switch_statement); - + /* The set of IO redirections that we construct for the process */ io_chain_t process_io_chain; bool errored = ! this->determine_io_chain(statement_node, &process_io_chain); @@ -1121,12 +1121,12 @@ parse_execution_result_t parse_execution_context_t::populate_job_process(job_t * { assert(statement_node.type == symbol_statement); assert(statement_node.child_count == 1); - + // Get the "specific statement" which is boolean / block / if / switch / decorated const parse_node_t &specific_statement = *get_child(statement_node, 0); - + parse_execution_result_t result = parse_execution_success; - + switch (specific_statement.type) { case symbol_boolean_statement: @@ -1134,7 +1134,7 @@ parse_execution_result_t parse_execution_context_t::populate_job_process(job_t * result = this->populate_boolean_process(job, proc, specific_statement); break; } - + case symbol_block_statement: case symbol_if_statement: case symbol_switch_statement: @@ -1142,7 +1142,7 @@ parse_execution_result_t parse_execution_context_t::populate_job_process(job_t * result = this->populate_block_process(job, proc, specific_statement); break; } - + case symbol_decorated_statement: { /* Get the plain statement. It will pull out the decoration itself */ @@ -1150,13 +1150,13 @@ parse_execution_result_t parse_execution_context_t::populate_job_process(job_t * result = this->populate_plain_process(job, proc, plain_statement); break; } - + default: fprintf(stderr, "'%ls' not handled by new parser yet\n", specific_statement.describe().c_str()); PARSER_DIE(); break; } - + return result; } @@ -1164,28 +1164,28 @@ parse_execution_result_t parse_execution_context_t::populate_job_process(job_t * parse_execution_result_t parse_execution_context_t::populate_job_from_job_node(job_t *j, const parse_node_t &job_node, const block_t *associated_block) { assert(job_node.type == symbol_job); - + /* Tell the job what its command is */ j->set_command(get_source(job_node)); - + /* We are going to construct process_t structures for every statement in the job. Get the first statement. */ const parse_node_t *statement_node = get_child(job_node, 0, symbol_statement); assert(statement_node != NULL); - + parse_execution_result_t result = parse_execution_success; - + /* Create processes. Each one may fail. */ std::vector processes; processes.push_back(new process_t()); result = this->populate_job_process(j, processes.back(), *statement_node); - + /* Construct process_ts for job continuations (pipelines), by walking the list until we hit the terminal (empty) job continuation */ const parse_node_t *job_cont = get_child(job_node, 1, symbol_job_continuation); assert(job_cont != NULL); while (result == parse_execution_success && job_cont->child_count > 0) { assert(job_cont->type == symbol_job_continuation); - + /* Handle the pipe, whose fd may not be the obvious stdout */ const parse_node_t &pipe_node = *get_child(*job_cont, 0, parse_token_type_pipe); int pipe_write_fd = fd_redirected_by_pipe(get_source(pipe_node)); @@ -1195,20 +1195,20 @@ parse_execution_result_t parse_execution_context_t::populate_job_from_job_node(j break; } processes.back()->pipe_write_fd = pipe_write_fd; - + /* Get the statement node and make a process from it */ const parse_node_t *statement_node = get_child(*job_cont, 1, symbol_statement); assert(statement_node != NULL); - + /* Store the new process (and maybe with an error) */ processes.push_back(new process_t()); result = this->populate_job_process(j, processes.back(), *statement_node); - + /* Get the next continuation */ job_cont = get_child(*job_cont, 2, symbol_job_continuation); assert(job_cont != NULL); } - + /* Return what happened */ if (result == parse_execution_success) { @@ -1236,7 +1236,7 @@ parse_execution_result_t parse_execution_context_t::populate_job_from_job_node(j parse_execution_result_t parse_execution_context_t::run_1_job(const parse_node_t &job_node, const block_t *associated_block) { parse_execution_result_t result = parse_execution_success; - + bool log_it = false; if (log_it) { @@ -1248,7 +1248,7 @@ parse_execution_result_t parse_execution_context_t::run_1_job(const parse_node_t { return parse_execution_cancelled; } - + // Get terminal modes struct termios tmodes = {}; if (get_is_interactive()) @@ -1260,10 +1260,10 @@ parse_execution_result_t parse_execution_context_t::run_1_job(const parse_node_t return parse_execution_errored; } } - + /* Increment the eval_level for the duration of this command */ scoped_push saved_eval_level(&eval_level, eval_level + 1); - + /* When we encounter a block construct (e.g. while loop) in the general case, we create a "block process" that has a pointer to its source. This allows us to handle block-level redirections. However, if there are no redirections, then we can just jump into the block directly, which is significantly faster. */ if (job_is_simple_block(job_node)) { @@ -1274,20 +1274,20 @@ parse_execution_result_t parse_execution_context_t::run_1_job(const parse_node_t { case symbol_block_statement: return this->run_block_statement(specific_statement); - + case symbol_if_statement: return this->run_if_statement(specific_statement); - + case symbol_switch_statement: return this->run_switch_statement(specific_statement); - + default: /* Other types should be impossible due to the specific_statement_type_is_redirectable_block check */ PARSER_DIE(); break; } } - + /* Profiling support */ long long start_time = 0, parse_time = 0, exec_time = 0; const bool do_profile = profile; @@ -1299,26 +1299,26 @@ parse_execution_result_t parse_execution_context_t::run_1_job(const parse_node_t profile_items.push_back(profile_item); start_time = get_time(); } - + job_t *j = new job_t(acquire_job_id(), block_io); j->tmodes = tmodes; job_set_flag(j, JOB_CONTROL, (job_control_mode==JOB_CONTROL_ALL) || ((job_control_mode == JOB_CONTROL_INTERACTIVE) && (get_is_interactive()))); - + job_set_flag(j, JOB_FOREGROUND, 1); - + job_set_flag(j, JOB_TERMINAL, job_get_flag(j, JOB_CONTROL) \ && (!is_subshell && !is_event)); - + job_set_flag(j, JOB_SKIP_NOTIFICATION, is_subshell \ || is_block \ || is_event \ || (!get_is_interactive())); - + /* Populate the job. This may fail for reasons like command_not_found. If this fails, an error will have been printed */ parse_execution_result_t pop_result = this->populate_job_from_job_node(j, job_node, associated_block); - + /* Clean up the job on failure or cancellation */ bool populated_job = (pop_result == parse_execution_success); if (! populated_job || this->should_cancel_execution(associated_block)) @@ -1327,8 +1327,8 @@ parse_execution_result_t parse_execution_context_t::run_1_job(const parse_node_t j = NULL; populated_job = false; } - - + + /* Store time it took to 'parse' the command */ if (do_profile) { @@ -1336,7 +1336,7 @@ parse_execution_result_t parse_execution_context_t::run_1_job(const parse_node_t profile_item->cmd = j->command(); profile_item->skipped=parser->current_block()->skip; } - + if (populated_job) { /* Success. Give the job to the parser - it will clean it up. */ @@ -1353,17 +1353,17 @@ parse_execution_result_t parse_execution_context_t::run_1_job(const parse_node_t break; } } - + /* Actually execute the job */ exec_job(*this->parser, j); - + /* Only external commands require a new fishd barrier */ if (!job_contained_external_command) { set_proc_had_barrier(false); } } - + /* If the job was skipped, we pretend it ran anyways */ if (result == parse_execution_skipped) { @@ -1378,10 +1378,10 @@ parse_execution_result_t parse_execution_context_t::run_1_job(const parse_node_t profile_item->exec=(int)(exec_time-parse_time); profile_item->skipped = ! populated_job; } - + /* Clean up jobs. */ job_reap(0); - + /* All done */ return result; } @@ -1389,7 +1389,7 @@ parse_execution_result_t parse_execution_context_t::run_1_job(const parse_node_t parse_execution_result_t parse_execution_context_t::run_job_list(const parse_node_t &job_list_node, const block_t *associated_block) { assert(job_list_node.type == symbol_job_list); - + parse_execution_result_t result = parse_execution_success; const parse_node_t *job_list = &job_list_node; while (job_list != NULL && ! should_cancel_execution(associated_block)) @@ -1398,13 +1398,13 @@ parse_execution_result_t parse_execution_context_t::run_job_list(const parse_nod // Try pulling out a job const parse_node_t *job = tree.next_node_in_node_list(*job_list, symbol_job, &job_list); - + if (job != NULL) { result = this->run_1_job(*job, associated_block); } } - + /* Returns the last job executed */ return result; } @@ -1412,24 +1412,24 @@ parse_execution_result_t parse_execution_context_t::run_job_list(const parse_nod parse_execution_result_t parse_execution_context_t::eval_node_at_offset(node_offset_t offset, const block_t *associated_block, const io_chain_t &io) { bool log_it = false; - + /* Don't ever expect to have an empty tree if this is called */ assert(! tree.empty()); assert(offset < tree.size()); - + /* Apply this block IO for the duration of this function */ scoped_push block_io_push(&block_io, io); - + const parse_node_t &node = tree.at(offset); - + if (log_it) { fprintf(stderr, "eval node: %ls\n", get_source(node).c_str()); } - + /* Currently, we only expect to execute the top level job list, or a block node. Assert that. */ assert(node.type == symbol_job_list || specific_statement_type_is_redirectable_block(node)); - + enum parse_execution_result_t status = parse_execution_success; switch (node.type) { @@ -1452,25 +1452,25 @@ parse_execution_result_t parse_execution_context_t::eval_node_at_offset(node_off } break; } - + case symbol_block_statement: status = this->run_block_statement(node); break; - + case symbol_if_statement: status = this->run_if_statement(node); break; - + case symbol_switch_statement: status = this->run_switch_statement(node); break; - + default: /* In principle, we could support other node types. However we never expect to be passed them - see above. */ fprintf(stderr, "Unexpected node %ls found in %s\n", node.describe().c_str(), __FUNCTION__); PARSER_DIE(); break; } - + return status; } diff --git a/parse_execution.h b/parse_execution.h index 6c022cb2b..89565d753 100644 --- a/parse_execution.h +++ b/parse_execution.h @@ -22,33 +22,33 @@ enum parse_execution_result_t /* The job did not execute due to some error (e.g. failed to wildcard expand). An error will have been printed and proc_last_status will have been set. */ parse_execution_errored, - + /* The job was cancelled (e.g. Ctrl-C) */ parse_execution_cancelled, - + /* The job was skipped (e.g. due to a not-taken 'and' command). This is a special return allowed only from the populate functions, not the run functions. */ parse_execution_skipped }; class parse_execution_context_t { - private: +private: const parse_node_tree_t tree; const wcstring src; io_chain_t block_io; parser_t * const parser; //parse_error_list_t errors; - + int eval_level; std::vector profile_items; - + /* No copying allowed */ parse_execution_context_t(const parse_execution_context_t&); parse_execution_context_t& operator=(const parse_execution_context_t&); - + /* Should I cancel? */ bool should_cancel_execution(const block_t *block) const; - + /* Ways that we can stop executing a block. These are in a sort of ascending order of importance, e.g. `exit` should trump `break` */ enum execution_cancellation_reason_t { @@ -58,32 +58,32 @@ class parse_execution_context_t execution_cancellation_exit }; execution_cancellation_reason_t cancellation_reason(const block_t *block) const; - + /* Report an error. Always returns true. */ parse_execution_result_t report_error(const parse_node_t &node, const wchar_t *fmt, ...); /* Wildcard error helper */ parse_execution_result_t report_unmatched_wildcard_error(const parse_node_t &unmatched_wildcard); - + /* Command not found support */ void handle_command_not_found(const wcstring &cmd, const parse_node_t &statement_node, int err_code); - + /* Utilities */ wcstring get_source(const parse_node_t &node) const; const parse_node_t *get_child(const parse_node_t &parent, node_offset_t which, parse_token_type_t expected_type = token_type_invalid) const; node_offset_t get_offset(const parse_node_t &node) const; const parse_node_t *infinite_recursive_statement_in_job_list(const parse_node_t &job_list, wcstring *out_func_name) const; - + /* Indicates whether a job is a simple block (one block, no redirections) */ bool job_is_simple_block(const parse_node_t &node) const; - + enum process_type_t process_type_for_command(const parse_node_t &plain_statement, const wcstring &cmd) const; - + /* These create process_t structures from statements */ parse_execution_result_t populate_job_process(job_t *job, process_t *proc, const parse_node_t &statement_node); parse_execution_result_t populate_boolean_process(job_t *job, process_t *proc, const parse_node_t &bool_statement); parse_execution_result_t populate_plain_process(job_t *job, process_t *proc, const parse_node_t &statement); parse_execution_result_t populate_block_process(job_t *job, process_t *proc, const parse_node_t &statement_node); - + /* These encapsulate the actual logic of various (block) statements. */ parse_execution_result_t run_block_statement(const parse_node_t &statement); parse_execution_result_t run_for_statement(const parse_node_t &header, const parse_node_t &contents); @@ -92,22 +92,22 @@ class parse_execution_context_t parse_execution_result_t run_while_statement(const parse_node_t &header, const parse_node_t &contents); parse_execution_result_t run_function_statement(const parse_node_t &header, const parse_node_t &contents); parse_execution_result_t run_begin_statement(const parse_node_t &header, const parse_node_t &contents); - + wcstring_list_t determine_arguments(const parse_node_t &parent, const parse_node_t **out_unmatched_wildcard_node); - + /* Determines the IO chain. Returns true on success, false on error */ bool determine_io_chain(const parse_node_t &statement, io_chain_t *out_chain); - + parse_execution_result_t run_1_job(const parse_node_t &job_node, const block_t *associated_block); parse_execution_result_t run_job_list(const parse_node_t &job_list_node, const block_t *associated_block); parse_execution_result_t populate_job_from_job_node(job_t *j, const parse_node_t &job_node, const block_t *associated_block); - - public: + +public: parse_execution_context_t(const parse_node_tree_t &t, const wcstring &s, parser_t *p); - + /* Start executing at the given node offset. Returns 0 if there was no error, 1 if there was an error */ parse_execution_result_t eval_node_at_offset(node_offset_t offset, const block_t *associated_block, const io_chain_t &io); - + }; diff --git a/parse_productions.cpp b/parse_productions.cpp index 4f40ce614..ff473268e 100644 --- a/parse_productions.cpp +++ b/parse_productions.cpp @@ -122,7 +122,7 @@ RESOLVE(statement) 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 @@ -135,14 +135,14 @@ RESOLVE(statement) { return 4; } - + // 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)) { return 4; } - + } switch (token1.type) @@ -173,12 +173,8 @@ RESOLVE(statement) case parse_keyword_end: return NO_PRODUCTION; - // 'in' is only special within a for_header - case parse_keyword_in: - case parse_keyword_none: - case parse_keyword_command: - case parse_keyword_builtin: - case parse_keyword_case: + // All other keywords fall through to decorated statement + default: return 4; } break; @@ -369,7 +365,7 @@ RESOLVE(decorated_statement) { return 0; } - + switch (token1.keyword) { default: diff --git a/parse_tree.cpp b/parse_tree.cpp index 5a24f455f..3643252cc 100644 --- a/parse_tree.cpp +++ b/parse_tree.cpp @@ -39,13 +39,13 @@ wcstring parse_error_t::describe(const wcstring &src, bool skip_caret) const { line_end = src.size(); } - + assert(line_end >= line_start); assert(source_start >= line_start); // Don't include the caret and line if we're interactive this is the first line, because then it's obvious bool skip_caret = (get_is_interactive() && source_start == 0); - + if (! skip_caret) { // Append the line of text. @@ -54,8 +54,8 @@ wcstring parse_error_t::describe(const wcstring &src, bool skip_caret) const result.push_back(L'\n'); } result.append(src, line_start, line_end - line_start); - - + + // Append the caret line. The input source may include tabs; for that reason we construct a "caret line" that has tabs in corresponding positions wcstring caret_space_line; caret_space_line.reserve(source_start - line_start); @@ -247,28 +247,28 @@ static wcstring token_type_user_presentable_description(parse_token_type_t type, { return format_string(L"keyword '%ls'", keyword_description(keyword).c_str()); } - + switch (type) { - /* Hackish. We only support the following types. */ + /* Hackish. We only support the following types. */ case symbol_statement: return L"a command"; - + case parse_token_type_string: return L"a string"; - + case parse_token_type_pipe: return L"a pipe"; - + case parse_token_type_redirection: return L"a redirection"; - + case parse_token_type_background: return L"a '&'"; - + case parse_token_type_end: return L"end of the statement"; - + default: return format_string(L"a %ls", token_type_description(type).c_str()); } @@ -351,14 +351,14 @@ static inline parse_token_type_t parse_token_type_from_tokenizer_token(enum toke static void dump_tree_recursive(const parse_node_tree_t &nodes, const wcstring &src, node_offset_t node_idx, size_t indent, wcstring *result, size_t *line, node_offset_t *inout_first_node_not_dumped) { assert(node_idx < nodes.size()); - + // Update first_node_not_dumped // This takes a bit of explanation. While it's true that a parse tree may be a "forest", its individual trees are "compact," meaning they are not interleaved. Thus we keep track of the largest node index as we descend a tree. One past the largest is the start of the next tree. if (*inout_first_node_not_dumped <= node_idx) { *inout_first_node_not_dumped = node_idx + 1; } - + const parse_node_t &node = nodes.at(node_idx); const size_t spacesPerIndent = 2; @@ -376,14 +376,14 @@ static void dump_tree_recursive(const parse_node_tree_t &nodes, const wcstring & { append_format(*result, L" <%lu children>", node.child_count); } - + if (node.has_source() && node.type == parse_token_type_string) { result->append(L": \""); result->append(src, node.source_start, node.source_length); result->append(L"\""); } - + if (node.type != parse_token_type_string) { if (node.has_source()) @@ -392,10 +392,10 @@ static void dump_tree_recursive(const parse_node_tree_t &nodes, const wcstring & } else { - append_format(*result, L" [no src]", (long)node.source_start, (long)node.source_length); + append_format(*result, L" [no src]", (long)node.source_start, (long)node.source_length); } } - + result->push_back(L'\n'); ++*line; for (size_t child_idx = node.child_start; child_idx < node.child_start + node.child_count; child_idx++) @@ -409,7 +409,7 @@ wcstring parse_dump_tree(const parse_node_tree_t &nodes, const wcstring &src) { if (nodes.empty()) return L"(empty!)"; - + node_offset_t first_node_not_dumped = 0; size_t line = 0; wcstring result; @@ -448,7 +448,7 @@ struct parse_stack_element_t } return result; } - + /* Returns a name that we can show to the user, e.g. "a command" */ wcstring user_presentable_description(void) const { @@ -461,19 +461,19 @@ class parse_ll_t { /* Traditional symbol stack of the LL parser */ std::vector symbol_stack; - + /* Parser output. This is a parse tree, but stored in an array. */ parse_node_tree_t nodes; /* Whether we ran into a fatal error, including parse errors or tokenizer errors */ bool fatal_errored; - + /* Whether we should collect error messages or not */ bool should_generate_error_messages; - + /* List of errors we have encountered */ parse_error_list_t errors; - + /* The symbol stack can contain terminal types or symbols. Symbols go on to do productions, but terminal types are just matched against input tokens. */ bool top_node_handle_terminal_types(parse_token_t token); @@ -521,7 +521,7 @@ class parse_ll_t } if (! count) fprintf(stderr, "\t\n"); } - + // Get the parent index. But we can't get the parent parse node yet, since it may be made invalid by adding children const size_t parent_node_idx = symbol_stack.back().node_idx; @@ -569,8 +569,8 @@ class parse_ll_t } } - public: - +public: + /* Constructor */ parse_ll_t() : fatal_errored(false), should_generate_error_messages(true) { @@ -581,31 +581,31 @@ class parse_ll_t /* Input */ void accept_tokens(parse_token_t token1, parse_token_t token2); - + /* Report tokenizer errors */ void report_tokenizer_error(parse_token_t token, int tok_err, const wchar_t *tok_error); - + /* Indicate if we hit a fatal error */ bool has_fatal_error(void) const { return this->fatal_errored; } - + /* Indicate whether we want to generate error messages */ void set_should_generate_error_messages(bool flag) { this->should_generate_error_messages = flag; } - + /* Clear the parse symbol stack (but not the node tree). Add a new job_list_t goal node. This is called from the constructor */ void reset_symbols(void); /* Clear the parse symbol stack and the node tree. Add a new job_list_t goal node. This is called from the constructor. */ void reset_symbols_and_nodes(void); - + /* Once parsing is complete, determine the ranges of intermediate nodes */ void determine_node_ranges(); - + /* Acquire output after parsing. This transfers directly from within self */ void acquire_output(parse_node_tree_t *output, parse_error_list_t *errors); }; @@ -684,7 +684,7 @@ void parse_ll_t::acquire_output(parse_node_tree_t *output, parse_error_list_t *e std::swap(*output, this->nodes); } this->nodes.clear(); - + if (errors != NULL) { std::swap(*errors, this->errors); @@ -727,7 +727,7 @@ void parse_ll_t::parse_error_unbalancing_token(parse_token_t token) case parse_keyword_end: this->parse_error(token, parse_error_unbalancing_end, L"'end' outside of a block"); break; - + case parse_keyword_else: this->parse_error(token, parse_error_unbalancing_else, L"'else' builtin not inside of if block"); break; @@ -735,7 +735,7 @@ void parse_ll_t::parse_error_unbalancing_token(parse_token_t token) case parse_keyword_case: this->parse_error(token, parse_error_unbalancing_case, L"'case' builtin not inside of switch block"); break; - + default: fprintf(stderr, "Unexpected token %ls passed to %s\n", token.describe().c_str(), __FUNCTION__); PARSER_DIE(); @@ -751,7 +751,7 @@ void parse_ll_t::parse_error_failed_production(struct parse_stack_element_t &sta if (this->should_generate_error_messages) { bool done = false; - + /* Check for || */ if (token.type == parse_token_type_pipe && token.source_start > 0) { @@ -764,7 +764,7 @@ void parse_ll_t::parse_error_failed_production(struct parse_stack_element_t &sta done = true; } } - + /* Check for && */ if (! done && token.type == parse_token_type_background && token.source_start > 0) { @@ -777,7 +777,7 @@ void parse_ll_t::parse_error_failed_production(struct parse_stack_element_t &sta done = true; } } - + if (! done) { const wcstring expected = stack_elem.user_presentable_description(); @@ -901,17 +901,17 @@ bool parse_ll_t::top_node_handle_terminal_types(parse_token_t token) { // Keyword failure. We should unify this with the 'matched' computation above. assert(stack_top.keyword != parse_keyword_none && stack_top.keyword != token.keyword); - + // Check to see which keyword we got which was considered wrong switch (token.keyword) { - // Some keywords are only valid in certain contexts. If this cascaded all the way down through the outermost job_list, it was not in a valid context. + // Some keywords are only valid in certain contexts. If this cascaded all the way down through the outermost job_list, it was not in a valid context. case parse_keyword_case: case parse_keyword_end: case parse_keyword_else: this->parse_error_unbalancing_token(token); break; - + case parse_keyword_none: { // This is a random other string (not a keyword) @@ -920,7 +920,7 @@ bool parse_ll_t::top_node_handle_terminal_types(parse_token_t token) break; } - + default: { // Got a real keyword we can report @@ -963,7 +963,7 @@ void parse_ll_t::accept_tokens(parse_token_t token1, parse_token_t token2) err_node.source_length = token1.source_length; nodes.push_back(err_node); consumed = true; - + /* tokenizer errors are fatal */ if (token1.type == parse_special_type_tokenizer_error) this->fatal_errored = true; @@ -999,22 +999,22 @@ void parse_ll_t::accept_tokens(parse_token_t token1, parse_token_t token2) else { bool is_terminate = (token1.type == parse_token_type_terminate); - + // When a job_list encounters something like 'else', it returns an empty production to return control to the outer block. But if it's unbalanced, then we'll end up with an empty stack! So make sure that doesn't happen. This is the primary mechanism by which we detect e.g. unbalanced end. However, if we get a true terminate token, then we allow (expect) this to empty the stack if (symbol_stack.size() == 1 && production_is_empty(production) && ! is_terminate) { this->parse_error_unbalancing_token(token1); break; } - + // Manipulate the symbol stack. // Note that stack_elem is invalidated by popping the stack. symbol_stack_pop_push_production(production); - + // Expect to not have an empty stack, unless this was the terminate type // Note we may not have an empty stack with the terminate type (i.e. incomplete input) assert(is_terminate || ! symbol_stack.empty()); - + if (symbol_stack.empty()) { break; @@ -1082,7 +1082,7 @@ static inline parse_token_t next_parse_token(tokenizer_t *tok) { return kTerminalToken; } - + token_type tok_type = static_cast(tok_last_type(tok)); int tok_start = tok_get_pos(tok); size_t tok_extent = tok_get_extent(tok); @@ -1090,7 +1090,7 @@ static inline parse_token_t next_parse_token(tokenizer_t *tok) const wchar_t *tok_txt = tok_last(tok); parse_token_t result; - + /* Set the type, keyword, and whether there's a dash prefix. Note that this is quite sketchy, because it ignores quotes. This is the historical behavior. For example, `builtin --names` lists builtins, but `builtin "--names"` attempts to run --names as a command. Amazingly as of this writing (10/12/13) nobody seems to have noticed this. Squint at it really hard and it even starts to look like a feature. */ result.type = parse_token_type_from_tokenizer_token(tok_type); result.keyword = keyword_for_token(tok_type, tok_txt); @@ -1098,7 +1098,7 @@ static inline parse_token_t next_parse_token(tokenizer_t *tok) result.is_help_argument = result.has_dash_prefix && is_help_argument(tok_txt); result.source_start = (size_t)tok_start; result.source_length = tok_extent; - + tok_next(tok); return result; } @@ -1112,15 +1112,15 @@ bool parse_tree_from_string(const wcstring &str, parse_tree_flags_t parse_flags, tok_flags_t tok_options = 0; if (parse_flags & parse_flag_include_comments) tok_options |= TOK_SHOW_COMMENTS; - + if (parse_flags & parse_flag_accept_incomplete_tokens) tok_options |= TOK_ACCEPT_UNFINISHED; - + if (errors == NULL) tok_options |= TOK_SQUASH_ERRORS; - + tokenizer_t tok = tokenizer_t(str.c_str(), tok_options); - + /* We are an LL(2) parser. We pass two tokens at a time. New tokens come in at index 1. Seed our queue with an initial token at index 1. */ parse_token_t queue[2] = {kInvalidToken, kInvalidToken}; @@ -1130,25 +1130,25 @@ bool parse_tree_from_string(const wcstring &str, parse_tree_flags_t parse_flags, /* Push a new token onto the queue */ queue[0] = queue[1]; queue[1] = next_parse_token(&tok); - + /* If we are leaving things unterminated, then don't pass parse_token_type_terminate */ if (queue[0].type == parse_token_type_terminate && (parse_flags & parse_flag_leave_unterminated)) { break; } - + /* Pass these two tokens, unless we're still loading the queue. We know that queue[0] is valid; queue[1] may be invalid. */ if (token_count > 0) { parser.accept_tokens(queue[0], queue[1]); } - + /* Handle tokenizer errors. This is a hack because really the parser should report this for itself; but it has no way of getting the tokenizer message */ if (queue[1].type == parse_special_type_tokenizer_error) { parser.report_tokenizer_error(queue[1], tok_get_error(&tok), tok_last(&tok)); } - + /* Handle errors */ if (parser.has_fatal_error()) { @@ -1172,7 +1172,7 @@ bool parse_tree_from_string(const wcstring &str, parse_tree_flags_t parse_flags, // Teach each node where its source range is parser.determine_node_ranges(); - + // Acquire the output from the parser parser.acquire_output(output, errors); @@ -1181,7 +1181,7 @@ bool parse_tree_from_string(const wcstring &str, parse_tree_flags_t parse_flags, //fprintf(stderr, "Tree (%ld nodes):\n%ls", this->parser->nodes.size(), result.c_str()); fprintf(stderr, "%lu nodes, node size %lu, %lu bytes\n", output->size(), sizeof(parse_node_t), output->size() * sizeof(parse_node_t)); #endif - + // Indicate if we had a fatal error return ! parser.has_fatal_error(); } @@ -1189,7 +1189,7 @@ bool parse_tree_from_string(const wcstring &str, parse_tree_flags_t parse_flags, const parse_node_t *parse_node_tree_t::get_child(const parse_node_t &parent, node_offset_t which, parse_token_type_t expected_type) const { const parse_node_t *result = NULL; - + /* We may get nodes with no children if we had an imcomplete parse. Don't consider than an error */ if (parent.child_count > 0) { @@ -1198,7 +1198,7 @@ const parse_node_t *parse_node_tree_t::get_child(const parse_node_t &parent, nod if (child_offset < this->size()) { result = &this->at(child_offset); - + /* If we are given an expected type, then the node must be null or that type */ assert(expected_type == token_type_invalid || expected_type == result->type); } @@ -1321,7 +1321,7 @@ const parse_node_t *parse_node_tree_t::find_node_matching_source_location(parse_ for (size_t idx=0; idx < len; idx++) { const parse_node_t &node = this->at(idx); - + /* Types must match */ if (node.type != type) continue; @@ -1329,11 +1329,11 @@ const parse_node_t *parse_node_tree_t::find_node_matching_source_location(parse_ /* Must contain source location */ if (! node.location_in_or_at_end_of_source_range(source_loc)) continue; - + /* If a parent is given, it must be an ancestor */ if (parent != NULL && node_has_ancestor(*this, node, *parent)) continue; - + /* Found it */ result = &node; break; @@ -1390,7 +1390,7 @@ bool parse_node_tree_t::statement_is_in_pipeline(const parse_node_t &node, bool // This accepts a few statement types bool result = false; const parse_node_t *ancestor = &node; - + // If we're given a plain statement, try to get its decorated statement parent if (ancestor && ancestor->type == symbol_plain_statement) ancestor = this->get_parent(*ancestor, symbol_decorated_statement); @@ -1398,7 +1398,7 @@ bool parse_node_tree_t::statement_is_in_pipeline(const parse_node_t &node, bool ancestor = this->get_parent(*ancestor, symbol_statement); if (ancestor) ancestor = this->get_parent(*ancestor); - + if (ancestor) { if (ancestor->type == symbol_job_continuation) @@ -1413,7 +1413,7 @@ bool parse_node_tree_t::statement_is_in_pipeline(const parse_node_t &node, bool result = (continuation != NULL && continuation->child_count > 0); } } - + return result; } @@ -1423,7 +1423,7 @@ enum token_type parse_node_tree_t::type_for_redirection(const parse_node_t &redi enum token_type result = TOK_NONE; const parse_node_t *redirection_primitive = this->get_child(redirection_node, 0, parse_token_type_redirection); //like 2> const parse_node_t *redirection_target = this->get_child(redirection_node, 1, parse_token_type_string); //like &1 or file path - + if (redirection_primitive != NULL && redirection_primitive->has_source()) { result = redirection_type_for_string(redirection_primitive->get_source(src), out_fd); @@ -1453,10 +1453,10 @@ parse_node_tree_t::parse_node_list_t parse_node_tree_t::specific_statements_for_ { assert(job.type == symbol_job); parse_node_list_t result; - + /* Initial statement (non-specific) */ result.push_back(get_child(job, 0, symbol_statement)); - + /* Our cursor variable. Walk over the list of continuations. */ const parse_node_t *continuation = get_child(job, 1, symbol_job_continuation); while (continuation != NULL && continuation->child_count > 0) @@ -1464,7 +1464,7 @@ parse_node_tree_t::parse_node_list_t parse_node_tree_t::specific_statements_for_ result.push_back(get_child(*continuation, 1, symbol_statement)); continuation = get_child(*continuation, 2, symbol_job_continuation); } - + /* Result now contains a list of statements. But we want a list of specific statements e.g. symbol_switch_statement. So replace them in-place in the vector. */ for (size_t i=0; i < result.size(); i++) { @@ -1472,25 +1472,25 @@ parse_node_tree_t::parse_node_list_t parse_node_tree_t::specific_statements_for_ assert(statement->type == symbol_statement); result.at(i) = this->get_child(*statement, 0); } - + return result; } const parse_node_t *parse_node_tree_t::next_node_in_node_list(const parse_node_t &node_list, parse_token_type_t entry_type, const parse_node_t **out_list_tail) const { parse_token_type_t list_type = node_list.type; - + /* Paranoia - it doesn't make sense for a list type to contain itself */ assert(list_type != entry_type); - + const parse_node_t *list_cursor = &node_list; const parse_node_t *list_entry = NULL; - + /* Loop while we don't have an item but do have a list. Note that not every node in the list may contain an in item that we care about - e.g. job_list contains blank lines as a production */ while (list_entry == NULL && list_cursor != NULL) { const parse_node_t *next_cursor = NULL; - + /* Walk through the children */ for (size_t i=0; i < list_cursor->child_count; i++) { @@ -1509,7 +1509,7 @@ const parse_node_t *parse_node_tree_t::next_node_in_node_list(const parse_node_t /* Go to the next entry, even if it's NULL */ list_cursor = next_cursor; } - + /* Return what we got */ assert(list_cursor == NULL || list_cursor->type == list_type); assert(list_entry == NULL || list_entry->type == entry_type); diff --git a/parse_tree.h b/parse_tree.h index 530b5c25c..255ab467f 100644 --- a/parse_tree.h +++ b/parse_tree.h @@ -25,7 +25,7 @@ struct parse_error_t { /** Text of the error */ wcstring text; - + /** Code for the error */ enum parse_error_code_t code; @@ -62,16 +62,16 @@ enum /* Attempt to build a "parse tree" no matter what. This may result in a 'forest' of disconnected trees. This is intended to be used by syntax highlighting. */ parse_flag_continue_after_error = 1 << 0, - + /* Include comment tokens */ parse_flag_include_comments = 1 << 1, - + /* Indicate that the tokenizer should accept incomplete tokens */ parse_flag_accept_incomplete_tokens = 1 << 2, - + /* Indicate that the parser should not generate the terminate token, allowing an 'unfinished' tree where some nodes may have no productions. */ parse_flag_leave_unterminated = 1 << 3 - + }; typedef unsigned int parse_tree_flags_t; @@ -93,7 +93,7 @@ public: /* Length of our range in the source code */ size_t source_length; - + /* Parent */ node_offset_t parent; @@ -123,7 +123,7 @@ public: { return source_start != (size_t)(-1); } - + /* Gets source for the node, or the empty string if it has no source */ wcstring get_source(const wcstring &str) const { @@ -132,7 +132,7 @@ public: else return wcstring(str, this->source_start, this->source_length); } - + /* Returns whether the given location is within the source range or at its end */ bool location_in_or_at_end_of_source_range(size_t loc) const { @@ -149,50 +149,50 @@ public: /* Get the node corresponding to a child of the given node, or NULL if there is no such child. If expected_type is provided, assert that the node has that type. */ const parse_node_t *get_child(const parse_node_t &parent, node_offset_t which, parse_token_type_t expected_type = token_type_invalid) const; - + /* Find the first direct child of the given node of the given type. asserts on failure */ const parse_node_t &find_child(const parse_node_t &parent, parse_token_type_t type) const; - + /* Get the node corresponding to the parent of the given node, or NULL if there is no such child. If expected_type is provided, only returns the parent if it is of that type. Note the asymmetry: get_child asserts since the children are known, but get_parent does not, since the parent may not be known. */ const parse_node_t *get_parent(const parse_node_t &node, parse_token_type_t expected_type = token_type_invalid) const; - + /* Returns the first ancestor of the given type, or NULL. */ const parse_node_t *get_first_ancestor_of_type(const parse_node_t &node, parse_token_type_t desired_type) const; /* Find all the nodes of a given type underneath a given node, up to max_count of them */ typedef std::vector parse_node_list_t; parse_node_list_t find_nodes(const parse_node_t &parent, parse_token_type_t type, size_t max_count = (size_t)(-1)) const; - + /* 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. */ const parse_node_t *find_last_node_of_type(parse_token_type_t type, const parse_node_t *parent = NULL) const; - + /* Finds a node containing the given source location. If 'parent' is not NULL, it must be an ancestor. */ const parse_node_t *find_node_matching_source_location(parse_token_type_t type, size_t source_loc, const parse_node_t *parent) const; - + /* Indicate if the given argument_list or arguments_or_redirections_list is a root list, or has a parent */ bool argument_list_is_root(const parse_node_t &node) const; - + /* Utilities */ - + /* Given a plain statement, get the decoration (from the parent node), or none if there is no decoration */ enum parse_statement_decoration_t decoration_for_plain_statement(const parse_node_t &node) const; - + /* Given a plain statement, get the command by reference (from the child node). Returns true if successful. Clears the command on failure. */ bool command_for_plain_statement(const parse_node_t &node, const wcstring &src, wcstring *out_cmd) const; - + /* Given a plain statement, return true if the statement is part of a pipeline. If include_first is set, the first command in a pipeline is considered part of it; otherwise only the second or additional commands are */ bool statement_is_in_pipeline(const parse_node_t &node, bool include_first) const; - + /* Given a redirection, get the redirection type (or TOK_NONE) and target (file path, or fd) */ enum token_type type_for_redirection(const parse_node_t &node, const wcstring &src, int *out_fd, wcstring *out_target) const; - + /* If the given node is a block statement, returns the header node (for_header, while_header, begin_header, or function_header). Otherwise returns NULL */ const parse_node_t *header_node_for_block_statement(const parse_node_t &node) const; - + /* 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 if we've exhausted the list. */ const parse_node_t *next_node_in_node_list(const parse_node_t &node_list, parse_token_type_t item_type, const parse_node_t **list_tail) const; - + /* Given a job, return all of its statements. These are 'specific statements' (e.g. symbol_decorated_statement, not symbol_statement) */ parse_node_list_t specific_statements_for_job(const parse_node_t &job) const; }; @@ -231,7 +231,7 @@ bool parse_tree_from_string(const wcstring &str, parse_tree_flags_t flags, parse case_item_list = | case_item case_item_list | case_item_list - + case_item = CASE argument_list STATEMENT_TERMINATOR job_list block_statement = block_header job_list end_command arguments_or_redirections_list @@ -239,7 +239,7 @@ bool parse_tree_from_string(const wcstring &str, parse_tree_flags_t flags, parse for_header = FOR var_name IN argument_list while_header = WHILE job begin_header = BEGIN - + # Functions take arguments, and require at least one (the name). No redirections allowed. function_header = FUNCTION argument argument_list @@ -258,13 +258,13 @@ bool parse_tree_from_string(const wcstring &str, parse_tree_flags_t flags, parse argument_or_redirection arguments_or_redirections_list argument_or_redirection = argument | redirection argument = - + redirection = terminator = | optional_background = | - + end_command = END */ diff --git a/parse_util.cpp b/parse_util.cpp index 466217175..42fbb49cf 100644 --- a/parse_util.cpp +++ b/parse_util.cpp @@ -241,11 +241,11 @@ int parse_util_locate_cmdsubst_range(const wcstring &str, size_t *inout_cursor_o out_contents->clear(); *out_start = 0; *out_end = str.size(); - + /* Nothing to do if the offset is at or past the end of the string. */ if (*inout_cursor_offset >= str.size()) return 0; - + /* Defer to the wonky version */ const wchar_t * const buff = str.c_str(); const wchar_t * const valid_range_start = buff + *inout_cursor_offset, *valid_range_end = buff + str.size(); @@ -256,15 +256,15 @@ int parse_util_locate_cmdsubst_range(const wcstring &str, size_t *inout_cursor_o /* The command substitutions must not be NULL and must be in the valid pointer range, and the end must be bigger than the beginning */ assert(cmdsub_begin != NULL && cmdsub_begin >= valid_range_start && cmdsub_begin <= valid_range_end); assert(cmdsub_end != NULL && cmdsub_end > cmdsub_begin && cmdsub_end >= valid_range_start && cmdsub_end <= valid_range_end); - + /* Assign the substring to the out_contents */ const wchar_t *interior_begin = cmdsub_begin + 1; out_contents->assign(interior_begin, cmdsub_end - interior_begin); - + /* Return the start and end */ *out_start = cmdsub_begin - buff; *out_end = cmdsub_end - buff; - + /* Update the inout_cursor_offset. Note this may cause it to exceed str.size(), though overflow is not likely */ *inout_cursor_offset = 1 + *out_end; } @@ -803,9 +803,9 @@ wcstring parse_util_escape_string_with_quote(const wcstring &cmd, wchar_t quote) /* We are given a parse tree, the index of a node within the tree, its indent, and a vector of indents the same size as the original source string. Set the indent correspdonding to the node's source range, if appropriate. trailing_indent is the indent for nodes with unrealized source, i.e. if I type 'if false ' then we have an if node with an empty job list (without source) but we want the last line to be indented anyways. - + switch statements also indent. - + max_visited_node_idx is the largest index we visited. */ static void compute_indents_recursive(const parse_node_tree_t &tree, node_offset_t node_idx, int node_indent, parse_token_type_t parent_type, std::vector *indents, int *trailing_indent, node_offset_t *max_visited_node_idx) @@ -813,16 +813,16 @@ static void compute_indents_recursive(const parse_node_tree_t &tree, node_offset /* Guard against incomplete trees */ if (node_idx > tree.size()) return; - + /* Update max_visited_node_idx */ if (node_idx > *max_visited_node_idx) *max_visited_node_idx = node_idx; /* We could implement this by utilizing the fish grammar. But there's an easy trick instead: almost everything that wraps a job list should be indented by 1. So just find all of the job lists. One exception is switch; the other exception is job_list itself: a job_list is a job and a job_list, and we want that child list to be indented the same as the parent. So just find all job_lists whose parent is not a job_list, and increment their indent by 1. */ - + const parse_node_t &node = tree.at(node_idx); const parse_token_type_t node_type = node.type; - + /* Increment the indent if we are either a root job_list, or root case_item_list */ const bool is_root_job_list = (node_type == symbol_job_list && parent_type != symbol_job_list); const bool is_root_case_item_list = (node_type == symbol_case_item_list && parent_type != symbol_case_item_list); @@ -830,22 +830,22 @@ static void compute_indents_recursive(const parse_node_tree_t &tree, node_offset { node_indent += 1; } - + /* If we have source, store the trailing indent unconditionally. If we do not have source, store the trailing indent only if ours is bigger; this prevents the trailing "run" of terminal job lists from affecting the trailing indent. For example, code like this: - + if foo - + will be parsed as this: - + job_list job if_statement job [if] job_list [empty] job_list [empty] - + There's two "terminal" job lists, and we want the innermost one. - + Note we are relying on the fact that nodes are in the same order as the source, i.e. an in-order traversal of the node tree also traverses the source from beginning to end. */ if (node.has_source() || node_indent > *trailing_indent) @@ -853,7 +853,7 @@ static void compute_indents_recursive(const parse_node_tree_t &tree, node_offset *trailing_indent = node_indent; } - + /* Store the indent into the indent array */ if (node.has_source()) { @@ -861,7 +861,7 @@ static void compute_indents_recursive(const parse_node_tree_t &tree, node_offset indents->at(node.source_start) = node_indent; } - + /* Recursive to all our children */ for (node_offset_t idx = 0; idx < node.child_count; idx++) { @@ -875,31 +875,31 @@ std::vector parse_util_compute_indents(const wcstring &src) /* Make a vector the same size as the input string, which contains the indents. Initialize them to -1. */ const size_t src_size = src.size(); std::vector indents(src_size, -1); - + /* Parse the string. We pass continue_after_error to produce a forest; the trailing indent of the last node we visited becomes the input indent of the next. I.e. in the case of 'switch foo ; cas', we get an invalid parse tree (since 'cas' is not valid) but we indent it as if it were a case item list */ parse_node_tree_t tree; parse_tree_from_string(src, parse_flag_continue_after_error | parse_flag_accept_incomplete_tokens, &tree, NULL /* errors */); - + /* Start indenting at the first node. If we have a parse error, we'll have to start indenting from the top again */ node_offset_t start_node_idx = 0; int last_trailing_indent = 0; - + while (start_node_idx < tree.size()) { /* The indent that we'll get for the last line */ int trailing_indent = 0; - + /* Biggest offset we visited */ node_offset_t max_visited_node_idx = 0; - + /* Invoke the recursive version. As a hack, pass job_list for the 'parent' token type, which will prevent the really-root job list from indenting */ compute_indents_recursive(tree, start_node_idx, last_trailing_indent, symbol_job_list, &indents, &trailing_indent, &max_visited_node_idx); - + /* We may have more to indent. The trailing indent becomes our current indent. Start at the node after the last we visited. */ last_trailing_indent = trailing_indent; start_node_idx = max_visited_node_idx + 1; } - + int last_indent = 0; for (size_t i=0; i parse_util_compute_indents(const wcstring &src) break; indents.at(suffix_idx) = last_trailing_indent; } - + return indents; } @@ -942,12 +942,12 @@ static bool append_syntax_error(parse_error_list_t *errors, const parse_node_t & error.source_start = node.source_start; error.source_length = node.source_length; error.code = parse_error_syntax; - + va_list va; va_start(va, fmt); error.text = vformat_string(fmt, va); va_end(va); - + errors->push_back(error); return true; } @@ -984,14 +984,14 @@ parser_test_error_bits_t parse_util_detect_errors(const wcstring &buff_src, pars { parse_node_tree_t node_tree; parse_error_list_t parse_errors; - + // Whether we encountered a parse error bool errored = false; - + // Whether we encountered an unclosed block // We detect this via an 'end_command' block without source bool has_unclosed_block = false; - + // Whether there's an unclosed quote, and therefore unfinished bool has_unclosed_quote = false; @@ -1017,12 +1017,12 @@ parser_test_error_bits_t parse_util_detect_errors(const wcstring &buff_src, pars { errored = true; } - + // Expand all commands // Verify 'or' and 'and' not used inside pipelines // Verify pipes via parser_is_pipe_forbidden // Verify return only within a function - + if (! errored) { const size_t node_tree_size = node_tree.size(); @@ -1054,7 +1054,7 @@ parser_test_error_bits_t parse_util_detect_errors(const wcstring &buff_src, pars { errored = append_syntax_error(&parse_errors, node, ILLEGAL_CMD_ERR_MSG, command.c_str()); } - + // Check that pipes are sound if (! errored && parser_is_pipe_forbidden(command)) { @@ -1064,7 +1064,7 @@ parser_test_error_bits_t parse_util_detect_errors(const wcstring &buff_src, pars errored = append_syntax_error(&parse_errors, node, EXEC_ERR_MSG, command.c_str()); } } - + // Check that we don't return from outside a function // But we allow it if it's 'return --help' if (! errored && command == L"return") @@ -1087,7 +1087,7 @@ parser_test_error_bits_t parse_util_detect_errors(const wcstring &buff_src, pars errored = append_syntax_error(&parse_errors, node, INVALID_RETURN_ERR_MSG); } } - + // Check that we don't break or continue from outside a loop if (! errored && (command == L"break" || command == L"continue")) { @@ -1108,13 +1108,13 @@ parser_test_error_bits_t parse_util_detect_errors(const wcstring &buff_src, pars found_loop = true; end_search = true; break; - + case symbol_function_header: // this is a function header, so we cannot break or continue. We stop our search here. found_loop = false; end_search = true; break; - + default: // most likely begin / end style block, which makes no difference break; @@ -1122,7 +1122,7 @@ parser_test_error_bits_t parse_util_detect_errors(const wcstring &buff_src, pars } ancestor = node_tree.get_parent(*ancestor); } - + if (! found_loop && ! first_argument_is_help(node_tree, node, buff_src)) { errored = append_syntax_error(&parse_errors, node, (command == L"break" ? INVALID_BREAK_ERR_MSG : INVALID_CONTINUE_ERR_MSG)); @@ -1134,13 +1134,13 @@ parser_test_error_bits_t parse_util_detect_errors(const wcstring &buff_src, pars } parser_test_error_bits_t res = 0; - + if (errored) res |= PARSER_TEST_ERROR; if (has_unclosed_block || has_unclosed_quote) res |= PARSER_TEST_INCOMPLETE; - + if (out_errors) { out_errors->swap(parse_errors); diff --git a/parse_util.h b/parse_util.h index 862e5a621..921932027 100644 --- a/parse_util.h +++ b/parse_util.h @@ -41,11 +41,11 @@ int parse_util_locate_cmdsubst(const wchar_t *in, */ int parse_util_locate_cmdsubst_range(const wcstring &str, - size_t *inout_cursor_offset, - wcstring *out_contents, - size_t *out_start, - size_t *out_end, - bool accept_incomplete); + size_t *inout_cursor_offset, + wcstring *out_contents, + size_t *out_start, + size_t *out_end, + bool accept_incomplete); /** Find the beginning and end of the command substitution under the diff --git a/parser.cpp b/parser.cpp index ffcec02ad..9bb1e6490 100644 --- a/parser.cpp +++ b/parser.cpp @@ -372,7 +372,7 @@ void parser_t::push_block(block_t *new_current) new_current->job = 0; new_current->loop_status=LOOP_NORMAL; - + this->block_stack.push_back(new_current); if ((new_current->type() != FUNCTION_DEF) && @@ -553,7 +553,7 @@ void parser_t::error(int ec, size_t p, const wchar_t *str, ...) CHECK(str,); error_code = ec; - + // note : p may be -1 err_pos = static_cast(p); @@ -802,7 +802,7 @@ void parser_t::stack_trace(size_t block_idx, wcstring &buff) const */ if (block_idx >= this->block_count()) return; - + const block_t *b = this->block_at_index(block_idx); if (b->type()==EVENT) @@ -968,7 +968,7 @@ const wchar_t *parser_t::current_filename() const return function_get_definition_file(fb->name); } } - + /* We query a global array for the current file name, but only do that if we are the principal parser */ if (this == &principal_parser()) { @@ -1193,7 +1193,7 @@ void parser_t::job_promote(job_t *job) { signal_block(); - job_list_t::iterator loc = std::find(my_job_list.begin(), my_job_list.end(), job); + job_list_t::iterator loc = std::find(my_job_list.begin(), my_job_list.end(), job); assert(loc != my_job_list.end()); /* Move the job to the beginning */ @@ -2557,32 +2557,32 @@ void parser_t::eval_job(tokenizer_t *tok) int parser_t::eval_new_parser(const wcstring &cmd, const io_chain_t &io, enum block_type_t block_type) { CHECK_BLOCK(1); - + if (block_type != TOP && block_type != SUBST) { debug(1, INVALID_SCOPE_ERR_MSG, parser_t::get_block_desc(block_type)); bugreport(); return 1; } - + /* Parse the source into a tree, if we can */ parse_node_tree_t tree; if (! parse_tree_from_string(cmd, parse_flag_none, &tree, NULL)) { return 1; } - + /* Append to the execution context stack */ parse_execution_context_t *ctx = new parse_execution_context_t(tree, cmd, this); execution_contexts.push_back(ctx); - + /* Execute the first node */ int result = 1; if (! tree.empty()) { result = this->eval_block_node(0, io, block_type); } - + /* Clean up the execution context stack */ assert(! execution_contexts.empty() && execution_contexts.back() == ctx); execution_contexts.pop_back(); @@ -2596,9 +2596,9 @@ int parser_t::eval_block_node(node_offset_t node_idx, const io_chain_t &io, enum // Paranoia. It's a little frightening that we're given only a node_idx and we interpret this in the topmost execution context's tree. What happens if these were to be interleaved? Fortunately that cannot happen. parse_execution_context_t *ctx = execution_contexts.back(); assert(ctx != NULL); - + CHECK_BLOCK(1); - + /* Handle cancellation requests. If our block stack is currently empty, then we already did successfully cancel (or there was nothing to cancel); clear the flag. If our block stack is not empty, we are still in the process of cancelling; refuse to evaluate anything */ if (this->cancellation_requested) { @@ -2611,7 +2611,7 @@ int parser_t::eval_block_node(node_offset_t node_idx, const io_chain_t &io, enum this->cancellation_requested = false; } } - + /* Only certain blocks are allowed */ if ((block_type != TOP) && (block_type != SUBST)) @@ -2622,16 +2622,16 @@ int parser_t::eval_block_node(node_offset_t node_idx, const io_chain_t &io, enum bugreport(); return 1; } - + /* Not sure why we reap jobs here */ job_reap(0); - + /* Start it up */ const block_t * const start_current_block = current_block(); block_t *scope_block = new scope_block_t(block_type); this->push_block(scope_block); int result = ctx->eval_node_at_offset(node_idx, scope_block, io); - + /* Clean up the block stack */ this->pop_block(); while (start_current_block != current_block()) @@ -2646,10 +2646,10 @@ int parser_t::eval_block_node(node_offset_t node_idx, const io_chain_t &io, enum } this->pop_block(); } - + /* Reap again */ job_reap(0); - + return result; } @@ -2659,7 +2659,7 @@ int parser_t::eval(const wcstring &cmd_str, const io_chain_t &io, enum block_typ if (parser_use_ast()) return this->eval_new_parser(cmd_str, io, block_type); - + const wchar_t * const cmd = cmd_str.c_str(); size_t forbid_count; int code; @@ -2993,11 +2993,11 @@ void parser_t::get_backtrace(const wcstring &src, const parse_error_list_t &erro if (! errors.empty()) { const parse_error_t &err = errors.at(0); - + // Determine which line we're on assert(err.source_start <= src.size()); size_t which_line = 1 + std::count(src.begin(), src.begin() + err.source_start, L'\n'); - + const wchar_t *filename = this->current_filename(); if (filename) { @@ -3007,13 +3007,13 @@ void parser_t::get_backtrace(const wcstring &src, const parse_error_list_t &erro { output->append(L"fish: "); } - + // Don't include the caret if we're interactive, this is the first line of text, and our source is at its beginning, because then it's obvious bool skip_caret = (get_is_interactive() && which_line == 1 && err.source_start == 0); - + output->append(err.describe(src, skip_caret)); output->push_back(L'\n'); - + this->stack_trace(0, *output); } } diff --git a/parser.h b/parser.h index 0c8c7334a..32add3fb1 100644 --- a/parser.h +++ b/parser.h @@ -95,7 +95,7 @@ public: bool skip; /**< Whether execution of the commands in this block should be skipped */ bool had_command; /**< Set to non-zero once a command has been executed in this block */ int tok_pos; /**< The start index of the block */ - + node_offset_t node_offset; /* Offset of the node */ /** Status for the current loop block. Can be any of the values from the loop_status enum. */ @@ -286,10 +286,10 @@ private: /** Position of last error */ int err_pos; - + /** Indication that we should skip all blocks */ bool cancellation_requested; - + /** Stack of execution contexts. We own these pointers and must delete them */ std::vector execution_contexts; @@ -313,7 +313,7 @@ private: /** The jobs associated with this parser */ job_list_t my_job_list; - + /** The list of blocks, allocated with new. It's our responsibility to delete these */ std::vector block_stack; @@ -327,7 +327,7 @@ private: /* No copying allowed */ parser_t(const parser_t&); parser_t& operator=(const parser_t&); - + void parse_job_argument_list(process_t *p, job_t *j, tokenizer_t *tok, std::vector&, bool); int parse_job(process_t *p, job_t *j, tokenizer_t *tok); @@ -339,7 +339,7 @@ private: /** Create a job */ job_t *job_create(const io_chain_t &io); - + /** Adds a job to the beginning of the job list. */ void job_add(job_t *job); @@ -382,10 +382,10 @@ public: */ int eval(const wcstring &cmd_str, const io_chain_t &io, enum block_type_t block_type); int eval_new_parser(const wcstring &cmd, const io_chain_t &io, enum block_type_t block_type); - + /** Evaluates a block node at the given node offset in the topmost execution context */ int eval_block_node(node_offset_t node_idx, const io_chain_t &io, enum block_type_t block_type); - + /** Evaluate line as a list of parameters, i.e. tokenize it and perform parameter expansion and cmdsubst execution on the tokens. The output is inserted into output. @@ -430,15 +430,15 @@ public: /** Set the current position in the latest string of the tokenizer. */ void set_pos(int p); - + /** Returns the block at the given index. 0 corresponds to the innermost block. Returns NULL when idx is at or equal to the number of blocks. */ const block_t *block_at_index(size_t idx) const; block_t *block_at_index(size_t idx); - + /** Returns the current (innermost) block */ const block_t *current_block() const; block_t *current_block(); - + /** Count of blocks */ size_t block_count() const { @@ -459,7 +459,7 @@ public: /** Remove the outermost block namespace */ void pop_block(); - + /** Remove the outermost block, asserting it's the given one */ void pop_block(const block_t *b); diff --git a/proc.cpp b/proc.cpp index 4a1bfd5b5..2ea0d1bd4 100644 --- a/proc.cpp +++ b/proc.cpp @@ -640,7 +640,7 @@ int job_reap(bool interactive) static int locked = 0; locked++; - + /* Preserve the exit status */ const int saved_status = proc_get_last_status(); diff --git a/proc.h b/proc.h index 510f549a2..05013e1db 100644 --- a/proc.h +++ b/proc.h @@ -73,10 +73,10 @@ enum process_type_t A block of commands */ INTERNAL_BLOCK, - + /** A block of commands, represented as a node */ INTERNAL_BLOCK_NODE, - + /** The exec builtin */ @@ -156,7 +156,7 @@ public: INTERNAL_EXEC, or INTERNAL_BUFFER */ enum process_type_t type; - + /* For internal block processes only, the node offset of the block */ node_offset_t internal_block_node; diff --git a/reader.cpp b/reader.cpp index 09617bcd9..9b82550b8 100644 --- a/reader.cpp +++ b/reader.cpp @@ -663,31 +663,31 @@ bool reader_expand_abbreviation_in_command(const wcstring &cmdline, size_t curso const wcstring subcmd = wcstring(cmdsub_begin, cmdsub_end - cmdsub_begin); const size_t subcmd_cursor_pos = cursor_pos - subcmd_offset; - + /* Parse this subcmd */ parse_node_tree_t parse_tree; parse_tree_from_string(subcmd, parse_flag_continue_after_error | parse_flag_accept_incomplete_tokens, &parse_tree, NULL); - + /* Look for plain statements where the cursor is at the end of the command */ const parse_node_t *matching_cmd_node = NULL; const size_t len = parse_tree.size(); for (size_t i=0; i < len; i++) { const parse_node_t &node = parse_tree.at(i); - + /* Only interested in plain statements with source */ if (node.type != symbol_plain_statement || ! node.has_source()) continue; - + /* Skip decorated statements */ if (parse_tree.decoration_for_plain_statement(node) != parse_statement_decoration_none) continue; - + /* Get the command node. Skip it if we can't or it has no source */ const parse_node_t *cmd_node = parse_tree.get_child(node, 0, parse_token_type_string); if (cmd_node == NULL || ! cmd_node->has_source()) continue; - + /* Now see if its source range contains our cursor, including at the end */ if (subcmd_cursor_pos >= cmd_node->source_start && subcmd_cursor_pos <= cmd_node->source_start + cmd_node->source_length) { @@ -696,7 +696,7 @@ bool reader_expand_abbreviation_in_command(const wcstring &cmdline, size_t curso break; } } - + /* Now if we found a command node, expand it */ bool result = false; if (matching_cmd_node != NULL) @@ -2484,10 +2484,10 @@ int reader_shell_test(const wchar_t *b) { assert(b != NULL); wcstring bstr = b; - + /* Append a newline, to act as a statement terminator */ bstr.push_back(L'\n'); - + parse_error_list_t errors; int res = parse_util_detect_errors(bstr, &errors); @@ -2495,7 +2495,7 @@ int reader_shell_test(const wchar_t *b) { wcstring error_desc; parser_t::principal_parser().get_backtrace(bstr, errors, &error_desc); - + // ensure we end with a newline. Also add an initial newline, because it's likely the user just hit enter and so there's junk on the current line if (! string_suffixes_string(L"\n", error_desc)) { @@ -3179,7 +3179,7 @@ const wchar_t *reader_readline(void) /* Figure out the extent of the token within the command substitution. Note we pass cmdsub_begin here, not buff */ const wchar_t *token_begin, *token_end; parse_util_token_extent(cmdsub_begin, data->buff_pos - (cmdsub_begin-buff), &token_begin, &token_end, 0, 0); - + /* Hack: the token may extend past the end of the command substitution, e.g. in (echo foo) the last token is 'foo)'. Don't let that happen. */ if (token_end > cmdsub_end) token_end = cmdsub_end; diff --git a/screen.cpp b/screen.cpp index fc2bcb6ff..f6c49b762 100644 --- a/screen.cpp +++ b/screen.cpp @@ -1215,7 +1215,8 @@ static screen_layout_t compute_layout(screen_t *s, // If the command wraps, and the prompt is not short, place the command on its own line. // A short prompt is 33% or less of the terminal's width. const size_t prompt_percent_width = (100 * left_prompt_width) / screen_width; - if (left_prompt_width + first_command_line_width + 1 > screen_width && prompt_percent_width > 33) { + if (left_prompt_width + first_command_line_width + 1 > screen_width && prompt_percent_width > 33) + { result.prompts_get_own_line = true; } diff --git a/tokenizer.cpp b/tokenizer.cpp index 4e2b402c6..67b35fff2 100644 --- a/tokenizer.cpp +++ b/tokenizer.cpp @@ -441,7 +441,7 @@ static size_t read_redirection_or_fd_pipe(const wchar_t *buff, enum token_type * enum token_type redirection_mode = TOK_NONE; size_t idx = 0; - + /* Determine the fd. This may be specified as a prefix like '2>...' or it may be implicit like '>' or '^'. Try parsing out a number; if we did not get any digits then infer it from the first character. Watch out for overflow. */ long long big_fd = 0; for (; iswdigit(buff[idx]); idx++) @@ -450,21 +450,29 @@ static size_t read_redirection_or_fd_pipe(const wchar_t *buff, enum token_type * if (big_fd <= INT_MAX) big_fd = big_fd * 10 + (buff[idx] - L'0'); } - + fd = (big_fd > INT_MAX ? -1 : static_cast(big_fd)); - + if (idx == 0) { /* We did not find a leading digit, so there's no explicit fd. Infer it from the type */ switch (buff[idx]) { - case L'>': fd = STDOUT_FILENO; break; - case L'<': fd = STDIN_FILENO; break; - case L'^': fd = STDERR_FILENO; break; - default: errored = true; break; + case L'>': + fd = STDOUT_FILENO; + break; + case L'<': + fd = STDIN_FILENO; + break; + case L'^': + fd = STDERR_FILENO; + break; + default: + errored = true; + break; } } - + /* Either way we should have ended on the redirection character itself like '>' */ wchar_t redirect_char = buff[idx++]; //note increment of idx if (redirect_char == L'>' || redirect_char == L'^') @@ -486,7 +494,7 @@ static size_t read_redirection_or_fd_pipe(const wchar_t *buff, enum token_type * /* Something else */ errored = true; } - + /* Optional characters like & or ?, or the pipe char | */ wchar_t opt_char = buff[idx]; if (opt_char == L'&') @@ -505,20 +513,20 @@ static size_t read_redirection_or_fd_pipe(const wchar_t *buff, enum token_type * redirection_mode = TOK_PIPE; idx++; } - + /* Don't return valid-looking stuff on error */ if (errored) { idx = 0; redirection_mode = TOK_NONE; } - + /* Return stuff */ if (out_redirection_mode != NULL) *out_redirection_mode = redirection_mode; if (out_fd != NULL) *out_fd = fd; - + return idx; } @@ -542,7 +550,7 @@ int fd_redirected_by_pipe(const wcstring &str) { return STDOUT_FILENO; } - + enum token_type mode = TOK_NONE; int fd = 0; read_redirection_or_fd_pipe(str.c_str(), &mode, &fd); @@ -556,11 +564,15 @@ int oflags_for_redirection_type(enum token_type type) { switch (type) { - case TOK_REDIRECT_APPEND: return O_CREAT | O_APPEND | O_WRONLY; - case TOK_REDIRECT_OUT: return O_CREAT | O_WRONLY | O_TRUNC; - case TOK_REDIRECT_NOCLOB: return O_CREAT | O_EXCL | O_WRONLY; - case TOK_REDIRECT_IN: return O_RDONLY; - + case TOK_REDIRECT_APPEND: + return O_CREAT | O_APPEND | O_WRONLY; + case TOK_REDIRECT_OUT: + return O_CREAT | O_WRONLY | O_TRUNC; + case TOK_REDIRECT_NOCLOB: + return O_CREAT | O_EXCL | O_WRONLY; + case TOK_REDIRECT_IN: + return O_RDONLY; + default: return -1; } @@ -703,7 +715,7 @@ void tok_next(tokenizer_t *tok) int fd = -1; if (iswdigit(*tok->buff)) consumed = read_redirection_or_fd_pipe(tok->buff, &mode, &fd); - + if (consumed > 0) { /* It looks like a redirection or a pipe. But we don't support piping fd 0. Note that fd 0 may be -1, indicating overflow; but we don't treat that as a tokenizer error. */