Matthew Booth
2012-Dec-13 15:22 UTC
[Libguestfs] [PATCH 1/2] daemon: NFC Use symbolic names in commandrvf
Improve readability of commandrvf() by replacing bare int values for file descriptors with their symbolic names STD{IN,OUT,ERR}_FILENO. Also add PIPE_READ and PIPE_WRITE for referencing relevant ends of a pipe. --- daemon/guestfsd.c | 79 +++++++++++++++++++++++++++++-------------------------- 1 file changed, 42 insertions(+), 37 deletions(-) diff --git a/daemon/guestfsd.c b/daemon/guestfsd.c index 5c84849..4b3dd5f 100644 --- a/daemon/guestfsd.c +++ b/daemon/guestfsd.c @@ -66,6 +66,10 @@ static char *read_cmdline (void); # define O_CLOEXEC 0 #endif +/* For improved readability dealing with pipe arrays */ +#define PIPE_READ 0 +#define PIPE_WRITE 1 + /* If root device is an ext2 filesystem, this is the major and minor. * This is so we can ignore this device from the point of view of the * user, eg. in guestfs_list_devices and many other places. @@ -834,22 +838,22 @@ commandrvf (char **stdoutput, char **stderror, int flags, signal (SIGPIPE, SIG_DFL); close (0); if (flag_copy_stdin) { - dup2 (stdin_fd[0], 0); - close (stdin_fd[0]); - close (stdin_fd[1]); + dup2 (stdin_fd[PIPE_READ], STDIN_FILENO); + close (stdin_fd[PIPE_READ]); + close (stdin_fd[PIPE_WRITE]); } else { /* Set stdin to /dev/null (ignore failure) */ ignore_value (open ("/dev/null", O_RDONLY|O_CLOEXEC)); } - close (so_fd[0]); - close (se_fd[0]); + close (so_fd[PIPE_READ]); + close (se_fd[PIPE_READ]); if (!(flags & COMMAND_FLAG_FOLD_STDOUT_ON_STDERR)) - dup2 (so_fd[1], 1); + dup2 (so_fd[PIPE_WRITE], STDOUT_FILENO); else - dup2 (se_fd[1], 1); - dup2 (se_fd[1], 2); - close (so_fd[1]); - close (se_fd[1]); + dup2 (se_fd[PIPE_WRITE], STDOUT_FILENO); + dup2 (se_fd[PIPE_WRITE], STDERR_FILENO); + close (so_fd[PIPE_WRITE]); + close (se_fd[PIPE_WRITE]); execvp (argv[0], (void *) argv); perror (argv[0]); @@ -866,15 +870,15 @@ commandrvf (char **stdoutput, char **stderror, int flags, } if (stdin_pid == 0) { /* Child process copying stdin. */ - close (so_fd[0]); - close (so_fd[1]); - close (se_fd[0]); - close (se_fd[1]); + close (so_fd[PIPE_READ]); + close (so_fd[PIPE_WRITE]); + close (se_fd[PIPE_READ]); + close (se_fd[PIPE_WRITE]); - close (1); - dup2 (stdin_fd[1], 1); - close (stdin_fd[0]); - close (stdin_fd[1]); + close (STDOUT_FILENO); + dup2 (stdin_fd[PIPE_WRITE], STDOUT_FILENO); + close (stdin_fd[PIPE_READ]); + close (stdin_fd[PIPE_WRITE]); if (chroot (sysroot) == -1) { perror ("chroot"); @@ -884,7 +888,7 @@ commandrvf (char **stdoutput, char **stderror, int flags, ssize_t n; char buffer[BUFSIZ]; while ((n = read (fd, buffer, sizeof buffer)) > 0) { - if (xwrite (1, buffer, n) == -1) + if (xwrite (STDOUT_FILENO, buffer, n) == -1) /* EPIPE error indicates the command process has exited * early. If the command process fails that will be caught * by the daemon, and if not, then it's not an error. @@ -906,23 +910,24 @@ commandrvf (char **stdoutput, char **stderror, int flags, } close (fd); - close (stdin_fd[0]); - close (stdin_fd[1]); + close (stdin_fd[PIPE_READ]); + close (stdin_fd[PIPE_WRITE]); } /* Parent process. */ - close (so_fd[1]); - close (se_fd[1]); + close (so_fd[PIPE_WRITE]); + close (se_fd[PIPE_WRITE]); FD_ZERO (&rset); - FD_SET (so_fd[0], &rset); - FD_SET (se_fd[0], &rset); + FD_SET (so_fd[PIPE_READ], &rset); + FD_SET (se_fd[PIPE_READ], &rset); quit = 0; while (quit < 2) { again: rset2 = rset; - r = select (MAX (so_fd[0], se_fd[0]) + 1, &rset2, NULL, NULL, NULL); + r = select (MAX (so_fd[PIPE_READ], se_fd[PIPE_READ]) + 1, &rset2, + NULL, NULL, NULL); if (r == -1) { if (errno == EINTR) goto again; @@ -943,20 +948,20 @@ commandrvf (char **stdoutput, char **stderror, int flags, *stderror = strdup ("error running external command, " "see debug output for details"); } - close (so_fd[0]); - close (se_fd[0]); + close (so_fd[PIPE_READ]); + close (se_fd[PIPE_READ]); waitpid (pid, NULL, 0); if (stdin_pid >= 0) waitpid (stdin_pid, NULL, 0); return -1; } - if (FD_ISSET (so_fd[0], &rset2)) { /* something on stdout */ - r = read (so_fd[0], buf, sizeof buf); + if (FD_ISSET (so_fd[PIPE_READ], &rset2)) { /* something on stdout */ + r = read (so_fd[PIPE_READ], buf, sizeof buf); if (r == -1) { perror ("read"); goto quit; } - if (r == 0) { FD_CLR (so_fd[0], &rset); quit++; } + if (r == 0) { FD_CLR (so_fd[PIPE_READ], &rset); quit++; } if (r > 0 && stdoutput) { so_size += r; @@ -970,17 +975,17 @@ commandrvf (char **stdoutput, char **stderror, int flags, } } - if (FD_ISSET (se_fd[0], &rset2)) { /* something on stderr */ - r = read (se_fd[0], buf, sizeof buf); + if (FD_ISSET (se_fd[PIPE_READ], &rset2)) { /* something on stderr */ + r = read (se_fd[PIPE_READ], buf, sizeof buf); if (r == -1) { perror ("read"); goto quit; } - if (r == 0) { FD_CLR (se_fd[0], &rset); quit++; } + if (r == 0) { FD_CLR (se_fd[PIPE_READ], &rset); quit++; } if (r > 0) { if (verbose) - ignore_value (write (2, buf, r)); + ignore_value (write (STDERR_FILENO, buf, r)); if (stderror) { se_size += r; @@ -996,8 +1001,8 @@ commandrvf (char **stdoutput, char **stderror, int flags, } } - close (so_fd[0]); - close (se_fd[0]); + close (so_fd[PIPE_READ]); + close (se_fd[PIPE_READ]); /* Make sure the output buffers are \0-terminated. Also remove any * trailing \n characters from the error buffer (not from stdout). -- 1.7.11.7
Matthew Booth
2012-Dec-13 15:22 UTC
[Libguestfs] [PATCH 2/2] daemon: Remove redundant fork in commandrvf
Currently the code is doing a redundant fork when passed the COMMAND_FLAG_CHROOT_COPY_FILE_TO_STDIN flag. The additional sub-process does a chroot() which has no effect because all file handles are already open at that point, then simply copies its input to its output. This change simply replaces the above with a dup2 of the passed file handle to STDIN of the command process. --- daemon/guestfsd.c | 89 +++++-------------------------------------------------- 1 file changed, 7 insertions(+), 82 deletions(-) diff --git a/daemon/guestfsd.c b/daemon/guestfsd.c index 4b3dd5f..7e83a2a 100644 --- a/daemon/guestfsd.c +++ b/daemon/guestfsd.c @@ -789,8 +789,8 @@ commandrvf (char **stdoutput, char **stderror, int flags, size_t so_size = 0, se_size = 0; int so_fd[2], se_fd[2]; int flag_copy_stdin = flags & COMMAND_FLAG_CHROOT_COPY_FILE_TO_STDIN; - int stdin_fd[2] = { -1, -1 }; - pid_t pid, stdin_pid = -1; + int flag_copy_fd = flags & COMMAND_FLAG_FD_MASK; + pid_t pid = -1; int r, quit, i; fd_set rset, rset2; char buf[256]; @@ -820,13 +820,6 @@ commandrvf (char **stdoutput, char **stderror, int flags, abort (); } - if (flag_copy_stdin) { - if (pipe (stdin_fd) == -1) { - error (0, errno, "pipe"); - abort (); - } - } - pid = fork (); if (pid == -1) { error (0, errno, "fork"); @@ -838,9 +831,7 @@ commandrvf (char **stdoutput, char **stderror, int flags, signal (SIGPIPE, SIG_DFL); close (0); if (flag_copy_stdin) { - dup2 (stdin_fd[PIPE_READ], STDIN_FILENO); - close (stdin_fd[PIPE_READ]); - close (stdin_fd[PIPE_WRITE]); + dup2 (flag_copy_fd, STDIN_FILENO); } else { /* Set stdin to /dev/null (ignore failure) */ ignore_value (open ("/dev/null", O_RDONLY|O_CLOEXEC)); @@ -860,60 +851,6 @@ commandrvf (char **stdoutput, char **stderror, int flags, _exit (EXIT_FAILURE); } - if (flag_copy_stdin) { - int fd = flags & COMMAND_FLAG_FD_MASK; - - stdin_pid = fork (); - if (stdin_pid == -1) { - error (0, errno, "fork"); - abort (); - } - - if (stdin_pid == 0) { /* Child process copying stdin. */ - close (so_fd[PIPE_READ]); - close (so_fd[PIPE_WRITE]); - close (se_fd[PIPE_READ]); - close (se_fd[PIPE_WRITE]); - - close (STDOUT_FILENO); - dup2 (stdin_fd[PIPE_WRITE], STDOUT_FILENO); - close (stdin_fd[PIPE_READ]); - close (stdin_fd[PIPE_WRITE]); - - if (chroot (sysroot) == -1) { - perror ("chroot"); - _exit (EXIT_FAILURE); - } - - ssize_t n; - char buffer[BUFSIZ]; - while ((n = read (fd, buffer, sizeof buffer)) > 0) { - if (xwrite (STDOUT_FILENO, buffer, n) == -1) - /* EPIPE error indicates the command process has exited - * early. If the command process fails that will be caught - * by the daemon, and if not, then it's not an error. - */ - _exit (errno == EPIPE ? EXIT_SUCCESS : EXIT_FAILURE); - } - - if (n == -1) { - perror ("read"); - _exit (EXIT_FAILURE); - } - - if (close (fd) == -1) { - perror ("close"); - _exit (EXIT_FAILURE); - } - - _exit (EXIT_SUCCESS); - } - - close (fd); - close (stdin_fd[PIPE_READ]); - close (stdin_fd[PIPE_WRITE]); - } - /* Parent process. */ close (so_fd[PIPE_WRITE]); close (se_fd[PIPE_WRITE]); @@ -950,8 +887,8 @@ commandrvf (char **stdoutput, char **stderror, int flags, } close (so_fd[PIPE_READ]); close (se_fd[PIPE_READ]); + if (flag_copy_stdin) close (flag_copy_fd); waitpid (pid, NULL, 0); - if (stdin_pid >= 0) waitpid (stdin_pid, NULL, 0); return -1; } @@ -1033,21 +970,9 @@ commandrvf (char **stdoutput, char **stderror, int flags, } } - if (flag_copy_stdin) { - /* Check copy process didn't fail. */ - if (waitpid (stdin_pid, &r, 0) != stdin_pid) { - perror ("waitpid"); - kill (pid, 9); - waitpid (pid, NULL, 0); - return -1; - } - - if (!WIFEXITED (r) || WEXITSTATUS (r) != 0) { - fprintf (stderr, "failed copying from input file, see earlier messages (r = %d)\n", r); - kill (pid, 9); - waitpid (pid, NULL, 0); - return -1; - } + if (flag_copy_stdin && close (flag_copy_fd) == -1) { + perror ("close"); + return -1; } /* Get the exit status of the command. */ -- 1.7.11.7
Richard W.M. Jones
2012-Dec-13 17:54 UTC
[Libguestfs] [PATCH 1/2] daemon: NFC Use symbolic names in commandrvf
On Thu, Dec 13, 2012 at 03:22:38PM +0000, Matthew Booth wrote:> Improve readability of commandrvf() by replacing bare int values for > file descriptors with their symbolic names STD{IN,OUT,ERR}_FILENO. > > Also add PIPE_READ and PIPE_WRITE for referencing relevant ends of a pipe.Thanks -- applied, will push it shortly. Rich. -- Richard Jones, Virtualization Group, Red Hat http://people.redhat.com/~rjones Read my programming blog: http://rwmj.wordpress.com Fedora now supports 80 OCaml packages (the OPEN alternative to F#) http://cocan.org/getting_started_with_ocaml_on_red_hat_and_fedora
Maybe Matching Threads
- [PATCH] daemon: improve internal commandrvf
- [PATCH] daemon: improve debugging for "stdout on stderr" flag
- [PATCH v3 2/6] daemon: Split out command() functions and CLEANUP_* macros into separate files.
- Re: [PATCH] daemon: improve internal commandrvf
- missing chdir before chroot in guestfsd