Ticket #4198: MC hangs on start randomly with zsh as subshell.

The bug happens because MC tells the shell to perform a "cd" command,
and waits indefinitely for the command to complete. If the shell, for
some reason, cannot complete the "cd" command, MC just freezes
indefinitely.

This patch removes all instances of MC waiting indefinitely for "cd"
commands. Now, if the shell freezes while doing a "cd", MC will timeout
after 1 second, and just set the subshell state to ACTIVE. If the user
tries to run a command, they will get the error "The shell is already
running a command".

Some other stuff where MC waits on the shell if an error occurred is
also simplified.

In feed_subshell(), the timeout time is changed from 10 seconds to
1 second. Ten seconds seemed like far too long to wait.

Signed-off-by: Andrew Borodin <aborodin@vmail.ru>
This commit is contained in:
Eric Roberts 2021-07-04 19:06:28 +03:00 committed by Andrew Borodin
parent 5c1d3c55dd
commit e89713a3d6

View File

@ -201,11 +201,6 @@ static int subshell_ready;
/* Flag to indicate if the subshell supports the persistent buffer feature. */ /* Flag to indicate if the subshell supports the persistent buffer feature. */
static gboolean use_persistent_buffer = FALSE; static gboolean use_persistent_buffer = FALSE;
/* Flag to indicate if the contents of the subshell command line need to be cleared before */
/* executing a command. This should only end up set to true if there is some sort of error. */
/* This allows us to recover gracefully from an error. */
static gboolean subshell_should_clear_command_line = FALSE;
/* This is the local variable where the subshell prompt is stored while we are working on it. */ /* This is the local variable where the subshell prompt is stored while we are working on it. */
static GString *subshell_prompt_temp_buffer = NULL; static GString *subshell_prompt_temp_buffer = NULL;
@ -752,10 +747,9 @@ feed_subshell (int how, gboolean fail_on_error)
struct timeval *wptr; struct timeval *wptr;
should_read_new_subshell_prompt = FALSE; should_read_new_subshell_prompt = FALSE;
subshell_should_clear_command_line = FALSE;
/* we wait up to 10 seconds if fail_on_error, forever otherwise */ /* we wait up to 1 second if fail_on_error, forever otherwise */
wtime.tv_sec = 10; wtime.tv_sec = 1;
wtime.tv_usec = 0; wtime.tv_usec = 0;
wptr = fail_on_error ? &wtime : NULL; wptr = fail_on_error ? &wtime : NULL;
@ -878,9 +872,17 @@ feed_subshell (int how, gboolean fail_on_error)
if (subshell_ready && !read_command_line_buffer (FALSE)) if (subshell_ready && !read_command_line_buffer (FALSE))
{ {
/* If we got here, some unforseen error must have occurred. */ /* If we got here, some unforseen error must have occurred. */
if (mc_global.shell->type != SHELL_FISH)
{
write_all (mc_global.tty.subshell_pty, "\003", 1);
subshell_state = RUNNING_COMMAND;
if (feed_subshell (QUIETLY, TRUE))
if (read_command_line_buffer (FALSE))
return TRUE;
}
subshell_state = ACTIVE;
flush_subshell (0, VISIBLY); flush_subshell (0, VISIBLY);
input_assign_text (cmdline, ""); input_assign_text (cmdline, "");
subshell_should_clear_command_line = TRUE;
} }
} }
@ -1455,14 +1457,16 @@ invoke_subshell (const char *command, int how, vfs_path_t ** new_dir_vpath)
/* data is there, but only if we are using one of the shells that */ /* data is there, but only if we are using one of the shells that */
/* doesn't support keeping command buffer contents, OR if there was */ /* doesn't support keeping command buffer contents, OR if there was */
/* some sort of error. */ /* some sort of error. */
if (!use_persistent_buffer || subshell_should_clear_command_line) if (!use_persistent_buffer)
{ {
write_all (mc_global.tty.subshell_pty, "\003", 1); /* We don't need to call feed_subshell here if we are using fish, because of a
subshell_state = RUNNING_COMMAND; * quirk in the behavior of that particular shell. */
/* We need to call feed_subshell here if we are using fish, because of a quirk
* in the behavioral of that particular shell. */
if (mc_global.shell->type != SHELL_FISH) if (mc_global.shell->type != SHELL_FISH)
{
write_all (mc_global.tty.subshell_pty, "\003", 1);
subshell_state = RUNNING_COMMAND;
feed_subshell (QUIETLY, FALSE); feed_subshell (QUIETLY, FALSE);
}
} }
if (how == QUIETLY) if (how == QUIETLY)
@ -1667,12 +1671,16 @@ do_subshell_chdir (const vfs_path_t * vpath, gboolean update_prompt)
/* If we are using a shell that doesn't support persistent command buffer, we need to clear /* If we are using a shell that doesn't support persistent command buffer, we need to clear
* the command prompt before we send the cd command. */ * the command prompt before we send the cd command. */
if (!use_persistent_buffer || subshell_should_clear_command_line) if (!use_persistent_buffer)
{ {
write_all (mc_global.tty.subshell_pty, "\003", 1); write_all (mc_global.tty.subshell_pty, "\003", 1);
subshell_state = RUNNING_COMMAND; subshell_state = RUNNING_COMMAND;
if (mc_global.shell->type != SHELL_FISH) if (mc_global.shell->type != SHELL_FISH)
feed_subshell (QUIETLY, FALSE); if (!feed_subshell (QUIETLY, TRUE))
{
subshell_state = ACTIVE;
return;
}
} }
/* The initial space keeps this out of the command history (in bash /* The initial space keeps this out of the command history (in bash
because we set "HISTCONTROL=ignorespace") */ because we set "HISTCONTROL=ignorespace") */
@ -1703,7 +1711,11 @@ do_subshell_chdir (const vfs_path_t * vpath, gboolean update_prompt)
write_all (mc_global.tty.subshell_pty, "\n", 1); write_all (mc_global.tty.subshell_pty, "\n", 1);
subshell_state = RUNNING_COMMAND; subshell_state = RUNNING_COMMAND;
feed_subshell (QUIETLY, FALSE); if (!feed_subshell (QUIETLY, TRUE))
{
subshell_state = ACTIVE;
return;
}
if (subshell_alive) if (subshell_alive)
{ {
@ -1745,7 +1757,7 @@ do_subshell_chdir (const vfs_path_t * vpath, gboolean update_prompt)
* type a space and press return. */ * type a space and press return. */
write_all (mc_global.tty.subshell_pty, " \n", 2); write_all (mc_global.tty.subshell_pty, " \n", 2);
subshell_state = RUNNING_COMMAND; subshell_state = RUNNING_COMMAND;
feed_subshell (QUIETLY, FALSE); feed_subshell (QUIETLY, TRUE);
} }
update_subshell_prompt = FALSE; update_subshell_prompt = FALSE;