8bb395b147
Use C89 size_t for sizes from sizeof() or to malloc() or realloc().
Remove obsolete (unsigned) casts. Sizes were unsigned int in
traditional C but are size_t in C89.
Silence some clang warnings. Add the second pair of round brackets in
`while ((ff = ff->ff_next))` to silence -Wparentheses. Change
`if (nc_first(...))/*nothing*/;` to `(void)nc_first(...);` to silence
-Wempty-body. The code in compute.c nc_first() had the form
`if (x) if (y) s; else t;`. The old indentation (before 10717cc
)
suggests that the "else" belongs to the 2nd "if", so add braces like
`if (x) { if (y) s; else t; }` to silence -Wdangling-else.
Shuffle extern function declarations. Add missing declaration for
LLparse(). Stop declaring RENAME(); it doesn't exist. Move some
declarations from main.c to extern.h, so the C compiler may check that
the declarations are compatible with the function definitions.
Assume that standard C89 remove() is available and doesn't need the
UNLINK() wrapper.
In lib/incl, don't need to include <stdio.h> nor <stdlib.h> to use
assert().
Remove alloc.h. If you don't clean your build, then an outdated
BUILDDIR/obj/util/LLgen/headers/alloc.h will survive but should not
cause harm, because nothing includes it. Don't need to remove alloc.h
from util/LLgen/distr.sh, because it isn't there.
Run the bootstrap to rebuild LLgen.c, Lpars.c, tokens.c.
1539 lines
35 KiB
Plaintext
1539 lines
35 KiB
Plaintext
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
/*
|
|
compile with -DNOFIRST to disable firstset optimization
|
|
-DFOLLOW_OPT to enable followset optimization
|
|
NOTE: Followset optimization is not
|
|
supported when using -s option of LLgen
|
|
-DDEBUG to print debug information
|
|
*/
|
|
|
|
extern int LLsymb;
|
|
extern int LLstartsymb;
|
|
|
|
void LLmessage(int);
|
|
|
|
|
|
struct stacks {
|
|
|
|
/* Acces to the stacks is through a 'dynamic array' of pointers
|
|
* to the heads. We implemented it this way to save on the number
|
|
* of Malloc() calls. nr_heads is the number of heads; heads_buf_size
|
|
* is the current size of heads_buf.
|
|
*/
|
|
|
|
int nr_heads;
|
|
struct stack_elt **heads_buf;
|
|
int heads_buf_size;
|
|
|
|
/* r_rec contains nonterminals already tried to build a new
|
|
* stack with, to prevent right-recursive rules making the
|
|
* process loop forever
|
|
*/
|
|
|
|
char r_rec[(LLNNONTERMINALS + 7)/8];
|
|
|
|
/* join_array contains pointers to already substituted stack
|
|
* elements, so that if the same nonterminal turns up again
|
|
* we can make a link
|
|
*/
|
|
|
|
struct stack_elt *join_array[LLNNONTERMINALS];
|
|
|
|
/* cleanup_buf contains pointerts to elements that can possibly
|
|
* be deleted. Again this is implemented as a `growing array'.
|
|
* Although it's not so clean to do it this way, it DOES save
|
|
* a lot of time, mainly because much less pointer manipulation
|
|
* is required, and because it's not necessary to deallocate
|
|
* the buffer after each turn. Just set nr_cleanups to 0...
|
|
*/
|
|
|
|
int nr_cleanups;
|
|
int cleanup_buf_size;
|
|
struct stack_elt **cleanup_buf;
|
|
|
|
/* visited_buf contains pointers to elements whose flags
|
|
* need to be cleared
|
|
*/
|
|
|
|
int nr_visited;
|
|
int visited_buf_size;
|
|
struct stack_elt **visited_buf;
|
|
|
|
|
|
/* start_seen indicates if the last prediction phase
|
|
* has matched the start symbol
|
|
*/
|
|
|
|
int start_seen;
|
|
|
|
/* exp_terminals will contain the terminals that are `on top'
|
|
* of the prediction graph after a prediction phase
|
|
*/
|
|
|
|
char exp_terminals[LLSETSIZE];
|
|
|
|
/* check_run_ok indicates whether a stack element can be deleted
|
|
* or not
|
|
*/
|
|
|
|
int check_run_ok;
|
|
};
|
|
|
|
|
|
#ifdef DEBUG
|
|
static int allocates = 0;
|
|
static int deallocates = 0;
|
|
static int max_in_use = 0;
|
|
static int edge_allocates = 0;
|
|
static int edge_deallocates = 0;
|
|
static int edge_max_in_use = 0;
|
|
#endif
|
|
|
|
static int grammar_index = 0;
|
|
|
|
/* The grammar should only be build the first time we enter the
|
|
* recovery routine. grammar_read == 0 indicates this has not
|
|
* been done yet
|
|
*/
|
|
|
|
static int grammar_read = 0;
|
|
|
|
/* 'terminals' is an array indexed by the number of a terminal and links
|
|
* all rules containing this terminal in the RHS
|
|
*/
|
|
|
|
static struct terminal *terminals;
|
|
|
|
/* 'nonterminals' is an array indexed by the number of a nonterminal
|
|
* and contains all rules with this nonterminal in the LHS and links all
|
|
* rules containing this nonterminal in the RHS
|
|
*/
|
|
|
|
static struct nonterminal *nonterminals;
|
|
|
|
|
|
/* These functions must be called instead of the original functions in
|
|
* 'malloc.h'. They offer a checking allocation mechanism.
|
|
*/
|
|
static void *Malloc(size_t);
|
|
static void *Realloc(void *, size_t);
|
|
|
|
|
|
|
|
/* These functions build the grammar */
|
|
static void init_grammar(void);
|
|
static void build_grammar(void);
|
|
static struct lhs *build_rule(void);
|
|
static struct symbol *build_rhs(struct lhs*);
|
|
static struct symbol *make_link(struct symbol*);
|
|
|
|
|
|
|
|
/* These functions operate on the stacks */
|
|
static int optimize(struct stacks*, struct symbol*, int);
|
|
static void read_token(void);
|
|
static void start_stack(struct stacks*, int, int);
|
|
static void continuation(struct stacks*, int, int);
|
|
static struct stack_elt *push_rule(struct stack_elt*, struct symbol*);
|
|
static void new_head(struct stacks*, struct stack_elt*);
|
|
static void to_delete(struct stacks*, struct stack_elt*);
|
|
static void substitute(struct stacks*, struct stack_elt*, int);
|
|
static int join(struct stacks*, struct stack_elt*, int);
|
|
static int path(struct stack_elt*, struct stack_elt*);
|
|
static int part_of_loop(struct stack_elt*);
|
|
static void generate_heads(struct stacks*, struct stack_elt*, int);
|
|
static void delete(struct stacks*, struct stack_elt*);
|
|
static void hyp_run(struct stack_elt*);
|
|
static void check_run(struct stacks*, struct stack_elt*);
|
|
static struct stack_elt *split(struct stack_elt*);
|
|
#ifdef DEBUG
|
|
static void test(struct stacks*);
|
|
static void dump_stack(struct stack_elt*, int);
|
|
#endif
|
|
static void clear_flags(struct stack_elt*, char);
|
|
static void clear_gen_flags(struct stacks*);
|
|
static void match_heads(struct stacks*, int);
|
|
static void cleanup(struct stacks*);
|
|
static void initialize(struct stacks*);
|
|
static void calculate(struct stacks*, int);
|
|
static void kill_stack(struct stacks *stack);
|
|
void LLnc_recover(void);
|
|
|
|
|
|
|
|
static void *Malloc(size_t size)
|
|
{
|
|
void *p;
|
|
|
|
if ((p = malloc(size)) == NULL) {
|
|
fprintf(stderr, "fatal error: out of memory\n");
|
|
exit(1);
|
|
}
|
|
return p;
|
|
}
|
|
|
|
|
|
static void *Realloc(void *ptr, size_t size)
|
|
{
|
|
void *p;
|
|
|
|
if ((p = realloc(ptr, size)) == NULL) {
|
|
fprintf(stderr, "fatal error: out of memory\n");
|
|
exit(1);
|
|
}
|
|
return p;
|
|
}
|
|
|
|
|
|
static void init_grammar(void)
|
|
{
|
|
/* Allocate and initialize an array for terminals and nonterminals */
|
|
|
|
int i;
|
|
|
|
terminals = (struct terminal *)
|
|
Malloc(LLFIRST_NT * sizeof(struct terminal));
|
|
for (i = 0; i < LLFIRST_NT; i++) {
|
|
(terminals + i)->link = (struct symbol *)0;
|
|
}
|
|
|
|
nonterminals = (struct nonterminal *)
|
|
Malloc(LLNNONTERMINALS * sizeof(struct nonterminal));
|
|
for (i = 0; i < LLNNONTERMINALS; i++) {
|
|
(nonterminals + i)->rule = (struct lhs *)0;
|
|
(nonterminals + i)->link = (struct symbol *)0;
|
|
}
|
|
}
|
|
|
|
|
|
static void build_grammar(void)
|
|
{
|
|
/* Build a rule for every nonterminal. The LHS must be saved first because
|
|
* of the fact that the right side of an assignment statement (in C) will
|
|
* be evaluated before the left side
|
|
*/
|
|
int nt, j;
|
|
|
|
init_grammar();
|
|
for (j = 0; j < LLNNONTERMINALS; j++) {
|
|
nt = LLgrammar[grammar_index];
|
|
(nonterminals + nt - LLFIRST_NT)->rule = build_rule();
|
|
}
|
|
}
|
|
|
|
|
|
static struct lhs *build_rule(void)
|
|
{
|
|
/* Build LHS and call a funcion to create RHS */
|
|
|
|
struct lhs *l;
|
|
int j;
|
|
|
|
l = (struct lhs *)Malloc(sizeof(struct lhs));
|
|
l->nr = LLgrammar[grammar_index++];
|
|
|
|
/* Build first set */
|
|
for (j = 0; j < LLSETSIZE; j++) {
|
|
l->first[j] = LLgrammar[grammar_index++];
|
|
}
|
|
|
|
/* Build follow set */
|
|
for (j = 0; j < LLSETSIZE; j++) {
|
|
l->follow[j] = LLgrammar[grammar_index++];
|
|
}
|
|
|
|
l->empty = LLgrammar[grammar_index++]; /* Can NT produce empty? */
|
|
|
|
l->rhs = build_rhs(l);
|
|
|
|
return l;
|
|
}
|
|
|
|
|
|
static struct symbol *build_rhs(struct lhs *l)
|
|
{
|
|
/* Build RHS by creating structs for all symbols including ALT and
|
|
* EORULE. Also call a function for linking same terminals and
|
|
* nonterminals.
|
|
*/
|
|
|
|
struct symbol *r;
|
|
|
|
r = (struct symbol *)Malloc(sizeof(struct symbol));
|
|
if (LLgrammar[grammar_index] == LLALT) {
|
|
grammar_index++;
|
|
r->x = LLALT;
|
|
r->nr = -1; /* Not applicable */
|
|
r->link = (struct symbol *)0; /* Not applicable */
|
|
r->next = build_rhs(l);
|
|
r->lhs = l;
|
|
}
|
|
else if (LLgrammar[grammar_index] == LLEORULE) {
|
|
grammar_index++;
|
|
r->x = LLEORULE;
|
|
r->nr = -1; /* Not applicable */
|
|
r->link = (struct symbol *)0; /* Not applicable */
|
|
r->next = (struct symbol *)0; /* Not applicable */
|
|
r->lhs = l;
|
|
}
|
|
else if (LLgrammar[grammar_index] < LLFIRST_NT) {
|
|
r->x = LLTERMINAL;
|
|
r->nr = LLgrammar[grammar_index++];
|
|
r->link = make_link(r);
|
|
r->next = build_rhs(l);
|
|
r->lhs = l;
|
|
}
|
|
else {
|
|
r->x = LLNONTERMINAL;
|
|
r->nr = LLgrammar[grammar_index++];
|
|
r->link = make_link(r);
|
|
r->next = build_rhs(l);
|
|
r->lhs = l;
|
|
}
|
|
return r;
|
|
}
|
|
|
|
|
|
static struct symbol *make_link(struct symbol *r)
|
|
{
|
|
/* Link same terminals and nonterminals. Every new symbol is appended
|
|
* in front of the corresponding list for efficiency.
|
|
*/
|
|
|
|
struct symbol *tmp;
|
|
|
|
if (r->nr < LLFIRST_NT) {
|
|
/* Terminal array */
|
|
tmp = (terminals + r->nr)->link;
|
|
(terminals + r->nr)->link = r;
|
|
}
|
|
else { /* Nonterminal array */
|
|
tmp = (nonterminals + (r->nr - LLFIRST_NT))->link;
|
|
(nonterminals + (r->nr - LLFIRST_NT))->link = r;
|
|
}
|
|
return tmp;
|
|
}
|
|
|
|
|
|
/*****************************************************************************/
|
|
|
|
|
|
static int optimize(struct stacks* stack, struct symbol *symb_ptr, int l_ahead)
|
|
/* Return 1 if rule symb_ptr can start with terminal l_ahead, else return 0.
|
|
* The array with expected terminals will also be filled in.
|
|
*/
|
|
{
|
|
struct lhs *l;
|
|
int i;
|
|
|
|
#ifdef NOFIRST
|
|
return(1);
|
|
#else
|
|
|
|
if ((l_ahead <= 0) || (l_ahead == EOFILE)) return 1;
|
|
|
|
switch(symb_ptr->x) {
|
|
case LLTERMINAL:
|
|
LLPUTIN(stack->exp_terminals, LLindex[symb_ptr->nr]);
|
|
if (symb_ptr->nr != l_ahead) return 0;
|
|
else return 1;/*???*/
|
|
|
|
case LLNONTERMINAL:
|
|
l = (nonterminals + symb_ptr->nr - LLFIRST_NT)->rule;
|
|
if (LLIN(l->first, LLindex[l_ahead])) return 1;
|
|
else if (l->empty) {
|
|
/* Try next symbol */
|
|
return optimize(stack, symb_ptr->next, l_ahead);
|
|
}
|
|
else {
|
|
for (i = 0; i < LLSETSIZE; i++) {
|
|
stack->exp_terminals[i] |= (l->first)[i];
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
default:
|
|
|
|
#ifndef FOLLOW_OPT
|
|
return(1);
|
|
#else
|
|
|
|
l = (nonterminals + symb_ptr->lhs->nr - LLFIRST_NT)->rule;
|
|
|
|
if (LLIN(l->follow, LLindex[l_ahead])) return 1;
|
|
else {
|
|
for (i = 0; i < LLSETSIZE; i++) {
|
|
stack->exp_terminals[i] |= (l->follow)[i];
|
|
}
|
|
return 0;
|
|
}
|
|
#endif /*FOLLOW_OPT */
|
|
}
|
|
#endif /* NOFIRST */
|
|
}
|
|
|
|
|
|
static void read_token(void)
|
|
/* Read token and put it in global variable LLsymb, skipping
|
|
* invalid tokens
|
|
*/
|
|
{
|
|
LLsymb = LL_LEXI();
|
|
while (LLindex[LLsymb] < 0) {
|
|
/* Skip garbage tokens */
|
|
LLmessage(0);
|
|
LLsymb = LL_LEXI();
|
|
}
|
|
}
|
|
|
|
|
|
static void start_stack(struct stacks *stack, int base, int l_ahead)
|
|
/* Start stack on base symbol base with lookahead l_ahead */
|
|
{
|
|
struct stack_elt *bottom, *top;
|
|
struct symbol *symb_ptr;
|
|
int i;
|
|
|
|
/* Find first applicable rule */
|
|
symb_ptr = (terminals + base)->link;
|
|
|
|
/* Now try all applicable rules */
|
|
while (symb_ptr != (struct symbol *)0) {
|
|
|
|
/* If the current rule cannot start with l_ahead,
|
|
* try the next one
|
|
*/
|
|
if (!optimize(stack, symb_ptr->next, l_ahead)) {
|
|
symb_ptr = symb_ptr->link;
|
|
continue;
|
|
}
|
|
|
|
if ( (symb_ptr->next->x == LLTERMINAL)
|
|
|| (symb_ptr->next->x == LLNONTERMINAL)
|
|
) {
|
|
/* Allocate an end-of-stack */
|
|
#ifdef DEBUG
|
|
allocates++;
|
|
if (allocates - deallocates > max_in_use) {
|
|
max_in_use = allocates - deallocates;
|
|
}
|
|
#endif
|
|
bottom = (struct stack_elt *)
|
|
Malloc(sizeof(struct stack_elt));
|
|
bottom->edges = (struct edge *)0;
|
|
bottom->nr = LLEOSTACK;
|
|
bottom->flags = 0;
|
|
bottom->nr_nexts = 0;
|
|
bottom->ref_count = 0;
|
|
bottom->hyp_ref_count = -1;
|
|
|
|
/* And use the rule to build a stack on it */
|
|
top = push_rule(bottom, symb_ptr->next);
|
|
|
|
/* Remember that we're now trying to match the LHS
|
|
* of the used rule
|
|
*/
|
|
bottom->matched = symb_ptr->lhs->nr;
|
|
|
|
if (top->nr >= LLFIRST_NT) {
|
|
substitute(stack, top, l_ahead);
|
|
}
|
|
else {
|
|
new_head(stack, top);
|
|
}
|
|
|
|
/* Perhaps this only has produced an empty stack, in
|
|
* that case bottom can be deallocated.
|
|
*/
|
|
if (bottom->ref_count == 0) {
|
|
to_delete(stack, bottom);
|
|
}
|
|
}
|
|
else {
|
|
/* base was the last element of the rule, so we
|
|
* figure we have matched the LHS of this rule.
|
|
*/
|
|
if (symb_ptr->lhs->nr == LLstartsymb) {
|
|
stack->start_seen = 1;
|
|
}
|
|
|
|
continuation(stack, symb_ptr->lhs->nr, l_ahead);
|
|
}
|
|
symb_ptr = symb_ptr->link;
|
|
}
|
|
|
|
|
|
/* Reinitialize some arrays */
|
|
for (i = 0; i < (LLNNONTERMINALS + 7)/8; i++) {
|
|
stack->r_rec[i] = (char) 0;
|
|
}
|
|
|
|
for (i = 0; i < LLNNONTERMINALS; i++) {
|
|
stack->join_array[i] = (struct stack_elt *)0;
|
|
}
|
|
|
|
/* Delete all HEAD flags */
|
|
for (i = 0; i < stack->nr_heads; i++) {
|
|
(*(stack->heads_buf + i))->flags &= ~LLHEAD;
|
|
}
|
|
|
|
/* Delete flags turned on by 'generate_heads()' */
|
|
clear_gen_flags(stack);
|
|
/* Try to delete elements on cleanup_buf */
|
|
cleanup(stack);
|
|
}
|
|
|
|
|
|
static void continuation(struct stacks *stack, int nt, int l_ahead)
|
|
/* We have 'eaten' a whole stack, and think we recognized nt. Now
|
|
look for rules that we can proceed with, ie containing nt in the RHS.
|
|
Each rule found will be developed untill a terminal is at the top
|
|
of the stack.*/
|
|
{
|
|
|
|
struct symbol *symb_ptr;
|
|
struct stack_elt *bottom, *top;
|
|
|
|
/* If we've already tried this nt, don't do it again.
|
|
* Otherwise we may loop forever on right-recursive rules
|
|
*/
|
|
if (LLIN(stack->r_rec, nt - LLFIRST_NT)) return;
|
|
|
|
/* Mark that we have looked for a continuation for nt */
|
|
LLPUTIN(stack->r_rec, nt - LLFIRST_NT);
|
|
|
|
/* Find first applicable rule */
|
|
symb_ptr = (nonterminals + nt - LLFIRST_NT)->link;
|
|
|
|
/* Try all applicable rules */
|
|
while (symb_ptr != (struct symbol *)0) {
|
|
|
|
/* If the current rule cannot start with l_ahead,
|
|
* try the next one
|
|
*/
|
|
if (!optimize(stack, symb_ptr->next, l_ahead)) {
|
|
symb_ptr = symb_ptr->link;
|
|
continue;
|
|
}
|
|
|
|
if ( (symb_ptr->next->x == LLTERMINAL)
|
|
|| (symb_ptr->next->x == LLNONTERMINAL)
|
|
) {
|
|
#ifdef DEBUG
|
|
allocates++;
|
|
if (allocates - deallocates > max_in_use) {
|
|
max_in_use = allocates - deallocates;
|
|
}
|
|
#endif
|
|
bottom = (struct stack_elt *)
|
|
Malloc(sizeof(struct stack_elt));
|
|
bottom->edges = (struct edge *)0;
|
|
bottom->nr = LLEOSTACK;
|
|
bottom->flags = 0;
|
|
bottom->nr_nexts = 0;
|
|
bottom->ref_count = 0;
|
|
bottom->hyp_ref_count = -1;
|
|
|
|
/* Use the rule to build a stack on bottom */
|
|
top = push_rule(bottom, symb_ptr->next);
|
|
|
|
/* Remember that we're now trying to match the LHS
|
|
* of the used rule
|
|
*/
|
|
bottom->matched = symb_ptr->lhs->nr;
|
|
|
|
if (top->nr >= LLFIRST_NT) {
|
|
substitute(stack, top, l_ahead);
|
|
}
|
|
else {
|
|
new_head(stack, top);
|
|
}
|
|
|
|
/* Perhaps this only has produced an empty stack, in
|
|
* that case bottom can be deallocated.
|
|
*/
|
|
if (bottom->ref_count == 0) {
|
|
delete(stack, bottom);
|
|
}
|
|
}
|
|
else {
|
|
/* Stack is still empty */
|
|
if (symb_ptr->lhs->nr == LLstartsymb) {
|
|
stack->start_seen = 1;
|
|
}
|
|
|
|
continuation(stack, symb_ptr->lhs->nr, l_ahead);
|
|
}
|
|
|
|
symb_ptr = symb_ptr->link;
|
|
}
|
|
}
|
|
|
|
|
|
static struct stack_elt *push_rule(struct stack_elt *element,
|
|
struct symbol *symb_ptr)
|
|
/* Append the rule symb_ptr to stack element 'element'. Return a
|
|
* pointer to the new top of the stack
|
|
*/
|
|
{
|
|
struct stack_elt *se, *top;
|
|
|
|
if ( (symb_ptr->next->x == LLTERMINAL)
|
|
|| (symb_ptr->next->x == LLNONTERMINAL)
|
|
) {
|
|
top = push_rule(element, symb_ptr->next);
|
|
}
|
|
else {
|
|
top = element;
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
allocates++;
|
|
if (allocates - deallocates > max_in_use) {
|
|
max_in_use = allocates - deallocates;
|
|
}
|
|
#endif
|
|
|
|
se = (struct stack_elt *)Malloc(sizeof(struct stack_elt));
|
|
se->flags = 0;
|
|
se->nr = symb_ptr->nr;
|
|
se->ref_count = 0;
|
|
se->hyp_ref_count = -1;
|
|
se->matched = -1;
|
|
se->nr_nexts = 1;
|
|
|
|
#ifdef DEBUG
|
|
edge_allocates++;
|
|
if (edge_allocates - edge_deallocates > edge_max_in_use) {
|
|
edge_max_in_use = edge_allocates - edge_deallocates;
|
|
}
|
|
#endif
|
|
|
|
se->edges = (struct edge *)Malloc(sizeof(struct edge));
|
|
se->edges->ptr = top;
|
|
se->edges->flags = 0;
|
|
|
|
top->ref_count++;
|
|
return se;
|
|
}
|
|
|
|
|
|
static void new_head(struct stacks *stack, struct stack_elt *ptr)
|
|
/* Make ptr a head of stack */
|
|
{
|
|
|
|
/* Is this already a head?*/
|
|
if (ptr->flags & LLHEAD) return;
|
|
|
|
if (stack->heads_buf_size == 0) {
|
|
stack->heads_buf_size = LLHEADS_BUF_INCR;
|
|
stack->heads_buf = (struct stack_elt **)
|
|
Malloc(LLHEADS_BUF_INCR * sizeof(struct stack_elt *));
|
|
}
|
|
else if (stack->nr_heads == stack->heads_buf_size) {
|
|
/* buffer full? */
|
|
stack->heads_buf_size += LLHEADS_BUF_INCR;
|
|
stack->heads_buf = (struct stack_elt **)
|
|
Realloc(stack->heads_buf,
|
|
stack->heads_buf_size *
|
|
sizeof(struct stack_elt *)
|
|
);
|
|
}
|
|
|
|
*(stack->heads_buf + stack->nr_heads) = ptr; /* Add at the tail */
|
|
stack->nr_heads++; /* Increase number of heads */
|
|
ptr->flags |= LLHEAD; /* Mark it as a head */
|
|
ptr->ref_count++; /* Increase reference count */
|
|
LLPUTIN(stack->exp_terminals, LLindex[ptr->nr]);
|
|
}
|
|
|
|
|
|
static void to_delete(struct stacks *stack, struct stack_elt *ptr)
|
|
/* Remember that ptr has to be deleted */
|
|
{
|
|
|
|
int i;
|
|
|
|
#ifdef NOCLEAN
|
|
return;
|
|
#endif
|
|
|
|
|
|
|
|
for (i = 0; i < stack->nr_cleanups; i++) {
|
|
/* Check if already in buffer */
|
|
if (*(stack->cleanup_buf + i) == ptr) return;
|
|
}
|
|
|
|
if (stack->cleanup_buf_size == 0) {
|
|
stack->cleanup_buf_size = LLCLEANUP_BUF_INCR;
|
|
stack->cleanup_buf = (struct stack_elt **)
|
|
Malloc(LLCLEANUP_BUF_INCR * sizeof(struct stack_elt *));
|
|
}
|
|
else if (stack->nr_cleanups == stack->cleanup_buf_size) {
|
|
stack->cleanup_buf_size += LLCLEANUP_BUF_INCR;
|
|
stack->cleanup_buf = (struct stack_elt **)
|
|
Realloc(stack->cleanup_buf,
|
|
stack->cleanup_buf_size *
|
|
sizeof(struct stack_elt *));
|
|
}
|
|
*(stack->cleanup_buf + stack->nr_cleanups) = ptr;
|
|
stack->nr_cleanups++;
|
|
}
|
|
|
|
|
|
static void substitute(struct stacks *stack, struct stack_elt *top, int l_ahead)
|
|
/* This function substitutes the NT pointed to by 'top'. 'top' should be a top
|
|
* of a stack
|
|
*/
|
|
{
|
|
struct symbol *symb_ptr;
|
|
struct stack_elt *new_top;
|
|
|
|
/* Try to join top NT */
|
|
if (join(stack, top, l_ahead)) return;
|
|
|
|
/* Find RHS of the rule of nonterminal 'top->nr' */
|
|
symb_ptr = (nonterminals + top->nr - LLFIRST_NT)->rule->rhs;
|
|
|
|
/* Mark top as dummy */
|
|
top->flags |= LLDUMMY;
|
|
|
|
while (1) {
|
|
/* If this an empty production, search down the stack for
|
|
* terminals
|
|
*/
|
|
if ((symb_ptr->x == LLALT) || (symb_ptr->x == LLEORULE)) {
|
|
generate_heads(stack, top, l_ahead);
|
|
}
|
|
|
|
/* Skip other empty productions, they have no effect. */
|
|
while (symb_ptr->x == LLALT) {
|
|
symb_ptr = symb_ptr->next;
|
|
}
|
|
|
|
if (symb_ptr->x == LLEORULE) {
|
|
/* If there are only empty productions, the NT on top
|
|
* can be deleted
|
|
*/
|
|
if (top->ref_count == 0) {
|
|
to_delete(stack, top);
|
|
}
|
|
return;
|
|
}
|
|
|
|
/* If this rule can produce 'l_ahead' on the top of the stack
|
|
* substitute the nonterminal
|
|
*/
|
|
if (optimize(stack, symb_ptr, l_ahead)) {
|
|
new_top = push_rule(top, symb_ptr);
|
|
|
|
/* If the new element on top is a nonterminal
|
|
* substitute it, else make it a head
|
|
*/
|
|
if (new_top->nr >= LLFIRST_NT) {
|
|
substitute(stack, new_top, l_ahead);
|
|
}
|
|
else {
|
|
new_head(stack, new_top);
|
|
}
|
|
}
|
|
|
|
/* Search to next alternative */
|
|
while ( (symb_ptr->x == LLTERMINAL)
|
|
|| (symb_ptr->x == LLNONTERMINAL)
|
|
) {
|
|
symb_ptr = symb_ptr->next;
|
|
}
|
|
|
|
if (symb_ptr->x == LLEORULE) {
|
|
if (top->ref_count == 0) {
|
|
to_delete(stack, top);
|
|
}
|
|
return;
|
|
}
|
|
else {
|
|
symb_ptr = symb_ptr->next;
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
|
|
static int join(struct stacks *stack, struct stack_elt *top, int l_ahead)
|
|
/* This function tries to connect a NT on top of a stack with another stack,
|
|
* which has already substituted this NT
|
|
*/
|
|
{
|
|
struct stack_elt *se;
|
|
int size;
|
|
|
|
if ( (se = stack->join_array[top->nr - LLFIRST_NT]) ==
|
|
(struct stack_elt *)0
|
|
) {
|
|
stack->join_array[top->nr - LLFIRST_NT] = top;
|
|
return 0; /* Join not possible */
|
|
}
|
|
else {
|
|
se->nr_nexts++; /* Increase number of descendants */
|
|
|
|
#ifdef DEBUG
|
|
edge_allocates++;
|
|
if (edge_allocates - edge_deallocates > edge_max_in_use) {
|
|
edge_max_in_use = edge_allocates - edge_deallocates;
|
|
}
|
|
#endif
|
|
|
|
/* Allocate one more pointer to descendants */
|
|
size = se->nr_nexts * sizeof(struct edge);
|
|
se->edges = (struct edge *)Realloc(se->edges, size);
|
|
|
|
/* Link it */
|
|
(se->edges + se->nr_nexts - 1)->ptr = top->edges->ptr;
|
|
(se->edges + se->nr_nexts - 1)->flags = 0;
|
|
|
|
/* The successor of 'top' gets an extra predecessor.
|
|
* 'top' has always only one successor because the stacks are
|
|
* constructed in 'depth first' order
|
|
*/
|
|
top->edges->ptr->ref_count++;
|
|
|
|
|
|
#ifndef NOLOOPS
|
|
/* If we have made a new loop find all stack elements of this
|
|
* loop and mark them
|
|
*/
|
|
if (path(top->edges->ptr, se)) {
|
|
(se->edges + se->nr_nexts - 1)->flags |= LLLOOP;
|
|
(se->edges + se->nr_nexts - 1)->flags |= LLYES;
|
|
}
|
|
clear_flags(top->edges->ptr, (LLNO | LLYES));
|
|
#endif
|
|
|
|
|
|
/* Check if joined NT produces empty */
|
|
if ((nonterminals + se->nr - LLFIRST_NT)->rule->empty) {
|
|
generate_heads(stack, top, l_ahead);
|
|
}
|
|
|
|
/* Deallocate top symbol */
|
|
if (top->ref_count == 0) {
|
|
to_delete(stack, top);
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
|
|
#ifndef NOLOOPS
|
|
|
|
static int path(struct stack_elt *se1, struct stack_elt *se2)
|
|
/* If there is a path from se1 to se2 it returns 1 and marks all the paths
|
|
* betweeen these two points, otherwise it returns 0. The flags LLYES and
|
|
* LLNO are used for optimization. */
|
|
{
|
|
int i, result = 0;
|
|
|
|
if (se1 == se2) return 1;
|
|
|
|
for (i = 0; i < se1->nr_nexts; i++) {
|
|
if ( (!((se1->edges + i)->flags & LLNO))
|
|
&& (!((se1->edges + i)->flags & LLLOOP_SEARCH))
|
|
) {
|
|
(se1->edges + i)->flags |= LLLOOP_SEARCH;
|
|
|
|
if (path((se1->edges + i)->ptr, se2)) {
|
|
(se1->edges + i)->flags |= LLLOOP;
|
|
result = 1;
|
|
}
|
|
else {
|
|
(se1->edges + i)->flags |= LLNO;
|
|
}
|
|
|
|
(se1->edges + i)->flags &= ~LLLOOP_SEARCH;
|
|
}
|
|
(se1->edges + i)->flags |= LLYES;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
|
|
static int part_of_loop(struct stack_elt *se)
|
|
/* Checks if 'se' belongs to a loop */
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < se->nr_nexts; i++) {
|
|
if ((se->edges + i)->flags & LLLOOP) return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
#endif /* NOLOOPS */
|
|
|
|
|
|
static void generate_heads(struct stacks *stack, struct stack_elt *se,
|
|
int l_ahead)
|
|
/* This funcion finds all heads starting at 'se'. */
|
|
{
|
|
int i;
|
|
struct stack_elt *next_se;
|
|
|
|
|
|
for (i = 0; i < se->nr_nexts; i++) {
|
|
|
|
if (!((se->edges + i)->ptr->flags & LLGEN_SEARCH)) {
|
|
|
|
(se->edges + i)->ptr->flags |= LLGEN_SEARCH;
|
|
|
|
next_se = (se->edges + i)->ptr;
|
|
|
|
/* Remember a flag has to be cleared later */
|
|
|
|
if (stack->visited_buf_size == 0) {
|
|
stack->visited_buf_size = LL_VIS_INCR;
|
|
stack->visited_buf = (struct stack_elt **)
|
|
Malloc(LL_VIS_INCR * sizeof(struct stack_elt *));
|
|
}
|
|
else if (stack->nr_visited == stack->visited_buf_size) {
|
|
stack->visited_buf_size += LL_VIS_INCR;
|
|
stack->visited_buf = (struct stack_elt **)
|
|
Realloc(stack->visited_buf,
|
|
stack->visited_buf_size *
|
|
sizeof(struct stack_elt *));
|
|
}
|
|
*(stack->visited_buf + stack->nr_visited) = next_se;
|
|
stack->nr_visited++;
|
|
|
|
if (next_se->flags & LLDUMMY) {
|
|
generate_heads(stack, next_se, l_ahead);
|
|
}
|
|
else if (next_se->nr == LLEOSTACK) {
|
|
/* We have matched a nt */
|
|
if (next_se->matched == LLstartsymb) {
|
|
stack->start_seen = 1;
|
|
}
|
|
|
|
continuation(stack, next_se->matched, l_ahead);
|
|
if (next_se->ref_count == 0) {
|
|
to_delete(stack, next_se);
|
|
}
|
|
}
|
|
else if (next_se->nr < LLFIRST_NT) {
|
|
/* terminal */
|
|
new_head(stack, next_se);
|
|
}
|
|
else {
|
|
if (next_se->ref_count > 0) {
|
|
next_se = split(next_se);
|
|
}
|
|
substitute(stack, next_se, l_ahead);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
static void delete(struct stacks *stack, struct stack_elt *se)
|
|
/* This function runs down the stack(s) deleting every element which cannot be
|
|
* reached anymore. */
|
|
{
|
|
int i;
|
|
|
|
#ifdef NOCLEAN
|
|
return;
|
|
#endif
|
|
|
|
if (se->ref_count == 0) {
|
|
|
|
/* Decrease reference counts of all successors */
|
|
for (i = 0; i < se->nr_nexts; i++) {
|
|
if ((se->edges + i)->ptr->ref_count != 0) {
|
|
(se->edges + i)->ptr->ref_count--;
|
|
|
|
/* Try to delete next element */
|
|
delete(stack, (se->edges + i)->ptr);
|
|
}
|
|
}
|
|
|
|
/* If this element is saved in the join_array clear it */
|
|
if (se->nr >= LLFIRST_NT) {
|
|
if (stack->join_array[se->nr - LLFIRST_NT] == se) {
|
|
stack->join_array[se->nr - LLFIRST_NT] =
|
|
(struct stack_elt *)0;
|
|
}
|
|
}
|
|
#ifdef DEBUG
|
|
deallocates++;
|
|
edge_deallocates += se->nr_nexts;
|
|
#endif
|
|
free((char *) se->edges);
|
|
free((char *) se);
|
|
}
|
|
|
|
#ifndef NOLOOPS
|
|
/* If this element belongs to a loop try to delete it */
|
|
else if (part_of_loop(se)) {
|
|
|
|
/* Do a temporary delete */
|
|
hyp_run(se);
|
|
|
|
/* Check it */
|
|
stack->check_run_ok = 1;
|
|
check_run(stack, se);
|
|
|
|
/* If it can be deleted delete it */
|
|
if (stack->check_run_ok) {
|
|
se->ref_count = 0;
|
|
delete(stack, se);
|
|
}
|
|
|
|
}
|
|
#endif
|
|
}
|
|
|
|
|
|
#ifndef NOLOOPS
|
|
|
|
static void hyp_run(struct stack_elt *se)
|
|
/* This function sets the 'hyp_ref_counts' of all elements of the loop that
|
|
* 'se' belongs to to the value that 'ref_count' will get when 'se' is
|
|
* deleted
|
|
*/
|
|
{
|
|
int i;
|
|
struct stack_elt *next_se;
|
|
|
|
for (i = 0; i < se->nr_nexts; i++) {
|
|
next_se = (se->edges + i)->ptr;
|
|
|
|
if ( (!((se->edges + i)->flags & LLHYP_SEARCH))
|
|
&& ((se->edges + i)->flags & LLLOOP)
|
|
) {
|
|
(se->edges + i)->flags |= LLHYP_SEARCH;
|
|
|
|
/* If this element is not yet visited initialize
|
|
* 'hyp_ref_count' else decrease it by one
|
|
*/
|
|
if (next_se->hyp_ref_count == -1) {
|
|
next_se->hyp_ref_count = next_se->ref_count - 1;
|
|
}
|
|
else {
|
|
next_se->hyp_ref_count--;
|
|
}
|
|
|
|
/* Continue searching */
|
|
hyp_run(next_se);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
static void check_run(struct stacks *stack, struct stack_elt *se)
|
|
/* This function checks all 'hyp_ref_counts' that 'hyp_run()' has set.
|
|
* If one of them is not 0, 'check_run_ok' will be set to 0 indicating
|
|
* that 'se' cannot be deleted. 'check_run()' also resets all 'hyp_ref_counts'
|
|
*/
|
|
{
|
|
int i;
|
|
|
|
if (se->hyp_ref_count > 0) {
|
|
stack->check_run_ok = 0;
|
|
}
|
|
|
|
/* Reset 'hyp_ref_count' */
|
|
se->hyp_ref_count = -1;
|
|
for (i = 0; i < se->nr_nexts; i++) {
|
|
if ((se->edges + i)->flags & LLHYP_SEARCH) {
|
|
(se->edges + i)->flags &= ~LLHYP_SEARCH;
|
|
check_run(stack, (se->edges + i)->ptr);
|
|
}
|
|
}
|
|
}
|
|
|
|
#endif /* NOLOOPS */
|
|
|
|
|
|
static struct stack_elt *split(struct stack_elt *se)
|
|
/* This function splits of a NT in de stack, and returns a pointer to it */
|
|
{
|
|
struct stack_elt *new_stack;
|
|
int i;
|
|
|
|
#ifdef DEBUG
|
|
allocates++;
|
|
if (allocates - deallocates > max_in_use) {
|
|
max_in_use = allocates - deallocates;
|
|
}
|
|
#endif
|
|
|
|
new_stack = (struct stack_elt *)Malloc(sizeof(struct stack_elt));
|
|
new_stack->flags = 0; /* Used by 'clear_gen_flags()' */
|
|
new_stack->nr = se->nr;
|
|
new_stack->ref_count = 0; /* Copy is new top */
|
|
new_stack->hyp_ref_count = -1;
|
|
new_stack->matched = -1;
|
|
new_stack->nr_nexts = se->nr_nexts;
|
|
|
|
#ifdef DEBUG
|
|
edge_allocates++;
|
|
if (edge_allocates - edge_deallocates > edge_max_in_use) {
|
|
edge_max_in_use = edge_allocates - edge_deallocates;
|
|
}
|
|
#endif
|
|
|
|
new_stack->edges = (struct edge *)
|
|
Malloc(se->nr_nexts * sizeof(struct edge));
|
|
|
|
/* Copy gets the same successors as the original */
|
|
memcpy((char *) new_stack->edges, (char *) se->edges,
|
|
se->nr_nexts * sizeof(struct edge));
|
|
|
|
/* Each successor gets a new predecessor */
|
|
for (i = 0; i < new_stack->nr_nexts; i++) {
|
|
(new_stack->edges + i)->ptr->ref_count++;
|
|
(new_stack->edges + i)->flags = 0;
|
|
}
|
|
|
|
return new_stack;
|
|
}
|
|
|
|
|
|
#ifdef DEBUG
|
|
static void test(struct stacks *stack)
|
|
{
|
|
struct stack_elt *se;
|
|
int i;
|
|
|
|
printf("STACKS:\n");
|
|
for (i = 0; i < stack->nr_heads; i++) {
|
|
printf("%2d: ", i + 1);
|
|
if (*(stack->heads_buf + i) == (struct stack_elt *)0) {
|
|
printf("NIL\n");
|
|
continue;
|
|
}
|
|
se = *(stack->heads_buf + i);
|
|
dump_stack(se, 1);
|
|
clear_flags(se, PRINT_SEARCH);
|
|
}
|
|
}
|
|
|
|
|
|
static void dump_stack(struct stack_elt *se, int level)
|
|
{
|
|
int i, j;
|
|
|
|
while (se->nr != LLEOSTACK) {
|
|
if ((se->flags & LLDUMMY) && (se->nr_nexts > 1)) {
|
|
printf("[%d] <%d,%d,%d>\n",
|
|
se->nr, se->ref_count,
|
|
se->hyp_ref_count,
|
|
se->flags
|
|
);
|
|
for (j = 0; j < se->nr_nexts; j++) {
|
|
for (i = 1; i <= level; i++) {
|
|
printf(" ");
|
|
}
|
|
printf("%d: ", j + 1);
|
|
if (!((se->edges + j)->flags & PRINT_SEARCH)) {
|
|
printf(" (%d) ", (se->edges + j)->flags);
|
|
(se->edges + j)->flags |= PRINT_SEARCH;
|
|
dump_stack((se->edges+j)->ptr,level+1);
|
|
/*clear_flags((se->edges+j)->ptr,PRINT_SEARCH);*/
|
|
}
|
|
else {
|
|
printf("LOOP\n");
|
|
}
|
|
}
|
|
return;
|
|
}
|
|
else {
|
|
if (se->flags & LLDUMMY) {
|
|
printf("[%d] <%d,%d,%d> ",
|
|
se->nr,se->ref_count,
|
|
se->hyp_ref_count,
|
|
se->flags
|
|
);
|
|
}
|
|
else {
|
|
printf("%d <%d,%d,%d> ",
|
|
se->nr, se->ref_count,
|
|
se->hyp_ref_count,
|
|
se->flags
|
|
);
|
|
}
|
|
if (!(se->edges->flags & PRINT_SEARCH)) {
|
|
printf(" (%d) ", se->edges->flags);
|
|
se->edges->flags |= PRINT_SEARCH;
|
|
se = se->edges->ptr;
|
|
}
|
|
else {
|
|
printf("LOOP\n");
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
printf("\n");
|
|
}
|
|
#endif
|
|
|
|
|
|
static void clear_flags(struct stack_elt *se, char flag)
|
|
/* Clears edge flag 'flag' */
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < se->nr_nexts; i++) {
|
|
if ((se->edges + i)->flags & flag) {
|
|
(se->edges + i)->flags &= ~flag; /* clear flag */
|
|
clear_flags((se->edges + i)->ptr, flag);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void clear_gen_flags(struct stacks *stack)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < stack->nr_visited; i++) {
|
|
(*(stack->visited_buf + i))->flags &= ~(LLGEN_SEARCH);
|
|
}
|
|
|
|
stack->nr_visited = 0;
|
|
}
|
|
|
|
|
|
static void match_heads(struct stacks *stack, int symb)
|
|
/* Match heads_buf against symb, leaving only matching heads,
|
|
* whilst deallocating the non-matching stacks
|
|
*/
|
|
{
|
|
int i;
|
|
|
|
int old_nr_heads;
|
|
struct stack_elt **old_heads_buf;
|
|
|
|
|
|
/* Copy the 'old' heads */
|
|
old_nr_heads = stack->nr_heads;
|
|
old_heads_buf = stack->heads_buf;
|
|
|
|
|
|
/* Set heads in stack to 0 */
|
|
stack->nr_heads = 0;
|
|
stack->heads_buf_size = 0;
|
|
stack->heads_buf = (struct stack_elt **) 0;
|
|
|
|
|
|
for (i = 0; i < old_nr_heads; i++) {
|
|
if ((*(old_heads_buf + i))->nr != symb) {
|
|
/* Does not match? */
|
|
(*(old_heads_buf + i))->ref_count--;
|
|
(*(old_heads_buf + i))->flags &= ~LLHEAD;
|
|
delete(stack, *(old_heads_buf + i));
|
|
}
|
|
else { /* Matches */
|
|
if (stack->heads_buf_size == 0) {
|
|
stack->heads_buf_size = LLHEADS_BUF_INCR;
|
|
stack->heads_buf = (struct stack_elt **)
|
|
Malloc(stack->heads_buf_size *
|
|
sizeof(struct stack_elt *));
|
|
}
|
|
else if (stack->nr_heads == stack->heads_buf_size) {
|
|
stack->heads_buf_size += LLHEADS_BUF_INCR;
|
|
stack->heads_buf = (struct stack_elt **)
|
|
Realloc(stack->heads_buf,
|
|
stack->heads_buf_size *
|
|
sizeof(struct stack_elt *));
|
|
}
|
|
*(stack->heads_buf + stack->nr_heads) =
|
|
*(old_heads_buf + i);
|
|
stack->nr_heads++;
|
|
}
|
|
}
|
|
free((char *) old_heads_buf);
|
|
}
|
|
|
|
|
|
static void cleanup(struct stacks *stack)
|
|
/* Deletes all elements in 'cleanup_buf()' */
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < stack->nr_cleanups; i++) {
|
|
delete(stack, *(stack->cleanup_buf + i));
|
|
}
|
|
|
|
stack->nr_cleanups = 0;
|
|
|
|
}
|
|
|
|
|
|
static void initialize(struct stacks *stack)
|
|
/* Initializes some variables and arrays */
|
|
{
|
|
int j;
|
|
|
|
stack->nr_heads = 0;
|
|
stack->heads_buf_size = 0;
|
|
stack->heads_buf = (struct stack_elt **)0;
|
|
|
|
stack->nr_cleanups = 0;
|
|
stack->cleanup_buf_size = 0;
|
|
stack->cleanup_buf = (struct stack_elt **)0;
|
|
|
|
stack->nr_visited = 0;
|
|
stack->visited_buf_size = 0;
|
|
stack->visited_buf = (struct stack_elt **)0;
|
|
|
|
for (j = 0; j < (LLNNONTERMINALS + 7)/8; j++) {
|
|
stack->r_rec[j] = (char) 0;
|
|
}
|
|
|
|
for (j = 0; j < LLNNONTERMINALS; j++) {
|
|
stack->join_array[j] = (struct stack_elt *)0;
|
|
}
|
|
|
|
for (j = 0; j < LLSETSIZE; j++) {
|
|
stack->exp_terminals[j] = 0;
|
|
}
|
|
|
|
stack->start_seen = 0;
|
|
}
|
|
|
|
|
|
static void calculate(struct stacks *stack, int l_ahead)
|
|
/* This function finds all new heads and deletes the old heads */
|
|
{
|
|
int i;
|
|
int old_nr_heads;
|
|
struct stack_elt **old_heads_buf;
|
|
|
|
/* Make a copy of the heads */
|
|
old_nr_heads = stack->nr_heads;
|
|
old_heads_buf = stack->heads_buf;
|
|
|
|
stack->nr_heads = 0;
|
|
stack->heads_buf = (struct stack_elt **) 0;
|
|
stack->heads_buf_size = 0;
|
|
|
|
for (i = 0; i < old_nr_heads; i++) {
|
|
/* Find all new heads */
|
|
generate_heads(stack, *(old_heads_buf + i), l_ahead);
|
|
clear_gen_flags(stack);
|
|
|
|
/* Old head can be deleted now */
|
|
(*(old_heads_buf + i))->ref_count--;
|
|
delete(stack, *(old_heads_buf + i));
|
|
}
|
|
|
|
|
|
cleanup(stack);
|
|
free((char *) old_heads_buf);
|
|
|
|
/* Reinitialize some things */
|
|
for (i = 0; i < (LLNNONTERMINALS + 7)/8; i++) {
|
|
stack->r_rec[i] = (char) 0;
|
|
}
|
|
|
|
for (i = 0; i < LLNNONTERMINALS; i++) {
|
|
stack->join_array[i] = (struct stack_elt *)0;
|
|
}
|
|
|
|
/* Delete all HEAD flags */
|
|
for (i = 0; i < stack->nr_heads; i++) {
|
|
(*(stack->heads_buf + i))->flags &= ~LLHEAD;
|
|
}
|
|
}
|
|
|
|
static void kill_stack(struct stacks *stack)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < stack->nr_heads; i++) {
|
|
(*(stack->heads_buf + i))->ref_count--;
|
|
delete(stack, *(stack->heads_buf + i));
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void LLnc_recover(void)
|
|
/* This function contains the main loop for non correcting syntax error
|
|
* recovery
|
|
*/
|
|
{
|
|
int j;
|
|
int base_symb;
|
|
struct stacks stack;
|
|
int max_nr_heads;
|
|
int max_nr_good_heads;
|
|
|
|
initialize(&stack);
|
|
max_nr_heads = 0;
|
|
max_nr_good_heads = 0;
|
|
|
|
/* Grammar has to be read only once */
|
|
if (!grammar_read) {
|
|
build_grammar();
|
|
grammar_read = 1;
|
|
}
|
|
|
|
/* Read first token */
|
|
read_token();
|
|
base_symb = LLsymb;
|
|
|
|
/* Check on end of file */
|
|
if ((base_symb <= 0) || (base_symb == EOFILE)) {
|
|
|
|
if ((nonterminals + LLstartsymb - LLFIRST_NT)->rule->empty != 1
|
|
) {
|
|
LLsymb = EOFILE;
|
|
LLmessage(0);
|
|
}
|
|
|
|
kill_stack(&stack);
|
|
return;
|
|
}
|
|
|
|
/* Read look ahead token */
|
|
read_token();
|
|
|
|
/* Now search applicable rules and starts the ball rolling */
|
|
start_stack(&stack, base_symb, LLsymb);
|
|
|
|
if (stack.nr_heads > max_nr_heads) {
|
|
max_nr_heads = stack.nr_heads;
|
|
}
|
|
|
|
|
|
/* Only matching heads are needed */
|
|
match_heads(&stack, LLsymb);
|
|
|
|
if (stack.nr_heads > max_nr_good_heads) {
|
|
max_nr_good_heads = stack.nr_heads;
|
|
}
|
|
|
|
|
|
#ifdef DEBUG
|
|
test(&stack);
|
|
#endif
|
|
|
|
/* Loop untill end of inputfile */
|
|
while ((LLsymb > 0) && (LLsymb != EOFILE)) {
|
|
/* When entering the loop LLsymb always contains the
|
|
* symbol that was used as look_ahead to construct the stacks,
|
|
* or, if optimization is OFF, it contains the symbol with
|
|
* which the current heads have been matched
|
|
*/
|
|
|
|
if (stack.nr_heads == 0) {
|
|
/* No more heads left */
|
|
LLmessage(0);
|
|
|
|
/* Restart the whole thing */
|
|
initialize(&stack);
|
|
|
|
/* The look-ahead caused the empty stack, don't
|
|
* use it to start a new one !
|
|
*/
|
|
|
|
read_token();
|
|
base_symb = LLsymb;
|
|
|
|
/* Check on end of file */
|
|
if ((base_symb <= 0) || (base_symb == EOFILE)) {
|
|
if ((nonterminals + LLstartsymb - LLFIRST_NT)->rule->empty != 1) {
|
|
LLsymb = EOFILE;
|
|
LLmessage(0);
|
|
}
|
|
kill_stack(&stack);
|
|
return;
|
|
}
|
|
|
|
read_token();
|
|
|
|
start_stack(&stack, base_symb, LLsymb);
|
|
|
|
if (stack.nr_heads > max_nr_heads) {
|
|
max_nr_heads = stack.nr_heads;
|
|
}
|
|
|
|
|
|
match_heads(&stack, LLsymb);
|
|
|
|
if (stack.nr_heads > max_nr_good_heads) {
|
|
max_nr_good_heads = stack.nr_heads;
|
|
}
|
|
|
|
continue;
|
|
}
|
|
|
|
|
|
/* Normal case starts here */
|
|
stack.start_seen = 0;
|
|
|
|
for (j = 0; j < LLSETSIZE; j++) {
|
|
stack.exp_terminals[j] = 0;
|
|
}
|
|
|
|
/* Read next symbol */
|
|
read_token();
|
|
|
|
/* Generate all new heads and delete old ones */
|
|
calculate(&stack, LLsymb);
|
|
|
|
/* Leave out not wanted heads */
|
|
|
|
if (stack.nr_heads > max_nr_heads) {
|
|
max_nr_heads = stack.nr_heads;
|
|
}
|
|
|
|
match_heads(&stack, LLsymb);
|
|
|
|
if (stack.nr_heads > max_nr_good_heads) {
|
|
max_nr_good_heads = stack.nr_heads;
|
|
}
|
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
test(&stack);
|
|
#endif
|
|
|
|
}
|
|
|
|
/* End of file reached, check if we have seen a start symbol */
|
|
if (stack.start_seen == 1) return;
|
|
else {
|
|
LLsymb = EOFILE;
|
|
LLmessage(0);
|
|
}
|
|
|
|
kill_stack(&stack);
|
|
|
|
#ifdef DEBUG
|
|
printf("Maximum number of heads: %d\n", max_nr_heads);
|
|
printf("Maximum number of good heads: %d\n", max_nr_good_heads);
|
|
printf("Number of node allocates: %d\n", allocates);
|
|
printf("Number of node deallocates: %d\n", deallocates);
|
|
printf("Maximum number of nodes in use: %8d\n", max_in_use);
|
|
printf("Sizeof(struct stack_elt) = %8d\n", sizeof(struct stack_elt));
|
|
printf(" --------x\n");
|
|
printf(" %8d\n", max_in_use * sizeof(
|
|
struct stack_elt));
|
|
printf("Number of edge allocates: %d\n", edge_allocates);
|
|
printf("Number of edge deallocates: %d\n", edge_deallocates);
|
|
printf("Maximum number of edges in use: %8d\n", edge_max_in_use);
|
|
printf("Sizeof(struct edge) = %8d\n", sizeof(struct edge));
|
|
printf(" --------x\n");
|
|
printf(" %8d\n", edge_max_in_use * sizeof(struct edge));
|
|
|
|
#endif
|
|
}
|
|
|