X-Git-Url: http://erislabs.net/gitweb/?a=blobdiff_plain;ds=sidebyside;f=m4%2Futimes.m4;h=dc93cf863c8e9125b5760b2d8e5421a1cf80356d;hb=423d3cac13c30bb0d5f85e5fac6ab1406ed1731a;hp=3b8d945ac25558fc0dd327cf9c4bf1128f5dff93;hpb=078bf4ac2813c5a3a93e5d57b5b42c3419aa63a0;p=gnulib.git diff --git a/m4/utimes.m4 b/m4/utimes.m4 index 3b8d945ac..dc93cf863 100644 --- a/m4/utimes.m4 +++ b/m4/utimes.m4 @@ -1,20 +1,34 @@ +# Detect some bugs in glibc's implementation of utimes. +# serial 3 + +dnl Copyright (C) 2003-2005, 2009-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, +dnl with or without modifications, as long as this notice is preserved. + # See if we need to work around bugs in glibc's implementation of -# utimes from around July/August 2003. +# utimes from 2003-07-12 to 2003-09-17. # First, there was a bug that would make utimes set mtime # and atime to zero (1970-01-01) unconditionally. -# Then, there is/was code to round rather than truncate. +# Then, there was code to round rather than truncate. +# Then, there was an implementation (sparc64, Linux-2.4.28, glibc-2.3.3) +# that didn't honor the NULL-means-set-to-current-time semantics. +# Finally, there was also a version of utimes that failed on read-only +# files, while utime worked fine (linux-2.2.20, glibc-2.2.5). # # From Jim Meyering, with suggestions from Paul Eggert. AC_DEFUN([gl_FUNC_UTIMES], [ - AC_CACHE_CHECK([determine whether the utimes function works], - gl_cv_func_working_utimes, + AC_CACHE_CHECK([whether the utimes function works], + [gl_cv_func_working_utimes], [ AC_RUN_IFELSE([AC_LANG_SOURCE([[ #include #include +#include #include +#include #include #include #include @@ -23,17 +37,71 @@ AC_DEFUN([gl_FUNC_UTIMES], int main () { - static struct timeval timeval[2] = {{9, 10}, {999999, 999999}}; - struct stat sbuf; + int result = 0; char const *file = "conftest.utimes"; - FILE *f; - - exit ( ! ((f = fopen (file, "w")) - && fclose (f) == 0 - && utimes (file, timeval) == 0 - && lstat (file, &sbuf) == 0 - && sbuf.st_atime == timeval[0].tv_sec - && sbuf.st_mtime == timeval[1].tv_sec) ); + + /* Test whether utimes() essentially works. */ + { + static struct timeval timeval[2] = {{9, 10}, {999999, 999999}}; + struct stat sbuf; + FILE *f = fopen (file, "w"); + if (f == NULL) + result |= 1; + else if (fclose (f) != 0) + result |= 1; + else if (utimes (file, timeval) != 0) + result |= 2; + else if (lstat (file, &sbuf) != 0) + result |= 1; + else if (!(sbuf.st_atime == timeval[0].tv_sec + && sbuf.st_mtime == timeval[1].tv_sec)) + result |= 4; + unlink (file); + } + + /* Test whether utimes() with a NULL argument sets the file's timestamp + to the current time. Note that this test fails on NFS file systems + if there is a time skew between the host and the NFS server. */ + { + struct stat sbuf; + FILE *f = fopen (file, "w"); + if (f == NULL) + result |= 1; + else if (fclose (f) != 0) + result |= 1; + else + { + time_t now; + if (time (&now) == (time_t)-1) + result |= 1; + else if (utimes (file, NULL) != 0) + result |= 8; + else if (lstat (file, &sbuf) != 0) + result |= 1; + else + { + if (!(now - sbuf.st_atime <= 2)) + result |= 16; + if (!(now - sbuf.st_mtime <= 2)) + result |= 32; + } + } + unlink (file); + } + + /* Test whether utimes() with a NULL argument works on read-only files. */ + { + int fd = open (file, O_WRONLY|O_CREAT, 0444); + if (fd < 0) + result |= 1; + else if (close (fd) != 0) + result |= 1; + else if (utimes (file, NULL) != 0) + result |= 64; + unlink (file); + } + + return result; } ]])], [gl_cv_func_working_utimes=yes], @@ -41,6 +109,6 @@ main () [gl_cv_func_working_utimes=no])]) if test $gl_cv_func_working_utimes = yes; then - AC_DEFINE([HAVE_WORKING_UTIMES], 1, [Define if utimes works properly. ]) + AC_DEFINE([HAVE_WORKING_UTIMES], [1], [Define if utimes works properly. ]) fi ])