X-Git-Url: http://git.ozlabs.org/?p=petitboot;a=blobdiff_plain;f=lib%2Fprocess%2Fprocess.c;h=02b6f490dad8318744ba8b4f43b30c6362ceae09;hp=c1606a52b342406de99d44d9cd07a678a327ba29;hb=51fb307dfdbb8f238d15520a74e33ef4fea52e4d;hpb=2914110af0cc490566446f18c929452368121b98 diff --git a/lib/process/process.c b/lib/process/process.c index c1606a5..02b6f49 100644 --- a/lib/process/process.c +++ b/lib/process/process.c @@ -92,7 +92,7 @@ static int process_read_stdout_once(struct process_info *procinfo, char **line) if (rc < 0) { if (errno == EINTR) return 1; - pb_log("%s: read failed: %s\n", __func__, strerror(errno)); + pb_log_fn("read failed: %s\n", strerror(errno)); return rc; } @@ -178,28 +178,7 @@ static int process_read_stdout(struct process_info *procinfo) return rc < 0 ? rc : 0; } -static int process_stdout_cb(void *arg) -{ - struct process_info *procinfo = arg; - int rc; - - rc = process_read_stdout_once(procinfo, NULL); - - /* if we're going to signal to the waitset that we're done (ie, non-zero - * return value), then the waiters will remove us, so we drop the - * reference */ - if (rc < 0) { - talloc_unlink(procset, procinfo); - procinfo->stdout_waiter = NULL; - rc = -1; - } else { - rc = 0; - } - - return rc; -} - -int process_stdout_custom(struct process_info *procinfo, char **line) +int process_process_stdout(struct process_info *procinfo, char **line) { int rc; @@ -232,7 +211,7 @@ static void sigchld_sigaction(int signo, siginfo_t *info, rc = write(procset->sigchld_pipe[1], &pid, sizeof(pid)); if (rc != sizeof(pid)) - pb_log("%s: write failed: %s\n", __func__, strerror(errno)); + pb_log_fn("write failed: %s\n", strerror(errno)); } static int sigchld_pipe_event(void *arg) @@ -306,7 +285,7 @@ struct procset *process_init(void *ctx, struct waitset *set, bool dry_run) rc = pipe(procset->sigchld_pipe); if (rc) { - pb_log("%s: pipe() failed: %s\n", __func__, strerror(errno)); + pb_log_fn("pipe() failed: %s\n", strerror(errno)); goto err_free; } @@ -322,7 +301,7 @@ struct procset *process_init(void *ctx, struct waitset *set, bool dry_run) rc = sigaction(SIGCHLD, &sa, NULL); if (rc) { - pb_log("%s: sigaction() failed: %s\n", __func__, + pb_log_fn("sigaction() failed: %s\n", strerror(errno)); goto err_remove; } @@ -374,7 +353,7 @@ static int process_run_common(struct process_info *procinfo) pid = fork(); if (pid < 0) { - pb_log("%s: fork failed: %s\n", __func__, strerror(errno)); + pb_log_fn("fork failed: %s\n", strerror(errno)); return pid; } @@ -410,17 +389,21 @@ int process_run_sync(struct process *process) if (errno == EINTR) continue; - pb_log("%s: waitpid failed: %s\n", __func__, strerror(errno)); + pb_log_fn("waitpid failed: %s\n", strerror(errno)); return rc; } return 0; } +static int process_stdout_cb(struct process_info *procinfo) +{ + return process_process_stdout(procinfo, NULL); +} + int process_run_async(struct process *process) { struct process_info *procinfo = get_info(process); - waiter_cb stdout_cb; int rc; rc = process_run_common(procinfo); @@ -428,7 +411,8 @@ int process_run_async(struct process *process) return rc; if (process->keep_stdout) { - stdout_cb = process->stdout_cb ?: process_stdout_cb; + waiter_cb stdout_cb = process->stdout_cb ?: + (waiter_cb)process_stdout_cb; procinfo->stdout_waiter = waiter_register_io(procset->waitset, procinfo->stdout_pipe[0], WAIT_IN, stdout_cb, procinfo); @@ -468,48 +452,76 @@ void process_stop_async_all(void) } } -int process_run_simple_argv(void *ctx, const char *argv[]) +int process_get_stdout_argv(void *ctx, struct process_stdout **stdout, + const char *argv[]) { - struct process *process; + struct process *p; int rc; - process = process_create(ctx); + p = process_create(NULL); + p->path = argv[0]; + p->argv = argv; - process->path = argv[0]; - process->argv = argv; + if (stdout) { + p->keep_stdout = true; + *stdout = NULL; + } - rc = process_run_sync(process); + rc = process_run_sync(p); if (!rc) - rc = process->exit_status; + rc = p->exit_status; + else { + pb_debug("%s: process_run_sync failed: %s.\n", __func__, + p->path); + if (stdout) + pb_debug("%s: stdout: %s\n\n", __func__, p->stdout_buf); + goto exit; + } + + if (!stdout) + goto exit; + + *stdout = talloc(ctx, struct process_stdout); + + if (!*stdout) { + rc = -1; + goto exit; + } - process_release(process); + (*stdout)->len = p->stdout_len; + (*stdout)->buf = talloc_memdup(*stdout, p->stdout_buf, + p->stdout_len + 1); + (*stdout)->buf[p->stdout_len] = 0; +exit: + process_release(p); return rc; } -int process_run_simple(void *ctx, const char *name, ...) +int process_get_stdout(void *ctx, struct process_stdout **stdout, + const char *path, ...) { int rc, i, n_argv = 1; const char **argv; va_list ap; - va_start(ap, name); + va_start(ap, path); while (va_arg(ap, char *)) n_argv++; va_end(ap); argv = talloc_array(ctx, const char *, n_argv + 1); - argv[0] = name; + argv[0] = path; - va_start(ap, name); + va_start(ap, path); for (i = 1; i < n_argv; i++) argv[i] = va_arg(ap, const char *); va_end(ap); argv[i] = NULL; - rc = process_run_simple_argv(ctx, argv); + rc = process_get_stdout_argv(ctx, stdout, argv); talloc_free(argv);