]> git.saurik.com Git - bison.git/blob - src/conflicts.c
warnings: useless semantic types
[bison.git] / src / conflicts.c
1 /* Find and resolve or report lookahead conflicts for bison,
2
3 Copyright (C) 1984, 1989, 1992, 2000-2012 Free Software Foundation,
4 Inc.
5
6 This file is part of Bison, the GNU Compiler Compiler.
7
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.
12
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.
17
18 You should have received a copy of the GNU General Public License
19 along with this program. If not, see <http://www.gnu.org/licenses/>. */
20
21 #include <config.h>
22 #include "system.h"
23
24 #include <bitset.h>
25
26 #include "LR0.h"
27 #include "complain.h"
28 #include "conflicts.h"
29 #include "files.h"
30 #include "getargs.h"
31 #include "gram.h"
32 #include "lalr.h"
33 #include "print-xml.h"
34 #include "reader.h"
35 #include "state.h"
36 #include "symtab.h"
37
38 /* -1 stands for not specified. */
39 int expected_sr_conflicts = -1;
40 int expected_rr_conflicts = -1;
41 static char *conflicts;
42 static struct obstack solved_conflicts_obstack;
43 static struct obstack solved_conflicts_xml_obstack;
44
45 static bitset shift_set;
46 static bitset lookahead_set;
47
48 \f
49
50 enum conflict_resolution
51 {
52 shift_resolution,
53 reduce_resolution,
54 left_resolution,
55 right_resolution,
56 nonassoc_resolution
57 };
58
59
60 /*----------------------------------------------------------------.
61 | Explain how an SR conflict between TOKEN and RULE was resolved: |
62 | RESOLUTION. |
63 `----------------------------------------------------------------*/
64
65 static inline void
66 log_resolution (rule *r, symbol_number token,
67 enum conflict_resolution resolution)
68 {
69 if (report_flag & report_solved_conflicts)
70 {
71 /* The description of the resolution. */
72 switch (resolution)
73 {
74 case shift_resolution:
75 case right_resolution:
76 obstack_fgrow2 (&solved_conflicts_obstack,
77 _(" Conflict between rule %d and token %s"
78 " resolved as shift"),
79 r->number,
80 symbols[token]->tag);
81 break;
82
83 case reduce_resolution:
84 case left_resolution:
85 obstack_fgrow2 (&solved_conflicts_obstack,
86 _(" Conflict between rule %d and token %s"
87 " resolved as reduce"),
88 r->number,
89 symbols[token]->tag);
90 break;
91
92 case nonassoc_resolution:
93 obstack_fgrow2 (&solved_conflicts_obstack,
94 _(" Conflict between rule %d and token %s"
95 " resolved as an error"),
96 r->number,
97 symbols[token]->tag);
98 break;
99 }
100
101 /* The reason. */
102 switch (resolution)
103 {
104 case shift_resolution:
105 obstack_fgrow2 (&solved_conflicts_obstack,
106 " (%s < %s)",
107 r->prec->tag,
108 symbols[token]->tag);
109 break;
110
111 case reduce_resolution:
112 obstack_fgrow2 (&solved_conflicts_obstack,
113 " (%s < %s)",
114 symbols[token]->tag,
115 r->prec->tag);
116 break;
117
118 case left_resolution:
119 obstack_fgrow1 (&solved_conflicts_obstack,
120 " (%%left %s)",
121 symbols[token]->tag);
122 break;
123
124 case right_resolution:
125 obstack_fgrow1 (&solved_conflicts_obstack,
126 " (%%right %s)",
127 symbols[token]->tag);
128 break;
129
130 case nonassoc_resolution:
131 obstack_fgrow1 (&solved_conflicts_obstack,
132 " (%%nonassoc %s)",
133 symbols[token]->tag);
134 break;
135 }
136
137 obstack_sgrow (&solved_conflicts_obstack, ".\n");
138 }
139
140 /* XML report */
141 if (xml_flag)
142 {
143 /* The description of the resolution. */
144 switch (resolution)
145 {
146 case shift_resolution:
147 case right_resolution:
148 obstack_fgrow2 (&solved_conflicts_xml_obstack,
149 " <resolution rule=\"%d\" symbol=\"%s\""
150 " type=\"shift\">",
151 r->number,
152 xml_escape (symbols[token]->tag));
153 break;
154
155 case reduce_resolution:
156 case left_resolution:
157 obstack_fgrow2 (&solved_conflicts_xml_obstack,
158 " <resolution rule=\"%d\" symbol=\"%s\""
159 " type=\"reduce\">",
160 r->number,
161 xml_escape (symbols[token]->tag));
162 break;
163
164 case nonassoc_resolution:
165 obstack_fgrow2 (&solved_conflicts_xml_obstack,
166 " <resolution rule=\"%d\" symbol=\"%s\""
167 " type=\"error\">",
168 r->number,
169 xml_escape (symbols[token]->tag));
170 break;
171 }
172
173 /* The reason. */
174 switch (resolution)
175 {
176 case shift_resolution:
177 obstack_fgrow2 (&solved_conflicts_xml_obstack,
178 "%s &lt; %s",
179 xml_escape_n (0, r->prec->tag),
180 xml_escape_n (1, symbols[token]->tag));
181 break;
182
183 case reduce_resolution:
184 obstack_fgrow2 (&solved_conflicts_xml_obstack,
185 "%s &lt; %s",
186 xml_escape_n (0, symbols[token]->tag),
187 xml_escape_n (1, r->prec->tag));
188 break;
189
190 case left_resolution:
191 obstack_fgrow1 (&solved_conflicts_xml_obstack,
192 "%%left %s",
193 xml_escape (symbols[token]->tag));
194 break;
195
196 case right_resolution:
197 obstack_fgrow1 (&solved_conflicts_xml_obstack,
198 "%%right %s",
199 xml_escape (symbols[token]->tag));
200 break;
201
202 case nonassoc_resolution:
203 obstack_fgrow1 (&solved_conflicts_xml_obstack,
204 "%%nonassoc %s",
205 xml_escape (symbols[token]->tag));
206 break;
207 }
208
209 obstack_sgrow (&solved_conflicts_xml_obstack, "</resolution>\n");
210 }
211 }
212
213
214 /*------------------------------------------------------------------.
215 | Turn off the shift recorded for the specified token in the |
216 | specified state. Used when we resolve a shift-reduce conflict in |
217 | favor of the reduction or as an error (%nonassoc). |
218 `------------------------------------------------------------------*/
219
220 static void
221 flush_shift (state *s, int token)
222 {
223 transitions *trans = s->transitions;
224 int i;
225
226 bitset_reset (lookahead_set, token);
227 for (i = 0; i < trans->num; i++)
228 if (!TRANSITION_IS_DISABLED (trans, i)
229 && TRANSITION_SYMBOL (trans, i) == token)
230 TRANSITION_DISABLE (trans, i);
231 }
232
233
234 /*--------------------------------------------------------------------.
235 | Turn off the reduce recorded for the specified token in the |
236 | specified lookahead set. Used when we resolve a shift-reduce |
237 | conflict in favor of the shift or as an error (%nonassoc). |
238 `--------------------------------------------------------------------*/
239
240 static void
241 flush_reduce (bitset lookahead_tokens, int token)
242 {
243 bitset_reset (lookahead_tokens, token);
244 }
245
246
247 /*------------------------------------------------------------------.
248 | Attempt to resolve shift-reduce conflict for one rule by means of |
249 | precedence declarations. It has already been checked that the |
250 | rule has a precedence. A conflict is resolved by modifying the |
251 | shift or reduce tables so that there is no longer a conflict. |
252 | |
253 | RULENO is the number of the lookahead bitset to consider. |
254 | |
255 | ERRORS and NERRS can be used to store discovered explicit |
256 | errors. |
257 `------------------------------------------------------------------*/
258
259 static void
260 resolve_sr_conflict (state *s, int ruleno, symbol **errors, int *nerrs)
261 {
262 symbol_number i;
263 reductions *reds = s->reductions;
264 /* Find the rule to reduce by to get precedence of reduction. */
265 rule *redrule = reds->rules[ruleno];
266 int redprec = redrule->prec->prec;
267 bitset lookahead_tokens = reds->lookahead_tokens[ruleno];
268
269 for (i = 0; i < ntokens; i++)
270 if (bitset_test (lookahead_tokens, i)
271 && bitset_test (lookahead_set, i)
272 && symbols[i]->prec)
273 {
274 /* Shift-reduce conflict occurs for token number i
275 and it has a precedence.
276 The precedence of shifting is that of token i. */
277 if (symbols[i]->prec < redprec)
278 {
279 log_resolution (redrule, i, reduce_resolution);
280 flush_shift (s, i);
281 }
282 else if (symbols[i]->prec > redprec)
283 {
284 log_resolution (redrule, i, shift_resolution);
285 flush_reduce (lookahead_tokens, i);
286 }
287 else
288 /* Matching precedence levels.
289 For non-defined associativity, keep both: unexpected
290 associativity conflict.
291 For left associativity, keep only the reduction.
292 For right associativity, keep only the shift.
293 For nonassociativity, keep neither. */
294
295 switch (symbols[i]->assoc)
296 {
297 case undef_assoc:
298 abort ();
299
300 case precedence_assoc:
301 break;
302
303 case right_assoc:
304 log_resolution (redrule, i, right_resolution);
305 flush_reduce (lookahead_tokens, i);
306 break;
307
308 case left_assoc:
309 log_resolution (redrule, i, left_resolution);
310 flush_shift (s, i);
311 break;
312
313 case non_assoc:
314 log_resolution (redrule, i, nonassoc_resolution);
315 flush_shift (s, i);
316 flush_reduce (lookahead_tokens, i);
317 /* Record an explicit error for this token. */
318 errors[(*nerrs)++] = symbols[i];
319 break;
320 }
321 }
322 }
323
324
325 /*-------------------------------------------------------------------.
326 | Solve the S/R conflicts of state S using the |
327 | precedence/associativity, and flag it inconsistent if it still has |
328 | conflicts. ERRORS can be used as storage to compute the list of |
329 | lookahead tokens on which S raises a syntax error (%nonassoc). |
330 `-------------------------------------------------------------------*/
331
332 static void
333 set_conflicts (state *s, symbol **errors)
334 {
335 int i;
336 transitions *trans = s->transitions;
337 reductions *reds = s->reductions;
338 int nerrs = 0;
339
340 if (s->consistent)
341 return;
342
343 bitset_zero (lookahead_set);
344
345 FOR_EACH_SHIFT (trans, i)
346 bitset_set (lookahead_set, TRANSITION_SYMBOL (trans, i));
347
348 /* Loop over all rules which require lookahead in this state. First
349 check for shift-reduce conflict, and try to resolve using
350 precedence. */
351 for (i = 0; i < reds->num; ++i)
352 if (reds->rules[i]->prec && reds->rules[i]->prec->prec
353 && !bitset_disjoint_p (reds->lookahead_tokens[i], lookahead_set))
354 resolve_sr_conflict (s, i, errors, &nerrs);
355
356 if (nerrs)
357 {
358 /* Some tokens have been explicitly made errors. Allocate a
359 permanent errs structure for this state, to record them. */
360 state_errs_set (s, nerrs, errors);
361 }
362 if (obstack_object_size (&solved_conflicts_obstack))
363 {
364 obstack_1grow (&solved_conflicts_obstack, '\0');
365 s->solved_conflicts = obstack_finish (&solved_conflicts_obstack);
366 }
367 if (obstack_object_size (&solved_conflicts_xml_obstack))
368 {
369 obstack_1grow (&solved_conflicts_xml_obstack, '\0');
370 s->solved_conflicts_xml = obstack_finish (&solved_conflicts_xml_obstack);
371 }
372
373 /* Loop over all rules which require lookahead in this state. Check
374 for conflicts not resolved above. */
375 for (i = 0; i < reds->num; ++i)
376 {
377 if (!bitset_disjoint_p (reds->lookahead_tokens[i], lookahead_set))
378 conflicts[s->number] = 1;
379 bitset_or (lookahead_set, lookahead_set, reds->lookahead_tokens[i]);
380 }
381 }
382
383
384 /*----------------------------------------------------------------.
385 | Solve all the S/R conflicts using the precedence/associativity, |
386 | and flag as inconsistent the states that still have conflicts. |
387 `----------------------------------------------------------------*/
388
389 void
390 conflicts_solve (void)
391 {
392 state_number i;
393 /* List of lookahead tokens on which we explicitly raise a syntax error. */
394 symbol **errors = xnmalloc (ntokens + 1, sizeof *errors);
395
396 conflicts = xcalloc (nstates, sizeof *conflicts);
397 shift_set = bitset_create (ntokens, BITSET_FIXED);
398 lookahead_set = bitset_create (ntokens, BITSET_FIXED);
399 obstack_init (&solved_conflicts_obstack);
400 obstack_init (&solved_conflicts_xml_obstack);
401
402 for (i = 0; i < nstates; i++)
403 {
404 set_conflicts (states[i], errors);
405
406 /* For uniformity of the code, make sure all the states have a valid
407 `errs' member. */
408 if (!states[i]->errs)
409 states[i]->errs = errs_new (0, 0);
410 }
411
412 free (errors);
413 }
414
415
416 void
417 conflicts_update_state_numbers (state_number old_to_new[],
418 state_number nstates_old)
419 {
420 state_number i;
421 for (i = 0; i < nstates_old; ++i)
422 if (old_to_new[i] != nstates_old)
423 conflicts[old_to_new[i]] = conflicts[i];
424 }
425
426
427 /*---------------------------------------------.
428 | Count the number of shift/reduce conflicts. |
429 `---------------------------------------------*/
430
431 static int
432 count_sr_conflicts (state *s)
433 {
434 int i;
435 int src_count = 0;
436 transitions *trans = s->transitions;
437 reductions *reds = s->reductions;
438
439 if (!trans)
440 return 0;
441
442 bitset_zero (lookahead_set);
443 bitset_zero (shift_set);
444
445 FOR_EACH_SHIFT (trans, i)
446 bitset_set (shift_set, TRANSITION_SYMBOL (trans, i));
447
448 for (i = 0; i < reds->num; ++i)
449 bitset_or (lookahead_set, lookahead_set, reds->lookahead_tokens[i]);
450
451 bitset_and (lookahead_set, lookahead_set, shift_set);
452
453 src_count = bitset_count (lookahead_set);
454
455 return src_count;
456 }
457
458
459 /*----------------------------------------------------------------.
460 | Count the number of reduce/reduce conflicts. If ONE_PER_TOKEN, |
461 | count one conflict for each token that has any reduce/reduce |
462 | conflicts. Otherwise, count one conflict for each pair of |
463 | conflicting reductions. |
464 +`----------------------------------------------------------------*/
465
466 static int
467 count_rr_conflicts (state *s, bool one_per_token)
468 {
469 int i;
470 reductions *reds = s->reductions;
471 int rrc_count = 0;
472
473 for (i = 0; i < ntokens; i++)
474 {
475 int count = 0;
476 int j;
477 for (j = 0; j < reds->num; ++j)
478 if (bitset_test (reds->lookahead_tokens[j], i))
479 count++;
480
481 if (count >= 2)
482 rrc_count += one_per_token ? 1 : count-1;
483 }
484
485 return rrc_count;
486 }
487
488
489 /*--------------------------------------------------------.
490 | Report the number of conflicts, using the Yacc format. |
491 `--------------------------------------------------------*/
492
493 static void
494 conflict_report (FILE *out, int src_num, int rrc_num)
495 {
496 if (src_num && rrc_num)
497 fprintf (out, _("conflicts: %d shift/reduce, %d reduce/reduce\n"),
498 src_num, rrc_num);
499 else if (src_num)
500 fprintf (out, _("conflicts: %d shift/reduce\n"), src_num);
501 else if (rrc_num)
502 fprintf (out, _("conflicts: %d reduce/reduce\n"), rrc_num);
503 }
504
505
506 /*-----------------------------------------------------------.
507 | Output the detailed description of states with conflicts. |
508 `-----------------------------------------------------------*/
509
510 void
511 conflicts_output (FILE *out)
512 {
513 bool printed_sth = false;
514 state_number i;
515 for (i = 0; i < nstates; i++)
516 {
517 state *s = states[i];
518 if (conflicts[i])
519 {
520 fprintf (out, _("State %d "), i);
521 conflict_report (out, count_sr_conflicts (s),
522 count_rr_conflicts (s, true));
523 printed_sth = true;
524 }
525 }
526 if (printed_sth)
527 fputs ("\n\n", out);
528 }
529
530 /*--------------------------------------------------------.
531 | Total the number of S/R and R/R conflicts. Unlike the |
532 | code in conflicts_output, however, count EACH pair of |
533 | reductions for the same state and lookahead as one |
534 | conflict. |
535 `--------------------------------------------------------*/
536
537 int
538 conflicts_total_count (void)
539 {
540 state_number i;
541 int count;
542
543 /* Conflicts by state. */
544 count = 0;
545 for (i = 0; i < nstates; i++)
546 if (conflicts[i])
547 {
548 count += count_sr_conflicts (states[i]);
549 count += count_rr_conflicts (states[i], false);
550 }
551 return count;
552 }
553
554
555 /*------------------------------------------.
556 | Reporting the total number of conflicts. |
557 `------------------------------------------*/
558
559 void
560 conflicts_print (void)
561 {
562 /* Is the number of SR conflicts OK? Either EXPECTED_CONFLICTS is
563 not set, and then we want 0 SR, or else it is specified, in which
564 case we want equality. */
565 bool src_ok;
566 bool rrc_ok;
567
568 int src_total = 0;
569 int rrc_total = 0;
570 int src_expected;
571 int rrc_expected;
572
573 /* Conflicts by state. */
574 {
575 state_number i;
576
577 for (i = 0; i < nstates; i++)
578 if (conflicts[i])
579 {
580 src_total += count_sr_conflicts (states[i]);
581 rrc_total += count_rr_conflicts (states[i], true);
582 }
583 }
584
585 if (! glr_parser && rrc_total > 0 && expected_rr_conflicts != -1)
586 {
587 warn (_("%%expect-rr applies only to GLR parsers"));
588 expected_rr_conflicts = -1;
589 }
590
591 src_expected = expected_sr_conflicts == -1 ? 0 : expected_sr_conflicts;
592 rrc_expected = expected_rr_conflicts == -1 ? 0 : expected_rr_conflicts;
593 src_ok = src_total == src_expected;
594 rrc_ok = rrc_total == rrc_expected;
595
596 /* If there are as many RR conflicts and SR conflicts as
597 expected, then there is nothing to report. */
598 if (rrc_ok & src_ok)
599 return;
600
601 /* Report the total number of conflicts on STDERR. */
602 if (expected_sr_conflicts == -1 && expected_rr_conflicts == -1)
603 {
604 if (!(warnings_flag & warnings_conflicts_sr))
605 src_total = 0;
606 if (!(warnings_flag & warnings_conflicts_rr))
607 rrc_total = 0;
608 }
609 if (src_total | rrc_total)
610 {
611 if (expected_sr_conflicts == -1 && expected_rr_conflicts == -1)
612 set_warning_issued ();
613 if (! yacc_flag)
614 fprintf (stderr, "%s: ", current_file);
615 conflict_report (stderr, src_total, rrc_total);
616 }
617
618 if (expected_sr_conflicts != -1 || expected_rr_conflicts != -1)
619 {
620 if (! src_ok)
621 complain (ngettext ("expected %d shift/reduce conflict",
622 "expected %d shift/reduce conflicts",
623 src_expected),
624 src_expected);
625 if (! rrc_ok)
626 complain (ngettext ("expected %d reduce/reduce conflict",
627 "expected %d reduce/reduce conflicts",
628 rrc_expected),
629 rrc_expected);
630 }
631 }
632
633
634 void
635 conflicts_free (void)
636 {
637 free (conflicts);
638 bitset_free (shift_set);
639 bitset_free (lookahead_set);
640 obstack_free (&solved_conflicts_obstack, NULL);
641 obstack_free (&solved_conflicts_xml_obstack, NULL);
642 }