mirror of
https://git.savannah.gnu.org/git/gnulib.git
synced 2025-08-20 11:01:20 +03:00
lstat: fix Solaris 9 bug
lstat("file/",buf) mistakenly succeeded. * lib/lstat.c (lstat): Also check for trailing slash on non-symlink, non-directories. Use stat module to simplify logic. * doc/posix-functions/lstat.texi (lstat): Document it. * modules/lstat-tests (Depends-on): Add errno, same-inode. (configure.ac): Check for symlink. * tests/test-lstat.c (main): Add more tests. Signed-off-by: Eric Blake <ebb9@byu.net>
This commit is contained in:
@@ -1,5 +1,13 @@
|
|||||||
2009-09-19 Eric Blake <ebb9@byu.net>
|
2009-09-19 Eric Blake <ebb9@byu.net>
|
||||||
|
|
||||||
|
lstat: fix Solaris 9 bug
|
||||||
|
* lib/lstat.c (lstat): Also check for trailing slash on
|
||||||
|
non-symlink, non-directories. Use stat module to simplify logic.
|
||||||
|
* doc/posix-functions/lstat.texi (lstat): Document it.
|
||||||
|
* modules/lstat-tests (Depends-on): Add errno, same-inode.
|
||||||
|
(configure.ac): Check for symlink.
|
||||||
|
* tests/test-lstat.c (main): Add more tests.
|
||||||
|
|
||||||
stat: add as dependency to other modules
|
stat: add as dependency to other modules
|
||||||
* modules/chown (Depends-on): Add stat.
|
* modules/chown (Depends-on): Add stat.
|
||||||
* modules/euidaccess (Depends-on): Likewise.
|
* modules/euidaccess (Depends-on): Likewise.
|
||||||
|
@@ -9,8 +9,13 @@ Gnulib module: lstat
|
|||||||
Portability problems fixed by Gnulib:
|
Portability problems fixed by Gnulib:
|
||||||
@itemize
|
@itemize
|
||||||
@item
|
@item
|
||||||
When the argument ends in a slash, some platforms don't dereference the
|
For symlinks, when the argument ends in a slash, some platforms don't
|
||||||
argument.
|
dereference the argument:
|
||||||
|
Solaris 9.
|
||||||
|
@item
|
||||||
|
On some platforms, @code{lstat("file/",buf)} succeeds instead of
|
||||||
|
failing with @code{ENOTDIR}.
|
||||||
|
Solaris 9.
|
||||||
@item
|
@item
|
||||||
On Windows platforms (excluding Cygwin), symlinks are not supported, so
|
On Windows platforms (excluding Cygwin), symlinks are not supported, so
|
||||||
@code{lstat} does not exist.
|
@code{lstat} does not exist.
|
||||||
@@ -22,4 +27,12 @@ Portability problems not fixed by Gnulib:
|
|||||||
On platforms where @code{off_t} is a 32-bit type, @code{lstat} may not
|
On platforms where @code{off_t} is a 32-bit type, @code{lstat} may not
|
||||||
correctly report the size of files or block devices larger than 2 GB. The fix
|
correctly report the size of files or block devices larger than 2 GB. The fix
|
||||||
is to use the @code{AC_SYS_LARGEFILE} macro.
|
is to use the @code{AC_SYS_LARGEFILE} macro.
|
||||||
|
@item
|
||||||
|
On Windows platforms (excluding Cygwin), @code{st_ino} is always 0.
|
||||||
|
@item
|
||||||
|
Because of the definition of @code{struct stat}, it is not possible to
|
||||||
|
portably replace @code{stat} via an object-like macro. Therefore,
|
||||||
|
expressions such as @code{(islnk ? lstat : stat) (name, buf)} are not
|
||||||
|
portable, and should instead be written @code{islnk ? lstat (name,
|
||||||
|
buf) : stat (name, buf)}.
|
||||||
@end itemize
|
@end itemize
|
||||||
|
37
lib/lstat.c
37
lib/lstat.c
@@ -1,6 +1,7 @@
|
|||||||
/* Work around a bug of lstat on some systems
|
/* Work around a bug of lstat on some systems
|
||||||
|
|
||||||
Copyright (C) 1997-1999, 2000-2006, 2008 Free Software Foundation, Inc.
|
Copyright (C) 1997-1999, 2000-2006, 2008-2009 Free Software
|
||||||
|
Foundation, Inc.
|
||||||
|
|
||||||
This program is free software: you can redistribute it and/or modify
|
This program is free software: you can redistribute it and/or modify
|
||||||
it under the terms of the GNU General Public License as published by
|
it under the terms of the GNU General Public License as published by
|
||||||
@@ -19,7 +20,7 @@
|
|||||||
|
|
||||||
#include <config.h>
|
#include <config.h>
|
||||||
|
|
||||||
/* Get the original definition of open. It might be defined as a macro. */
|
/* Get the original definition of lstat. It might be defined as a macro. */
|
||||||
#define __need_system_sys_stat_h
|
#define __need_system_sys_stat_h
|
||||||
#include <sys/types.h>
|
#include <sys/types.h>
|
||||||
#include <sys/stat.h>
|
#include <sys/stat.h>
|
||||||
@@ -56,27 +57,27 @@ rpl_lstat (const char *file, struct stat *sbuf)
|
|||||||
size_t len;
|
size_t len;
|
||||||
int lstat_result = orig_lstat (file, sbuf);
|
int lstat_result = orig_lstat (file, sbuf);
|
||||||
|
|
||||||
if (lstat_result != 0 || !S_ISLNK (sbuf->st_mode))
|
if (lstat_result != 0)
|
||||||
return lstat_result;
|
return lstat_result;
|
||||||
|
|
||||||
|
/* This replacement file can blindly check against '/' rather than
|
||||||
|
using the ISSLASH macro, because all platforms with '\\' either
|
||||||
|
lack symlinks (mingw) or have working lstat (cygwin) and thus do
|
||||||
|
not compile this file. 0 len should have already been filtered
|
||||||
|
out above, with a failure return of ENOENT. */
|
||||||
len = strlen (file);
|
len = strlen (file);
|
||||||
if (len == 0 || file[len - 1] != '/')
|
if (file[len - 1] != '/' || S_ISDIR (sbuf->st_mode))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
/* FILE refers to a symbolic link and the name ends with a slash.
|
/* At this point, a trailing slash is only permitted on
|
||||||
Call stat() to get info about the link's referent. */
|
symlink-to-dir; but it should have found information on the
|
||||||
|
directory, not the symlink. Call stat() to get info about the
|
||||||
/* If stat fails, then we do the same. */
|
link's referent. Our replacement stat guarantees valid results,
|
||||||
if (stat (file, sbuf) != 0)
|
even if the symlink is not pointing to a directory. */
|
||||||
return -1;
|
if (!S_ISLNK (sbuf->st_mode))
|
||||||
|
{
|
||||||
/* If FILE references a directory, return 0. */
|
|
||||||
if (S_ISDIR (sbuf->st_mode))
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
/* Here, we know stat succeeded and FILE references a non-directory.
|
|
||||||
But it was specified via a name including a trailing slash.
|
|
||||||
Fail with errno set to ENOTDIR to indicate the contradiction. */
|
|
||||||
errno = ENOTDIR;
|
errno = ENOTDIR;
|
||||||
return -1;
|
return -1;
|
||||||
|
}
|
||||||
|
return stat (file, sbuf);
|
||||||
}
|
}
|
||||||
|
@@ -2,8 +2,11 @@ Files:
|
|||||||
tests/test-lstat.c
|
tests/test-lstat.c
|
||||||
|
|
||||||
Depends-on:
|
Depends-on:
|
||||||
|
errno
|
||||||
|
same-inode
|
||||||
|
|
||||||
configure.ac:
|
configure.ac:
|
||||||
|
AC_CHECK_FUNCS_ONCE([symlink])
|
||||||
|
|
||||||
Makefile.am:
|
Makefile.am:
|
||||||
TESTS += test-lstat
|
TESTS += test-lstat
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
/* Test of lstat() function.
|
/* Test of lstat() function.
|
||||||
Copyright (C) 2008 Free Software Foundation, Inc.
|
Copyright (C) 2008, 2009 Free Software Foundation, Inc.
|
||||||
|
|
||||||
This program is free software: you can redistribute it and/or modify
|
This program is free software: you can redistribute it and/or modify
|
||||||
it under the terms of the GNU General Public License as published by
|
it under the terms of the GNU General Public License as published by
|
||||||
@@ -14,24 +14,129 @@
|
|||||||
You should have received a copy of the GNU General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
along with this program. If not, see <http://www.gnu.org/licenses/>. */
|
along with this program. If not, see <http://www.gnu.org/licenses/>. */
|
||||||
|
|
||||||
/* Written by Simon Josefsson, 2008. */
|
/* Written by Simon Josefsson, 2008; and Eric Blake, 2009. */
|
||||||
|
|
||||||
#include <config.h>
|
#include <config.h>
|
||||||
|
|
||||||
#include <sys/stat.h>
|
#include <sys/stat.h>
|
||||||
|
|
||||||
|
#include <fcntl.h>
|
||||||
|
#include <errno.h>
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
|
||||||
|
#include "same-inode.h"
|
||||||
|
|
||||||
|
#if !HAVE_SYMLINK
|
||||||
|
# define symlink(a,b) (-1)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define ASSERT(expr) \
|
||||||
|
do \
|
||||||
|
{ \
|
||||||
|
if (!(expr)) \
|
||||||
|
{ \
|
||||||
|
fprintf (stderr, "%s:%d: assertion failed\n", __FILE__, __LINE__); \
|
||||||
|
fflush (stderr); \
|
||||||
|
abort (); \
|
||||||
|
} \
|
||||||
|
} \
|
||||||
|
while (0)
|
||||||
|
|
||||||
|
#define BASE "test-lstat.t"
|
||||||
|
|
||||||
int
|
int
|
||||||
main (int argc, char **argv)
|
main ()
|
||||||
{
|
{
|
||||||
struct stat buf;
|
struct stat st1;
|
||||||
|
struct stat st2;
|
||||||
|
|
||||||
if (lstat ("/", &buf) != 0)
|
/* Remove any leftovers from a previous partial run. */
|
||||||
|
ASSERT (system ("rm -rf " BASE "*") == 0);
|
||||||
|
|
||||||
|
/* Test for common directories. */
|
||||||
|
ASSERT (lstat (".", &st1) == 0);
|
||||||
|
ASSERT (lstat ("./", &st2) == 0);
|
||||||
|
ASSERT (SAME_INODE (st1, st2));
|
||||||
|
ASSERT (S_ISDIR (st1.st_mode));
|
||||||
|
ASSERT (S_ISDIR (st2.st_mode));
|
||||||
|
ASSERT (lstat ("/", &st1) == 0);
|
||||||
|
ASSERT (lstat ("///", &st2) == 0);
|
||||||
|
ASSERT (SAME_INODE (st1, st2));
|
||||||
|
ASSERT (S_ISDIR (st1.st_mode));
|
||||||
|
ASSERT (S_ISDIR (st2.st_mode));
|
||||||
|
ASSERT (lstat ("..", &st1) == 0);
|
||||||
|
ASSERT (S_ISDIR (st1.st_mode));
|
||||||
|
|
||||||
|
/* Test for error conditions. */
|
||||||
|
errno = 0;
|
||||||
|
ASSERT (lstat ("", &st1) == -1);
|
||||||
|
ASSERT (errno == ENOENT);
|
||||||
|
errno = 0;
|
||||||
|
ASSERT (lstat ("nosuch", &st1) == -1);
|
||||||
|
ASSERT (errno == ENOENT);
|
||||||
|
errno = 0;
|
||||||
|
ASSERT (lstat ("nosuch/", &st1) == -1);
|
||||||
|
ASSERT (errno == ENOENT);
|
||||||
|
|
||||||
|
ASSERT (close (creat (BASE "file", 0600)) == 0);
|
||||||
|
ASSERT (lstat (BASE "file", &st1) == 0);
|
||||||
|
ASSERT (S_ISREG (st1.st_mode));
|
||||||
|
errno = 0;
|
||||||
|
ASSERT (lstat (BASE "file/", &st1) == -1);
|
||||||
|
ASSERT (errno == ENOTDIR);
|
||||||
|
|
||||||
|
/* Now for some symlink tests, where supported. We set up:
|
||||||
|
link1 -> directory
|
||||||
|
link2 -> file
|
||||||
|
link3 -> dangling
|
||||||
|
link4 -> loop
|
||||||
|
then test behavior both with and without trailing slash.
|
||||||
|
*/
|
||||||
|
if (symlink (".", BASE "link1") != 0)
|
||||||
{
|
{
|
||||||
perror ("lstat");
|
ASSERT (unlink (BASE "file") == 0);
|
||||||
return 1;
|
fputs ("skipping test: symlinks not supported on this filesystem\n",
|
||||||
|
stderr);
|
||||||
|
return 77;
|
||||||
}
|
}
|
||||||
|
ASSERT (symlink (BASE "file", BASE "link2") == 0);
|
||||||
|
ASSERT (symlink (BASE "nosuch", BASE "link3") == 0);
|
||||||
|
ASSERT (symlink (BASE "link4", BASE "link4") == 0);
|
||||||
|
|
||||||
|
ASSERT (lstat (BASE "link1", &st1) == 0);
|
||||||
|
ASSERT (S_ISLNK (st1.st_mode));
|
||||||
|
ASSERT (lstat (BASE "link1/", &st1) == 0);
|
||||||
|
ASSERT (stat (BASE "link1", &st2) == 0);
|
||||||
|
ASSERT (S_ISDIR (st1.st_mode));
|
||||||
|
ASSERT (S_ISDIR (st2.st_mode));
|
||||||
|
ASSERT (SAME_INODE (st1, st2));
|
||||||
|
|
||||||
|
ASSERT (lstat (BASE "link2", &st1) == 0);
|
||||||
|
ASSERT (S_ISLNK (st1.st_mode));
|
||||||
|
errno = 0;
|
||||||
|
ASSERT (lstat (BASE "link2/", &st1) == -1);
|
||||||
|
ASSERT (errno == ENOTDIR);
|
||||||
|
|
||||||
|
ASSERT (lstat (BASE "link3", &st1) == 0);
|
||||||
|
ASSERT (S_ISLNK (st1.st_mode));
|
||||||
|
errno = 0;
|
||||||
|
ASSERT (lstat (BASE "link3/", &st1) == -1);
|
||||||
|
ASSERT (errno == ENOENT);
|
||||||
|
|
||||||
|
ASSERT (lstat (BASE "link4", &st1) == 0);
|
||||||
|
ASSERT (S_ISLNK (st1.st_mode));
|
||||||
|
errno = 0;
|
||||||
|
ASSERT (lstat (BASE "link4/", &st1) == -1);
|
||||||
|
ASSERT (errno == ELOOP);
|
||||||
|
|
||||||
|
/* Cleanup. */
|
||||||
|
ASSERT (unlink (BASE "file") == 0);
|
||||||
|
ASSERT (unlink (BASE "link1") == 0);
|
||||||
|
ASSERT (unlink (BASE "link2") == 0);
|
||||||
|
ASSERT (unlink (BASE "link3") == 0);
|
||||||
|
ASSERT (unlink (BASE "link4") == 0);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user