Merge branch 'jk/execv-dashed-external' into maint

Typing ^C to pager, which usually does not kill it, killed Git and
took the pager down as a collateral damage in certain process-tree
structure.  This has been fixed.

* jk/execv-dashed-external:
  execv_dashed_external: wait for child on signal death
  execv_dashed_external: stop exiting with negative code
  execv_dashed_external: use child_process struct
This commit is contained in:
Junio C Hamano 2017-02-02 13:20:29 -08:00
commit 5816d3cdfb
3 changed files with 36 additions and 22 deletions

38
git.c
View File

@ -575,8 +575,7 @@ static void handle_builtin(int argc, const char **argv)
static void execv_dashed_external(const char **argv) static void execv_dashed_external(const char **argv)
{ {
struct strbuf cmd = STRBUF_INIT; struct child_process cmd = CHILD_PROCESS_INIT;
const char *tmp;
int status; int status;
if (get_super_prefix()) if (get_super_prefix())
@ -586,30 +585,25 @@ static void execv_dashed_external(const char **argv)
use_pager = check_pager_config(argv[0]); use_pager = check_pager_config(argv[0]);
commit_pager_choice(); commit_pager_choice();
strbuf_addf(&cmd, "git-%s", argv[0]); argv_array_pushf(&cmd.args, "git-%s", argv[0]);
argv_array_pushv(&cmd.args, argv + 1);
cmd.clean_on_exit = 1;
cmd.wait_after_clean = 1;
cmd.silent_exec_failure = 1;
trace_argv_printf(cmd.args.argv, "trace: exec:");
/* /*
* argv[0] must be the git command, but the argv array * If we fail because the command is not found, it is
* belongs to the caller, and may be reused in * OK to return. Otherwise, we just pass along the status code,
* subsequent loop iterations. Save argv[0] and * or our usual generic code if we were not even able to exec
* restore it on error. * the program.
*/ */
tmp = argv[0]; status = run_command(&cmd);
argv[0] = cmd.buf; if (status >= 0)
trace_argv_printf(argv, "trace: exec:");
/*
* if we fail because the command is not found, it is
* OK to return. Otherwise, we just pass along the status code.
*/
status = run_command_v_opt(argv, RUN_SILENT_EXEC_FAILURE | RUN_CLEAN_ON_EXIT);
if (status >= 0 || errno != ENOENT)
exit(status); exit(status);
else if (errno != ENOENT)
argv[0] = tmp; exit(128);
strbuf_release(&cmd);
} }
static int run_argv(int *argcp, const char ***argv) static int run_argv(int *argcp, const char ***argv)

View File

@ -29,6 +29,8 @@ static int installed_child_cleanup_handler;
static void cleanup_children(int sig, int in_signal) static void cleanup_children(int sig, int in_signal)
{ {
struct child_to_clean *children_to_wait_for = NULL;
while (children_to_clean) { while (children_to_clean) {
struct child_to_clean *p = children_to_clean; struct child_to_clean *p = children_to_clean;
children_to_clean = p->next; children_to_clean = p->next;
@ -45,6 +47,23 @@ static void cleanup_children(int sig, int in_signal)
} }
kill(p->pid, sig); kill(p->pid, sig);
if (p->process->wait_after_clean) {
p->next = children_to_wait_for;
children_to_wait_for = p;
} else {
if (!in_signal)
free(p);
}
}
while (children_to_wait_for) {
struct child_to_clean *p = children_to_wait_for;
children_to_wait_for = p->next;
while (waitpid(p->pid, NULL, 0) < 0 && errno == EINTR)
; /* spin waiting for process exit or error */
if (!in_signal) if (!in_signal)
free(p); free(p);
} }

View File

@ -43,6 +43,7 @@ struct child_process {
unsigned stdout_to_stderr:1; unsigned stdout_to_stderr:1;
unsigned use_shell:1; unsigned use_shell:1;
unsigned clean_on_exit:1; unsigned clean_on_exit:1;
unsigned wait_after_clean:1;
void (*clean_on_exit_handler)(struct child_process *process); void (*clean_on_exit_handler)(struct child_process *process);
void *clean_on_exit_handler_cbdata; void *clean_on_exit_handler_cbdata;
}; };