1 /* seq - print sequence of numbers to standard output.
2 Copyright (C) 1994 Free Software Foundation, Inc.
4 This program is free software; you can redistribute it and/or modify
5 it under the terms of the GNU General Public License as published by
6 the Free Software Foundation; either version 2, or (at your option)
9 This program is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 GNU General Public License for more details.
14 You should have received a copy of the GNU General Public License
15 along with this program; if not, write to the Free Software
16 Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */
28 static double scan_double_arg ();
29 static int check_format ();
30 static char *get_width_format ();
31 static int print_numbers ();
33 /* If non-zero print all number with equal width. */
34 static int equal_width;
36 /* The printf(3) format used for output. */
37 static char *format_str;
39 /* The starting number. */
42 /* The name that this program was run with. */
45 /* The string used to seperate two number. */
46 static char *seperator;
48 /* If non-zero, display usage information and exit. */
51 /* If non-zero, print the version on standard output and exit. */
52 static int show_version;
57 /* The last number. */
60 static struct option const long_options[] =
62 { "equal-width", no_argument, NULL, 'w'},
63 { "format", required_argument, NULL, 'f'},
64 { "help", no_argument, &show_help, 1},
65 { "seperator", required_argument, NULL, 's'},
66 { "version", no_argument, &show_version, 1},
75 (void) fprintf (stderr, "Try `%s --help' for more information.\n",
80 Usage: %s [OPTION]... [from [step]] to\n\
84 -f, --format FORMAT use printf(3) style FORMAT (default: %%g)\n\
85 --help display this help and exit\n\
86 -s, --seperator STRING use STRING for seperating numbers (default: \\n)\n\
87 --version output version information and exit\n\
88 -w, --equal-width equalize width by padding with leading zeroes\n\
90 FROM, STEP, TO are interpreted as floating point. STEP should be > 0 if\n\
91 FROM is smaller than TO and vice versa. When given, the FORMAT argument\n\
92 must contain exactly one of the float output formats %%e, %%f, or %%g.\n\
107 program_name = argv[0];
114 /* We have to handle negative numbers in the command line but this
115 conflicts with the command line arguments. So the getopt mode is
116 REQUIRE_ORDER (the '+' in the format string) and it abort on the
117 first non-option or negative number. */
118 while ((optc = getopt_long (argc, argv, "+0123456789f:s:w", long_options,
121 if ('0' <= optc && optc <= '9')
123 /* means negative number */
152 (void) printf ("seq - %s\n", version_string);
163 /* FIXME Give reason for failure. */
167 last = scan_double_arg (argv[optind++]);
172 last = scan_double_arg (argv[optind++]);
178 last = scan_double_arg (argv[optind++]);
190 step = from < last ? 1.0 : -1.0;
193 if (format_str != NULL)
195 if (!check_format (format_str))
197 error (0, 0, "invalid format string: `%s'", format_str);
204 format_str = get_width_format ();
209 errs = print_numbers (format_str);
215 /* Read an double value from the command line.
216 Return if the string is correct else signal error. */
219 scan_double_arg (arg)
225 ret_val = strtod (arg, &end_ptr);
226 if (end_ptr == arg || *end_ptr != '\0')
228 error (0, 0, "invalid float argument: %s\n", arg);
236 /* Check whether the format string is valid for a single double
238 Return 0 if not, 1 if correct. */
241 check_format (format_string)
244 while (*format_string != '\0')
246 if (*format_string == '%')
249 if (*format_string != '%')
255 if (*format_string == '\0')
258 format_string += strspn (format_string, "-+#0");
259 if (isdigit (*format_string))
261 format_string += strspn (format_string, "012345789");
263 if (*format_string == '.')
264 format_string += strspn (++format_string, "0123456789");
267 if (*format_string != 'e' && *format_string != 'f' &&
268 *format_string != 'g')
272 while (*format_string != '\0')
274 if (*format_string == '%')
277 if (*format_string != '%')
287 /* Returns the format for that all printed numbers have the same width. */
291 static char buffer[256];
301 min_val = from - step * floor ((from - last) / step);
307 max_val = from + step * floor ((last - from) / step);
310 (void) sprintf (buffer, "%g", rint (max_val));
311 if (buffer[strspn (buffer, "0123456789")] != '\0')
313 width1 = strlen (buffer);
317 (void) sprintf (buffer, "%g", rint (min_val));
318 if (buffer[strspn (buffer, "-0123456789")] != '\0')
320 width2 = strlen (buffer);
322 width1 = width1 > width2 ? width1 : width2;
326 (void) sprintf (buffer, "%g", 1.0 + modf (min_val, &temp));
327 width1 = strlen (buffer);
332 if (buffer[0] != '1' || buffer[1] != '.' ||
333 buffer[2 + strspn (&buffer[2], "0123456789")] != '\0')
338 (void) sprintf (buffer, "%g", 1.0 + modf (step, &temp));
339 width2 = strlen (buffer);
344 if (buffer[0] != '1' || buffer[1] != '.' ||
345 buffer[2 + strspn (&buffer[2], "0123456789")] != '\0')
349 frac_width = width1 > width2 ? width1 : width2;
352 (void) sprintf (buffer, "%%0%d.%df", full_width + 1 + frac_width, frac_width);
354 (void) sprintf (buffer, "%%0%dg", full_width);
359 /* Actually print the sequence of numbers in the specified range, with the
360 given or default stepping and format. */
362 print_numbers (format_str)
369 error (0, 0, "invalid increment: %g\n", step);
376 (void) printf (format_str, from);
382 (void) fputs (seperator, stdout);
389 error (0, 0, "invalid increment: %g\n", step);
396 (void) printf (format_str, from);
402 (void) fputs (seperator, stdout);