123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217 |
- /* vi: set sw=4 ts=4: */
- /*
- * cat implementation for busybox
- *
- * Copyright (C) 2003 Manuel Novoa III <mjn3@codepoet.org>
- *
- * Licensed under GPLv2, see file LICENSE in this source tree.
- */
- //config:config CAT
- //config: bool "cat (5.8 kb)"
- //config: default y
- //config: help
- //config: cat is used to concatenate files and print them to the standard
- //config: output. Enable this option if you wish to enable the 'cat' utility.
- //config:
- //config:config FEATURE_CATN
- //config: bool "Enable -n and -b options"
- //config: default y
- //config: depends on CAT
- //config: help
- //config: -n numbers all output lines while -b numbers nonempty output lines.
- //config:
- //config:config FEATURE_CATV
- //config: bool "cat -v[etA]"
- //config: default y
- //config: depends on CAT
- //config: help
- //config: Display nonprinting characters as escape sequences
- //applet:IF_CAT(APPLET(cat, BB_DIR_BIN, BB_SUID_DROP))
- //kbuild:lib-$(CONFIG_CAT) += cat.o
- /* BB_AUDIT SUSv3 compliant */
- /* http://www.opengroup.org/onlinepubs/007904975/utilities/cat.html */
- //usage:#if ENABLE_FEATURE_CATN || ENABLE_FEATURE_CATV
- //usage:#define cat_trivial_usage
- //usage: "[-" IF_FEATURE_CATN("nb") IF_FEATURE_CATV("vteA") "] [FILE]..."
- //usage:#else
- //usage:#define cat_trivial_usage
- //usage: "[FILE]..."
- //usage:#endif
- //usage:#define cat_full_usage "\n\n"
- //usage: "Print FILEs to stdout\n"
- //usage: IF_FEATURE_CATN(
- //usage: "\n -n Number output lines"
- //usage: "\n -b Number nonempty lines"
- //usage: )
- //usage: IF_FEATURE_CATV(
- //usage: "\n -v Show nonprinting characters as ^x or M-x"
- //usage: "\n -t ...and tabs as ^I"
- //usage: "\n -e ...and end lines with $"
- //usage: "\n -A Same as -vte"
- //usage: )
- /*
- Longopts not implemented yet:
- --number-nonblank number nonempty output lines, overrides -n
- --number number all output lines
- --show-nonprinting use ^ and M- notation, except for LFD and TAB
- --show-all equivalent to -vet
- Not implemented yet:
- -E, --show-ends display $ at end of each line (-e sans -v)
- -T, --show-tabs display TAB characters as ^I (-t sans -v)
- -s, --squeeze-blank suppress repeated empty output lines
- */
- //usage:
- //usage:#define cat_example_usage
- //usage: "$ cat /proc/uptime\n"
- //usage: "110716.72 17.67"
- #include "libbb.h"
- #include "common_bufsiz.h"
- #if ENABLE_FEATURE_CATV
- /*
- * cat -v implementation for busybox
- *
- * Copyright (C) 2006 Rob Landley <rob@landley.net>
- *
- * Licensed under GPLv2 or later, see file LICENSE in this source tree.
- */
- /* Rob had "cat -v" implemented as a separate applet, catv.
- * See "cat -v considered harmful" at
- * http://cm.bell-labs.com/cm/cs/doc/84/kp.ps.gz
- * From USENIX Summer Conference Proceedings, 1983
- * """
- * The talk reviews reasons for UNIX's popularity and shows, using UCB cat
- * as a primary example, how UNIX has grown fat. cat isn't for printing
- * files with line numbers, it isn't for compressing multiple blank lines,
- * it's not for looking at non-printing ASCII characters, it's for
- * concatenating files.
- * We are reminded that ls isn't the place for code to break a single column
- * into multiple ones, and that mailnews shouldn't have its own more
- * processing or joke encryption code.
- * """
- *
- * I agree with the argument. Unfortunately, this ship has sailed (1983...).
- * There are dozens of Linux distros and each of them has "cat" which supports -v.
- * It's unrealistic for us to "reeducate" them to use our, incompatible way
- * to achieve "cat -v" effect. The actual effect would be "users pissed off
- * by gratuitous incompatibility".
- */
- #define CAT_OPT_e (1<<0)
- #define CAT_OPT_t (1<<1)
- #define CAT_OPT_v (1<<2)
- /* -A occupies bit (1<<3) */
- #define CAT_OPT_n ((1<<4) * ENABLE_FEATURE_CATN)
- #define CAT_OPT_b ((1<<5) * ENABLE_FEATURE_CATN)
- static int catv(unsigned opts, char **argv)
- {
- int retval = EXIT_SUCCESS;
- int fd;
- #if ENABLE_FEATURE_CATN
- bool eol_seen = (opts & (CAT_OPT_n|CAT_OPT_b));
- unsigned eol_char = (eol_seen ? '\n' : 0x100);
- unsigned skip_num_on = (opts & CAT_OPT_b) ? '\n' : 0x100;
- unsigned lineno = 0;
- #endif
- BUILD_BUG_ON(CAT_OPT_e != VISIBLE_ENDLINE);
- BUILD_BUG_ON(CAT_OPT_t != VISIBLE_SHOW_TABS);
- #if 0 /* These consts match, we can just pass "opts" to visible() */
- if (opts & CAT_OPT_e)
- flags |= VISIBLE_ENDLINE;
- if (opts & CAT_OPT_t)
- flags |= VISIBLE_SHOW_TABS;
- #endif
- #define read_buf bb_common_bufsiz1
- setup_common_bufsiz();
- do {
- fd = open_or_warn_stdin(*argv);
- if (fd < 0) {
- retval = EXIT_FAILURE;
- continue;
- }
- for (;;) {
- int i, res;
- res = read(fd, read_buf, COMMON_BUFSIZE);
- if (res < 0)
- retval = EXIT_FAILURE;
- if (res <= 0)
- break;
- for (i = 0; i < res; i++) {
- unsigned char c = read_buf[i];
- char buf[sizeof("M-^c")];
- #if ENABLE_FEATURE_CATN
- if (eol_seen && c != skip_num_on)
- printf("%6u ", ++lineno);
- eol_seen = (c == eol_char);
- #endif
- visible(c, buf, opts);
- fputs_stdout(buf);
- }
- }
- if (ENABLE_FEATURE_CLEAN_UP && fd)
- close(fd);
- } while (*++argv);
- fflush_stdout_and_exit(retval);
- }
- #undef CAT_OPT_n
- #undef CAT_OPT_b
- #endif
- int cat_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
- int cat_main(int argc UNUSED_PARAM, char **argv)
- {
- #if ENABLE_FEATURE_CATV || ENABLE_FEATURE_CATN
- unsigned opts;
- opts =
- #endif
- getopt32(argv, IF_FEATURE_CATV("^")
- /* -u is ignored ("unbuffered") */
- IF_FEATURE_CATV("etvA")IF_FEATURE_CATN("nb")"u"
- IF_FEATURE_CATV("\0" "Aetv" /* -A == -vet */)
- );
- argv += optind;
- /* Read from stdin if there's nothing else to do. */
- if (!argv[0])
- *--argv = (char*)"-";
- #if ENABLE_FEATURE_CATV
- if (opts & 7)
- return catv(opts, argv);
- opts >>= 4;
- #endif
- #if ENABLE_FEATURE_CATN
- # define CAT_OPT_n (1<<0)
- # define CAT_OPT_b (1<<1)
- if (opts & (CAT_OPT_n|CAT_OPT_b)) { /* -n or -b */
- struct number_state ns;
- int exitcode;
- ns.width = 6;
- ns.start = 1;
- ns.inc = 1;
- ns.sep = "\t";
- ns.empty_str = NULL;
- ns.all = !(opts & CAT_OPT_b); /* -n without -b */
- ns.nonempty = (opts & CAT_OPT_b); /* -b (with or without -n) */
- exitcode = EXIT_SUCCESS;
- do {
- exitcode |= print_numbered_lines(&ns, *argv);
- } while (*++argv);
- fflush_stdout_and_exit(exitcode);
- }
- /*opts >>= 2;*/
- #endif
- return bb_cat(argv);
- }
|