101 lines
		
	
	
		
			3.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			101 lines
		
	
	
		
			3.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/* vi: set sw=4 ts=4: */
 | 
						|
/*
 | 
						|
 * Mini readlink implementation for busybox
 | 
						|
 *
 | 
						|
 * Copyright (C) 2000,2001 Matt Kraai <kraai@alumni.carnegiemellon.edu>
 | 
						|
 *
 | 
						|
 * Licensed under GPLv2 or later, see file LICENSE in this source tree.
 | 
						|
 */
 | 
						|
//config:config READLINK
 | 
						|
//config:	bool "readlink (4 kb)"
 | 
						|
//config:	default y
 | 
						|
//config:	help
 | 
						|
//config:	This program reads a symbolic link and returns the name
 | 
						|
//config:	of the file it points to
 | 
						|
//config:
 | 
						|
//config:config FEATURE_READLINK_FOLLOW
 | 
						|
//config:	bool "Enable canonicalization by following all symlinks (-f)"
 | 
						|
//config:	default y
 | 
						|
//config:	depends on READLINK
 | 
						|
//config:	help
 | 
						|
//config:	Enable the readlink option (-f).
 | 
						|
 | 
						|
//applet:IF_READLINK(APPLET_NOFORK(readlink, readlink, BB_DIR_USR_BIN, BB_SUID_DROP, readlink))
 | 
						|
 | 
						|
//kbuild:lib-$(CONFIG_READLINK) += readlink.o
 | 
						|
 | 
						|
//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:	)
 | 
						|
 | 
						|
#include "libbb.h"
 | 
						|
 | 
						|
/*
 | 
						|
 * # 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
 | 
						|
 *
 | 
						|
 * 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 :(.
 | 
						|
 */
 | 
						|
 | 
						|
int readlink_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
 | 
						|
int readlink_main(int argc UNUSED_PARAM, char **argv)
 | 
						|
{
 | 
						|
	char *buf;
 | 
						|
	char *fname;
 | 
						|
 | 
						|
	IF_FEATURE_READLINK_FOLLOW(
 | 
						|
		unsigned opt;
 | 
						|
		/* We need exactly one non-option argument.  */
 | 
						|
		opt = getopt32(argv, "^" "fnvsq" "\0" "=1");
 | 
						|
		fname = argv[optind];
 | 
						|
	)
 | 
						|
	IF_NOT_FEATURE_READLINK_FOLLOW(
 | 
						|
		const unsigned opt = 0;
 | 
						|
		if (argc != 2) bb_show_usage();
 | 
						|
		fname = argv[1];
 | 
						|
	)
 | 
						|
 | 
						|
	/* compat: coreutils readlink reports errors silently via exit code */
 | 
						|
	if (!(opt & 4)) /* not -v */
 | 
						|
		logmode = LOGMODE_NONE;
 | 
						|
 | 
						|
	/* NOFORK: only one alloc is allowed; must free */
 | 
						|
	if (opt & 1) { /* -f */
 | 
						|
		buf = xmalloc_realpath_coreutils(fname);
 | 
						|
	} else {
 | 
						|
		buf = xmalloc_readlink_or_warn(fname);
 | 
						|
	}
 | 
						|
 | 
						|
	if (!buf)
 | 
						|
		return EXIT_FAILURE;
 | 
						|
	printf((opt & 2) ? "%s" : "%s\n", buf);
 | 
						|
	free(buf);
 | 
						|
 | 
						|
	fflush_stdout_and_exit(EXIT_SUCCESS);
 | 
						|
}
 |