2000-10-10 00:26:47 +05:30
|
|
|
/* vi: set sw=4 ts=4: */
|
|
|
|
/*
|
|
|
|
* Mini readlink implementation for busybox
|
|
|
|
*
|
2001-10-24 10:30:29 +05:30
|
|
|
* Copyright (C) 2000,2001 Matt Kraai <kraai@alumni.carnegiemellon.edu>
|
2000-10-10 00:26:47 +05:30
|
|
|
*
|
2010-08-16 23:44:46 +05:30
|
|
|
* Licensed under GPLv2 or later, see file LICENSE in this source tree.
|
2000-10-10 00:26:47 +05:30
|
|
|
*/
|
2016-11-23 19:16:56 +05:30
|
|
|
//config:config READLINK
|
2018-12-28 07:50:17 +05:30
|
|
|
//config: bool "readlink (4 kb)"
|
2016-11-23 19:16:56 +05:30
|
|
|
//config: default y
|
|
|
|
//config: help
|
2017-07-21 13:20:55 +05:30
|
|
|
//config: This program reads a symbolic link and returns the name
|
|
|
|
//config: of the file it points to
|
2016-11-23 19:16:56 +05:30
|
|
|
//config:
|
|
|
|
//config:config FEATURE_READLINK_FOLLOW
|
|
|
|
//config: bool "Enable canonicalization by following all symlinks (-f)"
|
|
|
|
//config: default y
|
|
|
|
//config: depends on READLINK
|
|
|
|
//config: help
|
2017-07-21 13:20:55 +05:30
|
|
|
//config: Enable the readlink option (-f).
|
2016-11-23 19:16:56 +05:30
|
|
|
|
2017-08-03 22:30:01 +05:30
|
|
|
//applet:IF_READLINK(APPLET_NOFORK(readlink, readlink, BB_DIR_USR_BIN, BB_SUID_DROP, readlink))
|
2016-11-23 19:16:56 +05:30
|
|
|
|
|
|
|
//kbuild:lib-$(CONFIG_READLINK) += readlink.o
|
2011-03-31 18:13:25 +05:30
|
|
|
|
|
|
|
//usage:#define readlink_trivial_usage
|
|
|
|
//usage: IF_FEATURE_READLINK_FOLLOW("[-fnv] ") "FILE"
|
|
|
|
//usage:#define readlink_full_usage "\n\n"
|
|
|
|
//usage: "Display the value of a symlink"
|
|
|
|
//usage: IF_FEATURE_READLINK_FOLLOW( "\n"
|
|
|
|
//usage: "\n -f Canonicalize by following all symlinks"
|
|
|
|
//usage: "\n -n Don't add newline"
|
|
|
|
//usage: "\n -v Verbose"
|
|
|
|
//usage: )
|
|
|
|
|
2007-05-27 00:30:18 +05:30
|
|
|
#include "libbb.h"
|
|
|
|
|
2009-06-12 16:46:21 +05:30
|
|
|
/*
|
|
|
|
* # readlink --version
|
|
|
|
* readlink (GNU coreutils) 6.10
|
|
|
|
* # readlink --help
|
|
|
|
* -f, --canonicalize
|
|
|
|
* canonicalize by following every symlink in
|
|
|
|
* every component of the given name recursively;
|
|
|
|
* all but the last component must exist
|
|
|
|
* -e, --canonicalize-existing
|
|
|
|
* canonicalize by following every symlink in
|
|
|
|
* every component of the given name recursively,
|
|
|
|
* all components must exist
|
|
|
|
* -m, --canonicalize-missing
|
|
|
|
* canonicalize by following every symlink in
|
|
|
|
* every component of the given name recursively,
|
|
|
|
* without requirements on components existence
|
|
|
|
* -n, --no-newline do not output the trailing newline
|
|
|
|
* -q, --quiet, -s, --silent suppress most error messages
|
|
|
|
* -v, --verbose report error messages
|
|
|
|
*
|
2013-03-12 21:08:03 +05:30
|
|
|
* bbox supports: -f (partially) -n -v (fully), -q -s (accepts but ignores)
|
|
|
|
* Note: we export the -f flag, but our -f behaves like coreutils' -e.
|
|
|
|
* Unfortunately, there isn't a C lib function we can leverage to get this
|
|
|
|
* behavior which means we'd have to implement the full stack ourselves :(.
|
2009-06-12 16:46:21 +05:30
|
|
|
*/
|
|
|
|
|
2007-10-11 15:35:36 +05:30
|
|
|
int readlink_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
|
2008-07-05 14:48:54 +05:30
|
|
|
int readlink_main(int argc UNUSED_PARAM, char **argv)
|
2000-10-10 00:26:47 +05:30
|
|
|
{
|
2005-09-12 05:15:28 +05:30
|
|
|
char *buf;
|
2006-10-21 00:06:55 +05:30
|
|
|
char *fname;
|
|
|
|
|
2009-04-21 16:39:40 +05:30
|
|
|
IF_FEATURE_READLINK_FOLLOW(
|
2006-10-21 00:06:55 +05:30
|
|
|
unsigned opt;
|
|
|
|
/* We need exactly one non-option argument. */
|
2017-08-09 01:25:02 +05:30
|
|
|
opt = getopt32(argv, "^" "fnvsq" "\0" "=1");
|
2006-10-21 00:06:55 +05:30
|
|
|
fname = argv[optind];
|
|
|
|
)
|
2009-04-21 16:39:40 +05:30
|
|
|
IF_NOT_FEATURE_READLINK_FOLLOW(
|
2006-10-21 00:06:55 +05:30
|
|
|
const unsigned opt = 0;
|
|
|
|
if (argc != 2) bb_show_usage();
|
|
|
|
fname = argv[1];
|
|
|
|
)
|
|
|
|
|
|
|
|
/* compat: coreutils readlink reports errors silently via exit code */
|
2009-06-12 16:46:21 +05:30
|
|
|
if (!(opt & 4)) /* not -v */
|
|
|
|
logmode = LOGMODE_NONE;
|
2006-10-21 00:06:55 +05:30
|
|
|
|
2017-08-03 22:30:01 +05:30
|
|
|
/* NOFORK: only one alloc is allowed; must free */
|
2009-06-12 16:46:21 +05:30
|
|
|
if (opt & 1) { /* -f */
|
2018-05-24 20:59:14 +05:30
|
|
|
buf = xmalloc_realpath_coreutils(fname);
|
2006-10-21 00:06:55 +05:30
|
|
|
} else {
|
2007-02-11 21:49:28 +05:30
|
|
|
buf = xmalloc_readlink_or_warn(fname);
|
2006-10-21 00:06:55 +05:30
|
|
|
}
|
2004-12-08 22:17:28 +05:30
|
|
|
|
2001-05-07 23:18:28 +05:30
|
|
|
if (!buf)
|
|
|
|
return EXIT_FAILURE;
|
2009-06-12 16:46:21 +05:30
|
|
|
printf((opt & 2) ? "%s" : "%s\n", buf);
|
2017-08-03 22:30:01 +05:30
|
|
|
free(buf);
|
2000-10-10 00:26:47 +05:30
|
|
|
|
2006-10-27 04:51:47 +05:30
|
|
|
fflush_stdout_and_exit(EXIT_SUCCESS);
|
2000-10-10 00:26:47 +05:30
|
|
|
}
|