]> git.saurik.com Git - bison.git/blame - src/closure.c
Merge branch 'maint'
[bison.git] / src / closure.c
CommitLineData
c7bd07f7
PE
1/* Closures for Bison
2
34136e65 3 Copyright (C) 1984, 1989, 2000-2002, 2004-2005, 2007, 2009-2012 Free
575619af 4 Software Foundation, Inc.
d0fb370f 5
8dc26b76 6 This file is part of Bison, the GNU Compiler Compiler.
d0fb370f 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.
d0fb370f 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.
d0fb370f 17
8dc26b76 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/>. */
d0fb370f 20
2cec9080 21#include <config.h>
d0fb370f 22#include "system.h"
c7bd07f7
PE
23
24#include <bitset.h>
25#include <bitsetv-print.h>
26#include <bitsetv.h>
c7bd07f7
PE
27
28#include "closure.h"
29#include "derives.h"
9bfe901c 30#include "getargs.h"
d0fb370f 31#include "gram.h"
2c5f66ed 32#include "reader.h"
c7bd07f7 33#include "symtab.h"
d0fb370f 34
b2872512 35/* NITEMSET is the size of the array ITEMSET. */
c7bd07f7 36item_number *itemset;
b09f4f48 37size_t nitemset;
fb908786 38
dfdb1797 39static bitset ruleset;
d0fb370f
RS
40
41/* internal data. See comments before set_fderives and set_firsts. */
914feea9
AD
42static bitsetv fderives = NULL;
43static bitsetv firsts = NULL;
d0fb370f 44
03ec521c 45/* Retrieve the FDERIVES/FIRSTS sets of the nonterminals numbered Var. */
7086e707 46#define FDERIVES(Var) fderives[(Var) - ntokens]
d8a0245c 47#define FIRSTS(Var) firsts[(Var) - ntokens]
2fa6973e 48\f
d0fb370f 49
2fa6973e
AD
50/*-----------------.
51| Debugging code. |
52`-----------------*/
d0fb370f 53
2fa6973e 54static void
c7bd07f7 55print_closure (char const *title, item_number *array, size_t size)
2fa6973e 56{
23cbcc6c 57 size_t i;
427c0dda 58 fprintf (stderr, "Closure: %s\n", title);
23cbcc6c
AD
59 for (i = 0; i < size; ++i)
60 {
c7bd07f7 61 item_number *rp;
23cbcc6c 62 fprintf (stderr, " %2d: .", array[i]);
75142d45 63 for (rp = &ritem[array[i]]; *rp >= 0; ++rp)
e9690142 64 fprintf (stderr, " %s", symbols[*rp]->tag);
427c0dda 65 fprintf (stderr, " (rule %d)\n", -*rp - 1);
23cbcc6c
AD
66 }
67 fputs ("\n\n", stderr);
2fa6973e
AD
68}
69
70
71static void
72print_firsts (void)
d0fb370f 73{
c7bd07f7 74 symbol_number i, j;
2fa6973e 75
2f4f028d 76 fprintf (stderr, "FIRSTS\n");
2fa6973e
AD
77 for (i = ntokens; i < nsyms; i++)
78 {
613f5e1a 79 bitset_iterator iter;
97650f4e 80 fprintf (stderr, "\t%s firsts\n", symbols[i]->tag);
613f5e1a 81 BITSET_FOR_EACH (iter, FIRSTS (i), j, 0)
e9690142
JD
82 {
83 fprintf (stderr, "\t\t%s\n",
84 symbols[j + ntokens]->tag);
85 }
2fa6973e 86 }
2f4f028d 87 fprintf (stderr, "\n\n");
d0fb370f
RS
88}
89
90
2fa6973e
AD
91static void
92print_fderives (void)
93{
9222837b 94 int i;
c7bd07f7 95 rule_number r;
2fa6973e 96
2f4f028d 97 fprintf (stderr, "FDERIVES\n");
2fa6973e
AD
98 for (i = ntokens; i < nsyms; i++)
99 {
613f5e1a 100 bitset_iterator iter;
97650f4e 101 fprintf (stderr, "\t%s derives\n", symbols[i]->tag);
613f5e1a 102 BITSET_FOR_EACH (iter, FDERIVES (i), r, 0)
e9690142
JD
103 {
104 fprintf (stderr, "\t\t%3d ", r);
105 rule_rhs_print (&rules[r], stderr);
3e153163 106 fprintf (stderr, "\n");
e9690142 107 }
2fa6973e 108 }
2f4f028d 109 fprintf (stderr, "\n\n");
2fa6973e 110}
2fa6973e 111\f
d8a0245c
AD
112/*------------------------------------------------------------------.
113| Set FIRSTS to be an NVARS array of NVARS bitsets indicating which |
114| items can represent the beginning of the input corresponding to |
115| which other items. |
116| |
117| For example, if some rule expands symbol 5 into the sequence of |
118| symbols 8 3 20, the symbol 8 can be the beginning of the data for |
119| symbol 5, so the bit [8 - ntokens] in first[5 - ntokens] (= FIRST |
120| (5)) is set. |
121`------------------------------------------------------------------*/
2fa6973e
AD
122
123static void
124set_firsts (void)
125{
c7bd07f7 126 symbol_number i, j;
2fa6973e 127
914feea9
AD
128 firsts = bitsetv_create (nvars, nvars, BITSET_FIXED);
129
2fa6973e 130 for (i = ntokens; i < nsyms; i++)
bb7c2bc6 131 for (j = 0; derives[i - ntokens][j]; ++j)
914feea9 132 {
e9690142
JD
133 item_number sym = derives[i - ntokens][j]->rhs[0];
134 if (ISVAR (sym))
135 bitset_set (FIRSTS (i), sym - ntokens);
914feea9 136 }
2fa6973e 137
273a74fa 138 if (trace_flag & trace_sets)
427c0dda 139 bitsetv_matrix_dump (stderr, "RTC: Firsts Input", firsts);
65ccf9fc 140 bitsetv_reflexive_transitive_closure (firsts);
273a74fa 141 if (trace_flag & trace_sets)
427c0dda 142 bitsetv_matrix_dump (stderr, "RTC: Firsts Output", firsts);
2fa6973e 143
273a74fa 144 if (trace_flag & trace_sets)
9bfe901c 145 print_firsts ();
2fa6973e
AD
146}
147
148/*-------------------------------------------------------------------.
149| Set FDERIVES to an NVARS by NRULES matrix of bits indicating which |
150| rules can help derive the beginning of the data for each |
151| nonterminal. |
152| |
153| For example, if symbol 5 can be derived as the sequence of symbols |
154| 8 3 20, and one of the rules for deriving symbol 8 is rule 4, then |
155| the [5 - NTOKENS, 4] bit in FDERIVES is set. |
156`-------------------------------------------------------------------*/
d0fb370f 157
4a120d45 158static void
d2729d44 159set_fderives (void)
d0fb370f 160{
c7bd07f7
PE
161 symbol_number i, j;
162 rule_number k;
d0fb370f 163
4b3d3a8e 164 fderives = bitsetv_create (nvars, nrules, BITSET_FIXED);
d0fb370f 165
2fa6973e 166 set_firsts ();
d0fb370f 167
1cbcf2e7
AD
168 for (i = ntokens; i < nsyms; ++i)
169 for (j = ntokens; j < nsyms; ++j)
d8a0245c 170 if (bitset_test (FIRSTS (i), j - ntokens))
e9690142
JD
171 for (k = 0; derives[j - ntokens][k]; ++k)
172 bitset_set (FDERIVES (i), derives[j - ntokens][k]->number);
d0fb370f 173
273a74fa 174 if (trace_flag & trace_sets)
9bfe901c 175 print_fderives ();
d0fb370f 176
914feea9 177 bitsetv_free (firsts);
d0fb370f 178}
1565b720 179
2fa6973e 180\f
d0fb370f 181
2fa6973e 182void
da2a7671 183new_closure (unsigned int n)
d0fb370f 184{
da2a7671 185 itemset = xnmalloc (n, sizeof *itemset);
d0fb370f 186
4b3d3a8e 187 ruleset = bitset_create (nrules, BITSET_FIXED);
d0fb370f 188
2fa6973e 189 set_fderives ();
d0fb370f
RS
190}
191
192
2fa6973e 193
d0fb370f 194void
f6fbd3da 195closure (item_number *core, size_t n)
d0fb370f 196{
576890b7 197 /* Index over CORE. */
f6fbd3da 198 size_t c;
576890b7 199
d2b04478 200 /* A bit index over RULESET. */
c7bd07f7 201 rule_number ruleno;
d0fb370f 202
613f5e1a
AD
203 bitset_iterator iter;
204
273a74fa 205 if (trace_flag & trace_sets)
427c0dda 206 print_closure ("input", core, n);
c87d4863 207
643a5994 208 bitset_zero (ruleset);
d0fb370f 209
643a5994
AD
210 for (c = 0; c < n; ++c)
211 if (ISVAR (ritem[core[c]]))
212 bitset_or (ruleset, ruleset, FDERIVES (ritem[core[c]]));
d0fb370f 213
6ce2d93a 214 /* core is sorted on item index in ritem, which is sorted on rule number.
71b61d4d 215 Compute itemset with the same sort. */
b09f4f48 216 nitemset = 0;
576890b7 217 c = 0;
613f5e1a
AD
218 BITSET_FOR_EACH (iter, ruleset, ruleno, 0)
219 {
c7bd07f7 220 item_number itemno = rules[ruleno].rhs - ritem;
613f5e1a 221 while (c < n && core[c] < itemno)
e9690142
JD
222 {
223 itemset[nitemset] = core[c];
224 nitemset++;
225 c++;
226 }
b09f4f48
JD
227 itemset[nitemset] = itemno;
228 nitemset++;
613f5e1a 229 };
d0fb370f 230
576890b7
AD
231 while (c < n)
232 {
b09f4f48
JD
233 itemset[nitemset] = core[c];
234 nitemset++;
576890b7
AD
235 c++;
236 }
d0fb370f 237
273a74fa 238 if (trace_flag & trace_sets)
b09f4f48 239 print_closure ("output", itemset, nitemset);
d0fb370f
RS
240}
241
242
243void
2fa6973e 244free_closure (void)
d0fb370f 245{
afbb696d 246 free (itemset);
dfdb1797 247 bitset_free (ruleset);
914feea9 248 bitsetv_free (fderives);
d0fb370f 249}