Blame gnulib-tests/test-dup-safer.c

Packit 709fb3
/* Test that dup_safer leaves standard fds alone.
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 <fcntl.h>
Packit 709fb3
#include <errno.h>
Packit 709fb3
#include <stdbool.h>
Packit 709fb3
#include <stdio.h>
Packit 709fb3
#include <unistd.h>
Packit 709fb3
Packit 709fb3
#include "binary-io.h"
Packit 709fb3
#include "cloexec.h"
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
#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
/* This test intentionally closes stderr.  So, we arrange to have fd 10
Packit 709fb3
   (outside the range of interesting fd's during the test) set up to
Packit 709fb3
   duplicate the original stderr.  */
Packit 709fb3
Packit 709fb3
#define BACKUP_STDERR_FILENO 10
Packit 709fb3
#define ASSERT_STREAM myerr
Packit 709fb3
#include "macros.h"
Packit 709fb3
Packit 709fb3
static FILE *myerr;
Packit 709fb3
Packit 709fb3
/* Return true if FD is open.  */
Packit 709fb3
static bool
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
/* Return true if FD is open and inheritable across exec/spawn.  */
Packit 709fb3
static bool
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
Packit 709fb3
/* Return true if FD is open in the given MODE, which is either
Packit 709fb3
   O_TEXT or O_BINARY.  */
Packit 709fb3
static bool
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
#define witness "test-dup-safer.txt"
Packit 709fb3
Packit 709fb3
int
Packit 709fb3
main (void)
Packit 709fb3
{
Packit 709fb3
  int i;
Packit 709fb3
  int fd;
Packit 709fb3
  int bad_fd = getdtablesize ();
Packit 709fb3
Packit 709fb3
  /* We close fd 2 later, so save it in fd 10.  */
Packit 709fb3
  if (dup2 (STDERR_FILENO, BACKUP_STDERR_FILENO) != BACKUP_STDERR_FILENO
Packit 709fb3
      || (myerr = fdopen (BACKUP_STDERR_FILENO, "w")) == NULL)
Packit 709fb3
    return 2;
Packit 709fb3
Packit 709fb3
  /* Create file for later checks.  */
Packit 709fb3
  fd = creat (witness, 0600);
Packit 709fb3
  ASSERT (STDERR_FILENO < fd);
Packit 709fb3
Packit 709fb3
  /* Four iterations, with progressively more standard descriptors
Packit 709fb3
     closed.  */
Packit 709fb3
  for (i = -1; i <= STDERR_FILENO; i++)
Packit 709fb3
    {
Packit 709fb3
      if (0 <= i)
Packit 709fb3
        ASSERT (close (i) == 0);
Packit 709fb3
Packit 709fb3
      /* Detect errors.  */
Packit 709fb3
      errno = 0;
Packit 709fb3
      ASSERT (dup (-1) == -1);
Packit 709fb3
      ASSERT (errno == EBADF);
Packit 709fb3
      errno = 0;
Packit 709fb3
      ASSERT (dup (bad_fd) == -1);
Packit 709fb3
      ASSERT (errno == EBADF);
Packit 709fb3
      close (fd + 1);
Packit 709fb3
      errno = 0;
Packit 709fb3
      ASSERT (dup (fd + 1) == -1);
Packit 709fb3
      ASSERT (errno == EBADF);
Packit 709fb3
Packit 709fb3
      /* Preserve text vs. binary.  */
Packit 709fb3
      setmode (fd, O_BINARY);
Packit 709fb3
      ASSERT (dup (fd) == fd + 1);
Packit 709fb3
      ASSERT (is_open (fd + 1));
Packit 709fb3
      ASSERT (is_inheritable (fd + 1));
Packit 709fb3
      ASSERT (is_mode (fd + 1, O_BINARY));
Packit 709fb3
Packit 709fb3
      ASSERT (close (fd + 1) == 0);
Packit 709fb3
      setmode (fd, O_TEXT);
Packit 709fb3
      ASSERT (dup (fd) == fd + 1);
Packit 709fb3
      ASSERT (is_open (fd + 1));
Packit 709fb3
      ASSERT (is_inheritable (fd + 1));
Packit 709fb3
      ASSERT (is_mode (fd + 1, O_TEXT));
Packit 709fb3
Packit 709fb3
      /* Create cloexec copy.  */
Packit 709fb3
      ASSERT (close (fd + 1) == 0);
Packit 709fb3
      ASSERT (fd_safer_flag (dup_cloexec (fd), O_CLOEXEC) == fd + 1);
Packit 709fb3
      ASSERT (set_cloexec_flag (fd + 1, true) == 0);
Packit 709fb3
      ASSERT (is_open (fd + 1));
Packit 709fb3
      ASSERT (!is_inheritable (fd + 1));
Packit 709fb3
      ASSERT (close (fd) == 0);
Packit 709fb3
Packit 709fb3
      /* dup always creates inheritable copies.  Also, check that
Packit 709fb3
         earliest slot past std fds is used.  */
Packit 709fb3
      ASSERT (dup (fd + 1) == fd);
Packit 709fb3
      ASSERT (is_open (fd));
Packit 709fb3
      ASSERT (is_inheritable (fd));
Packit 709fb3
      ASSERT (close (fd + 1) == 0);
Packit 709fb3
    }
Packit 709fb3
Packit 709fb3
  /* Cleanup.  */
Packit 709fb3
  ASSERT (close (fd) == 0);
Packit 709fb3
  ASSERT (unlink (witness) == 0);
Packit 709fb3
Packit 709fb3
  return 0;
Packit 709fb3
}