aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorDenis Vlasenko <vda.linux@googlemail.com>2007-08-13 14:10:24 +0000
committerDenis Vlasenko <vda.linux@googlemail.com>2007-08-13 14:10:24 +0000
commit3952f20c24ff3eed2ef6f4a9abd2d8a0aaa34a6e (patch)
tree038567c7d5d0a3c9bfe40eeb534ffc9559423b9d
parent08294dbf5bc6fb1ea0b185488675d91169e231ce (diff)
downloadbusybox-w32-3952f20c24ff3eed2ef6f4a9abd2d8a0aaa34a6e.tar.gz
busybox-w32-3952f20c24ff3eed2ef6f4a9abd2d8a0aaa34a6e.tar.bz2
busybox-w32-3952f20c24ff3eed2ef6f4a9abd2d8a0aaa34a6e.zip
expand, unexpand: new applets from Tito <farmatito@tiscali.it>
-rw-r--r--coreutils/Config.in26
-rw-r--r--coreutils/Kbuild2
-rw-r--r--coreutils/expand.c209
-rw-r--r--coreutils/expr.c2
-rw-r--r--include/applets.h2
-rw-r--r--include/usage.h22
-rw-r--r--libbb/fflush_stdout_and_exit.c2
-rw-r--r--scripts/defconfig4
-rw-r--r--testsuite/expand/expand-works-like-GNU18
-rw-r--r--testsuite/unexpand/unexpand-works-like-GNU52
10 files changed, 336 insertions, 3 deletions
diff --git a/coreutils/Config.in b/coreutils/Config.in
index cb5241ef6..3370b2a5d 100644
--- a/coreutils/Config.in
+++ b/coreutils/Config.in
@@ -200,6 +200,19 @@ config FEATURE_ENV_LONG_OPTIONS
200 help 200 help
201 Support long options for the env applet. 201 Support long options for the env applet.
202 202
203config EXPAND
204 bool "expand"
205 default n
206 help
207 By default, convert all tabs to spaces.
208
209config FEATURE_EXPAND_LONG_OPTIONS
210 bool "Enable long options"
211 default n
212 depends on EXPAND && GETOPT_LONG
213 help
214 Support long options for the expand applet.
215
203config EXPR 216config EXPR
204 bool "expr" 217 bool "expr"
205 default n 218 default n
@@ -655,6 +668,19 @@ config UNAME
655 help 668 help
656 uname is used to print system information. 669 uname is used to print system information.
657 670
671config UNEXPAND
672 bool "unexpand"
673 default n
674 help
675 By default, convert only leading sequences of blanks to tabs.
676
677config FEATURE_UNEXPAND_LONG_OPTIONS
678 bool "Enable long options"
679 default n
680 depends on UNEXPAND && GETOPT_LONG
681 help
682 Support long options for the unexpand applet.
683
658config UNIQ 684config UNIQ
659 bool "uniq" 685 bool "uniq"
660 default n 686 default n
diff --git a/coreutils/Kbuild b/coreutils/Kbuild
index fd67d6c85..ce21b3a89 100644
--- a/coreutils/Kbuild
+++ b/coreutils/Kbuild
@@ -31,6 +31,7 @@ lib-$(CONFIG_ECHO) += echo.o
31lib-$(CONFIG_ASH) += echo.o # used by ash 31lib-$(CONFIG_ASH) += echo.o # used by ash
32lib-$(CONFIG_ENV) += env.o 32lib-$(CONFIG_ENV) += env.o
33lib-$(CONFIG_EXPR) += expr.o 33lib-$(CONFIG_EXPR) += expr.o
34lib-$(CONFIG_EXPAND) += expand.o
34lib-$(CONFIG_FALSE) += false.o 35lib-$(CONFIG_FALSE) += false.o
35lib-$(CONFIG_FOLD) += fold.o 36lib-$(CONFIG_FOLD) += fold.o
36lib-$(CONFIG_HEAD) += head.o 37lib-$(CONFIG_HEAD) += head.o
@@ -74,6 +75,7 @@ lib-$(CONFIG_TR) += tr.o
74lib-$(CONFIG_TRUE) += true.o 75lib-$(CONFIG_TRUE) += true.o
75lib-$(CONFIG_TTY) += tty.o 76lib-$(CONFIG_TTY) += tty.o
76lib-$(CONFIG_UNAME) += uname.o 77lib-$(CONFIG_UNAME) += uname.o
78lib-$(CONFIG_UNEXPAND) += expand.o
77lib-$(CONFIG_UNIQ) += uniq.o 79lib-$(CONFIG_UNIQ) += uniq.o
78lib-$(CONFIG_USLEEP) += usleep.o 80lib-$(CONFIG_USLEEP) += usleep.o
79lib-$(CONFIG_UUDECODE) += uudecode.o 81lib-$(CONFIG_UUDECODE) += uudecode.o
diff --git a/coreutils/expand.c b/coreutils/expand.c
new file mode 100644
index 000000000..30815ff63
--- /dev/null
+++ b/coreutils/expand.c
@@ -0,0 +1,209 @@
1/* expand - convert tabs to spaces
2 * unexpand - convert spaces to tabs
3 *
4 * Copyright (C) 89, 91, 1995-2006 Free Software Foundation, Inc.
5 *
6 * Licensed under GPLv2 or later, see file LICENSE in this tarball for details.
7 *
8 * David MacKenzie <djm@gnu.ai.mit.edu>
9 *
10 * Options for expand:
11 * -t num --tabs=NUM Convert tabs to num spaces (default 8 spaces).
12 * -i --initial Only convert initial tabs on each line to spaces.
13 *
14 * Options for unexpand:
15 * -a --all Convert all blanks, instead of just initial blanks.
16 * -f --first-only Convert only leading sequences of blanks (default).
17 * -t num --tabs=NUM Have tabs num characters apart instead of 8.
18 *
19 * Busybox version (C) 2007 by Tito Ragusa <farmatito@tiscali.it>
20 *
21 * Caveat: this versions of expand and unexpand don't accept tab lists.
22 */
23
24#include "libbb.h"
25
26enum {
27 OPT_INITIAL = 1 << 0,
28 OPT_TABS = 1 << 1,
29 OPT_ALL = 1 << 2,
30};
31
32static void xputchar(char c)
33{
34 if (putchar(c) < 0)
35 bb_error_msg_and_die(bb_msg_write_error);
36}
37
38#if ENABLE_EXPAND
39static void expand(FILE *file, unsigned tab_size, unsigned opt)
40{
41 char *line;
42 char *ptr;
43 int convert;
44 int pos;
45
46 /* Increment tab_size by 1 locally.*/
47 tab_size++;
48
49 while ((line = xmalloc_fgets(file)) != NULL) {
50 convert = 1;
51 pos = 0;
52 ptr = line;
53 while (*line) {
54 pos++;
55 if (*line == '\t' && convert) {
56 for (; pos < tab_size; pos++) {
57 xputchar(' ');
58 }
59 } else {
60 if ((opt & OPT_INITIAL) && !isblank(*line)) {
61 convert = 0;
62 }
63 xputchar(*line);
64 }
65 if (pos == tab_size) {
66 pos = 0;
67 }
68 line++;
69 }
70 free(ptr);
71 }
72}
73#endif
74
75#if ENABLE_UNEXPAND
76static void unexpand(FILE *file, unsigned int tab_size, unsigned opt)
77{
78 char *line;
79 char *ptr;
80 int convert;
81 int pos;
82 int i = 0;
83 int column = 0;
84
85 while ((line = xmalloc_fgets(file)) != NULL) {
86 convert = 1;
87 pos = 0;
88 ptr = line;
89 while (*line) {
90 while ((*line == ' ' || *line == '\t') && convert) {
91 pos += (*line == ' ') ? 1 : tab_size;
92 line++;
93 column++;
94 if ((opt & OPT_ALL) && column == tab_size) {
95 column = 0;
96 goto put_tab;
97 }
98 }
99 if (pos) {
100 i = pos / tab_size;
101 if (i) {
102 for (; i > 0; i--) {
103 put_tab:
104 xputchar('\t');
105 }
106 } else {
107 for (i = pos % tab_size; i > 0; i--) {
108 xputchar(' ');
109 }
110 }
111 pos = 0;
112 } else {
113 if (opt & OPT_INITIAL) {
114 convert = 0;
115 }
116 if (opt & OPT_ALL) {
117 column++;
118 }
119 xputchar(*line);
120 line++;
121 }
122 }
123 free(ptr);
124 }
125}
126#endif
127
128int expand_main(int argc, char **argv);
129int expand_main(int argc, char **argv)
130{
131 /* Default 8 spaces for 1 tab */
132 const char *opt_t = "8";
133 FILE *file;
134 unsigned tab_size;
135 unsigned opt;
136 int exit_status = EXIT_SUCCESS;
137
138#if ENABLE_FEATURE_EXPAND_LONG_OPTIONS
139 static const char expand_longopts[] ALIGN1 =
140 /* name, has_arg, val */
141 "initial\0" No_argument "i"
142 "tabs\0" Required_argument "t"
143 ;
144#endif
145#if ENABLE_FEATURE_UNEXPAND_LONG_OPTIONS
146 static const char unexpand_longopts[] ALIGN1 =
147 /* name, has_arg, val */
148 "first-only\0" No_argument "i"
149 "tabs\0" Required_argument "t"
150 "all\0" No_argument "a"
151 ;
152#endif
153
154 if (ENABLE_EXPAND && (!ENABLE_UNEXPAND || applet_name[0] == 'e')) {
155 USE_FEATURE_EXPAND_LONG_OPTIONS(applet_long_options = expand_longopts);
156 opt = getopt32(argc, argv, "it:", &opt_t);
157 } else if (ENABLE_UNEXPAND) {
158 USE_FEATURE_UNEXPAND_LONG_OPTIONS(applet_long_options = unexpand_longopts);
159 /* -t NUM sets also -a */
160 opt_complementary = "ta";
161 opt = getopt32(argc, argv, "ft:a", &opt_t);
162 /* -f --first-only is the default */
163 if (!(opt & OPT_ALL)) opt |= OPT_INITIAL;
164 }
165 tab_size = xatou_range(opt_t, 1, UINT_MAX);
166
167 argv += optind;
168
169 /* If no args are given, read from stdin */
170 if (!*argv) {
171 *--argv = (char*)bb_msg_standard_input;
172 goto use_stdin;
173 }
174
175 do {
176 if (NOT_LONE_CHAR(*argv, '-')) {
177 file = fopen_or_warn(*argv, "r");
178 if (!file) {
179 exit_status = EXIT_FAILURE;
180 continue;
181 }
182 } else {
183 use_stdin:
184 file = stdin;
185 }
186
187 if (ENABLE_EXPAND && (!ENABLE_UNEXPAND || applet_name[0] == 'e'))
188 USE_EXPAND(expand(file, tab_size, opt));
189 else if (ENABLE_UNEXPAND)
190 USE_UNEXPAND(unexpand(file, tab_size, opt));
191
192 /* Check and close the file */
193 /* We do want all of them to execute, thus | instead of || */
194 if (ferror(file) | fclose_if_not_stdin(file)) {
195 bb_perror_msg("%s", *argv);
196 exit_status = EXIT_FAILURE;
197 }
198 /* If stdin also clear EOF */
199 if (file == stdin)
200 clearerr(file);
201 } while (*++argv);
202
203 /* Now close stdin also */
204 /* (if we didn't read from it, it's a no-op) */
205 if (fclose(stdin))
206 bb_perror_msg_and_die(bb_msg_standard_input);
207
208 fflush_stdout_and_exit(exit_status);
209}
diff --git a/coreutils/expr.c b/coreutils/expr.c
index 318fee721..c2d966966 100644
--- a/coreutils/expr.c
+++ b/coreutils/expr.c
@@ -505,5 +505,3 @@ int expr_main(int argc, char **argv)
505 505
506 fflush_stdout_and_exit(null(v)); 506 fflush_stdout_and_exit(null(v));
507} 507}
508
509
diff --git a/include/applets.h b/include/applets.h
index b5b9fc884..f92ce4b13 100644
--- a/include/applets.h
+++ b/include/applets.h
@@ -134,6 +134,7 @@ USE_ENV(APPLET(env, _BB_DIR_USR_BIN, _BB_SUID_NEVER))
134USE_ENVDIR(APPLET_ODDNAME(envdir, chpst, _BB_DIR_USR_BIN, _BB_SUID_NEVER, envdir)) 134USE_ENVDIR(APPLET_ODDNAME(envdir, chpst, _BB_DIR_USR_BIN, _BB_SUID_NEVER, envdir))
135USE_ENVUIDGID(APPLET_ODDNAME(envuidgid, chpst, _BB_DIR_USR_BIN, _BB_SUID_NEVER, envuidgid)) 135USE_ENVUIDGID(APPLET_ODDNAME(envuidgid, chpst, _BB_DIR_USR_BIN, _BB_SUID_NEVER, envuidgid))
136USE_ETHER_WAKE(APPLET_ODDNAME(ether-wake, ether_wake, _BB_DIR_USR_BIN, _BB_SUID_NEVER, ether_wake)) 136USE_ETHER_WAKE(APPLET_ODDNAME(ether-wake, ether_wake, _BB_DIR_USR_BIN, _BB_SUID_NEVER, ether_wake))
137USE_EXPAND(APPLET(expand, _BB_DIR_USR_BIN, _BB_SUID_NEVER))
137USE_EXPR(APPLET(expr, _BB_DIR_USR_BIN, _BB_SUID_NEVER)) 138USE_EXPR(APPLET(expr, _BB_DIR_USR_BIN, _BB_SUID_NEVER))
138USE_FAKEIDENTD(APPLET(fakeidentd, _BB_DIR_USR_SBIN, _BB_SUID_NEVER)) 139USE_FAKEIDENTD(APPLET(fakeidentd, _BB_DIR_USR_SBIN, _BB_SUID_NEVER))
139USE_FALSE(APPLET_NOFORK(false, false, _BB_DIR_BIN, _BB_SUID_NEVER, false)) 140USE_FALSE(APPLET_NOFORK(false, false, _BB_DIR_BIN, _BB_SUID_NEVER, false))
@@ -345,6 +346,7 @@ USE_UDPSVD(APPLET_ODDNAME(udpsvd, tcpudpsvd, _BB_DIR_USR_BIN, _BB_SUID_NEVER, ud
345USE_UMOUNT(APPLET(umount, _BB_DIR_BIN, _BB_SUID_NEVER)) 346USE_UMOUNT(APPLET(umount, _BB_DIR_BIN, _BB_SUID_NEVER))
346USE_UNAME(APPLET(uname, _BB_DIR_BIN, _BB_SUID_NEVER)) 347USE_UNAME(APPLET(uname, _BB_DIR_BIN, _BB_SUID_NEVER))
347USE_UNCOMPRESS(APPLET(uncompress, _BB_DIR_BIN, _BB_SUID_NEVER)) 348USE_UNCOMPRESS(APPLET(uncompress, _BB_DIR_BIN, _BB_SUID_NEVER))
349USE_UNEXPAND(APPLET_ODDNAME(unexpand, expand, _BB_DIR_USR_BIN, _BB_SUID_NEVER, unexpand))
348USE_UNIQ(APPLET(uniq, _BB_DIR_USR_BIN, _BB_SUID_NEVER)) 350USE_UNIQ(APPLET(uniq, _BB_DIR_USR_BIN, _BB_SUID_NEVER))
349USE_UNIX2DOS(APPLET_ODDNAME(unix2dos, dos2unix, _BB_DIR_USR_BIN, _BB_SUID_NEVER, unix2dos)) 351USE_UNIX2DOS(APPLET_ODDNAME(unix2dos, dos2unix, _BB_DIR_USR_BIN, _BB_SUID_NEVER, unix2dos))
350USE_UNLZMA(APPLET(unlzma, _BB_DIR_USR_BIN, _BB_SUID_NEVER)) 352USE_UNLZMA(APPLET(unlzma, _BB_DIR_USR_BIN, _BB_SUID_NEVER))
diff --git a/include/usage.h b/include/usage.h
index f5025b70a..d01e3e4f3 100644
--- a/include/usage.h
+++ b/include/usage.h
@@ -844,6 +844,16 @@
844 " -i iface Use interface ifname instead of the default \"eth0\"\n" \ 844 " -i iface Use interface ifname instead of the default \"eth0\"\n" \
845 " -p pass Append the four or six byte password PW to the packet" 845 " -p pass Append the four or six byte password PW to the packet"
846 846
847#define expand_trivial_usage \
848 "[-i] [-t NUM] [FILE|-]"
849#define expand_full_usage \
850 "Convert tabs to spaces, writing to standard output." \
851 "\n\nOptions:" \
852 "\n -i" USE_FEATURE_EXPAND_LONG_OPTIONS(",--initial") \
853 " Do not convert tabs after non blanks" \
854 "\n -t" USE_FEATURE_EXPAND_LONG_OPTIONS(",--tabs=N") \
855 " Tabstops every N chars"
856
847#define expr_trivial_usage \ 857#define expr_trivial_usage \
848 "EXPRESSION" 858 "EXPRESSION"
849#define expr_full_usage \ 859#define expr_full_usage \
@@ -3707,6 +3717,18 @@ USE_FEATURE_RUN_PARTS_FANCY("\n -l Prints names of all matching files even when
3707 " -c Extract to stdout\n" \ 3717 " -c Extract to stdout\n" \
3708 " -f Force overwrite an existing file" 3718 " -f Force overwrite an existing file"
3709 3719
3720#define unexpand_trivial_usage \
3721 "[-f][-a][-t NUM] [FILE|-]"
3722#define unexpand_full_usage \
3723 "Convert spaces to tabs, writing to standard output." \
3724 "\n\nOptions:" \
3725 "\n -a" USE_FEATURE_UNEXPAND_LONG_OPTIONS(",--all") \
3726 " Convert all blanks" \
3727 "\n -f" USE_FEATURE_UNEXPAND_LONG_OPTIONS(",--first-only") \
3728 " Convert only leading sequences of blanks" \
3729 "\n -t" USE_FEATURE_UNEXPAND_LONG_OPTIONS(",--tabs=N") \
3730 " Tabstops every N chars"
3731
3710#define uniq_trivial_usage \ 3732#define uniq_trivial_usage \
3711 "[-fscdu]... [INPUT [OUTPUT]]" 3733 "[-fscdu]... [INPUT [OUTPUT]]"
3712#define uniq_full_usage \ 3734#define uniq_full_usage \
diff --git a/libbb/fflush_stdout_and_exit.c b/libbb/fflush_stdout_and_exit.c
index 48889dae8..9f05500f3 100644
--- a/libbb/fflush_stdout_and_exit.c
+++ b/libbb/fflush_stdout_and_exit.c
@@ -16,7 +16,7 @@
16void fflush_stdout_and_exit(int retval) 16void fflush_stdout_and_exit(int retval)
17{ 17{
18 if (fflush(stdout)) 18 if (fflush(stdout))
19 xfunc_die(); 19 bb_perror_msg_and_die(bb_msg_standard_output);
20 20
21 if (ENABLE_FEATURE_PREFER_APPLETS && die_sleep < 0) { 21 if (ENABLE_FEATURE_PREFER_APPLETS && die_sleep < 0) {
22 /* We are in NOFORK applet. Do not exit() directly, 22 /* We are in NOFORK applet. Do not exit() directly,
diff --git a/scripts/defconfig b/scripts/defconfig
index 4709b2234..f24fe9586 100644
--- a/scripts/defconfig
+++ b/scripts/defconfig
@@ -152,6 +152,8 @@ CONFIG_ECHO=y
152CONFIG_FEATURE_FANCY_ECHO=y 152CONFIG_FEATURE_FANCY_ECHO=y
153CONFIG_ENV=y 153CONFIG_ENV=y
154CONFIG_FEATURE_ENV_LONG_OPTIONS=y 154CONFIG_FEATURE_ENV_LONG_OPTIONS=y
155CONFIG_EXPAND=y
156CONFIG_FEATURE_EXPAND_LONG_OPTIONS=y
155CONFIG_EXPR=y 157CONFIG_EXPR=y
156CONFIG_EXPR_MATH_SUPPORT_64=y 158CONFIG_EXPR_MATH_SUPPORT_64=y
157CONFIG_FALSE=y 159CONFIG_FALSE=y
@@ -218,6 +220,8 @@ CONFIG_FEATURE_TR_EQUIV=y
218CONFIG_TRUE=y 220CONFIG_TRUE=y
219CONFIG_TTY=y 221CONFIG_TTY=y
220CONFIG_UNAME=y 222CONFIG_UNAME=y
223CONFIG_UNEXPAND=y
224CONFIG_FEATURE_UNEXPAND_LONG_OPTIONS=y
221CONFIG_UNIQ=y 225CONFIG_UNIQ=y
222CONFIG_USLEEP=y 226CONFIG_USLEEP=y
223CONFIG_UUDECODE=y 227CONFIG_UUDECODE=y
diff --git a/testsuite/expand/expand-works-like-GNU b/testsuite/expand/expand-works-like-GNU
new file mode 100644
index 000000000..ee8c793ed
--- /dev/null
+++ b/testsuite/expand/expand-works-like-GNU
@@ -0,0 +1,18 @@
1rm -f foo bar
2echo -e "\ty" | expand -t 3 ../../busybox > foo
3echo -e "\ty" | busybox unexpand -t 3 ../../busybox > bar
4set +e
5test ! -f foo -a -f bar
6if [ $? = 0 ] ; then
7 set -e
8 diff -q foo bar
9fi
10rm -f foo bar
11echo -e "\ty\tx" | expand -it 3 ../../busybox > foo
12echo -e "\ty\tx" | busybox unexpand -it 3 ../../busybox > bar
13set +e
14test ! -f foo -a -f bar
15if [ $? = 0 ] ; then
16 set -e
17 diff -q foo bar
18fi
diff --git a/testsuite/unexpand/unexpand-works-like-GNU b/testsuite/unexpand/unexpand-works-like-GNU
new file mode 100644
index 000000000..a5258363f
--- /dev/null
+++ b/testsuite/unexpand/unexpand-works-like-GNU
@@ -0,0 +1,52 @@
1rm -f foo bar
2echo " y" | unexpand ../../busybox > foo
3echo " y" | busybox unexpand ../../busybox > bar
4set +e
5test ! -f foo -a -f bar
6if [ $? = 0 ] ; then
7 set -e
8 diff -q foo bar
9fi
10rm -f foo bar
11echo " y" | unexpand ../../busybox > foo
12echo " y" | busybox unexpand ../../busybox > bar
13set +e
14test ! -f foo -a -f bar
15if [ $? = 0 ] ; then
16 set -e
17 diff -q foo bar
18fi
19echo " y y" | unexpand ../../busybox > foo
20echo " y y" | busybox unexpand ../../busybox > bar
21set +e
22test ! -f foo -a -f bar
23if [ $? = 0 ] ; then
24 set -e
25 diff -q foo bar
26fi
27rm -f foo bar
28echo " y y" | unexpand ../../busybox > foo
29echo " y y" | busybox unexpand ../../busybox > bar
30set +e
31test ! -f foo -a -f bar
32if [ $? = 0 ] ; then
33 set -e
34 diff -q foo bar
35fi
36echo " y y" | unexpand -a ../../busybox > foo
37echo " y y" | busybox unexpand -a ../../busybox > bar
38set +e
39test ! -f foo -a -f bar
40if [ $? = 0 ] ; then
41 set -e
42 diff -q foo bar
43fi
44rm -f foo bar
45echo " y y" | unexpand -a ../../busybox > foo
46echo " y y" | busybox unexpand -a ../../busybox > bar
47set +e
48test ! -f foo -a -f bar
49if [ $? = 0 ] ; then
50 set -e
51 diff -q foo bar
52fi