1# fcntl.m4 serial 10 2dnl Copyright (C) 2009-2020 Free Software Foundation, Inc. 3dnl This file is free software; the Free Software Foundation 4dnl gives unlimited permission to copy and/or distribute it, 5dnl with or without modifications, as long as this notice is preserved. 6 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 16AC_DEFUN([gl_FUNC_FCNTL], 17[ 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; 66 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 83 84 dnl Many systems lack F_DUPFD_CLOEXEC 85 AC_CACHE_CHECK([whether fcntl understands F_DUPFD_CLOEXEC], 86 [gl_cv_func_fcntl_f_dupfd_cloexec], 87 [AC_COMPILE_IFELSE([AC_LANG_PROGRAM([[ 88#include <fcntl.h> 89#ifndef F_DUPFD_CLOEXEC 90choke me 91#endif 92 ]])], 93 [AC_COMPILE_IFELSE([AC_LANG_PROGRAM([[ 94#ifdef __linux__ 95/* The Linux kernel only added F_DUPFD_CLOEXEC in 2.6.24, so we always replace 96 it to support the semantics on older kernels that failed with EINVAL. */ 97choke me 98#endif 99 ]])], 100 [gl_cv_func_fcntl_f_dupfd_cloexec=yes], 101 [gl_cv_func_fcntl_f_dupfd_cloexec="needs runtime check"])], 102 [gl_cv_func_fcntl_f_dupfd_cloexec=no])]) 103 if test "$gl_cv_func_fcntl_f_dupfd_cloexec" != yes; then 104 gl_REPLACE_FCNTL 105 dnl No witness macro needed for this bug. 106 fi 107 fi 108 dnl Replace fcntl() for supporting the gnulib-defined fchdir() function, 109 dnl to keep fchdir's bookkeeping up-to-date. 110 m4_ifdef([gl_FUNC_FCHDIR], [ 111 gl_TEST_FCHDIR 112 if test $HAVE_FCHDIR = 0; then 113 gl_REPLACE_FCNTL 114 fi 115 ]) 116]) 117 118AC_DEFUN([gl_REPLACE_FCNTL], 119[ 120 AC_REQUIRE([gl_FCNTL_H_DEFAULTS]) 121 AC_CHECK_FUNCS_ONCE([fcntl]) 122 if test $ac_cv_func_fcntl = no; then 123 HAVE_FCNTL=0 124 else 125 REPLACE_FCNTL=1 126 fi 127]) 128