diff options
Diffstat (limited to 'src/aig/gia/giaRetime.c')
-rw-r--r-- | src/aig/gia/giaRetime.c | 296 |
1 files changed, 296 insertions, 0 deletions
diff --git a/src/aig/gia/giaRetime.c b/src/aig/gia/giaRetime.c new file mode 100644 index 00000000..4f2c6e08 --- /dev/null +++ b/src/aig/gia/giaRetime.c @@ -0,0 +1,296 @@ +/**CFile**************************************************************** + + FileName [giaRetime.c] + + SystemName [ABC: Logic synthesis and verification system.] + + PackageName [Scalable AIG package.] + + Synopsis [Performs most-forward retiming for AIG with flop classes.] + + Author [Alan Mishchenko] + + Affiliation [UC Berkeley] + + Date [Ver. 1.0. Started - June 20, 2005.] + + Revision [$Id: giaRetime.c,v 1.00 2005/06/20 00:00:00 alanmi Exp $] + +***********************************************************************/ + +#include "gia.h" + +//////////////////////////////////////////////////////////////////////// +/// DECLARATIONS /// +//////////////////////////////////////////////////////////////////////// + +//////////////////////////////////////////////////////////////////////// +/// FUNCTION DEFINITIONS /// +//////////////////////////////////////////////////////////////////////// + +/**Function************************************************************* + + Synopsis [Marks objects reachables from Const0 and PIs/ + + Description [] + + SideEffects [] + + SeeAlso [] + +***********************************************************************/ +int Gia_ManMarkAutonomous_rec( Gia_Man_t * p, Gia_Obj_t * pObj ) +{ + if ( Gia_ObjIsTravIdCurrent(p, pObj) ) + return pObj->fMark0; + Gia_ObjSetTravIdCurrent(p, pObj); + assert( pObj->fMark0 == 0 ); + if ( Gia_ObjIsPi(p, pObj) || Gia_ObjIsConst0(pObj) ) + return pObj->fMark0 = 1; + if ( Gia_ObjIsCo(pObj) ) + return pObj->fMark0 = Gia_ManMarkAutonomous_rec( p, Gia_ObjFanin0(pObj) ); + if ( Gia_ObjIsCi(pObj) ) + return pObj->fMark0 = Gia_ManMarkAutonomous_rec( p, Gia_ObjRoToRi(p, pObj) ); + assert( Gia_ObjIsAnd(pObj) ); + if ( Gia_ManMarkAutonomous_rec( p, Gia_ObjFanin0(pObj) ) ) + return pObj->fMark0 = 1; + return pObj->fMark0 = Gia_ManMarkAutonomous_rec( p, Gia_ObjFanin1(pObj) ); +} + +/**Function************************************************************* + + Synopsis [Marks with current trav ROs reachable from Const0 and PIs.] + + Description [] + + SideEffects [] + + SeeAlso [] + +***********************************************************************/ +void Gia_ManMarkAutonomous( Gia_Man_t * p ) +{ + Gia_Obj_t * pObj; + int i; + Gia_ManCleanMark0( p ); + Gia_ManIncrementTravId( p ); + Gia_ManForEachRo( p, pObj, i ) + Gia_ManMarkAutonomous_rec( p, pObj ); + Gia_ManIncrementTravId( p ); + Gia_ManForEachRo( p, pObj, i ) + if ( pObj->fMark0 ) + Gia_ObjSetTravIdCurrent( p, pObj ); + Gia_ManCleanMark0( p ); +} + +/**Function************************************************************* + + Synopsis [Duplicates the AIG recursively.] + + Description [] + + SideEffects [] + + SeeAlso [] + +***********************************************************************/ +void Gia_ManRetimeDup_rec( Gia_Man_t * pNew, Gia_Obj_t * pObj ) +{ + if ( ~pObj->Value ) + return; + assert( Gia_ObjIsAnd(pObj) ); + Gia_ManRetimeDup_rec( pNew, Gia_ObjFanin0(pObj) ); + Gia_ManRetimeDup_rec( pNew, Gia_ObjFanin1(pObj) ); + pObj->Value = Gia_ManHashAnd( pNew, Gia_ObjFanin0Copy(pObj), Gia_ObjFanin1Copy(pObj) ); +} + +/**Function************************************************************* + + Synopsis [Duplicates the AIG while retiming the registers to the cut.] + + Description [] + + SideEffects [] + + SeeAlso [] + +***********************************************************************/ +Gia_Man_t * Gia_ManRetimeDupForward( Gia_Man_t * p, Vec_Ptr_t * vCut ) +{ + Gia_Man_t * pNew, * pTemp; + Gia_Obj_t * pObj, * pObjRi, * pObjRo; + int i; + // create the new manager + pNew = Gia_ManStart( Gia_ManObjNum(p) ); + pNew->pName = Aig_UtilStrsav( p->pName ); + Gia_ManHashAlloc( pNew ); + // create the true PIs + Gia_ManFillValue( p ); + Gia_ManSetPhase( p ); + Gia_ManConst0(p)->Value = 0; + Gia_ManForEachPi( p, pObj, i ) + pObj->Value = Gia_ManAppendCi( pNew ); + // create the registers + Vec_PtrForEachEntry( vCut, pObj, i ) + pObj->Value = Gia_LitNotCond( Gia_ManAppendCi(pNew), pObj->fPhase ); + // duplicate logic above the cut + Gia_ManForEachCo( p, pObj, i ) + Gia_ManRetimeDup_rec( pNew, Gia_ObjFanin0(pObj) ); + // create the true POs + Gia_ManForEachPo( p, pObj, i ) + Gia_ManAppendCo( pNew, Gia_ObjFanin0Copy(pObj) ); + // remember value in LI + Gia_ManForEachRi( p, pObj, i ) + pObj->Value = Gia_ObjFanin0Copy(pObj); + // transfer values from the LIs to the LOs + Gia_ManForEachRiRo( p, pObjRi, pObjRo, i ) + pObjRo->Value = pObjRi->Value; + // erase the data values on the internal nodes of the cut + Vec_PtrForEachEntry( vCut, pObj, i ) + if ( Gia_ObjIsAnd(pObj) ) + pObj->Value = ~0; + // duplicate logic below the cut + Vec_PtrForEachEntry( vCut, pObj, i ) + { + Gia_ManRetimeDup_rec( pNew, pObj ); + Gia_ManAppendCo( pNew, Gia_LitNotCond( pObj->Value, pObj->fPhase ) ); + } + Gia_ManHashStop( pNew ); + Gia_ManSetRegNum( pNew, Vec_PtrSize(vCut) ); + pNew = Gia_ManCleanup( pTemp = pNew ); + Gia_ManStop( pTemp ); + return pNew; +} + +/**Function************************************************************* + + Synopsis [Derives the cut for forward retiming.] + + Description [Assumes topological ordering of the nodes.] + + SideEffects [] + + SeeAlso [] + +***********************************************************************/ +Gia_Man_t * Gia_ManRetimeForwardOne( Gia_Man_t * p, int * pnRegFixed, int * pnRegMoves ) +{ + Vec_Int_t * vFlopClasses = NULL; + Vec_Int_t * vObjClasses = NULL; + Gia_Man_t * pNew; + Vec_Ptr_t * vCut; + Gia_Obj_t * pObj; + int i; + if ( p->vFlopClasses ) + { + printf( "Performing retiming with register classes.\n" ); + vObjClasses = Vec_IntAlloc( Gia_ManObjNum(p) ); + for ( i = 0; i < Gia_ManObjNum(p); i++ ) + Vec_IntPush( vObjClasses, -1 ); + Gia_ManForEachRo( p, pObj, i ) + Vec_IntWriteEntry( vObjClasses, Gia_ObjId(p, pObj), Vec_IntEntry(p->vFlopClasses, i) ); + vFlopClasses = Vec_IntAlloc( Gia_ManRegNum(p) ); + } + // mark the retimable nodes + Gia_ManResetTravId( p ); + Gia_ManMarkAutonomous( p ); + // mark the retimable registers with the fresh trav ID + Gia_ManIncrementTravId( p ); + *pnRegFixed = 0; + Gia_ManForEachRo( p, pObj, i ) + if ( Gia_ObjIsTravIdPrevious(p, pObj) ) + Gia_ObjSetTravIdCurrent(p, pObj); + else + (*pnRegFixed)++; + // mark all the nodes that can be retimed forward + *pnRegMoves = 0; + Gia_ManForEachAnd( p, pObj, i ) + if ( Gia_ObjIsTravIdCurrent(p, Gia_ObjFanin0(pObj)) && Gia_ObjIsTravIdCurrent(p, Gia_ObjFanin1(pObj)) ) + { + if ( vObjClasses && Vec_IntEntry(vObjClasses, Gia_ObjFaninId0(pObj, i)) != Vec_IntEntry(vObjClasses, Gia_ObjFaninId1(pObj, i)) ) + continue; + if ( vObjClasses ) + Vec_IntWriteEntry( vObjClasses, Gia_ObjId(p, pObj), Vec_IntEntry(vObjClasses, Gia_ObjFaninId0(pObj, i)) ); + Gia_ObjSetTravIdCurrent(p, pObj); + (*pnRegMoves)++; + } + // mark the remaining registers + Gia_ManForEachRo( p, pObj, i ) + Gia_ObjSetTravIdCurrent(p, pObj); + // find the cut (all such marked objects that fanout into unmarked nodes) + vCut = Vec_PtrAlloc( 1000 ); + Gia_ManIncrementTravId( p ); + Gia_ManForEachObj( p, pObj, i ) + { + if ( Gia_ObjIsTravIdPrevious(p, pObj) ) + continue; + if ( (Gia_ObjIsCo(pObj) || Gia_ObjIsAnd(pObj)) && Gia_ObjIsTravIdPrevious(p, Gia_ObjFanin0(pObj)) ) + { + if ( vFlopClasses ) + Vec_IntPush( vFlopClasses, Vec_IntEntry(vObjClasses, Gia_ObjFaninId0(pObj, i)) ); + Vec_PtrPush( vCut, Gia_ObjFanin0(pObj) ); + Gia_ObjSetTravIdCurrent( p, Gia_ObjFanin0(pObj) ); + } + if ( Gia_ObjIsAnd(pObj) && Gia_ObjIsTravIdPrevious(p, Gia_ObjFanin1(pObj)) ) + { + if ( vFlopClasses ) + Vec_IntPush( vFlopClasses, Vec_IntEntry(vObjClasses, Gia_ObjFaninId1(pObj, i)) ); + Vec_PtrPush( vCut, Gia_ObjFanin1(pObj) ); + Gia_ObjSetTravIdCurrent( p, Gia_ObjFanin1(pObj) ); + } + } + assert( vFlopClasses == NULL || Vec_IntSize(vFlopClasses) == Vec_PtrSize(vCut) ); + // finally derive the new manager + pNew = Gia_ManRetimeDupForward( p, vCut ); + Vec_PtrFree( vCut ); + Vec_IntFree( vObjClasses ); + pNew->vFlopClasses = vFlopClasses; + return pNew; +} + +/**Function************************************************************* + + Synopsis [Derives the cut for forward retiming.] + + Description [Assumes topological ordering of the nodes.] + + SideEffects [] + + SeeAlso [] + +***********************************************************************/ +Gia_Man_t * Gia_ManRetimeForward( Gia_Man_t * p, int nMaxIters, int fVerbose ) +{ + Gia_Man_t * pNew, * pTemp; + int i, clk, nRegFixed, nRegMoves = 1; + pNew = p; + for ( i = 0; i < nMaxIters && nRegMoves > 0; i++ ) + { + clk = clock(); + pNew = Gia_ManRetimeForwardOne( pTemp = pNew, &nRegFixed, &nRegMoves ); + if ( fVerbose ) + { + printf( "%2d : And = %6d. Reg = %5d. Unret = %5d. Move = %6d. ", + i + 1, Gia_ManAndNum(pTemp), Gia_ManRegNum(pTemp), nRegFixed, nRegMoves ); + ABC_PRT( "Time", clock() - clk ); + } + if ( pTemp != p ) + Gia_ManStop( pTemp ); + } +/* + clk = clock(); + pNew = Gia_ManReduceLaches( pNew, fVerbose ); + if ( fVerbose ) + { + ABC_PRT( "Register sharing time", clock() - clk ); + } +*/ + return pNew; +} + + +//////////////////////////////////////////////////////////////////////// +/// END OF FILE /// +//////////////////////////////////////////////////////////////////////// + + |