From 872abb203f6a4ddde61e37912d97e622b89a2f7a Mon Sep 17 00:00:00 2001 From: Bruno Haible Date: Sat, 1 Jan 2011 02:25:00 +0100 Subject: [PATCH] pwrite: Work around HP-UX 11.11 bug. * m4/pwrite.m4 (gl_FUNC_PWRITE): When pwrite exists, test whether it works and set REPLACE_PWRITE if not. * lib/pwrite.c (pwrite): Add an implementation that uses the system function. * doc/posix-functions/pwrite.texi: Document the HP-UX 11 bug. --- ChangeLog | 9 +++++++ doc/posix-functions/pwrite.texi | 4 ++++ lib/pwrite.c | 24 ++++++++++++++++--- m4/pwrite.m4 | 53 +++++++++++++++++++++++++++++++++++++++-- 4 files changed, 85 insertions(+), 5 deletions(-) diff --git a/ChangeLog b/ChangeLog index 6c7d45300..f3dcc8f5b 100644 --- a/ChangeLog +++ b/ChangeLog @@ -1,5 +1,14 @@ 2010-12-31 Bruno Haible + pwrite: Work around HP-UX 11.11 bug. + * m4/pwrite.m4 (gl_FUNC_PWRITE): When pwrite exists, test whether it + works and set REPLACE_PWRITE if not. + * lib/pwrite.c (pwrite): Add an implementation that uses the system + function. + * doc/posix-functions/pwrite.texi: Document the HP-UX 11 bug. + +2010-12-31 Bruno Haible + pread: Work around HP-UX 11 bugs. * m4/pread.m4 (gl_FUNC_PREAD): When pread exists, test whether it works and set REPLACE_PREAD if not. diff --git a/doc/posix-functions/pwrite.texi b/doc/posix-functions/pwrite.texi index 24fb4fb92..720482136 100644 --- a/doc/posix-functions/pwrite.texi +++ b/doc/posix-functions/pwrite.texi @@ -11,6 +11,10 @@ Portability problems fixed by Gnulib: @item This function is missing on some platforms: HP-UX 10, mingw, BeOS. +@item +This function does not fail when an invalid (negative) offset is passed when +large file support is enabled on some platforms: +HP-UX 11.11. @end itemize Portability problems not fixed by Gnulib: diff --git a/lib/pwrite.c b/lib/pwrite.c index 00714791c..4a913571d 100644 --- a/lib/pwrite.c +++ b/lib/pwrite.c @@ -24,9 +24,25 @@ #include -#define __libc_lseek(f,o,w) lseek (f, o, w) -#define __set_errno(Val) errno = (Val) -#define __libc_write(f,b,n) write (f, b, n) +#if HAVE_PWRITE + +ssize_t +pwrite (int fd, const void *buf, size_t nbyte, off_t offset) +# undef pwrite +{ + if (offset < 0) + { + errno = EINVAL; + return -1; + } + return pwrite (fd, buf, nbyte, offset); +} + +#else + +# define __libc_lseek(f,o,w) lseek (f, o, w) +# define __set_errno(Val) errno = (Val) +# define __libc_write(f,b,n) write (f, b, n) /* Note: This implementation of pwrite is not multithread-safe. */ @@ -62,3 +78,5 @@ pwrite (int fd, const void *buf, size_t nbyte, off_t offset) return result; } + +#endif diff --git a/m4/pwrite.m4 b/m4/pwrite.m4 index 90d8caca6..4814c5d10 100644 --- a/m4/pwrite.m4 +++ b/m4/pwrite.m4 @@ -1,4 +1,4 @@ -# pwrite.m4 serial 1 +# pwrite.m4 serial 2 dnl Copyright (C) 2010 Free Software Foundation, Inc. dnl This file is free software; the Free Software Foundation dnl gives unlimited permission to copy and/or distribute it, @@ -7,13 +7,62 @@ dnl with or without modifications, as long as this notice is preserved. AC_DEFUN([gl_FUNC_PWRITE], [ AC_REQUIRE([gl_UNISTD_H_DEFAULTS]) + AC_REQUIRE([AC_CANONICAL_HOST]) dnl for cross-compiles dnl Persuade glibc to declare pwrite(). AC_REQUIRE([AC_USE_SYSTEM_EXTENSIONS]) AC_CHECK_FUNCS_ONCE([pwrite]) - if test $ac_cv_func_pwrite = no; then + if test $ac_cv_func_pwrite = yes; then + dnl On HP-UX 11.11 with _FILE_OFFSET_BITS=64, pwrite() on a file does not + dnl fail when an invalid (negative) offset is passed. + AC_CACHE_CHECK([whether pwrite works], + [gl_cv_func_pwrite_works], + [ + dnl Initial guess, used when cross-compiling. +changequote(,)dnl + case "$host_os" in + # Guess no on HP-UX. + hpux*) gl_cv_func_pwrite_works="guessing no" ;; + # Guess yes otherwise. + *) gl_cv_func_pwrite_works="guessing yes" ;; + esac +changequote([,])dnl + gl_save_CPPFLAGS="$CPPFLAGS" + CPPFLAGS="$CPPFLAGS -D_FILE_OFFSET_BITS=64" + rm -f conftest.out + AC_RUN_IFELSE( + [AC_LANG_SOURCE([[ +#include +#include +#include + ]], + [[ +{ + /* This test fails on HP-UX 11.00..11.11. */ + { + int fd = open ("conftest.out", O_RDWR | O_CREAT | O_TRUNC, 0600); + if (fd < 0) + return 1; + if (pwrite (fd, "b", 1, (off_t) -1) >= 0) + return 2; + } + return 0; +}]])], + [gl_cv_func_pwrite_works=yes], + [gl_cv_func_pwrite_works=no], + [:]) + rm -f conftest.out + CPPFLAGS="$gl_save_CPPFLAGS" + ]) + case "$gl_cv_func_pwrite_works" in + *yes) ;; + *) REPLACE_PWRITE=1 ;; + esac + else HAVE_PWRITE=0 + fi + if test $HAVE_PWRITE = 0 || test $REPLACE_PWRITE = 1; then AC_LIBOBJ([pwrite]) fi ]) -- 2.11.0