]> git.saurik.com Git - bison.git/blame - src/getargs.c
Simplify last patch slightly.
[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,
59da312b 4 2005, 2006, 2007, 2008 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.",
327afc7c
AD
179 "tools - m4 invocation",
180 "m4 - m4 traces",
c5e3e510
AD
181 "skeleton - skeleton postprocessing",
182 "time - time consumption",
273a74fa
AD
183 "all - all of the above",
184 0
185};
186
187static const int trace_types[] =
188{
189 trace_none,
473d0a75
AD
190 trace_scan,
191 trace_parse,
273a74fa
AD
192 trace_automaton,
193 trace_bitsets,
194 trace_grammar,
195 trace_resource,
196 trace_sets,
197 trace_tools,
327afc7c 198 trace_m4,
c5e3e510
AD
199 trace_skeleton,
200 trace_time,
273a74fa
AD
201 trace_all
202};
203
8a6f72f3 204ARGMATCH_VERIFY (trace_args, trace_types);
273a74fa 205
7b42569e
AD
206
207/*------------------------.
208| --warnings's handling. |
209`------------------------*/
210
211static const char * const warnings_args[] =
273a74fa 212{
7b42569e
AD
213 /* In a series of synonyms, present the most meaningful first, so
214 that argmatch_valid be more readable. */
17bd8a73
JD
215 "none - no warnings",
216 "midrule-values - unset or unused midrule values",
217 "yacc - incompatibilities with POSIX YACC",
218 "all - all of the above",
219 "error - warnings are errors",
7b42569e
AD
220 0
221};
222
223static const int warnings_types[] =
224{
225 warnings_none,
17bd8a73 226 warnings_midrule_values,
7b42569e 227 warnings_yacc,
89eb3c76
JD
228 warnings_all,
229 warnings_error
7b42569e
AD
230};
231
232ARGMATCH_VERIFY (warnings_args, warnings_types);
273a74fa
AD
233
234
0e575721
AD
235/*-------------------------------------------.
236| Display the help message and exit STATUS. |
237`-------------------------------------------*/
e79137ac 238
0df27e8b
PE
239static void usage (int) ATTRIBUTE_NORETURN;
240
4a120d45 241static void
0e575721 242usage (int status)
cbd8ffc5 243{
0e575721
AD
244 if (status != 0)
245 fprintf (stderr, _("Try `%s --help' for more information.\n"),
246 program_name);
247 else
248 {
a92be413 249 printf (_("Usage: %s [OPTION]... FILE\n"), program_name);
0e575721 250 fputs (_("\
a92be413
PE
251Generate LALR(1) and GLR parsers.\n\
252\n\
253"), stdout);
9f306f2a 254
0e575721 255 fputs (_("\
a92be413 256Mandatory arguments to long options are mandatory for short options too.\n\
8e55b3aa
JD
257"), stdout);
258 fputs (_("\
259The same is true for optional arguments.\n\
a92be413 260"), stdout);
9f306f2a 261
0e575721 262 fputs (_("\
a92be413 263\n\
9f306f2a 264Operation modes:\n\
f7ab6a50
PE
265 -h, --help display this help and exit\n\
266 -V, --version output version information and exit\n\
267 --print-localedir output directory containing locale-dependent data\n\
d4bd2295 268 --print-datadir output directory containing skeletons and XSLT\n\
a92be413 269 -y, --yacc emulate POSIX Yacc\n\
f67c4037 270 -W, --warnings[=CATEGORY] report the warnings falling in CATEGORY\n\
a92be413
PE
271\n\
272"), stdout);
9f306f2a 273
0e575721 274 fputs (_("\
9f306f2a 275Parser:\n\
59da312b 276 -L, --language=LANGUAGE specify the output programming language\n\
ed4d67dc 277 (this is an experimental feature)\n\
cd5bd6ac 278 -S, --skeleton=FILE specify the skeleton to use\n\
9f306f2a 279 -t, --debug instrument the parser for debugging\n\
e14c6831
AD
280 --locations enable location support\n\
281 -D, --define=NAME[=VALUE] same as `%define NAME \"VALUE\"'\n\
9f306f2a
AD
282 -p, --name-prefix=PREFIX prepend PREFIX to the external symbols\n\
283 -l, --no-lines don't generate `#line' directives\n\
9f306f2a 284 -k, --token-table include a table of token names\n\
a92be413 285\n\
0e575721 286"), stdout);
9f306f2a 287
8e55b3aa
JD
288 /* Keep -d and --defines separate so that ../build-aux/cross-options.pl
289 * won't assume that -d also takes an argument. */
0e575721 290 fputs (_("\
9f306f2a 291Output:\n\
8e55b3aa
JD
292 --defines[=FILE] also produce a header file\n\
293 -d likewise but cannot specify FILE (for POSIX Yacc)\n\
ec3bc396 294 -r, --report=THINGS also produce details on the automaton\n\
1bb2bd75 295 --report-file=FILE write report to FILE\n\
ec3bc396 296 -v, --verbose same as `--report=state'\n\
9f306f2a 297 -b, --file-prefix=PREFIX specify a PREFIX for output files\n\
951366c1 298 -o, --output=FILE leave output to FILE\n\
6aeb9c57
AD
299 -g, --graph[=FILE] also output a graph of the automaton\n\
300 -x, --xml[=FILE] also output an XML report of the automaton\n\
59da312b 301 (the XML schema is experimental)\n\
a92be413 302\n\
0e575721 303"), stdout);
86eff183 304
6aeb9c57
AD
305 fputs (_("\
306Warning categories include:\n\
307 `midrule-values' unset or unused midrule values\n\
308 `yacc' incompatibilities with POSIX YACC\n\
309 `all' all the warnings\n\
310 `no-CATEGORY' turn off warnings in CATEGORY\n\
311 `none' turn off all the warnings\n\
312 `error' treat warnings as errors\n\
66f0441d 313\n\
6aeb9c57
AD
314"), stdout);
315
0e575721 316 fputs (_("\
ec3bc396
AD
317THINGS is a list of comma separated words that can include:\n\
318 `state' describe the states\n\
319 `itemset' complete the core item sets with their closure\n\
742e4900 320 `lookahead' explicitly associate lookahead tokens to items\n\
b408954b 321 `solved' describe shift/reduce conflicts solving\n\
ec3bc396
AD
322 `all' include all the above information\n\
323 `none' disable the report\n\
0e575721 324"), stdout);
9f306f2a 325
a92be413 326 printf (_("\nReport bugs to <%s>.\n"), PACKAGE_BUGREPORT);
0e575721
AD
327 }
328
329 exit (status);
cbd8ffc5
DM
330}
331
e79137ac
AD
332
333/*------------------------------.
334| Display the version message. |
335`------------------------------*/
336
337static void
0e575721 338version (void)
e79137ac
AD
339{
340 /* Some efforts were made to ease the translators' task, please
341 continue. */
0e575721
AD
342 printf (_("bison (GNU Bison) %s"), VERSION);
343 putc ('\n', stdout);
344 fputs (_("Written by Robert Corbett and Richard Stallman.\n"), stdout);
345 putc ('\n', stdout);
e79137ac 346
0e575721 347 fprintf (stdout,
a005a9c4
AD
348 _("Copyright (C) %d Free Software Foundation, Inc.\n"),
349 PACKAGE_COPYRIGHT_YEAR);
e79137ac
AD
350
351 fputs (_("\
352This is free software; see the source for copying conditions. There is NO\n\
353warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.\n\
354"),
0e575721 355 stdout);
e79137ac
AD
356}
357
358
0e021770
PE
359/*-------------------------------------.
360| --skeleton and --language handling. |
361`--------------------------------------*/
362
363void
e186a284 364skeleton_arg (char const *arg, int prio, location loc)
0e021770
PE
365{
366 if (prio < skeleton_prio)
367 {
368 skeleton_prio = prio;
369 skeleton = arg;
370 }
371 else if (prio == skeleton_prio)
372 {
373 char const *msg =
374 _("multiple skeleton declarations are invalid");
e186a284 375 complain_at (loc, msg);
0e021770
PE
376 }
377}
378
379void
e186a284 380language_argmatch (char const *arg, int prio, location loc)
0e021770
PE
381{
382 char const *msg;
383
384 if (prio < language_prio)
385 {
386 int i;
387 for (i = 0; valid_languages[i].language[0]; i++)
d7e0a1a7 388 if (c_strcasecmp (arg, valid_languages[i].language) == 0)
0e021770
PE
389 {
390 language_prio = prio;
391 language = &valid_languages[i];
392 return;
393 }
394 msg = _("invalid language `%s'");
395 }
396 else if (language_prio == prio)
397 msg = _("multiple language declarations are invalid");
398 else
399 return;
400
e186a284 401 complain_at (loc, msg, arg);
0e021770
PE
402}
403
e79137ac
AD
404/*----------------------.
405| Process the options. |
406`----------------------*/
407
7020f1e9
AD
408/* Shorts options.
409 Should be computed from long_options. */
410static char const short_options[] =
e14c6831 411 "D:"
7020f1e9
AD
412 "L:"
413 "S:"
414 "T::"
415 "V"
8e55b3aa 416 "W::"
7020f1e9
AD
417 "b:"
418 "d"
419 "e"
420 "g::"
421 "h"
422 "k"
423 "l"
424 "n"
425 "o:"
426 "p:"
427 "r:"
428 "t"
429 "v"
430 "x::"
431 "y"
432 ;
e2aaf4c4 433
d0829076
PE
434/* Values for long options that do not have single-letter equivalents. */
435enum
436{
f7ab6a50 437 LOCATIONS_OPTION = CHAR_MAX + 1,
d4bd2295 438 PRINT_LOCALEDIR_OPTION,
1bb2bd75
JD
439 PRINT_DATADIR_OPTION,
440 REPORT_FILE_OPTION
d0829076
PE
441};
442
e2aaf4c4
AD
443static struct option const long_options[] =
444{
445 /* Operation modes. */
7b42569e
AD
446 { "help", no_argument, 0, 'h' },
447 { "version", no_argument, 0, 'V' },
448 { "print-localedir", no_argument, 0, PRINT_LOCALEDIR_OPTION },
d4bd2295 449 { "print-datadir", no_argument, 0, PRINT_DATADIR_OPTION },
7b42569e 450 { "warnings", optional_argument, 0, 'W' },
e2aaf4c4
AD
451
452 /* Parser. */
453 { "name-prefix", required_argument, 0, 'p' },
454 { "include", required_argument, 0, 'I' },
455
456 /* Output. */
457 { "file-prefix", required_argument, 0, 'b' },
458 { "output", required_argument, 0, 'o' },
459 { "output-file", required_argument, 0, 'o' },
460 { "graph", optional_argument, 0, 'g' },
41d7a5f2 461 { "xml", optional_argument, 0, 'x' },
e2aaf4c4 462 { "report", required_argument, 0, 'r' },
1bb2bd75 463 { "report-file", required_argument, 0, REPORT_FILE_OPTION },
e2aaf4c4
AD
464 { "verbose", no_argument, 0, 'v' },
465
466 /* Hidden. */
273a74fa 467 { "trace", optional_argument, 0, 'T' },
e2aaf4c4 468
e2aaf4c4
AD
469 /* Output. */
470 { "defines", optional_argument, 0, 'd' },
471
472 /* Operation modes. */
473 { "fixed-output-files", no_argument, 0, 'y' },
474 { "yacc", no_argument, 0, 'y' },
475
476 /* Parser. */
477 { "debug", no_argument, 0, 't' },
e14c6831 478 { "define", required_argument, 0, 'D' },
d0829076 479 { "locations", no_argument, 0, LOCATIONS_OPTION },
e2aaf4c4 480 { "no-lines", no_argument, 0, 'l' },
e2aaf4c4
AD
481 { "raw", no_argument, 0, 0 },
482 { "skeleton", required_argument, 0, 'S' },
0e021770 483 { "language", required_argument, 0, 'L' },
e2aaf4c4
AD
484 { "token-table", no_argument, 0, 'k' },
485
486 {0, 0, 0, 0}
487};
488
ae404801
AD
489/* Under DOS, there is no difference on the case. This can be
490 troublesome when looking for `.tab' etc. */
491#ifdef MSDOS
492# define AS_FILE_NAME(File) (strlwr (File), (File))
493#else
494# define AS_FILE_NAME(File) (File)
495#endif
496
e14c6831
AD
497/* Build a location for the current command line argument. */
498static
499location
6f5be1ab 500command_line_location (void)
e14c6831
AD
501{
502 location res;
503 /* "<command line>" is used in GCC's messages about -D. */
504 boundary_set (&res.start, uniqstr_new ("<command line>"), optind, -1);
505 res.end = res.start;
506 return res;
507}
508
509
3d8fc6ca 510void
d2729d44 511getargs (int argc, char *argv[])
3d8fc6ca 512{
1916f98e 513 int c;
3d8fc6ca 514
08721544
PE
515 while ((c = getopt_long (argc, argv, short_options, long_options, NULL))
516 != -1)
cd5bd6ac
AD
517 switch (c)
518 {
e14c6831
AD
519 /* ASCII Sorting for short options (i.e., upper case then
520 lower case), and then long-only options. */
521
cd5bd6ac
AD
522 case 0:
523 /* Certain long options cause getopt_long to return 0. */
524 break;
525
e14c6831
AD
526 case 'D': /* -DNAME[=VALUE]. */
527 {
528 char* name = optarg;
529 char* value = strchr (optarg, '=');
530 if (value)
c4eb1e84 531 *value++ = 0;
10fa0146 532 muscle_percent_define_insert (name, command_line_location (),
c4eb1e84 533 value ? value : "");
e14c6831 534 }
22c2cbc0
AD
535 break;
536
8e55b3aa
JD
537 case 'I':
538 include = AS_FILE_NAME (optarg);
41d7a5f2
PE
539 break;
540
0e021770 541 case 'L':
e186a284
AD
542 language_argmatch (optarg, command_line_prio,
543 command_line_location ());
0e021770
PE
544 break;
545
cd5bd6ac 546 case 'S':
e186a284
AD
547 skeleton_arg (AS_FILE_NAME (optarg), command_line_prio,
548 command_line_location ());
cd5bd6ac
AD
549 break;
550
8e55b3aa
JD
551 case 'T':
552 FLAGS_ARGMATCH (trace, optarg);
f6bd5427
MA
553 break;
554
8e55b3aa
JD
555 case 'V':
556 version ();
557 exit (EXIT_SUCCESS);
558
559 case 'W':
ae404801 560 if (optarg)
8e55b3aa
JD
561 FLAGS_ARGMATCH (warnings, optarg);
562 else
563 warnings_flag |= warnings_all;
564 break;
565
566 case 'b':
567 spec_file_prefix = AS_FILE_NAME (optarg);
568 break;
569
e14c6831
AD
570 case 'd':
571 /* Here, the -d and --defines options are differentiated. */
572 defines_flag = true;
573 if (optarg)
574 spec_defines_file = xstrdup (AS_FILE_NAME (optarg));
575 break;
576
8e55b3aa
JD
577 case 'g':
578 graph_flag = true;
579 if (optarg)
580 spec_graph_file = xstrdup (AS_FILE_NAME (optarg));
cd5bd6ac
AD
581 break;
582
8e55b3aa
JD
583 case 'h':
584 usage (EXIT_SUCCESS);
585
7b42569e
AD
586 case 'k':
587 token_table_flag = true;
588 break;
589
cd5bd6ac 590 case 'l':
d0829076
PE
591 no_lines_flag = true;
592 break;
593
7b42569e
AD
594 case 'o':
595 spec_outfile = AS_FILE_NAME (optarg);
cd5bd6ac
AD
596 break;
597
7b42569e
AD
598 case 'p':
599 spec_name_prefix = optarg;
600 break;
601
602 case 'r':
603 FLAGS_ARGMATCH (report, optarg);
604 break;
605
cd5bd6ac 606 case 't':
d0829076 607 debug_flag = true;
cd5bd6ac
AD
608 break;
609
7b42569e
AD
610 case 'v':
611 report_flag |= report_states;
cd5bd6ac
AD
612 break;
613
8e55b3aa
JD
614 case 'x':
615 xml_flag = true;
616 if (optarg)
617 spec_xml_file = xstrdup (AS_FILE_NAME (optarg));
cd5bd6ac
AD
618 break;
619
8e55b3aa
JD
620 case 'y':
621 yacc_flag = true;
ec3bc396
AD
622 break;
623
7b42569e
AD
624 case LOCATIONS_OPTION:
625 locations_flag = true;
273a74fa
AD
626 break;
627
7b42569e
AD
628 case PRINT_LOCALEDIR_OPTION:
629 printf ("%s\n", LOCALEDIR);
630 exit (EXIT_SUCCESS);
631
d4bd2295
JD
632 case PRINT_DATADIR_OPTION:
633 printf ("%s\n", compute_pkgdatadir ());
634 exit (EXIT_SUCCESS);
635
8e55b3aa
JD
636 case REPORT_FILE_OPTION:
637 spec_verbose_file = xstrdup (AS_FILE_NAME (optarg));
638 break;
639
cd5bd6ac 640 default:
0df27e8b 641 usage (EXIT_FAILURE);
cd5bd6ac 642 }
3d8fc6ca 643
a4b6efd4 644 if (argc - optind != 1)
3d8fc6ca 645 {
a4b6efd4
PE
646 if (argc - optind < 1)
647 error (0, 0, _("missing operand after `%s'"), argv[argc - 1]);
648 else
649 error (0, 0, _("extra operand `%s'"), argv[optind + 1]);
0e575721 650 usage (EXIT_FAILURE);
3d8fc6ca 651 }
3d8fc6ca 652
d38a11a6 653 current_file = grammar_file = uniqstr_new (argv[optind]);
75c21b61 654 MUSCLE_INSERT_C_STRING ("file_name", grammar_file);
3d8fc6ca 655}