#include "fsmgraph.h"
#include <iostream>
using
std::cerr;
using
std::endl;
CondData *condData = 0;
KeyOps *keyOps = 0;
void
ActionTable::setAction(
int
ordering, Action *action )
{
insertMulti( ordering, action );
}
void
ActionTable::setActions(
const
ActionTable &other )
{
for
( ActionTable::Iter action = other; action.lte(); action++ )
insertMulti( action->key, action->value );
}
void
ActionTable::setActions(
int
*orderings, Action **actions,
int
nActs )
{
for
(
int
a = 0; a < nActs; a++ )
insertMulti( orderings[a], actions[a] );
}
bool
ActionTable::hasAction( Action *action )
{
for
(
int
a = 0; a < length(); a++ ) {
if
( data[a].value == action )
return
true
;
}
return
false
;
}
void
LmActionTable::setAction(
int
ordering, LongestMatchPart *action )
{
insertMulti( ordering, action );
}
void
LmActionTable::setActions(
const
LmActionTable &other )
{
for
( LmActionTable::Iter action = other; action.lte(); action++ )
insertMulti( action->key, action->value );
}
void
ErrActionTable::setAction(
int
ordering, Action *action,
int
transferPoint )
{
insertMulti( ErrActionTableEl( action, ordering, transferPoint ) );
}
void
ErrActionTable::setActions(
const
ErrActionTable &other )
{
for
( ErrActionTable::Iter act = other; act.lte(); act++ )
insertMulti( ErrActionTableEl( act->action, act->ordering, act->transferPoint ) );
}
void
PriorTable::setPrior(
int
ordering, PriorDesc *desc )
{
PriorEl *lastHit = 0;
PriorEl *insed = insert( PriorEl(ordering, desc), &lastHit );
if
( insed == 0 ) {
if
( ordering >= lastHit->ordering )
*lastHit = PriorEl( ordering, desc );
}
}
void
PriorTable::setPriors(
const
PriorTable &other )
{
PriorTable::Iter priorIt = other;
for
( ; priorIt.lte(); priorIt++ )
setPrior( priorIt->ordering, priorIt->desc );
}
void
FsmAp::startFsmPrior(
int
ordering, PriorDesc *prior )
{
isolateStartState();
for
( TransList::Iter trans = startState->outList; trans.lte(); trans++ ) {
if
( trans->toState != 0 )
trans->priorTable.setPrior( ordering, prior );
}
if
( startState->stateBits & STB_ISFINAL )
startState->outPriorTable.setPrior( ordering, prior );
}
void
FsmAp::allTransPrior(
int
ordering, PriorDesc *prior )
{
for
( StateList::Iter state = stateList; state.lte(); state++ ) {
for
( TransList::Iter trans = state->outList; trans.lte(); trans++ ) {
if
( trans->toState != 0 )
trans->priorTable.setPrior( ordering, prior );
}
}
}
void
FsmAp::finishFsmPrior(
int
ordering, PriorDesc *prior )
{
for
( StateSet::Iter state = finStateSet; state.lte(); state++ ) {
for
( TransInList::Iter trans = (*state)->inList; trans.lte(); trans++ )
trans->priorTable.setPrior( ordering, prior );
}
}
void
FsmAp::leaveFsmPrior(
int
ordering, PriorDesc *prior )
{
for
( StateSet::Iter state = finStateSet; state.lte(); state++ )
(*state)->outPriorTable.setPrior( ordering, prior );
}
void
FsmAp::startFsmAction(
int
ordering, Action *action )
{
isolateStartState();
for
( TransList::Iter trans = startState->outList; trans.lte(); trans++ ) {
if
( trans->toState != 0 )
trans->actionTable.setAction( ordering, action );
}
if
( startState->stateBits & STB_ISFINAL )
startState->outActionTable.setAction( ordering, action );
}
void
FsmAp::allTransAction(
int
ordering, Action *action )
{
for
( StateList::Iter state = stateList; state.lte(); state++ ) {
for
( TransList::Iter trans = state->outList; trans.lte(); trans++ ) {
if
( trans->toState != 0 )
trans->actionTable.setAction( ordering, action );
}
}
}
void
FsmAp::finishFsmAction(
int
ordering, Action *action )
{
for
( StateSet::Iter state = finStateSet; state.lte(); state++ ) {
for
( TransInList::Iter trans = (*state)->inList; trans.lte(); trans++ )
trans->actionTable.setAction( ordering, action );
}
}
void
FsmAp::leaveFsmAction(
int
ordering, Action *action )
{
for
( StateSet::Iter state = finStateSet; state.lte(); state++ )
(*state)->outActionTable.setAction( ordering, action );
}
void
FsmAp::longMatchAction(
int
ordering, LongestMatchPart *lmPart )
{
for
( StateSet::Iter state = finStateSet; state.lte(); state++ ) {
for
( TransInList::Iter trans = (*state)->inList; trans.lte(); trans++ )
trans->lmActionTable.setAction( ordering, lmPart );
}
}
void
FsmAp::fillGaps( StateAp *state )
{
if
( state->outList.length() == 0 ) {
attachNewTrans( state, 0, keyOps->minKey, keyOps->maxKey );
}
else
{
TransList srcList;
srcList.transfer( state->outList );
TransList::Iter trans = srcList, next;
if
( keyOps->minKey < trans->lowKey ) {
Key highKey = trans->lowKey;
highKey.decrement();
attachNewTrans( state, 0, keyOps->minKey, highKey );
}
next = trans.next();
state->outList.append( trans );
Key lastHigh = trans->highKey;
for
( trans = next; trans.lte(); trans = next ) {
Key nextKey = lastHigh;
nextKey.increment();
if
( nextKey < trans->lowKey ) {
Key highKey = trans->lowKey;
highKey.decrement();
attachNewTrans( state, 0, nextKey, highKey );
}
next = trans.next();
state->outList.append( trans );
lastHigh = trans->highKey;
}
if
( lastHigh < keyOps->maxKey ) {
lastHigh.increment();
attachNewTrans( state, 0, lastHigh, keyOps->maxKey );
}
}
}
void
FsmAp::setErrorActions( StateAp *state,
const
ActionTable &other )
{
fillGaps( state );
for
( TransList::Iter trans = state->outList; trans.lte(); trans++ ) {
if
( trans->toState == 0 )
trans->actionTable.setActions( other );
}
}
void
FsmAp::setErrorAction( StateAp *state,
int
ordering, Action *action )
{
fillGaps( state );
for
( TransList::Iter trans = state->outList; trans.lte(); trans++ ) {
if
( trans->toState == 0 )
trans->actionTable.setAction( ordering, action );
}
}
void
FsmAp::setErrorTarget( StateAp *state, StateAp *target,
int
*orderings,
Action **actions,
int
nActs )
{
fillGaps( state );
for
( TransList::Iter trans = state->outList; trans.lte(); trans++ ) {
if
( trans->toState == 0 ) {
redirectErrorTrans( trans->fromState, target, trans );
trans->actionTable.setActions( orderings, actions, nActs );
}
}
}
void
FsmAp::transferOutActions( StateAp *state )
{
for
( ActionTable::Iter act = state->outActionTable; act.lte(); act++ )
state->eofActionTable.setAction( act->key, act->value );
state->outActionTable.empty();
}
void
FsmAp::transferErrorActions( StateAp *state,
int
transferPoint )
{
for
(
int
i = 0; i < state->errActionTable.length(); ) {
ErrActionTableEl *act = state->errActionTable.data + i;
if
( act->transferPoint == transferPoint ) {
setErrorAction( state, act->ordering, act->action );
if
( ! state->isFinState() )
state->eofActionTable.setAction( act->ordering, act->action );
state->errActionTable.vremove( i );
}
else
{
i += 1;
}
}
}
void
FsmAp::startErrorAction(
int
ordering, Action *action,
int
transferPoint )
{
isolateStartState();
startState->errActionTable.setAction( ordering, action, transferPoint );
}
void
FsmAp::allErrorAction(
int
ordering, Action *action,
int
transferPoint )
{
for
( StateList::Iter state = stateList; state.lte(); state++ )
state->errActionTable.setAction( ordering, action, transferPoint );
}
void
FsmAp::finalErrorAction(
int
ordering, Action *action,
int
transferPoint )
{
for
( StateSet::Iter state = finStateSet; state.lte(); state++ )
(*state)->errActionTable.setAction( ordering, action, transferPoint );
}
void
FsmAp::notStartErrorAction(
int
ordering, Action *action,
int
transferPoint )
{
for
( StateList::Iter state = stateList; state.lte(); state++ ) {
if
( state != startState )
state->errActionTable.setAction( ordering, action, transferPoint );
}
}
void
FsmAp::notFinalErrorAction(
int
ordering, Action *action,
int
transferPoint )
{
for
( StateList::Iter state = stateList; state.lte(); state++ ) {
if
( ! state->isFinState() )
state->errActionTable.setAction( ordering, action, transferPoint );
}
}
void
FsmAp::middleErrorAction(
int
ordering, Action *action,
int
transferPoint )
{
for
( StateList::Iter state = stateList; state.lte(); state++ ) {
if
( state != startState && ! state->isFinState() )
state->errActionTable.setAction( ordering, action, transferPoint );
}
}
void
FsmAp::startEOFAction(
int
ordering, Action *action )
{
isolateStartState();
startState->eofActionTable.setAction( ordering, action );
}
void
FsmAp::allEOFAction(
int
ordering, Action *action )
{
for
( StateList::Iter state = stateList; state.lte(); state++ )
state->eofActionTable.setAction( ordering, action );
}
void
FsmAp::finalEOFAction(
int
ordering, Action *action )
{
for
( StateSet::Iter state = finStateSet; state.lte(); state++ )
(*state)->eofActionTable.setAction( ordering, action );
}
void
FsmAp::notStartEOFAction(
int
ordering, Action *action )
{
for
( StateList::Iter state = stateList; state.lte(); state++ ) {
if
( state != startState )
state->eofActionTable.setAction( ordering, action );
}
}
void
FsmAp::notFinalEOFAction(
int
ordering, Action *action )
{
for
( StateList::Iter state = stateList; state.lte(); state++ ) {
if
( ! state->isFinState() )
state->eofActionTable.setAction( ordering, action );
}
}
void
FsmAp::middleEOFAction(
int
ordering, Action *action )
{
for
( StateList::Iter state = stateList; state.lte(); state++ ) {
if
( state != startState && ! state->isFinState() )
state->eofActionTable.setAction( ordering, action );
}
}
void
FsmAp::startToStateAction(
int
ordering, Action *action )
{
isolateStartState();
startState->toStateActionTable.setAction( ordering, action );
}
void
FsmAp::allToStateAction(
int
ordering, Action *action )
{
for
( StateList::Iter state = stateList; state.lte(); state++ )
state->toStateActionTable.setAction( ordering, action );
}
void
FsmAp::finalToStateAction(
int
ordering, Action *action )
{
for
( StateSet::Iter state = finStateSet; state.lte(); state++ )
(*state)->toStateActionTable.setAction( ordering, action );
}
void
FsmAp::notStartToStateAction(
int
ordering, Action *action )
{
for
( StateList::Iter state = stateList; state.lte(); state++ ) {
if
( state != startState )
state->toStateActionTable.setAction( ordering, action );
}
}
void
FsmAp::notFinalToStateAction(
int
ordering, Action *action )
{
for
( StateList::Iter state = stateList; state.lte(); state++ ) {
if
( ! state->isFinState() )
state->toStateActionTable.setAction( ordering, action );
}
}
void
FsmAp::middleToStateAction(
int
ordering, Action *action )
{
for
( StateList::Iter state = stateList; state.lte(); state++ ) {
if
( state != startState && ! state->isFinState() )
state->toStateActionTable.setAction( ordering, action );
}
}
void
FsmAp::startFromStateAction(
int
ordering, Action *action )
{
isolateStartState();
startState->fromStateActionTable.setAction( ordering, action );
}
void
FsmAp::allFromStateAction(
int
ordering, Action *action )
{
for
( StateList::Iter state = stateList; state.lte(); state++ )
state->fromStateActionTable.setAction( ordering, action );
}
void
FsmAp::finalFromStateAction(
int
ordering, Action *action )
{
for
( StateSet::Iter state = finStateSet; state.lte(); state++ )
(*state)->fromStateActionTable.setAction( ordering, action );
}
void
FsmAp::notStartFromStateAction(
int
ordering, Action *action )
{
for
( StateList::Iter state = stateList; state.lte(); state++ ) {
if
( state != startState )
state->fromStateActionTable.setAction( ordering, action );
}
}
void
FsmAp::notFinalFromStateAction(
int
ordering, Action *action )
{
for
( StateList::Iter state = stateList; state.lte(); state++ ) {
if
( ! state->isFinState() )
state->fromStateActionTable.setAction( ordering, action );
}
}
void
FsmAp::middleFromStateAction(
int
ordering, Action *action )
{
for
( StateList::Iter state = stateList; state.lte(); state++ ) {
if
( state != startState && ! state->isFinState() )
state->fromStateActionTable.setAction( ordering, action );
}
}
int
FsmAp::shiftStartActionOrder(
int
fromOrder )
{
int
maxUsed = 0;
for
( TransList::Iter trans = startState->outList; trans.lte(); trans++ ) {
int
curFromOrder = fromOrder;
ActionTable::Iter action = trans->actionTable;
for
( ; action.lte(); action++ )
action->key = curFromOrder++;
if
( curFromOrder - fromOrder > maxUsed )
maxUsed = curFromOrder - fromOrder;
}
return
maxUsed;
}
void
FsmAp::clearAllPriorities()
{
for
( StateList::Iter state = stateList; state.lte(); state++ ) {
state->outPriorTable.empty();
for
( TransList::Iter trans = state->outList; trans.lte(); trans++ )
trans->priorTable.empty();
}
}
void
FsmAp::nullActionKeys( )
{
for
( StateList::Iter state = stateList; state.lte(); state++ ) {
for
( TransList::Iter trans = state->outList; trans.lte(); trans++ ) {
for
( ActionTable::Iter action = trans->actionTable;
action.lte(); action++ )
action->key = 0;
for
( LmActionTable::Iter action = trans->lmActionTable;
action.lte(); action++ )
action->key = 0;
}
for
( ActionTable::Iter action = state->toStateActionTable;
action.lte(); action++ )
action->key = 0;
for
( ActionTable::Iter action = state->fromStateActionTable;
action.lte(); action++ )
action->key = 0;
for
( ActionTable::Iter action = state->outActionTable;
action.lte(); action++ )
action->key = 0;
for
( ErrActionTable::Iter action = state->errActionTable;
action.lte(); action++ )
action->ordering = 0;
for
( ActionTable::Iter action = state->eofActionTable;
action.lte(); action++ )
action->key = 0;
}
}
void
FsmAp::verifyStates()
{
for
( StateList::Iter state = stateList; state.lte(); state++ ) {
if
( ! (state->stateBits & STB_ISFINAL) ) {
assert
( state->outActionTable.length() == 0 );
assert
( state->outCondSet.length() == 0 );
assert
( state->outPriorTable.length() == 0 );
}
assert
( (state->stateBits & STB_BOTH) == 0 );
assert
( state->foreignInTrans > 0 );
}
}
int
FsmAp::comparePrior(
const
PriorTable &priorTable1,
const
PriorTable &priorTable2 )
{
PriorTable::Iter pd1 = priorTable1;
PriorTable::Iter pd2 = priorTable2;
while
( pd1.lte() && pd2.lte() ) {
if
( pd1->desc->key < pd2->desc->key )
pd1.increment();
else
if
( pd1->desc->key > pd2->desc->key )
pd2.increment();
else
if
( pd1->desc->priority < pd2->desc->priority )
return
-1;
else
if
( pd1->desc->priority > pd2->desc->priority )
return
1;
else
{
pd1.increment();
pd2.increment();
}
}
return
0;
}
int
FsmAp::compareTransData( TransAp *trans1, TransAp *trans2 )
{
int
cmpRes = CmpPriorTable::compare( trans1->priorTable,
trans2->priorTable );
if
( cmpRes != 0 )
return
cmpRes;
cmpRes = CmpLmActionTable::compare(trans1->lmActionTable,
trans2->lmActionTable);
if
( cmpRes != 0 )
return
cmpRes;
return
CmpActionTable::compare(trans1->actionTable,
trans2->actionTable);
}
void
FsmAp::addInTrans( TransAp *destTrans, TransAp *srcTrans )
{
if
( srcTrans == destTrans ) {
destTrans->lmActionTable.setActions( LmActionTable(srcTrans->lmActionTable) );
destTrans->actionTable.setActions( ActionTable(srcTrans->actionTable) );
}
else
{
destTrans->lmActionTable.setActions( srcTrans->lmActionTable );
destTrans->actionTable.setActions( srcTrans->actionTable );
destTrans->priorTable.setPriors( srcTrans->priorTable );
}
}
int
FsmAp::compareStateData(
const
StateAp *state1,
const
StateAp *state2 )
{
int
cmpRes = CmpPriorTable::
compare( state1->outPriorTable, state2->outPriorTable );
if
( cmpRes != 0 )
return
cmpRes;
cmpRes = CmpActionTable::compare( state1->toStateActionTable,
state2->toStateActionTable );
if
( cmpRes != 0 )
return
cmpRes;
cmpRes = CmpActionTable::compare( state1->fromStateActionTable,
state2->fromStateActionTable );
if
( cmpRes != 0 )
return
cmpRes;
cmpRes = CmpActionTable::compare( state1->outActionTable,
state2->outActionTable );
if
( cmpRes != 0 )
return
cmpRes;
cmpRes = CmpOutCondSet::compare( state1->outCondSet,
state2->outCondSet );
if
( cmpRes != 0 )
return
cmpRes;
cmpRes = CmpErrActionTable::compare( state1->errActionTable,
state2->errActionTable );
if
( cmpRes != 0 )
return
cmpRes;
return
CmpActionTable::compare( state1->eofActionTable,
state2->eofActionTable );
}
void
FsmAp::clearOutData( StateAp *state )
{
state->outActionTable.empty();
state->outCondSet.empty();
state->outPriorTable.empty();
}
bool
FsmAp::hasOutData( StateAp *state )
{
return
( state->outActionTable.length() > 0 ||
state->outCondSet.length() > 0 ||
state->outPriorTable.length() > 0 );
}
void
logNewExpansion( Expansion *
exp
);
void
logCondSpace( CondSpace *condSpace );
CondSpace *FsmAp::addCondSpace(
const
CondSet &condSet )
{
CondSpace *condSpace = condData->condSpaceMap.find( condSet );
if
( condSpace == 0 ) {
Size availableSpace = condData->lastCondKey.availableSpace();
Size neededSpace = (1 << condSet.length() ) * keyOps->alphSize();
if
( neededSpace > availableSpace )
throw
FsmConstructFail( FsmConstructFail::CondNoKeySpace );
Key baseKey = condData->lastCondKey;
baseKey.increment();
condData->lastCondKey += (1 << condSet.length() ) * keyOps->alphSize();
condSpace =
new
CondSpace( condSet );
condSpace->baseKey = baseKey;
condData->condSpaceMap.insert( condSpace );
#ifdef LOG_CONDS
cerr <<
"adding new condition space"
<< endl;
cerr <<
" condition set: "
;
logCondSpace( condSpace );
cerr << endl;
cerr <<
" baseKey: "
<< baseKey.getVal() << endl;
#endif
}
return
condSpace;
}
void
FsmAp::startFsmCondition( Action *condAction,
bool
sense )
{
isolateStartState();
embedCondition( startState, condAction, sense );
}
void
FsmAp::allTransCondition( Action *condAction,
bool
sense )
{
for
( StateList::Iter state = stateList; state.lte(); state++ )
embedCondition( state, condAction, sense );
}
void
FsmAp::leaveFsmCondition( Action *condAction,
bool
sense )
{
for
( StateSet::Iter state = finStateSet; state.lte(); state++ )
(*state)->outCondSet.insert( OutCond( condAction, sense ) );
}