1 /* vi: set sw=4 ts=4: */
3 * seq implementation for busybox
5 * Copyright (C) 2004, Glenn McGrath
7 * Licensed under GPLv2, see file LICENSE in this source tree.
10 //config: bool "seq (4 kb)"
13 //config: print a sequence of numbers
15 //applet:IF_SEQ(APPLET_NOEXEC(seq, seq, BB_DIR_USR_BIN, BB_SUID_DROP, seq))
16 /* was NOFORK, but then "seq 1 999999999" can't be ^C'ed if run by hush */
18 //kbuild:lib-$(CONFIG_SEQ) += seq.o
20 //usage:#define seq_trivial_usage
21 //usage: "[-w] [-s SEP] [FIRST [INC]] LAST"
22 //usage:#define seq_full_usage "\n\n"
23 //usage: "Print numbers from FIRST to LAST, in steps of INC.\n"
24 //usage: "FIRST, INC default to 1.\n"
25 //usage: "\n -w Pad with leading zeros"
26 //usage: "\n -s SEP String separator"
30 /* This is a NOEXEC applet. Be very careful! */
32 int seq_main(int argc
, char **argv
) MAIN_EXTERNALLY_VISIBLE
;
33 int seq_main(int argc
, char **argv
)
39 double first
, last
, increment
, v
;
43 const char *sep
, *opt_s
= "\n";
47 #if ENABLE_LOCALE_SUPPORT
48 /* Undo busybox.c: on input, we want to use dot
49 * as fractional separator, regardless of current locale */
50 setlocale(LC_NUMERIC
, "C");
53 /* Cater for negative arguments: if we see one, truncate argv[] on it */
60 if (saved
[0] != '-') {
61 // break; // "seq -s : -1 1" won't be treated correctly
64 // "seq -s -1 1 9" is not treated correctly, but such usage
65 // (delimiter string which looks like negative number) is very unlikely
67 if (c
== '.' || (c
>= '0' && c
<= '9')) {
72 opt
= getopt32(argv
, "+ws:", &opt_s
); /* "+": stop at first non-option */
73 /* Restore possibly truncated argv[] */
78 first
= increment
= 1;
83 increment
= strtod(argv
[1], &pp
);
86 first
= strtod(argv
[0], &pp
);
89 last
= strtod(argv
[argc
-1], &pp
);
90 if (!errno
&& *pp
== '\0')
96 #if ENABLE_LOCALE_SUPPORT
97 setlocale(LC_NUMERIC
, "");
100 /* Last checked to be compatible with: coreutils-6.10 */
104 char *dot
= strchrnul(*argv
, '.');
105 int w
= (dot
- *argv
);
112 /* Why do the above _before_ frac check below?
113 * Try "seq 1 2.0" and "seq 1.0 2.0":
114 * coreutils never pay attention to the number
115 * of fractional digits in last arg. */
122 width
+= frac_part
+ 1;
130 while (increment
>= 0 ? v
<= last
: v
>= last
) {
131 if (printf("%s%0*.*f", sep
, width
, frac_part
, v
) < 0)
132 break; /* I/O error, bail out (yes, this really happens) */
134 /* v += increment; - would accumulate floating point errors */
136 v
= first
+ n
* increment
;
138 if (n
) /* if while loop executed at least once */