Blame src/unix/process.c

Packit b5b901
/* Copyright Joyent, Inc. and other Node contributors. All rights reserved.
Packit b5b901
 *
Packit b5b901
 * Permission is hereby granted, free of charge, to any person obtaining a copy
Packit b5b901
 * of this software and associated documentation files (the "Software"), to
Packit b5b901
 * deal in the Software without restriction, including without limitation the
Packit b5b901
 * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
Packit b5b901
 * sell copies of the Software, and to permit persons to whom the Software is
Packit b5b901
 * furnished to do so, subject to the following conditions:
Packit b5b901
 *
Packit b5b901
 * The above copyright notice and this permission notice shall be included in
Packit b5b901
 * all copies or substantial portions of the Software.
Packit b5b901
 *
Packit b5b901
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
Packit b5b901
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
Packit b5b901
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
Packit b5b901
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
Packit b5b901
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
Packit b5b901
 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
Packit b5b901
 * IN THE SOFTWARE.
Packit b5b901
 */
Packit b5b901
Packit b5b901
#include "uv.h"
Packit b5b901
#include "internal.h"
Packit b5b901
Packit b5b901
#include <stdio.h>
Packit b5b901
#include <stdlib.h>
Packit b5b901
#include <assert.h>
Packit b5b901
#include <errno.h>
Packit b5b901
Packit b5b901
#include <sys/types.h>
Packit b5b901
#include <sys/wait.h>
Packit b5b901
#include <unistd.h>
Packit b5b901
#include <fcntl.h>
Packit b5b901
#include <poll.h>
Packit b5b901
Packit b5b901
#if defined(__APPLE__) && !TARGET_OS_IPHONE
Packit b5b901
# include <crt_externs.h>
Packit b5b901
# define environ (*_NSGetEnviron())
Packit b5b901
#else
Packit b5b901
extern char **environ;
Packit b5b901
#endif
Packit b5b901
Packit b5b901
#if defined(__linux__) || defined(__GLIBC__)
Packit b5b901
# include <grp.h>
Packit b5b901
#endif
Packit b5b901
Packit b5b901
Packit b5b901
static void uv__chld(uv_signal_t* handle, int signum) {
Packit b5b901
  uv_process_t* process;
Packit b5b901
  uv_loop_t* loop;
Packit b5b901
  int exit_status;
Packit b5b901
  int term_signal;
Packit b5b901
  int status;
Packit b5b901
  pid_t pid;
Packit b5b901
  QUEUE pending;
Packit b5b901
  QUEUE* q;
Packit b5b901
  QUEUE* h;
Packit b5b901
Packit b5b901
  assert(signum == SIGCHLD);
Packit b5b901
Packit b5b901
  QUEUE_INIT(&pending);
Packit b5b901
  loop = handle->loop;
Packit b5b901
Packit b5b901
  h = &loop->process_handles;
Packit b5b901
  q = QUEUE_HEAD(h);
Packit b5b901
  while (q != h) {
Packit b5b901
    process = QUEUE_DATA(q, uv_process_t, queue);
Packit b5b901
    q = QUEUE_NEXT(q);
Packit b5b901
Packit b5b901
    do
Packit b5b901
      pid = waitpid(process->pid, &status, WNOHANG);
Packit b5b901
    while (pid == -1 && errno == EINTR);
Packit b5b901
Packit b5b901
    if (pid == 0)
Packit b5b901
      continue;
Packit b5b901
Packit b5b901
    if (pid == -1) {
Packit b5b901
      if (errno != ECHILD)
Packit b5b901
        abort();
Packit b5b901
      continue;
Packit b5b901
    }
Packit b5b901
Packit b5b901
    process->status = status;
Packit b5b901
    QUEUE_REMOVE(&process->queue);
Packit b5b901
    QUEUE_INSERT_TAIL(&pending, &process->queue);
Packit b5b901
  }
Packit b5b901
Packit b5b901
  h = &pending;
Packit b5b901
  q = QUEUE_HEAD(h);
Packit b5b901
  while (q != h) {
Packit b5b901
    process = QUEUE_DATA(q, uv_process_t, queue);
Packit b5b901
    q = QUEUE_NEXT(q);
Packit b5b901
Packit b5b901
    QUEUE_REMOVE(&process->queue);
Packit b5b901
    QUEUE_INIT(&process->queue);
Packit b5b901
    uv__handle_stop(process);
Packit b5b901
Packit b5b901
    if (process->exit_cb == NULL)
Packit b5b901
      continue;
Packit b5b901
Packit b5b901
    exit_status = 0;
Packit b5b901
    if (WIFEXITED(process->status))
Packit b5b901
      exit_status = WEXITSTATUS(process->status);
Packit b5b901
Packit b5b901
    term_signal = 0;
Packit b5b901
    if (WIFSIGNALED(process->status))
Packit b5b901
      term_signal = WTERMSIG(process->status);
Packit b5b901
Packit b5b901
    process->exit_cb(process, exit_status, term_signal);
Packit b5b901
  }
Packit b5b901
  assert(QUEUE_EMPTY(&pending));
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit Service e08953
static int uv__make_socketpair(int fds[2]) {
Packit Service e08953
#if defined(__FreeBSD__) || defined(__linux__)
Packit Service e08953
  if (socketpair(AF_UNIX, SOCK_STREAM | SOCK_CLOEXEC, 0, fds))
Packit b5b901
    return UV__ERR(errno);
Packit b5b901
Packit Service e08953
  return 0;
Packit Service e08953
#else
Packit Service e08953
  int err;
Packit b5b901
Packit b5b901
  if (socketpair(AF_UNIX, SOCK_STREAM, 0, fds))
Packit b5b901
    return UV__ERR(errno);
Packit b5b901
Packit Service e08953
  err = uv__cloexec(fds[0], 1);
Packit Service e08953
  if (err == 0)
Packit Service e08953
    err = uv__cloexec(fds[1], 1);
Packit b5b901
Packit Service e08953
  if (err != 0) {
Packit Service e08953
    uv__close(fds[0]);
Packit Service e08953
    uv__close(fds[1]);
Packit Service e08953
    return UV__ERR(errno);
Packit b5b901
  }
Packit b5b901
Packit b5b901
  return 0;
Packit Service e08953
#endif
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv__make_pipe(int fds[2], int flags) {
Packit Service e08953
#if defined(__FreeBSD__) || defined(__linux__)
Packit Service e08953
  if (pipe2(fds, flags | O_CLOEXEC))
Packit b5b901
    return UV__ERR(errno);
Packit b5b901
Packit Service e08953
  return 0;
Packit Service e08953
#else
Packit b5b901
  if (pipe(fds))
Packit b5b901
    return UV__ERR(errno);
Packit b5b901
Packit Service e08953
  if (uv__cloexec(fds[0], 1))
Packit Service e08953
    goto fail;
Packit Service e08953
Packit Service e08953
  if (uv__cloexec(fds[1], 1))
Packit Service e08953
    goto fail;
Packit b5b901
Packit b5b901
  if (flags & UV__F_NONBLOCK) {
Packit Service e08953
    if (uv__nonblock(fds[0], 1))
Packit Service e08953
      goto fail;
Packit Service e08953
Packit Service e08953
    if (uv__nonblock(fds[1], 1))
Packit Service e08953
      goto fail;
Packit b5b901
  }
Packit b5b901
Packit b5b901
  return 0;
Packit Service e08953
Packit Service e08953
fail:
Packit Service e08953
  uv__close(fds[0]);
Packit Service e08953
  uv__close(fds[1]);
Packit Service e08953
  return UV__ERR(errno);
Packit Service e08953
#endif
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
/*
Packit b5b901
 * Used for initializing stdio streams like options.stdin_stream. Returns
Packit b5b901
 * zero on success. See also the cleanup section in uv_spawn().
Packit b5b901
 */
Packit b5b901
static int uv__process_init_stdio(uv_stdio_container_t* container, int fds[2]) {
Packit b5b901
  int mask;
Packit b5b901
  int fd;
Packit b5b901
Packit b5b901
  mask = UV_IGNORE | UV_CREATE_PIPE | UV_INHERIT_FD | UV_INHERIT_STREAM;
Packit b5b901
Packit b5b901
  switch (container->flags & mask) {
Packit b5b901
  case UV_IGNORE:
Packit b5b901
    return 0;
Packit b5b901
Packit b5b901
  case UV_CREATE_PIPE:
Packit b5b901
    assert(container->data.stream != NULL);
Packit b5b901
    if (container->data.stream->type != UV_NAMED_PIPE)
Packit b5b901
      return UV_EINVAL;
Packit b5b901
    else
Packit Service e08953
      return uv__make_socketpair(fds);
Packit b5b901
Packit b5b901
  case UV_INHERIT_FD:
Packit b5b901
  case UV_INHERIT_STREAM:
Packit b5b901
    if (container->flags & UV_INHERIT_FD)
Packit b5b901
      fd = container->data.fd;
Packit b5b901
    else
Packit b5b901
      fd = uv__stream_fd(container->data.stream);
Packit b5b901
Packit b5b901
    if (fd == -1)
Packit b5b901
      return UV_EINVAL;
Packit b5b901
Packit b5b901
    fds[1] = fd;
Packit b5b901
    return 0;
Packit b5b901
Packit b5b901
  default:
Packit b5b901
    assert(0 && "Unexpected flags");
Packit b5b901
    return UV_EINVAL;
Packit b5b901
  }
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
static int uv__process_open_stream(uv_stdio_container_t* container,
Packit b5b901
                                   int pipefds[2]) {
Packit b5b901
  int flags;
Packit b5b901
  int err;
Packit b5b901
Packit b5b901
  if (!(container->flags & UV_CREATE_PIPE) || pipefds[0] < 0)
Packit b5b901
    return 0;
Packit b5b901
Packit b5b901
  err = uv__close(pipefds[1]);
Packit b5b901
  if (err != 0)
Packit b5b901
    abort();
Packit b5b901
Packit b5b901
  pipefds[1] = -1;
Packit b5b901
  uv__nonblock(pipefds[0], 1);
Packit b5b901
Packit b5b901
  flags = 0;
Packit b5b901
  if (container->flags & UV_WRITABLE_PIPE)
Packit b5b901
    flags |= UV_HANDLE_READABLE;
Packit b5b901
  if (container->flags & UV_READABLE_PIPE)
Packit b5b901
    flags |= UV_HANDLE_WRITABLE;
Packit b5b901
Packit b5b901
  return uv__stream_open(container->data.stream, pipefds[0], flags);
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
static void uv__process_close_stream(uv_stdio_container_t* container) {
Packit b5b901
  if (!(container->flags & UV_CREATE_PIPE)) return;
Packit Service e08953
  uv__stream_close(container->data.stream);
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
static void uv__write_int(int fd, int val) {
Packit b5b901
  ssize_t n;
Packit b5b901
Packit b5b901
  do
Packit b5b901
    n = write(fd, &val, sizeof(val));
Packit b5b901
  while (n == -1 && errno == EINTR);
Packit b5b901
Packit b5b901
  if (n == -1 && errno == EPIPE)
Packit b5b901
    return; /* parent process has quit */
Packit b5b901
Packit b5b901
  assert(n == sizeof(val));
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
#if !(defined(__APPLE__) && (TARGET_OS_TV || TARGET_OS_WATCH))
Packit b5b901
/* execvp is marked __WATCHOS_PROHIBITED __TVOS_PROHIBITED, so must be
Packit b5b901
 * avoided. Since this isn't called on those targets, the function
Packit b5b901
 * doesn't even need to be defined for them.
Packit b5b901
 */
Packit b5b901
static void uv__process_child_init(const uv_process_options_t* options,
Packit b5b901
                                   int stdio_count,
Packit b5b901
                                   int (*pipes)[2],
Packit b5b901
                                   int error_fd) {
Packit b5b901
  sigset_t set;
Packit b5b901
  int close_fd;
Packit b5b901
  int use_fd;
Packit b5b901
  int err;
Packit b5b901
  int fd;
Packit b5b901
  int n;
Packit b5b901
Packit b5b901
  if (options->flags & UV_PROCESS_DETACHED)
Packit b5b901
    setsid();
Packit b5b901
Packit b5b901
  /* First duplicate low numbered fds, since it's not safe to duplicate them,
Packit b5b901
   * they could get replaced. Example: swapping stdout and stderr; without
Packit b5b901
   * this fd 2 (stderr) would be duplicated into fd 1, thus making both
Packit b5b901
   * stdout and stderr go to the same fd, which was not the intention. */
Packit b5b901
  for (fd = 0; fd < stdio_count; fd++) {
Packit b5b901
    use_fd = pipes[fd][1];
Packit b5b901
    if (use_fd < 0 || use_fd >= fd)
Packit b5b901
      continue;
Packit b5b901
    pipes[fd][1] = fcntl(use_fd, F_DUPFD, stdio_count);
Packit b5b901
    if (pipes[fd][1] == -1) {
Packit b5b901
      uv__write_int(error_fd, UV__ERR(errno));
Packit b5b901
      _exit(127);
Packit b5b901
    }
Packit b5b901
  }
Packit b5b901
Packit b5b901
  for (fd = 0; fd < stdio_count; fd++) {
Packit b5b901
    close_fd = pipes[fd][0];
Packit b5b901
    use_fd = pipes[fd][1];
Packit b5b901
Packit b5b901
    if (use_fd < 0) {
Packit b5b901
      if (fd >= 3)
Packit b5b901
        continue;
Packit b5b901
      else {
Packit b5b901
        /* redirect stdin, stdout and stderr to /dev/null even if UV_IGNORE is
Packit b5b901
         * set
Packit b5b901
         */
Packit b5b901
        use_fd = open("/dev/null", fd == 0 ? O_RDONLY : O_RDWR);
Packit b5b901
        close_fd = use_fd;
Packit b5b901
Packit Service e08953
        if (use_fd < 0) {
Packit b5b901
          uv__write_int(error_fd, UV__ERR(errno));
Packit b5b901
          _exit(127);
Packit b5b901
        }
Packit b5b901
      }
Packit b5b901
    }
Packit b5b901
Packit b5b901
    if (fd == use_fd)
Packit b5b901
      uv__cloexec_fcntl(use_fd, 0);
Packit b5b901
    else
Packit b5b901
      fd = dup2(use_fd, fd);
Packit b5b901
Packit b5b901
    if (fd == -1) {
Packit b5b901
      uv__write_int(error_fd, UV__ERR(errno));
Packit b5b901
      _exit(127);
Packit b5b901
    }
Packit b5b901
Packit b5b901
    if (fd <= 2)
Packit b5b901
      uv__nonblock_fcntl(fd, 0);
Packit b5b901
Packit b5b901
    if (close_fd >= stdio_count)
Packit b5b901
      uv__close(close_fd);
Packit b5b901
  }
Packit b5b901
Packit b5b901
  for (fd = 0; fd < stdio_count; fd++) {
Packit b5b901
    use_fd = pipes[fd][1];
Packit b5b901
Packit b5b901
    if (use_fd >= stdio_count)
Packit b5b901
      uv__close(use_fd);
Packit b5b901
  }
Packit b5b901
Packit b5b901
  if (options->cwd != NULL && chdir(options->cwd)) {
Packit b5b901
    uv__write_int(error_fd, UV__ERR(errno));
Packit b5b901
    _exit(127);
Packit b5b901
  }
Packit b5b901
Packit b5b901
  if (options->flags & (UV_PROCESS_SETUID | UV_PROCESS_SETGID)) {
Packit b5b901
    /* When dropping privileges from root, the `setgroups` call will
Packit b5b901
     * remove any extraneous groups. If we don't call this, then
Packit b5b901
     * even though our uid has dropped, we may still have groups
Packit b5b901
     * that enable us to do super-user things. This will fail if we
Packit b5b901
     * aren't root, so don't bother checking the return value, this
Packit b5b901
     * is just done as an optimistic privilege dropping function.
Packit b5b901
     */
Packit b5b901
    SAVE_ERRNO(setgroups(0, NULL));
Packit b5b901
  }
Packit b5b901
Packit b5b901
  if ((options->flags & UV_PROCESS_SETGID) && setgid(options->gid)) {
Packit b5b901
    uv__write_int(error_fd, UV__ERR(errno));
Packit b5b901
    _exit(127);
Packit b5b901
  }
Packit b5b901
Packit b5b901
  if ((options->flags & UV_PROCESS_SETUID) && setuid(options->uid)) {
Packit b5b901
    uv__write_int(error_fd, UV__ERR(errno));
Packit b5b901
    _exit(127);
Packit b5b901
  }
Packit b5b901
Packit b5b901
  if (options->env != NULL) {
Packit b5b901
    environ = options->env;
Packit b5b901
  }
Packit b5b901
Packit b5b901
  /* Reset signal disposition.  Use a hard-coded limit because NSIG
Packit b5b901
   * is not fixed on Linux: it's either 32, 34 or 64, depending on
Packit b5b901
   * whether RT signals are enabled.  We are not allowed to touch
Packit b5b901
   * RT signal handlers, glibc uses them internally.
Packit b5b901
   */
Packit b5b901
  for (n = 1; n < 32; n += 1) {
Packit b5b901
    if (n == SIGKILL || n == SIGSTOP)
Packit b5b901
      continue;  /* Can't be changed. */
Packit b5b901
Packit Service e08953
#if defined(__HAIKU__)
Packit Service e08953
    if (n == SIGKILLTHR)
Packit Service e08953
      continue;  /* Can't be changed. */
Packit Service e08953
#endif
Packit Service e08953
Packit b5b901
    if (SIG_ERR != signal(n, SIG_DFL))
Packit b5b901
      continue;
Packit b5b901
Packit b5b901
    uv__write_int(error_fd, UV__ERR(errno));
Packit b5b901
    _exit(127);
Packit b5b901
  }
Packit b5b901
Packit b5b901
  /* Reset signal mask. */
Packit b5b901
  sigemptyset(&set);
Packit b5b901
  err = pthread_sigmask(SIG_SETMASK, &set, NULL);
Packit b5b901
Packit b5b901
  if (err != 0) {
Packit b5b901
    uv__write_int(error_fd, UV__ERR(err));
Packit b5b901
    _exit(127);
Packit b5b901
  }
Packit b5b901
Packit b5b901
  execvp(options->file, options->args);
Packit b5b901
  uv__write_int(error_fd, UV__ERR(errno));
Packit b5b901
  _exit(127);
Packit b5b901
}
Packit b5b901
#endif
Packit b5b901
Packit b5b901
Packit b5b901
int uv_spawn(uv_loop_t* loop,
Packit b5b901
             uv_process_t* process,
Packit b5b901
             const uv_process_options_t* options) {
Packit b5b901
#if defined(__APPLE__) && (TARGET_OS_TV || TARGET_OS_WATCH)
Packit b5b901
  /* fork is marked __WATCHOS_PROHIBITED __TVOS_PROHIBITED. */
Packit b5b901
  return UV_ENOSYS;
Packit b5b901
#else
Packit b5b901
  int signal_pipe[2] = { -1, -1 };
Packit b5b901
  int pipes_storage[8][2];
Packit b5b901
  int (*pipes)[2];
Packit b5b901
  int stdio_count;
Packit b5b901
  ssize_t r;
Packit b5b901
  pid_t pid;
Packit b5b901
  int err;
Packit b5b901
  int exec_errorno;
Packit b5b901
  int i;
Packit b5b901
  int status;
Packit b5b901
Packit b5b901
  assert(options->file != NULL);
Packit b5b901
  assert(!(options->flags & ~(UV_PROCESS_DETACHED |
Packit b5b901
                              UV_PROCESS_SETGID |
Packit b5b901
                              UV_PROCESS_SETUID |
Packit b5b901
                              UV_PROCESS_WINDOWS_HIDE |
Packit Service e08953
                              UV_PROCESS_WINDOWS_HIDE_CONSOLE |
Packit Service e08953
                              UV_PROCESS_WINDOWS_HIDE_GUI |
Packit b5b901
                              UV_PROCESS_WINDOWS_VERBATIM_ARGUMENTS)));
Packit b5b901
Packit b5b901
  uv__handle_init(loop, (uv_handle_t*)process, UV_PROCESS);
Packit b5b901
  QUEUE_INIT(&process->queue);
Packit b5b901
Packit b5b901
  stdio_count = options->stdio_count;
Packit b5b901
  if (stdio_count < 3)
Packit b5b901
    stdio_count = 3;
Packit b5b901
Packit b5b901
  err = UV_ENOMEM;
Packit b5b901
  pipes = pipes_storage;
Packit b5b901
  if (stdio_count > (int) ARRAY_SIZE(pipes_storage))
Packit b5b901
    pipes = uv__malloc(stdio_count * sizeof(*pipes));
Packit b5b901
Packit b5b901
  if (pipes == NULL)
Packit b5b901
    goto error;
Packit b5b901
Packit b5b901
  for (i = 0; i < stdio_count; i++) {
Packit b5b901
    pipes[i][0] = -1;
Packit b5b901
    pipes[i][1] = -1;
Packit b5b901
  }
Packit b5b901
Packit b5b901
  for (i = 0; i < options->stdio_count; i++) {
Packit b5b901
    err = uv__process_init_stdio(options->stdio + i, pipes[i]);
Packit b5b901
    if (err)
Packit b5b901
      goto error;
Packit b5b901
  }
Packit b5b901
Packit b5b901
  /* This pipe is used by the parent to wait until
Packit b5b901
   * the child has called `execve()`. We need this
Packit b5b901
   * to avoid the following race condition:
Packit b5b901
   *
Packit b5b901
   *    if ((pid = fork()) > 0) {
Packit b5b901
   *      kill(pid, SIGTERM);
Packit b5b901
   *    }
Packit b5b901
   *    else if (pid == 0) {
Packit b5b901
   *      execve("/bin/cat", argp, envp);
Packit b5b901
   *    }
Packit b5b901
   *
Packit b5b901
   * The parent sends a signal immediately after forking.
Packit b5b901
   * Since the child may not have called `execve()` yet,
Packit b5b901
   * there is no telling what process receives the signal,
Packit b5b901
   * our fork or /bin/cat.
Packit b5b901
   *
Packit b5b901
   * To avoid ambiguity, we create a pipe with both ends
Packit b5b901
   * marked close-on-exec. Then, after the call to `fork()`,
Packit b5b901
   * the parent polls the read end until it EOFs or errors with EPIPE.
Packit b5b901
   */
Packit b5b901
  err = uv__make_pipe(signal_pipe, 0);
Packit b5b901
  if (err)
Packit b5b901
    goto error;
Packit b5b901
Packit b5b901
  uv_signal_start(&loop->child_watcher, uv__chld, SIGCHLD);
Packit b5b901
Packit b5b901
  /* Acquire write lock to prevent opening new fds in worker threads */
Packit b5b901
  uv_rwlock_wrlock(&loop->cloexec_lock);
Packit b5b901
  pid = fork();
Packit b5b901
Packit b5b901
  if (pid == -1) {
Packit b5b901
    err = UV__ERR(errno);
Packit b5b901
    uv_rwlock_wrunlock(&loop->cloexec_lock);
Packit b5b901
    uv__close(signal_pipe[0]);
Packit b5b901
    uv__close(signal_pipe[1]);
Packit b5b901
    goto error;
Packit b5b901
  }
Packit b5b901
Packit b5b901
  if (pid == 0) {
Packit b5b901
    uv__process_child_init(options, stdio_count, pipes, signal_pipe[1]);
Packit b5b901
    abort();
Packit b5b901
  }
Packit b5b901
Packit b5b901
  /* Release lock in parent process */
Packit b5b901
  uv_rwlock_wrunlock(&loop->cloexec_lock);
Packit b5b901
  uv__close(signal_pipe[1]);
Packit b5b901
Packit b5b901
  process->status = 0;
Packit b5b901
  exec_errorno = 0;
Packit b5b901
  do
Packit b5b901
    r = read(signal_pipe[0], &exec_errorno, sizeof(exec_errorno));
Packit b5b901
  while (r == -1 && errno == EINTR);
Packit b5b901
Packit b5b901
  if (r == 0)
Packit b5b901
    ; /* okay, EOF */
Packit b5b901
  else if (r == sizeof(exec_errorno)) {
Packit b5b901
    do
Packit b5b901
      err = waitpid(pid, &status, 0); /* okay, read errorno */
Packit b5b901
    while (err == -1 && errno == EINTR);
Packit b5b901
    assert(err == pid);
Packit b5b901
  } else if (r == -1 && errno == EPIPE) {
Packit b5b901
    do
Packit b5b901
      err = waitpid(pid, &status, 0); /* okay, got EPIPE */
Packit b5b901
    while (err == -1 && errno == EINTR);
Packit b5b901
    assert(err == pid);
Packit b5b901
  } else
Packit b5b901
    abort();
Packit b5b901
Packit b5b901
  uv__close_nocheckstdio(signal_pipe[0]);
Packit b5b901
Packit b5b901
  for (i = 0; i < options->stdio_count; i++) {
Packit b5b901
    err = uv__process_open_stream(options->stdio + i, pipes[i]);
Packit b5b901
    if (err == 0)
Packit b5b901
      continue;
Packit b5b901
Packit b5b901
    while (i--)
Packit b5b901
      uv__process_close_stream(options->stdio + i);
Packit b5b901
Packit b5b901
    goto error;
Packit b5b901
  }
Packit b5b901
Packit b5b901
  /* Only activate this handle if exec() happened successfully */
Packit b5b901
  if (exec_errorno == 0) {
Packit b5b901
    QUEUE_INSERT_TAIL(&loop->process_handles, &process->queue);
Packit b5b901
    uv__handle_start(process);
Packit b5b901
  }
Packit b5b901
Packit b5b901
  process->pid = pid;
Packit b5b901
  process->exit_cb = options->exit_cb;
Packit b5b901
Packit b5b901
  if (pipes != pipes_storage)
Packit b5b901
    uv__free(pipes);
Packit b5b901
Packit b5b901
  return exec_errorno;
Packit b5b901
Packit b5b901
error:
Packit b5b901
  if (pipes != NULL) {
Packit b5b901
    for (i = 0; i < stdio_count; i++) {
Packit b5b901
      if (i < options->stdio_count)
Packit b5b901
        if (options->stdio[i].flags & (UV_INHERIT_FD | UV_INHERIT_STREAM))
Packit b5b901
          continue;
Packit b5b901
      if (pipes[i][0] != -1)
Packit b5b901
        uv__close_nocheckstdio(pipes[i][0]);
Packit b5b901
      if (pipes[i][1] != -1)
Packit b5b901
        uv__close_nocheckstdio(pipes[i][1]);
Packit b5b901
    }
Packit b5b901
Packit b5b901
    if (pipes != pipes_storage)
Packit b5b901
      uv__free(pipes);
Packit b5b901
  }
Packit b5b901
Packit b5b901
  return err;
Packit b5b901
#endif
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_process_kill(uv_process_t* process, int signum) {
Packit b5b901
  return uv_kill(process->pid, signum);
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
int uv_kill(int pid, int signum) {
Packit b5b901
  if (kill(pid, signum))
Packit b5b901
    return UV__ERR(errno);
Packit b5b901
  else
Packit b5b901
    return 0;
Packit b5b901
}
Packit b5b901
Packit b5b901
Packit b5b901
void uv__process_close(uv_process_t* handle) {
Packit b5b901
  QUEUE_REMOVE(&handle->queue);
Packit b5b901
  uv__handle_stop(handle);
Packit b5b901
  if (QUEUE_EMPTY(&handle->loop->process_handles))
Packit b5b901
    uv_signal_stop(&handle->loop->child_watcher);
Packit b5b901
}