blob: 1b223b3c52254fedd1a49821a4316310c2e6791e [file] [log] [blame]
Mark Whitley872138d2000-10-09 18:56:47 +00001/* vi: set sw=4 ts=4: */
2/*
3 * Mini readlink implementation for busybox
4 *
Eric Andersenbdfd0d72001-10-24 05:00:29 +00005 * Copyright (C) 2000,2001 Matt Kraai <kraai@alumni.carnegiemellon.edu>
Mark Whitley872138d2000-10-09 18:56:47 +00006 *
Denys Vlasenko0ef64bd2010-08-16 20:14:46 +02007 * Licensed under GPLv2 or later, see file LICENSE in this source tree.
Mark Whitley872138d2000-10-09 18:56:47 +00008 */
Denys Vlasenkoaf3f4202016-11-23 14:46:56 +01009//config:config READLINK
10//config: bool "readlink"
11//config: default y
12//config: help
13//config: This program reads a symbolic link and returns the name
14//config: of the file it points to
15//config:
16//config:config FEATURE_READLINK_FOLLOW
17//config: bool "Enable canonicalization by following all symlinks (-f)"
18//config: default y
19//config: depends on READLINK
20//config: help
21//config: Enable the readlink option (-f).
22
23//applet:IF_READLINK(APPLET(readlink, BB_DIR_USR_BIN, BB_SUID_DROP))
24
25//kbuild:lib-$(CONFIG_READLINK) += readlink.o
Pere Orga34425382011-03-31 14:43:25 +020026
27//usage:#define readlink_trivial_usage
28//usage: IF_FEATURE_READLINK_FOLLOW("[-fnv] ") "FILE"
29//usage:#define readlink_full_usage "\n\n"
30//usage: "Display the value of a symlink"
31//usage: IF_FEATURE_READLINK_FOLLOW( "\n"
Pere Orga34425382011-03-31 14:43:25 +020032//usage: "\n -f Canonicalize by following all symlinks"
33//usage: "\n -n Don't add newline"
34//usage: "\n -v Verbose"
35//usage: )
36
Denis Vlasenkob6adbf12007-05-26 19:00:18 +000037#include "libbb.h"
38
Denys Vlasenkoa99aa6e2009-06-12 13:16:21 +020039/*
40 * # readlink --version
41 * readlink (GNU coreutils) 6.10
42 * # readlink --help
43 * -f, --canonicalize
44 * canonicalize by following every symlink in
45 * every component of the given name recursively;
46 * all but the last component must exist
47 * -e, --canonicalize-existing
48 * canonicalize by following every symlink in
49 * every component of the given name recursively,
50 * all components must exist
51 * -m, --canonicalize-missing
52 * canonicalize by following every symlink in
53 * every component of the given name recursively,
54 * without requirements on components existence
55 * -n, --no-newline do not output the trailing newline
56 * -q, --quiet, -s, --silent suppress most error messages
57 * -v, --verbose report error messages
58 *
Mike Frysinger1b49c252013-03-12 11:38:03 -040059 * bbox supports: -f (partially) -n -v (fully), -q -s (accepts but ignores)
60 * Note: we export the -f flag, but our -f behaves like coreutils' -e.
61 * Unfortunately, there isn't a C lib function we can leverage to get this
62 * behavior which means we'd have to implement the full stack ourselves :(.
Denys Vlasenkoa99aa6e2009-06-12 13:16:21 +020063 */
64
Denis Vlasenko9b49a5e2007-10-11 10:05:36 +000065int readlink_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
Denis Vlasenkoa60f84e2008-07-05 09:18:54 +000066int readlink_main(int argc UNUSED_PARAM, char **argv)
Mark Whitley872138d2000-10-09 18:56:47 +000067{
Rob Landleyba502172005-09-11 23:45:28 +000068 char *buf;
Denis Vlasenko456fa6c2006-10-20 18:36:55 +000069 char *fname;
Mark Whitley8a633262001-04-30 18:17:00 +000070
Denis Vlasenko5e34ff22009-04-21 11:09:40 +000071 IF_FEATURE_READLINK_FOLLOW(
Denis Vlasenko456fa6c2006-10-20 18:36:55 +000072 unsigned opt;
73 /* We need exactly one non-option argument. */
74 opt_complementary = "=1";
Denys Vlasenkoa99aa6e2009-06-12 13:16:21 +020075 opt = getopt32(argv, "fnvsq");
Denis Vlasenko456fa6c2006-10-20 18:36:55 +000076 fname = argv[optind];
77 )
Denis Vlasenko5e34ff22009-04-21 11:09:40 +000078 IF_NOT_FEATURE_READLINK_FOLLOW(
Denis Vlasenko456fa6c2006-10-20 18:36:55 +000079 const unsigned opt = 0;
80 if (argc != 2) bb_show_usage();
81 fname = argv[1];
82 )
Mark Whitley872138d2000-10-09 18:56:47 +000083
Denis Vlasenko456fa6c2006-10-20 18:36:55 +000084 /* compat: coreutils readlink reports errors silently via exit code */
Denys Vlasenkoa99aa6e2009-06-12 13:16:21 +020085 if (!(opt & 4)) /* not -v */
86 logmode = LOGMODE_NONE;
Denis Vlasenko456fa6c2006-10-20 18:36:55 +000087
Denys Vlasenkoa99aa6e2009-06-12 13:16:21 +020088 if (opt & 1) { /* -f */
Jeremie Koenigb1754622010-05-27 15:32:19 +020089 buf = xmalloc_realpath(fname);
Denis Vlasenko456fa6c2006-10-20 18:36:55 +000090 } else {
Denis Vlasenko6ca04442007-02-11 16:19:28 +000091 buf = xmalloc_readlink_or_warn(fname);
Denis Vlasenko456fa6c2006-10-20 18:36:55 +000092 }
Ned Luddc6fbed52004-12-08 16:47:28 +000093
Eric Andersen28355a32001-05-07 17:48:28 +000094 if (!buf)
95 return EXIT_FAILURE;
Denys Vlasenkoa99aa6e2009-06-12 13:16:21 +020096 printf((opt & 2) ? "%s" : "%s\n", buf);
Rob Landleyb7128c62005-09-11 01:05:30 +000097
Jeremie Koenigb1754622010-05-27 15:32:19 +020098 if (ENABLE_FEATURE_CLEAN_UP)
Bernhard Reutner-Fischere15d7572006-06-02 20:56:16 +000099 free(buf);
Mark Whitley872138d2000-10-09 18:56:47 +0000100
Denis Vlasenkof0ed3762006-10-26 23:21:47 +0000101 fflush_stdout_and_exit(EXIT_SUCCESS);
Mark Whitley872138d2000-10-09 18:56:47 +0000102}