git: use run_command() to execute dashed externals
We used to simply try calling execvp(); if it succeeded, then we were done and the new program was running. If it didn't, then we knew that it wasn't a valid command. Unfortunately, this interacted badly with the new pager handling. Now that git remains the parent process and the pager is spawned, git has to hang around until the pager is finished. We install an atexit handler to do this, but that handler never gets called if we successfully run execvp. You could see this behavior by running any dashed external using a pager (e.g., "git -p stash list"). The command finishes running, but the pager is still going. In the case of less, it then gets an error reading from the terminal and exits, potentially leaving the terminal in a broken state (and not showing the output). This patch just uses run_command() to try running the dashed external. The parent git process then waits for the external process to complete and then handles the pager cleanup as it would for an internal command. Signed-off-by: Jeff King <peff@peff.net> Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
parent
1d64f21d99
commit
d8e96fd86d
17
git.c
17
git.c
@ -2,6 +2,7 @@
|
|||||||
#include "exec_cmd.h"
|
#include "exec_cmd.h"
|
||||||
#include "cache.h"
|
#include "cache.h"
|
||||||
#include "quote.h"
|
#include "quote.h"
|
||||||
|
#include "run-command.h"
|
||||||
|
|
||||||
const char git_usage_string[] =
|
const char git_usage_string[] =
|
||||||
"git [--version] [--exec-path[=GIT_EXEC_PATH]] [-p|--paginate|--no-pager] [--bare] [--git-dir=GIT_DIR] [--work-tree=GIT_WORK_TREE] [--help] COMMAND [ARGS]";
|
"git [--version] [--exec-path[=GIT_EXEC_PATH]] [-p|--paginate|--no-pager] [--bare] [--git-dir=GIT_DIR] [--work-tree=GIT_WORK_TREE] [--help] COMMAND [ARGS]";
|
||||||
@ -392,6 +393,7 @@ static void execv_dashed_external(const char **argv)
|
|||||||
{
|
{
|
||||||
struct strbuf cmd = STRBUF_INIT;
|
struct strbuf cmd = STRBUF_INIT;
|
||||||
const char *tmp;
|
const char *tmp;
|
||||||
|
int status;
|
||||||
|
|
||||||
strbuf_addf(&cmd, "git-%s", argv[0]);
|
strbuf_addf(&cmd, "git-%s", argv[0]);
|
||||||
|
|
||||||
@ -406,10 +408,17 @@ static void execv_dashed_external(const char **argv)
|
|||||||
|
|
||||||
trace_argv_printf(argv, "trace: exec:");
|
trace_argv_printf(argv, "trace: exec:");
|
||||||
|
|
||||||
/* execvp() can only ever return if it fails */
|
/*
|
||||||
execvp(cmd.buf, (char **)argv);
|
* if we fail because the command is not found, it is
|
||||||
|
* OK to return. Otherwise, we just pass along the status code.
|
||||||
trace_printf("trace: exec failed: %s\n", strerror(errno));
|
*/
|
||||||
|
status = run_command_v_opt(argv, 0);
|
||||||
|
if (status != -ERR_RUN_COMMAND_EXEC) {
|
||||||
|
if (IS_RUN_COMMAND_ERR(status))
|
||||||
|
die("unable to run '%s'", argv[0]);
|
||||||
|
exit(-status);
|
||||||
|
}
|
||||||
|
errno = ENOENT; /* as if we called execvp */
|
||||||
|
|
||||||
argv[0] = tmp;
|
argv[0] = tmp;
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user