mirror of
https://github.com/MidnightCommander/mc
synced 2025-01-18 17:29:28 +03:00
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:
parent
5c1d3c55dd
commit
e89713a3d6
@ -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;
|
||||||
|
Loading…
Reference in New Issue
Block a user