#include <string.h>
#include <assert.h>
#include "fsmgraph.h"
#include <iostream>
using
namespace
std;
MarkIndex::MarkIndex(
int
states ) : numStates(states)
{
int
total = states * states;
array =
new
bool
[total];
memset
( array, 0,
sizeof
(
bool
) * total );
}
MarkIndex::~MarkIndex()
{
delete
[] array;
}
void
MarkIndex::markPair(
int
state1,
int
state2)
{
int
pos = ( state1 >= state2 ) ?
( state1 * numStates ) + state2 :
( state2 * numStates ) + state1;
array[pos] =
true
;
}
bool
MarkIndex::isPairMarked(
int
state1,
int
state2)
{
int
pos = ( state1 >= state2 ) ?
( state1 * numStates ) + state2 :
( state2 * numStates ) + state1;
return
array[pos];
}
StateAp::StateAp()
:
outList(),
inList(),
eofTarget(0),
entryIds(),
epsilonTrans(),
stateCondList(),
foreignInTrans(0),
stateDictEl(0),
eptVect(0),
stateBits(0),
outPriorTable(),
toStateActionTable(),
fromStateActionTable(),
outActionTable(),
outCondSet(),
errActionTable(),
eofActionTable()
{
}
StateAp::StateAp(
const
StateAp &other)
:
outList(),
inList(),
eofTarget(other.eofTarget),
entryIds(other.entryIds),
epsilonTrans(other.epsilonTrans),
stateCondList( other.stateCondList ),
foreignInTrans(0),
stateDictEl(0),
eptVect(0),
stateBits(other.stateBits),
outPriorTable(other.outPriorTable),
toStateActionTable(other.toStateActionTable),
fromStateActionTable(other.fromStateActionTable),
outActionTable(other.outActionTable),
outCondSet(other.outCondSet),
errActionTable(other.errActionTable),
eofActionTable(other.eofActionTable)
{
for
( TransList::Iter trans = other.outList; trans.lte(); trans++ ) {
TransAp *newTrans =
new
TransAp(*trans);
assert
( trans->lmActionTable.length() == 0 );
newTrans->toState = trans->toState;
outList.append( newTrans );
}
}
StateAp::~StateAp()
{
if
( stateDictEl != 0 )
delete
stateDictEl;
}
int
ApproxCompare::compare(
const
StateAp *state1,
const
StateAp *state2 )
{
int
compareRes;
if
( (state1->stateBits & STB_ISFINAL) && !(state2->stateBits & STB_ISFINAL) )
return
-1;
else
if
( !(state1->stateBits & STB_ISFINAL) && (state2->stateBits & STB_ISFINAL) )
return
1;
compareRes = CmpEpsilonTrans::compare( state1->epsilonTrans,
state2->epsilonTrans );
if
( compareRes != 0 )
return
compareRes;
compareRes = FsmAp::compareStateData( state1, state2 );
if
( compareRes != 0 )
return
compareRes;
PairIter<TransAp> outPair( state1->outList.head, state2->outList.head );
for
( ; !outPair.end(); outPair++ ) {
switch
( outPair.userState ) {
case
RangeInS1:
compareRes = FsmAp::compareFullPtr( outPair.s1Tel.trans, 0 );
if
( compareRes != 0 )
return
compareRes;
break
;
case
RangeInS2:
compareRes = FsmAp::compareFullPtr( 0, outPair.s2Tel.trans );
if
( compareRes != 0 )
return
compareRes;
break
;
case
RangeOverlap:
compareRes = FsmAp::compareFullPtr(
outPair.s1Tel.trans, outPair.s2Tel.trans );
if
( compareRes != 0 )
return
compareRes;
break
;
case
BreakS1:
case
BreakS2:
break
;
}
}
if
( state1->eofTarget < state2->eofTarget )
return
-1;
else
if
( state1->eofTarget > state2->eofTarget )
return
1;
return
0;
}
int
InitPartitionCompare::compare(
const
StateAp *state1 ,
const
StateAp *state2 )
{
int
compareRes;
if
( (state1->stateBits & STB_ISFINAL) && !(state2->stateBits & STB_ISFINAL) )
return
-1;
else
if
( !(state1->stateBits & STB_ISFINAL) && (state2->stateBits & STB_ISFINAL) )
return
1;
compareRes = CmpEpsilonTrans::compare( state1->epsilonTrans,
state2->epsilonTrans );
if
( compareRes != 0 )
return
compareRes;
compareRes = FsmAp::compareStateData( state1, state2 );
if
( compareRes != 0 )
return
compareRes;
PairIter<StateCond> condPair( state1->stateCondList.head, state2->stateCondList.head );
for
( ; !condPair.end(); condPair++ ) {
switch
( condPair.userState ) {
case
RangeInS1:
return
1;
case
RangeInS2:
return
-1;
case
RangeOverlap: {
CondSpace *condSpace1 = condPair.s1Tel.trans->condSpace;
CondSpace *condSpace2 = condPair.s2Tel.trans->condSpace;
if
( condSpace1 < condSpace2 )
return
-1;
else
if
( condSpace1 > condSpace2 )
return
1;
break
;
}
case
BreakS1:
case
BreakS2:
break
;
}
}
PairIter<TransAp> outPair( state1->outList.head, state2->outList.head );
for
( ; !outPair.end(); outPair++ ) {
switch
( outPair.userState ) {
case
RangeInS1:
compareRes = FsmAp::compareDataPtr( outPair.s1Tel.trans, 0 );
if
( compareRes != 0 )
return
compareRes;
break
;
case
RangeInS2:
compareRes = FsmAp::compareDataPtr( 0, outPair.s2Tel.trans );
if
( compareRes != 0 )
return
compareRes;
break
;
case
RangeOverlap:
compareRes = FsmAp::compareDataPtr(
outPair.s1Tel.trans, outPair.s2Tel.trans );
if
( compareRes != 0 )
return
compareRes;
break
;
case
BreakS1:
case
BreakS2:
break
;
}
}
return
0;
}
int
PartitionCompare::compare(
const
StateAp *state1,
const
StateAp *state2 )
{
int
compareRes;
PairIter<TransAp> outPair( state1->outList.head, state2->outList.head );
for
( ; !outPair.end(); outPair++ ) {
switch
( outPair.userState ) {
case
RangeInS1:
compareRes = FsmAp::comparePartPtr( outPair.s1Tel.trans, 0 );
if
( compareRes != 0 )
return
compareRes;
break
;
case
RangeInS2:
compareRes = FsmAp::comparePartPtr( 0, outPair.s2Tel.trans );
if
( compareRes != 0 )
return
compareRes;
break
;
case
RangeOverlap:
compareRes = FsmAp::comparePartPtr(
outPair.s1Tel.trans, outPair.s2Tel.trans );
if
( compareRes != 0 )
return
compareRes;
break
;
case
BreakS1:
case
BreakS2:
break
;
}
}
if
( state1->eofTarget == 0 && state2->eofTarget != 0 )
return
-1;
else
if
( state1->eofTarget != 0 && state2->eofTarget == 0 )
return
1;
else
if
( state1->eofTarget != 0 ) {
compareRes = CmpOrd< MinPartition* >::compare(
state1->eofTarget->alg.partition, state2->eofTarget->alg.partition );
if
( compareRes != 0 )
return
compareRes;
}
return
0;
}
bool
MarkCompare::shouldMark( MarkIndex &markIndex,
const
StateAp *state1,
const
StateAp *state2 )
{
PairIter<TransAp> outPair( state1->outList.head, state2->outList.head );
for
( ; !outPair.end(); outPair++ ) {
switch
( outPair.userState ) {
case
RangeInS1:
if
( FsmAp::shouldMarkPtr( markIndex, outPair.s1Tel.trans, 0 ) )
return
true
;
break
;
case
RangeInS2:
if
( FsmAp::shouldMarkPtr( markIndex, 0, outPair.s2Tel.trans ) )
return
true
;
break
;
case
RangeOverlap:
if
( FsmAp::shouldMarkPtr( markIndex,
outPair.s1Tel.trans, outPair.s2Tel.trans ) )
return
true
;
break
;
case
BreakS1:
case
BreakS2:
break
;
}
}
return
false
;
}
int
FsmAp::comparePartPtr( TransAp *trans1, TransAp *trans2 )
{
if
( trans1 != 0 ) {
if
( trans1->toState == 0 && trans2->toState != 0 )
return
-1;
else
if
( trans1->toState != 0 && trans2->toState == 0 )
return
1;
else
if
( trans1->toState != 0 ) {
return
CmpOrd< MinPartition* >::compare(
trans1->toState->alg.partition, trans2->toState->alg.partition );
}
}
return
0;
}
int
FsmAp::compareDataPtr( TransAp *trans1, TransAp *trans2 )
{
if
( trans1 == 0 && trans2 != 0 )
return
-1;
else
if
( trans1 != 0 && trans2 == 0 )
return
1;
else
if
( trans1 != 0 ) {
int
compareRes = compareTransData( trans1, trans2 );
if
( compareRes != 0 )
return
compareRes;
}
return
0;
}
int
FsmAp::compareFullPtr( TransAp *trans1, TransAp *trans2 )
{
if
( (trans1 != 0) ^ (trans2 != 0) ) {
if
( trans1 != 0 )
return
-1;
else
return
1;
}
else
if
( trans1 != 0 ) {
if
( trans1->toState < trans2->toState )
return
-1;
else
if
( trans1->toState > trans2->toState )
return
1;
else
if
( trans1->toState != 0 ) {
int
compareRes = compareTransData( trans1, trans2 );
if
( compareRes != 0 )
return
compareRes;
}
}
return
0;
}
bool
FsmAp::shouldMarkPtr( MarkIndex &markIndex, TransAp *trans1,
TransAp *trans2 )
{
if
( (trans1 != 0) ^ (trans2 != 0) ) {
assert
(
false
);
}
else
if
( trans1 != 0 ) {
return
markIndex.isPairMarked( trans1->toState->alg.stateNum,
trans2->toState->alg.stateNum );
}
return
false
;
}