fcntl: work around Haiku F_DUPFD bugs
authorEric Blake <eblake@redhat.com>
Wed, 26 Jan 2011 05:06:16 +0000 (22:06 -0700)
committerEric Blake <eblake@redhat.com>
Wed, 26 Jan 2011 17:05:43 +0000 (10:05 -0700)
* m4/fcntl.m4 (gl_FUNC_FCNTL): Also catch Haiku errno bug.
* lib/fcntl.c (rpl_fcntl) [F_DUPFD]: Work around Haiku losing
cloexec bit on duplication.
* doc/posix-functions/fcntl.texi (fcntl): Document the bug.

Signed-off-by: Eric Blake <eblake@redhat.com>
ChangeLog
doc/posix-functions/fcntl.texi
lib/fcntl.c
m4/fcntl.m4

index c6700ec..4528062 100644 (file)
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,3 +1,11 @@
+2011-01-26  Eric Blake  <eblake@redhat.com>
+
+       fcntl: work around Haiku F_DUPFD bugs
+       * m4/fcntl.m4 (gl_FUNC_FCNTL): Also catch Haiku errno bug.
+       * lib/fcntl.c (rpl_fcntl) [F_DUPFD]: Work around Haiku losing
+       cloexec bit on duplication.
+       * doc/posix-functions/fcntl.texi (fcntl): Document the bug.
+
 2011-01-26  Bruno Haible  <bruno@clisp.org>
 
        Enable memory leak tests on AIX.
index a0511ec..161af1e 100644 (file)
@@ -19,7 +19,12 @@ Note that the gnulib replacement code is functional but not atomic.
 @item
 The @code{F_DUPFD} action of this function does not reject
 out-of-range targets properly on some platforms:
-Cygwin 1.5.x.
+Cygwin 1.5.x, Haiku.
+
+@item
+The @code{F_DUPFD} action of this function mistakenly clears
+FD_CLOEXEC on the source descriptor on some platforms:
+Haiku.
 
 @item
 This function is missing on some platforms:
index 25a79c5..d6a328c 100644 (file)
@@ -187,7 +187,21 @@ rpl_fcntl (int fd, int action, /* arg */...)
           errno = EINVAL;
         else
           {
+            /* Haiku alpha 2 loses fd flags on original.  */
+            int flags = fcntl (fd, F_GETFD);
+            if (flags < 0)
+              {
+                result = -1;
+                break;
+              }
             result = fcntl (fd, action, target);
+            if (0 <= result && fcntl (fd, F_SETFD, flags) == -1)
+              {
+                int saved_errno = errno;
+                close (result);
+                result = -1;
+                errno = saved_errno;
+              }
 # if REPLACE_FCHDIR
             if (0 <= result)
               result = _gl_register_dup (fd, result);
index 3a86f2c..a93ed85 100644 (file)
@@ -1,4 +1,4 @@
-# fcntl.m4 serial 3
+# fcntl.m4 serial 4
 dnl Copyright (C) 2009-2011 Free Software Foundation, Inc.
 dnl This file is free software; the Free Software Foundation
 dnl gives unlimited permission to copy and/or distribute it,
@@ -24,11 +24,16 @@ AC_DEFUN([gl_FUNC_FCNTL],
     gl_REPLACE_FCNTL
   else
     dnl cygwin 1.5.x F_DUPFD has wrong errno, and allows negative target
+    dnl haiku alpha 2 F_DUPFD has wrong errno
     AC_CACHE_CHECK([whether fcntl handles F_DUPFD correctly],
       [gl_cv_func_fcntl_f_dupfd_works],
       [AC_RUN_IFELSE([AC_LANG_PROGRAM([[
 #include <fcntl.h>
-]], [[return fcntl (0, F_DUPFD, -1) != -1;
+#include <errno.h>
+]], [[int result = 0;
+      if (fcntl (0, F_DUPFD, -1) != -1) result |= 1;
+      if (errno != EINVAL) result |= 2;
+      return result;
          ]])],
          [gl_cv_func_fcntl_f_dupfd_works=yes],
          [gl_cv_func_fcntl_f_dupfd_works=no],