Blame lib/dup2.c

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