#include <assert.h>
#include <iostream>
#include "fsmgraph.h"
#include "mergesort.h"
#include "parsedata.h"
using
std::cerr;
using
std::endl;
StateAp *FsmAp::addState()
{
StateAp *state =
new
StateAp();
if
( misfitAccounting ) {
misfitList.append( state );
}
else
{
stateList.append( state );
}
return
state;
}
void
FsmAp::concatFsm( Key *str,
int
len )
{
StateAp *last = addState();
setStartState( last );
for
(
int
i = 0; i < len; i++ ) {
StateAp *newState = addState();
attachNewTrans( last, newState, str[i], str[i] );
last = newState;
}
setFinState( last );
}
void
FsmAp::concatFsmCI( Key *str,
int
len )
{
StateAp *last = addState();
setStartState( last );
for
(
int
i = 0; i < len; i++ ) {
StateAp *newState = addState();
KeySet keySet;
if
( str[i].isLower() )
keySet.insert( str[i].toUpper() );
if
( str[i].isUpper() )
keySet.insert( str[i].toLower() );
keySet.insert( str[i] );
for
(
int
i = 0; i < keySet.length(); i++ )
attachNewTrans( last, newState, keySet[i], keySet[i] );
last = newState;
}
setFinState( last );
}
void
FsmAp::concatFsm( Key chr )
{
setStartState( addState() );
StateAp *end = addState();
setFinState( end );
attachNewTrans( startState, end, chr, chr );
}
void
FsmAp::orFsm( Key *set,
int
len )
{
setStartState( addState() );
StateAp *end = addState();
setFinState( end );
for
(
int
i = 1; i < len; i++ )
assert
( set[i-1] < set[i] );
for
(
int
i = 0; i < len; i++ )
attachNewTrans( startState, end, set[i], set[i] );
}
void
FsmAp::rangeFsm( Key low, Key high )
{
setStartState( addState() );
StateAp *end = addState();
setFinState( end );
attachNewTrans( startState, end, low, high );
}
void
FsmAp::rangeStarFsm( Key low, Key high)
{
setStartState( addState() );
setFinState( startState );
attachNewTrans( startState, startState, low, high );
}
void
FsmAp::lambdaFsm( )
{
setStartState( addState() );
setFinState( startState );
}
void
FsmAp::emptyFsm( )
{
setStartState( addState() );
}
void
FsmAp::transferOutData( StateAp *destState, StateAp *srcState )
{
for
( TransList::Iter trans = destState->outList; trans.lte(); trans++ ) {
if
( trans->toState != 0 ) {
trans->actionTable.setActions( srcState->outActionTable );
trans->priorTable.setPriors( srcState->outPriorTable );
}
}
}
void
FsmAp::starOp( )
{
MergeData md;
setMisfitAccounting(
true
);
StateAp *prevStartState = startState;
unsetStartState();
setStartState( addState() );
mergeStates( md, startState, prevStartState );
for
( StateSet::Iter st = finStateSet; st.lte(); st++ ) {
if
( *st != startState )
mergeStatesLeaving( md, *st, startState );
}
if
( startState->isFinState() )
mergeStatesLeaving( md, startState, startState );
setFinState( startState );
fillInStates( md );
removeMisfits();
setMisfitAccounting(
false
);
}
void
FsmAp::repeatOp(
int
times )
{
assert
( times > 0 );
if
( times == 1 )
return
;
FsmAp *copyFrom =
new
FsmAp( *
this
);
for
(
int
i = 1; i < times-1; i++ ) {
FsmAp *dup =
new
FsmAp( *copyFrom );
doConcat( dup, 0,
false
);
}
doConcat( copyFrom, 0,
false
);
}
void
FsmAp::optionalRepeatOp(
int
times )
{
assert
( times > 0 );
if
( times == 1 ) {
setFinState( startState );
return
;
}
FsmAp *copyFrom =
new
FsmAp( *
this
);
StateSet lastFinSet( finStateSet );
setFinState( startState );
for
(
int
i = 1; i < times-1; i++ ) {
FsmAp *dup =
new
FsmAp( *copyFrom );
dup->setFinBits( STB_GRAPH2 );
doConcat( dup, &lastFinSet,
true
);
lastFinSet.empty();
for
(
int
i = 0; i < finStateSet.length(); i++ ) {
StateAp *fs = finStateSet[i];
if
( fs->stateBits & STB_GRAPH2 ) {
lastFinSet.insert( fs );
fs->stateBits &= ~STB_GRAPH2;
}
}
}
doConcat( copyFrom, &lastFinSet,
true
);
}
void
FsmAp::doConcat( FsmAp *other, StateSet *fromStates,
bool
optional )
{
StateSet finStateSetCopy, startStateSet;
MergeData md;
setMisfitAccounting(
true
);
other->setMisfitAccounting(
true
);
StateAp *otherStartState = other->startState;
other->unsetStartState();
copyInEntryPoints( other );
other->entryPoints.empty();
stateList.append( other->stateList );
misfitList.append( other->misfitList );
if
( fromStates == 0 ) {
finStateSetCopy = finStateSet;
fromStates = &finStateSetCopy;
}
if
( !optional )
unsetAllFinStates();
finStateSet.insert( other->finStateSet );
delete
other;
for
(
int
i = 0; i < fromStates->length(); i++ ) {
StateAp *state = fromStates->data[i];
mergeStatesLeaving( md, state, otherStartState );
if
( ! state->isFinState() )
clearOutData( state );
}
fillInStates( md );
removeMisfits();
setMisfitAccounting(
false
);
}
void
FsmAp::concatOp( FsmAp *other )
{
doConcat( other, 0,
false
);
}
void
FsmAp::doOr( FsmAp *other )
{
MergeData md;
StateSet startStateSet;
startStateSet.insert( startState );
startStateSet.insert( other->startState );
unsetStartState();
other->unsetStartState();
copyInEntryPoints( other );
other->entryPoints.empty();
stateList.append( other->stateList );
misfitList.append( other->misfitList );
finStateSet.insert(other->finStateSet);
other->finStateSet.empty();
delete
other;
setStartState( addState() );
mergeStates( md, startState, startStateSet.data, startStateSet.length() );
fillInStates( md );
}
void
FsmAp::unionOp( FsmAp *other )
{
setMisfitAccounting(
true
);
other->setMisfitAccounting(
true
);
doOr( other );
removeMisfits();
setMisfitAccounting(
false
);
}
void
FsmAp::intersectOp( FsmAp *other )
{
setMisfitAccounting(
true
);
other->setMisfitAccounting(
true
);
setFinBits( STB_GRAPH1 );
other->setFinBits( STB_GRAPH2 );
doOr( other );
unsetIncompleteFinals();
removeMisfits();
setMisfitAccounting(
false
);
removeDeadEndStates();
}
void
FsmAp::subtractOp( FsmAp *other )
{
setMisfitAccounting(
true
);
other->setMisfitAccounting(
true
);
other->setFinBits( STB_GRAPH1 );
doOr( other );
unsetKilledFinals();
removeMisfits();
setMisfitAccounting(
false
);
removeDeadEndStates();
}
bool
FsmAp::inEptVect( EptVect *eptVect, StateAp *state )
{
if
( eptVect != 0 ) {
for
(
int
i = 0; i < eptVect->length(); i++ ) {
if
( eptVect->data[i].targ == state )
return
true
;
}
}
return
false
;
}
void
FsmAp::epsilonFillEptVectFrom( StateAp *root, StateAp *from,
bool
parentLeaving )
{
for
( EpsilonTrans::Iter ep = from->epsilonTrans; ep.lte(); ep++ ) {
EntryMapEl *enLow, *enHigh;
if
( entryPoints.findMulti( *ep, enLow, enHigh ) ) {
for
( EntryMapEl *en = enLow; en <= enHigh; en++ ) {
StateAp *targ = en->value;
if
( targ != from && !inEptVect(root->eptVect, targ) ) {
if
( root->eptVect == 0 )
root->eptVect =
new
EptVect();
bool
leaving = parentLeaving ||
root->owningGraph != targ->owningGraph;
root->eptVect->append( EptVectEl(targ, leaving) );
epsilonFillEptVectFrom( root, targ, leaving );
}
}
}
}
}
void
FsmAp::shadowReadWriteStates( MergeData &md )
{
for
( StateList::Iter st = stateList; st.lte(); st++ )
st->isolatedShadow = 0;
for
( StateList::Iter st = stateList; st.lte(); st++ ) {
if
( st->eptVect != 0 ) {
for
( EptVect::Iter ept = *st->eptVect; ept.lte(); ept++ ) {
StateAp *targ = ept->targ;
if
( targ->eptVect != 0 ) {
if
( targ->isolatedShadow == 0 ) {
StateAp *shadow = addState();
mergeStates( md, shadow, targ );
targ->isolatedShadow = shadow;
}
ept->targ = targ->isolatedShadow;
}
}
}
}
}
void
FsmAp::resolveEpsilonTrans( MergeData &md )
{
for
( StateList::Iter st = stateList; st.lte(); st++ )
epsilonFillEptVectFrom( st, st,
false
);
shadowReadWriteStates( md );
for
( StateList::Iter st = stateList; st.lte(); st++ ) {
if
( st->eptVect != 0 ) {
for
( EptVect::Iter ept = *st->eptVect; ept.lte(); ept++ ) {
if
( ept->leaving )
mergeStatesLeaving( md, st, ept->targ );
else
mergeStates( md, st, ept->targ );
}
delete
st->eptVect;
st->eptVect = 0;
}
st->epsilonTrans.empty();
}
}
void
FsmAp::epsilonOp()
{
MergeData md;
setMisfitAccounting(
true
);
for
( StateList::Iter st = stateList; st.lte(); st++ )
st->owningGraph = 0;
resolveEpsilonTrans( md );
fillInStates( md );
removeMisfits();
setMisfitAccounting(
false
);
}
void
FsmAp::joinOp(
int
startId,
int
finalId, FsmAp **others,
int
numOthers )
{
MergeData md;
for
( StateList::Iter st = stateList; st.lte(); st++ )
st->owningGraph = 1;
for
(
int
m = 0; m < numOthers; m++ ) {
for
( StateList::Iter st = others[m]->stateList; st.lte(); st++ )
st->owningGraph = 2+m;
}
unsetStartState();
for
(
int
m = 0; m < numOthers; m++ )
others[m]->unsetStartState();
for
(
int
m = 0; m < numOthers; m++ ) {
copyInEntryPoints( others[m] );
others[m]->entryPoints.empty();
stateList.append( others[m]->stateList );
assert
( others[m]->misfitList.length() == 0 );
finStateSet.insert( others[m]->finStateSet );
others[m]->finStateSet.empty();
delete
others[m];
}
EntryMapEl *enLow = 0, *enHigh = 0;
bool
findRes = entryPoints.findMulti( startId, enLow, enHigh );
if
( ! findRes ) {
setStartState( addState() );
}
else
{
StateAp *newStart = addState();
setStartState( newStart );
newStart->owningGraph = 0;
StateSet stateSet;
for
( EntryMapEl *en = enLow; en <= enHigh; en++ )
stateSet.insert( en->value );
mergeStates( md, newStart, stateSet.data, stateSet.length() );
}
StateSet finStateSetCopy = finStateSet;
unsetAllFinStates();
if
( finalId >= 0 ) {
StateAp *finState = addState();
setFinState( finState );
setEntry( finalId, finState );
finState->owningGraph = 0;
}
resolveEpsilonTrans( md );
for
( StateSet::Iter st = finStateSetCopy; st.lte(); st++ ) {
if
( !((*st)->stateBits & STB_ISFINAL) )
clearOutData( *st );
}
fillInStates( md );
removeUnreachableStates();
}
void
FsmAp::globOp( FsmAp **others,
int
numOthers )
{
for
(
int
m = 0; m < numOthers; m++ )
others[m]->unsetStartState();
for
(
int
m = 0; m < numOthers; m++ ) {
copyInEntryPoints( others[m] );
others[m]->entryPoints.empty();
stateList.append( others[m]->stateList );
assert
( others[m]->misfitList.length() == 0 );
finStateSet.insert( others[m]->finStateSet );
others[m]->finStateSet.empty();
delete
others[m];
}
}
void
FsmAp::deterministicEntry()
{
MergeData md;
setMisfitAccounting(
true
);
EntryMap prevEntry = entryPoints;
unsetAllEntryPoints();
for
(
int
enId = 0; enId < prevEntry.length(); ) {
int
highId = enId;
while
( highId < prevEntry.length() && prevEntry[enId].key == prevEntry[highId].key )
highId += 1;
int
numIds = highId - enId;
if
( numIds == 1 ) {
setEntry( prevEntry[enId].key, prevEntry[enId].value );
}
else
{
StateAp *newEntry = addState();
for
(
int
en = enId; en < highId; en++ )
mergeStates( md, newEntry, prevEntry[en].value );
setEntry( prevEntry[enId].key, newEntry );
}
enId += numIds;
}
removeMisfits();
setMisfitAccounting(
false
);
}
void
FsmAp::unsetKilledFinals()
{
StateSet fin( finStateSet );
for
(
int
s = 0; s < fin.length(); s++ ) {
StateAp *state = fin.data[s];
if
( state->stateBits & STB_GRAPH1 ) {
unsetFinState( state );
}
state->stateBits &= ~STB_GRAPH1;
}
}
void
FsmAp::unsetIncompleteFinals()
{
StateSet fin( finStateSet );
for
(
int
s = 0; s < fin.length(); s++ ) {
StateAp *state = fin.data[s];
if
( state->stateBits & STB_BOTH &&
(state->stateBits & STB_BOTH) != STB_BOTH )
{
unsetFinState( state );
}
state->stateBits &= ~STB_BOTH;
}
}
void
FsmAp::isolateStartState( )
{
MergeData md;
if
( isStartStateIsolated() )
return
;
setMisfitAccounting(
true
);
StateAp *prevStartState = startState;
unsetStartState();
setStartState( addState() );
mergeStates( md, startState, prevStartState );
assert
( md.stateDict.treeSize == 0 );
assert
( md.stfillHead == 0 );
removeMisfits();
setMisfitAccounting(
false
);
}
#ifdef LOG_CONDS
void
logCondSpace( CondSpace *condSpace )
{
if
( condSpace == 0 )
cerr <<
"<empty>"
;
else
{
for
( CondSet::Iter csi = condSpace->condSet.last(); csi.gtb(); csi-- ) {
if
( ! csi.last() )
cerr <<
','
;
(*csi)->actionName( cerr );
}
}
}
void
logNewExpansion( Expansion *
exp
)
{
cerr <<
"created expansion:"
<< endl;
cerr <<
" range: "
<<
exp
->lowKey.getVal() <<
" .. "
<<
exp
->highKey.getVal() << endl;
cerr <<
" fromCondSpace: "
;
logCondSpace(
exp
->fromCondSpace );
cerr << endl;
cerr <<
" fromVals: "
<<
exp
->fromVals << endl;
cerr <<
" toCondSpace: "
;
logCondSpace(
exp
->toCondSpace );
cerr << endl;
cerr <<
" toValsList: "
;
for
( LongVect::Iter to =
exp
->toValsList; to.lte(); to++ )
cerr <<
" "
<< *to;
cerr << endl;
}
#endif
void
FsmAp::findTransExpansions( ExpansionList &expansionList,
StateAp *destState, StateAp *srcState )
{
PairIter<TransAp, StateCond> transCond( destState->outList.head,
srcState->stateCondList.head );
for
( ; !transCond.end(); transCond++ ) {
if
( transCond.userState == RangeOverlap ) {
Expansion *expansion =
new
Expansion( transCond.s1Tel.lowKey,
transCond.s1Tel.highKey );
expansion->fromTrans =
new
TransAp(*transCond.s1Tel.trans);
expansion->fromTrans->fromState = 0;
expansion->fromTrans->toState = transCond.s1Tel.trans->toState;
expansion->fromCondSpace = 0;
expansion->fromVals = 0;
CondSpace *srcCS = transCond.s2Tel.trans->condSpace;
expansion->toCondSpace = srcCS;
long
numTargVals = (1 << srcCS->condSet.length());
for
(
long
targVals = 0; targVals < numTargVals; targVals++ )
expansion->toValsList.append( targVals );
#ifdef LOG_CONDS
logNewExpansion( expansion );
#endif
expansionList.append( expansion );
}
}
}
void
FsmAp::findCondExpInTrans( ExpansionList &expansionList, StateAp *state,
Key lowKey, Key highKey, CondSpace *fromCondSpace, CondSpace *toCondSpace,
long
fromVals, LongVect &toValsList )
{
TransAp searchTrans;
searchTrans.lowKey = fromCondSpace->baseKey + fromVals * keyOps->alphSize() +
(lowKey - keyOps->minKey);
searchTrans.highKey = fromCondSpace->baseKey + fromVals * keyOps->alphSize() +
(highKey - keyOps->minKey);
searchTrans.prev = searchTrans.next = 0;
PairIter<TransAp> pairIter( state->outList.head, &searchTrans );
for
( ; !pairIter.end(); pairIter++ ) {
if
( pairIter.userState == RangeOverlap ) {
Key expLowKey = pairIter.s1Tel.lowKey - fromCondSpace->baseKey - fromVals *
keyOps->alphSize() + keyOps->minKey;
Key expHighKey = pairIter.s1Tel.highKey - fromCondSpace->baseKey - fromVals *
keyOps->alphSize() + keyOps->minKey;
Expansion *expansion =
new
Expansion( expLowKey, expHighKey );
expansion->fromTrans =
new
TransAp(*pairIter.s1Tel.trans);
expansion->fromTrans->fromState = 0;
expansion->fromTrans->toState = pairIter.s1Tel.trans->toState;
expansion->fromCondSpace = fromCondSpace;
expansion->fromVals = fromVals;
expansion->toCondSpace = toCondSpace;
expansion->toValsList = toValsList;
expansionList.append( expansion );
#ifdef LOG_CONDS
logNewExpansion( expansion );
#endif
}
}
}
void
FsmAp::findCondExpansions( ExpansionList &expansionList,
StateAp *destState, StateAp *srcState )
{
PairIter<StateCond, StateCond> condCond( destState->stateCondList.head,
srcState->stateCondList.head );
for
( ; !condCond.end(); condCond++ ) {
if
( condCond.userState == RangeOverlap ) {
CondSet &destCS = condCond.s1Tel.trans->condSpace->condSet;
long
destLen = destCS.length();
CondSet srcOnlyCS = condCond.s2Tel.trans->condSpace->condSet;
for
( CondSet::Iter dcsi = destCS; dcsi.lte(); dcsi++ )
srcOnlyCS.
remove
( *dcsi );
long
srcOnlyLen = srcOnlyCS.length();
if
( srcOnlyCS.length() > 0 ) {
#ifdef LOG_CONDS
cerr <<
"there are "
<< srcOnlyCS.length() <<
" item(s) that are "
"only in the srcCS"
<< endl;
#endif
CondSet mergedCS = destCS;
mergedCS.insert( condCond.s2Tel.trans->condSpace->condSet );
CondSpace *fromCondSpace = addCondSpace( destCS );
CondSpace *toCondSpace = addCondSpace( mergedCS );
for
(
long
destVals = 0; destVals < (1 << destLen); destVals++ ) {
long
basicVals = 0;
for
( CondSet::Iter csi = destCS; csi.lte(); csi++ ) {
if
( destVals & (1 << csi.pos()) ) {
Action **cim = mergedCS.find( *csi );
long
bitPos = (cim - mergedCS.data);
basicVals |= 1 << bitPos;
}
}
LongVect expandToVals;
for
(
long
soVals = 0; soVals < (1 << srcOnlyLen); soVals++ ) {
long
targVals = basicVals;
for
( CondSet::Iter csi = srcOnlyCS; csi.lte(); csi++ ) {
if
( soVals & (1 << csi.pos()) ) {
Action **cim = mergedCS.find( *csi );
long
bitPos = (cim - mergedCS.data);
targVals |= 1 << bitPos;
}
}
expandToVals.append( targVals );
}
findCondExpInTrans( expansionList, destState,
condCond.s1Tel.lowKey, condCond.s1Tel.highKey,
fromCondSpace, toCondSpace, destVals, expandToVals );
}
}
}
}
}
void
FsmAp::doExpand( MergeData &md, StateAp *destState, ExpansionList &expList1 )
{
for
( ExpansionList::Iter
exp
= expList1;
exp
.lte();
exp
++ ) {
for
( LongVect::Iter to =
exp
->toValsList; to.lte(); to++ ) {
long
targVals = *to;
TransAp *srcTrans =
exp
->fromTrans;
srcTrans->lowKey =
exp
->toCondSpace->baseKey +
targVals * keyOps->alphSize() + (
exp
->lowKey - keyOps->minKey);
srcTrans->highKey =
exp
->toCondSpace->baseKey +
targVals * keyOps->alphSize() + (
exp
->highKey - keyOps->minKey);
TransList srcList;
srcList.append( srcTrans );
outTransCopy( md, destState, srcList.head );
srcList.abandon();
}
}
}
void
FsmAp::doRemove( MergeData &md, StateAp *destState, ExpansionList &expList1 )
{
for
( ExpansionList::Iter
exp
= expList1;
exp
.lte();
exp
++ ) {
Removal removal;
if
(
exp
->fromCondSpace == 0 ) {
removal.lowKey =
exp
->lowKey;
removal.highKey =
exp
->highKey;
}
else
{
removal.lowKey =
exp
->fromCondSpace->baseKey +
exp
->fromVals * keyOps->alphSize() + (
exp
->lowKey - keyOps->minKey);
removal.highKey =
exp
->fromCondSpace->baseKey +
exp
->fromVals * keyOps->alphSize() + (
exp
->highKey - keyOps->minKey);
}
removal.next = 0;
TransList destList;
PairIter<TransAp, Removal> pairIter( destState->outList.head, &removal );
for
( ; !pairIter.end(); pairIter++ ) {
switch
( pairIter.userState ) {
case
RangeInS1: {
TransAp *destTrans = pairIter.s1Tel.trans;
destTrans->lowKey = pairIter.s1Tel.lowKey;
destTrans->highKey = pairIter.s1Tel.highKey;
destList.append( destTrans );
break
;
}
case
RangeInS2:
break
;
case
RangeOverlap: {
TransAp *trans = pairIter.s1Tel.trans;
detachTrans( trans->fromState, trans->toState, trans );
delete
trans;
break
;
}
case
BreakS1: {
pairIter.s1Tel.trans = dupTrans( destState,
pairIter.s1Tel.trans );
break
;
}
case
BreakS2:
break
;
}
}
destState->outList.transfer( destList );
}
}
void
FsmAp::mergeStateConds( StateAp *destState, StateAp *srcState )
{
StateCondList destList;
PairIter<StateCond> pairIter( destState->stateCondList.head,
srcState->stateCondList.head );
for
( ; !pairIter.end(); pairIter++ ) {
switch
( pairIter.userState ) {
case
RangeInS1: {
StateCond *destCond = pairIter.s1Tel.trans;
destCond->lowKey = pairIter.s1Tel.lowKey;
destCond->highKey = pairIter.s1Tel.highKey;
destList.append( destCond );
break
;
}
case
RangeInS2: {
StateCond *newCond =
new
StateCond( *pairIter.s2Tel.trans );
newCond->lowKey = pairIter.s2Tel.lowKey;
newCond->highKey = pairIter.s2Tel.highKey;
destList.append( newCond );
break
;
}
case
RangeOverlap: {
StateCond *destCond = pairIter.s1Tel.trans;
StateCond *srcCond = pairIter.s2Tel.trans;
CondSet mergedCondSet;
mergedCondSet.insert( destCond->condSpace->condSet );
mergedCondSet.insert( srcCond->condSpace->condSet );
destCond->condSpace = addCondSpace( mergedCondSet );
destCond->lowKey = pairIter.s1Tel.lowKey;
destCond->highKey = pairIter.s1Tel.highKey;
destList.append( destCond );
break
;
}
case
BreakS1:
pairIter.s1Tel.trans =
new
StateCond( *pairIter.s1Tel.trans );
break
;
case
BreakS2:
break
;
}
}
destState->stateCondList.transfer( destList );
}
void
FsmAp::mergeStatesLeaving( MergeData &md, StateAp *destState, StateAp *srcState )
{
if
( !hasOutData( destState ) )
mergeStates( md, destState, srcState );
else
{
StateAp *ssMutable = addState();
mergeStates( md, ssMutable, srcState );
transferOutData( ssMutable, destState );
for
( OutCondSet::Iter cond = destState->outCondSet; cond.lte(); cond++ )
embedCondition( md, ssMutable, cond->action, cond->sense );
mergeStates( md, destState, ssMutable );
}
}
void
FsmAp::mergeStates( MergeData &md, StateAp *destState,
StateAp **srcStates,
int
numSrc )
{
for
(
int
s = 0; s < numSrc; s++ )
mergeStates( md, destState, srcStates[s] );
}
void
FsmAp::mergeStates( MergeData &md, StateAp *destState, StateAp *srcState )
{
ExpansionList expList1;
ExpansionList expList2;
findTransExpansions( expList1, destState, srcState );
findCondExpansions( expList1, destState, srcState );
findTransExpansions( expList2, srcState, destState );
findCondExpansions( expList2, srcState, destState );
mergeStateConds( destState, srcState );
outTransCopy( md, destState, srcState->outList.head );
doExpand( md, destState, expList1 );
doExpand( md, destState, expList2 );
doRemove( md, destState, expList1 );
doRemove( md, destState, expList2 );
expList1.empty();
expList2.empty();
destState->stateBits |= ( srcState->stateBits & ~STB_ISFINAL );
if
( srcState->isFinState() )
setFinState( destState );
if
( srcState == destState ) {
destState->epsilonTrans.append( EpsilonTrans( srcState->epsilonTrans ) );
destState->toStateActionTable.setActions(
ActionTable( srcState->toStateActionTable ) );
destState->fromStateActionTable.setActions(
ActionTable( srcState->fromStateActionTable ) );
destState->outActionTable.setActions( ActionTable( srcState->outActionTable ) );
destState->outCondSet.insert( OutCondSet( srcState->outCondSet ) );
destState->errActionTable.setActions( ErrActionTable( srcState->errActionTable ) );
destState->eofActionTable.setActions( ActionTable( srcState->eofActionTable ) );
}
else
{
destState->epsilonTrans.append( srcState->epsilonTrans );
destState->outPriorTable.setPriors( srcState->outPriorTable );
destState->toStateActionTable.setActions( srcState->toStateActionTable );
destState->fromStateActionTable.setActions( srcState->fromStateActionTable );
destState->outActionTable.setActions( srcState->outActionTable );
destState->outCondSet.insert( srcState->outCondSet );
destState->errActionTable.setActions( srcState->errActionTable );
destState->eofActionTable.setActions( srcState->eofActionTable );
}
}
void
FsmAp::fillInStates( MergeData &md )
{
StateAp *state = md.stfillHead;
while
( state != 0 ) {
StateSet *stateSet = &state->stateDictEl->stateSet;
mergeStates( md, state, stateSet->data, stateSet->length() );
state = state->alg.next;
}
state = md.stfillHead;
while
( state != 0 ) {
delete
state->stateDictEl;
state->stateDictEl = 0;
state = state->alg.next;
}
}
void
FsmAp::findEmbedExpansions( ExpansionList &expansionList,
StateAp *destState, Action *condAction,
bool
sense )
{
StateCondList destList;
PairIter<TransAp, StateCond> transCond( destState->outList.head,
destState->stateCondList.head );
for
( ; !transCond.end(); transCond++ ) {
switch
( transCond.userState ) {
case
RangeInS1: {
if
( transCond.s1Tel.lowKey <= keyOps->maxKey ) {
assert
( transCond.s1Tel.highKey <= keyOps->maxKey );
StateCond *newStateCond =
new
StateCond( transCond.s1Tel.lowKey,
transCond.s1Tel.highKey );
newStateCond->condSpace = addCondSpace( CondSet( condAction ) );
destList.append( newStateCond );
Expansion *expansion =
new
Expansion( transCond.s1Tel.lowKey,
transCond.s1Tel.highKey );
expansion->fromTrans =
new
TransAp(*transCond.s1Tel.trans);
expansion->fromTrans->fromState = 0;
expansion->fromTrans->toState = transCond.s1Tel.trans->toState;
expansion->fromCondSpace = 0;
expansion->fromVals = 0;
expansion->toCondSpace = newStateCond->condSpace;
expansion->toValsList.append( sense?1:0 );
#ifdef LOG_CONDS
logNewExpansion( expansion );
#endif
expansionList.append( expansion );
}
break
;
}
case
RangeInS2: {
StateCond *stateCond = transCond.s2Tel.trans;
stateCond->lowKey = transCond.s2Tel.lowKey;
stateCond->highKey = transCond.s2Tel.highKey;
CondSet &destCS = stateCond->condSpace->condSet;
long
destLen = destCS.length();
CondSpace *fromCondSpace = stateCond->condSpace;
CondSet mergedCS = destCS;
mergedCS.insert( condAction );
CondSpace *toCondSpace = addCondSpace( mergedCS );
stateCond->condSpace = toCondSpace;
destList.append( stateCond );
for
(
long
destVals = 0; destVals < (1 << destLen); destVals++ ) {
long
basicVals = 0;
for
( CondSet::Iter csi = destCS; csi.lte(); csi++ ) {
if
( destVals & (1 << csi.pos()) ) {
Action **cim = mergedCS.find( *csi );
long
bitPos = (cim - mergedCS.data);
basicVals |= 1 << bitPos;
}
}
long
targVals = basicVals;
Action **cim = mergedCS.find( condAction );
long
bitPos = (cim - mergedCS.data);
targVals |= (sense?1:0) << bitPos;
LongVect expandToVals( targVals );
findCondExpInTrans( expansionList, destState,
transCond.s2Tel.lowKey, transCond.s2Tel.highKey,
fromCondSpace, toCondSpace, destVals, expandToVals );
}
break
;
}
case
RangeOverlap:
case
BreakS1:
case
BreakS2:
assert
(
false
);
break
;
}
}
destState->stateCondList.transfer( destList );
}
void
FsmAp::embedCondition( StateAp *state, Action *condAction,
bool
sense )
{
MergeData md;
ExpansionList expList;
setMisfitAccounting(
true
);
embedCondition( md, state, condAction, sense );
fillInStates( md );
removeMisfits();
setMisfitAccounting(
false
);
}
void
FsmAp::embedCondition( MergeData &md, StateAp *state, Action *condAction,
bool
sense )
{
ExpansionList expList;
findEmbedExpansions( expList, state, condAction, sense );
doExpand( md, state, expList );
doRemove( md, state, expList );
expList.empty();
}
bool
FsmAp::checkSingleCharMachine()
{
if
( stateList.length() != 2 )
return
false
;
if
( startState->isFinState() )
return
false
;
if
( finStateSet.length() != 1 )
return
false
;
if
( finStateSet[0]->outList.length() != 0 )
return
false
;
if
( startState->outList.length() != 1 )
return
false
;
TransAp *startTrans = startState->outList.head;
if
( startTrans->lowKey != startTrans->highKey )
return
false
;
return
true
;
}