]> git.saurik.com Git - bison.git/blame - src/getargs.c
maint: run "make update-copyright"
[bison.git] / src / getargs.c
CommitLineData
08721544
PE
1/* Parse command line arguments for Bison.
2
1462fcee
JD
3 Copyright (C) 1984, 1986, 1989, 1992, 2000-2010 Free Software
4 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"
23ec25b7 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 (_("\
51c7ca01
JD
260Generate a deterministic LR or generalized LR (GLR) parser employing\n\
261LALR(1), IELR(1), or canonical 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\
620b2e36
JD
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\
278 --print-datadir output directory containing skeletons and XSLT\n\
279 -y, --yacc emulate POSIX Yacc\n\
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\
34d41938
JD
286 -L, --language=LANGUAGE specify the output programming language\n\
287 (this is an experimental feature)\n\
288 -S, --skeleton=FILE specify the skeleton to use\n\
289 -t, --debug instrument the parser for debugging\n\
290 --locations enable location support\n\
291 -D, --define=NAME[=VALUE] similar to `%define NAME \"VALUE\"'\n\
292 -F, --force-define=NAME[=VALUE] override `%define NAME \"VALUE\"'\n\
293 -p, --name-prefix=PREFIX prepend PREFIX to the external symbols\n\
294 -l, --no-lines don't generate `#line' directives\n\
295 -k, --token-table include a table of token names\n\
a92be413 296\n\
0e575721 297"), stdout);
9f306f2a 298
8e55b3aa
JD
299 /* Keep -d and --defines separate so that ../build-aux/cross-options.pl
300 * won't assume that -d also takes an argument. */
0e575721 301 fputs (_("\
9f306f2a 302Output:\n\
620b2e36
JD
303 --defines[=FILE] also produce a header file\n\
304 -d likewise but cannot specify FILE (for POSIX Yacc)\n\
305 -r, --report=THINGS also produce details on the automaton\n\
306 --report-file=FILE write report to FILE\n\
307 -v, --verbose same as `--report=state'\n\
308 -b, --file-prefix=PREFIX specify a PREFIX for output files\n\
309 -o, --output=FILE leave output to FILE\n\
310 -g, --graph[=FILE] also output a graph of the automaton\n\
311 -x, --xml[=FILE] also output an XML report of the automaton\n\
312 (the XML schema is experimental)\n\
a92be413 313\n\
0e575721 314"), stdout);
86eff183 315
6aeb9c57
AD
316 fputs (_("\
317Warning categories include:\n\
318 `midrule-values' unset or unused midrule values\n\
44c124a3 319 `yacc' incompatibilities with POSIX Yacc\n\
6aeb9c57
AD
320 `all' all the warnings\n\
321 `no-CATEGORY' turn off warnings in CATEGORY\n\
322 `none' turn off all the warnings\n\
323 `error' treat warnings as errors\n\
66f0441d 324\n\
6aeb9c57
AD
325"), stdout);
326
0e575721 327 fputs (_("\
ec3bc396
AD
328THINGS is a list of comma separated words that can include:\n\
329 `state' describe the states\n\
330 `itemset' complete the core item sets with their closure\n\
742e4900 331 `lookahead' explicitly associate lookahead tokens to items\n\
b408954b 332 `solved' describe shift/reduce conflicts solving\n\
ec3bc396
AD
333 `all' include all the above information\n\
334 `none' disable the report\n\
0e575721 335"), stdout);
9f306f2a 336
a92be413 337 printf (_("\nReport bugs to <%s>.\n"), PACKAGE_BUGREPORT);
0e575721
AD
338 }
339
340 exit (status);
cbd8ffc5
DM
341}
342
e79137ac
AD
343
344/*------------------------------.
345| Display the version message. |
346`------------------------------*/
347
348static void
0e575721 349version (void)
e79137ac
AD
350{
351 /* Some efforts were made to ease the translators' task, please
352 continue. */
0e575721
AD
353 printf (_("bison (GNU Bison) %s"), VERSION);
354 putc ('\n', stdout);
355 fputs (_("Written by Robert Corbett and Richard Stallman.\n"), stdout);
356 putc ('\n', stdout);
e79137ac 357
0e575721 358 fprintf (stdout,
a005a9c4
AD
359 _("Copyright (C) %d Free Software Foundation, Inc.\n"),
360 PACKAGE_COPYRIGHT_YEAR);
e79137ac
AD
361
362 fputs (_("\
363This is free software; see the source for copying conditions. There is NO\n\
364warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.\n\
365"),
0e575721 366 stdout);
e79137ac
AD
367}
368
369
0e021770
PE
370/*-------------------------------------.
371| --skeleton and --language handling. |
372`--------------------------------------*/
373
374void
e186a284 375skeleton_arg (char const *arg, int prio, location loc)
0e021770
PE
376{
377 if (prio < skeleton_prio)
378 {
379 skeleton_prio = prio;
380 skeleton = arg;
381 }
382 else if (prio == skeleton_prio)
c8bf65f0 383 complain_at (loc, _("multiple skeleton declarations are invalid"));
0e021770
PE
384}
385
386void
e186a284 387language_argmatch (char const *arg, int prio, location loc)
0e021770
PE
388{
389 char const *msg;
390
391 if (prio < language_prio)
392 {
393 int i;
394 for (i = 0; valid_languages[i].language[0]; i++)
d7e0a1a7 395 if (c_strcasecmp (arg, valid_languages[i].language) == 0)
0e021770
PE
396 {
397 language_prio = prio;
398 language = &valid_languages[i];
399 return;
400 }
401 msg = _("invalid language `%s'");
402 }
403 else if (language_prio == prio)
404 msg = _("multiple language declarations are invalid");
405 else
406 return;
407
e186a284 408 complain_at (loc, msg, arg);
0e021770
PE
409}
410
e79137ac
AD
411/*----------------------.
412| Process the options. |
413`----------------------*/
414
7020f1e9
AD
415/* Shorts options.
416 Should be computed from long_options. */
417static char const short_options[] =
e14c6831 418 "D:"
34d41938 419 "F:"
7020f1e9
AD
420 "L:"
421 "S:"
422 "T::"
423 "V"
8e55b3aa 424 "W::"
7020f1e9
AD
425 "b:"
426 "d"
427 "e"
428 "g::"
429 "h"
430 "k"
431 "l"
432 "n"
433 "o:"
434 "p:"
435 "r:"
436 "t"
437 "v"
438 "x::"
439 "y"
440 ;
e2aaf4c4 441
d0829076
PE
442/* Values for long options that do not have single-letter equivalents. */
443enum
444{
f7ab6a50 445 LOCATIONS_OPTION = CHAR_MAX + 1,
d4bd2295 446 PRINT_LOCALEDIR_OPTION,
1bb2bd75
JD
447 PRINT_DATADIR_OPTION,
448 REPORT_FILE_OPTION
d0829076
PE
449};
450
e2aaf4c4
AD
451static struct option const long_options[] =
452{
453 /* Operation modes. */
7b42569e
AD
454 { "help", no_argument, 0, 'h' },
455 { "version", no_argument, 0, 'V' },
456 { "print-localedir", no_argument, 0, PRINT_LOCALEDIR_OPTION },
d4bd2295 457 { "print-datadir", no_argument, 0, PRINT_DATADIR_OPTION },
7b42569e 458 { "warnings", optional_argument, 0, 'W' },
e2aaf4c4
AD
459
460 /* Parser. */
461 { "name-prefix", required_argument, 0, 'p' },
462 { "include", required_argument, 0, 'I' },
463
464 /* Output. */
465 { "file-prefix", required_argument, 0, 'b' },
466 { "output", required_argument, 0, 'o' },
467 { "output-file", required_argument, 0, 'o' },
468 { "graph", optional_argument, 0, 'g' },
41d7a5f2 469 { "xml", optional_argument, 0, 'x' },
e2aaf4c4 470 { "report", required_argument, 0, 'r' },
1bb2bd75 471 { "report-file", required_argument, 0, REPORT_FILE_OPTION },
e2aaf4c4
AD
472 { "verbose", no_argument, 0, 'v' },
473
474 /* Hidden. */
273a74fa 475 { "trace", optional_argument, 0, 'T' },
e2aaf4c4 476
e2aaf4c4
AD
477 /* Output. */
478 { "defines", optional_argument, 0, 'd' },
479
480 /* Operation modes. */
481 { "fixed-output-files", no_argument, 0, 'y' },
482 { "yacc", no_argument, 0, 'y' },
483
484 /* Parser. */
485 { "debug", no_argument, 0, 't' },
e14c6831 486 { "define", required_argument, 0, 'D' },
34d41938 487 { "force-define", required_argument, 0, 'F' },
d0829076 488 { "locations", no_argument, 0, LOCATIONS_OPTION },
e2aaf4c4 489 { "no-lines", no_argument, 0, 'l' },
e2aaf4c4
AD
490 { "raw", no_argument, 0, 0 },
491 { "skeleton", required_argument, 0, 'S' },
0e021770 492 { "language", required_argument, 0, 'L' },
e2aaf4c4
AD
493 { "token-table", no_argument, 0, 'k' },
494
495 {0, 0, 0, 0}
496};
497
ae404801
AD
498/* Under DOS, there is no difference on the case. This can be
499 troublesome when looking for `.tab' etc. */
500#ifdef MSDOS
501# define AS_FILE_NAME(File) (strlwr (File), (File))
502#else
503# define AS_FILE_NAME(File) (File)
504#endif
505
e14c6831
AD
506/* Build a location for the current command line argument. */
507static
508location
6f5be1ab 509command_line_location (void)
e14c6831
AD
510{
511 location res;
512 /* "<command line>" is used in GCC's messages about -D. */
513 boundary_set (&res.start, uniqstr_new ("<command line>"), optind, -1);
514 res.end = res.start;
515 return res;
516}
517
518
3d8fc6ca 519void
d2729d44 520getargs (int argc, char *argv[])
3d8fc6ca 521{
1916f98e 522 int c;
3d8fc6ca 523
08721544
PE
524 while ((c = getopt_long (argc, argv, short_options, long_options, NULL))
525 != -1)
cd5bd6ac
AD
526 switch (c)
527 {
e14c6831
AD
528 /* ASCII Sorting for short options (i.e., upper case then
529 lower case), and then long-only options. */
530
cd5bd6ac
AD
531 case 0:
532 /* Certain long options cause getopt_long to return 0. */
533 break;
534
e14c6831 535 case 'D': /* -DNAME[=VALUE]. */
34d41938 536 case 'F': /* -FNAME[=VALUE]. */
e14c6831
AD
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 (),
34d41938
JD
543 value ? value : "",
544 c == 'D' ? MUSCLE_PERCENT_DEFINE_D
545 : MUSCLE_PERCENT_DEFINE_F);
e14c6831 546 }
22c2cbc0
AD
547 break;
548
8e55b3aa
JD
549 case 'I':
550 include = AS_FILE_NAME (optarg);
41d7a5f2
PE
551 break;
552
0e021770 553 case 'L':
e186a284
AD
554 language_argmatch (optarg, command_line_prio,
555 command_line_location ());
0e021770
PE
556 break;
557
cd5bd6ac 558 case 'S':
e186a284
AD
559 skeleton_arg (AS_FILE_NAME (optarg), command_line_prio,
560 command_line_location ());
cd5bd6ac
AD
561 break;
562
8e55b3aa
JD
563 case 'T':
564 FLAGS_ARGMATCH (trace, optarg);
f6bd5427
MA
565 break;
566
8e55b3aa
JD
567 case 'V':
568 version ();
569 exit (EXIT_SUCCESS);
570
571 case 'W':
ae404801 572 if (optarg)
8e55b3aa
JD
573 FLAGS_ARGMATCH (warnings, optarg);
574 else
575 warnings_flag |= warnings_all;
576 break;
577
578 case 'b':
579 spec_file_prefix = AS_FILE_NAME (optarg);
580 break;
581
e14c6831
AD
582 case 'd':
583 /* Here, the -d and --defines options are differentiated. */
584 defines_flag = true;
585 if (optarg)
586 spec_defines_file = xstrdup (AS_FILE_NAME (optarg));
587 break;
588
8e55b3aa
JD
589 case 'g':
590 graph_flag = true;
591 if (optarg)
592 spec_graph_file = xstrdup (AS_FILE_NAME (optarg));
cd5bd6ac
AD
593 break;
594
8e55b3aa
JD
595 case 'h':
596 usage (EXIT_SUCCESS);
597
7b42569e
AD
598 case 'k':
599 token_table_flag = true;
600 break;
601
cd5bd6ac 602 case 'l':
d0829076
PE
603 no_lines_flag = true;
604 break;
605
7b42569e
AD
606 case 'o':
607 spec_outfile = AS_FILE_NAME (optarg);
cd5bd6ac
AD
608 break;
609
7b42569e
AD
610 case 'p':
611 spec_name_prefix = optarg;
612 break;
613
614 case 'r':
615 FLAGS_ARGMATCH (report, optarg);
616 break;
617
cd5bd6ac 618 case 't':
d0829076 619 debug_flag = true;
cd5bd6ac
AD
620 break;
621
7b42569e
AD
622 case 'v':
623 report_flag |= report_states;
cd5bd6ac
AD
624 break;
625
8e55b3aa
JD
626 case 'x':
627 xml_flag = true;
628 if (optarg)
629 spec_xml_file = xstrdup (AS_FILE_NAME (optarg));
cd5bd6ac
AD
630 break;
631
8e55b3aa
JD
632 case 'y':
633 yacc_flag = true;
ec3bc396
AD
634 break;
635
7b42569e
AD
636 case LOCATIONS_OPTION:
637 locations_flag = true;
273a74fa
AD
638 break;
639
7b42569e
AD
640 case PRINT_LOCALEDIR_OPTION:
641 printf ("%s\n", LOCALEDIR);
642 exit (EXIT_SUCCESS);
643
d4bd2295
JD
644 case PRINT_DATADIR_OPTION:
645 printf ("%s\n", compute_pkgdatadir ());
646 exit (EXIT_SUCCESS);
647
8e55b3aa
JD
648 case REPORT_FILE_OPTION:
649 spec_verbose_file = xstrdup (AS_FILE_NAME (optarg));
650 break;
651
cd5bd6ac 652 default:
0df27e8b 653 usage (EXIT_FAILURE);
cd5bd6ac 654 }
3d8fc6ca 655
a4b6efd4 656 if (argc - optind != 1)
3d8fc6ca 657 {
a4b6efd4
PE
658 if (argc - optind < 1)
659 error (0, 0, _("missing operand after `%s'"), argv[argc - 1]);
660 else
661 error (0, 0, _("extra operand `%s'"), argv[optind + 1]);
0e575721 662 usage (EXIT_FAILURE);
3d8fc6ca 663 }
3d8fc6ca 664
d38a11a6 665 current_file = grammar_file = uniqstr_new (argv[optind]);
75c21b61 666 MUSCLE_INSERT_C_STRING ("file_name", grammar_file);
3d8fc6ca 667}
812775a0
JD
668
669void
670tr (char *s, char from, char to)
671{
672 for (; *s; s++)
673 if (*s == from)
674 *s = to;
675}