access tests: Fix test failure on native Windows.
[gnulib.git] / m4 / fcntl.m4
blob547ff40b804f0e9f936a20aa94b7f30b359dc316
1 # fcntl.m4 serial 11
2 dnl Copyright (C) 2009-2020 Free Software Foundation, Inc.
3 dnl This file is free software; the Free Software Foundation
4 dnl gives unlimited permission to copy and/or distribute it,
5 dnl with or without modifications, as long as this notice is preserved.
7 # For now, this module ensures that fcntl()
8 # - supports F_DUPFD correctly
9 # - supports or emulates F_DUPFD_CLOEXEC
10 # - supports F_GETFD
11 # Still to be ported to mingw:
12 # - F_SETFD
13 # - F_GETFL, F_SETFL
14 # - F_GETOWN, F_SETOWN
15 # - F_GETLK, F_SETLK, F_SETLKW
16 AC_DEFUN([gl_FUNC_FCNTL],
18   dnl Persuade glibc to expose F_DUPFD_CLOEXEC.
19   AC_REQUIRE([gl_USE_SYSTEM_EXTENSIONS])
20   AC_REQUIRE([gl_FCNTL_H_DEFAULTS])
21   AC_REQUIRE([AC_CANONICAL_HOST])
22   AC_CHECK_FUNCS_ONCE([fcntl])
23   if test $ac_cv_func_fcntl = no; then
24     gl_REPLACE_FCNTL
25   else
26     dnl cygwin 1.5.x F_DUPFD has wrong errno, and allows negative target
27     dnl haiku alpha 2 F_DUPFD has wrong errno
28     AC_CACHE_CHECK([whether fcntl handles F_DUPFD correctly],
29       [gl_cv_func_fcntl_f_dupfd_works],
30       [AC_RUN_IFELSE(
31          [AC_LANG_PROGRAM(
32             [[#include <errno.h>
33               #include <fcntl.h>
34               #include <limits.h>
35               #include <sys/resource.h>
36               #include <unistd.h>
37               ]GL_MDA_DEFINES[
38               #ifndef RLIM_SAVED_CUR
39               # define RLIM_SAVED_CUR RLIM_INFINITY
40               #endif
41               #ifndef RLIM_SAVED_MAX
42               # define RLIM_SAVED_MAX RLIM_INFINITY
43               #endif
44             ]],
45             [[int result = 0;
46               int bad_fd = INT_MAX;
47               struct rlimit rlim;
48               if (getrlimit (RLIMIT_NOFILE, &rlim) == 0
49                   && 0 <= rlim.rlim_cur && rlim.rlim_cur <= INT_MAX
50                   && rlim.rlim_cur != RLIM_INFINITY
51                   && rlim.rlim_cur != RLIM_SAVED_MAX
52                   && rlim.rlim_cur != RLIM_SAVED_CUR)
53                 bad_fd = rlim.rlim_cur;
54               if (fcntl (0, F_DUPFD, -1) != -1) result |= 1;
55               if (errno != EINVAL) result |= 2;
56               if (fcntl (0, F_DUPFD, bad_fd) != -1) result |= 4;
57               if (errno != EINVAL) result |= 8;
58               /* On OS/2 kLIBC, F_DUPFD does not work on a directory fd */
59               {
60                 int fd;
61                 fd = open (".", O_RDONLY);
62                 if (fd == -1)
63                   result |= 16;
64                 else if (fcntl (fd, F_DUPFD, STDERR_FILENO + 1) == -1)
65                   result |= 32;
67                 close (fd);
68               }
69               return result;]])],
70          [gl_cv_func_fcntl_f_dupfd_works=yes],
71          [gl_cv_func_fcntl_f_dupfd_works=no],
72          [case $host_os in
73             aix* | cygwin* | haiku*)
74                gl_cv_func_fcntl_f_dupfd_works="guessing no" ;;
75             *) gl_cv_func_fcntl_f_dupfd_works="guessing yes" ;;
76           esac])])
77     case $gl_cv_func_fcntl_f_dupfd_works in
78       *yes) ;;
79       *) gl_REPLACE_FCNTL
80         AC_DEFINE([FCNTL_DUPFD_BUGGY], [1], [Define this to 1 if F_DUPFD
81           behavior does not match POSIX]) ;;
82     esac
84     dnl Many systems lack F_DUPFD_CLOEXEC.
85     dnl NetBSD 9.0 declares F_DUPFD_CLOEXEC but it works only like F_DUPFD.
86     AC_CACHE_CHECK([whether fcntl understands F_DUPFD_CLOEXEC],
87       [gl_cv_func_fcntl_f_dupfd_cloexec],
88       [AC_RUN_IFELSE(
89          [AC_LANG_SOURCE(
90             [[#include <fcntl.h>
91               #include <unistd.h>
92               int main (int argc, char *argv[])
93               {
94                 if (argc == 1)
95                   /* parent process */
96                   {
97                     if (fcntl (1, F_DUPFD_CLOEXEC, 10) < 0)
98                       return 1;
99                     return execl ("./conftest", "./conftest", "child", NULL);
100                   }
101                 else
102                   /* child process */
103                   return (fcntl (10, F_GETFL) < 0 ? 0 : 42);
104               }
105             ]])
106          ],
107          [AC_COMPILE_IFELSE([AC_LANG_PROGRAM([[
108 #ifdef __linux__
109 /* The Linux kernel only added F_DUPFD_CLOEXEC in 2.6.24, so we always replace
110    it to support the semantics on older kernels that failed with EINVAL.  */
111 choke me
112 #endif
113            ]])],
114            [gl_cv_func_fcntl_f_dupfd_cloexec=yes],
115            [gl_cv_func_fcntl_f_dupfd_cloexec="needs runtime check"])
116          ],
117          [gl_cv_func_fcntl_f_dupfd_cloexec=no],
118          [case "$host_os" in
119                      # Guess no on NetBSD.
120             netbsd*) gl_cv_func_fcntl_f_dupfd_cloexec="guessing no" ;;
121             *)       gl_cv_func_fcntl_f_dupfd_cloexec="$gl_cross_guess_normal" ;;
122           esac
123          ])
124       ])
125     case "$gl_cv_func_fcntl_f_dupfd_cloexec" in
126       *yes) ;;
127       *)    gl_REPLACE_FCNTL
128             dnl No witness macro needed for this bug.
129             ;;
130     esac
131   fi
132   dnl Replace fcntl() for supporting the gnulib-defined fchdir() function,
133   dnl to keep fchdir's bookkeeping up-to-date.
134   m4_ifdef([gl_FUNC_FCHDIR], [
135     gl_TEST_FCHDIR
136     if test $HAVE_FCHDIR = 0; then
137       gl_REPLACE_FCNTL
138     fi
139   ])
142 AC_DEFUN([gl_REPLACE_FCNTL],
144   AC_REQUIRE([gl_FCNTL_H_DEFAULTS])
145   AC_CHECK_FUNCS_ONCE([fcntl])
146   if test $ac_cv_func_fcntl = no; then
147     HAVE_FCNTL=0
148   else
149     REPLACE_FCNTL=1
150   fi