mirror of https://github.com/acidanthera/audk.git
3031 lines
81 KiB
C
3031 lines
81 KiB
C
/*
|
|
* mrhoist.c
|
|
*
|
|
* SOFTWARE RIGHTS
|
|
*
|
|
* We reserve no LEGAL rights to the Purdue Compiler Construction Tool
|
|
* Set (PCCTS) -- PCCTS is in the public domain. An individual or
|
|
* company may do whatever they wish with source code distributed with
|
|
* PCCTS or the code generated by PCCTS, including the incorporation of
|
|
* PCCTS, or its output, into commerical software.
|
|
*
|
|
* We encourage users to develop software with PCCTS. However, we do ask
|
|
* that credit is given to us for developing PCCTS. By "credit",
|
|
* we mean that if you incorporate our source code into one of your
|
|
* programs (commercial product, research project, or otherwise) that you
|
|
* acknowledge this fact somewhere in the documentation, research report,
|
|
* etc... If you like PCCTS and have developed a nice tool with the
|
|
* output, please mention that you developed it using PCCTS. In
|
|
* addition, we ask that this header remain intact in our source code.
|
|
* As long as these guidelines are kept, we expect to continue enhancing
|
|
* this system and expect to make other tools available as they are
|
|
* completed.
|
|
*
|
|
* ANTLR 1.33MR10
|
|
*
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
|
|
#include "pcctscfg.h"
|
|
|
|
#include "set.h"
|
|
#include "syn.h"
|
|
#include "hash.h"
|
|
#include "generic.h"
|
|
#include "dlgdef.h"
|
|
#include <ctype.h>
|
|
|
|
#ifdef __USE_PROTOS
|
|
void dumppred(Predicate *);
|
|
#else
|
|
void dumppred();
|
|
#endif
|
|
|
|
/*
|
|
Try to determine whether predicate "first" is true for
|
|
all cases where "second" is true. Comparison takes place
|
|
without regard to context.
|
|
Assumes that predicate symbols have been expanded.
|
|
Assumes that there are no NAND or NOR nodes
|
|
|
|
*/
|
|
|
|
#ifdef __USE_PROTOS
|
|
int MR_secondPredicateUnreachable(Predicate *first,Predicate *second)
|
|
#else
|
|
int MR_secondPredicateUnreachable(first,second)
|
|
Predicate *first;
|
|
Predicate *second;
|
|
#endif
|
|
{
|
|
Predicate *f;
|
|
Predicate *s;
|
|
|
|
if (first == NULL) {
|
|
return 1;
|
|
} else if (second == NULL) {
|
|
return 0;
|
|
} else if (first->down == NULL && second->down == NULL) {
|
|
if (first->source == second->source &&
|
|
first->inverted == second->inverted) {
|
|
return 1; /* look identical - will never reach alt2 */
|
|
} else {
|
|
return 0; /* look different */
|
|
};
|
|
} else if (first->down == NULL && second->down != NULL) {
|
|
|
|
if (second->expr == PRED_AND_LIST) {
|
|
|
|
/* unreachable if first covers any child of second */
|
|
|
|
for (s=second->down; s != NULL; s=s->right) {
|
|
if (MR_secondPredicateUnreachable(first,s)) {
|
|
return 1;
|
|
};
|
|
};
|
|
return 0;
|
|
} else if (second->expr == PRED_OR_LIST) {
|
|
|
|
/* unreachable if first covers every child of second */
|
|
|
|
for (s=second->down; s != NULL; s=s->right) {
|
|
if (!MR_secondPredicateUnreachable(first,s)) {
|
|
return 0;
|
|
};
|
|
};
|
|
return 1;
|
|
} else {
|
|
require (0,"Illegal pred->expr");
|
|
return 0; /* MR20 Make compiler happy */
|
|
};
|
|
} else if (first->down != NULL && second->down == NULL) {
|
|
if (first->expr == PRED_AND_LIST) {
|
|
|
|
/* unreachable if every child of first covers second */
|
|
|
|
for (f=first->down; f != NULL; f=f->right) {
|
|
if (!MR_secondPredicateUnreachable(f,second)) {
|
|
return 0;
|
|
};
|
|
};
|
|
return 1;
|
|
} else if (first->expr == PRED_OR_LIST) {
|
|
|
|
/* unreachable if any child of first covers second */
|
|
|
|
for (f=first->down; f != NULL; f=f->right) {
|
|
if (MR_secondPredicateUnreachable(f,second)) {
|
|
return 1;
|
|
};
|
|
};
|
|
return 0;
|
|
} else {
|
|
require (0,"Illegal predicate->expr");
|
|
return 0; /* MR20 Make compiler happy */
|
|
};
|
|
} else {
|
|
|
|
if (first->expr == PRED_AND_LIST && second->expr == PRED_AND_LIST) {
|
|
|
|
/* unreachable if each child of first covers at least one child of second */
|
|
|
|
for (f=first->down; f != NULL ; f=f->right) {
|
|
for (s=second->down; s != NULL ; s=s->right) {
|
|
if (MR_secondPredicateUnreachable(f,s)) goto A_next_f;
|
|
};
|
|
return 0;
|
|
A_next_f:
|
|
continue;
|
|
};
|
|
return 1;
|
|
|
|
} else if (first->expr == PRED_AND_LIST && second->expr == PRED_OR_LIST) {
|
|
|
|
/* unreachable if each child of first covers ALL of second's children */
|
|
|
|
for (f=first->down; f != NULL ; f=f->right) {
|
|
for (s=second->down; s != NULL ; s=s->right) {
|
|
if (!MR_secondPredicateUnreachable(f,s)) return 0;
|
|
};
|
|
};
|
|
return 1;
|
|
|
|
} else if (first->expr == PRED_OR_LIST && second->expr == PRED_AND_LIST) {
|
|
|
|
/* unreachable if any child of second is covered by any child of first */
|
|
|
|
for (f=first->down; f != NULL ; f=f->right) {
|
|
for (s=second->down; s != NULL ; s=s->right) {
|
|
if (MR_secondPredicateUnreachable(f,s)) return 1;
|
|
};
|
|
};
|
|
return 0;
|
|
|
|
} else if (first->expr == PRED_OR_LIST && second->expr == PRED_OR_LIST) {
|
|
|
|
/* unreachable if every child of second is covered by some child of first */
|
|
|
|
for (f=first->down; f != NULL ; f=f->right) {
|
|
for (s=second->down; s != NULL ; s=s->right) {
|
|
if (MR_secondPredicateUnreachable(f,s)) goto B_next_f;
|
|
};
|
|
return 0;
|
|
B_next_f:
|
|
continue;
|
|
};
|
|
return 1;
|
|
|
|
} else {
|
|
require (0,"Illegal predicate->expr");
|
|
return 0; /* MR20 Make compiler happy */
|
|
};
|
|
};
|
|
return 0; /* MR20 MSVC 5.0 complains about missing return statement */
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_xxxIndent(FILE *f,int depth)
|
|
#else
|
|
void MR_xxxIndent(f,depth)
|
|
FILE *f;
|
|
int depth;
|
|
#endif
|
|
{
|
|
int i;
|
|
|
|
for (i=0; i<depth ; i++) {
|
|
fprintf(f," ");
|
|
};
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_stderrIndent(int depth)
|
|
#else
|
|
void MR_stderrIndent(depth)
|
|
int depth;
|
|
#endif
|
|
{
|
|
MR_xxxIndent(stderr,depth);
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_outputIndent(int depth)
|
|
#else
|
|
void MR_outputIndent(depth)
|
|
int depth;
|
|
#endif
|
|
{
|
|
MR_xxxIndent(output,depth);
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_set_reuse(set *s)
|
|
#else
|
|
void MR_set_reuse(s)
|
|
set *s;
|
|
#endif
|
|
{
|
|
set_free(*s);
|
|
*s=empty;
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_dumpPredRuleRefStack(FILE *iounit,int indent)
|
|
#else
|
|
void MR_dumpPredRuleRefStack(iounit,indent)
|
|
FILE *iounit;
|
|
int indent;
|
|
#endif
|
|
{
|
|
int i;
|
|
int j;
|
|
int count=MR_PredRuleRefStack.count;
|
|
RuleRefNode *rrn=NULL;
|
|
Junction *lastOne;
|
|
|
|
if (count == 0) {
|
|
fprintf(iounit,"empty\n");
|
|
return;
|
|
};
|
|
for (i=0; i < count; i++) {
|
|
rrn=(RuleRefNode *) MR_PredRuleRefStack.data[i];
|
|
for (j=0; j<indent; j++) fprintf(iounit," ");
|
|
fprintf(iounit,"#%-2d in rule %s (line %d %s) to rule %s\n",
|
|
i,rrn->rname,rrn->line,FileStr[rrn->file],rrn->text);
|
|
};
|
|
lastOne=MR_ruleReferenced(rrn);
|
|
if (lastOne != NULL) {
|
|
for (j=0; j<indent; j++) fprintf(iounit," ");
|
|
fprintf(iounit,"#%-2d in rule %s (line %d %s)\n",
|
|
count,lastOne->rname,lastOne->line,FileStr[lastOne->file]);
|
|
};
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_dumpTreeF(FILE *f,int depth,Tree *tree,int across)
|
|
#else
|
|
void MR_dumpTreeF(f,depth,tree,across)
|
|
FILE *f;
|
|
Tree *tree;
|
|
int depth;
|
|
int across;
|
|
#endif
|
|
{
|
|
int newAcross=across;
|
|
|
|
if (tree == NULL ) return;
|
|
if (tree->down != NULL ) {
|
|
fprintf(output,"\n");
|
|
MR_outputIndent(depth);
|
|
fprintf(output, "(root =");
|
|
};
|
|
if (tree->token == ALT ) {
|
|
fprintf(output," %-16s","Alt");
|
|
} else if (tree->token==EpToken ) {
|
|
fprintf(output,"(%d)%13s",tree->v.rk," ");
|
|
} else {
|
|
fprintf(output," %-16s",TerminalString(tree->token));
|
|
};
|
|
if (tree->down != NULL) {
|
|
fprintf(output,"\n");
|
|
MR_outputIndent(depth+1);
|
|
MR_dumpTreeF(f,depth+1,tree->down,1);
|
|
newAcross=0;
|
|
fprintf(output,"\n");
|
|
MR_outputIndent(depth);
|
|
fprintf(output,")");
|
|
};
|
|
if (newAcross > 3) {
|
|
fprintf(output,"\n");
|
|
MR_outputIndent(depth);
|
|
newAcross=0;
|
|
};
|
|
MR_dumpTreeF(f,depth,tree->right,newAcross+1);
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_dumpTreeX(int depth,Tree *tree,int across)
|
|
#else
|
|
void MR_dumpTreeX(depth,tree,across)
|
|
Tree *tree;
|
|
int depth;
|
|
int across;
|
|
#endif
|
|
{
|
|
MR_dumpTreeF(output,depth,tree,across);
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_dumpTokenSet(FILE *f,int depth,set s)
|
|
#else
|
|
void MR_dumpTokenSet(f,depth,s)
|
|
FILE *f;
|
|
int depth;
|
|
set s;
|
|
#endif
|
|
{
|
|
int i;
|
|
int j;
|
|
|
|
unsigned *pdq;
|
|
|
|
if (set_nil(s)) {
|
|
fprintf(f,"\n");
|
|
MR_xxxIndent(f,depth+1);
|
|
fprintf(f,"nil\n");
|
|
return;
|
|
};
|
|
|
|
pdq=set_pdq(s);
|
|
require(pdq != NULL,"set_pdq failed");
|
|
i=0;
|
|
for (i=0 ; ; i=i+4) {
|
|
fprintf(f,"\n");
|
|
MR_xxxIndent(f,depth+1);
|
|
for (j=0; j < 4 ; j++) {
|
|
if (pdq[i+j] == nil) break;
|
|
fprintf(f," %-16s",TerminalString(pdq[i+j]));
|
|
};
|
|
if (pdq[i+j] == nil) break;
|
|
};
|
|
fprintf(f,"\n");
|
|
free( (char *) pdq);
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_dumpPred1(int depth,Predicate *p,int withContext)
|
|
#else
|
|
void MR_dumpPred1(depth,p,withContext)
|
|
int depth;
|
|
Predicate *p;
|
|
int withContext;
|
|
#endif
|
|
{
|
|
unsigned k;
|
|
|
|
if (p == NULL) {
|
|
MR_outputIndent(depth);
|
|
fprintf(output,"The predicate is empty (or always true)\n\n");
|
|
return;
|
|
};
|
|
if (p->down != NULL) {
|
|
MR_outputIndent(depth);
|
|
if (p->inverted) {
|
|
|
|
/* MR14a Left out print expression in fprintf
|
|
Reported by Manuel Kessler (mlkessle@cip.physik.uni-wuerzburg.de)
|
|
*/
|
|
|
|
if (p->expr == PRED_AND_LIST) fprintf(output,"%s NAND (not AND) expr\n\n",p->expr);
|
|
if (p->expr == PRED_OR_LIST) fprintf(output,"%s NOR (not OR) expr\n\n",p->expr);
|
|
} else {
|
|
fprintf(output,"%s expr\n\n",p->expr);
|
|
};
|
|
} else {
|
|
MR_outputIndent(depth);
|
|
fprintf(output,"pred %s <<%s>>?\n",
|
|
(p->inverted ? " *not*" : ""),
|
|
(p->expr == NULL ? "null expr" : p->expr));
|
|
MR_outputIndent(depth+1);
|
|
fprintf(output," ");
|
|
fprintf(output," depth=k=%d",p->k);
|
|
if (p->source != NULL && p->source->guardpred) {
|
|
fprintf(output," (\"=>\" guard)");
|
|
}
|
|
if (p->source != NULL && p->source->ampersandPred != NULL) {
|
|
fprintf(output," (\"&&\" guard)");
|
|
};
|
|
k=set_int(p->completionSet);
|
|
if (k != nil) {
|
|
fprintf(output," Incomplete Set at k=%d !",k);
|
|
};
|
|
k=set_int(p->completionTree);
|
|
if (k != nil) {
|
|
fprintf(output," Incomplete Tree at k=%d !",k);
|
|
};
|
|
if (p->source != NULL) {
|
|
fprintf(output," rule %s line %d %s",
|
|
p->source->rname,p->source->line,FileStr[p->source->file]);
|
|
};
|
|
fprintf(output,"\n");
|
|
if (withContext &&
|
|
(HoistPredicateContext ||
|
|
! set_nil(p->scontext[1]) ||
|
|
p->tcontext != NULL)) {
|
|
if (p->k == 1) {
|
|
MR_outputIndent(depth+1);
|
|
fprintf(output,"set context: ");
|
|
MR_dumpTokenSet(output,depth+1,p->scontext[1]);
|
|
}
|
|
if (p->k != 1) {
|
|
MR_outputIndent(depth+1);
|
|
fprintf(output,"tree context:");
|
|
if (p->tcontext == NULL) {
|
|
fprintf(output," null");
|
|
} else {
|
|
MR_dumpTreeX(depth+2,p->tcontext,0);
|
|
};
|
|
fprintf(output,"\n");
|
|
};
|
|
};
|
|
fprintf(output,"\n");
|
|
};
|
|
if (p->down != NULL) {
|
|
MR_dumpPred1(depth+1,p->down,withContext);
|
|
};
|
|
if (p->right != NULL) {
|
|
MR_dumpPred1(depth,p->right,withContext);
|
|
};
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_dumpPred(Predicate *p,int withContext)
|
|
#else
|
|
void MR_dumpPred(p,withContext)
|
|
Predicate *p;
|
|
int withContext;
|
|
#endif
|
|
{
|
|
MR_dumpPred1(0,p,withContext);
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
Tree * MR_make_tree_from_set(set s)
|
|
#else
|
|
Tree * MR_make_tree_from_set(s)
|
|
set s;
|
|
#endif
|
|
{
|
|
Tree *t=NULL;
|
|
Tree *node;
|
|
Tree **tp=&t;
|
|
int i;
|
|
|
|
unsigned *pdq=set_pdq(s);
|
|
|
|
if (pdq != NULL) {
|
|
for (i=0 ; pdq[i] != nil ; i++) {
|
|
node=tnode( (int) pdq[i]);
|
|
*tp=node;
|
|
tp=&(node->right);
|
|
};
|
|
*tp=NULL;
|
|
free ( (char *) pdq);
|
|
};
|
|
return t;
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_check_pred_too_long(Predicate *p,set completion)
|
|
#else
|
|
void MR_check_pred_too_long(p,completion)
|
|
Predicate *p;
|
|
set completion;
|
|
#endif
|
|
{
|
|
if (p != NULL &&
|
|
p->source != NULL &&
|
|
! p->source->predTooLong) {
|
|
if ( !set_nil(completion)) {
|
|
p->source->predTooLong=1;
|
|
warnFL("It is unusual (but ok) for a semantic predicate to test context past the end of its own rule",
|
|
FileStr[p->source->file],p->source->line);
|
|
};
|
|
};
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
int MR_predicate_context_completed(Predicate *p)
|
|
#else
|
|
int MR_predicate_context_completed(p)
|
|
Predicate *p;
|
|
#endif
|
|
{
|
|
if (p == NULL) return 1;
|
|
if (p->expr != PRED_AND_LIST &&
|
|
p->expr != PRED_OR_LIST) {
|
|
if ( ! set_nil(p->completionSet)) return 0;
|
|
if ( ! set_nil(p->completionTree)) return 0;
|
|
};
|
|
return MR_predicate_context_completed(p->down) &
|
|
MR_predicate_context_completed(p->right);
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
Node * MR_advance(Node *n)
|
|
#else
|
|
Node * MR_advance(n)
|
|
Node *n;
|
|
#endif
|
|
{
|
|
if (n == NULL) return NULL;
|
|
switch (n->ntype) {
|
|
case nJunction: return ((Junction *)n)->p1;
|
|
case nToken: return ((TokNode *)n)->next;
|
|
case nRuleRef: return ((RuleRefNode *)n)->next;
|
|
case nAction: return ((ActionNode *)n)->next;
|
|
default: return NULL;
|
|
};
|
|
return NULL; /* MSVC 5.0 complains about missing return statement */
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
Junction * MR_find_endRule(Node *n)
|
|
#else
|
|
Junction * MR_find_endRule(n)
|
|
Node *n;
|
|
#endif
|
|
{
|
|
Node *next;
|
|
if (n == NULL) return NULL;
|
|
for (next=n; next != NULL; next=MR_advance(next)) {
|
|
if (next->ntype == nJunction &&
|
|
( (Junction *) next)->jtype == EndRule) {
|
|
break;
|
|
};
|
|
};
|
|
return (Junction *)next;
|
|
}
|
|
|
|
/*
|
|
Intersection: a branch which is shorter is chosen
|
|
over one which is longer: (A B C) intersect (A B) yields (A B).
|
|
|
|
AND: a branch which is longer is chosen over the one
|
|
which is shorter: (A B C) AND (A B) yields (A B C)
|
|
|
|
*/
|
|
|
|
#ifdef __USE_PROTOS
|
|
Tree *MR_computeTreeIntersection(Tree *l,Tree *r)
|
|
#else
|
|
Tree *MR_computeTreeIntersection(l,r)
|
|
Tree *l;
|
|
Tree *r;
|
|
#endif
|
|
{
|
|
Tree *result=NULL;
|
|
Tree **tail;
|
|
Tree *p;
|
|
Tree *q;
|
|
Tree *match;
|
|
|
|
if (l == NULL || r == NULL) return NULL;
|
|
for (p=l; p != NULL; p=p->right) {
|
|
require(p->token != EpToken,"MR_computeTreeIntersection: p->EpToken unexpected\n");
|
|
require (p->token != ALT,"MR_computeTreeIntersection: p->ALT unexpected\n");
|
|
};
|
|
for (q=r; q != NULL; q=q->right) {
|
|
require(q->token != EpToken,"MR_computeTreeIntersection: q->EpToken unexpected\n");
|
|
require(q->token != ALT,"MR_computeTreeIntersection: q->ALT unexpected\n");
|
|
};
|
|
|
|
result=tnode(ALT);
|
|
tail=&(result->down);
|
|
|
|
for (p=l; p != NULL ; p=p->right) {
|
|
for (q=r; q != NULL ; q=q->right) {
|
|
if (p->token == q->token) {
|
|
match=tnode(p->token);
|
|
match->down=MR_computeTreeIntersection(p->down,q->down);
|
|
*tail=match;
|
|
tail=&(match->right);
|
|
};
|
|
};
|
|
};
|
|
|
|
*tail=NULL;
|
|
result=tshrink(result);
|
|
result=tflatten( result );
|
|
result=tleft_factor( result );
|
|
return result;
|
|
}
|
|
|
|
/* the predicates which are ANDed together have a common
|
|
context: they must all have common roots. Thus the
|
|
AND operation is more like an OR operation because
|
|
branches which are longer are grafted onto shorter
|
|
branches of the AND tree. For instance combining
|
|
(A B C) with (A B C D) gives (A B C D). There
|
|
should never be a case of (A B C) and (A B D) because
|
|
they have the same context.
|
|
|
|
Actually, this may not be true once one throws in
|
|
guard predicates which are defined by the user, not
|
|
the context.
|
|
*/
|
|
|
|
/* requires input trees to be in "canonical" format */
|
|
|
|
#ifdef __USE_PROTOS
|
|
Tree *MR_computeTreeAND(Tree *l,Tree *r)
|
|
#else
|
|
Tree *MR_computeTreeAND(l,r)
|
|
Tree *l;
|
|
Tree *r;
|
|
#endif
|
|
{
|
|
Tree *result=NULL;
|
|
Tree **tail;
|
|
Tree *p;
|
|
Tree *q;
|
|
Tree *match;
|
|
|
|
if (l == NULL) return tdup(r);
|
|
if (r == NULL) return tdup(l);
|
|
|
|
for (p=l; p != NULL; p=p->right) {
|
|
/**** require(p->token != EpToken,"MR_computeTreeAND: p->EpToken unexpected\n"); ****/
|
|
require (p->token != ALT,"MR_computeTreeAND: p->ALT unexpected\n");
|
|
};
|
|
for (q=r; q != NULL; q=q->right) {
|
|
/**** require(q->token != EpToken,"MR_computeTreeAND: q->EpToken unexpected\n"); ****/
|
|
require(q->token != ALT,"MR_computeTreeAND: q->ALT unexpected\n");
|
|
};
|
|
|
|
result=tnode(ALT);
|
|
tail=&(result->down);
|
|
|
|
for (p=l; p != NULL ; p=p->right) {
|
|
for (q=r; q != NULL ; q=q->right) {
|
|
if (p->token == q->token) {
|
|
match=tnode(p->token);
|
|
match->down=MR_computeTreeAND(p->down,q->down);
|
|
*tail=match;
|
|
tail=&(match->right);
|
|
};
|
|
};
|
|
};
|
|
|
|
*tail=NULL;
|
|
result=tshrink(result);
|
|
result=tflatten( result );
|
|
result=tleft_factor( result );
|
|
return result;
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_union_plain_sets1(Predicate *p,set *theUnion)
|
|
#else
|
|
void MR_union_plain_sets1(p,theUnion)
|
|
Predicate *p;
|
|
set *theUnion;
|
|
#endif
|
|
{
|
|
if (p == NULL) return;
|
|
MR_union_plain_sets1(p->down,theUnion);
|
|
MR_union_plain_sets1(p->right,theUnion);
|
|
set_orin(theUnion,p->plainSet);
|
|
return;
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
set MR_union_plain_sets(Predicate *p)
|
|
#else
|
|
set MR_union_plain_sets(p)
|
|
Predicate *p;
|
|
#endif
|
|
{
|
|
set theUnion;
|
|
|
|
theUnion=empty;
|
|
|
|
MR_union_plain_sets1(p,&theUnion);
|
|
return theUnion;
|
|
}
|
|
|
|
/* does NOT left factor: do not want to merge
|
|
(A B) with (A) to get (A B)
|
|
in fact the opposite: (A B) with (A) gives (A)
|
|
*/
|
|
|
|
#ifdef __USE_PROTOS
|
|
Tree *MR_compute_pred_tree_ctxXX(Predicate *p)
|
|
#else
|
|
Tree *MR_compute_pred_tree_ctxXX(p)
|
|
Predicate *p;
|
|
#endif
|
|
{
|
|
Tree *result=NULL;
|
|
Predicate *q;
|
|
Tree *t;
|
|
|
|
if (p == NULL) return NULL;
|
|
|
|
/* this appears strange: why do we OR the context
|
|
of an AND predicate ? It is because of the way
|
|
that predicates are evaluated: if the context is
|
|
wrong then it's the same as if the predicate was
|
|
true. That means that even when one leg of an
|
|
AND has unmatched context, if the other leg has
|
|
matched context and is true then the predicate
|
|
succeeds. It's only when all the legs have unmatched
|
|
context that this one can skip evaluation of the
|
|
predicates.
|
|
*/
|
|
if (p->expr == PRED_OR_LIST ||
|
|
p->expr == PRED_AND_LIST) {
|
|
for (q=p->down; q != NULL ; q=q->right) {
|
|
t=MR_compute_pred_tree_ctxXX(q);
|
|
result=tappend(result,t);
|
|
t=NULL;
|
|
};
|
|
|
|
result=tshrink(result);
|
|
result=tflatten( result );
|
|
|
|
/* does NOT left factor: do not want to merge
|
|
(A B) with (A) to get (A B)
|
|
in fact the opposite: (A B) with (A) gives (A)
|
|
*/
|
|
|
|
/**** result=tleft_factor( result ); ****/
|
|
return result;
|
|
};
|
|
|
|
#if 0
|
|
** if (p->expr == PRED_AND_LIST) {
|
|
**
|
|
** Predicate *l;
|
|
** Predicate *r;
|
|
** Tree *l1;
|
|
** Tree *r1;
|
|
** Tree *prevl1;
|
|
**
|
|
** l=p->down;
|
|
** require (l->right != NULL,"MR_compute_pred_tree - AND has only one child");
|
|
**
|
|
**/* l1 and r1 should already be in "canonical" format */
|
|
**
|
|
** l1=MR_compute_pred_tree(l);
|
|
** for (r=l->right; r != NULL; r=r->right) {
|
|
** r1=MR_compute_pred_tree(r);
|
|
** prevl1=l1;
|
|
** l1=MR_computeTreeAND(l1,r1);
|
|
** Tfree(r1);
|
|
** Tfree(prevl1);
|
|
** };
|
|
**
|
|
**/* result from computeTreeAND should be in "canonical" format */
|
|
**
|
|
** result=l1;
|
|
**
|
|
**/* result of MR_computeTreeAND should be in "canonical" format */
|
|
**
|
|
** return result;
|
|
** };
|
|
#endif
|
|
|
|
if (p->k == 1) {
|
|
result=MR_make_tree_from_set(p->scontext[1]);
|
|
} else {
|
|
result=tdup(p->tcontext);
|
|
result=MR_remove_epsilon_from_tree(result);
|
|
result=tshrink(result);
|
|
result=tflatten(result);
|
|
result=tleft_factor(result);
|
|
};
|
|
return result;
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_pred_depth(Predicate *p,int *maxDepth)
|
|
#else
|
|
void MR_pred_depth(p,maxDepth)
|
|
Predicate *p;
|
|
int *maxDepth;
|
|
#endif
|
|
{
|
|
if (p == NULL) return;
|
|
if (p->expr != PRED_OR_LIST &&
|
|
p->expr != PRED_AND_LIST) {
|
|
if (p->k > *maxDepth) *maxDepth=p->k;
|
|
};
|
|
MR_pred_depth(p->down,maxDepth);
|
|
MR_pred_depth(p->right,maxDepth);
|
|
}
|
|
|
|
/* this computes the OR of all the contexts */
|
|
|
|
#ifdef __USE_PROTOS
|
|
set MR_compute_pred_set(Predicate *p)
|
|
#else
|
|
set MR_compute_pred_set(p)
|
|
Predicate *p;
|
|
#endif
|
|
{
|
|
set result;
|
|
Predicate *q;
|
|
|
|
result=empty;
|
|
|
|
if (p == NULL) return empty;
|
|
|
|
if (p->expr == PRED_OR_LIST ||
|
|
p->expr == PRED_AND_LIST) { /* yes, I do mean PRED_AND_LIST ! */
|
|
/* remember: r1: (A)? => <<p>>? r2; */
|
|
/* r2: (B)? => <<q>>? r3; */
|
|
set t;
|
|
|
|
t=empty;
|
|
result=empty;
|
|
|
|
for (q=p->down; q != NULL; q=q->right) {
|
|
t=MR_compute_pred_set(q);
|
|
set_orin(&result,t);
|
|
set_free(t);
|
|
};
|
|
return result;
|
|
} else if (p->k > 1) {
|
|
return empty;
|
|
} else {
|
|
return set_dup(p->scontext[1]);
|
|
};
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
set MR_First(int ck,Junction *j,set *incomplete)
|
|
#else
|
|
set MR_First(ck,j,incomplete)
|
|
int ck;
|
|
Junction *j;
|
|
set *incomplete;
|
|
#endif
|
|
{
|
|
Junction *p;
|
|
set tokensUsed;
|
|
|
|
tokensUsed=empty;
|
|
|
|
require(j->ntype==nJunction, "MR_First: non junction passed");
|
|
|
|
p = analysis_point((Junction *)j->p1);
|
|
|
|
REACH(p,ck,incomplete,tokensUsed);
|
|
|
|
return tokensUsed;
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_cleanup_pred_trees(Predicate *p)
|
|
#else
|
|
void MR_cleanup_pred_trees(p)
|
|
Predicate *p;
|
|
#endif
|
|
{
|
|
Tree *t;
|
|
|
|
if (p == NULL) return;
|
|
if (p->expr != PRED_OR_LIST &&
|
|
p->expr != PRED_AND_LIST) {
|
|
t=p->tcontext;
|
|
t=tshrink(t);
|
|
t=tflatten(t);
|
|
t=tleft_factor(t);
|
|
p->tcontext=t;
|
|
};
|
|
MR_cleanup_pred_trees(p->down);
|
|
MR_cleanup_pred_trees(p->right);
|
|
}
|
|
|
|
/* does NOT return canonical tree */
|
|
|
|
#ifdef __USE_PROTOS
|
|
Tree * MR_remove_epsilon_from_tree(Tree *t)
|
|
#else
|
|
Tree * MR_remove_epsilon_from_tree(t)
|
|
Tree *t;
|
|
#endif
|
|
{
|
|
if (t == NULL) return NULL;
|
|
|
|
/* I think ALT can be ignored as a special case */
|
|
|
|
if (t->token != EpToken) {
|
|
t->down=MR_remove_epsilon_from_tree(t->down);
|
|
t->right=MR_remove_epsilon_from_tree(t->right);
|
|
return t;
|
|
} else {
|
|
Tree *u;
|
|
u=MR_remove_epsilon_from_tree(t->right);
|
|
t->right=NULL;
|
|
Tfree(t);
|
|
return u;
|
|
};
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_complete_set(int predDepth,set *tokensUsed,set *incomplete)
|
|
#else
|
|
void MR_complete_set(predDepth,tokensUsed,incomplete)
|
|
int predDepth;
|
|
set *tokensUsed;
|
|
set *incomplete;
|
|
#endif
|
|
{
|
|
int i;
|
|
RuleRefNode *ruleRef;
|
|
set rk2;
|
|
set b;
|
|
int k2;
|
|
Junction *save_MR_RuleBlkWithHalt;
|
|
|
|
if (set_int(*incomplete) > (unsigned) predDepth) {
|
|
return;
|
|
};
|
|
|
|
require(MR_PredRuleRefStack.count == MR_RuleBlkWithHaltStack.count,
|
|
"RuleRefStack and RuleBlkWithHaltStack not same size");
|
|
|
|
require(MR_RuleBlkWithHalt == NULL ||
|
|
(MR_RuleBlkWithHalt->jtype == RuleBlk && MR_RuleBlkWithHalt->end->halt == TRUE),
|
|
"RuleBlkWithHalt has no halt set");
|
|
|
|
save_MR_RuleBlkWithHalt=MR_RuleBlkWithHalt;
|
|
|
|
if (MR_RuleBlkWithHalt != NULL) {
|
|
MR_RuleBlkWithHalt->end->halt=FALSE;
|
|
};
|
|
|
|
for (i=MR_PredRuleRefStack.count-1; i >= 0 ; i--) {
|
|
ruleRef=(RuleRefNode *)MR_PredRuleRefStack.data[i];
|
|
if (ruleRef == NULL) continue;
|
|
|
|
MR_RuleBlkWithHalt=(Junction *)MR_RuleBlkWithHaltStack.data[i];
|
|
if (MR_RuleBlkWithHalt != NULL) MR_RuleBlkWithHalt->end->halt=TRUE;
|
|
|
|
rk2=empty;
|
|
b=empty;
|
|
|
|
while ( !set_nil(*incomplete) ) {
|
|
k2=set_int(*incomplete);
|
|
if (k2 > predDepth) break; /* <=== another exit from loop */
|
|
set_rm(k2,*incomplete);
|
|
REACH(ruleRef->next,k2,&rk2,b);
|
|
set_orin(tokensUsed,b);
|
|
set_free(b);
|
|
};
|
|
|
|
if (MR_RuleBlkWithHalt != NULL) MR_RuleBlkWithHalt->end->halt=FALSE;
|
|
|
|
set_orin(incomplete,rk2); /* remember what we couldn't do */
|
|
set_free(rk2);
|
|
if (set_int(*incomplete) > (unsigned) predDepth) break; /* <=== another exit from loop */
|
|
};
|
|
|
|
MR_RuleBlkWithHalt=save_MR_RuleBlkWithHalt;
|
|
if (MR_RuleBlkWithHalt != NULL) {
|
|
MR_RuleBlkWithHalt->end->halt=TRUE;
|
|
};
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_complete_tree(int predDepth,Tree **t,set *incomplete)
|
|
#else
|
|
void MR_complete_tree(predDepth,t,incomplete)
|
|
int predDepth;
|
|
Tree **t;
|
|
set *incomplete;
|
|
#endif
|
|
{
|
|
int i;
|
|
RuleRefNode *ruleRef;
|
|
set rk2;
|
|
Tree *u;
|
|
unsigned k2;
|
|
Junction *save_MR_RuleBlkWithHalt;
|
|
int saveConstrainSearch;
|
|
|
|
if (set_int(*incomplete) > (unsigned) predDepth) {
|
|
return;
|
|
};
|
|
|
|
require(MR_PredRuleRefStack.count == MR_RuleBlkWithHaltStack.count,
|
|
"RuleRefStack and RuleBlkWithHaltStack not same size");
|
|
|
|
require(MR_RuleBlkWithHalt == NULL ||
|
|
(MR_RuleBlkWithHalt->jtype == RuleBlk && MR_RuleBlkWithHalt->end->halt == TRUE),
|
|
"RuleBlkWithHalt has no halt set");
|
|
|
|
save_MR_RuleBlkWithHalt=MR_RuleBlkWithHalt;
|
|
saveConstrainSearch=ConstrainSearch;
|
|
ConstrainSearch=0;
|
|
|
|
if (MR_RuleBlkWithHalt != NULL) {
|
|
MR_RuleBlkWithHalt->end->halt=FALSE;
|
|
};
|
|
|
|
for (i=MR_PredRuleRefStack.count-1; i >= 0 ; i--) {
|
|
ruleRef=(RuleRefNode *)MR_PredRuleRefStack.data[i];
|
|
if (ruleRef == NULL) continue;
|
|
|
|
MR_RuleBlkWithHalt=(Junction *)MR_RuleBlkWithHaltStack.data[i];
|
|
|
|
if (MR_RuleBlkWithHalt != NULL) MR_RuleBlkWithHalt->end->halt=TRUE;
|
|
|
|
rk2=empty;
|
|
|
|
while ( !set_nil(*incomplete) ) {
|
|
k2 = set_int(*incomplete);
|
|
if (k2 > (unsigned) predDepth) break; /* <=== another exit from loop */
|
|
set_rm(k2,*incomplete);
|
|
u = NULL;
|
|
|
|
TRAV(ruleRef->next,k2,&rk2,u);
|
|
|
|
/* any subtrees missing k2 tokens, add u onto end */
|
|
|
|
*t=tlink(*t,u,k2);
|
|
Tfree(u);
|
|
}
|
|
|
|
set_orin(incomplete,rk2); /* remember what we couldn't do */
|
|
set_free(rk2);
|
|
|
|
if (MR_RuleBlkWithHalt != NULL) MR_RuleBlkWithHalt->end->halt=FALSE;
|
|
|
|
if (set_int(*incomplete) > (unsigned) predDepth) break; /* <=== another exit from loop */
|
|
};
|
|
|
|
MR_RuleBlkWithHalt=save_MR_RuleBlkWithHalt;
|
|
|
|
if (MR_RuleBlkWithHalt != NULL) {
|
|
MR_RuleBlkWithHalt->end->halt=TRUE;
|
|
};
|
|
ConstrainSearch=saveConstrainSearch;
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_complete_predicates(int predDepth,Predicate *pred)
|
|
#else
|
|
void MR_complete_predicates(predDepth,pred)
|
|
int predDepth;
|
|
Predicate *pred;
|
|
#endif
|
|
{
|
|
if (pred == NULL) return;
|
|
if (pred->expr != PRED_AND_LIST &&
|
|
pred->expr != PRED_OR_LIST) {
|
|
MR_complete_set(predDepth,&(pred->scontext[1]),&(pred->completionSet));
|
|
MR_complete_tree(predDepth,&(pred->tcontext),&(pred->completionTree));
|
|
};
|
|
MR_complete_predicates(predDepth,pred->down);
|
|
MR_complete_predicates(predDepth,pred->right);
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
Junction * MR_junctionWithoutP2(Junction *j)
|
|
#else
|
|
Junction * MR_junctionWithoutP2(j)
|
|
Junction *j;
|
|
#endif
|
|
{
|
|
Junction *thisAlt;
|
|
|
|
/* don't want to follow p2 to the next alternative of this rule */
|
|
/* insert a generic node with null p2 if necessary */
|
|
/* however FIRST requires a junction */
|
|
|
|
thisAlt=j;
|
|
if (thisAlt->p2 != NULL) {
|
|
if (thisAlt->p1->ntype == nJunction) {
|
|
thisAlt=(Junction *) thisAlt->p1;
|
|
} else {
|
|
thisAlt=newJunction();
|
|
thisAlt->p1=j->p1;
|
|
thisAlt->rname=j->rname;
|
|
thisAlt->file=j->file;
|
|
thisAlt->line=j->line;
|
|
j->p1=(Node *)thisAlt;
|
|
};
|
|
};
|
|
return thisAlt;
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
int MR_tree_equ(Tree *big, Tree *small) {
|
|
#else
|
|
int MR_tree_equ(big,small)
|
|
Tree *big;
|
|
Tree *small;
|
|
{
|
|
#endif
|
|
|
|
Tree *b;
|
|
Tree *s;
|
|
int bcount=0;
|
|
int scount=0;
|
|
|
|
if (small == NULL && big == NULL) return 1;
|
|
if (small == NULL) return 0;
|
|
if (big == NULL) return 0;
|
|
|
|
if (small->token == ALT) {
|
|
require(small->right == NULL,
|
|
"MR_tree_equ: small: ALT node has siblings");
|
|
return MR_tree_equ(big,small->down);
|
|
};
|
|
if (big->token == ALT) {
|
|
require(big->right == NULL,
|
|
"MR_tree_equ: big: ALT node has siblings");
|
|
return MR_tree_equ(big->down,small);
|
|
};
|
|
for (s=small; s != NULL; s=s->right) {
|
|
scount++;
|
|
require(s->token != EpToken,"MR_tree_equ: s->EpToken unexpected\n");
|
|
};
|
|
for (b=big; b != NULL; b=b->right) {
|
|
bcount++;
|
|
require(b->token != EpToken,"MR_tree_equ: b->EpToken unexpected\n");
|
|
};
|
|
|
|
if (bcount != scount) return 0;
|
|
|
|
for (s=small; s != NULL; s=s->right) {
|
|
for (b=big; b!= NULL; b=b->right) {
|
|
if (s->token == b->token) {
|
|
if (MR_tree_equ(b->down,s->down)) goto next_s;
|
|
};
|
|
};
|
|
return 0;
|
|
next_s:
|
|
continue;
|
|
};
|
|
return 1;
|
|
}
|
|
|
|
/* this does not compare sources - only contexts ! */
|
|
|
|
#ifdef __USE_PROTOS
|
|
int MR_identicalContext(Predicate *p,Predicate *q)
|
|
#else
|
|
int MR_identicalContext(p,q)
|
|
Predicate *p;
|
|
Predicate *q;
|
|
#endif
|
|
{
|
|
if (p->k != q->k) return 0;
|
|
require ( (p->tcontext == NULL) == (q->tcontext == NULL),
|
|
"tcontext inconsistent");
|
|
if (p->k == 1) {
|
|
return set_equ(p->scontext[1],q->scontext[1]);
|
|
} else {
|
|
return MR_tree_equ(p->tcontext,q->tcontext);
|
|
};
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_reportSetSuppression(int predDepth,
|
|
set predSet,set plainSet,Junction *jPred,Junction *jPlain,Predicate *p)
|
|
#else
|
|
void MR_reportSetSuppression(predDepth,predSet,plainSet,jPred,jPlain,p)
|
|
int predDepth;
|
|
set predSet;
|
|
set plainSet;
|
|
Junction *jPred;
|
|
Junction *jPlain;
|
|
Predicate *p;
|
|
#endif
|
|
{
|
|
if (InfoP) {
|
|
fprintf(output,"\n#if 0\n\n");
|
|
fprintf(output,"Hoisting of predicate suppressed by alternative without predicate.\n");
|
|
fprintf(output,"The alt without the predicate includes all cases where the predicate is false.\n\n");
|
|
fprintf(output," WITH predicate: line %d %s\n",jPred->line,FileStr[jPred->file]);
|
|
if (jPlain != NULL) {
|
|
fprintf(output," WITHOUT predicate: line %d %s\n",jPlain->line,FileStr[jPlain->file]);
|
|
} else {
|
|
fprintf(output," WITHOUT predicate: all alternatives without predicates (combined)\n");
|
|
};
|
|
if (predDepth == 1) {
|
|
fprintf(output,"\nThe context set for the predicate:\n");
|
|
MR_dumpTokenSet(output,1,predSet);
|
|
};
|
|
fprintf(output,"\nThe lookahead set for the alt WITHOUT the semantic predicate:\n");
|
|
MR_dumpTokenSet(output,1,plainSet);
|
|
fprintf(output,"\nThe predicate:\n\n");
|
|
MR_dumpPred1(1,p,1);
|
|
fprintf(output,"Chain of referenced rules:\n\n");
|
|
MR_dumpPredRuleRefStack(output,4);
|
|
fprintf(output,"\n#endif\n");
|
|
};
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_reportSetRestriction(int predDepth,set predSet,set plainSet,
|
|
Junction *jPred,Junction *jPlain,Predicate *origPred,Predicate *newPred)
|
|
#else
|
|
void MR_reportSetRestriction(predDepth,predSet,plainSet,jPred,jPlain,origPred,newPred)
|
|
int predDepth;
|
|
set predSet;
|
|
set plainSet;
|
|
Junction *jPred;
|
|
Junction *jPlain;
|
|
Predicate *origPred;
|
|
Predicate *newPred;
|
|
#endif
|
|
{
|
|
set intersect;
|
|
|
|
intersect=empty;
|
|
|
|
if (! InfoP) return;
|
|
fprintf(output,"\n#if 0\n\n");
|
|
fprintf(output,"Restricting the context of a predicate because of overlap in the lookahead set\n");
|
|
fprintf(output," between the alternative with the semantic predicate and one without\n");
|
|
fprintf(output,"Without this restriction the alternative without the predicate could not\n");
|
|
fprintf(output," be reached when input matched the context of the predicate and the predicate\n");
|
|
fprintf(output," was false.\n\n");
|
|
|
|
fprintf(output," WITH predicate: line %d %s\n",jPred->line,FileStr[jPred->file]);
|
|
if (jPlain != NULL) {
|
|
fprintf(output," WITHOUT predicate: line %d %s\n",jPlain->line,FileStr[jPlain->file]);
|
|
} else {
|
|
fprintf(output," WITHOUT predicate: all alternatives without predicates (combined)\n");
|
|
};
|
|
if (predDepth == 1) {
|
|
fprintf(output,"\nThe original context set for the predicate:\n");
|
|
MR_dumpTokenSet(output,1,predSet);
|
|
};
|
|
fprintf(output,"\nThe lookahead set for the alt WITHOUT the semantic predicate:\n");
|
|
MR_dumpTokenSet(output,1,plainSet);
|
|
if (predDepth == 1) {
|
|
fprintf(output,"\nThe intersection of the two sets\n");
|
|
intersect=set_and(predSet,plainSet);
|
|
MR_dumpTokenSet(output,1,intersect);
|
|
set_free(intersect);
|
|
};
|
|
fprintf(output,"\nThe original predicate:\n\n");
|
|
MR_dumpPred1(1,origPred,1);
|
|
fprintf(output,"The new (modified) form of the predicate:\n\n");
|
|
MR_dumpPred1(1,newPred,1);
|
|
fprintf(output,"#endif\n");
|
|
}
|
|
|
|
/* don't use Pass3 by itself unless you know that inverted is not important */
|
|
|
|
#ifdef __USE_PROTOS
|
|
Predicate * MR_removeRedundantPredPass3(Predicate *p)
|
|
#else
|
|
Predicate * MR_removeRedundantPredPass3(p)
|
|
Predicate *p;
|
|
#endif
|
|
{
|
|
Predicate *q;
|
|
|
|
if (p == NULL) return NULL;
|
|
p->right=MR_removeRedundantPredPass3(p->right);
|
|
p->down=MR_removeRedundantPredPass3(p->down);
|
|
if (p->redundant) {
|
|
q=p->right;
|
|
p->right=NULL;
|
|
predicate_free(p);
|
|
return q;
|
|
};
|
|
if (p->expr == PRED_AND_LIST ||
|
|
p->expr == PRED_OR_LIST) {
|
|
if (p->down == NULL) {
|
|
q=p->right;
|
|
p->right=NULL;
|
|
predicate_free(p);
|
|
return q;
|
|
};
|
|
if (p->down != NULL && p->down->right == NULL) {
|
|
q=p->down;
|
|
q->right=p->right;
|
|
p->right=NULL;
|
|
p->down=NULL;
|
|
return q;
|
|
};
|
|
};
|
|
return p;
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_removeRedundantPredPass2(Predicate *p)
|
|
#else
|
|
void MR_removeRedundantPredPass2(p)
|
|
Predicate *p;
|
|
#endif
|
|
{
|
|
Predicate *q;
|
|
|
|
if (p == NULL) return;
|
|
|
|
if (p->expr == PRED_AND_LIST) {
|
|
for (q=p->down ; q != NULL ; q=q->right) {
|
|
MR_removeRedundantPredPass2(q);
|
|
if (q->isConst) {
|
|
if (q->constValue == 0) {
|
|
p->isConst=1;
|
|
p->constValue=0;
|
|
return;
|
|
} else {
|
|
q->redundant=1;
|
|
};
|
|
};
|
|
};
|
|
};
|
|
|
|
if (p->expr == PRED_OR_LIST) {
|
|
for (q=p->down ; q != NULL ; q=q->right) {
|
|
MR_removeRedundantPredPass2(q);
|
|
if (q->isConst) {
|
|
if (q->constValue == 0) {
|
|
q->redundant=1;
|
|
} else {
|
|
p->isConst=1;
|
|
p->constValue=1;
|
|
return;
|
|
};
|
|
};
|
|
};
|
|
};
|
|
|
|
return;
|
|
}
|
|
|
|
#if 0
|
|
this totally ignores the implications of guarded predicates
|
|
in which the part after the guard could possibly cover a predicate.
|
|
that would be much harder:
|
|
|
|
rule : (A)? => <<p>>? sub1; /* 1 */
|
|
| (B)? => <<r>>? sub2 /* 2 */
|
|
sub1 : (A)? => <<q>>? A B /* 3 */
|
|
| B /* 4 - suppresses line 2 */
|
|
;
|
|
#endif
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_apply_restriction1(Predicate *pred,set *plainSet,int *changed)
|
|
#else
|
|
void MR_apply_restriction1(pred,plainSet,changed)
|
|
Predicate *pred;
|
|
set *plainSet;
|
|
int *changed;
|
|
#endif
|
|
{
|
|
if (pred == NULL) return;
|
|
MR_apply_restriction1(pred->right,plainSet,changed);
|
|
if (pred->down != NULL) {
|
|
MR_apply_restriction1(pred->down,plainSet,changed);
|
|
} else {
|
|
set t;
|
|
if (pred->k == 1) {
|
|
t=set_dif(pred->scontext[1],*plainSet);
|
|
if (*changed == 0 &&
|
|
!set_equ(t,pred->scontext[1])) {
|
|
*changed=1;
|
|
};
|
|
if (set_nil(t)) {
|
|
pred->redundant=1;
|
|
};
|
|
set_free(pred->scontext[1]);
|
|
pred->scontext[1]=t;
|
|
};
|
|
};
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_orin_plainSet(Predicate *p,set plainSet)
|
|
#else
|
|
void MR_orin_plainSet(p,plainSet)
|
|
Predicate *p;
|
|
set plainSet;
|
|
#endif
|
|
{
|
|
if (p == NULL) return;
|
|
MR_orin_plainSet(p->down,plainSet);
|
|
MR_orin_plainSet(p->right,plainSet);
|
|
set_orin(&p->plainSet,plainSet);
|
|
}
|
|
|
|
Predicate *PRED_SUPPRESS;
|
|
|
|
#ifdef __USE_PROTOS
|
|
Predicate * MR_find_in_aSubBlk(Junction *alt)
|
|
#else
|
|
Predicate * MR_find_in_aSubBlk(alt)
|
|
Junction *alt;
|
|
#endif
|
|
{
|
|
Predicate *root=NULL;
|
|
Predicate **tail=NULL;
|
|
|
|
Junction *p;
|
|
|
|
int nAlts=0;
|
|
Junction **jList;
|
|
Predicate **predList;
|
|
int *matchList;
|
|
set predSet;
|
|
int i;
|
|
int j;
|
|
int m;
|
|
int predDepth;
|
|
set incomplete;
|
|
set union_plainSet;
|
|
set setChange;
|
|
int changed;
|
|
Predicate *newPred;
|
|
set setDif;
|
|
Predicate *origPred;
|
|
int depth1=1; /* const int */
|
|
set *plainContext;
|
|
set plainSet;
|
|
|
|
predSet=empty;
|
|
incomplete=empty;
|
|
union_plainSet=empty;
|
|
setChange=empty;
|
|
setDif=empty;
|
|
plainSet=empty;
|
|
|
|
if (PRED_SUPPRESS == NULL) {
|
|
PRED_SUPPRESS=new_pred();
|
|
PRED_SUPPRESS->expr="Predicate Suppressed";
|
|
};
|
|
|
|
/* this section just counts the number of "interesting" alternatives */
|
|
/* in order to allocate arrays */
|
|
|
|
for (p=alt; p!=NULL; p=(Junction *)p->p2) {
|
|
/* ignore empty alts */
|
|
if ( p->p1->ntype != nJunction ||
|
|
((Junction *)p->p1)->jtype != EndBlk ) {
|
|
nAlts++;
|
|
};
|
|
};
|
|
|
|
/* if this is a (...)+ block then don't count the last alt because
|
|
it can't be taken until at least one time through the block.
|
|
In other words it isn't a real choice until the (...)+ is entered
|
|
at which point the hoisting issue is moot.
|
|
Maybe look at "ignore" instead ?
|
|
*/
|
|
|
|
if (alt->jtype == aPlusBlk) {
|
|
nAlts--;
|
|
};
|
|
|
|
jList=(Junction **)calloc(nAlts,sizeof(Junction *));
|
|
require(jList!=NULL,"cannot allocate MR_find_in_aSubBlk jList");
|
|
|
|
plainContext=(set *)calloc(nAlts,sizeof(set));
|
|
require(plainContext!=NULL,"cannot allocate MR_find_in_aSubBlk plainContext");
|
|
for (m=0; m < nAlts; m++) plainContext[m]=empty;
|
|
|
|
predList=(Predicate **)calloc(nAlts,sizeof(Predicate *));
|
|
require(predList!=NULL,"cannot allocate MR_find_in_aSubBlk predList");
|
|
|
|
matchList=(int *)calloc(nAlts,sizeof(int));
|
|
require(matchList!=NULL,"cannot allocate MR_find_in_aSubBlk matchList");
|
|
|
|
/* this section just fills in the arrays previously allocated */
|
|
/* the most interesting one is matchList[] */
|
|
/* */
|
|
/* bit 0 => this alt has a semantic pred which is "covered" */
|
|
/* by an alt without a semantic pred. Don't hoist. */
|
|
|
|
for (i=0,p=alt;
|
|
p!=NULL && i<nAlts;
|
|
i++,p=(Junction *)p->p2) {
|
|
|
|
/* ignore empty alts */
|
|
|
|
if ( p->p1->ntype != nJunction ||
|
|
((Junction *)p->p1)->jtype != EndBlk ) {
|
|
jList[i]=MR_junctionWithoutP2(p);
|
|
predList[i]=find_predicates(p->p1); /* should be jList ????? */
|
|
if (predList[i] != NULL) {
|
|
MR_cleanup_pred_trees(predList[i]); /* flatten & left factor */
|
|
plainContext[i]=MR_union_plain_sets(predList[i]);
|
|
} else {
|
|
MR_set_reuse(&plainSet);
|
|
MR_set_reuse(&incomplete);
|
|
plainSet=MR_First(depth1,jList[i],&incomplete);
|
|
MR_complete_set(depth1,&plainSet,&incomplete);
|
|
require(set_nil(incomplete),"couldn't complete k=1");
|
|
plainContext[i]=plainSet;
|
|
plainSet=empty;
|
|
};
|
|
set_orin(&union_plainSet,plainContext[i]);
|
|
};
|
|
};
|
|
|
|
if (nAlts == 1) {
|
|
goto EXIT_SIMPLE;
|
|
};
|
|
|
|
/*
|
|
* Looking for cases where alt i has a semantic pred and alt j does not.
|
|
* Don't care about cases where lookahead for semantic predicates overlap
|
|
* because normal predicate hoisting does the correct thing automatically.
|
|
* Don't care about cases where lookahead for alts without semantic predicates
|
|
* overlap because normal prediction does the correct thing automatically.
|
|
*
|
|
* When we find such a case check for one of three subcases:
|
|
*
|
|
* 1. if lookahead for alt i is contained in the lookahead for any
|
|
* alt j then ignore semantic predicate of alt i
|
|
* 2. if lookahead for alt i is not contained in the lookahead for
|
|
* any alt j then add predicate i to the OR list to be hoisted
|
|
* 3. if lookahead for alt i overlaps the lookahead for some alt j then
|
|
* add a dummy semantic predicate for alt j
|
|
*
|
|
* There is an implicit assumption that the context of all alternatives following
|
|
* the rule being processed here are identical (but may vary from hoist to
|
|
* hoist depending on the place where the rule was invoked that led to hoisting
|
|
* these predicates. In othere words in the fragment:
|
|
*
|
|
* ( <<a>>? a1 a2 a3 | <<b>>? b1 b2 b3 )
|
|
*
|
|
* both a3 and b3 have the same follow sets because they are both at the end of
|
|
* alternatives in the same block.
|
|
*/
|
|
|
|
for (i=0; i < nAlts; i++) {
|
|
if (jList[i] == NULL) continue;
|
|
if (predList[i] == NULL) continue;
|
|
|
|
/* if the predicate depth turns out to be one token only */
|
|
/* then it is can be easily represented as a set and */
|
|
/* compared to the junction set create by MR_First() */
|
|
|
|
predDepth=0;
|
|
MR_pred_depth(predList[i],&predDepth);
|
|
require (predDepth >= 1,"MR_find_in_aSubBlk: pred depth < 1");
|
|
require (predDepth <= CLL_k,"MR_find_in_aSubBlk: predDepth > CLL_k");
|
|
|
|
/* complete predicates to predDepth
|
|
If completed to depth=1 then the context would be incomplete.
|
|
The context would be truncated and the predicate simplify routine
|
|
would have incomplete information. It would lead to
|
|
either false matches of failure to find true matches.
|
|
*/
|
|
|
|
MR_complete_predicates(predDepth,predList[i]);
|
|
|
|
if (predList[i] != NULL) {
|
|
MR_cleanup_pred_trees(predList[i]); /* flatten & left factor */
|
|
};
|
|
|
|
/* If the predicate depth is 1 then it is possible to suppress
|
|
a predicate completely using a single plain alt. Check for suppression
|
|
by a single plain alt first because it gives better messages. If that
|
|
fails try the union of all the plain alts.
|
|
*/
|
|
|
|
if (predDepth == 1) {
|
|
|
|
MR_set_reuse(&predSet);
|
|
predSet=MR_compute_pred_set(predList[i]); /* ignores k>1 predicates */
|
|
|
|
for (j=0; j < nAlts; j++) {
|
|
if (jList[j] == NULL) continue;
|
|
if (j == i) continue;
|
|
|
|
MR_set_reuse(&setDif);
|
|
setDif=set_dif(predSet,plainContext[j]);
|
|
if (set_nil(setDif)) {
|
|
matchList[i] |= 1;
|
|
MR_reportSetSuppression(predDepth,predSet,plainContext[j],jList[i],jList[j],predList[i]);
|
|
predicate_free(predList[i]);
|
|
predList[i]=PRED_SUPPRESS;
|
|
goto next_i;
|
|
};
|
|
|
|
}; /* end loop on j */
|
|
|
|
changed=0;
|
|
|
|
/* predicate_dup is only to give good error messages */
|
|
/* remember to do a predicate_free() */
|
|
|
|
origPred=predicate_dup(predList[i]);
|
|
MR_apply_restriction1(predList[i],&union_plainSet,&changed);
|
|
if (changed) {
|
|
|
|
/* don't use Pass3 by itself unless you know that inverted is not important */
|
|
|
|
newPred=MR_removeRedundantPredPass3(predList[i]);
|
|
newPred=MR_predSimplifyALL(newPred);
|
|
if (newPred == NULL) {
|
|
matchList[i] |= 1;
|
|
MR_reportSetSuppression(predDepth,predSet,union_plainSet,jList[i],
|
|
NULL,origPred);
|
|
predList[i]=PRED_SUPPRESS;
|
|
} else {
|
|
MR_reportSetRestriction(predDepth,predSet,union_plainSet,jList[i],
|
|
NULL,origPred,newPred);
|
|
predList[i]=newPred;
|
|
};
|
|
};
|
|
predicate_free(origPred);
|
|
origPred=NULL;
|
|
};
|
|
|
|
/*
|
|
If the predicate depth is > 1 then it can't be suppressed completely
|
|
because the code doesn't support inspection of such things. They're
|
|
much messier than k=1 sets.
|
|
*/
|
|
|
|
if (predDepth > 1 ) {
|
|
|
|
changed=0;
|
|
|
|
/* predicate_dup is only to give good error messages */
|
|
/* remember to do a predicate_free() */
|
|
|
|
origPred=predicate_dup(predList[i]);
|
|
MR_apply_restriction1(predList[i],&union_plainSet,&changed);
|
|
if (changed) {
|
|
newPred=MR_removeRedundantPredPass3(predList[i]);
|
|
newPred=MR_predSimplifyALL(newPred);
|
|
if (newPred == NULL) {
|
|
matchList[i] |= 1;
|
|
MR_reportSetSuppression(predDepth,predSet,union_plainSet,jList[i],
|
|
NULL,origPred);
|
|
predList[i]=PRED_SUPPRESS;
|
|
} else {
|
|
MR_reportSetRestriction(predDepth,predSet,union_plainSet,jList[i],
|
|
NULL,origPred,newPred);
|
|
predList[i]=newPred;
|
|
};
|
|
};
|
|
predicate_free(origPred);
|
|
origPred=NULL;
|
|
};
|
|
next_i:
|
|
continue;
|
|
};
|
|
|
|
EXIT_SIMPLE:
|
|
|
|
root = new_pred();
|
|
root->expr=PRED_OR_LIST;
|
|
tail = &(root->down);
|
|
|
|
for (i=0 ; i< nAlts ; i++) {
|
|
if (jList[i] == NULL) continue;
|
|
|
|
if (predList[i] == NULL) {
|
|
continue;
|
|
} else if ( (matchList[i] & 1) != 0) {
|
|
if (predList[i] != PRED_SUPPRESS) {
|
|
predicate_free(predList[i]);
|
|
};
|
|
continue;
|
|
};
|
|
|
|
/* make an OR list of predicates */
|
|
|
|
*tail=predList[i];
|
|
tail=&(predList[i]->right);
|
|
};
|
|
|
|
/* if just one pred, remove OR root */
|
|
|
|
if (root->down == NULL) {
|
|
predicate_free(root);
|
|
root=NULL;
|
|
} else if (root->down->right == NULL) {
|
|
Predicate *p=root->down;
|
|
root->down=NULL;
|
|
predicate_free(root);
|
|
root=p;
|
|
}
|
|
|
|
root=MR_predSimplifyALL(root);
|
|
|
|
MR_orin_plainSet(root,union_plainSet);
|
|
|
|
set_free(predSet);
|
|
set_free(union_plainSet);
|
|
set_free(incomplete);
|
|
set_free(setChange);
|
|
set_free(setDif);
|
|
|
|
for (m=0; m < nAlts; m++) set_free(plainContext[m]);
|
|
|
|
free ( (char *) jList);
|
|
free ( (char *) predList);
|
|
free ( (char *) matchList);
|
|
free ( (char *) plainContext);
|
|
|
|
return root;
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_predContextPresent(Predicate *p,int *allHaveContext,int *noneHaveContext)
|
|
#else
|
|
void MR_predContextPresent(p,allHaveContext,noneHaveContext)
|
|
Predicate *p;
|
|
int *allHaveContext;
|
|
int *noneHaveContext;
|
|
#endif
|
|
{
|
|
if (p == NULL) return;
|
|
MR_predContextPresent(p->right,allHaveContext,noneHaveContext);
|
|
if (p->expr != PRED_AND_LIST &&
|
|
p->expr != PRED_OR_LIST) {
|
|
if (set_nil(p->scontext[1]) == 0 ||
|
|
(p->tcontext != NULL)) {
|
|
*noneHaveContext=0;
|
|
} else {
|
|
*allHaveContext=0;
|
|
};
|
|
};
|
|
MR_predContextPresent(p->down,allHaveContext,noneHaveContext);
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
int MR_pointerStackPush(PointerStack *ps,void *dataPointer)
|
|
#else
|
|
int MR_pointerStackPush(ps,dataPointer)
|
|
PointerStack *ps;
|
|
void *dataPointer;
|
|
#endif
|
|
{
|
|
void **newStack;
|
|
int newSize;
|
|
int i;
|
|
|
|
if (ps->count == ps->size) {
|
|
newSize=20+ps->size*2;
|
|
newStack=(void **)calloc(newSize,sizeof(void *));
|
|
require (newStack != NULL,"cannot allocate PointerStack");
|
|
for (i=0; i < ps->size; i++) {
|
|
newStack[i]=ps->data[i];
|
|
};
|
|
if (ps->data != NULL) free( (char *) ps->data);
|
|
ps->data=newStack;
|
|
ps->size=newSize;
|
|
};
|
|
ps->data[ps->count]=dataPointer;
|
|
ps->count++;
|
|
return ps->count-1;
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void * MR_pointerStackPop(PointerStack *ps)
|
|
#else
|
|
void * MR_pointerStackPop(ps)
|
|
PointerStack *ps;
|
|
#endif
|
|
{
|
|
void *dataPointer;
|
|
|
|
require(ps->count > 0,"MR_pointerStackPop underflow");
|
|
|
|
dataPointer=ps->data[ps->count-1];
|
|
ps->data[ps->count-1]=NULL;
|
|
(ps->count)--;
|
|
return dataPointer;
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void * MR_pointerStackTop(PointerStack *ps)
|
|
#else
|
|
void * MR_pointerStackTop(ps)
|
|
PointerStack *ps;
|
|
#endif
|
|
{
|
|
require(ps->count > 0,"MR_pointerStackTop underflow");
|
|
return ps->data[ps->count-1];
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_pointerStackReset(PointerStack *ps)
|
|
#else
|
|
void MR_pointerStackReset(ps)
|
|
PointerStack *ps;
|
|
#endif
|
|
{
|
|
int i;
|
|
if (ps->data != NULL) {
|
|
for (i=0; i < ps->count ; i++) {
|
|
ps->data[i]=NULL;
|
|
};
|
|
};
|
|
ps->count=0;
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
Junction *MR_nameToRuleBlk(char *name)
|
|
#else
|
|
Junction *MR_nameToRuleBlk(name)
|
|
char *name;
|
|
#endif
|
|
{
|
|
RuleEntry *q;
|
|
|
|
require (RulePtr != NULL,"MR_nameToRule: RulePtr not initialized");
|
|
|
|
if (name == NULL) return NULL;
|
|
|
|
q = (RuleEntry *) hash_get(Rname,name);
|
|
|
|
if ( q == NULL ) {
|
|
return NULL;
|
|
} else {
|
|
return RulePtr[q->rulenum];
|
|
};
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
Junction * MR_ruleReferenced(RuleRefNode *rrn)
|
|
#else
|
|
Junction * MR_ruleReferenced(rrn)
|
|
RuleRefNode *rrn;
|
|
#endif
|
|
{
|
|
return MR_nameToRuleBlk(rrn->text);
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_comparePredLeaves(Predicate *me,Predicate *myParent,Predicate *him,Predicate *hisParent)
|
|
#else
|
|
void MR_comparePredLeaves(me,myParent,him,hisParent)
|
|
Predicate *me;
|
|
Predicate *myParent;
|
|
Predicate *him;
|
|
Predicate *hisParent;
|
|
#endif
|
|
{
|
|
if (me == NULL) return;
|
|
if (me == him) {
|
|
MR_comparePredLeaves(me->right,myParent,him,hisParent);
|
|
return;
|
|
} else if (me->expr == PRED_AND_LIST ||
|
|
me->expr == PRED_OR_LIST) {
|
|
MR_comparePredLeaves(me->down,me,him,hisParent);
|
|
MR_comparePredLeaves(me->right,myParent,him,hisParent);
|
|
return;
|
|
} else {
|
|
if (me->source != NULL) {
|
|
|
|
/* predicate->invert can be set only in the predEntry predicates */
|
|
/* thus they are only visible after the predEntry predicates have been "unfolded" */
|
|
|
|
int sameSource=(me->source == him->source);
|
|
int sameInvert=1 &
|
|
(1 + me->inverted + him->inverted + me->source->inverted + him->source->inverted);
|
|
int samePredEntry=(me->source->predEntry != NULL
|
|
&& him->source->predEntry != NULL
|
|
&& me->source->predEntry == him->source->predEntry);
|
|
if (sameInvert && (sameSource || samePredEntry)) {
|
|
if (MR_identicalContext(me,him)) {
|
|
|
|
/* identical predicates */
|
|
|
|
if (hisParent->expr == PRED_OR_LIST &&
|
|
myParent->expr == PRED_OR_LIST) {
|
|
me->redundant=1;
|
|
} else if (hisParent->expr == PRED_AND_LIST &&
|
|
myParent->expr == PRED_AND_LIST) {
|
|
me->redundant=1;
|
|
} else if ( (hisParent->expr == PRED_OR_LIST &&
|
|
myParent->expr == PRED_AND_LIST)
|
|
||
|
|
(hisParent->expr == PRED_AND_LIST &&
|
|
myParent->expr == PRED_OR_LIST)
|
|
) {
|
|
myParent->redundant=1;
|
|
} else {
|
|
require (0,"MR_comparePredLeaves: not both PRED_LIST");
|
|
};
|
|
};
|
|
}; /* end same source or same predEntrr with same invert sense */
|
|
|
|
/* same predEntry but opposite invert sense */
|
|
|
|
if (!sameInvert && (sameSource || samePredEntry)) {
|
|
if (MR_identicalContext(me,him)) {
|
|
if (hisParent->expr == PRED_OR_LIST &&
|
|
myParent->expr == PRED_OR_LIST) {
|
|
myParent->isConst=1;
|
|
myParent->constValue=1;
|
|
} else if (hisParent->expr == PRED_AND_LIST &&
|
|
myParent->expr == PRED_AND_LIST) {
|
|
myParent->isConst=1;
|
|
myParent->constValue=0;
|
|
} else if ( (hisParent->expr == PRED_OR_LIST &&
|
|
myParent->expr == PRED_AND_LIST)
|
|
||
|
|
(hisParent->expr == PRED_AND_LIST &&
|
|
myParent->expr == PRED_OR_LIST)
|
|
) {
|
|
me->redundant=1;
|
|
} else {
|
|
require (0,"MR_comparePredLeaves: not both PRED_LIST");
|
|
};
|
|
};
|
|
}; /* end same predEntry with opposite invert sense */
|
|
};
|
|
|
|
MR_comparePredLeaves(me->right,myParent,him,hisParent);
|
|
return;
|
|
};
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_removeRedundantPredPass1(Predicate *me,Predicate *myParent)
|
|
#else
|
|
void MR_removeRedundantPredPass1(me,myParent)
|
|
Predicate *me;
|
|
Predicate *myParent;
|
|
#endif
|
|
{
|
|
if (me == NULL) return;
|
|
if (me->redundant) {
|
|
MR_removeRedundantPredPass1(me->right,myParent);
|
|
return;
|
|
};
|
|
if (me->expr == PRED_AND_LIST ||
|
|
me->expr == PRED_OR_LIST) {
|
|
MR_removeRedundantPredPass1(me->down,me);
|
|
MR_removeRedundantPredPass1(me->right,myParent);
|
|
} else {
|
|
require (me->source != NULL,"me->source == NULL");
|
|
if (myParent != NULL) {
|
|
MR_comparePredLeaves(myParent->down,myParent,me,myParent);
|
|
};
|
|
MR_removeRedundantPredPass1(me->right,myParent);
|
|
};
|
|
}
|
|
|
|
/* pretty much ignores things with the inverted bit set */
|
|
|
|
#ifdef __USE_PROTOS
|
|
Predicate *MR_predFlatten(Predicate *p)
|
|
#else
|
|
Predicate *MR_predFlatten(p)
|
|
Predicate *p;
|
|
#endif
|
|
{
|
|
if (p == NULL) return NULL;
|
|
if (p->expr == PRED_OR_LIST
|
|
|| p->expr == PRED_AND_LIST) {
|
|
|
|
Predicate *child;
|
|
Predicate *gchild;
|
|
Predicate **tail;
|
|
Predicate *next;
|
|
char *PRED_XXX_LIST=p->expr;
|
|
|
|
require (p->down != NULL,"MR_predFlatten AND/OR no child");
|
|
|
|
|
|
p->down=MR_predFlatten(p->down);
|
|
p->right=MR_predFlatten(p->right);
|
|
child=p->down;
|
|
if (child->right == NULL) {
|
|
child->right=p->right;
|
|
p->right=NULL;
|
|
p->down=NULL;
|
|
if (p->inverted) child->inverted=!child->inverted;
|
|
predicate_free(p);
|
|
return child;
|
|
};
|
|
|
|
/* make a single list of all children and grandchildren */
|
|
|
|
tail=&(p->down);
|
|
for (child=p->down; child != NULL; child=next) {
|
|
if (child->expr != PRED_XXX_LIST
|
|
|| child->inverted
|
|
|| child->predEntry != NULL) {
|
|
*tail=child;
|
|
tail=&(child->right);
|
|
next=child->right;
|
|
} else {
|
|
for (gchild=child->down;
|
|
gchild != NULL;
|
|
gchild=gchild->right) {
|
|
*tail=gchild;
|
|
tail=&(gchild->right);
|
|
};
|
|
next=child->right;
|
|
child->right=NULL;
|
|
child->down=NULL;
|
|
predicate_free(child);
|
|
};
|
|
};
|
|
*tail=NULL;
|
|
return p;
|
|
} else {
|
|
p->right=MR_predFlatten(p->right);
|
|
return p;
|
|
};
|
|
}
|
|
|
|
static char *alwaysFalseWarning=NULL;
|
|
|
|
#ifdef __USE_PROTOS
|
|
Predicate *checkPredicateConflict(Predicate *p)
|
|
#else
|
|
Predicate *checkPredicateConflict(p)
|
|
Predicate *p;
|
|
#endif
|
|
{
|
|
if (p->isConst) {
|
|
if (p->constValue == 1) {
|
|
predicate_free(p);
|
|
return NULL;
|
|
} else {
|
|
if (InfoP && !p->conflictReported) {
|
|
p->conflictReported=1;
|
|
fprintf(output,"\n#if 0\n\n");
|
|
fprintf(output,"The following predicate expression will always be false:\n\n");
|
|
MR_dumpPred1(1,p,1);
|
|
fprintf(output,"\n#endif\n");
|
|
};
|
|
|
|
if (alwaysFalseWarning != CurRule) {
|
|
alwaysFalseWarning=CurRule;
|
|
if (InfoP) {
|
|
warnNoFL(eMsg1("one (or more) predicate expression hoisted into rule \"%s\" are always false \
|
|
- see output file for more information",CurRule));
|
|
} else {
|
|
warnNoFL(eMsg1("one (or more) predicate expressions hoisted into rule \"%s\" are always false \
|
|
- use \"-info p\" for more information",CurRule));
|
|
};
|
|
};
|
|
};
|
|
};
|
|
return p;
|
|
}
|
|
|
|
|
|
#ifdef __USE_PROTOS
|
|
int MR_countPredNodes(Predicate *p)
|
|
#else
|
|
int MR_countPredNodes(p)
|
|
Predicate *p;
|
|
#endif
|
|
{
|
|
if (p == NULL) return 0;
|
|
return 1 + MR_countPredNodes(p->down) + MR_countPredNodes(p->right);
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
Predicate *MR_predSimplifyALLX(Predicate *p,int skipPass3)
|
|
#else
|
|
Predicate *MR_predSimplifyALLX(p,skipPass3)
|
|
Predicate *p;
|
|
int skipPass3;
|
|
#endif
|
|
{
|
|
int countBefore;
|
|
int countAfter;
|
|
|
|
countAfter=MR_countPredNodes(p);
|
|
|
|
do {
|
|
if (p == NULL) return NULL;
|
|
if (p->right == NULL && p->down == NULL) return p;
|
|
countBefore=countAfter;
|
|
MR_simplifyInverted(p,0);
|
|
p=MR_predFlatten(p);
|
|
MR_removeRedundantPredPass1(p,NULL);
|
|
MR_removeRedundantPredPass2(p);
|
|
if (! skipPass3) {
|
|
p=checkPredicateConflict(p);
|
|
p=MR_removeRedundantPredPass3(p);
|
|
};
|
|
countAfter=MR_countPredNodes(p);
|
|
} while (countBefore != countAfter);
|
|
|
|
return p;
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
Predicate *MR_predSimplifyALL(Predicate *p)
|
|
#else
|
|
Predicate *MR_predSimplifyALL(p)
|
|
Predicate *p;
|
|
#endif
|
|
{
|
|
return MR_predSimplifyALLX(p,0);
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_releaseResourcesUsedInRule(Node *n)
|
|
#else
|
|
void MR_releaseResourcesUsedInRule(n)
|
|
Node *n;
|
|
#endif
|
|
{
|
|
Node *next;
|
|
Junction *j;
|
|
int i;
|
|
|
|
if (n == NULL) return;
|
|
if (n->ntype == nJunction) {
|
|
j=(Junction *) n;
|
|
|
|
if (j->predicate != NULL) {
|
|
predicate_free(j->predicate);
|
|
j->predicate=NULL;
|
|
};
|
|
for (i=0; i< CLL_k; i++) {
|
|
set_free(j->fset[i]);
|
|
j->fset[i]=empty;
|
|
};
|
|
if (j->ftree != NULL) {
|
|
Tfree(j->ftree);
|
|
j->ftree=NULL;
|
|
};
|
|
if (j->jtype == EndRule) return;
|
|
if (j->jtype != RuleBlk && j->jtype != EndBlk) {
|
|
if (j->p2 != NULL && !j->ignore) { /* MR11 */
|
|
MR_releaseResourcesUsedInRule(j->p2);
|
|
};
|
|
};
|
|
};
|
|
next=MR_advance(n);
|
|
MR_releaseResourcesUsedInRule(next);
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
int MR_allPredLeaves(Predicate *p)
|
|
#else
|
|
int MR_allPredLeaves(p)
|
|
Predicate *p;
|
|
#endif
|
|
{
|
|
Predicate *q;
|
|
|
|
if (p == NULL) return 1;
|
|
|
|
for (q=p; q != NULL; q=q->right) {
|
|
if (q->down != NULL) return 0;
|
|
};
|
|
return 1;
|
|
}
|
|
|
|
/* make sure it works for the last rule in a file */
|
|
|
|
#ifdef __USE_PROTOS
|
|
int MR_offsetFromRule(Node *n)
|
|
#else
|
|
int MR_offsetFromRule(n)
|
|
Node *n;
|
|
#endif
|
|
{
|
|
Junction *j;
|
|
int offset=(-1);
|
|
|
|
for (j=SynDiag; j != NULL; j=(Junction *)j->p2) {
|
|
|
|
require (j->ntype == nJunction && j->jtype == RuleBlk,"Not a rule block");
|
|
|
|
if (n->file < j->file) {
|
|
return offset;
|
|
};
|
|
if (n->file == j->file) {
|
|
if (n->line < j->line) {
|
|
return (offset < 0) ? 0 : offset;
|
|
} else {
|
|
offset=n->line - j->line;
|
|
if (offset == 0) return 0;
|
|
};
|
|
};
|
|
};
|
|
return offset;
|
|
}
|
|
|
|
#define ruleNameMax 50
|
|
|
|
static char ruleNameStatic1[ruleNameMax];
|
|
static char ruleNameStatic2[ruleNameMax+10];
|
|
|
|
#ifdef __USE_PROTOS
|
|
char * MR_ruleNamePlusOffset(Node *n)
|
|
#else
|
|
char * MR_ruleNamePlusOffset(n)
|
|
Node *n;
|
|
#endif
|
|
{
|
|
int offset=MR_offsetFromRule(n);
|
|
|
|
strncpy(ruleNameStatic1,n->rname,ruleNameMax);
|
|
if (offset < 0) {
|
|
sprintf(ruleNameStatic2,"%s/?",ruleNameStatic1);
|
|
} else {
|
|
sprintf(ruleNameStatic2,"%s/%d",ruleNameStatic1,offset+1);
|
|
};
|
|
return ruleNameStatic2;
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
int MR_max_height_of_tree(Tree *t)
|
|
#else
|
|
int MR_max_height_of_tree(t)
|
|
Tree *t;
|
|
#endif
|
|
{
|
|
int h;
|
|
int height=0;
|
|
Tree *u;
|
|
|
|
if (t == NULL) return 0;
|
|
|
|
require (t->token != ALT && t->token != EpToken,"MR_max_height_of_tree ALT or EpToken");
|
|
|
|
for (u=t; u != NULL; u=u->right) {
|
|
h=MR_max_height_of_tree(u->down)+1;
|
|
if (h > height) height=h;
|
|
};
|
|
return height;
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
int MR_all_leaves_same_height(Tree *t,int depth)
|
|
#else
|
|
int MR_all_leaves_same_height(t,depth)
|
|
Tree *t;
|
|
int depth;
|
|
#endif
|
|
{
|
|
if (t == NULL) {
|
|
return (depth==0);
|
|
};
|
|
|
|
require (t->token != ALT && t->token != EpToken,"MR_all_leaves_same_height ALT or EpToken");
|
|
|
|
if (depth == 0) {
|
|
return 0;
|
|
} else {
|
|
if ( ! MR_all_leaves_same_height(t->down,depth-1)) {
|
|
return 0;
|
|
};
|
|
if (t->right == NULL) {
|
|
return 1;
|
|
} else {
|
|
return MR_all_leaves_same_height(t->right,depth);
|
|
};
|
|
};
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_projectTreeOntoSet(Tree *tree,int ck,set *ckset)
|
|
#else
|
|
void MR_projectTreeOntoSet(tree,ck,ckset)
|
|
Tree *tree;
|
|
int ck;
|
|
set *ckset;
|
|
#endif
|
|
{
|
|
if (tree == NULL) return;
|
|
|
|
require(tree->token != EpToken,"MR_projectTreeOntoSet: EpToken unexpected\n");
|
|
|
|
MR_projectTreeOntoSet(tree->right,ck,ckset);
|
|
if (tree->token == ALT) {
|
|
MR_projectTreeOntoSet(tree->down,ck,ckset);
|
|
} else {
|
|
if (ck > 1) {
|
|
MR_projectTreeOntoSet(tree->down,ck-1,ckset);
|
|
} else {
|
|
set_orel(tree->token,ckset);
|
|
};
|
|
};
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
int MR_comparePredicates(Predicate *a,Predicate *b)
|
|
#else
|
|
int MR_comparePredicates(a,b)
|
|
Predicate *a;
|
|
Predicate *b;
|
|
#endif
|
|
{
|
|
Predicate *p;
|
|
Predicate *q;
|
|
|
|
if (a == b) return 1;
|
|
if (a == NULL || b == NULL ) return 0;
|
|
if (a->down == NULL && b->down == NULL) {
|
|
|
|
/* predicate->invert can be set only in the predEntry predicates */
|
|
/* thus they are only visible after the predEntry predicates have been "unfolded" */
|
|
|
|
int sameSource=(a->source == b->source);
|
|
int sameInvert= 1 & (1 +a->inverted + b->inverted +
|
|
a->source->inverted + b->source->inverted);
|
|
int samePredEntry=(a->source->predEntry != NULL
|
|
&& b->source->predEntry != NULL
|
|
&& a->source->predEntry == b->source->predEntry);
|
|
if (sameInvert && (sameSource || samePredEntry)) {
|
|
if (MR_identicalContext(a,b)) {
|
|
return 1;
|
|
};
|
|
};
|
|
return 0;
|
|
};
|
|
if (a->down == NULL || b->down == NULL) return 0;
|
|
if (a->expr != b->expr) return 0;
|
|
|
|
for (p=a->down; p != NULL; p=p->right) {
|
|
for (q=b->down; q != NULL; q=q->right) {
|
|
if (MR_comparePredicates(p,q)) goto NEXT_P;
|
|
};
|
|
return 0;
|
|
NEXT_P:
|
|
continue;
|
|
};
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* action->inverted can be set only when a predicate symbol appears in
|
|
* a rule: "rule : <<!XXX>>? X". It cannot be set under any
|
|
* other circumstances. In particular it cannot be set by
|
|
* "#pred NotA !A" or by "#pred Nota <<!A>>?". The first case
|
|
* creates a predEntry and the predicate expression of that predEntry
|
|
* has inverted set. In the second case, the code for handling "!"
|
|
* is only present in buildAction, which is not called by the #pred
|
|
* semantic routines, only when a <<...>>? is recognized as part of
|
|
* a rule definition.
|
|
*
|
|
* predicate->inverted can only be set by a predicate created by a #pred
|
|
* expression, such as "#pred NotA !A" or "#pred NotXY ! (X && Y) or
|
|
* "#pred XbarY !(X && Y)". In particular, it cannot be set by any
|
|
* predicate expression occurring under any other circumstances.
|
|
* The #pred predicate expressions are stored with in predEntry->pred
|
|
* and do not normally appear anywhere else until the predicates are
|
|
* "unfolded" in order to recognize redundancies, conflicts, and
|
|
* tautologies.
|
|
*
|
|
* The unfold routine expands all references to #pred expressions.
|
|
*
|
|
* The simplifyInvert goes through and propagates the invert bit so that
|
|
* all OR and AND nodes are un-inverted.
|
|
*
|
|
* Note that !(A and B) => (!A or !B)
|
|
* !(A or B) => (!A and !B)
|
|
*
|
|
* MR_unfold() is called to expand predicate symbols by replacing predicates
|
|
* that reference predicate entries with the copies of the predicate entries.
|
|
* Each reference receives a duplicate of the original. This is necessary
|
|
* because the next phase involves simplification and removal of redundant
|
|
* predicate nodes. Anyway, the point I'm making is that predicate->invert
|
|
* should not be set in any predicate until it has been expanded.
|
|
*
|
|
* This is a recursive structure, but there is no need for "recursive expansion"
|
|
* by which I mean a predicate symbol refers to other predicate symbols which
|
|
* must also be expanded.
|
|
*
|
|
* Recursive expansion is *not* performed by this routine because it is not
|
|
* necessary. Expansion of references is performed by predPrimary when
|
|
* a new predicate symbol is created by referring to others in the pred expr.
|
|
*/
|
|
|
|
#ifdef __USE_PROTOS
|
|
Predicate *MR_unfold(Predicate *pred)
|
|
#else
|
|
Predicate *MR_unfold(pred)
|
|
Predicate *pred;
|
|
#endif
|
|
{
|
|
Predicate *result;
|
|
|
|
if (pred == NULL) return NULL;
|
|
|
|
pred->right=MR_unfold(pred->right);
|
|
|
|
if (pred->down == NULL) {
|
|
if (pred->source->predEntry != NULL) {
|
|
if (pred->source->predEntry->pred == NULL) {
|
|
; /* do nothing */ /* a reference to a literal #pred (perhaps with "!" */
|
|
} else {
|
|
result=predicate_dup_without_context(pred->source->predEntry->pred);
|
|
if (pred->inverted) {
|
|
result->inverted=!result->inverted;
|
|
};
|
|
if (pred->source->inverted) {
|
|
result->inverted=!result->inverted;
|
|
};
|
|
result->right=pred->right;
|
|
pred->right=NULL;
|
|
predicate_free(pred);
|
|
/*** result=MR_unfold(result); *** not necessary */ /* recursive expansion */
|
|
return result;
|
|
};
|
|
} else {
|
|
; /* do nothing */ /* an inline literal predicate */
|
|
};
|
|
} else {
|
|
pred->down=MR_unfold(pred->down);
|
|
};
|
|
return pred;
|
|
}
|
|
|
|
/* this should be called immediately after MR_unfold() and
|
|
at no other times
|
|
*/
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_simplifyInverted(Predicate *pred,int inverted)
|
|
#else
|
|
void MR_simplifyInverted(pred,inverted)
|
|
Predicate *pred;
|
|
int inverted;
|
|
#endif
|
|
{
|
|
int newInverted;
|
|
|
|
if (pred == NULL) return;
|
|
|
|
MR_simplifyInverted(pred->right,inverted);
|
|
|
|
newInverted= 1 & (inverted + pred->inverted);
|
|
|
|
if (pred->down == NULL) {
|
|
pred->inverted=newInverted;
|
|
} else {
|
|
if (newInverted != 0) {
|
|
if (pred->expr == PRED_AND_LIST) {
|
|
pred->expr=PRED_OR_LIST;
|
|
} else {
|
|
pred->expr=PRED_AND_LIST;
|
|
};
|
|
};
|
|
pred->inverted=0;
|
|
MR_simplifyInverted(pred->down,newInverted);
|
|
};
|
|
}
|
|
|
|
/* only remove it from AND and OR nodes, not leaves */
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_clearPredEntry(Predicate *p)
|
|
#else
|
|
void MR_clearPredEntry(p)
|
|
Predicate *p;
|
|
#endif
|
|
{
|
|
if (p == NULL) return;
|
|
MR_clearPredEntry(p->down);
|
|
MR_clearPredEntry(p->right);
|
|
if (p->down != NULL) p->predEntry=NULL;
|
|
}
|
|
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_orphanRules(FILE *f)
|
|
#else
|
|
void MR_orphanRules(f)
|
|
FILE *f;
|
|
#endif
|
|
{
|
|
set a;
|
|
Junction *p;
|
|
unsigned e;
|
|
RuleEntry *re;
|
|
|
|
a=empty;
|
|
|
|
if (! InfoO) return;
|
|
|
|
for (p=SynDiag; p!=NULL; p = (Junction *)p->p2) {
|
|
if ( (Junction *) (p->end)->p1 == NULL) {
|
|
re=(RuleEntry *) hash_get(Rname,p->rname);
|
|
require (re != NULL,"RuleEntry == NULL");
|
|
set_orel(re->rulenum, &a);
|
|
}
|
|
}
|
|
|
|
if (set_deg(a) > 1) {
|
|
fprintf(f,"note: Start rules: {");
|
|
for (; !set_nil(a); set_rm(e,a)) {
|
|
e=set_int(a);
|
|
fprintf(f," %s",RulePtr[e]->rname);
|
|
};
|
|
fprintf(f," }\n");
|
|
};
|
|
set_free( a );
|
|
}
|
|
|
|
/* merge (X Y) and (X) to create (X) */
|
|
|
|
static int *mergeChain;
|
|
static Tree *mergeTree;
|
|
|
|
#ifdef __USE_PROTOS
|
|
Tree *MR_merge_tree_contexts_client(Tree *t,int chain[])
|
|
#else
|
|
Tree *MR_merge_tree_contexts_client(t,chain)
|
|
Tree *t;
|
|
int chain[];
|
|
#endif
|
|
{
|
|
if (t == NULL) return NULL;
|
|
if (chain[0] == 0) {
|
|
Tree *u=t->right;
|
|
t->right=NULL;
|
|
Tfree(t);
|
|
return MR_merge_tree_contexts_client(u,&chain[0]);
|
|
}
|
|
if (chain[0] == t->token) {
|
|
t->down=MR_merge_tree_contexts_client(t->down,&chain[1]);
|
|
};
|
|
t->right=MR_merge_tree_contexts_client(t->right,&chain[0]);
|
|
return t;
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_iterateOverTreeContexts(Tree *t,int chain[])
|
|
#else
|
|
void MR_iterateOverTreeContexts(t,chain)
|
|
Tree *t;
|
|
int chain[];
|
|
#endif
|
|
{
|
|
if (t == NULL) return;
|
|
chain[0]=t->token;
|
|
if (t->down != NULL) {
|
|
MR_iterateOverTreeContexts(t->down,&chain[1]);
|
|
} else {
|
|
MR_merge_tree_contexts_client(mergeTree,mergeChain);
|
|
};
|
|
MR_iterateOverTreeContexts(t->right,&chain[0]);
|
|
chain[0]=0;
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
Tree *MR_merge_tree_contexts(Tree *t)
|
|
#else
|
|
Tree *MR_merge_tree_contexts(t)
|
|
Tree *t;
|
|
#endif
|
|
{
|
|
int h=MR_max_height_of_tree(t);
|
|
|
|
mergeTree=t;
|
|
mergeChain=(int *) calloc(h+1,sizeof(int));
|
|
require (mergeChain != NULL,"MR_merge_tree_contexts: can't alloc chain");
|
|
MR_iterateOverTreeContexts(t,mergeChain);
|
|
t=tshrink(t);
|
|
t=tflatten(t);
|
|
t=tleft_factor(t);
|
|
free ( (char *) mergeChain);
|
|
mergeChain=NULL;
|
|
return t;
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
Tree *MR_compute_pred_tree_context(Predicate *p)
|
|
#else
|
|
Tree *MR_compute_pred_tree_context(p)
|
|
Predicate *p;
|
|
#endif
|
|
{
|
|
Tree *t;
|
|
|
|
t=MR_compute_pred_tree_ctxXX(p);
|
|
MR_merge_tree_contexts(t);
|
|
return t;
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_guardPred_plainSet(ActionNode *anode,Predicate *pred)
|
|
#else
|
|
void MR_guardPred_plainSet(anode,pred)
|
|
ActionNode *anode;
|
|
Predicate *pred;
|
|
#endif
|
|
{
|
|
Junction *j;
|
|
Predicate *workPred;
|
|
set maskSet;
|
|
|
|
maskSet=empty;
|
|
|
|
if (!MRhoisting) return;
|
|
|
|
/* it doesn't really matter whether the predicate has
|
|
depth k=1 or k>1 because we're not really looking
|
|
at the predicate itself, just the stuff "behind"
|
|
the predicate.
|
|
*/
|
|
|
|
/* shouldn't have to worry about REACHing off the end
|
|
of the rule containing the predicate because the
|
|
Rule->end->halt should have been set already by the
|
|
the code which handles RuleRef nodes.
|
|
|
|
We don't want to REACH off the end of the rule because
|
|
this would give the "global" follow context rather than
|
|
the "local" context.
|
|
|
|
r1a : (A)? => <<p>>? r2 (A|B)
|
|
r1b : (A)? => <<p>>? r2 (A|C)
|
|
r2 : ();
|
|
|
|
For r1a we want follow of predicate = {A B}
|
|
we want plainSet = {B}
|
|
For r1b we want follow of predicate = {A C}
|
|
we want plainSet = {C}
|
|
*/
|
|
|
|
require (anode->next->ntype == nJunction,"MR_guardpred_plainSet not Junction");
|
|
j=(Junction *)(anode->next);
|
|
|
|
workPred=predicate_dup_without_context(pred);
|
|
workPred->k=1;
|
|
workPred->scontext[1]=MR_First(1,j, &(workPred->completionSet) );
|
|
MR_complete_predicates(1,workPred);
|
|
if (pred->k == 1) {
|
|
maskSet=pred->scontext[1];
|
|
} else {
|
|
MR_projectTreeOntoSet(pred->tcontext,1,&maskSet);
|
|
}
|
|
pred->plainSet=set_dif(workPred->scontext[1],maskSet);
|
|
predicate_free(workPred);
|
|
}
|
|
|
|
/*******************************************************************************/
|
|
|
|
static Tree * suppressTree;
|
|
static int * suppressChain; /* element 0 not used */
|
|
static set * suppressSets;
|
|
static Node * suppressNode;
|
|
static int suppressChainLength;
|
|
int MR_SuppressSearch=0;
|
|
static int suppressSucceeded;
|
|
static Predicate * suppressPredicate;
|
|
|
|
#ifdef __USE_PROTOS
|
|
int MR_isChain(Tree *t)
|
|
#else
|
|
int MR_isChain(t)
|
|
Tree *t;
|
|
#endif
|
|
{
|
|
Tree *u;
|
|
|
|
for (u=t; u != NULL; u=u->down) {
|
|
if (u->right != NULL) return 0;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
int MR_suppressK_client(Tree *tree,int tokensInChain[])
|
|
#else
|
|
int MR_suppressK_client(tree,tokensInChain)
|
|
Tree *tree;
|
|
int tokensInChain[];
|
|
#endif
|
|
{
|
|
int i;
|
|
set *save_fset;
|
|
int save_ConstrainSearch;
|
|
set incomplete;
|
|
Tree *t;
|
|
|
|
suppressSucceeded=0; /* volatile */
|
|
|
|
if (suppressSets == NULL) {
|
|
suppressSets=(set *) calloc (CLL_k+1,sizeof(set));
|
|
require (suppressSets != NULL,"MR_suppressK_client: suppressSets alloc");
|
|
};
|
|
|
|
for (suppressChainLength=1;
|
|
tokensInChain[suppressChainLength+1] != 0;
|
|
suppressChainLength++) {};
|
|
|
|
require (suppressChainLength != 0,"MR_suppressK_client: chain empty");
|
|
|
|
for (i=1 ; i <= suppressChainLength ; i++) {
|
|
set_clr(suppressSets[i]);
|
|
set_orel( (unsigned) tokensInChain[i],
|
|
&suppressSets[i]);
|
|
};
|
|
|
|
save_fset=fset;
|
|
save_ConstrainSearch=ConstrainSearch;
|
|
|
|
fset=suppressSets;
|
|
|
|
MR_SuppressSearch=1;
|
|
MR_AmbSourceSearch=1;
|
|
MR_MaintainBackTrace=1;
|
|
ConstrainSearch=1;
|
|
|
|
maxk = suppressChainLength;
|
|
|
|
incomplete=empty;
|
|
t=NULL;
|
|
|
|
/*** constrain = &(fset[1]); ***/
|
|
|
|
MR_setConstrainPointer(&(fset[1])); /* MR18 */
|
|
|
|
MR_pointerStackReset(&MR_BackTraceStack);
|
|
|
|
TRAV(suppressNode,maxk,&incomplete,t);
|
|
|
|
Tfree(t);
|
|
|
|
require (set_nil(incomplete),"MR_suppressK_client TRAV incomplete");
|
|
require (MR_BackTraceStack.count == 0,
|
|
"MR_suppressK_client: MR_BackTraceStack.count != 0");
|
|
set_free(incomplete);
|
|
|
|
ConstrainSearch=save_ConstrainSearch;
|
|
fset=save_fset;
|
|
|
|
MR_AmbSourceSearch=0;
|
|
MR_MaintainBackTrace=0;
|
|
MR_SuppressSearch=0;
|
|
return suppressSucceeded;
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
Tree * MR_iterateOverTreeSuppressK(Tree *t,int chain[])
|
|
#else
|
|
Tree * MR_iterateOverTreeSuppressK(t,chain)
|
|
Tree *t;
|
|
int chain[];
|
|
#endif
|
|
{
|
|
if (t == NULL) return NULL;
|
|
t->right=MR_iterateOverTreeSuppressK(t->right,&chain[0]);
|
|
chain[0]=t->token;
|
|
if (t->down != NULL) {
|
|
t->down=MR_iterateOverTreeSuppressK(t->down,&chain[1]);
|
|
if (t->down == NULL) {
|
|
Tree *u=t->right;
|
|
t->right=NULL;
|
|
Tfree(t);
|
|
chain[0]=0;
|
|
return u;
|
|
};
|
|
} else {
|
|
MR_suppressK_client(suppressTree,suppressChain);
|
|
if (suppressSucceeded) {
|
|
Tree *u=t->right;
|
|
t->right=NULL;
|
|
Tfree(t);
|
|
chain[0]=0;
|
|
return u;
|
|
};
|
|
};
|
|
chain[0]=0;
|
|
return t;
|
|
}
|
|
|
|
/* @@@ */
|
|
|
|
#ifdef __USE_PROTOS
|
|
Predicate * MR_suppressK(Node *j,Predicate *p)
|
|
#else
|
|
Predicate * MR_suppressK(j,p)
|
|
Node *j;
|
|
Predicate *p;
|
|
#endif
|
|
{
|
|
Predicate *result;
|
|
int guardPred=0;
|
|
int ampersandPred=0;
|
|
Node *nodePrime;
|
|
|
|
if (! MRhoistingk) {
|
|
return p;
|
|
}
|
|
|
|
if (! MRhoisting) return p;
|
|
if (CLL_k == 1) return p;
|
|
|
|
if (suppressChain == NULL) {
|
|
suppressChain=(int *) calloc(CLL_k+2,sizeof(int));
|
|
require (suppressChain != NULL,"MR_suppressK: can't allocate chain");
|
|
}
|
|
|
|
if (p == NULL) return NULL;
|
|
|
|
if (j->ntype == nJunction) {
|
|
nodePrime=(Node *) MR_junctionWithoutP2( (Junction *) j);
|
|
} else {
|
|
nodePrime=j;
|
|
};
|
|
|
|
p->down=MR_suppressK(j,p->down);
|
|
p->right=MR_suppressK(j,p->right);
|
|
if (p->down != NULL) {
|
|
result=p;
|
|
goto EXIT;
|
|
};
|
|
if (p->k == 1) {
|
|
result=p;
|
|
goto EXIT;
|
|
};
|
|
|
|
if (p->source != NULL) {
|
|
if (p->source->guardpred != NULL) guardPred=1;
|
|
if (p->source->ampersandPred != NULL) ampersandPred=1;
|
|
}
|
|
|
|
suppressPredicate=p;
|
|
suppressNode=nodePrime; /* was j*/
|
|
|
|
suppressTree=p->tcontext;
|
|
|
|
if (guardPred || ampersandPred) {
|
|
p->tcontext=MR_iterateOverTreeSuppressK(suppressTree,&suppressChain[1]);
|
|
if (p->tcontext == NULL) {
|
|
predicate_free(p);
|
|
result=NULL;
|
|
goto EXIT;
|
|
};
|
|
} else {
|
|
if (MR_isChain(p->tcontext)) {
|
|
p->tcontext=MR_iterateOverTreeSuppressK(suppressTree,&suppressChain[1]);
|
|
if (p->tcontext == NULL) {
|
|
predicate_free(p);
|
|
result=NULL;
|
|
goto EXIT;
|
|
};
|
|
}
|
|
}
|
|
result=p;
|
|
EXIT:
|
|
return result;
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_suppressSearchReport(void)
|
|
#else
|
|
void MR_suppressSearchReport()
|
|
#endif
|
|
{
|
|
int i;
|
|
Node *p;
|
|
TokNode *tn;
|
|
int depth;
|
|
set setAnd;
|
|
|
|
/* number of tokens in back trace stack matches length of chain */
|
|
|
|
depth=0;
|
|
for (i=0; i < MR_BackTraceStack.count ; i++) {
|
|
p=(Node *) MR_BackTraceStack.data[i];
|
|
if (p->ntype == nToken) depth++;
|
|
};
|
|
|
|
require (depth == suppressChainLength,"depth > suppressChainLength");
|
|
|
|
/* token codes match chain */
|
|
|
|
depth=0;
|
|
for (i=0; i < MR_BackTraceStack.count ; i++) {
|
|
p=(Node *) MR_BackTraceStack.data[i];
|
|
if (p->ntype != nToken) continue;
|
|
tn=(TokNode *) p;
|
|
depth++;
|
|
if (set_nil(tn->tset)) {
|
|
require(set_el( (unsigned) tn->token,fset[depth]),
|
|
"MR_suppressSearchReport: no match to #token in chain");
|
|
} else {
|
|
setAnd=set_and(fset[depth],tn->tset);
|
|
require(!set_nil(setAnd),
|
|
"MR_suppressSearchReport: no match to #token set in chain");
|
|
set_free(setAnd);
|
|
};
|
|
};
|
|
|
|
/* have a match - now remove it from the predicate */
|
|
|
|
suppressSucceeded=1;
|
|
|
|
if (suppressSucceeded) {
|
|
fprintf(output,"\n");
|
|
fprintf(output,"#if 0\n");
|
|
fprintf(output,"\n");
|
|
fprintf(output,"Part (or all) of predicate with depth > 1 suppressed by ");
|
|
fprintf(output,"alternative without predicate\n\n");
|
|
MR_dumpPred(suppressPredicate,1);
|
|
fprintf(output,"The token sequence which is suppressed:");
|
|
fprintf(output," (");
|
|
for (i=1; i <= suppressChainLength; i++) {
|
|
fprintf(output," %s",TerminalString(suppressChain[i]));
|
|
};
|
|
fprintf(output," )\n");
|
|
fprintf(output,"The sequence of references which generate that sequence of tokens:\n\n");
|
|
|
|
MR_backTraceDumpItemReset();
|
|
|
|
for (i=0; i < MR_BackTraceStack.count ; i++) {
|
|
MR_backTraceDumpItem(output,0,(Node *) MR_BackTraceStack.data[i]);
|
|
};
|
|
fprintf(output,"\n");
|
|
fprintf(output,"#endif\n");
|
|
}
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_markCompromisedRule(Node *n)
|
|
#else
|
|
void MR_markCompromisedRule(n)
|
|
Node *n;
|
|
#endif
|
|
{
|
|
RuleEntry *q;
|
|
Node *mark=NULL;
|
|
Junction *j;
|
|
|
|
if (n->ntype == nRuleRef) {
|
|
mark=(Node *) MR_ruleReferenced( (RuleRefNode *) n);
|
|
} else if (n->ntype == nToken) {
|
|
mark=n;
|
|
} else if (n->ntype == nJunction) {
|
|
j=(Junction *)n;
|
|
switch (j->jtype) {
|
|
case aOptBlk:
|
|
case aLoopBlk:
|
|
case RuleBlk:
|
|
case EndRule:
|
|
case aPlusBlk:
|
|
case aLoopBegin:
|
|
mark=n;
|
|
break;
|
|
default:
|
|
break;
|
|
};
|
|
}
|
|
|
|
if (mark == NULL) return;
|
|
|
|
require (RulePtr != NULL,"RulePtr not initialized");
|
|
|
|
q = (RuleEntry *) hash_get(Rname,mark->rname);
|
|
require (q != NULL,"RuleEntry not found");
|
|
set_orel(q->rulenum,&MR_CompromisedRules);
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_alphaBetaTraceReport(void)
|
|
#else
|
|
void MR_alphaBetaTraceReport()
|
|
#endif
|
|
{
|
|
int i;
|
|
|
|
if (! AlphaBetaTrace) return;
|
|
|
|
MR_AlphaBetaMessageCount++;
|
|
|
|
fprintf(output,"\n");
|
|
fprintf(output,"#if 0\n");
|
|
fprintf(output,"\n");
|
|
fprintf(output,"Trace of references leading to attempt to compute the follow set of\n");
|
|
fprintf(output,"alpha in an \"(alpha)? beta\" block. It is not possible for antlr to\n");
|
|
fprintf(output,"compute this follow set because it is not known what part of beta has\n");
|
|
fprintf(output,"already been matched by alpha and what part remains to be matched.\n");
|
|
fprintf(output,"\n");
|
|
fprintf(output,"Rules which make use of the incorrect follow set will also be incorrect\n");
|
|
fprintf(output,"\n");
|
|
|
|
MR_backTraceDumpItemReset();
|
|
|
|
for (i=0; i < MR_BackTraceStack.count ; i++) {
|
|
MR_backTraceDumpItem(output,0,(Node *) MR_BackTraceStack.data[i]);
|
|
if (i < MR_BackTraceStack.count-1) {
|
|
MR_markCompromisedRule( (Node *) MR_BackTraceStack.data[i]);
|
|
};
|
|
};
|
|
fprintf(output,"\n");
|
|
fprintf(output,"#endif\n");
|
|
}
|
|
|
|
#ifdef __USE_PROTOS
|
|
void MR_dumpRuleSet(set s)
|
|
#else
|
|
void MR_dumpRuleSet(s)
|
|
set s;
|
|
#endif
|
|
{
|
|
unsigned *cursor;
|
|
unsigned *origin=set_pdq(s);
|
|
|
|
require(origin != NULL,"set_pdq failed");
|
|
|
|
if (RulePtr == NULL) {
|
|
fprintf(stderr,"RulePtr[] not yet initialized");
|
|
} else {
|
|
for (cursor=origin; *cursor != nil ; cursor++) {
|
|
/**** if (cursor != origin) fprintf(stderr,","); ****/
|
|
fprintf(stderr," %s",RulePtr[*cursor]->rname);
|
|
fprintf(stderr,"\n");
|
|
};
|
|
free( (char *) origin);
|
|
};
|
|
}
|