Blame lib/dup2.c

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