Blame gnulib-tests/test-dup2.c

Packit 33f14e
/* Test duplicating file descriptors.
Packit 33f14e
   Copyright (C) 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 Eric Blake <ebb9@byu.net>, 2009.  */
Packit 33f14e
Packit 33f14e
#include <config.h>
Packit 33f14e
Packit 33f14e
#include <unistd.h>
Packit 33f14e
Packit 33f14e
#include "signature.h"
Packit 33f14e
SIGNATURE_CHECK (dup2, int, (int, int));
Packit 33f14e
Packit 33f14e
#include <errno.h>
Packit 33f14e
#include <fcntl.h>
Packit 33f14e
Packit 33f14e
#if HAVE_SYS_RESOURCE_H
Packit 33f14e
# include <sys/resource.h>
Packit 33f14e
#endif
Packit 33f14e
Packit 33f14e
#include "binary-io.h"
Packit 33f14e
Packit 33f14e
#if GNULIB_TEST_CLOEXEC
Packit 33f14e
# include "cloexec.h"
Packit 33f14e
#endif
Packit 33f14e
Packit 33f14e
#if (defined _WIN32 || defined __WIN32__) && ! defined __CYGWIN__
Packit 33f14e
/* Get declarations of the native Windows API functions.  */
Packit 33f14e
# define WIN32_LEAN_AND_MEAN
Packit 33f14e
# include <windows.h>
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
#endif
Packit 33f14e
Packit 33f14e
#include "macros.h"
Packit 33f14e
Packit 33f14e
/* Return non-zero if FD is open.  */
Packit 33f14e
static int
Packit 33f14e
is_open (int fd)
Packit 33f14e
{
Packit 33f14e
#if (defined _WIN32 || defined __WIN32__) && ! defined __CYGWIN__
Packit 33f14e
  /* On native Windows, the initial state of unassigned standard file
Packit 33f14e
     descriptors is that they are open but point to an
Packit 33f14e
     INVALID_HANDLE_VALUE, and there is no fcntl.  */
Packit 33f14e
  return (HANDLE) _get_osfhandle (fd) != INVALID_HANDLE_VALUE;
Packit 33f14e
#else
Packit 33f14e
# ifndef F_GETFL
Packit 33f14e
#  error Please port fcntl to your platform
Packit 33f14e
# endif
Packit 33f14e
  return 0 <= fcntl (fd, F_GETFL);
Packit 33f14e
#endif
Packit 33f14e
}
Packit 33f14e
Packit 33f14e
#if GNULIB_TEST_CLOEXEC
Packit 33f14e
/* Return non-zero if FD is open and inheritable across exec/spawn.  */
Packit 33f14e
static int
Packit 33f14e
is_inheritable (int fd)
Packit 33f14e
{
Packit 33f14e
# if (defined _WIN32 || defined __WIN32__) && ! defined __CYGWIN__
Packit 33f14e
  /* On native Windows, the initial state of unassigned standard file
Packit 33f14e
     descriptors is that they are open but point to an
Packit 33f14e
     INVALID_HANDLE_VALUE, and there is no fcntl.  */
Packit 33f14e
  HANDLE h = (HANDLE) _get_osfhandle (fd);
Packit 33f14e
  DWORD flags;
Packit 33f14e
  if (h == INVALID_HANDLE_VALUE || GetHandleInformation (h, &flags) == 0)
Packit 33f14e
    return 0;
Packit 33f14e
  return (flags & HANDLE_FLAG_INHERIT) != 0;
Packit 33f14e
# else
Packit 33f14e
#  ifndef F_GETFD
Packit 33f14e
#   error Please port fcntl to your platform
Packit 33f14e
#  endif
Packit 33f14e
  int i = fcntl (fd, F_GETFD);
Packit 33f14e
  return 0 <= i && (i & FD_CLOEXEC) == 0;
Packit 33f14e
# endif
Packit 33f14e
}
Packit 33f14e
#endif /* GNULIB_TEST_CLOEXEC */
Packit 33f14e
Packit 33f14e
#if !O_BINARY
Packit 33f14e
# define setmode(f,m) zero ()
Packit 33f14e
static int zero (void) { return 0; }
Packit 33f14e
#endif
Packit 33f14e
Packit 33f14e
/* Return non-zero if FD is open in the given MODE, which is either
Packit 33f14e
   O_TEXT or O_BINARY.  */
Packit 33f14e
static int
Packit 33f14e
is_mode (int fd, int mode)
Packit 33f14e
{
Packit 33f14e
  int value = setmode (fd, O_BINARY);
Packit 33f14e
  setmode (fd, value);
Packit 33f14e
  return mode == value;
Packit 33f14e
}
Packit 33f14e
Packit 33f14e
int
Packit 33f14e
main (void)
Packit 33f14e
{
Packit 33f14e
  const char *file = "test-dup2.tmp";
Packit 33f14e
  char buffer[1];
Packit 33f14e
  int bad_fd = getdtablesize ();
Packit 33f14e
  int fd = open (file, O_CREAT | O_TRUNC | O_RDWR, 0600);
Packit 33f14e
Packit 33f14e
  /* Assume std descriptors were provided by invoker.  */
Packit 33f14e
  ASSERT (STDERR_FILENO < fd);
Packit 33f14e
  ASSERT (is_open (fd));
Packit 33f14e
  /* Ignore any other fd's leaked into this process.  */
Packit 33f14e
  close (fd + 1);
Packit 33f14e
  close (fd + 2);
Packit 33f14e
  ASSERT (!is_open (fd + 1));
Packit 33f14e
  ASSERT (!is_open (fd + 2));
Packit 33f14e
Packit 33f14e
  /* Assigning to self must be a no-op.  */
Packit 33f14e
  ASSERT (dup2 (fd, fd) == fd);
Packit 33f14e
  ASSERT (is_open (fd));
Packit 33f14e
Packit 33f14e
  /* The source must be valid.  */
Packit 33f14e
  errno = 0;
Packit 33f14e
  ASSERT (dup2 (-1, fd) == -1);
Packit 33f14e
  ASSERT (errno == EBADF);
Packit 33f14e
  close (99);
Packit 33f14e
  errno = 0;
Packit 33f14e
  ASSERT (dup2 (99, fd) == -1);
Packit 33f14e
  ASSERT (errno == EBADF);
Packit 33f14e
  errno = 0;
Packit 33f14e
  ASSERT (dup2 (AT_FDCWD, fd) == -1);
Packit 33f14e
  ASSERT (errno == EBADF);
Packit 33f14e
  ASSERT (is_open (fd));
Packit 33f14e
Packit 33f14e
  /* If the source is not open, then the destination is unaffected.  */
Packit 33f14e
  errno = 0;
Packit 33f14e
  ASSERT (dup2 (fd + 1, fd + 1) == -1);
Packit 33f14e
  ASSERT (errno == EBADF);
Packit 33f14e
  ASSERT (!is_open (fd + 1));
Packit 33f14e
  errno = 0;
Packit 33f14e
  ASSERT (dup2 (fd + 1, fd) == -1);
Packit 33f14e
  ASSERT (errno == EBADF);
Packit 33f14e
  ASSERT (is_open (fd));
Packit 33f14e
Packit 33f14e
  /* The destination must be valid.  */
Packit 33f14e
  errno = 0;
Packit 33f14e
  ASSERT (dup2 (fd, -2) == -1);
Packit 33f14e
  ASSERT (errno == EBADF);
Packit 33f14e
  if (bad_fd > 256)
Packit 33f14e
    {
Packit 33f14e
      ASSERT (dup2 (fd, 255) == 255);
Packit 33f14e
      ASSERT (dup2 (fd, 256) == 256);
Packit 33f14e
      ASSERT (close (255) == 0);
Packit 33f14e
      ASSERT (close (256) == 0);
Packit 33f14e
    }
Packit 33f14e
  ASSERT (dup2 (fd, bad_fd - 1) == bad_fd - 1);
Packit 33f14e
  ASSERT (close (bad_fd - 1) == 0);
Packit 33f14e
  errno = 0;
Packit 33f14e
  ASSERT (dup2 (fd, bad_fd) == -1);
Packit 33f14e
  ASSERT (errno == EBADF);
Packit 33f14e
Packit 33f14e
  /* Using dup2 can skip fds.  */
Packit 33f14e
  ASSERT (dup2 (fd, fd + 2) == fd + 2);
Packit 33f14e
  ASSERT (is_open (fd));
Packit 33f14e
  ASSERT (!is_open (fd + 1));
Packit 33f14e
  ASSERT (is_open (fd + 2));
Packit 33f14e
Packit 33f14e
  /* Verify that dup2 closes the previous occupant of a fd.  */
Packit 33f14e
  ASSERT (open ("/dev/null", O_WRONLY, 0600) == fd + 1);
Packit 33f14e
  ASSERT (dup2 (fd + 1, fd) == fd);
Packit 33f14e
  ASSERT (close (fd + 1) == 0);
Packit 33f14e
  ASSERT (write (fd, "1", 1) == 1);
Packit 33f14e
  ASSERT (dup2 (fd + 2, fd) == fd);
Packit 33f14e
  ASSERT (lseek (fd, 0, SEEK_END) == 0);
Packit 33f14e
  ASSERT (write (fd + 2, "2", 1) == 1);
Packit 33f14e
  ASSERT (lseek (fd, 0, SEEK_SET) == 0);
Packit 33f14e
  ASSERT (read (fd, buffer, 1) == 1);
Packit 33f14e
  ASSERT (*buffer == '2');
Packit 33f14e
Packit 33f14e
#if GNULIB_TEST_CLOEXEC
Packit 33f14e
  /* Any new fd created by dup2 must not be cloexec.  */
Packit 33f14e
  ASSERT (close (fd + 2) == 0);
Packit 33f14e
  ASSERT (dup_cloexec (fd) == fd + 1);
Packit 33f14e
  ASSERT (!is_inheritable (fd + 1));
Packit 33f14e
  ASSERT (dup2 (fd + 1, fd + 1) == fd + 1);
Packit 33f14e
  ASSERT (!is_inheritable (fd + 1));
Packit 33f14e
  ASSERT (dup2 (fd + 1, fd + 2) == fd + 2);
Packit 33f14e
  ASSERT (!is_inheritable (fd + 1));
Packit 33f14e
  ASSERT (is_inheritable (fd + 2));
Packit 33f14e
  errno = 0;
Packit 33f14e
  ASSERT (dup2 (fd + 1, -1) == -1);
Packit 33f14e
  ASSERT (errno == EBADF);
Packit 33f14e
  ASSERT (!is_inheritable (fd + 1));
Packit 33f14e
#endif
Packit 33f14e
Packit 33f14e
  /* On systems that distinguish between text and binary mode, dup2
Packit 33f14e
     reuses the mode of the source.  */
Packit 33f14e
  setmode (fd, O_BINARY);
Packit 33f14e
  ASSERT (is_mode (fd, O_BINARY));
Packit 33f14e
  ASSERT (dup2 (fd, fd + 1) == fd + 1);
Packit 33f14e
  ASSERT (is_mode (fd + 1, O_BINARY));
Packit 33f14e
  setmode (fd, O_TEXT);
Packit 33f14e
  ASSERT (is_mode (fd, O_TEXT));
Packit 33f14e
  ASSERT (dup2 (fd, fd + 1) == fd + 1);
Packit 33f14e
  ASSERT (is_mode (fd + 1, O_TEXT));
Packit 33f14e
Packit 33f14e
  /* Clean up.  */
Packit 33f14e
  ASSERT (close (fd + 2) == 0);
Packit 33f14e
  ASSERT (close (fd + 1) == 0);
Packit 33f14e
  ASSERT (close (fd) == 0);
Packit 33f14e
  ASSERT (unlink (file) == 0);
Packit 33f14e
Packit 33f14e
  return 0;
Packit 33f14e
}