Blame lib/dup2.c

Packit 33f14e
/* Duplicate an open file descriptor to a specified file descriptor.
Packit 33f14e
Packit 33f14e
   Copyright (C) 1999, 2004-2007, 2009-2017 Free Software Foundation, Inc.
Packit 33f14e
Packit 33f14e
   This program is free software: you can redistribute it and/or modify
Packit 33f14e
   it under the terms of the GNU General Public License as published by
Packit 33f14e
   the Free Software Foundation; either version 3 of the License, or
Packit 33f14e
   (at your option) any later version.
Packit 33f14e
Packit 33f14e
   This program is distributed in the hope that it will be useful,
Packit 33f14e
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 33f14e
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit 33f14e
   GNU General Public License for more details.
Packit 33f14e
Packit 33f14e
   You should have received a copy of the GNU General Public License
Packit 33f14e
   along with this program.  If not, see <http://www.gnu.org/licenses/>.  */
Packit 33f14e
Packit 33f14e
/* written by Paul Eggert */
Packit 33f14e
Packit 33f14e
#include <config.h>
Packit 33f14e
Packit 33f14e
/* Specification.  */
Packit 33f14e
#include <unistd.h>
Packit 33f14e
Packit 33f14e
#include <errno.h>
Packit 33f14e
#include <fcntl.h>
Packit 33f14e
Packit 33f14e
#if HAVE_DUP2
Packit 33f14e
Packit 33f14e
# undef dup2
Packit 33f14e
Packit 33f14e
# if (defined _WIN32 || defined __WIN32__) && ! defined __CYGWIN__
Packit 33f14e
Packit 33f14e
/* Get declarations of the native Windows API functions.  */
Packit 33f14e
#  define WIN32_LEAN_AND_MEAN
Packit 33f14e
#  include <windows.h>
Packit 33f14e
Packit 33f14e
#  if HAVE_MSVC_INVALID_PARAMETER_HANDLER
Packit 33f14e
#   include "msvc-inval.h"
Packit 33f14e
#  endif
Packit 33f14e
Packit 33f14e
/* Get _get_osfhandle.  */
Packit 33f14e
#  if GNULIB_MSVC_NOTHROW
Packit 33f14e
#   include "msvc-nothrow.h"
Packit 33f14e
#  else
Packit 33f14e
#   include <io.h>
Packit 33f14e
#  endif
Packit 33f14e
Packit 33f14e
#  if HAVE_MSVC_INVALID_PARAMETER_HANDLER
Packit 33f14e
static int
Packit 33f14e
dup2_nothrow (int fd, int desired_fd)
Packit 33f14e
{
Packit 33f14e
  int result;
Packit 33f14e
Packit 33f14e
  TRY_MSVC_INVAL
Packit 33f14e
    {
Packit 33f14e
      result = dup2 (fd, desired_fd);
Packit 33f14e
    }
Packit 33f14e
  CATCH_MSVC_INVAL
Packit 33f14e
    {
Packit 33f14e
      errno = EBADF;
Packit 33f14e
      result = -1;
Packit 33f14e
    }
Packit 33f14e
  DONE_MSVC_INVAL;
Packit 33f14e
Packit 33f14e
  return result;
Packit 33f14e
}
Packit 33f14e
#  else
Packit 33f14e
#   define dup2_nothrow dup2
Packit 33f14e
#  endif
Packit 33f14e
Packit 33f14e
static int
Packit 33f14e
ms_windows_dup2 (int fd, int desired_fd)
Packit 33f14e
{
Packit 33f14e
  int result;
Packit 33f14e
Packit 33f14e
  /* If fd is closed, mingw hangs on dup2 (fd, fd).  If fd is open,
Packit 33f14e
     dup2 (fd, fd) returns 0, but all further attempts to use fd in
Packit 33f14e
     future dup2 calls will hang.  */
Packit 33f14e
  if (fd == desired_fd)
Packit 33f14e
    {
Packit 33f14e
      if ((HANDLE) _get_osfhandle (fd) == INVALID_HANDLE_VALUE)
Packit 33f14e
        {
Packit 33f14e
          errno = EBADF;
Packit 33f14e
          return -1;
Packit 33f14e
        }
Packit 33f14e
      return fd;
Packit 33f14e
    }
Packit 33f14e
Packit 33f14e
  /* Wine 1.0.1 return 0 when desired_fd is negative but not -1:
Packit 33f14e
     http://bugs.winehq.org/show_bug.cgi?id=21289 */
Packit 33f14e
  if (desired_fd < 0)
Packit 33f14e
    {
Packit 33f14e
      errno = EBADF;
Packit 33f14e
      return -1;
Packit 33f14e
    }
Packit 33f14e
Packit 33f14e
  result = dup2_nothrow (fd, desired_fd);
Packit 33f14e
Packit 33f14e
  if (result == 0)
Packit 33f14e
    result = desired_fd;
Packit 33f14e
Packit 33f14e
  return result;
Packit 33f14e
}
Packit 33f14e
Packit 33f14e
#  define dup2 ms_windows_dup2
Packit 33f14e
Packit 33f14e
# elif defined __KLIBC__
Packit 33f14e
Packit 33f14e
#  include <InnoTekLIBC/backend.h>
Packit 33f14e
Packit 33f14e
static int
Packit 33f14e
klibc_dup2dirfd (int fd, int desired_fd)
Packit 33f14e
{
Packit 33f14e
  int tempfd;
Packit 33f14e
  int dupfd;
Packit 33f14e
Packit 33f14e
  tempfd = open ("NUL", O_RDONLY);
Packit 33f14e
  if (tempfd == -1)
Packit 33f14e
    return -1;
Packit 33f14e
Packit 33f14e
  if (tempfd == desired_fd)
Packit 33f14e
    {
Packit 33f14e
      close (tempfd);
Packit 33f14e
Packit 33f14e
      char path[_MAX_PATH];
Packit 33f14e
      if (__libc_Back_ioFHToPath (fd, path, sizeof (path)))
Packit 33f14e
        return -1;
Packit 33f14e
Packit 33f14e
      return open(path, O_RDONLY);
Packit 33f14e
    }
Packit 33f14e
Packit 33f14e
  dupfd = klibc_dup2dirfd (fd, desired_fd);
Packit 33f14e
Packit 33f14e
  close (tempfd);
Packit 33f14e
Packit 33f14e
  return dupfd;
Packit 33f14e
}
Packit 33f14e
Packit 33f14e
static int
Packit 33f14e
klibc_dup2 (int fd, int desired_fd)
Packit 33f14e
{
Packit 33f14e
  int dupfd;
Packit 33f14e
  struct stat sbuf;
Packit 33f14e
Packit 33f14e
  dupfd = dup2 (fd, desired_fd);
Packit 33f14e
  if (dupfd == -1 && errno == ENOTSUP \
Packit 33f14e
      && !fstat (fd, &sbuf) && S_ISDIR (sbuf.st_mode))
Packit 33f14e
    {
Packit 33f14e
      close (desired_fd);
Packit 33f14e
Packit 33f14e
      return klibc_dup2dirfd (fd, desired_fd);
Packit 33f14e
    }
Packit 33f14e
Packit 33f14e
  return dupfd;
Packit 33f14e
}
Packit 33f14e
Packit 33f14e
#  define dup2 klibc_dup2
Packit 33f14e
# endif
Packit 33f14e
Packit 33f14e
int
Packit 33f14e
rpl_dup2 (int fd, int desired_fd)
Packit 33f14e
{
Packit 33f14e
  int result;
Packit 33f14e
Packit 33f14e
# ifdef F_GETFL
Packit 33f14e
  /* On Linux kernels 2.6.26-2.6.29, dup2 (fd, fd) returns -EBADF.
Packit 33f14e
     On Cygwin 1.5.x, dup2 (1, 1) returns 0.
Packit 33f14e
     On Cygwin 1.7.17, dup2 (1, -1) dumps core.
Packit 33f14e
     On Cygwin 1.7.25, dup2 (1, 256) can dump core.
Packit 33f14e
     On Haiku, dup2 (fd, fd) mistakenly clears FD_CLOEXEC.  */
Packit 33f14e
#  if HAVE_SETDTABLESIZE
Packit 33f14e
  setdtablesize (desired_fd + 1);
Packit 33f14e
#  endif
Packit 33f14e
  if (desired_fd < 0)
Packit 33f14e
    fd = desired_fd;
Packit 33f14e
  if (fd == desired_fd)
Packit 33f14e
    return fcntl (fd, F_GETFL) == -1 ? -1 : fd;
Packit 33f14e
# endif
Packit 33f14e
Packit 33f14e
  result = dup2 (fd, desired_fd);
Packit 33f14e
Packit 33f14e
  /* Correct an errno value on FreeBSD 6.1 and Cygwin 1.5.x.  */
Packit 33f14e
  if (result == -1 && errno == EMFILE)
Packit 33f14e
    errno = EBADF;
Packit 33f14e
# if REPLACE_FCHDIR
Packit 33f14e
  if (fd != desired_fd && result != -1)
Packit 33f14e
    result = _gl_register_dup (fd, result);
Packit 33f14e
# endif
Packit 33f14e
  return result;
Packit 33f14e
}
Packit 33f14e
Packit 33f14e
#else /* !HAVE_DUP2 */
Packit 33f14e
Packit 33f14e
/* On older platforms, dup2 did not exist.  */
Packit 33f14e
Packit 33f14e
# ifndef F_DUPFD
Packit 33f14e
static int
Packit 33f14e
dupfd (int fd, int desired_fd)
Packit 33f14e
{
Packit 33f14e
  int duplicated_fd = dup (fd);
Packit 33f14e
  if (duplicated_fd < 0 || duplicated_fd == desired_fd)
Packit 33f14e
    return duplicated_fd;
Packit 33f14e
  else
Packit 33f14e
    {
Packit 33f14e
      int r = dupfd (fd, desired_fd);
Packit 33f14e
      int e = errno;
Packit 33f14e
      close (duplicated_fd);
Packit 33f14e
      errno = e;
Packit 33f14e
      return r;
Packit 33f14e
    }
Packit 33f14e
}
Packit 33f14e
# endif
Packit 33f14e
Packit 33f14e
int
Packit 33f14e
dup2 (int fd, int desired_fd)
Packit 33f14e
{
Packit 33f14e
  int result = fcntl (fd, F_GETFL) < 0 ? -1 : fd;
Packit 33f14e
  if (result == -1 || fd == desired_fd)
Packit 33f14e
    return result;
Packit 33f14e
  close (desired_fd);
Packit 33f14e
# ifdef F_DUPFD
Packit 33f14e
  result = fcntl (fd, F_DUPFD, desired_fd);
Packit 33f14e
#  if REPLACE_FCHDIR
Packit 33f14e
  if (0 <= result)
Packit 33f14e
    result = _gl_register_dup (fd, result);
Packit 33f14e
#  endif
Packit 33f14e
# else
Packit 33f14e
  result = dupfd (fd, desired_fd);
Packit 33f14e
# endif
Packit 33f14e
  if (result == -1 && (errno == EMFILE || errno == EINVAL))
Packit 33f14e
    errno = EBADF;
Packit 33f14e
  return result;
Packit 33f14e
}
Packit 33f14e
#endif /* !HAVE_DUP2 */