blob: 2cfc5751cacf55b60fcae6648df4d59dee843e1b [file] [log] [blame]
"Robert P. J. Day"63fc1a92006-07-02 19:47:05 +00001/* vi: set sw=4 ts=4: */
Mark Whitley8a633262001-04-30 18:17:00 +00002/*
Eric Andersen28355a32001-05-07 17:48:28 +00003 * xreadlink.c - safe implementation of readlink.
4 * Returns a NULL on failure...
Mark Whitley8a633262001-04-30 18:17:00 +00005 */
6
Denis Vlasenkoa9b60e92007-01-04 17:59:59 +00007#include "libbb.h"
Mark Whitley8a633262001-04-30 18:17:00 +00008
9/*
10 * NOTE: This function returns a malloced char* that you will have to free
Bernhard Reutner-Fischer9bd8d0c2007-11-08 21:11:43 +000011 * yourself.
Mark Whitley8a633262001-04-30 18:17:00 +000012 */
Denis Vlasenkodefc1ea2008-06-27 02:52:20 +000013char* FAST_FUNC xmalloc_readlink(const char *path)
Tim Rikerc1ef7bd2006-01-25 00:08:53 +000014{
Rob Landleybc68cd12006-03-10 19:22:06 +000015 enum { GROWBY = 80 }; /* how large we will grow strings by */
Mark Whitley8a633262001-04-30 18:17:00 +000016
Eric Andersenc7bda1c2004-03-15 08:29:22 +000017 char *buf = NULL;
Mark Whitley8a633262001-04-30 18:17:00 +000018 int bufsize = 0, readsize = 0;
19
20 do {
Denis Vlasenkob68979a2007-11-02 23:31:10 +000021 bufsize += GROWBY;
22 buf = xrealloc(buf, bufsize);
Denis Vlasenkobeffd432007-09-05 11:30:34 +000023 readsize = readlink(path, buf, bufsize);
Eric Andersen28355a32001-05-07 17:48:28 +000024 if (readsize == -1) {
Glenn L McGrath18bbd9b2004-08-11 03:50:30 +000025 free(buf);
26 return NULL;
Eric Andersen28355a32001-05-07 17:48:28 +000027 }
Denis Vlasenkobeffd432007-09-05 11:30:34 +000028 } while (bufsize < readsize + 1);
Mark Whitley8a633262001-04-30 18:17:00 +000029
30 buf[readsize] = '\0';
31
32 return buf;
Eric Andersenc7bda1c2004-03-15 08:29:22 +000033}
Denis Vlasenkoa9b60e92007-01-04 17:59:59 +000034
Paul Fox459a2ba2007-11-08 01:11:41 +000035/*
Bernhard Reutner-Fischer9bd8d0c2007-11-08 21:11:43 +000036 * This routine is not the same as realpath(), which
37 * canonicalizes the given path completely. This routine only
38 * follows trailing symlinks until a real file is reached and
39 * returns its name. If the path ends in a dangling link or if
40 * the target doesn't exist, the path is returned in any case.
41 * Intermediate symlinks in the path are not expanded -- only
Paul Fox599bbfb2007-11-08 20:00:36 +000042 * those at the tail.
Bernhard Reutner-Fischer9bd8d0c2007-11-08 21:11:43 +000043 * A malloced char* is returned, which must be freed by the caller.
Paul Fox459a2ba2007-11-08 01:11:41 +000044 */
Denis Vlasenkodefc1ea2008-06-27 02:52:20 +000045char* FAST_FUNC xmalloc_follow_symlinks(const char *path)
Paul Fox459a2ba2007-11-08 01:11:41 +000046{
Denis Vlasenkoabbd3632007-11-08 17:40:23 +000047 char *buf;
48 char *lpc;
49 char *linkpath;
Paul Fox459a2ba2007-11-08 01:11:41 +000050 int bufsize;
Denis Vlasenkoabbd3632007-11-08 17:40:23 +000051 int looping = MAXSYMLINKS + 1;
Paul Fox459a2ba2007-11-08 01:11:41 +000052
Paul Fox599bbfb2007-11-08 20:00:36 +000053 buf = xstrdup(path);
Denis Vlasenkoabbd3632007-11-08 17:40:23 +000054 goto jump_in;
Paul Fox459a2ba2007-11-08 01:11:41 +000055
Denis Vlasenkoabbd3632007-11-08 17:40:23 +000056 while (1) {
Paul Fox599bbfb2007-11-08 20:00:36 +000057 linkpath = xmalloc_readlink(buf);
58 if (!linkpath) {
59 /* not a symlink, or doesn't exist */
60 if (errno == EINVAL || errno == ENOENT)
61 return buf;
Bernhard Reutner-Fischer9bd8d0c2007-11-08 21:11:43 +000062 goto free_buf_ret_null;
63 }
Paul Fox599bbfb2007-11-08 20:00:36 +000064
Denis Vlasenkoabbd3632007-11-08 17:40:23 +000065 if (!--looping) {
66 free(linkpath);
Denis Vlasenkod031b202007-11-10 01:28:19 +000067 free_buf_ret_null:
Denis Vlasenkoabbd3632007-11-08 17:40:23 +000068 free(buf);
69 return NULL;
70 }
Paul Fox599bbfb2007-11-08 20:00:36 +000071
72 if (*linkpath != '/') {
Paul Fox459a2ba2007-11-08 01:11:41 +000073 bufsize += strlen(linkpath);
Paul Fox459a2ba2007-11-08 01:11:41 +000074 buf = xrealloc(buf, bufsize);
75 lpc = bb_get_last_path_component_strip(buf);
76 strcpy(lpc, linkpath);
77 free(linkpath);
Denis Vlasenkoabbd3632007-11-08 17:40:23 +000078 } else {
79 free(buf);
Denis Vlasenkoabbd3632007-11-08 17:40:23 +000080 buf = linkpath;
Paul Fox599bbfb2007-11-08 20:00:36 +000081 jump_in:
Denis Vlasenkoabbd3632007-11-08 17:40:23 +000082 bufsize = strlen(buf) + 1;
Paul Fox459a2ba2007-11-08 01:11:41 +000083 }
84 }
Paul Fox459a2ba2007-11-08 01:11:41 +000085}
86
Denis Vlasenkodefc1ea2008-06-27 02:52:20 +000087char* FAST_FUNC xmalloc_readlink_or_warn(const char *path)
Denis Vlasenkobeffd432007-09-05 11:30:34 +000088{
89 char *buf = xmalloc_readlink(path);
90 if (!buf) {
91 /* EINVAL => "file: Invalid argument" => puzzled user */
92 bb_error_msg("%s: cannot read link (not a symlink?)", path);
93 }
94 return buf;
95}
96
97/* UNUSED */
98#if 0
Denis Vlasenkodefc1ea2008-06-27 02:52:20 +000099char* FAST_FUNC xmalloc_realpath(const char *path)
Denis Vlasenkoa9b60e92007-01-04 17:59:59 +0000100{
Denis Vlasenko218f2f42007-01-24 22:02:01 +0000101#if defined(__GLIBC__) && !defined(__UCLIBC__)
Denis Vlasenkoa9b60e92007-01-04 17:59:59 +0000102 /* glibc provides a non-standard extension */
103 return realpath(path, NULL);
104#else
105 char buf[PATH_MAX+1];
106
107 /* on error returns NULL (xstrdup(NULL) ==NULL) */
108 return xstrdup(realpath(path, buf));
109#endif
110}
Denis Vlasenkobeffd432007-09-05 11:30:34 +0000111#endif