summaryrefslogtreecommitdiff
path: root/coreutils
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 /coreutils
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>
Diffstat (limited to 'coreutils')
-rw-r--r--coreutils/Config.in26
-rw-r--r--coreutils/Kbuild2
-rw-r--r--coreutils/expand.c209
-rw-r--r--coreutils/expr.c2
4 files changed, 237 insertions, 2 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