1
0
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:
Eric Blake
2009-09-17 15:55:24 -06:00
parent ba16529251
commit b4caad347f
5 changed files with 159 additions and 29 deletions

View File

@@ -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.

View File

@@ -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

View File

@@ -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);
} }

View File

@@ -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

View File

@@ -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;
} }