#include "redfsm.h"
#include "avlmap.h"
#include "mergesort.h"
#include <iostream>
#include <sstream>
using
std::ostringstream;
using
std::string;
string GenAction::nameOrLoc()
{
if
( name != 0 )
return
string(name);
else
{
ostringstream ret;
ret << loc.line <<
":"
<< loc.col;
return
ret.str();
}
}
RedFsmAp::RedFsmAp()
:
forcedErrorState(
false
),
nextActionId(0),
nextTransId(0),
startState(0),
errState(0),
errTrans(0),
firstFinState(0),
numFinStates(0),
bAnyToStateActions(
false
),
bAnyFromStateActions(
false
),
bAnyRegActions(
false
),
bAnyEofActions(
false
),
bAnyEofTrans(
false
),
bAnyActionGotos(
false
),
bAnyActionCalls(
false
),
bAnyActionRets(
false
),
bAnyActionByValControl(
false
),
bAnyRegActionRets(
false
),
bAnyRegActionByValControl(
false
),
bAnyRegNextStmt(
false
),
bAnyRegCurStateRef(
false
),
bAnyRegBreak(
false
),
bAnyConditions(
false
)
{
}
bool
RedFsmAp::anyActions()
{
return
actionMap.length() > 0;
}
void
RedFsmAp::depthFirstOrdering( RedStateAp *state )
{
if
( state->onStateList )
return
;
state->onStateList =
true
;
stateList.append( state );
assert
( state->outSingle.length() == 0 );
assert
( state->defTrans == 0 );
for
( RedTransList::Iter rtel = state->outRange; rtel.lte(); rtel++ ) {
if
( rtel->value->targ != 0 )
depthFirstOrdering( rtel->value->targ );
}
}
void
RedFsmAp::depthFirstOrdering()
{
for
( RedStateList::Iter st = stateList; st.lte(); st++ )
st->onStateList =
false
;
int
stateListLen = stateList.length();
stateList.abandon();
if
( startState != 0 )
depthFirstOrdering( startState );
for
( RedStateSet::Iter en = entryPoints; en.lte(); en++ )
depthFirstOrdering( *en );
if
( forcedErrorState )
depthFirstOrdering( errState );
assert
( stateListLen == stateList.length() );
}
void
RedFsmAp::sequentialStateIds()
{
nextStateId = 0;
for
( RedStateList::Iter st = stateList; st.lte(); st++ )
st->id = nextStateId++;
}
void
RedFsmAp::sortStatesByFinal()
{
RedStateAp *state = 0;
RedStateAp *next = stateList.head;
RedStateAp *last = stateList.tail;
while
( state != last ) {
state = next;
next = state->next;
if
( state->isFinal ) {
stateList.detach( state );
stateList.append( state );
}
}
}
void
RedFsmAp::sortStateIdsByFinal()
{
nextStateId = 0;
for
( RedStateList::Iter st = stateList; st.lte(); st++ ) {
if
( ! st->isFinal )
st->id = nextStateId++;
}
for
( RedStateList::Iter st = stateList; st.lte(); st++ ) {
if
( st->isFinal )
st->id = nextStateId++;
}
}
struct
CmpStateById
{
static
int
compare( RedStateAp *st1, RedStateAp *st2 )
{
if
( st1->id < st2->id )
return
-1;
else
if
( st1->id > st2->id )
return
1;
else
return
0;
}
};
void
RedFsmAp::sortByStateId()
{
int
pos = 0;
RedStateAp **ptrList =
new
RedStateAp*[stateList.length()];
for
( RedStateList::Iter st = stateList; st.lte(); st++, pos++ )
ptrList[pos] = st;
MergeSort<RedStateAp*, CmpStateById> mergeSort;
mergeSort.sort( ptrList, stateList.length() );
stateList.abandon();
for
(
int
st = 0; st < pos; st++ )
stateList.append( ptrList[st] );
delete
[] ptrList;
}
void
RedFsmAp::findFirstFinState()
{
for
( RedStateList::Iter st = stateList; st.lte(); st++ ) {
if
( st->isFinal && (firstFinState == 0 || st->id < firstFinState->id) )
firstFinState = st;
}
}
void
RedFsmAp::assignActionLocs()
{
int
nextLocation = 0;
for
( GenActionTableMap::Iter act = actionMap; act.lte(); act++ ) {
act->location = nextLocation;
nextLocation += act->key.length() + 1;
}
}
bool
RedFsmAp::canExtend(
const
RedTransList &list,
int
pos )
{
RedTransAp *extendTrans = list[pos].value;
for
(
int
next = pos + 1; next < list.length(); pos++, next++ ) {
Key nextKey = list[next].lowKey;
nextKey.decrement();
if
( list[pos].highKey != nextKey )
break
;
if
( extendTrans == list[next].value )
return
true
;
unsigned
long
long
nextSpan = keyOps->span( list[next].lowKey, list[next].highKey );
if
( nextSpan > 1 )
break
;
}
return
false
;
}
void
RedFsmAp::moveTransToSingle( RedStateAp *state )
{
RedTransList &range = state->outRange;
RedTransList &single = state->outSingle;
for
(
int
rpos = 0; rpos < range.length(); ) {
if
( canExtend( range, rpos ) ) {
while
( range[rpos].value != range[rpos+1].value ) {
single.append( range[rpos+1] );
range.
remove
( rpos+1 );
}
range[rpos].highKey = range[rpos+1].highKey;
range.
remove
( rpos+1 );
}
else
if
( keyOps->span( range[rpos].lowKey, range[rpos].highKey ) == 1 ) {
single.append( range[rpos] );
range.
remove
( rpos );
}
else
{
rpos += 1;
}
}
}
void
RedFsmAp::chooseSingle()
{
for
( RedStateList::Iter st = stateList; st.lte(); st++ ) {
moveTransToSingle( st );
}
}
void
RedFsmAp::makeFlat()
{
for
( RedStateList::Iter st = stateList; st.lte(); st++ ) {
if
( st->stateCondList.length() == 0 ) {
st->condLowKey = 0;
st->condHighKey = 0;
}
else
{
st->condLowKey = st->stateCondList.head->lowKey;
st->condHighKey = st->stateCondList.tail->highKey;
unsigned
long
long
span = keyOps->span( st->condLowKey, st->condHighKey );
st->condList =
new
GenCondSpace*[ span ];
memset
( st->condList, 0,
sizeof
(GenCondSpace*)*span );
for
( GenStateCondList::Iter sci = st->stateCondList; sci.lte(); sci++ ) {
unsigned
long
long
base, trSpan;
base = keyOps->span( st->condLowKey, sci->lowKey )-1;
trSpan = keyOps->span( sci->lowKey, sci->highKey );
for
( unsigned
long
long
pos = 0; pos < trSpan; pos++ )
st->condList[base+pos] = sci->condSpace;
}
}
if
( st->outRange.length() == 0 ) {
st->lowKey = st->highKey = 0;
st->transList = 0;
}
else
{
st->lowKey = st->outRange[0].lowKey;
st->highKey = st->outRange[st->outRange.length()-1].highKey;
unsigned
long
long
span = keyOps->span( st->lowKey, st->highKey );
st->transList =
new
RedTransAp*[ span ];
memset
( st->transList, 0,
sizeof
(RedTransAp*)*span );
for
( RedTransList::Iter trans = st->outRange; trans.lte(); trans++ ) {
unsigned
long
long
base, trSpan;
base = keyOps->span( st->lowKey, trans->lowKey )-1;
trSpan = keyOps->span( trans->lowKey, trans->highKey );
for
( unsigned
long
long
pos = 0; pos < trSpan; pos++ )
st->transList[base+pos] = trans->value;
}
for
( unsigned
long
long
pos = 0; pos < span; pos++ ) {
if
( st->transList[pos] == 0 )
st->transList[pos] = st->defTrans;
}
}
}
}
void
RedFsmAp::moveToDefault( RedTransAp *defTrans, RedStateAp *state )
{
RedTransList outRange;
for
( RedTransList::Iter rtel = state->outRange; rtel.lte(); rtel++ ) {
if
( rtel->value != defTrans )
outRange.append( *rtel );
}
state->outRange.transfer( outRange );
state->defTrans = defTrans;
}
bool
RedFsmAp::alphabetCovered( RedTransList &outRange )
{
if
( outRange.length() == 0 )
return
false
;
RedTransList::Iter rtel = outRange;
if
( keyOps->minKey < rtel->lowKey )
return
false
;
rtel.increment();
for
( ; rtel.lte(); rtel++ ) {
Key highKey = rtel[-1].highKey;
highKey.increment();
if
( highKey != rtel->lowKey )
return
false
;
}
RedTransEl *last = &outRange[outRange.length()-1];
if
( last->highKey < keyOps->maxKey )
return
false
;
return
true
;
}
RedTransAp *RedFsmAp::chooseDefaultSpan( RedStateAp *state )
{
RedTransSet stateTransSet;
for
( RedTransList::Iter rtel = state->outRange; rtel.lte(); rtel++ )
stateTransSet.insert( rtel->value );
unsigned
long
long
*span =
new
unsigned
long
long
[stateTransSet.length()];
memset
( span, 0,
sizeof
(unsigned
long
long
) * stateTransSet.length() );
for
( RedTransList::Iter rtel = state->outRange; rtel.lte(); rtel++ ) {
RedTransAp **inSet = stateTransSet.find( rtel->value );
int
pos = inSet - stateTransSet.data;
span[pos] += keyOps->span( rtel->lowKey, rtel->highKey );
}
RedTransAp *maxTrans = 0;
unsigned
long
long
maxSpan = 0;
for
( RedTransSet::Iter rtel = stateTransSet; rtel.lte(); rtel++ ) {
if
( span[rtel.pos()] > maxSpan ) {
maxSpan = span[rtel.pos()];
maxTrans = *rtel;
}
}
delete
[] span;
return
maxTrans;
}
void
RedFsmAp::chooseDefaultSpan()
{
for
( RedStateList::Iter st = stateList; st.lte(); st++ ) {
if
( alphabetCovered( st->outRange ) ) {
RedTransAp *defTrans = chooseDefaultSpan( st );
moveToDefault( defTrans, st );
}
}
}
RedTransAp *RedFsmAp::chooseDefaultGoto( RedStateAp *state )
{
RedTransSet stateTransSet;
for
( RedTransList::Iter rtel = state->outRange; rtel.lte(); rtel++ ) {
if
( rtel->value->targ == state->next )
return
rtel->value;
}
return
0;
}
void
RedFsmAp::chooseDefaultGoto()
{
for
( RedStateList::Iter st = stateList; st.lte(); st++ ) {
RedTransAp *defTrans = chooseDefaultGoto( st );
if
( defTrans == 0 )
defTrans = chooseDefaultSpan( st );
moveToDefault( defTrans, st );
}
}
RedTransAp *RedFsmAp::chooseDefaultNumRanges( RedStateAp *state )
{
RedTransSet stateTransSet;
for
( RedTransList::Iter rtel = state->outRange; rtel.lte(); rtel++ )
stateTransSet.insert( rtel->value );
int
*numRanges =
new
int
[stateTransSet.length()];
memset
( numRanges, 0,
sizeof
(
int
) * stateTransSet.length() );
for
( RedTransList::Iter rtel = state->outRange; rtel.lte(); rtel++ ) {
RedTransAp **inSet = stateTransSet.find( rtel->value );
numRanges[inSet - stateTransSet.data] += 1;
}
RedTransAp *maxTrans = 0;
int
maxNumRanges = 0;
for
( RedTransSet::Iter rtel = stateTransSet; rtel.lte(); rtel++ ) {
if
( numRanges[rtel.pos()] > maxNumRanges ) {
maxNumRanges = numRanges[rtel.pos()];
maxTrans = *rtel;
}
}
delete
[] numRanges;
return
maxTrans;
}
void
RedFsmAp::chooseDefaultNumRanges()
{
for
( RedStateList::Iter st = stateList; st.lte(); st++ ) {
RedTransAp *defTrans = chooseDefaultNumRanges( st );
moveToDefault( defTrans, st );
}
}
RedTransAp *RedFsmAp::getErrorTrans( )
{
if
( errTrans == 0 ) {
errTrans =
new
RedTransAp( getErrorState(), 0, nextTransId++ );
RedTransAp *inRes = transSet.insert( errTrans );
assert
( inRes != 0 );
}
return
errTrans;
}
RedStateAp *RedFsmAp::getErrorState()
{
assert
( errState != 0 );
return
errState;
}
RedTransAp *RedFsmAp::allocateTrans( RedStateAp *targ, RedAction *action )
{
RedTransAp redTrans( targ, action, 0 );
RedTransAp *inDict = transSet.find( &redTrans );
if
( inDict == 0 ) {
inDict =
new
RedTransAp( targ, action, nextTransId++ );
transSet.insert( inDict );
}
return
inDict;
}
void
RedFsmAp::partitionFsm(
int
nparts )
{
this
->nParts = nparts;
int
partSize = stateList.length() / nparts;
int
remainder = stateList.length() % nparts;
int
numInPart = partSize;
int
partition = 0;
if
( remainder-- > 0 )
numInPart += 1;
for
( RedStateList::Iter st = stateList; st.lte(); st++ ) {
st->partition = partition;
numInPart -= 1;
if
( numInPart == 0 ) {
partition += 1;
numInPart = partSize;
if
( remainder-- > 0 )
numInPart += 1;
}
}
}
void
RedFsmAp::setInTrans()
{
for
( TransApSet::Iter trans = transSet; trans.lte(); trans++ )
trans->targ->numInTrans += 1;
for
( RedStateList::Iter st = stateList; st.lte(); st++ ) {
st->inTrans =
new
RedTransAp*[st->numInTrans];
st->numInTrans = 0;
}
for
( TransApSet::Iter trans = transSet; trans.lte(); trans++ )
trans->targ->inTrans[trans->targ->numInTrans++] = trans;
}