]> git.saurik.com Git - bison.git/blame - src/getargs.c
tests: check token numbers.
[bison.git] / src / getargs.c
CommitLineData
08721544
PE
1/* Parse command line arguments for Bison.
2
e2a21b6f 3 Copyright (C) 1984, 1986, 1989, 1992, 2000, 2001, 2002, 2003, 2004,
b3a2272a 4 2005, 2006, 2007, 2008, 2009 Free Software Foundation, Inc.
3d8fc6ca 5
9f306f2a 6 This file is part of Bison, the GNU Compiler Compiler.
3d8fc6ca 7
f16b0819
PE
8 This program is free software: you can redistribute it and/or modify
9 it under the terms of the GNU General Public License as published by
10 the Free Software Foundation, either version 3 of the License, or
11 (at your option) any later version.
3d8fc6ca 12
f16b0819
PE
13 This program is distributed in the hope that it will be useful,
14 but WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 GNU General Public License for more details.
3d8fc6ca 17
9f306f2a 18 You should have received a copy of the GNU General Public License
f16b0819 19 along with this program. If not, see <http://www.gnu.org/licenses/>. */
3d8fc6ca 20
2cec9080 21#include <config.h>
3d8fc6ca 22#include "system.h"
d4bd2295 23#include "output.h"
d38a11a6
PE
24
25#include <argmatch.h>
d7e0a1a7 26#include <c-strcase.h>
3b2942e6 27#include <configmake.h>
d38a11a6 28#include <error.h>
9fe5a457 29#include <quotearg.h>
f47dbf6b
PE
30
31/* Hack to get <getopt.h> to declare getopt with a prototype. */
32#if lint && ! defined __GNU_LIBRARY__
33# define __GNU_LIBRARY__
34# define HACK_FOR___GNU_LIBRARY___PROTOTYPE 1
35#endif
36
d38a11a6
PE
37#include <getopt.h>
38
f47dbf6b
PE
39#ifdef HACK_FOR___GNU_LIBRARY___PROTOTYPE
40# undef __GNU_LIBRARY__
41# undef HACK_FOR___GNU_LIBRARY___PROTOTYPE
42#endif
43
b0ce6046 44#include "complain.h"
ec3bc396 45#include "files.h"
d38a11a6 46#include "getargs.h"
00f5d575 47#include "muscle-tab.h"
d38a11a6 48#include "uniqstr.h"
3d8fc6ca 49
d0829076 50bool defines_flag;
4e83ea15 51bool graph_flag;
41d7a5f2 52bool xml_flag;
d0829076 53bool no_lines_flag;
d0829076
PE
54bool token_table_flag;
55bool yacc_flag; /* for -y */
4e83ea15 56
916708d5
AD
57bool nondeterministic_parser = false;
58bool glr_parser = false;
916708d5 59
4e83ea15
AD
60int report_flag = report_none;
61int trace_flag = trace_none;
7b42569e 62int warnings_flag = warnings_none;
4e83ea15 63
0e021770
PE
64static struct bison_language const valid_languages[] = {
65 { "c", "c-skel.m4", ".c", ".h", true },
66 { "c++", "c++-skel.m4", ".cc", ".hh", true },
8405b70c 67 { "java", "java-skel.m4", ".java", ".java", false },
0e021770
PE
68 { "", "", "", "", false }
69};
70
51365192 71int skeleton_prio = default_prio;
b0ce6046 72const char *skeleton = NULL;
51365192 73int language_prio = default_prio;
0e021770 74struct bison_language const *language = &valid_languages[0];
f6bd5427 75const char *include = NULL;
b0ce6046 76
0e021770 77char *program_name;
3d8fc6ca 78
273a74fa 79
7b42569e
AD
80/** Decode an option's set of keys.
81 *
82 * \param option option being decoded.
07c39ae9 83 * \param keys array of valid subarguments.
7b42569e 84 * \param values array of corresponding (int) values.
07c39ae9 85 * \param flags the flags to update
4182a0a1 86 * \param args comma separated list of effective subarguments to decode.
7b42569e
AD
87 * If 0, then activate all the flags.
88 *
4182a0a1
DJ
89 * If VALUE != 0 then KEY sets flags and no-KEY clears them.
90 * If VALUE == 0 then KEY clears all flags and no-KEY sets all flags.
91 * Thus no-none = all and no-all = none.
7b42569e 92 */
80ac75bc
PE
93static void
94flags_argmatch (const char *option,
7b42569e
AD
95 const char * const keys[], const int values[],
96 int *flags, char *args)
97{
98 if (args)
99 {
100 args = strtok (args, ",");
ba7560e2 101 while (args)
7b42569e 102 {
4182a0a1
DJ
103 int no = strncmp (args, "no-", 3) == 0 ? 3 : 0;
104 int value = XARGMATCH (option, args + no, keys, values);
7b42569e 105 if (value == 0)
4182a0a1
DJ
106 if (no)
107 *flags = ~0;
108 else
109 *flags = 0;
7b42569e 110 else
4182a0a1
DJ
111 if (no)
112 *flags &= ~value;
113 else
114 *flags |= value;
eb095650 115 args = strtok (NULL, ",");
7b42569e 116 }
7b42569e
AD
117 }
118 else
119 *flags = ~0;
120}
121
80ac75bc 122/** Decode a set of sub arguments.
7b42569e
AD
123 *
124 * \param FlagName the flag familly to update.
07c39ae9 125 * \param Args the effective sub arguments to decode.
7b42569e
AD
126 *
127 * \arg FlagName_args the list of keys.
128 * \arg FlagName_types the list of values.
129 * \arg FlagName_flag the flag to update.
130 */
131#define FLAGS_ARGMATCH(FlagName, Args) \
132 flags_argmatch ("--" #FlagName, FlagName ## _args, FlagName ## _types, \
133 &FlagName ## _flag, Args)
134
135
b8a41559
AD
136/*----------------------.
137| --report's handling. |
138`----------------------*/
139
140static const char * const report_args[] =
141{
142 /* In a series of synonyms, present the most meaningful first, so
143 that argmatch_valid be more readable. */
144 "none",
145 "state", "states",
146 "itemset", "itemsets",
147 "lookahead", "lookaheads", "look-ahead",
148 "solved",
149 "all",
150 0
151};
152
153static const int report_types[] =
154{
155 report_none,
156 report_states, report_states,
157 report_states | report_itemsets, report_states | report_itemsets,
158 report_states | report_lookahead_tokens,
159 report_states | report_lookahead_tokens,
160 report_states | report_lookahead_tokens,
161 report_states | report_solved_conflicts,
162 report_all
163};
164
165ARGMATCH_VERIFY (report_args, report_types);
166
b8a41559 167
273a74fa
AD
168/*---------------------.
169| --trace's handling. |
170`---------------------*/
171
172static const char * const trace_args[] =
173{
174 /* In a series of synonyms, present the most meaningful first, so
175 that argmatch_valid be more readable. */
b8a41559 176 "none - no traces",
c5e3e510
AD
177 "scan - grammar scanner traces",
178 "parse - grammar parser traces",
b8a41559 179 "automaton - construction of the automaton",
273a74fa 180 "bitsets - use of bitsets",
b8a41559 181 "grammar - reading, reducing the grammar",
1509d42f 182 "resource - memory consumption (where available)",
273a74fa 183 "sets - grammar sets: firsts, nullable etc.",
5263bea9 184 "muscles - m4 definitions passed to the skeleton",
327afc7c
AD
185 "tools - m4 invocation",
186 "m4 - m4 traces",
c5e3e510
AD
187 "skeleton - skeleton postprocessing",
188 "time - time consumption",
db34f798 189 "ielr - IELR conversion",
273a74fa
AD
190 "all - all of the above",
191 0
192};
193
194static const int trace_types[] =
195{
196 trace_none,
473d0a75
AD
197 trace_scan,
198 trace_parse,
273a74fa
AD
199 trace_automaton,
200 trace_bitsets,
201 trace_grammar,
202 trace_resource,
203 trace_sets,
5263bea9 204 trace_muscles,
273a74fa 205 trace_tools,
327afc7c 206 trace_m4,
c5e3e510
AD
207 trace_skeleton,
208 trace_time,
db34f798 209 trace_ielr,
273a74fa
AD
210 trace_all
211};
212
8a6f72f3 213ARGMATCH_VERIFY (trace_args, trace_types);
273a74fa 214
7b42569e
AD
215
216/*------------------------.
217| --warnings's handling. |
218`------------------------*/
219
220static const char * const warnings_args[] =
273a74fa 221{
7b42569e
AD
222 /* In a series of synonyms, present the most meaningful first, so
223 that argmatch_valid be more readable. */
17bd8a73
JD
224 "none - no warnings",
225 "midrule-values - unset or unused midrule values",
184e3179 226 "yacc - incompatibilities with POSIX Yacc",
17bd8a73
JD
227 "all - all of the above",
228 "error - warnings are errors",
7b42569e
AD
229 0
230};
231
232static const int warnings_types[] =
233{
234 warnings_none,
17bd8a73 235 warnings_midrule_values,
7b42569e 236 warnings_yacc,
89eb3c76
JD
237 warnings_all,
238 warnings_error
7b42569e
AD
239};
240
241ARGMATCH_VERIFY (warnings_args, warnings_types);
273a74fa
AD
242
243
0e575721
AD
244/*-------------------------------------------.
245| Display the help message and exit STATUS. |
246`-------------------------------------------*/
e79137ac 247
0df27e8b
PE
248static void usage (int) ATTRIBUTE_NORETURN;
249
4a120d45 250static void
0e575721 251usage (int status)
cbd8ffc5 252{
0e575721
AD
253 if (status != 0)
254 fprintf (stderr, _("Try `%s --help' for more information.\n"),
255 program_name);
256 else
257 {
a7c09cba
DJ
258 /* For ../build-aux/cross-options.pl to work, use the format:
259 ^ -S, --long[=ARGS] (whitespace)
260 A --long option is required.
261 Otherwise, add exceptions to ../build-aux/cross-options.pl. */
262
a92be413 263 printf (_("Usage: %s [OPTION]... FILE\n"), program_name);
0e575721 264 fputs (_("\
eb45ef3b
JD
265Generate a deterministic or GLR parser employing LALR(1), IELR(1), or\n\
266canonical LR(1) parser tables.\n\
a92be413
PE
267\n\
268"), stdout);
9f306f2a 269
0e575721 270 fputs (_("\
a92be413 271Mandatory arguments to long options are mandatory for short options too.\n\
8e55b3aa
JD
272"), stdout);
273 fputs (_("\
274The same is true for optional arguments.\n\
a92be413 275"), stdout);
9f306f2a 276
0e575721 277 fputs (_("\
a92be413 278\n\
9f306f2a 279Operation modes:\n\
f7ab6a50
PE
280 -h, --help display this help and exit\n\
281 -V, --version output version information and exit\n\
282 --print-localedir output directory containing locale-dependent data\n\
d4bd2295 283 --print-datadir output directory containing skeletons and XSLT\n\
a92be413 284 -y, --yacc emulate POSIX Yacc\n\
9b9e0a7d 285 -W, --warnings[=CATEGORY] report the warnings falling in CATEGORY\n\
a92be413
PE
286\n\
287"), stdout);
9f306f2a 288
0e575721 289 fputs (_("\
9f306f2a 290Parser:\n\
59da312b 291 -L, --language=LANGUAGE specify the output programming language\n\
ed4d67dc 292 (this is an experimental feature)\n\
cd5bd6ac 293 -S, --skeleton=FILE specify the skeleton to use\n\
fa819509
AD
294 -t, --debug instrument the parser for tracing\n\
295 same as `-Dparse.trace'\n\
58697c6d
AD
296 --locations enable location support\n\
297 -D, --define=NAME[=VALUE] same as `%define NAME \"VALUE\"'\n\
9f306f2a
AD
298 -p, --name-prefix=PREFIX prepend PREFIX to the external symbols\n\
299 -l, --no-lines don't generate `#line' directives\n\
9f306f2a 300 -k, --token-table include a table of token names\n\
a92be413 301\n\
0e575721 302"), stdout);
9f306f2a 303
8e55b3aa
JD
304 /* Keep -d and --defines separate so that ../build-aux/cross-options.pl
305 * won't assume that -d also takes an argument. */
0e575721 306 fputs (_("\
9f306f2a 307Output:\n\
8e55b3aa
JD
308 --defines[=FILE] also produce a header file\n\
309 -d likewise but cannot specify FILE (for POSIX Yacc)\n\
ec3bc396 310 -r, --report=THINGS also produce details on the automaton\n\
1bb2bd75 311 --report-file=FILE write report to FILE\n\
ec3bc396 312 -v, --verbose same as `--report=state'\n\
9f306f2a 313 -b, --file-prefix=PREFIX specify a PREFIX for output files\n\
951366c1 314 -o, --output=FILE leave output to FILE\n\
6aeb9c57
AD
315 -g, --graph[=FILE] also output a graph of the automaton\n\
316 -x, --xml[=FILE] also output an XML report of the automaton\n\
59da312b 317 (the XML schema is experimental)\n\
a92be413 318\n\
0e575721 319"), stdout);
86eff183 320
6aeb9c57
AD
321 fputs (_("\
322Warning categories include:\n\
323 `midrule-values' unset or unused midrule values\n\
184e3179 324 `yacc' incompatibilities with POSIX Yacc\n\
6aeb9c57
AD
325 `all' all the warnings\n\
326 `no-CATEGORY' turn off warnings in CATEGORY\n\
327 `none' turn off all the warnings\n\
328 `error' treat warnings as errors\n\
66f0441d 329\n\
6aeb9c57
AD
330"), stdout);
331
0e575721 332 fputs (_("\
ec3bc396
AD
333THINGS is a list of comma separated words that can include:\n\
334 `state' describe the states\n\
335 `itemset' complete the core item sets with their closure\n\
742e4900 336 `lookahead' explicitly associate lookahead tokens to items\n\
b408954b 337 `solved' describe shift/reduce conflicts solving\n\
ec3bc396
AD
338 `all' include all the above information\n\
339 `none' disable the report\n\
0e575721 340"), stdout);
9f306f2a 341
a92be413 342 printf (_("\nReport bugs to <%s>.\n"), PACKAGE_BUGREPORT);
0e575721
AD
343 }
344
345 exit (status);
cbd8ffc5
DM
346}
347
e79137ac
AD
348
349/*------------------------------.
350| Display the version message. |
351`------------------------------*/
352
353static void
0e575721 354version (void)
e79137ac
AD
355{
356 /* Some efforts were made to ease the translators' task, please
357 continue. */
0e575721
AD
358 printf (_("bison (GNU Bison) %s"), VERSION);
359 putc ('\n', stdout);
360 fputs (_("Written by Robert Corbett and Richard Stallman.\n"), stdout);
361 putc ('\n', stdout);
e79137ac 362
0e575721 363 fprintf (stdout,
a005a9c4
AD
364 _("Copyright (C) %d Free Software Foundation, Inc.\n"),
365 PACKAGE_COPYRIGHT_YEAR);
e79137ac
AD
366
367 fputs (_("\
368This is free software; see the source for copying conditions. There is NO\n\
369warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.\n\
370"),
0e575721 371 stdout);
e79137ac
AD
372}
373
374
0e021770
PE
375/*-------------------------------------.
376| --skeleton and --language handling. |
377`--------------------------------------*/
378
379void
5a893c2b 380skeleton_arg (char const *arg, int prio, location loc)
0e021770
PE
381{
382 if (prio < skeleton_prio)
383 {
384 skeleton_prio = prio;
385 skeleton = arg;
386 }
387 else if (prio == skeleton_prio)
388 {
389 char const *msg =
390 _("multiple skeleton declarations are invalid");
5a893c2b 391 complain_at (loc, msg);
0e021770
PE
392 }
393}
394
395void
5a893c2b 396language_argmatch (char const *arg, int prio, location loc)
0e021770
PE
397{
398 char const *msg;
399
400 if (prio < language_prio)
401 {
402 int i;
403 for (i = 0; valid_languages[i].language[0]; i++)
d7e0a1a7 404 if (c_strcasecmp (arg, valid_languages[i].language) == 0)
0e021770
PE
405 {
406 language_prio = prio;
407 language = &valid_languages[i];
408 return;
409 }
410 msg = _("invalid language `%s'");
411 }
412 else if (language_prio == prio)
413 msg = _("multiple language declarations are invalid");
414 else
415 return;
416
5a893c2b 417 complain_at (loc, msg, arg);
0e021770
PE
418}
419
e79137ac
AD
420/*----------------------.
421| Process the options. |
422`----------------------*/
423
7020f1e9
AD
424/* Shorts options.
425 Should be computed from long_options. */
426static char const short_options[] =
58697c6d 427 "D:"
7020f1e9
AD
428 "L:"
429 "S:"
430 "T::"
431 "V"
8e55b3aa 432 "W::"
7020f1e9
AD
433 "b:"
434 "d"
435 "e"
436 "g::"
437 "h"
438 "k"
439 "l"
440 "n"
441 "o:"
442 "p:"
443 "r:"
444 "t"
445 "v"
446 "x::"
447 "y"
448 ;
e2aaf4c4 449
d0829076
PE
450/* Values for long options that do not have single-letter equivalents. */
451enum
452{
f7ab6a50 453 LOCATIONS_OPTION = CHAR_MAX + 1,
d4bd2295 454 PRINT_LOCALEDIR_OPTION,
1bb2bd75
JD
455 PRINT_DATADIR_OPTION,
456 REPORT_FILE_OPTION
d0829076
PE
457};
458
e2aaf4c4
AD
459static struct option const long_options[] =
460{
461 /* Operation modes. */
7b42569e
AD
462 { "help", no_argument, 0, 'h' },
463 { "version", no_argument, 0, 'V' },
464 { "print-localedir", no_argument, 0, PRINT_LOCALEDIR_OPTION },
d4bd2295 465 { "print-datadir", no_argument, 0, PRINT_DATADIR_OPTION },
7b42569e 466 { "warnings", optional_argument, 0, 'W' },
e2aaf4c4
AD
467
468 /* Parser. */
469 { "name-prefix", required_argument, 0, 'p' },
470 { "include", required_argument, 0, 'I' },
471
472 /* Output. */
473 { "file-prefix", required_argument, 0, 'b' },
474 { "output", required_argument, 0, 'o' },
475 { "output-file", required_argument, 0, 'o' },
476 { "graph", optional_argument, 0, 'g' },
41d7a5f2 477 { "xml", optional_argument, 0, 'x' },
e2aaf4c4 478 { "report", required_argument, 0, 'r' },
1bb2bd75 479 { "report-file", required_argument, 0, REPORT_FILE_OPTION },
e2aaf4c4
AD
480 { "verbose", no_argument, 0, 'v' },
481
482 /* Hidden. */
273a74fa 483 { "trace", optional_argument, 0, 'T' },
e2aaf4c4 484
e2aaf4c4
AD
485 /* Output. */
486 { "defines", optional_argument, 0, 'd' },
487
488 /* Operation modes. */
489 { "fixed-output-files", no_argument, 0, 'y' },
490 { "yacc", no_argument, 0, 'y' },
491
492 /* Parser. */
493 { "debug", no_argument, 0, 't' },
58697c6d 494 { "define", required_argument, 0, 'D' },
d0829076 495 { "locations", no_argument, 0, LOCATIONS_OPTION },
e2aaf4c4 496 { "no-lines", no_argument, 0, 'l' },
e2aaf4c4
AD
497 { "raw", no_argument, 0, 0 },
498 { "skeleton", required_argument, 0, 'S' },
0e021770 499 { "language", required_argument, 0, 'L' },
e2aaf4c4
AD
500 { "token-table", no_argument, 0, 'k' },
501
502 {0, 0, 0, 0}
503};
504
ae404801
AD
505/* Under DOS, there is no difference on the case. This can be
506 troublesome when looking for `.tab' etc. */
507#ifdef MSDOS
508# define AS_FILE_NAME(File) (strlwr (File), (File))
509#else
510# define AS_FILE_NAME(File) (File)
511#endif
512
58697c6d
AD
513/* Build a location for the current command line argument. */
514static
515location
d73e55e0 516command_line_location (void)
58697c6d
AD
517{
518 location res;
519 /* "<command line>" is used in GCC's messages about -D. */
520 boundary_set (&res.start, uniqstr_new ("<command line>"), optind, -1);
521 res.end = res.start;
522 return res;
523}
524
525
3d8fc6ca 526void
d2729d44 527getargs (int argc, char *argv[])
3d8fc6ca 528{
1916f98e 529 int c;
3d8fc6ca 530
08721544
PE
531 while ((c = getopt_long (argc, argv, short_options, long_options, NULL))
532 != -1)
cd5bd6ac
AD
533 switch (c)
534 {
58697c6d
AD
535 /* ASCII Sorting for short options (i.e., upper case then
536 lower case), and then long-only options. */
537
cd5bd6ac
AD
538 case 0:
539 /* Certain long options cause getopt_long to return 0. */
540 break;
541
58697c6d
AD
542 case 'D': /* -DNAME[=VALUE]. */
543 {
544 char* name = optarg;
545 char* value = strchr (optarg, '=');
546 if (value)
9ce405ce 547 *value++ = 0;
a8beef7e 548 muscle_percent_define_insert (name, command_line_location (),
9ce405ce 549 value ? value : "");
58697c6d 550 }
22c2cbc0
AD
551 break;
552
8e55b3aa
JD
553 case 'I':
554 include = AS_FILE_NAME (optarg);
41d7a5f2
PE
555 break;
556
0e021770 557 case 'L':
51365192
JD
558 language_argmatch (optarg, command_line_prio,
559 command_line_location ());
0e021770
PE
560 break;
561
cd5bd6ac 562 case 'S':
51365192
JD
563 skeleton_arg (AS_FILE_NAME (optarg), command_line_prio,
564 command_line_location ());
cd5bd6ac
AD
565 break;
566
8e55b3aa
JD
567 case 'T':
568 FLAGS_ARGMATCH (trace, optarg);
f6bd5427
MA
569 break;
570
8e55b3aa
JD
571 case 'V':
572 version ();
573 exit (EXIT_SUCCESS);
574
575 case 'W':
ae404801 576 if (optarg)
8e55b3aa
JD
577 FLAGS_ARGMATCH (warnings, optarg);
578 else
579 warnings_flag |= warnings_all;
580 break;
581
582 case 'b':
583 spec_file_prefix = AS_FILE_NAME (optarg);
584 break;
585
58697c6d
AD
586 case 'd':
587 /* Here, the -d and --defines options are differentiated. */
588 defines_flag = true;
589 if (optarg)
590 spec_defines_file = xstrdup (AS_FILE_NAME (optarg));
591 break;
592
8e55b3aa
JD
593 case 'g':
594 graph_flag = true;
595 if (optarg)
596 spec_graph_file = xstrdup (AS_FILE_NAME (optarg));
cd5bd6ac
AD
597 break;
598
8e55b3aa
JD
599 case 'h':
600 usage (EXIT_SUCCESS);
601
7b42569e
AD
602 case 'k':
603 token_table_flag = true;
604 break;
605
cd5bd6ac 606 case 'l':
d0829076
PE
607 no_lines_flag = true;
608 break;
609
7b42569e
AD
610 case 'o':
611 spec_outfile = AS_FILE_NAME (optarg);
cd5bd6ac
AD
612 break;
613
7b42569e
AD
614 case 'p':
615 spec_name_prefix = optarg;
616 break;
617
618 case 'r':
619 FLAGS_ARGMATCH (report, optarg);
620 break;
621
cd5bd6ac 622 case 't':
fa819509
AD
623 muscle_percent_define_insert ("parse.trace",
624 command_line_location (), "");
cd5bd6ac
AD
625 break;
626
7b42569e
AD
627 case 'v':
628 report_flag |= report_states;
cd5bd6ac
AD
629 break;
630
8e55b3aa
JD
631 case 'x':
632 xml_flag = true;
633 if (optarg)
634 spec_xml_file = xstrdup (AS_FILE_NAME (optarg));
cd5bd6ac
AD
635 break;
636
8e55b3aa
JD
637 case 'y':
638 yacc_flag = true;
ec3bc396
AD
639 break;
640
7b42569e 641 case LOCATIONS_OPTION:
bc0f5737
AD
642 muscle_percent_define_ensure ("locations",
643 command_line_location (), true);
273a74fa
AD
644 break;
645
7b42569e
AD
646 case PRINT_LOCALEDIR_OPTION:
647 printf ("%s\n", LOCALEDIR);
648 exit (EXIT_SUCCESS);
649
d4bd2295
JD
650 case PRINT_DATADIR_OPTION:
651 printf ("%s\n", compute_pkgdatadir ());
652 exit (EXIT_SUCCESS);
653
8e55b3aa
JD
654 case REPORT_FILE_OPTION:
655 spec_verbose_file = xstrdup (AS_FILE_NAME (optarg));
656 break;
657
cd5bd6ac 658 default:
0df27e8b 659 usage (EXIT_FAILURE);
cd5bd6ac 660 }
3d8fc6ca 661
a4b6efd4 662 if (argc - optind != 1)
3d8fc6ca 663 {
a4b6efd4
PE
664 if (argc - optind < 1)
665 error (0, 0, _("missing operand after `%s'"), argv[argc - 1]);
666 else
667 error (0, 0, _("extra operand `%s'"), argv[optind + 1]);
0e575721 668 usage (EXIT_FAILURE);
3d8fc6ca 669 }
3d8fc6ca 670
d38a11a6 671 current_file = grammar_file = uniqstr_new (argv[optind]);
9fe5a457 672 MUSCLE_INSERT_C_STRING ("file_name", grammar_file);
3d8fc6ca 673}
67212941
JD
674
675void
676tr (char *s, char from, char to)
677{
678 for (; *s; s++)
679 if (*s == from)
680 *s = to;
681}