Blame gnulib-tests/test-dup2.c

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