diff options
author | Alan Mishchenko <alanmi@berkeley.edu> | 2015-10-08 23:27:56 -0700 |
---|---|---|
committer | Alan Mishchenko <alanmi@berkeley.edu> | 2015-10-08 23:27:56 -0700 |
commit | 1ca82c87b49312fdf0091a77db67f55fd53fba97 (patch) | |
tree | 9dc1fbf0149ad584162004cca2c37fafdad91d37 /src/opt/sfm | |
parent | 46223f903b69349135ccb1345720f2b4d6530a71 (diff) | |
download | abc-1ca82c87b49312fdf0091a77db67f55fd53fba97.tar.gz abc-1ca82c87b49312fdf0091a77db67f55fd53fba97.tar.bz2 abc-1ca82c87b49312fdf0091a77db67f55fd53fba97.zip |
Experiments with functional matching.
Diffstat (limited to 'src/opt/sfm')
-rw-r--r-- | src/opt/sfm/sfm.h | 2 | ||||
-rw-r--r-- | src/opt/sfm/sfmDec.c | 629 |
2 files changed, 451 insertions, 180 deletions
diff --git a/src/opt/sfm/sfm.h b/src/opt/sfm/sfm.h index 1d97cd8b..38744ae1 100644 --- a/src/opt/sfm/sfm.h +++ b/src/opt/sfm/sfm.h @@ -43,8 +43,10 @@ typedef struct Sfm_Par_t_ Sfm_Par_t; struct Sfm_Par_t_ { int nTfoLevMax; // the maximum fanout levels + int nTfiLevMax; // the maximum fanin levels int nFanoutMax; // the maximum number of fanouts int nDepthMax; // the maximum depth to try + int nMffcMax; // the maximum MFFC size int nWinSizeMax; // the maximum window size int nGrowthLevel; // the maximum allowed growth in level int nBTLimit; // the maximum number of conflicts in one SAT run diff --git a/src/opt/sfm/sfmDec.c b/src/opt/sfm/sfmDec.c index b4376718..fd9109c3 100644 --- a/src/opt/sfm/sfmDec.c +++ b/src/opt/sfm/sfmDec.c @@ -46,19 +46,25 @@ struct Sfm_Dec_t_ int GateConst1; // special gates int GateBuffer; // special gates int GateInvert; // special gates + int GateAnd[4]; // special gates + int GateOr[4]; // special gates // objects + int nDivs; // the number of divisors + int nMffc; // the number of divisors int iTarget; // target node - Vec_Int_t vObjTypes; // PI (1), PO (2) + Vec_Int_t vObjRoots; // roots of the window Vec_Int_t vObjGates; // functionality Vec_Wec_t vObjFanins; // fanin IDs + Vec_Int_t vObjMap; // object map + Vec_Int_t vObjDec; // decomposition // solver sat_solver * pSat; // reusable solver Vec_Wec_t vClauses; // CNF clauses for the node - Vec_Int_t vPols[2]; // onset/offset polarity - Vec_Int_t vTaken[2]; // onset/offset implied nodes Vec_Int_t vImpls[2]; // onset/offset implications + Vec_Int_t vCounts[2]; // onset/offset counters Vec_Wrd_t vSets[2]; // onset/offset patterns - int nPats[3]; + int nPats[2]; // CEX count + word uMask[2]; // mask count // temporary Vec_Int_t vTemp; Vec_Int_t vTemp2; @@ -74,6 +80,32 @@ struct Sfm_Dec_t_ /**Function************************************************************* + Synopsis [Setup parameter structure.] + + Description [] + + SideEffects [] + + SeeAlso [] + +***********************************************************************/ +void Sfm_ParSetDefault3( Sfm_Par_t * pPars ) +{ + memset( pPars, 0, sizeof(Sfm_Par_t) ); + pPars->nTfoLevMax = 1000; // the maximum fanout levels + pPars->nTfiLevMax = 1000; // the maximum fanin levels + pPars->nFanoutMax = 30; // the maximum number of fanoutsp + pPars->nMffcMax = 3; // the maximum MFFC size + pPars->nWinSizeMax = 300; // the maximum window size + pPars->nGrowthLevel = 0; // the maximum allowed growth in level + pPars->nBTLimit = 5000; // the maximum number of conflicts in one SAT run + pPars->fArea = 0; // performs optimization for area + pPars->fVerbose = 0; // enable basic stats + pPars->fVeryVerbose = 0; // enable detailed stats +} + +/**Function************************************************************* + Synopsis [] Description [] @@ -83,9 +115,10 @@ struct Sfm_Dec_t_ SeeAlso [] ***********************************************************************/ -Sfm_Dec_t * Sfm_DecStart() +Sfm_Dec_t * Sfm_DecStart( Sfm_Par_t * pPars ) { Sfm_Dec_t * p = ABC_CALLOC( Sfm_Dec_t, 1 ); + p->pPars = pPars; p->pSat = sat_solver_new(); return p; } @@ -97,18 +130,18 @@ void Sfm_DecStop( Sfm_Dec_t * p ) Vec_WecErase( &p->vGateCnfs ); Vec_PtrErase( &p->vGateHands ); // objects - Vec_IntErase( &p->vObjTypes ); + Vec_IntErase( &p->vObjRoots ); Vec_IntErase( &p->vObjGates ); Vec_WecErase( &p->vObjFanins ); + Vec_IntErase( &p->vObjMap ); + Vec_IntErase( &p->vObjDec ); // solver sat_solver_delete( p->pSat ); Vec_WecErase( &p->vClauses ); - Vec_IntErase( &p->vPols[0] ); - Vec_IntErase( &p->vPols[1] ); - Vec_IntErase( &p->vTaken[0] ); - Vec_IntErase( &p->vTaken[1] ); Vec_IntErase( &p->vImpls[0] ); Vec_IntErase( &p->vImpls[1] ); + Vec_IntErase( &p->vCounts[0] ); + Vec_IntErase( &p->vCounts[1] ); Vec_WrdErase( &p->vSets[0] ); Vec_WrdErase( &p->vSets[1] ); // temporary @@ -131,32 +164,25 @@ void Sfm_DecStop( Sfm_Dec_t * p ) int Sfm_DecPrepareSolver( Sfm_Dec_t * p ) { abctime clk = Abc_Clock(); - Vec_Int_t * vRoots = &p->vTemp; + Vec_Int_t * vRoots = &p->vObjRoots; Vec_Int_t * vFaninVars = &p->vTemp2; Vec_Int_t * vLevel, * vClause; - int i, k, Type, Gate, iObj, RetValue; + int i, k, Gate, iObj, RetValue; int nTfiSize = p->iTarget + 1; // including node - int nWinSize = Vec_IntSize(&p->vObjTypes); + int nWinSize = Vec_IntSize(&p->vObjGates); int nSatVars = 2 * nWinSize - nTfiSize; assert( nWinSize == Vec_IntSize(&p->vObjGates) ); assert( p->iTarget < nWinSize ); - // collect variables of root nodes - Vec_IntClear( vRoots ); - Vec_IntForEachEntryStart( &p->vObjTypes, Type, i, p->iTarget ) - if ( Type == 2 ) - Vec_IntPush( vRoots, i ); - assert( Vec_IntSize(vRoots) > 0 ); // create SAT solver sat_solver_restart( p->pSat ); sat_solver_setnvars( p->pSat, nSatVars + Vec_IntSize(vRoots) ); // add CNF clauses for the TFI - Vec_IntForEachEntryStop( &p->vObjTypes, Type, i, nTfiSize ) + Vec_IntForEachEntry( &p->vObjGates, Gate, i ) { - if ( Type == 1 ) + if ( Gate == -1 ) continue; - vLevel = Vec_WecEntry( &p->vObjFanins, i ); // generate CNF - Gate = Vec_IntEntry( &p->vObjGates, i ); + vLevel = Vec_WecEntry( &p->vObjFanins, i ); Vec_IntPush( vLevel, i ); Sfm_TranslateCnf( &p->vClauses, (Vec_Str_t *)Vec_WecEntry(&p->vGateCnfs, Gate), vLevel, -1 ); Vec_IntPop( vLevel ); @@ -171,16 +197,15 @@ int Sfm_DecPrepareSolver( Sfm_Dec_t * p ) } } // add CNF clauses for the TFO - Vec_IntForEachEntryStart( &p->vObjTypes, Type, i, nTfiSize ) + Vec_IntForEachEntryStart( &p->vObjGates, Gate, i, nTfiSize ) { - assert( Type != 1 ); + assert( Gate != -1 ); vLevel = Vec_WecEntry( &p->vObjFanins, i ); Vec_IntClear( vFaninVars ); Vec_IntForEachEntry( vLevel, iObj, k ) Vec_IntPush( vFaninVars, iObj <= p->iTarget ? iObj : iObj + nWinSize - nTfiSize ); Vec_IntPush( vFaninVars, i + nWinSize - nTfiSize ); // generate CNF - Gate = Vec_IntEntry( &p->vObjGates, i ); Sfm_TranslateCnf( &p->vClauses, (Vec_Str_t *)Vec_WecEntry(&p->vGateCnfs, Gate), vFaninVars, p->iTarget ); // add clauses Vec_WecForEachLevel( &p->vClauses, vClause, k ) @@ -192,16 +217,17 @@ int Sfm_DecPrepareSolver( Sfm_Dec_t * p ) return 0; } } - if ( p->iTarget + 1 < nWinSize ) + if ( nTfiSize < nWinSize ) { // create XOR clauses for the roots + Vec_IntClear( vFaninVars ); Vec_IntForEachEntry( vRoots, iObj, i ) { + Vec_IntPush( vFaninVars, Abc_Var2Lit(nSatVars, 0) ); sat_solver_add_xor( p->pSat, iObj, iObj + nWinSize - nTfiSize, nSatVars++, 0 ); - Vec_IntWriteEntry( vRoots, i, Abc_Var2Lit(nSatVars-1, 0) ); } // make OR clause for the last nRoots variables - RetValue = sat_solver_addclause( p->pSat, Vec_IntArray(vRoots), Vec_IntLimit(vRoots) ); + RetValue = sat_solver_addclause( p->pSat, Vec_IntArray(vFaninVars), Vec_IntLimit(vFaninVars) ); if ( RetValue == 0 ) return 0; assert( nSatVars == sat_solver_nvars(p->pSat) ); @@ -224,55 +250,66 @@ int Sfm_DecPrepareSolver( Sfm_Dec_t * p ) SeeAlso [] ***********************************************************************/ -int Sfm_DecPeformDec( Sfm_Dec_t * p ) +int Sfm_DecFindWeight( Sfm_Dec_t * p, int c, int iLit ) { - int fVerbose = 1; + int Value = Vec_IntEntry( &p->vCounts[!c], Abc_Lit2Var(iLit) ); + return Abc_LitIsCompl(iLit) ? Value : p->nPats[!c] - Value; +} +int Sfm_DecPeformDecOne( Sfm_Dec_t * p, int * pfConst ) +{ + int fVerbose = p->pPars->fVeryVerbose; int nBTLimit = 0; abctime clk = Abc_Clock(); - int i, j, k, c, n, Pol, Pol2, Entry, Entry2, status, Lits[3]; + int i, k, c, Entry, status, Lits[3], RetValue; + int iLitBest = -1, iCBest = -1, WeightBest = -1, Weight; + *pfConst = -1; // check stuck-at-0/1 (on/off-set empty) p->nPats[0] = p->nPats[1] = 0; + p->uMask[0] = p->uMask[1] = 0; + Vec_IntClear( &p->vImpls[0] ); + Vec_IntClear( &p->vImpls[1] ); + Vec_IntClear( &p->vCounts[0] ); + Vec_IntClear( &p->vCounts[1] ); + Vec_WrdClear( &p->vSets[0] ); + Vec_WrdClear( &p->vSets[1] ); for ( c = 0; c < 2; c++ ) { Lits[0] = Abc_Var2Lit( p->iTarget, c ); status = sat_solver_solve( p->pSat, Lits, Lits + 1, nBTLimit, 0, 0, 0 ); if ( status == l_Undef ) - return 0; + return -2; if ( status == l_False ) { - Vec_IntPush( &p->vObjTypes, 0 ); - Vec_IntPush( &p->vObjGates, c ? p->GateConst1 : p->GateConst0 ); - Vec_WecPushLevel( &p->vObjFanins ); - return 1; + *pfConst = c; + return -1; } assert( status == l_True ); // record this status for ( i = 0; i <= p->iTarget; i++ ) { - Vec_IntPush( &p->vPols[c], sat_solver_var_value(p->pSat, i) ); - Vec_WrdPush( &p->vSets[c], 0 ); + Entry = sat_solver_var_value(p->pSat, i); + Vec_IntPush( &p->vCounts[c], Entry ); + Vec_WrdPush( &p->vSets[c], (word)Entry ); } p->nPats[c]++; - Vec_IntClear( &p->vImpls[c] ); - Vec_IntFill( &p->vTaken[c], p->iTarget, 0 ); + p->uMask[c] = 1; } // proceed checking divisors based on their values for ( c = 0; c < 2; c++ ) { Lits[0] = Abc_Var2Lit( p->iTarget, c ); - for ( i = 0; i < p->iTarget; i++ ) + for ( i = 0; i < p->nDivs; i++ ) { - if ( Vec_WrdEntry(&p->vSets[c], i) ) // diff value is possible + word Column = Vec_WrdEntry(&p->vSets[c], i); + if ( Column != 0 && Column != p->uMask[c] ) // diff value is possible continue; - Pol = Vec_IntEntry(&p->vPols[c], i); - Lits[1] = Abc_Var2Lit( i, Pol ); + Lits[1] = Abc_Var2Lit( i, Column != 0 ); status = sat_solver_solve( p->pSat, Lits, Lits + 2, nBTLimit, 0, 0, 0 ); if ( status == l_Undef ) return 0; if ( status == l_False ) { - Vec_IntWriteEntry( &p->vTaken[c], i, 1 ); - Vec_IntPushTwo( &p->vImpls[c], Abc_Var2Lit(i, Pol), -1 ); + Vec_IntPush( &p->vImpls[c], Abc_LitNot(Lits[1]) ); continue; } assert( status == l_True ); @@ -280,73 +317,64 @@ int Sfm_DecPeformDec( Sfm_Dec_t * p ) continue; // record this status for ( k = 0; k <= p->iTarget; k++ ) - if ( sat_solver_var_value(p->pSat, k) ^ Vec_IntEntry(&p->vPols[c], k) ) + if ( sat_solver_var_value(p->pSat, k) ) + { + Vec_IntAddToEntry( &p->vCounts[c], k, 1 ); *Vec_WrdEntryP(&p->vSets[c], k) |= ((word)1 << p->nPats[c]); - p->nPats[c]++; + } + p->uMask[c] |= ((word)1 << p->nPats[c]++); } } - // proceed checking divisor pairs + // find the best decomposition for ( c = 0; c < 2; c++ ) { - Lits[0] = Abc_Var2Lit( p->iTarget, c ); - for ( i = 0; i < p->iTarget; i++ ) - if ( !Vec_IntEntry(&p->vTaken[c], i) ) - for ( j = 0; j < i; j++ ) - if ( !Vec_IntEntry(&p->vTaken[c], j) ) + Vec_IntForEachEntry( &p->vImpls[c], Entry, i ) { - word SignI = Vec_WrdEntry(&p->vSets[c], i); - word SignJ = Vec_WrdEntry(&p->vSets[c], j); - for ( n = 0; n < 3; n++ ) + Weight = Sfm_DecFindWeight(p, c, Entry); + if ( WeightBest < Weight ) { - if ( ((n&1) ? ~SignI : SignI) & ((n>>1) ? ~SignJ : SignJ) ) // diff value is possible - continue; - Pol = Vec_IntEntry(&p->vPols[c], i) ^ (n&1); - Pol2 = Vec_IntEntry(&p->vPols[c], j) ^ (n>>1); - Lits[1] = Abc_Var2Lit( i, Pol ); - Lits[2] = Abc_Var2Lit( j, Pol2 ); - status = sat_solver_solve( p->pSat, Lits, Lits + 3, nBTLimit, 0, 0, 0 ); - if ( status == l_Undef ) - return 0; - if ( status == l_False ) - { - Vec_IntPushTwo( &p->vImpls[c], Abc_Var2Lit(i, Pol), Abc_Var2Lit(j, Pol2) ); - continue; - } - assert( status == l_True ); - if ( p->nPats[c] == 64 ) - continue; - // record this status - for ( k = 0; k <= p->iTarget; k++ ) - if ( sat_solver_var_value(p->pSat, k) ^ Vec_IntEntry(&p->vPols[c], k) ) - *Vec_WrdEntryP(&p->vSets[c], k) |= ((word)1 << p->nPats[c]); - p->nPats[c]++; + WeightBest = Weight; + iLitBest = Entry; + iCBest = c; } } } + if ( WeightBest == -1 ) + return -2; + + // add clause + Lits[0] = Abc_Var2Lit( p->iTarget, iCBest ); + Lits[1] = iLitBest; + RetValue = sat_solver_addclause( p->pSat, Lits, Lits + 2 ); + if ( RetValue == 0 ) + return -1; + + p->timeSat += Abc_Clock() - clk; // print the results - if ( fVerbose ) + if ( !fVerbose ) + return Abc_Var2Lit( iLitBest, iCBest ); + + printf( "\nBest literal (%d; %s%d) with weight %d.\n\n", iCBest, Abc_LitIsCompl(iLitBest)? "!":"", Abc_Lit2Var(iLitBest), WeightBest ); + for ( c = 0; c < 2; c++ ) { Vec_Int_t * vLevel = Vec_WecEntry( &p->vObjFanins, p->iTarget ); - printf( "\n%s-SET of object %d with gate \"%s\" and fanins: ", c ? "OFF": "ON", p->iTarget, Mio_GateReadName((Mio_Gate_t *)Vec_PtrEntry(&p->vGateHands, Vec_IntEntry(&p->vObjGates,p->iTarget))) ); + printf( "\n%s-SET of object %d (divs = %d) with gate \"%s\" and fanins: ", + c ? "OFF": "ON", p->iTarget, p->nDivs, + Mio_GateReadName((Mio_Gate_t *)Vec_PtrEntry(&p->vGateHands, Vec_IntEntry(&p->vObjGates,p->iTarget))) ); Vec_IntForEachEntry( vLevel, Entry, i ) printf( "%d ", Entry ); printf( "\n" ); printf( "Implications: " ); - Vec_IntForEachEntryDouble( &p->vImpls[c], Entry, Entry2, i ) - { - if ( Entry2 == -1 ) - printf( "%s%d ", Abc_LitIsCompl(Entry)? "!":"", Abc_Lit2Var(Entry) ); - else - printf( "%s%d:%s%d ", Abc_LitIsCompl(Entry)? "!":"", Abc_Lit2Var(Entry), Abc_LitIsCompl(Entry2)? "!":"", Abc_Lit2Var(Entry2) ); - } + Vec_IntForEachEntry( &p->vImpls[c], Entry, i ) + printf( "%s%d(%d) ", Abc_LitIsCompl(Entry)? "!":"", Abc_Lit2Var(Entry), Sfm_DecFindWeight(p, c, Entry) ); printf( "\n" ); printf( " " ); for ( i = 0; i <= p->iTarget; i++ ) - printf( "%d", Vec_IntEntry(&p->vPols[c], i) ); - printf( "\n\n" ); + printf( "%d", i / 10 ); + printf( "\n" ); printf( " " ); for ( i = 0; i <= p->iTarget; i++ ) printf( "%d", i % 10 ); @@ -360,8 +388,117 @@ int Sfm_DecPeformDec( Sfm_Dec_t * p ) } printf( "\n" ); } - p->timeSat += Abc_Clock() - clk; - return 1; + return Abc_Var2Lit( iLitBest, iCBest ); +} +int Sfm_DecPeformDec( Sfm_Dec_t * p, Mio_Library_t * pLib ) +{ + Vec_Int_t * vLevel; + int i, Dec, Last, fCompl, Pol, fConst = -1, nNodes = Vec_IntSize(&p->vObjGates); + // perform decomposition + Vec_IntClear( &p->vObjDec ); + for ( i = 0; i <= p->nMffc; i++ ) + { + Dec = Sfm_DecPeformDecOne( p, &fConst ); + if ( Dec == -2 ) + { + if ( p->pPars->fVerbose ) + printf( "There is no decomposition (or time out occurred).\n" ); + return -1; + } + if ( Dec == -1 ) + break; + Vec_IntPush( &p->vObjDec, Dec ); + } + if ( i == p->nMffc + 1 ) + { + if ( p->pPars->fVerbose ) + printf( "Area-reducing decomposition is not found.\n" ); + return -1; + } + // check constant + if ( Vec_IntSize(&p->vObjDec) == 0 ) + { + assert( fConst >= 0 ); + // report + if ( p->pPars->fVerbose ) + printf( "Create constant %d.\n", fConst ); + // add gate + Vec_IntPush( &p->vObjGates, fConst ? p->GateConst1 : p->GateConst0 ); + vLevel = Vec_WecPushLevel( &p->vObjFanins ); + return Vec_IntSize(&p->vObjDec); + } + // create network + Last = Vec_IntPop( &p->vObjDec ); + fCompl = Abc_LitIsCompl(Last); + Last = Abc_LitNotCond( Abc_Lit2Var(Last), fCompl ); + if ( Vec_IntSize(&p->vObjDec) == 0 ) + { + // report + if ( p->pPars->fVerbose ) +// printf( "Create node %d = %s%d.\n", nNodes, (Abc_LitIsCompl(Last) ^ fCompl) ? "!":"", Abc_Lit2Var(Last) ); + printf( "Create node %d = %s%d.\n", nNodes, Abc_LitIsCompl(Last) ? "!":"", Abc_Lit2Var(Last) ); + // add gate +// Vec_IntPush( &p->vObjGates, (Abc_LitIsCompl(Last) ^ fCompl) ? p->GateInvert : p->GateBuffer ); + Vec_IntPush( &p->vObjGates, Abc_LitIsCompl(Last) ? p->GateInvert : p->GateBuffer ); + vLevel = Vec_WecPushLevel( &p->vObjFanins ); + Vec_IntPush( vLevel, Abc_Lit2Var(Last) ); + return Vec_IntSize(&p->vObjDec); + } + Vec_IntForEachEntryReverse( &p->vObjDec, Dec, i ) + { + fCompl = Abc_LitIsCompl(Dec); +// Dec = Abc_Lit2Var(Dec); + Dec = Abc_LitNotCond( Abc_Lit2Var(Dec), fCompl ); + // add gate + Pol = (Abc_LitIsCompl(Last) << 1) | Abc_LitIsCompl(Dec); +// Pol = Abc_LitIsCompl(Dec); + if ( fCompl ) + Vec_IntPush( &p->vObjGates, p->GateOr[Pol] ); + else + Vec_IntPush( &p->vObjGates, p->GateAnd[Pol] ); + vLevel = Vec_WecPushLevel( &p->vObjFanins ); + Vec_IntPush( vLevel, Abc_Lit2Var(Dec) ); + Vec_IntPush( vLevel, Abc_Lit2Var(Last) ); + // report + if ( p->pPars->fVerbose ) + printf( "Create node %s%d = %s%d and %s%d (gate %d).\n", + fCompl? "!":"", nNodes, + Abc_LitIsCompl(Last)? "!":"", Abc_Lit2Var(Last), + Abc_LitIsCompl(Dec)? "!":"", Abc_Lit2Var(Dec), + Pol ); + // prepare for the next one + Last = Abc_Var2Lit( nNodes, 0 ); + nNodes++; + } + return Vec_IntSize(&p->vObjDec); +} + +/**Function************************************************************* + + Synopsis [Incremental level update.] + + Description [] + + SideEffects [] + + SeeAlso [] + +***********************************************************************/ +void Abc_NtkUpdateIncLevel_rec( Abc_Obj_t * pObj ) +{ + Abc_Obj_t * pFanout; + int i, LevelNew = Abc_ObjLevelNew(pObj); + if ( LevelNew == (int)pObj->Level ) + return; + pObj->Level = LevelNew; + if ( Abc_ObjIsCo(pObj) ) + return; + Abc_ObjForEachFanout( pObj, pFanout, i ) + Abc_NtkUpdateIncLevel_rec( pFanout ); +} +void Abc_NtkUpdateIncLevel( Abc_Obj_t * pObj ) +{ + Abc_NtkUpdateIncLevel_rec( pObj ); } /**Function************************************************************* @@ -375,102 +512,208 @@ int Sfm_DecPeformDec( Sfm_Dec_t * p ) SeeAlso [] ***********************************************************************/ -void Abc_NtkDfsReverseOne_rec( Abc_Obj_t * pNode, Vec_Int_t * vTfo ) +#define SFM_MASK_PI 1 // supp(node) is contained in supp(TFI(pivot)) +#define SFM_MASK_INPUT 2 // supp(node) does not overlap with supp(TFI(pivot)) +#define SFM_MASK_FANIN 4 // the same as above (pointed to by node with SFM_MASK_PI | SFM_MASK_INPUT) +#define SFM_MASK_MFFC 8 // MFFC nodes, including the target node + +void Abc_NtkDfsReverseOne_rec( Abc_Obj_t * pObj, Vec_Int_t * vTfo, int nLevelMax, int nFanoutMax ) { Abc_Obj_t * pFanout; int i; - if ( Abc_NodeIsTravIdCurrent( pNode ) ) + if ( Abc_NodeIsTravIdCurrent( pObj ) ) return; - Abc_NodeSetTravIdCurrent( pNode ); - if ( Abc_ObjIsCo(pNode) ) - { - Vec_IntPush( vTfo, Abc_ObjId(pNode) ); + Abc_NodeSetTravIdCurrent( pObj ); + if ( Abc_ObjIsCo(pObj) || Abc_ObjLevel(pObj) > nLevelMax ) return; + assert( Abc_ObjIsNode( pObj ) ); + if ( Abc_ObjFanoutNum(pObj) <= nFanoutMax ) + { + Abc_ObjForEachFanout( pObj, pFanout, i ) + if ( Abc_ObjIsCo(pFanout) ) + break; + if ( i == Abc_ObjFanoutNum(pObj) ) + Abc_ObjForEachFanout( pObj, pFanout, i ) + Abc_NtkDfsReverseOne_rec( pFanout, vTfo, nLevelMax, nFanoutMax ); } - assert( Abc_ObjIsNode( pNode ) ); - Abc_ObjForEachFanout( pNode, pFanout, i ) - Abc_NtkDfsReverseOne_rec( pFanout, vTfo ); - Vec_IntPush( vTfo, Abc_ObjId(pNode) ); + Vec_IntPush( vTfo, Abc_ObjId(pObj) ); + pObj->iTemp = 0; } -void Abc_NtkDfsOne_rec( Abc_Obj_t * pNode, Vec_Int_t * vTfi, Vec_Int_t * vTypes ) +int Abc_NtkDfsOne_rec( Abc_Obj_t * pObj, Vec_Int_t * vTfi, int nLevelMin, int CiLabel ) { - Abc_Obj_t * pFanin; int i; - if ( Abc_NodeIsTravIdCurrent( pNode ) ) - return; - Abc_NodeSetTravIdCurrent( pNode ); - if ( Abc_ObjIsCi(pNode) ) + Abc_Obj_t * pFanin; int i, Mask = 0; + if ( Abc_NodeIsTravIdCurrent( pObj ) ) + return pObj->iTemp; + Abc_NodeSetTravIdCurrent( pObj ); + if ( Abc_ObjIsCi(pObj) || Abc_ObjLevel(pObj) < nLevelMin ) { - pNode->iTemp = Vec_IntSize(vTfi); - Vec_IntPush( vTfi, Abc_ObjId(pNode) ); - Vec_IntPush( vTypes, 1 ); - return; + Vec_IntPush( vTfi, Abc_ObjId(pObj) ); + return (pObj->iTemp = CiLabel); } - assert( Abc_ObjIsNode(pNode) ); - Abc_ObjForEachFanin( pNode, pFanin, i ) - Abc_NtkDfsOne_rec( pFanin, vTfi, vTypes ); - pNode->iTemp = Vec_IntSize(vTfi); - Vec_IntPush( vTfi, Abc_ObjId(pNode) ); - Vec_IntPush( vTypes, 0 ); + assert( Abc_ObjIsNode(pObj) ); + Abc_ObjForEachFanin( pObj, pFanin, i ) + Mask |= Abc_NtkDfsOne_rec( pFanin, vTfi, nLevelMin, CiLabel ); + Vec_IntPush( vTfi, Abc_ObjId(pObj) ); + //assert( Mask > 0 ); + return (pObj->iTemp = Mask); +} +void Sfm_DecAddNode( Abc_Obj_t * pObj, Vec_Int_t * vMap, Vec_Int_t * vGates, int fSkip, int fVeryVerbose ) +{ + if ( fVeryVerbose ) + printf( "%d:%d(%d) ", Vec_IntSize(vMap), Abc_ObjId(pObj), pObj->iTemp ); + if ( fVeryVerbose ) + Abc_ObjPrint( stdout, pObj ); + Vec_IntPush( vMap, Abc_ObjId(pObj) ); + Vec_IntPush( vGates, fSkip ? -1 : Mio_GateReadValue((Mio_Gate_t *)pObj->pData) ); } -int Sfm_DecExtract( Abc_Ntk_t * pNtk, Abc_Obj_t * pNode, Vec_Int_t * vTypes, Vec_Int_t * vGates, Vec_Wec_t * vFanins, Vec_Int_t * vMap, Vec_Int_t * vTfo ) +int Sfm_DecMarkMffc( Abc_Obj_t * pPivot, int nMffcMax, int fVeryVerbose ) +{ + Abc_Obj_t * pFanin, * pFanin2; + int i, k, nMffc = 1; + pPivot->iTemp |= SFM_MASK_MFFC; +if ( fVeryVerbose ) +printf( "Mffc = %d.\n", pPivot->Id ); + Abc_ObjForEachFanin( pPivot, pFanin, i ) + if ( Abc_ObjIsNode(pFanin) && Abc_ObjFanoutNum(pFanin) == 1 && Abc_NodeIsTravIdCurrent(pFanin) ) + { + if ( nMffc == nMffcMax ) + return nMffc; + pFanin->iTemp |= SFM_MASK_MFFC; + nMffc++; +if ( fVeryVerbose ) +printf( "Mffc = %d.\n", pFanin->Id ); + } + Abc_ObjForEachFanin( pPivot, pFanin, i ) + if ( Abc_ObjIsNode(pFanin) && Abc_ObjFanoutNum(pFanin) == 1 && Abc_NodeIsTravIdCurrent(pFanin) ) + { + if ( nMffc == nMffcMax ) + return nMffc; + Abc_ObjForEachFanin( pFanin, pFanin2, k ) + if ( Abc_ObjIsNode(pFanin2) && Abc_ObjFanoutNum(pFanin2) == 1 && Abc_NodeIsTravIdCurrent(pFanin2) ) + { + if ( nMffc == nMffcMax ) + return nMffc; + pFanin2->iTemp |= SFM_MASK_MFFC; + nMffc++; +if ( fVeryVerbose ) +printf( "Mffc = %d.\n", pFanin2->Id ); + } + } + return nMffc; +} +int Abc_NtkDfsCheck_rec( Abc_Obj_t * pObj, Abc_Obj_t * pPivot ) +{ + Abc_Obj_t * pFanin; int i; + if ( pObj == pPivot ) + return 0; + if ( Abc_NodeIsTravIdCurrent( pObj ) ) + return 1; + Abc_NodeSetTravIdCurrent( pObj ); + if ( Abc_ObjIsCi(pObj) ) + return 1; + assert( Abc_ObjIsNode(pObj) ); + Abc_ObjForEachFanin( pObj, pFanin, i ) + if ( !Abc_NtkDfsCheck_rec(pFanin, pPivot) ) + return 0; + return 1; +} + +int Sfm_DecExtract( Abc_Ntk_t * pNtk, Sfm_Par_t * pPars, Abc_Obj_t * pPivot, Vec_Int_t * vRoots, Vec_Int_t * vGates, Vec_Wec_t * vFanins, Vec_Int_t * vMap, Vec_Int_t * vTfi, Vec_Int_t * vTfo, int * pnMffc ) { Vec_Int_t * vLevel; - Abc_Obj_t * pFanin; - int i, k, iObj, iTarget; - assert( Abc_ObjIsNode(pNode) ); - // collect transitive fanout including COs + Abc_Obj_t * pObj, * pFanin; + int nLevelMax = pPivot->Level + pPars->nTfoLevMax; + int nLevelMin = pPivot->Level - pPars->nTfiLevMax; + int i, k, nTfiSize, nDivs = -1; + assert( Abc_ObjIsNode(pPivot) ); +if ( pPars->fVerbose ) +printf( "\n\nTarget %d\n", Abc_ObjId(pPivot) ); + // collect TFO nodes Vec_IntClear( vTfo ); Abc_NtkIncrementTravId( pNtk ); - Abc_NtkDfsReverseOne_rec( pNode, vTfo ); - // collect transitive fanin - Vec_IntClear( vMap ); - Vec_IntClear( vTypes ); + Abc_NtkDfsReverseOne_rec( pPivot, vTfo, nLevelMax, pPars->nFanoutMax ); + // count internal fanouts + Abc_NtkForEachObjVec( vTfo, pNtk, pObj, i ) + Abc_ObjForEachFanin( pObj, pFanin, k ) + pFanin->iTemp++; + // comput roots + Vec_IntClear( vRoots ); + Abc_NtkForEachObjVec( vTfo, pNtk, pObj, i ) + if ( pObj->iTemp != Abc_ObjFanoutNum(pObj) ) + Vec_IntPush( vRoots, Abc_ObjId(pObj) ); + assert( Vec_IntSize(vRoots) > 0 ); + // collect TFI and mark nodes + Vec_IntClear( vTfi ); Abc_NtkIncrementTravId( pNtk ); - Abc_NtkDfsOne_rec( pNode, vMap, vTypes ); - Vec_IntPop( vMap ); - Vec_IntPop( vTypes ); - assert( Vec_IntSize(vMap) == Vec_IntSize(vTypes) ); - // remember target node - iTarget = Vec_IntSize( vMap ); - // add transitive fanout - Vec_IntForEachEntryReverse( vTfo, iObj, i ) - { - pNode = Abc_NtkObj( pNtk, iObj ); - if ( Abc_ObjIsCo(pNode) ) - { - assert( Vec_IntEntry(vTypes, Abc_ObjFanin0(pNode)->iTemp) == 0 ); // CO points to a unique node - Vec_IntWriteEntry( vTypes, Abc_ObjFanin0(pNode)->iTemp, 2 ); - continue; - } - pNode->iTemp = Vec_IntSize(vMap); - Vec_IntPush( vMap, Abc_ObjId(pNode) ); - Vec_IntPush( vTypes, 0 ); - } - assert( Vec_IntSize(vMap) == Vec_IntSize(vTypes) ); - // create gates and fanins + Abc_NtkDfsOne_rec( pPivot, vTfi, nLevelMin, SFM_MASK_PI ); + nTfiSize = Vec_IntSize(vTfi); + // additinally mark MFFC + *pnMffc = Sfm_DecMarkMffc( pPivot, pPars->nMffcMax, pPars->fVeryVerbose ); + assert( *pnMffc <= pPars->nMffcMax ); +if ( pPars->fVerbose ) +printf( "Mffc size = %d.\n", *pnMffc ); + // collect TFI(TFO) + Abc_NtkForEachObjVec( vTfo, pNtk, pObj, i ) + Abc_NtkDfsOne_rec( pObj, vTfi, nLevelMin, SFM_MASK_INPUT ); + // mark input-only nodes pointed to by mixed nodes + Abc_NtkForEachObjVecStart( vTfi, pNtk, pObj, i, nTfiSize ) + if ( pObj->iTemp != SFM_MASK_INPUT ) + Abc_ObjForEachFanin( pObj, pFanin, k ) + if ( pFanin->iTemp == SFM_MASK_INPUT ) + pFanin->iTemp = SFM_MASK_FANIN; + // collect nodes supported only on TFI fanins and not MFFC + Vec_IntClear( vMap ); Vec_IntClear( vGates ); + Abc_NtkForEachObjVec( vTfi, pNtk, pObj, i ) + if ( pObj->iTemp == SFM_MASK_PI ) + Sfm_DecAddNode( pObj, vMap, vGates, Abc_ObjIsCi(pObj) || Abc_ObjLevel(pObj) < nLevelMin, pPars->fVeryVerbose ); + nDivs = Vec_IntSize(vMap); +if ( pPars->fVeryVerbose ) +printf( "\nFinish divs\n" ); + // add other nodes that are not in TFO and not in MFFC + Abc_NtkForEachObjVec( vTfi, pNtk, pObj, i ) + if ( pObj->iTemp == (SFM_MASK_PI | SFM_MASK_INPUT) || pObj->iTemp == SFM_MASK_FANIN || pObj->iTemp == 0 ) // const + Sfm_DecAddNode( pObj, vMap, vGates, pObj->iTemp == SFM_MASK_FANIN, pPars->fVeryVerbose ); +if ( pPars->fVeryVerbose ) +printf( "\nFinish side\n" ); + // add the TFO nodes + Abc_NtkForEachObjVec( vTfi, pNtk, pObj, i ) + if ( pObj->iTemp >= SFM_MASK_MFFC ) + Sfm_DecAddNode( pObj, vMap, vGates, 0, pPars->fVeryVerbose ); + assert( Vec_IntSize(vMap) == Vec_IntSize(vGates) ); +if ( pPars->fVeryVerbose ) +printf( "\nFinish all\n" ); + // create node IDs Vec_WecClear( vFanins ); - Vec_IntForEachEntry( vMap, iObj, i ) + Abc_NtkForEachObjVec( vMap, pNtk, pObj, i ) { + pObj->iTemp = i; vLevel = Vec_WecPushLevel( vFanins ); - pNode = Abc_NtkObj( pNtk, iObj ); - if ( Abc_ObjIsCi(pNode) ) - { - Vec_IntPush( vGates, -1 ); - continue; - } - Abc_ObjForEachFanin( pNode, pFanin, k ) - Vec_IntPush( vLevel, pFanin->iTemp ); - Vec_IntPush( vGates, Mio_GateReadValue((Mio_Gate_t *)pNode->pData) ); + if ( Vec_IntEntry(vGates, i) >= 0 ) + Abc_ObjForEachFanin( pObj, pFanin, k ) + Vec_IntPush( vLevel, pFanin->iTemp ); } - return iTarget; + // remap roots + Abc_NtkForEachObjVec( vRoots, pNtk, pObj, i ) + Vec_IntWriteEntry( vRoots, i, pObj->iTemp ); +/* + // check + Abc_NtkForEachObjVec( vMap, pNtk, pObj, i ) + { + if ( i == nDivs ) + break; + Abc_NtkIncrementTravId( pNtk ); + assert( Abc_NtkDfsCheck_rec(pObj, pPivot) ); + } +*/ + return nDivs; } -void Sfm_DecInsert( Abc_Ntk_t * pNtk, int iNode, int Limit, Vec_Int_t * vTypes, Vec_Int_t * vGates, Vec_Wec_t * vFanins, Vec_Int_t * vMap, Vec_Ptr_t * vGateHandles ) +void Sfm_DecInsert( Abc_Ntk_t * pNtk, Abc_Obj_t * pPivot, int Limit, Vec_Int_t * vGates, Vec_Wec_t * vFanins, Vec_Int_t * vMap, Vec_Ptr_t * vGateHandles ) { - Abc_Obj_t * pTarget = Abc_NtkObj( pNtk, iNode ); Abc_Obj_t * pObjNew = NULL; int i, k, iObj, Gate; // assuming that new gates are appended at the end - assert( Limit < Vec_IntSize(vTypes) ); + assert( Limit < Vec_IntSize(vGates) ); + assert( Limit == Vec_IntSize(vMap) ); // introduce new gates Vec_IntForEachEntryStart( vGates, Gate, i, Limit ) { @@ -479,20 +722,30 @@ void Sfm_DecInsert( Abc_Ntk_t * pNtk, int iNode, int Limit, Vec_Int_t * vTypes, Vec_IntForEachEntry( vLevel, iObj, k ) Abc_ObjAddFanin( pObjNew, Abc_NtkObj(pNtk, Vec_IntEntry(vMap, iObj)) ); pObjNew->pData = Vec_PtrEntry( vGateHandles, Gate ); + Vec_IntPush( vMap, Abc_ObjId(pObjNew) ); } // transfer the fanout - Abc_ObjTransferFanout( pTarget, pObjNew ); - assert( Abc_ObjFanoutNum(pTarget) == 0 ); - Abc_NtkDeleteObj_rec( pTarget, 1 ); + Abc_ObjTransferFanout( pPivot, pObjNew ); + assert( Abc_ObjFanoutNum(pPivot) == 0 ); + Abc_NtkDeleteObj_rec( pPivot, 1 ); + // update level + Abc_NtkForEachObjVecStart( vMap, pNtk, pObjNew, i, Limit ) + Abc_NtkUpdateIncLevel( pObjNew ); } -void Sfm_DecTestBench( Abc_Ntk_t * pNtk, int iNode ) +void Abc_NtkPerformMfs3( Abc_Ntk_t * pNtk, Sfm_Par_t * pPars ) { extern void Sfm_LibPreprocess( Mio_Library_t * pLib, Vec_Int_t * vGateSizes, Vec_Wrd_t * vGateFuncs, Vec_Wec_t * vGateCnfs, Vec_Ptr_t * vGateHands ); Mio_Library_t * pLib = (Mio_Library_t *)pNtk->pManFunc; - Sfm_Dec_t * p = Sfm_DecStart(); - Vec_Int_t * vMap = Vec_IntAlloc( Abc_NtkObjNumMax(pNtk) ); // Sfm->Ntk + Sfm_Dec_t * p = Sfm_DecStart( pPars ); Abc_Obj_t * pObj; - int i, Limit; + int i, Limit, Count = 0, nStop = Abc_NtkObjNumMax(pNtk); + //int iNode = 2299;//8;//70; + printf( "Running remapping with parameters: " ); + printf( "TFO = %d. ", pPars->nTfoLevMax ); + printf( "TFI = %d. ", pPars->nTfiLevMax ); + printf( "FanMax = %d. ", pPars->nFanoutMax ); + printf( "MffcMax = %d. ", pPars->nMffcMax ); + printf( "\n" ); // enter library assert( Abc_NtkIsMappedLogic(pNtk) ); Sfm_LibPreprocess( pLib, &p->vGateSizes, &p->vGateFuncs, &p->vGateCnfs, &p->vGateHands ); @@ -500,21 +753,37 @@ void Sfm_DecTestBench( Abc_Ntk_t * pNtk, int iNode ) p->GateConst1 = Mio_GateReadValue( Mio_LibraryReadConst1(pLib) ); p->GateBuffer = Mio_GateReadValue( Mio_LibraryReadBuf(pLib) ); p->GateInvert = Mio_GateReadValue( Mio_LibraryReadInv(pLib) ); + p->GateAnd[0] = Mio_GateReadValue( Mio_LibraryReadGateByName(pLib, "and00", NULL) ); + p->GateAnd[1] = Mio_GateReadValue( Mio_LibraryReadGateByName(pLib, "and01", NULL) ); + p->GateAnd[2] = Mio_GateReadValue( Mio_LibraryReadGateByName(pLib, "and10", NULL) ); + p->GateAnd[3] = Mio_GateReadValue( Mio_LibraryReadGateByName(pLib, "and11", NULL) ); + p->GateOr[0] = Mio_GateReadValue( Mio_LibraryReadGateByName(pLib, "or00", NULL) ); + p->GateOr[1] = Mio_GateReadValue( Mio_LibraryReadGateByName(pLib, "or01", NULL) ); + p->GateOr[2] = Mio_GateReadValue( Mio_LibraryReadGateByName(pLib, "or10", NULL) ); + p->GateOr[3] = Mio_GateReadValue( Mio_LibraryReadGateByName(pLib, "or11", NULL) ); // iterate over nodes -// Abc_NtkForEachNode( pNtk, pObj, i ) - for ( ; pObj = Abc_NtkObj(pNtk, iNode); ) + Abc_NtkLevel( pNtk ); + Abc_NtkForEachNode( pNtk, pObj, i ) +// for ( ; (pObj = Abc_NtkObj(pNtk, iNode)); ) { - p->iTarget = Sfm_DecExtract( pNtk, pObj, &p->vObjTypes, &p->vObjGates, &p->vObjFanins, vMap, &p->vTemp ); - Limit = Vec_IntSize( &p->vObjTypes ); + if ( i >= nStop ) + break; + //if ( Abc_ObjFaninNum(pObj) == 0 || (Abc_ObjFaninNum(pObj) == 1 && Abc_ObjFanoutNum(Abc_ObjFanin0(pObj)) > 1) ) + // continue; + p->nDivs = Sfm_DecExtract( pNtk, pPars, pObj, &p->vObjRoots, &p->vObjGates, &p->vObjFanins, &p->vObjMap, &p->vTemp, &p->vTemp2, &p->nMffc ); + p->iTarget = pObj->iTemp; + Limit = Vec_IntSize( &p->vObjGates ); if ( !Sfm_DecPrepareSolver( p ) ) continue; - if ( !Sfm_DecPeformDec( p ) ) + if ( Sfm_DecPeformDec( p, pLib ) == -1 ) continue; -// Sfm_DecInsert( pNtk, p->iTarget, Limit, &p->vObjTypes, &p->vObjGates, &p->vObjFanins, vMap, vGateHandles ); - - break; + Sfm_DecInsert( pNtk, pObj, Limit, &p->vObjGates, &p->vObjFanins, &p->vObjMap, &p->vGateHands ); +if ( pPars->fVerbose ) +printf( "This was modification %d\n", Count ); + //if ( Count == 2 ) + // break; + Count++; } - Vec_IntFree( vMap ); Sfm_DecStop( p ); } |