]> git.saurik.com Git - bison.git/blame_incremental - src/conflicts.c
(symbol_make_alias): Set type of SYMVAL to be that of SYM's type.
[bison.git] / src / conflicts.c
... / ...
CommitLineData
1/* Find and resolve or report look-ahead conflicts for bison,
2
3 Copyright (C) 1984, 1989, 1992, 2000, 2001, 2002
4 Free Software Foundation, Inc.
5
6 This file is part of Bison, the GNU Compiler Compiler.
7
8 Bison is free software; you can redistribute it and/or modify it
9 under the terms of the GNU General Public License as published by
10 the Free Software Foundation; either version 2, or (at your option)
11 any later version.
12
13 Bison is distributed in the hope that it will be useful, but
14 WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 General Public License for more details.
17
18 You should have received a copy of the GNU General Public License
19 along with Bison; see the file COPYING. If not, write to the Free
20 Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
21 02111-1307, USA. */
22
23#include "system.h"
24
25#include <bitset.h>
26
27#include "LR0.h"
28#include "complain.h"
29#include "conflicts.h"
30#include "files.h"
31#include "getargs.h"
32#include "gram.h"
33#include "lalr.h"
34#include "reader.h"
35#include "state.h"
36#include "symtab.h"
37
38/* -1 stands for not specified. */
39int expected_conflicts = -1;
40static char *conflicts = NULL;
41struct obstack solved_conflicts_obstack;
42
43static bitset shiftset;
44static bitset lookaheadset;
45
46\f
47
48enum conflict_resolution
49 {
50 shift_resolution,
51 reduce_resolution,
52 left_resolution,
53 right_resolution,
54 nonassoc_resolution
55 };
56
57
58/*----------------------------------------------------------------.
59| Explain how an SR conflict between TOKEN and RULE was resolved: |
60| RESOLUTION. |
61`----------------------------------------------------------------*/
62
63static inline void
64log_resolution (rule *r, symbol_number token,
65 enum conflict_resolution resolution)
66{
67 if (report_flag & report_solved_conflicts)
68 {
69 /* The description of the resolution. */
70 switch (resolution)
71 {
72 case shift_resolution:
73 case right_resolution:
74 obstack_fgrow2 (&solved_conflicts_obstack,
75 _("\
76 Conflict between rule %d and token %s resolved as shift"),
77 r->number,
78 symbols[token]->tag);
79 break;
80 case reduce_resolution:
81 case left_resolution:
82 obstack_fgrow2 (&solved_conflicts_obstack,
83 _("\
84 Conflict between rule %d and token %s resolved as reduce"),
85 r->number,
86 symbols[token]->tag);
87 break;
88 case nonassoc_resolution:
89 obstack_fgrow2 (&solved_conflicts_obstack,
90 _("\
91 Conflict between rule %d and token %s resolved as an error"),
92 r->number,
93 symbols[token]->tag);
94 break;
95 }
96
97 /* The reason. */
98 switch (resolution)
99 {
100 case shift_resolution:
101 obstack_fgrow2 (&solved_conflicts_obstack,
102 " (%s < %s)",
103 r->prec->tag,
104 symbols[token]->tag);
105 break;
106
107 case reduce_resolution:
108 obstack_fgrow2 (&solved_conflicts_obstack,
109 " (%s < %s)",
110 symbols[token]->tag,
111 r->prec->tag);
112 break;
113
114 case left_resolution:
115 obstack_fgrow1 (&solved_conflicts_obstack,
116 " (%%left %s)",
117 symbols[token]->tag);
118 break;
119
120 case right_resolution:
121 obstack_fgrow1 (&solved_conflicts_obstack,
122 " (%%right %s)",
123 symbols[token]->tag);
124 break;
125 case nonassoc_resolution:
126 obstack_fgrow1 (&solved_conflicts_obstack,
127 " (%%nonassoc %s)",
128 symbols[token]->tag);
129 break;
130 }
131 obstack_sgrow (&solved_conflicts_obstack, ".\n");
132 }
133}
134
135
136/*------------------------------------------------------------------.
137| Turn off the shift recorded for the specified token in the |
138| specified state. Used when we resolve a shift-reduce conflict in |
139| favor of the reduction. |
140`------------------------------------------------------------------*/
141
142static void
143flush_shift (state *s, int token)
144{
145 transitions *trans = s->transitions;
146 int i;
147
148 bitset_reset (lookaheadset, token);
149 for (i = 0; i < trans->num; i++)
150 if (!TRANSITION_IS_DISABLED (trans, i)
151 && TRANSITION_SYMBOL (trans, i) == token)
152 TRANSITION_DISABLE (trans, i);
153}
154
155
156/*-------------------------------------------------------------------.
157| Turn off the reduce recorded for the specified token for the |
158| specified lookahead. Used when we resolve a shift-reduce conflict |
159| in favor of the shift. |
160`-------------------------------------------------------------------*/
161
162static void
163flush_reduce (bitset lookaheads, int token)
164{
165 bitset_reset (lookaheads, token);
166}
167
168
169/*------------------------------------------------------------------.
170| Attempt to resolve shift-reduce conflict for one rule by means of |
171| precedence declarations. It has already been checked that the |
172| rule has a precedence. A conflict is resolved by modifying the |
173| shift or reduce tables so that there is no longer a conflict. |
174| |
175| LOOKAHEAD is the number of the lookahead bitset to consider. |
176| |
177| ERRORS can be used to store discovered explicit errors. |
178`------------------------------------------------------------------*/
179
180static void
181resolve_sr_conflict (state *s, int ruleno, symbol **errors)
182{
183 symbol_number i;
184 reductions *reds = s->reductions;
185 /* Find the rule to reduce by to get precedence of reduction. */
186 rule *redrule = reds->rules[ruleno];
187 int redprec = redrule->prec->prec;
188 bitset lookaheads = reds->lookaheads[ruleno];
189 int nerrs = 0;
190
191 for (i = 0; i < ntokens; i++)
192 if (bitset_test (lookaheads, i)
193 && bitset_test (lookaheadset, i)
194 && symbols[i]->prec)
195 {
196 /* Shift-reduce conflict occurs for token number i
197 and it has a precedence.
198 The precedence of shifting is that of token i. */
199 if (symbols[i]->prec < redprec)
200 {
201 log_resolution (redrule, i, reduce_resolution);
202 flush_shift (s, i);
203 }
204 else if (symbols[i]->prec > redprec)
205 {
206 log_resolution (redrule, i, shift_resolution);
207 flush_reduce (lookaheads, i);
208 }
209 else
210 /* Matching precedence levels.
211 For left association, keep only the reduction.
212 For right association, keep only the shift.
213 For nonassociation, keep neither. */
214
215 switch (symbols[i]->assoc)
216 {
217 case right_assoc:
218 log_resolution (redrule, i, right_resolution);
219 flush_reduce (lookaheads, i);
220 break;
221
222 case left_assoc:
223 log_resolution (redrule, i, left_resolution);
224 flush_shift (s, i);
225 break;
226
227 case non_assoc:
228 log_resolution (redrule, i, nonassoc_resolution);
229 flush_shift (s, i);
230 flush_reduce (lookaheads, i);
231 /* Record an explicit error for this token. */
232 errors[nerrs++] = symbols[i];
233 break;
234
235 case undef_assoc:
236 abort ();
237 }
238 }
239
240 /* Some tokens have been explicitly made errors. Allocate a
241 permanent errs structure for this state, to record them. */
242 state_errs_set (s, nerrs, errors);
243
244 if (obstack_object_size (&solved_conflicts_obstack))
245 {
246 obstack_1grow (&solved_conflicts_obstack, '\0');
247 s->solved_conflicts = obstack_finish (&solved_conflicts_obstack);
248 }
249}
250
251
252/*-------------------------------------------------------------------.
253| Solve the S/R conflicts of state S using the |
254| precedence/associativity, and flag it inconsistent if it still has |
255| conflicts. ERRORS can be used as storage to compute the list of |
256| lookaheads on which S raises a syntax error (%nonassoc). |
257`-------------------------------------------------------------------*/
258
259static void
260set_conflicts (state *s, symbol **errors)
261{
262 int i;
263 transitions *trans = s->transitions;
264 reductions *reds = s->reductions;
265
266 if (s->consistent)
267 return;
268
269 bitset_zero (lookaheadset);
270
271 FOR_EACH_SHIFT (trans, i)
272 bitset_set (lookaheadset, TRANSITION_SYMBOL (trans, i));
273
274 /* Loop over all rules which require lookahead in this state. First
275 check for shift-reduce conflict, and try to resolve using
276 precedence. */
277 for (i = 0; i < reds->num; ++i)
278 if (reds->rules[i]->prec && reds->rules[i]->prec->prec
279 && !bitset_disjoint_p (reds->lookaheads[i], lookaheadset))
280 {
281 resolve_sr_conflict (s, i, errors);
282 break;
283 }
284
285 /* Loop over all rules which require lookahead in this state. Check
286 for conflicts not resolved above. */
287 for (i = 0; i < reds->num; ++i)
288 {
289 if (!bitset_disjoint_p (reds->lookaheads[i], lookaheadset))
290 conflicts[s->number] = 1;
291
292 bitset_or (lookaheadset, lookaheadset, reds->lookaheads[i]);
293 }
294}
295
296
297/*----------------------------------------------------------------.
298| Solve all the S/R conflicts using the precedence/associativity, |
299| and flag as inconsistent the states that still have conflicts. |
300`----------------------------------------------------------------*/
301
302void
303conflicts_solve (void)
304{
305 state_number i;
306 /* List of lookaheads on which we explicitly raise a syntax error. */
307 symbol **errors = MALLOC (errors, ntokens + 1);
308
309 CALLOC (conflicts, nstates);
310 shiftset = bitset_create (ntokens, BITSET_FIXED);
311 lookaheadset = bitset_create (ntokens, BITSET_FIXED);
312 obstack_init (&solved_conflicts_obstack);
313
314 for (i = 0; i < nstates; i++)
315 {
316 set_conflicts (states[i], errors);
317
318 /* For uniformity of the code, make sure all the states have a valid
319 `errs' member. */
320 if (!states[i]->errs)
321 states[i]->errs = errs_new (0, 0);
322 }
323
324 free (errors);
325}
326
327
328/*---------------------------------------------.
329| Count the number of shift/reduce conflicts. |
330`---------------------------------------------*/
331
332static int
333count_sr_conflicts (state *s)
334{
335 int i;
336 int src_count = 0;
337 transitions *trans = s->transitions;
338 reductions *reds = s->reductions;
339
340 if (!trans)
341 return 0;
342
343 bitset_zero (lookaheadset);
344 bitset_zero (shiftset);
345
346 FOR_EACH_SHIFT (trans, i)
347 bitset_set (shiftset, TRANSITION_SYMBOL (trans, i));
348
349 for (i = 0; i < reds->num; ++i)
350 bitset_or (lookaheadset, lookaheadset, reds->lookaheads[i]);
351
352 bitset_and (lookaheadset, lookaheadset, shiftset);
353
354 src_count = bitset_count (lookaheadset);
355
356 return src_count;
357}
358
359
360/*----------------------------------------------------------------.
361| Count the number of reduce/reduce conflicts. If ONE_PER_TOKEN, |
362| count one conflict for each token that has any reduce/reduce |
363| conflicts. Otherwise, count one conflict for each pair of |
364| conflicting reductions. |
365+`----------------------------------------------------------------*/
366
367static int
368count_rr_conflicts (state *s, int one_per_token)
369{
370 int i;
371 reductions *reds = s->reductions;
372 int rrc_count = 0;
373
374 for (i = 0; i < ntokens; i++)
375 {
376 int count = 0;
377 int j;
378 for (j = 0; j < reds->num; ++j)
379 if (bitset_test (reds->lookaheads[j], i))
380 count++;
381
382 if (count >= 2)
383 rrc_count += one_per_token ? 1 : count-1;
384 }
385
386 return rrc_count;
387}
388
389
390/*--------------------------------------------------------.
391| Report the number of conflicts, using the Yacc format. |
392`--------------------------------------------------------*/
393
394static void
395conflict_report (FILE *out, int src_num, int rrc_num)
396{
397 if (src_num && rrc_num)
398 fprintf (out, _("conflicts: %d shift/reduce, %d reduce/reduce\n"),
399 src_num, rrc_num);
400 else if (src_num)
401 fprintf (out, _("conflicts: %d shift/reduce\n"), src_num);
402 else if (rrc_num)
403 fprintf (out, _("conflicts: %d reduce/reduce\n"), rrc_num);
404}
405
406
407/*-----------------------------------------------------------.
408| Output the detailed description of states with conflicts. |
409`-----------------------------------------------------------*/
410
411void
412conflicts_output (FILE *out)
413{
414 bool printed_sth = false;
415 state_number i;
416 for (i = 0; i < nstates; i++)
417 {
418 state *s = states[i];
419 if (conflicts[i])
420 {
421 fprintf (out, _("State %d "), i);
422 conflict_report (out, count_sr_conflicts (s),
423 count_rr_conflicts (s, true));
424 printed_sth = true;
425 }
426 }
427 if (printed_sth)
428 fputs ("\n\n", out);
429}
430
431/*--------------------------------------------------------.
432| Total the number of S/R and R/R conflicts. Unlike the |
433| code in conflicts_output, however, count EACH pair of |
434| reductions for the same state and lookahead as one |
435| conflict. |
436`--------------------------------------------------------*/
437
438int
439conflicts_total_count (void)
440{
441 state_number i;
442 int count;
443
444 /* Conflicts by state. */
445 count = 0;
446 for (i = 0; i < nstates; i++)
447 if (conflicts[i])
448 {
449 count += count_sr_conflicts (states[i]);
450 count += count_rr_conflicts (states[i], false);
451 }
452 return count;
453}
454
455
456/*------------------------------------------.
457| Reporting the total number of conflicts. |
458`------------------------------------------*/
459
460void
461conflicts_print (void)
462{
463 /* Is the number of SR conflicts OK? Either EXPECTED_CONFLICTS is
464 not set, and then we want 0 SR, or else it is specified, in which
465 case we want equality. */
466 int src_ok = 0;
467
468 int src_total = 0;
469 int rrc_total = 0;
470
471 /* Conflicts by state. */
472 {
473 state_number i;
474
475 for (i = 0; i < nstates; i++)
476 if (conflicts[i])
477 {
478 src_total += count_sr_conflicts (states[i]);
479 rrc_total += count_rr_conflicts (states[i], true);
480 }
481 }
482
483 src_ok = src_total == (expected_conflicts == -1 ? 0 : expected_conflicts);
484
485 /* If there are no RR conflicts, and as many SR conflicts as
486 expected, then there is nothing to report. */
487 if (!rrc_total && src_ok)
488 return;
489
490 /* Report the total number of conflicts on STDERR. */
491 if (! yacc_flag)
492 fprintf (stderr, "%s: ", current_file);
493 conflict_report (stderr, src_total, rrc_total);
494
495 if (expected_conflicts != -1)
496 {
497 if (! src_ok)
498 complain (ngettext ("expected %d shift/reduce conflict",
499 "expected %d shift/reduce conflicts",
500 expected_conflicts),
501 expected_conflicts);
502 if (rrc_total)
503 complain (_("expected 0 reduce/reduce conflicts"));
504 }
505}
506
507
508void
509conflicts_free (void)
510{
511 XFREE (conflicts);
512 bitset_free (shiftset);
513 bitset_free (lookaheadset);
514 obstack_free (&solved_conflicts_obstack, NULL);
515}