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:
commit
5816d3cdfb
38
git.c
38
git.c
@ -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)
|
||||||
|
@ -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);
|
||||||
}
|
}
|
||||||
|
@ -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;
|
||||||
};
|
};
|
||||||
|
Loading…
Reference in New Issue
Block a user