Blame gnulib/lib/readlink.c

Packit Service a2ae7a
/* Stub for readlink().
Packit Service a2ae7a
   Copyright (C) 2003-2007, 2009-2019 Free Software Foundation, Inc.
Packit Service a2ae7a
Packit Service a2ae7a
   This program is free software: you can redistribute it and/or modify
Packit Service a2ae7a
   it under the terms of the GNU Lesser General Public License as published by
Packit Service a2ae7a
   the Free Software Foundation; either version 2.1 of the License, or
Packit Service a2ae7a
   (at your option) any later version.
Packit Service a2ae7a
Packit Service a2ae7a
   This program is distributed in the hope that it will be useful,
Packit Service a2ae7a
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service a2ae7a
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit Service a2ae7a
   GNU Lesser General Public License for more details.
Packit Service a2ae7a
Packit Service a2ae7a
   You should have received a copy of the GNU Lesser General Public License
Packit Service a2ae7a
   along with this program.  If not, see <https://www.gnu.org/licenses/>.  */
Packit Service a2ae7a
Packit Service a2ae7a
#include <config.h>
Packit Service a2ae7a
Packit Service a2ae7a
/* Specification.  */
Packit Service a2ae7a
#include <unistd.h>
Packit Service a2ae7a
Packit Service a2ae7a
#include <errno.h>
Packit Service a2ae7a
#include <string.h>
Packit Service a2ae7a
#include <sys/stat.h>
Packit Service a2ae7a
Packit Service a2ae7a
#if !HAVE_READLINK
Packit Service a2ae7a
Packit Service a2ae7a
/* readlink() substitute for systems that don't have a readlink() function,
Packit Service a2ae7a
   such as DJGPP 2.03 and mingw32.  */
Packit Service a2ae7a
Packit Service a2ae7a
ssize_t
Packit Service a2ae7a
readlink (const char *name, char *buf _GL_UNUSED,
Packit Service a2ae7a
          size_t bufsize _GL_UNUSED)
Packit Service a2ae7a
{
Packit Service a2ae7a
  struct stat statbuf;
Packit Service a2ae7a
Packit Service a2ae7a
  /* In general we should use lstat() here, not stat().  But on platforms
Packit Service a2ae7a
     without symbolic links, lstat() - if it exists - would be equivalent to
Packit Service a2ae7a
     stat(), therefore we can use stat().  This saves us a configure check.  */
Packit Service a2ae7a
  if (stat (name, &statbuf) >= 0)
Packit Service a2ae7a
    errno = EINVAL;
Packit Service a2ae7a
  return -1;
Packit Service a2ae7a
}
Packit Service a2ae7a
Packit Service a2ae7a
#else /* HAVE_READLINK */
Packit Service a2ae7a
Packit Service a2ae7a
# undef readlink
Packit Service a2ae7a
Packit Service a2ae7a
/* readlink() wrapper that uses correct types, for systems like cygwin
Packit Service a2ae7a
   1.5.x where readlink returns int, and which rejects trailing slash,
Packit Service a2ae7a
   for Solaris 9.  */
Packit Service a2ae7a
Packit Service a2ae7a
ssize_t
Packit Service a2ae7a
rpl_readlink (const char *name, char *buf, size_t bufsize)
Packit Service a2ae7a
{
Packit Service a2ae7a
# if READLINK_TRAILING_SLASH_BUG
Packit Service a2ae7a
  size_t len = strlen (name);
Packit Service a2ae7a
  if (len && name[len - 1] == '/')
Packit Service a2ae7a
    {
Packit Service a2ae7a
      /* Even if name without the slash is a symlink to a directory,
Packit Service a2ae7a
         both lstat() and stat() must resolve the trailing slash to
Packit Service a2ae7a
         the directory rather than the symlink.  We can therefore
Packit Service a2ae7a
         safely use stat() to distinguish between EINVAL and
Packit Service a2ae7a
         ENOTDIR/ENOENT, avoiding extra overhead of rpl_lstat().  */
Packit Service a2ae7a
      struct stat st;
Packit Service a2ae7a
      if (stat (name, &st) == 0)
Packit Service a2ae7a
        errno = EINVAL;
Packit Service a2ae7a
      return -1;
Packit Service a2ae7a
    }
Packit Service a2ae7a
# endif /* READLINK_TRAILING_SLASH_BUG */
Packit Service a2ae7a
  return readlink (name, buf, bufsize);
Packit Service a2ae7a
}
Packit Service a2ae7a
Packit Service a2ae7a
#endif /* HAVE_READLINK */