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