Blame gnulib-tests/test-dup2.c

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