123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378 |
- /* Subroutines for bison
- Copyright (C) 1984 Bob Corbett and Free Software Foundation, Inc.
- BISON is distributed in the hope that it will be useful, but WITHOUT ANY
- WARRANTY. No author or distributor accepts responsibility to anyone
- for the consequences of using it or for whether it serves any
- particular purpose or works at all, unless he says so in writing.
- Refer to the BISON General Public License for full details.
- Everyone is granted permission to copy, modify and redistribute BISON,
- but only under the conditions described in the BISON General Public
- License. A copy of this license is supposed to have been given to you
- along with BISON so you can know your rights and responsibilities. It
- should be in a file named COPYING. Among other things, the copyright
- notice and this notice must be preserved on all copies.
- In other words, you are welcome to use, share and improve this program.
- You are forbidden to forbid anyone else to use, share and improve
- what you give them. Help stamp out software-hoarding! */
- /* subroutines of file LR0.c.
- Entry points:
- closure (items, n)
- Given a vector of item numbers items, of length n,
- set up ruleset and itemset to indicate what rules could be run
- and which items could be accepted when those items are the active ones.
- ruleset contains a bit for each rule. closure sets the bits
- for all rules which could potentially describe the next input to be read.
- itemset is a vector of item numbers; itemsetend points to just beyond the end
- of the part of it that is significant.
- closure places there the indices of all items which represent units of
- input that could arrive next.
- initialize_closure (n)
- Allocates the itemset and ruleset vectors,
- and precomputes useful data so that closure can be called.
- n is the number of elements to allocate for itemset.
- finalize_closure ()
- Frees itemset, ruleset and internal data.
- */
- #include <stdio.h>
- #include "machine.h"
- #include "new.h"
- #include "gram.h"
- extern short **derives;
- short *itemset;
- short *itemsetend;
- static unsigned *ruleset;
- /* internal data. See comments before set_fderives and set_firsts. */
- static unsigned *fderives;
- static unsigned *firsts;
- /* number of words required to hold a bit for each rule */
- static int rulesetsize;
- /* number of words required to hold a bit for each variable */
- static int varsetsize;
- initialize_closure(n)
- int n;
- {
- itemset = NEW2(n, short);
- rulesetsize = WORDSIZE(nrules + 1);
- ruleset = NEW2(rulesetsize, unsigned);
- set_fderives();
- }
- /* set fderives to an nvars by nrules matrix of bits
- indicating which rules can help derive the beginning of the data
- for each nonterminal. For example, if symbol 5 can be derived as
- the sequence of symbols 8 3 20, and one of the rules for deriving
- symbol 8 is rule 4, then the [5 - ntokens, 4] bit in fderives is set. */
- set_fderives()
- {
- register unsigned *rrow;
- register unsigned *vrow;
- register int j;
- register unsigned mask;
- register unsigned cword;
- register short *rp;
- int ruleno;
- int i;
- fderives = NEW2(nvars * rulesetsize, unsigned) - ntokens * rulesetsize;
- set_firsts();
- rrow = fderives + ntokens * rulesetsize;
- for (i = ntokens; i < nsyms; i++)
- {
- vrow = firsts + ((i - ntokens) * varsetsize);
- cword = *vrow++;
- mask = 1;
- for (j = ntokens; j < nsyms; j++)
- {
- if (cword & mask)
- {
- rp = derives[j];
- while ((ruleno = *rp++) > 0)
- {
- SETBIT(rrow, ruleno);
- }
- }
- mask <<= 1;
- if (mask == 0 && j + 1 < nsyms)
- {
- cword = *vrow++;
- mask = 1;
- }
- }
- vrow += varsetsize;
- rrow += rulesetsize;
- }
- #ifdef DEBUG
- print_fderives();
- #endif
- FREE(firsts);
- }
- /* set firsts to be an nvars by nvars bit matrix indicating which items
- can represent the beginning of the input corresponding to which other items.
- For example, if some rule expands symbol 5 into the sequence of symbols 8 3 20,
- the symbol 8 can be the beginning of the data for symbol 5,
- so the bit [8 - ntokens, 5 - ntokens] in firsts is set. */
- set_firsts()
- {
- register unsigned *row;
- /* register int done; JF unused */
- register int symbol;
- register short *sp;
- register int rowsize;
- int i;
- varsetsize = rowsize = WORDSIZE(nvars);
- firsts = NEW2(nvars * rowsize, unsigned);
- row = firsts;
- for (i = ntokens; i < nsyms; i++)
- {
- sp = derives[i];
- while (*sp >= 0)
- {
- symbol = ritem[rrhs[*sp++]];
- if (ISVAR(symbol))
- {
- symbol -= ntokens;
- SETBIT(row, symbol);
- }
- }
- row += rowsize;
- }
- RTC(firsts, nvars);
- #ifdef DEBUG
- print_firsts();
- #endif
- }
- closure(core, n)
- short *core;
- int n;
- {
- register int ruleno;
- register unsigned word;
- register unsigned mask;
- register short *csp;
- register unsigned *dsp;
- register unsigned *rsp;
- short *csend;
- unsigned *rsend;
- int symbol;
- int itemno;
- rsp = ruleset;
- rsend = ruleset + rulesetsize;
- csend = core + n;
- if (n == 0)
- {
- dsp = fderives + start_symbol * rulesetsize;
- while (rsp < rsend)
- *rsp++ = *dsp++;
- }
- else
- {
- while (rsp < rsend)
- *rsp++ = 0;
- csp = core;
- while (csp < csend)
- {
- symbol = ritem[*csp++];
- if (ISVAR(symbol))
- {
- dsp = fderives + symbol * rulesetsize;
- rsp = ruleset;
- while (rsp < rsend)
- *rsp++ |= *dsp++;
- }
- }
- }
- ruleno = 0;
- itemsetend = itemset;
- csp = core;
- rsp = ruleset;
- while (rsp < rsend)
- {
- word = *rsp++;
- if (word == 0)
- {
- ruleno += BITS_PER_WORD;
- }
- else
- {
- mask = 1;
- while (mask)
- {
- if (word & mask)
- {
- itemno = rrhs[ruleno];
- while (csp < csend && *csp < itemno)
- *itemsetend++ = *csp++;
- *itemsetend++ = itemno;
- }
- mask <<= 1;
- ruleno++;
- }
- }
- }
- while (csp < csend)
- *itemsetend++ = *csp++;
- #ifdef DEBUG
- print_closure(n);
- #endif
- }
- finalize_closure()
- {
- FREE(itemset);
- FREE(ruleset);
- FREE(fderives + ntokens * rulesetsize);
- }
- #ifdef DEBUG
- print_closure(n)
- int n;
- {
- register short *isp;
- printf("\n\nn = %d\n\n", n);
- for (isp = itemset; isp < itemsetend; isp++)
- printf(" %d\n", *isp);
- }
- print_firsts()
- {
- register int i;
- register int j;
- register unsigned *rowp;
- register unsigned cword;
- register unsigned mask;
- extern char **tags;
- printf("\n\n\nFIRSTS\n\n");
- for (i = ntokens; i < nsyms; i++)
- {
- printf("\n\n%s firsts\n\n", tags[i]);
- rowp = firsts + ((i - ntokens) * vrowsize);
- cword = *rowp++;
- mask = 1;
- for (j = 0; j < nsyms; j++)
- {
- if (cword & mask)
- printf(" %s\n", tags[j + ntokens]);
- mask <<= 1;
- if (mask == 0 && j + 1 < nsyms)
- {
- cword = *rowp++;
- mask = 1;
- }
- }
- }
- }
- print_fderives()
- {
- register int i;
- register int j;
- register unsigned *rp;
- register unsigned cword;
- register unsigned mask;
- extern char **tags;
- printf("\n\n\nFDERIVES\n");
- for (i = ntokens; i < nsyms; i++)
- {
- printf("\n\n%s derives\n\n", tags[i]);
- rp = fderives + i * rrowsize;
- cword = *rp++;
- mask = 1;
- for (j = 0; j <= nrules; j++)
- {
- if (cword & mask)
- printf(" %d\n", j);
- mask <<= 1;
- if (mask == 0 && j + 1 < nrules)
- {
- cword = *rp++;
- mask = 1;
- }
- }
- }
- fflush(stdout);
- }
- #endif
|