Mark Whitley | 872138d | 2000-10-09 18:56:47 +0000 | [diff] [blame] | 1 | /* vi: set sw=4 ts=4: */ |
| 2 | /* |
| 3 | * Mini readlink implementation for busybox |
| 4 | * |
Eric Andersen | bdfd0d7 | 2001-10-24 05:00:29 +0000 | [diff] [blame] | 5 | * Copyright (C) 2000,2001 Matt Kraai <kraai@alumni.carnegiemellon.edu> |
Mark Whitley | 872138d | 2000-10-09 18:56:47 +0000 | [diff] [blame] | 6 | * |
Denys Vlasenko | 0ef64bd | 2010-08-16 20:14:46 +0200 | [diff] [blame] | 7 | * Licensed under GPLv2 or later, see file LICENSE in this source tree. |
Mark Whitley | 872138d | 2000-10-09 18:56:47 +0000 | [diff] [blame] | 8 | */ |
Denys Vlasenko | af3f420 | 2016-11-23 14:46:56 +0100 | [diff] [blame] | 9 | //config:config READLINK |
Denys Vlasenko | 4eed2c6 | 2017-07-18 22:01:24 +0200 | [diff] [blame] | 10 | //config: bool "readlink (3.6 kb)" |
Denys Vlasenko | af3f420 | 2016-11-23 14:46:56 +0100 | [diff] [blame] | 11 | //config: default y |
| 12 | //config: help |
Denys Vlasenko | 72089cf | 2017-07-21 09:50:55 +0200 | [diff] [blame] | 13 | //config: This program reads a symbolic link and returns the name |
| 14 | //config: of the file it points to |
Denys Vlasenko | af3f420 | 2016-11-23 14:46:56 +0100 | [diff] [blame] | 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 |
Denys Vlasenko | 72089cf | 2017-07-21 09:50:55 +0200 | [diff] [blame] | 21 | //config: Enable the readlink option (-f). |
Denys Vlasenko | af3f420 | 2016-11-23 14:46:56 +0100 | [diff] [blame] | 22 | |
Denys Vlasenko | 39194f0 | 2017-08-03 19:00:01 +0200 | [diff] [blame] | 23 | //applet:IF_READLINK(APPLET_NOFORK(readlink, readlink, BB_DIR_USR_BIN, BB_SUID_DROP, readlink)) |
Denys Vlasenko | af3f420 | 2016-11-23 14:46:56 +0100 | [diff] [blame] | 24 | |
| 25 | //kbuild:lib-$(CONFIG_READLINK) += readlink.o |
Pere Orga | 3442538 | 2011-03-31 14:43:25 +0200 | [diff] [blame] | 26 | |
| 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 Orga | 3442538 | 2011-03-31 14:43:25 +0200 | [diff] [blame] | 32 | //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 Vlasenko | b6adbf1 | 2007-05-26 19:00:18 +0000 | [diff] [blame] | 37 | #include "libbb.h" |
| 38 | |
Denys Vlasenko | a99aa6e | 2009-06-12 13:16:21 +0200 | [diff] [blame] | 39 | /* |
| 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 Frysinger | 1b49c25 | 2013-03-12 11:38:03 -0400 | [diff] [blame] | 59 | * 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 Vlasenko | a99aa6e | 2009-06-12 13:16:21 +0200 | [diff] [blame] | 63 | */ |
| 64 | |
Denis Vlasenko | 9b49a5e | 2007-10-11 10:05:36 +0000 | [diff] [blame] | 65 | int readlink_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE; |
Denis Vlasenko | a60f84e | 2008-07-05 09:18:54 +0000 | [diff] [blame] | 66 | int readlink_main(int argc UNUSED_PARAM, char **argv) |
Mark Whitley | 872138d | 2000-10-09 18:56:47 +0000 | [diff] [blame] | 67 | { |
Rob Landley | ba50217 | 2005-09-11 23:45:28 +0000 | [diff] [blame] | 68 | char *buf; |
Denis Vlasenko | 456fa6c | 2006-10-20 18:36:55 +0000 | [diff] [blame] | 69 | char *fname; |
Mark Whitley | 8a63326 | 2001-04-30 18:17:00 +0000 | [diff] [blame] | 70 | |
Denis Vlasenko | 5e34ff2 | 2009-04-21 11:09:40 +0000 | [diff] [blame] | 71 | IF_FEATURE_READLINK_FOLLOW( |
Denis Vlasenko | 456fa6c | 2006-10-20 18:36:55 +0000 | [diff] [blame] | 72 | unsigned opt; |
| 73 | /* We need exactly one non-option argument. */ |
Denys Vlasenko | 22542ec | 2017-08-08 21:55:02 +0200 | [diff] [blame] | 74 | opt = getopt32(argv, "^" "fnvsq" "\0" "=1"); |
Denis Vlasenko | 456fa6c | 2006-10-20 18:36:55 +0000 | [diff] [blame] | 75 | fname = argv[optind]; |
| 76 | ) |
Denis Vlasenko | 5e34ff2 | 2009-04-21 11:09:40 +0000 | [diff] [blame] | 77 | IF_NOT_FEATURE_READLINK_FOLLOW( |
Denis Vlasenko | 456fa6c | 2006-10-20 18:36:55 +0000 | [diff] [blame] | 78 | const unsigned opt = 0; |
| 79 | if (argc != 2) bb_show_usage(); |
| 80 | fname = argv[1]; |
| 81 | ) |
Mark Whitley | 872138d | 2000-10-09 18:56:47 +0000 | [diff] [blame] | 82 | |
Denis Vlasenko | 456fa6c | 2006-10-20 18:36:55 +0000 | [diff] [blame] | 83 | /* compat: coreutils readlink reports errors silently via exit code */ |
Denys Vlasenko | a99aa6e | 2009-06-12 13:16:21 +0200 | [diff] [blame] | 84 | if (!(opt & 4)) /* not -v */ |
| 85 | logmode = LOGMODE_NONE; |
Denis Vlasenko | 456fa6c | 2006-10-20 18:36:55 +0000 | [diff] [blame] | 86 | |
Denys Vlasenko | 39194f0 | 2017-08-03 19:00:01 +0200 | [diff] [blame] | 87 | /* NOFORK: only one alloc is allowed; must free */ |
Denys Vlasenko | a99aa6e | 2009-06-12 13:16:21 +0200 | [diff] [blame] | 88 | if (opt & 1) { /* -f */ |
Jeremie Koenig | b175462 | 2010-05-27 15:32:19 +0200 | [diff] [blame] | 89 | buf = xmalloc_realpath(fname); |
Denis Vlasenko | 456fa6c | 2006-10-20 18:36:55 +0000 | [diff] [blame] | 90 | } else { |
Denis Vlasenko | 6ca0444 | 2007-02-11 16:19:28 +0000 | [diff] [blame] | 91 | buf = xmalloc_readlink_or_warn(fname); |
Denis Vlasenko | 456fa6c | 2006-10-20 18:36:55 +0000 | [diff] [blame] | 92 | } |
Ned Ludd | c6fbed5 | 2004-12-08 16:47:28 +0000 | [diff] [blame] | 93 | |
Eric Andersen | 28355a3 | 2001-05-07 17:48:28 +0000 | [diff] [blame] | 94 | if (!buf) |
| 95 | return EXIT_FAILURE; |
Denys Vlasenko | a99aa6e | 2009-06-12 13:16:21 +0200 | [diff] [blame] | 96 | printf((opt & 2) ? "%s" : "%s\n", buf); |
Denys Vlasenko | 39194f0 | 2017-08-03 19:00:01 +0200 | [diff] [blame] | 97 | free(buf); |
Mark Whitley | 872138d | 2000-10-09 18:56:47 +0000 | [diff] [blame] | 98 | |
Denis Vlasenko | f0ed376 | 2006-10-26 23:21:47 +0000 | [diff] [blame] | 99 | fflush_stdout_and_exit(EXIT_SUCCESS); |
Mark Whitley | 872138d | 2000-10-09 18:56:47 +0000 | [diff] [blame] | 100 | } |