2011-04-01 06:28:48 +00:00
|
|
|
/* Read symbolic links into a buffer without size limitation, relative to fd.
|
|
|
|
|
Update from gnulib
This incorporates:
2019-12-23 mktime, nstrftime: tweak division performance
2019-12-22 count-leading-zeros: assume 'long long'
2019-12-22 count-one-bits: assume 'long long'
2019-12-22 count-trailing-zeros: assume 'long long'
2019-12-12 inttypes-incomplete: assume 'long long'
2019-12-22 malloca: assume 'long long'
2019-12-22 stdint: assume 'long long'
2019-12-22 strtoll, strtoimax, strtoumax: assume 'long long'
2019-12-22 prefer lib_SOURCES to unconditional AC_LIBOBJ
2019-12-19 nstrftime: avoid a shadowing warning
2019-12-18 improve port of AC_C_RESTRICT to Oracle C++
2019-12-12 stdalign: port to xlclang 16.01
2019-12-11 stddef, unistd: fix compilation error in C++ mode on MSVC
2019-12-08 fix compilation errors in C++ mode on Haiku
2019-12-08 fix compilation errors in 32-bit C++ mode on HP-UX 11/ia64
2019-12-08 fix compilation error in C++ mode on OpenBSD
* build-aux/config.guess, doc/misc/texinfo.tex:
* lib/count-leading-zeros.h, lib/count-one-bits.h:
* lib/count-trailing-zeros.h, lib/inttypes.in.h, lib/malloca.h:
* lib/mktime.c, lib/nstrftime.c, lib/signal.in.h, lib/stdalign.in.h:
* lib/stddef.in.h, lib/stdint.in.h, lib/stdio.in.h, lib/stdlib.in.h:
* lib/strtoimax.c, lib/unistd.in.h, m4/gnulib-common.m4:
* m4/inttypes.m4, m4/largefile.m4, m4/malloca.m4, m4/strtoimax.m4:
* m4/strtoll.m4:
Copy from Gnulib. Also, change copyright notices in some other
Gnulib-copied files to exactly match Gnulib, as Gnulib updated
them in a trivially different way.
* lib/gnulib.mk.in, m4/gnulib-comp.m4: Regenerate.
2020-01-01 03:11:22 +00:00
|
|
|
Copyright (C) 2001, 2003-2004, 2007, 2009-2020 Free Software Foundation,
|
|
|
|
Inc.
|
2011-04-01 06:28:48 +00:00
|
|
|
|
|
|
|
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
|
|
|
|
the Free Software Foundation; either version 3 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
2017-09-13 09:07:03 +00:00
|
|
|
along with this program. If not, see <https://www.gnu.org/licenses/>. */
|
2011-04-01 06:28:48 +00:00
|
|
|
|
|
|
|
/* Written by Paul Eggert, Bruno Haible, and Jim Meyering. */
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
|
|
|
#include "careadlinkat.h"
|
|
|
|
|
|
|
|
#include <errno.h>
|
|
|
|
#include <limits.h>
|
|
|
|
#include <string.h>
|
2013-02-08 23:37:17 +00:00
|
|
|
#include <unistd.h>
|
2011-04-01 06:28:48 +00:00
|
|
|
|
|
|
|
/* Define this independently so that stdint.h is not a prerequisite. */
|
|
|
|
#ifndef SIZE_MAX
|
|
|
|
# define SIZE_MAX ((size_t) -1)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef SSIZE_MAX
|
|
|
|
# define SSIZE_MAX ((ssize_t) (SIZE_MAX / 2))
|
|
|
|
#endif
|
|
|
|
|
2011-04-10 16:00:46 +00:00
|
|
|
#include "allocator.h"
|
|
|
|
|
2011-04-01 06:28:48 +00:00
|
|
|
/* Assuming the current directory is FD, get the symbolic link value
|
|
|
|
of FILENAME as a null-terminated string and put it into a buffer.
|
|
|
|
If FD is AT_FDCWD, FILENAME is interpreted relative to the current
|
|
|
|
working directory, as in openat.
|
|
|
|
|
|
|
|
If the link is small enough to fit into BUFFER put it there.
|
|
|
|
BUFFER's size is BUFFER_SIZE, and BUFFER can be null
|
|
|
|
if BUFFER_SIZE is zero.
|
|
|
|
|
|
|
|
If the link is not small, put it into a dynamically allocated
|
|
|
|
buffer managed by ALLOC. It is the caller's responsibility to free
|
|
|
|
the returned value if it is nonnull and is not BUFFER. A null
|
|
|
|
ALLOC stands for the standard allocator.
|
|
|
|
|
2011-04-09 18:44:05 +00:00
|
|
|
The PREADLINKAT function specifies how to read links. It operates
|
|
|
|
like POSIX readlinkat()
|
Update from Gnulib
This incorporates:
2019-09-22 Update some URLs
2019-09-15 fcntl-h: fix compilation error of creat.c on MSVC
2019-09-15 creat: new module
2019-09-15 access: new module
2019-09-09 Add option to assume best, not worst, when cross-compiling.
* build-aux/config.guess, build-aux/config.sub, doc/misc/texinfo.tex:
* lib/careadlinkat.c, lib/careadlinkat.h, lib/count-leading-zeros.h:
* lib/count-trailing-zeros.h, lib/diffseq.h, lib/fcntl.in.h:
* lib/ftoastr.c, lib/get-permissions.c:
* lib/ieee754.in.h, lib/inttypes.in.h, lib/mktime.c, lib/open.c:
* lib/pathmax.h, lib/pipe2.c, lib/stddef.in.h, lib/stdint.in.h:
* lib/stdlib.in.h, lib/str-two-way.h, lib/string.in.h, lib/time.in.h:
* lib/timegm.c, lib/unistd.in.h, m4/canonicalize.m4:
* m4/extern-inline.m4, m4/fcntl_h.m4, m4/fdopendir.m4:
* m4/getgroups.m4, m4/getopt.m4, m4/gettimeofday.m4:
* m4/gnulib-common.m4, m4/largefile.m4:
* m4/lstat.m4, m4/memmem.m4, m4/mktime.m4, m4/nocrash.m4, m4/open.m4:
* m4/pselect.m4, m4/putenv.m4, m4/readlink.m4, m4/regex.m4:
* m4/symlink.m4, m4/unistd_h.m4, m4/utimens.m4, m4/utimes.m4:
Copy from Gnulib.
* lib/gnulib.mk.in, m4/gnulib-comp.m4: Regenerate.
* m4/open-slash.m4: New file, copied from Gnulib.
2019-09-23 06:50:59 +00:00
|
|
|
<https://pubs.opengroup.org/onlinepubs/9699919799/functions/readlink.html>
|
2011-04-09 18:44:05 +00:00
|
|
|
but can assume that its first argument is the same as FD.
|
2011-04-01 06:28:48 +00:00
|
|
|
|
|
|
|
If successful, return the buffer address; otherwise return NULL and
|
|
|
|
set errno. */
|
|
|
|
|
|
|
|
char *
|
|
|
|
careadlinkat (int fd, char const *filename,
|
|
|
|
char *buffer, size_t buffer_size,
|
|
|
|
struct allocator const *alloc,
|
|
|
|
ssize_t (*preadlinkat) (int, char const *, char *, size_t))
|
|
|
|
{
|
|
|
|
char *buf;
|
|
|
|
size_t buf_size;
|
|
|
|
size_t buf_size_max =
|
|
|
|
SSIZE_MAX < SIZE_MAX ? (size_t) SSIZE_MAX + 1 : SIZE_MAX;
|
|
|
|
char stack_buf[1024];
|
|
|
|
|
2011-04-05 18:19:19 +00:00
|
|
|
if (! alloc)
|
2011-04-08 21:53:30 +00:00
|
|
|
alloc = &stdlib_allocator;
|
2011-04-01 06:28:48 +00:00
|
|
|
|
|
|
|
if (! buffer_size)
|
|
|
|
{
|
|
|
|
/* Allocate the initial buffer on the stack. This way, in the
|
|
|
|
common case of a symlink of small size, we get away with a
|
|
|
|
single small malloc() instead of a big malloc() followed by a
|
|
|
|
shrinking realloc(). */
|
|
|
|
buffer = stack_buf;
|
|
|
|
buffer_size = sizeof stack_buf;
|
|
|
|
}
|
|
|
|
|
|
|
|
buf = buffer;
|
|
|
|
buf_size = buffer_size;
|
|
|
|
|
|
|
|
do
|
|
|
|
{
|
|
|
|
/* Attempt to read the link into the current buffer. */
|
|
|
|
ssize_t link_length = preadlinkat (fd, filename, buf, buf_size);
|
|
|
|
size_t link_size;
|
|
|
|
if (link_length < 0)
|
|
|
|
{
|
|
|
|
/* On AIX 5L v5.3 and HP-UX 11i v2 04/09, readlink returns -1
|
|
|
|
with errno == ERANGE if the buffer is too small. */
|
|
|
|
int readlinkat_errno = errno;
|
|
|
|
if (readlinkat_errno != ERANGE)
|
|
|
|
{
|
|
|
|
if (buf != buffer)
|
|
|
|
{
|
2011-04-05 18:19:19 +00:00
|
|
|
alloc->free (buf);
|
2011-04-01 06:28:48 +00:00
|
|
|
errno = readlinkat_errno;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
link_size = link_length;
|
|
|
|
|
|
|
|
if (link_size < buf_size)
|
|
|
|
{
|
|
|
|
buf[link_size++] = '\0';
|
|
|
|
|
|
|
|
if (buf == stack_buf)
|
|
|
|
{
|
2011-04-08 21:53:30 +00:00
|
|
|
char *b = (char *) alloc->allocate (link_size);
|
2011-06-02 08:22:57 +00:00
|
|
|
buf_size = link_size;
|
2011-04-01 06:28:48 +00:00
|
|
|
if (! b)
|
|
|
|
break;
|
|
|
|
memcpy (b, buf, link_size);
|
|
|
|
buf = b;
|
|
|
|
}
|
2011-04-08 21:53:30 +00:00
|
|
|
else if (link_size < buf_size && buf != buffer && alloc->reallocate)
|
2011-04-01 06:28:48 +00:00
|
|
|
{
|
|
|
|
/* Shrink BUF before returning it. */
|
2011-04-08 21:53:30 +00:00
|
|
|
char *b = (char *) alloc->reallocate (buf, link_size);
|
2011-04-01 06:28:48 +00:00
|
|
|
if (b)
|
|
|
|
buf = b;
|
|
|
|
}
|
|
|
|
|
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (buf != buffer)
|
2011-04-05 18:19:19 +00:00
|
|
|
alloc->free (buf);
|
2011-04-01 06:28:48 +00:00
|
|
|
|
|
|
|
if (buf_size <= buf_size_max / 2)
|
|
|
|
buf_size *= 2;
|
|
|
|
else if (buf_size < buf_size_max)
|
|
|
|
buf_size = buf_size_max;
|
2011-06-02 08:22:57 +00:00
|
|
|
else if (buf_size_max < SIZE_MAX)
|
|
|
|
{
|
|
|
|
errno = ENAMETOOLONG;
|
|
|
|
return NULL;
|
|
|
|
}
|
2011-04-01 06:28:48 +00:00
|
|
|
else
|
|
|
|
break;
|
2011-04-08 21:53:30 +00:00
|
|
|
buf = (char *) alloc->allocate (buf_size);
|
2011-04-01 06:28:48 +00:00
|
|
|
}
|
|
|
|
while (buf);
|
|
|
|
|
2011-04-05 18:19:19 +00:00
|
|
|
if (alloc->die)
|
2011-06-02 08:22:57 +00:00
|
|
|
alloc->die (buf_size);
|
2011-04-01 06:28:48 +00:00
|
|
|
errno = ENOMEM;
|
|
|
|
return NULL;
|
|
|
|
}
|