Blame gnulib-tests/test-fcntl.c

Packit Service fdd496
/* Test of fcntl(2).
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
/* Specification.  */
Packit Service fdd496
#include <fcntl.h>
Packit Service fdd496
Packit Service fdd496
#include "signature.h"
Packit Service fdd496
SIGNATURE_CHECK (fcntl, int, (int, int, ...));
Packit Service fdd496
Packit Service fdd496
/* Helpers.  */
Packit Service fdd496
#include <errno.h>
Packit Service fdd496
#include <stdarg.h>
Packit Service fdd496
#include <stdbool.h>
Packit Service fdd496
#include <unistd.h>
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 "binary-io.h"
Packit Service fdd496
#include "macros.h"
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 true if FD is open.  */
Packit Service fdd496
static bool
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
/* Return true if FD is open and inheritable across exec/spawn.  */
Packit Service fdd496
static bool
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 false;
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
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 bool
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
/* Since native fcntl can have more supported operations than our
Packit Service fdd496
   replacement is aware of, and since various operations assign
Packit Service fdd496
   different types to the vararg argument, a wrapper around fcntl must
Packit Service fdd496
   be able to pass a vararg of unknown type on through to the original
Packit Service fdd496
   fcntl.  Make sure that this works properly: func1 behaves like the
Packit Service fdd496
   original fcntl interpreting the vararg as an int or a pointer to a
Packit Service fdd496
   struct, and func2 behaves like rpl_fcntl that doesn't know what
Packit Service fdd496
   type to forward.  */
Packit Service fdd496
struct dummy_struct
Packit Service fdd496
{
Packit Service fdd496
  long filler;
Packit Service fdd496
  int value;
Packit Service fdd496
};
Packit Service fdd496
static int
Packit Service fdd496
func1 (int a, ...)
Packit Service fdd496
{
Packit Service fdd496
  va_list arg;
Packit Service fdd496
  int i;
Packit Service fdd496
  va_start (arg, a);
Packit Service fdd496
  if (a < 4)
Packit Service fdd496
    i = va_arg (arg, int);
Packit Service fdd496
  else
Packit Service fdd496
    {
Packit Service fdd496
      struct dummy_struct *s = va_arg (arg, struct dummy_struct *);
Packit Service fdd496
      i = s->value;
Packit Service fdd496
    }
Packit Service fdd496
  va_end (arg);
Packit Service fdd496
  return i;
Packit Service fdd496
}
Packit Service fdd496
static int
Packit Service fdd496
func2 (int a, ...)
Packit Service fdd496
{
Packit Service fdd496
  va_list arg;
Packit Service fdd496
  void *p;
Packit Service fdd496
  va_start (arg, a);
Packit Service fdd496
  p = va_arg (arg, void *);
Packit Service fdd496
  va_end (arg);
Packit Service fdd496
  return func1 (a, p);
Packit Service fdd496
}
Packit Service fdd496
Packit Service fdd496
/* Ensure that all supported fcntl actions are distinct, and
Packit Service fdd496
   usable in preprocessor expressions.  */
Packit Service fdd496
static void
Packit Service fdd496
check_flags (void)
Packit Service fdd496
{
Packit Service fdd496
  switch (0)
Packit Service fdd496
    {
Packit Service fdd496
    case F_DUPFD:
Packit Service fdd496
#if F_DUPFD
Packit Service fdd496
#endif
Packit Service fdd496
Packit Service fdd496
    case F_DUPFD_CLOEXEC:
Packit Service fdd496
#if F_DUPFD_CLOEXEC
Packit Service fdd496
#endif
Packit Service fdd496
Packit Service fdd496
    case F_GETFD:
Packit Service fdd496
#if F_GETFD
Packit Service fdd496
#endif
Packit Service fdd496
Packit Service fdd496
#ifdef F_SETFD
Packit Service fdd496
    case F_SETFD:
Packit Service fdd496
# if F_SETFD
Packit Service fdd496
# endif
Packit Service fdd496
#endif
Packit Service fdd496
Packit Service fdd496
#ifdef F_GETFL
Packit Service fdd496
    case F_GETFL:
Packit Service fdd496
# if F_GETFL
Packit Service fdd496
# endif
Packit Service fdd496
#endif
Packit Service fdd496
Packit Service fdd496
#ifdef F_SETFL
Packit Service fdd496
    case F_SETFL:
Packit Service fdd496
# if F_SETFL
Packit Service fdd496
# endif
Packit Service fdd496
#endif
Packit Service fdd496
Packit Service fdd496
#ifdef F_GETOWN
Packit Service fdd496
    case F_GETOWN:
Packit Service fdd496
# if F_GETOWN
Packit Service fdd496
# endif
Packit Service fdd496
#endif
Packit Service fdd496
Packit Service fdd496
#ifdef F_SETOWN
Packit Service fdd496
    case F_SETOWN:
Packit Service fdd496
# if F_SETOWN
Packit Service fdd496
# endif
Packit Service fdd496
#endif
Packit Service fdd496
Packit Service fdd496
#ifdef F_GETLK
Packit Service fdd496
    case F_GETLK:
Packit Service fdd496
# if F_GETLK
Packit Service fdd496
# endif
Packit Service fdd496
#endif
Packit Service fdd496
Packit Service fdd496
#ifdef F_SETLK
Packit Service fdd496
    case F_SETLK:
Packit Service fdd496
# if F_SETLK
Packit Service fdd496
# endif
Packit Service fdd496
#endif
Packit Service fdd496
Packit Service fdd496
#ifdef F_SETLKW
Packit Service fdd496
    case F_SETLKW:
Packit Service fdd496
# if F_SETLKW
Packit Service fdd496
# endif
Packit Service fdd496
#endif
Packit Service fdd496
Packit Service fdd496
      ;
Packit Service fdd496
    }
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-fcntl.tmp";
Packit Service fdd496
  int fd;
Packit Service fdd496
  int bad_fd = getdtablesize ();
Packit Service fdd496
Packit Service fdd496
  /* Sanity check that rpl_fcntl is likely to work.  */
Packit Service fdd496
  ASSERT (func2 (1, 2) == 2);
Packit Service fdd496
  ASSERT (func2 (2, -2) == -2);
Packit Service fdd496
  ASSERT (func2 (3, 0x80000000) == 0x80000000);
Packit Service fdd496
  {
Packit Service fdd496
    struct dummy_struct s = { 0L, 4 };
Packit Service fdd496
    ASSERT (func2 (4, &s) == 4);
Packit Service fdd496
  }
Packit Service fdd496
  check_flags ();
Packit Service fdd496
Packit Service fdd496
  /* Assume std descriptors were provided by invoker, and ignore fds
Packit Service fdd496
     that might have been inherited.  */
Packit Service fdd496
  fd = creat (file, 0600);
Packit Service fdd496
  ASSERT (STDERR_FILENO < fd);
Packit Service fdd496
  close (fd + 1);
Packit Service fdd496
  close (fd + 2);
Packit Service fdd496
Packit Service fdd496
  /* For F_DUPFD*, the source must be valid.  */
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  ASSERT (fcntl (-1, F_DUPFD, 0) == -1);
Packit Service fdd496
  ASSERT (errno == EBADF);
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  ASSERT (fcntl (fd + 1, F_DUPFD, 0) == -1);
Packit Service fdd496
  ASSERT (errno == EBADF);
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  ASSERT (fcntl (bad_fd, F_DUPFD, 0) == -1);
Packit Service fdd496
  ASSERT (errno == EBADF);
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  ASSERT (fcntl (-1, F_DUPFD_CLOEXEC, 0) == -1);
Packit Service fdd496
  ASSERT (errno == EBADF);
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  ASSERT (fcntl (fd + 1, F_DUPFD_CLOEXEC, 0) == -1);
Packit Service fdd496
  ASSERT (errno == EBADF);
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  ASSERT (fcntl (bad_fd, F_DUPFD_CLOEXEC, 0) == -1);
Packit Service fdd496
  ASSERT (errno == EBADF);
Packit Service fdd496
Packit Service fdd496
  /* For F_DUPFD*, the destination must be valid.  */
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  ASSERT (fcntl (fd, F_DUPFD, -1) == -1);
Packit Service fdd496
  ASSERT (errno == EINVAL);
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  ASSERT (fcntl (fd, F_DUPFD, bad_fd) == -1);
Packit Service fdd496
  ASSERT (errno == EINVAL);
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  ASSERT (fcntl (fd, F_DUPFD_CLOEXEC, -1) == -1);
Packit Service fdd496
  ASSERT (errno == EINVAL);
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  ASSERT (fcntl (fd, F_DUPFD_CLOEXEC, bad_fd) == -1);
Packit Service fdd496
  ASSERT (errno == EINVAL);
Packit Service fdd496
Packit Service fdd496
  /* For F_DUPFD*, check for correct inheritance, as well as
Packit Service fdd496
     preservation of text vs. binary.  */
Packit Service fdd496
  setmode (fd, O_BINARY);
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
  ASSERT (is_inheritable (fd));
Packit Service fdd496
  ASSERT (is_mode (fd, O_BINARY));
Packit Service fdd496
Packit Service fdd496
  ASSERT (fcntl (fd, F_DUPFD, fd) == fd + 1);
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
  ASSERT (is_inheritable (fd + 1));
Packit Service fdd496
  ASSERT (is_mode (fd, O_BINARY));
Packit Service fdd496
  ASSERT (is_mode (fd + 1, O_BINARY));
Packit Service fdd496
  ASSERT (close (fd + 1) == 0);
Packit Service fdd496
Packit Service fdd496
  ASSERT (fcntl (fd, F_DUPFD_CLOEXEC, 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
  ASSERT (is_inheritable (fd));
Packit Service fdd496
  ASSERT (!is_inheritable (fd + 2));
Packit Service fdd496
  ASSERT (is_mode (fd, O_BINARY));
Packit Service fdd496
  ASSERT (is_mode (fd + 2, O_BINARY));
Packit Service fdd496
  ASSERT (close (fd) == 0);
Packit Service fdd496
Packit Service fdd496
  setmode (fd + 2, O_TEXT);
Packit Service fdd496
  ASSERT (fcntl (fd + 2, F_DUPFD, fd + 1) == fd + 1);
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
  ASSERT (is_inheritable (fd + 1));
Packit Service fdd496
  ASSERT (!is_inheritable (fd + 2));
Packit Service fdd496
  ASSERT (is_mode (fd + 1, O_TEXT));
Packit Service fdd496
  ASSERT (is_mode (fd + 2, O_TEXT));
Packit Service fdd496
  ASSERT (close (fd + 1) == 0);
Packit Service fdd496
Packit Service fdd496
  ASSERT (fcntl (fd + 2, F_DUPFD_CLOEXEC, 0) == fd);
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
  ASSERT (!is_inheritable (fd));
Packit Service fdd496
  ASSERT (!is_inheritable (fd + 2));
Packit Service fdd496
  ASSERT (is_mode (fd, O_TEXT));
Packit Service fdd496
  ASSERT (is_mode (fd + 2, O_TEXT));
Packit Service fdd496
  ASSERT (close (fd + 2) == 0);
Packit Service fdd496
Packit Service fdd496
  /* Test F_GETFD on invalid file descriptors.  */
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  ASSERT (fcntl (-1, F_GETFD) == -1);
Packit Service fdd496
  ASSERT (errno == EBADF);
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  ASSERT (fcntl (fd + 1, F_GETFD) == -1);
Packit Service fdd496
  ASSERT (errno == EBADF);
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  ASSERT (fcntl (bad_fd, F_GETFD) == -1);
Packit Service fdd496
  ASSERT (errno == EBADF);
Packit Service fdd496
Packit Service fdd496
  /* Test F_GETFD, the FD_CLOEXEC bit.  */
Packit Service fdd496
  {
Packit Service fdd496
    int result = fcntl (fd, F_GETFD);
Packit Service fdd496
    ASSERT (0 <= result);
Packit Service fdd496
    ASSERT ((result & FD_CLOEXEC) == FD_CLOEXEC);
Packit Service fdd496
    ASSERT (dup (fd) == fd + 1);
Packit Service fdd496
    result = fcntl (fd + 1, F_GETFD);
Packit Service fdd496
    ASSERT (0 <= result);
Packit Service fdd496
    ASSERT ((result & FD_CLOEXEC) == 0);
Packit Service fdd496
    ASSERT (close (fd + 1) == 0);
Packit Service fdd496
  }
Packit Service fdd496
Packit Service fdd496
#ifdef F_SETFD
Packit Service fdd496
  /* Test F_SETFD on invalid file descriptors.  */
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  ASSERT (fcntl (-1, F_SETFD, 0) == -1);
Packit Service fdd496
  ASSERT (errno == EBADF);
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  ASSERT (fcntl (fd + 1, F_SETFD, 0) == -1);
Packit Service fdd496
  ASSERT (errno == EBADF);
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  ASSERT (fcntl (bad_fd, F_SETFD, 0) == -1);
Packit Service fdd496
  ASSERT (errno == EBADF);
Packit Service fdd496
#endif
Packit Service fdd496
Packit Service fdd496
#ifdef F_GETFL
Packit Service fdd496
  /* Test F_GETFL on invalid file descriptors.  */
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  ASSERT (fcntl (-1, F_GETFL) == -1);
Packit Service fdd496
  ASSERT (errno == EBADF);
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  ASSERT (fcntl (fd + 1, F_GETFL) == -1);
Packit Service fdd496
  ASSERT (errno == EBADF);
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  ASSERT (fcntl (bad_fd, F_GETFL) == -1);
Packit Service fdd496
  ASSERT (errno == EBADF);
Packit Service fdd496
#endif
Packit Service fdd496
Packit Service fdd496
#ifdef F_SETFL
Packit Service fdd496
  /* Test F_SETFL on invalid file descriptors.  */
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  ASSERT (fcntl (-1, F_SETFL, 0) == -1);
Packit Service fdd496
  ASSERT (errno == EBADF);
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  ASSERT (fcntl (fd + 1, F_SETFL, 0) == -1);
Packit Service fdd496
  ASSERT (errno == EBADF);
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  ASSERT (fcntl (bad_fd, F_SETFL, 0) == -1);
Packit Service fdd496
  ASSERT (errno == EBADF);
Packit Service fdd496
#endif
Packit Service fdd496
Packit Service fdd496
#ifdef F_GETOWN
Packit Service fdd496
  /* Test F_GETOWN on invalid file descriptors.  */
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  ASSERT (fcntl (-1, F_GETOWN) == -1);
Packit Service fdd496
  ASSERT (errno == EBADF);
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  ASSERT (fcntl (fd + 1, F_GETOWN) == -1);
Packit Service fdd496
  ASSERT (errno == EBADF);
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  ASSERT (fcntl (bad_fd, F_GETOWN) == -1);
Packit Service fdd496
  ASSERT (errno == EBADF);
Packit Service fdd496
#endif
Packit Service fdd496
Packit Service fdd496
#ifdef F_SETOWN
Packit Service fdd496
  /* Test F_SETFL on invalid file descriptors.  */
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  ASSERT (fcntl (-1, F_SETOWN, 0) == -1);
Packit Service fdd496
  ASSERT (errno == EBADF);
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  ASSERT (fcntl (fd + 1, F_SETOWN, 0) == -1);
Packit Service fdd496
  ASSERT (errno == EBADF);
Packit Service fdd496
  errno = 0;
Packit Service fdd496
  ASSERT (fcntl (bad_fd, F_SETOWN, 0) == -1);
Packit Service fdd496
  ASSERT (errno == EBADF);
Packit Service fdd496
#endif
Packit Service fdd496
Packit Service fdd496
  /* Cleanup.  */
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
}