Blame m4/fcntl.m4

Packit 33f14e
# fcntl.m4 serial 9
Packit 33f14e
dnl Copyright (C) 2009-2017 Free Software Foundation, Inc.
Packit 33f14e
dnl This file is free software; the Free Software Foundation
Packit 33f14e
dnl gives unlimited permission to copy and/or distribute it,
Packit 33f14e
dnl with or without modifications, as long as this notice is preserved.
Packit 33f14e
Packit 33f14e
# For now, this module ensures that fcntl()
Packit 33f14e
# - supports F_DUPFD correctly
Packit 33f14e
# - supports or emulates F_DUPFD_CLOEXEC
Packit 33f14e
# - supports F_GETFD
Packit 33f14e
# Still to be ported to mingw:
Packit 33f14e
# - F_SETFD
Packit 33f14e
# - F_GETFL, F_SETFL
Packit 33f14e
# - F_GETOWN, F_SETOWN
Packit 33f14e
# - F_GETLK, F_SETLK, F_SETLKW
Packit 33f14e
AC_DEFUN([gl_FUNC_FCNTL],
Packit 33f14e
[
Packit 33f14e
  dnl Persuade glibc to expose F_DUPFD_CLOEXEC.
Packit 33f14e
  AC_REQUIRE([gl_USE_SYSTEM_EXTENSIONS])
Packit 33f14e
  AC_REQUIRE([gl_FCNTL_H_DEFAULTS])
Packit 33f14e
  AC_REQUIRE([AC_CANONICAL_HOST])
Packit 33f14e
  AC_CHECK_FUNCS_ONCE([fcntl])
Packit 33f14e
  if test $ac_cv_func_fcntl = no; then
Packit 33f14e
    gl_REPLACE_FCNTL
Packit 33f14e
  else
Packit 33f14e
    dnl cygwin 1.5.x F_DUPFD has wrong errno, and allows negative target
Packit 33f14e
    dnl haiku alpha 2 F_DUPFD has wrong errno
Packit 33f14e
    AC_CACHE_CHECK([whether fcntl handles F_DUPFD correctly],
Packit 33f14e
      [gl_cv_func_fcntl_f_dupfd_works],
Packit 33f14e
      [AC_RUN_IFELSE(
Packit 33f14e
         [AC_LANG_PROGRAM(
Packit 33f14e
            [[#include <errno.h>
Packit 33f14e
              #include <fcntl.h>
Packit 33f14e
              #include <limits.h>
Packit 33f14e
              #include <sys/resource.h>
Packit 33f14e
              #include <unistd.h>
Packit 33f14e
              #ifndef RLIM_SAVED_CUR
Packit 33f14e
              # define RLIM_SAVED_CUR RLIM_INFINITY
Packit 33f14e
              #endif
Packit 33f14e
              #ifndef RLIM_SAVED_MAX
Packit 33f14e
              # define RLIM_SAVED_MAX RLIM_INFINITY
Packit 33f14e
              #endif
Packit 33f14e
            ]],
Packit 33f14e
            [[int result = 0;
Packit 33f14e
              int bad_fd = INT_MAX;
Packit 33f14e
              struct rlimit rlim;
Packit 33f14e
              if (getrlimit (RLIMIT_NOFILE, &rlim) == 0
Packit 33f14e
                  && 0 <= rlim.rlim_cur && rlim.rlim_cur <= INT_MAX
Packit 33f14e
                  && rlim.rlim_cur != RLIM_INFINITY
Packit 33f14e
                  && rlim.rlim_cur != RLIM_SAVED_MAX
Packit 33f14e
                  && rlim.rlim_cur != RLIM_SAVED_CUR)
Packit 33f14e
                bad_fd = rlim.rlim_cur;
Packit 33f14e
              if (fcntl (0, F_DUPFD, -1) != -1) result |= 1;
Packit 33f14e
              if (errno != EINVAL) result |= 2;
Packit 33f14e
              if (fcntl (0, F_DUPFD, bad_fd) != -1) result |= 4;
Packit 33f14e
              if (errno != EINVAL) result |= 8;
Packit 33f14e
              /* On OS/2 kLIBC, F_DUPFD does not work on a directory fd */
Packit 33f14e
              {
Packit 33f14e
                int fd;
Packit 33f14e
                fd = open (".", O_RDONLY);
Packit 33f14e
                if (fd == -1)
Packit 33f14e
                  result |= 16;
Packit 33f14e
                else if (fcntl (fd, F_DUPFD, STDERR_FILENO + 1) == -1)
Packit 33f14e
                  result |= 32;
Packit 33f14e
Packit 33f14e
                close (fd);
Packit 33f14e
              }
Packit 33f14e
              return result;]])],
Packit 33f14e
         [gl_cv_func_fcntl_f_dupfd_works=yes],
Packit 33f14e
         [gl_cv_func_fcntl_f_dupfd_works=no],
Packit 33f14e
         [case $host_os in
Packit 33f14e
            aix* | cygwin* | haiku*)
Packit 33f14e
               gl_cv_func_fcntl_f_dupfd_works="guessing no" ;;
Packit 33f14e
            *) gl_cv_func_fcntl_f_dupfd_works="guessing yes" ;;
Packit 33f14e
          esac])])
Packit 33f14e
    case $gl_cv_func_fcntl_f_dupfd_works in
Packit 33f14e
      *yes) ;;
Packit 33f14e
      *) gl_REPLACE_FCNTL
Packit 33f14e
        AC_DEFINE([FCNTL_DUPFD_BUGGY], [1], [Define this to 1 if F_DUPFD
Packit 33f14e
          behavior does not match POSIX]) ;;
Packit 33f14e
    esac
Packit 33f14e
Packit 33f14e
    dnl Many systems lack F_DUPFD_CLOEXEC
Packit 33f14e
    AC_CACHE_CHECK([whether fcntl understands F_DUPFD_CLOEXEC],
Packit 33f14e
      [gl_cv_func_fcntl_f_dupfd_cloexec],
Packit 33f14e
      [AC_COMPILE_IFELSE([AC_LANG_PROGRAM([[
Packit 33f14e
#include <fcntl.h>
Packit 33f14e
#ifndef F_DUPFD_CLOEXEC
Packit 33f14e
choke me
Packit 33f14e
#endif
Packit 33f14e
         ]])],
Packit 33f14e
         [AC_COMPILE_IFELSE([AC_LANG_PROGRAM([[
Packit 33f14e
#ifdef __linux__
Packit 33f14e
/* The Linux kernel only added F_DUPFD_CLOEXEC in 2.6.24, so we always replace
Packit 33f14e
   it to support the semantics on older kernels that failed with EINVAL.  */
Packit 33f14e
choke me
Packit 33f14e
#endif
Packit 33f14e
           ]])],
Packit 33f14e
           [gl_cv_func_fcntl_f_dupfd_cloexec=yes],
Packit 33f14e
           [gl_cv_func_fcntl_f_dupfd_cloexec="needs runtime check"])],
Packit 33f14e
         [gl_cv_func_fcntl_f_dupfd_cloexec=no])])
Packit 33f14e
    if test "$gl_cv_func_fcntl_f_dupfd_cloexec" != yes; then
Packit 33f14e
      gl_REPLACE_FCNTL
Packit 33f14e
      dnl No witness macro needed for this bug.
Packit 33f14e
    fi
Packit 33f14e
  fi
Packit 33f14e
  dnl Replace fcntl() for supporting the gnulib-defined fchdir() function,
Packit 33f14e
  dnl to keep fchdir's bookkeeping up-to-date.
Packit 33f14e
  m4_ifdef([gl_FUNC_FCHDIR], [
Packit 33f14e
    gl_TEST_FCHDIR
Packit 33f14e
    if test $HAVE_FCHDIR = 0; then
Packit 33f14e
      gl_REPLACE_FCNTL
Packit 33f14e
    fi
Packit 33f14e
  ])
Packit 33f14e
])
Packit 33f14e
Packit 33f14e
AC_DEFUN([gl_REPLACE_FCNTL],
Packit 33f14e
[
Packit 33f14e
  AC_REQUIRE([gl_FCNTL_H_DEFAULTS])
Packit 33f14e
  AC_CHECK_FUNCS_ONCE([fcntl])
Packit 33f14e
  if test $ac_cv_func_fcntl = no; then
Packit 33f14e
    HAVE_FCNTL=0
Packit 33f14e
  else
Packit 33f14e
    REPLACE_FCNTL=1
Packit 33f14e
  fi
Packit 33f14e
])