dup2.m4 4.09 KB
Newer Older
Paul Eggert's avatar
Paul Eggert committed
1
#serial 25
Paul Eggert's avatar
Paul Eggert committed
2
dnl Copyright (C) 2002, 2005, 2007, 2009-2019 Free Software Foundation, Inc.
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
dnl This file is free software; the Free Software Foundation
dnl gives unlimited permission to copy and/or distribute it,
dnl with or without modifications, as long as this notice is preserved.

AC_DEFUN([gl_FUNC_DUP2],
[
  AC_REQUIRE([gl_UNISTD_H_DEFAULTS])
  AC_REQUIRE([AC_CANONICAL_HOST])
  m4_ifdef([gl_FUNC_DUP2_OBSOLETE], [
    AC_CHECK_FUNCS_ONCE([dup2])
    if test $ac_cv_func_dup2 = no; then
      HAVE_DUP2=0
    fi
  ], [
    AC_DEFINE([HAVE_DUP2], [1], [Define to 1 if you have the 'dup2' function.])
  ])
  if test $HAVE_DUP2 = 1; then
    AC_CACHE_CHECK([whether dup2 works], [gl_cv_func_dup2_works],
      [AC_RUN_IFELSE([
Paul Eggert's avatar
Paul Eggert committed
22 23 24 25 26 27
         AC_LANG_PROGRAM(
           [[#include <errno.h>
             #include <fcntl.h>
             #include <limits.h>
             #include <sys/resource.h>
             #include <unistd.h>
Paul Eggert's avatar
Paul Eggert committed
28 29 30 31 32 33
             #ifndef RLIM_SAVED_CUR
             # define RLIM_SAVED_CUR RLIM_INFINITY
             #endif
             #ifndef RLIM_SAVED_MAX
             # define RLIM_SAVED_MAX RLIM_INFINITY
             #endif
Paul Eggert's avatar
Paul Eggert committed
34 35 36 37 38 39 40 41 42 43 44 45 46 47
           ]],
           [[int result = 0;
             int bad_fd = INT_MAX;
             struct rlimit rlim;
             if (getrlimit (RLIMIT_NOFILE, &rlim) == 0
                 && 0 <= rlim.rlim_cur && rlim.rlim_cur <= INT_MAX
                 && rlim.rlim_cur != RLIM_INFINITY
                 && rlim.rlim_cur != RLIM_SAVED_MAX
                 && rlim.rlim_cur != RLIM_SAVED_CUR)
               bad_fd = rlim.rlim_cur;
             #ifdef FD_CLOEXEC
               if (fcntl (1, F_SETFD, FD_CLOEXEC) == -1)
                 result |= 1;
             #endif
Paul Eggert's avatar
Paul Eggert committed
48
             if (dup2 (1, 1) != 1)
Paul Eggert's avatar
Paul Eggert committed
49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64
               result |= 2;
             #ifdef FD_CLOEXEC
               if (fcntl (1, F_GETFD) != FD_CLOEXEC)
                 result |= 4;
             #endif
             close (0);
             if (dup2 (0, 0) != -1)
               result |= 8;
             /* Many gnulib modules require POSIX conformance of EBADF.  */
             if (dup2 (2, bad_fd) == -1 && errno != EBADF)
               result |= 16;
             /* Flush out some cygwin core dumps.  */
             if (dup2 (2, -1) != -1 || errno != EBADF)
               result |= 32;
             dup2 (2, 255);
             dup2 (2, 256);
Paul Eggert's avatar
Paul Eggert committed
65 66 67 68 69 70 71 72 73 74
             /* On OS/2 kLIBC, dup2() does not work on a directory fd.  */
             {
               int fd = open (".", O_RDONLY);
               if (fd == -1)
                 result |= 64;
               else if (dup2 (fd, fd + 1) == -1)
                 result |= 128;

               close (fd);
             }
Paul Eggert's avatar
Paul Eggert committed
75
             return result;]])
76 77 78 79
        ],
        [gl_cv_func_dup2_works=yes], [gl_cv_func_dup2_works=no],
        [case "$host_os" in
           mingw*) # on this platform, dup2 always returns 0 for success
Paul Eggert's avatar
Paul Eggert committed
80
             gl_cv_func_dup2_works="guessing no" ;;
81
           cygwin*) # on cygwin 1.5.x, dup2(1,1) returns 0
Paul Eggert's avatar
Paul Eggert committed
82
             gl_cv_func_dup2_works="guessing no" ;;
Paul Eggert's avatar
Paul Eggert committed
83 84 85
           aix* | freebsd*)
                   # on AIX 7.1 and FreeBSD 6.1, dup2 (1,toobig) gives EMFILE,
                   # not EBADF.
Paul Eggert's avatar
Paul Eggert committed
86
             gl_cv_func_dup2_works="guessing no" ;;
87
           haiku*) # on Haiku alpha 2, dup2(1, 1) resets FD_CLOEXEC.
Paul Eggert's avatar
Paul Eggert committed
88
             gl_cv_func_dup2_works="guessing no" ;;
Paul Eggert's avatar
Paul Eggert committed
89 90
           *-android*) # implemented using dup3(), which fails if oldfd == newfd
             gl_cv_func_dup2_works="guessing no" ;;
Paul Eggert's avatar
Paul Eggert committed
91 92
           os2*) # on OS/2 kLIBC, dup2() does not work on a directory fd.
             gl_cv_func_dup2_works="guessing no" ;;
Paul Eggert's avatar
Paul Eggert committed
93
           *) gl_cv_func_dup2_works="guessing yes" ;;
94 95
         esac])
      ])
Paul Eggert's avatar
Paul Eggert committed
96 97 98 99
    case "$gl_cv_func_dup2_works" in
      *yes) ;;
      *)
        REPLACE_DUP2=1
Paul Eggert's avatar
Paul Eggert committed
100
        AC_CHECK_FUNCS([setdtablesize])
Paul Eggert's avatar
Paul Eggert committed
101 102
        ;;
    esac
103
  fi
Paul Eggert's avatar
Paul Eggert committed
104 105 106 107 108
  dnl Replace dup2() for supporting the gnulib-defined fchdir() function,
  dnl to keep fchdir's bookkeeping up-to-date.
  m4_ifdef([gl_FUNC_FCHDIR], [
    gl_TEST_FCHDIR
    if test $HAVE_FCHDIR = 0; then
Paul Eggert's avatar
Paul Eggert committed
109 110 111
      if test $HAVE_DUP2 = 1; then
        REPLACE_DUP2=1
      fi
Paul Eggert's avatar
Paul Eggert committed
112 113
    fi
  ])
114
])
115 116 117

# Prerequisites of lib/dup2.c.
AC_DEFUN([gl_PREREQ_DUP2], [])