mirror of https://github.com/YosysHQ/abc.git
857 lines
27 KiB
C
857 lines
27 KiB
C
/**CFile****************************************************************
|
|
|
|
FileName [ntlExtract.c]
|
|
|
|
SystemName [ABC: Logic synthesis and verification system.]
|
|
|
|
PackageName [Netlist representation.]
|
|
|
|
Synopsis [Netlist SOP to AIG conversion.]
|
|
|
|
Author [Alan Mishchenko]
|
|
|
|
Affiliation [UC Berkeley]
|
|
|
|
Date [Ver. 1.0. Started - June 20, 2005.]
|
|
|
|
Revision [$Id: ntlExtract.c,v 1.00 2005/06/20 00:00:00 alanmi Exp $]
|
|
|
|
***********************************************************************/
|
|
|
|
#include "ntl.h"
|
|
#include "dec.h"
|
|
#include "kit.h"
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
/// DECLARATIONS ///
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
/// FUNCTION DEFINITIONS ///
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
/**Function*************************************************************
|
|
|
|
Synopsis [Strashes one logic node using its SOP.]
|
|
|
|
Description []
|
|
|
|
SideEffects []
|
|
|
|
SeeAlso []
|
|
|
|
***********************************************************************/
|
|
Aig_Obj_t * Ntl_ConvertSopToAigInternal( Aig_Man_t * pMan, Ntl_Obj_t * pNode, char * pSop )
|
|
{
|
|
Ntl_Net_t * pNet;
|
|
Aig_Obj_t * pAnd, * pSum;
|
|
int i, Value, nFanins;
|
|
char * pCube;
|
|
// get the number of variables
|
|
nFanins = Kit_PlaGetVarNum(pSop);
|
|
// go through the cubes of the node's SOP
|
|
pSum = Aig_ManConst0(pMan);
|
|
Kit_PlaForEachCube( pSop, nFanins, pCube )
|
|
{
|
|
// create the AND of literals
|
|
pAnd = Aig_ManConst1(pMan);
|
|
Kit_PlaCubeForEachVar( pCube, Value, i )
|
|
{
|
|
pNet = Ntl_ObjFanin( pNode, i );
|
|
if ( Value == '1' )
|
|
pAnd = Aig_And( pMan, pAnd, pNet->pCopy );
|
|
else if ( Value == '0' )
|
|
pAnd = Aig_And( pMan, pAnd, Aig_Not(pNet->pCopy) );
|
|
}
|
|
// add to the sum of cubes
|
|
pSum = Aig_Or( pMan, pSum, pAnd );
|
|
}
|
|
// decide whether to complement the result
|
|
if ( Kit_PlaIsComplement(pSop) )
|
|
pSum = Aig_Not(pSum);
|
|
return pSum;
|
|
}
|
|
|
|
/**Function*************************************************************
|
|
|
|
Synopsis [Transforms the decomposition graph into the AIG.]
|
|
|
|
Description []
|
|
|
|
SideEffects []
|
|
|
|
SeeAlso []
|
|
|
|
***********************************************************************/
|
|
Aig_Obj_t * Ntl_GraphToNetworkAig( Aig_Man_t * pMan, Dec_Graph_t * pGraph )
|
|
{
|
|
Dec_Node_t * pNode;
|
|
Aig_Obj_t * pAnd0, * pAnd1;
|
|
int i;
|
|
// check for constant function
|
|
if ( Dec_GraphIsConst(pGraph) )
|
|
return Aig_NotCond( Aig_ManConst1(pMan), Dec_GraphIsComplement(pGraph) );
|
|
// check for a literal
|
|
if ( Dec_GraphIsVar(pGraph) )
|
|
return Aig_NotCond( Dec_GraphVar(pGraph)->pFunc, Dec_GraphIsComplement(pGraph) );
|
|
// build the AIG nodes corresponding to the AND gates of the graph
|
|
Dec_GraphForEachNode( pGraph, pNode, i )
|
|
{
|
|
pAnd0 = Aig_NotCond( Dec_GraphNode(pGraph, pNode->eEdge0.Node)->pFunc, pNode->eEdge0.fCompl );
|
|
pAnd1 = Aig_NotCond( Dec_GraphNode(pGraph, pNode->eEdge1.Node)->pFunc, pNode->eEdge1.fCompl );
|
|
pNode->pFunc = Aig_And( pMan, pAnd0, pAnd1 );
|
|
}
|
|
// complement the result if necessary
|
|
return Aig_NotCond( pNode->pFunc, Dec_GraphIsComplement(pGraph) );
|
|
}
|
|
|
|
/**Function*************************************************************
|
|
|
|
Synopsis [Converts the network from AIG to BDD representation.]
|
|
|
|
Description []
|
|
|
|
SideEffects []
|
|
|
|
SeeAlso []
|
|
|
|
***********************************************************************/
|
|
Aig_Obj_t * Ntl_ManBuildNodeAig( Ntl_Obj_t * pNode )
|
|
{
|
|
Aig_Man_t * pMan = pNode->pModel->pMan->pAig;
|
|
int fUseFactor = 1;
|
|
// consider the constant node
|
|
if ( Kit_PlaGetVarNum(pNode->pSop) == 0 )
|
|
return Aig_NotCond( Aig_ManConst1(pMan), Kit_PlaIsConst0(pNode->pSop) );
|
|
// decide when to use factoring
|
|
if ( fUseFactor && Kit_PlaGetVarNum(pNode->pSop) > 2 && Kit_PlaGetCubeNum(pNode->pSop) > 1 )
|
|
{
|
|
Dec_Graph_t * pFForm;
|
|
Dec_Node_t * pFFNode;
|
|
Aig_Obj_t * pFunc;
|
|
int i;
|
|
// perform factoring
|
|
pFForm = Dec_Factor( pNode->pSop );
|
|
// collect the fanins
|
|
Dec_GraphForEachLeaf( pFForm, pFFNode, i )
|
|
pFFNode->pFunc = Ntl_ObjFanin(pNode, i)->pCopy;
|
|
// perform strashing
|
|
pFunc = Ntl_GraphToNetworkAig( pMan, pFForm );
|
|
Dec_GraphFree( pFForm );
|
|
return pFunc;
|
|
}
|
|
return Ntl_ConvertSopToAigInternal( pMan, pNode, pNode->pSop );
|
|
}
|
|
|
|
/**Function*************************************************************
|
|
|
|
Synopsis [Collects the nodes in a topological order.]
|
|
|
|
Description []
|
|
|
|
SideEffects []
|
|
|
|
SeeAlso []
|
|
|
|
***********************************************************************/
|
|
int Ntl_ManExtract_rec( Ntl_Man_t * p, Ntl_Net_t * pNet )
|
|
{
|
|
Ntl_Obj_t * pObj;
|
|
Ntl_Net_t * pNetFanin;
|
|
int i;
|
|
// skip visited
|
|
if ( pNet->nVisits == 2 )
|
|
return 1;
|
|
// if the node is on the path, this is a combinational loop
|
|
if ( pNet->nVisits == 1 )
|
|
return 0;
|
|
// mark the node as the one on the path
|
|
pNet->nVisits = 1;
|
|
// derive the box
|
|
pObj = pNet->pDriver;
|
|
assert( Ntl_ObjIsNode(pObj) || Ntl_ObjIsBox(pObj) );
|
|
// visit the input nets of the box
|
|
Ntl_ObjForEachFanin( pObj, pNetFanin, i )
|
|
if ( !Ntl_ManExtract_rec( p, pNetFanin ) )
|
|
return 0;
|
|
// add box inputs/outputs to COs/CIs
|
|
if ( Ntl_ObjIsBox(pObj) )
|
|
{
|
|
int LevelCur, LevelMax = -TIM_ETERNITY;
|
|
Vec_IntPush( p->vBox1Cos, Aig_ManPoNum(p->pAig) );
|
|
Ntl_ObjForEachFanin( pObj, pNetFanin, i )
|
|
{
|
|
LevelCur = Aig_ObjLevel( Aig_Regular(pNetFanin->pCopy) );
|
|
LevelMax = AIG_MAX( LevelMax, LevelCur );
|
|
Vec_PtrPush( p->vCos, pNetFanin );
|
|
Aig_ObjCreatePo( p->pAig, pNetFanin->pCopy );
|
|
}
|
|
Ntl_ObjForEachFanout( pObj, pNetFanin, i )
|
|
{
|
|
Vec_PtrPush( p->vCis, pNetFanin );
|
|
pNetFanin->pCopy = Aig_ObjCreatePi( p->pAig );
|
|
Aig_ObjSetLevel( pNetFanin->pCopy, LevelMax + 1 );
|
|
}
|
|
}
|
|
Vec_PtrPush( p->vNodes, pObj );
|
|
if ( Ntl_ObjIsNode(pObj) )
|
|
pNet->pCopy = Ntl_ManBuildNodeAig( pObj );
|
|
pNet->nVisits = 2;
|
|
return 1;
|
|
}
|
|
|
|
/**Function*************************************************************
|
|
|
|
Synopsis [Performs DFS.]
|
|
|
|
Description [Checks for combinational loops. Collects PI/PO nets.
|
|
Collects nodes in the topological order.]
|
|
|
|
SideEffects []
|
|
|
|
SeeAlso []
|
|
|
|
***********************************************************************/
|
|
Aig_Man_t * Ntl_ManExtract( Ntl_Man_t * p )
|
|
{
|
|
Aig_Man_t * pAig;
|
|
Ntl_Mod_t * pRoot;
|
|
Ntl_Obj_t * pObj;
|
|
Ntl_Net_t * pNet;
|
|
int i, nUselessObjects;
|
|
Ntl_ManCleanup( p );
|
|
assert( Vec_PtrSize(p->vCis) == 0 );
|
|
assert( Vec_PtrSize(p->vCos) == 0 );
|
|
assert( Vec_PtrSize(p->vNodes) == 0 );
|
|
assert( Vec_IntSize(p->vBox1Cos) == 0 );
|
|
// start the AIG manager
|
|
assert( p->pAig == NULL );
|
|
p->pAig = Aig_ManStart( 10000 );
|
|
p->pAig->pName = Aig_UtilStrsav( p->pName );
|
|
p->pAig->pSpec = Aig_UtilStrsav( p->pSpec );
|
|
// get the root model
|
|
pRoot = Ntl_ManRootModel( p );
|
|
// clear net visited flags
|
|
Ntl_ModelForEachNet( pRoot, pNet, i )
|
|
pNet->nVisits = 0;
|
|
// collect primary inputs
|
|
Ntl_ModelForEachPi( pRoot, pObj, i )
|
|
{
|
|
assert( Ntl_ObjFanoutNum(pObj) == 1 );
|
|
pNet = Ntl_ObjFanout0(pObj);
|
|
Vec_PtrPush( p->vCis, pNet );
|
|
pNet->pCopy = Aig_ObjCreatePi( p->pAig );
|
|
if ( pNet->nVisits )
|
|
{
|
|
printf( "Ntl_ManExtract(): Primary input appears twice in the list.\n" );
|
|
return 0;
|
|
}
|
|
pNet->nVisits = 2;
|
|
}
|
|
// collect latch outputs
|
|
Ntl_ModelForEachLatch( pRoot, pObj, i )
|
|
{
|
|
assert( Ntl_ObjFanoutNum(pObj) == 1 );
|
|
pNet = Ntl_ObjFanout0(pObj);
|
|
Vec_PtrPush( p->vCis, pNet );
|
|
pNet->pCopy = Aig_ObjCreatePi( p->pAig );
|
|
if ( pNet->nVisits )
|
|
{
|
|
printf( "Ntl_ManExtract(): Latch output is duplicated or defined as a primary input.\n" );
|
|
return 0;
|
|
}
|
|
pNet->nVisits = 2;
|
|
}
|
|
// visit the nodes starting from primary outputs
|
|
Ntl_ModelForEachPo( pRoot, pObj, i )
|
|
{
|
|
pNet = Ntl_ObjFanin0(pObj);
|
|
if ( !Ntl_ManExtract_rec( p, pNet ) )
|
|
{
|
|
printf( "Ntl_ManExtract(): Error: Combinational loop is detected.\n" );
|
|
return 0;
|
|
}
|
|
Vec_PtrPush( p->vCos, pNet );
|
|
Aig_ObjCreatePo( p->pAig, pNet->pCopy );
|
|
}
|
|
// visit the nodes starting from latch inputs outputs
|
|
Ntl_ModelForEachLatch( pRoot, pObj, i )
|
|
{
|
|
pNet = Ntl_ObjFanin0(pObj);
|
|
if ( !Ntl_ManExtract_rec( p, pNet ) )
|
|
{
|
|
printf( "Ntl_ManExtract(): Error: Combinational loop is detected.\n" );
|
|
return 0;
|
|
}
|
|
Vec_PtrPush( p->vCos, pNet );
|
|
Aig_ObjCreatePo( p->pAig, pNet->pCopy );
|
|
}
|
|
// report the number of dangling objects
|
|
nUselessObjects = Ntl_ModelNodeNum(pRoot) + Ntl_ModelLut1Num(pRoot) + Ntl_ModelBoxNum(pRoot) - Vec_PtrSize(p->vNodes);
|
|
// if ( nUselessObjects )
|
|
// printf( "The number of nodes that do not feed into POs = %d.\n", nUselessObjects );
|
|
// cleanup the AIG
|
|
Aig_ManCleanup( p->pAig );
|
|
// extract the timing manager
|
|
assert( p->pManTime == NULL );
|
|
p->pManTime = Ntl_ManCreateTiming( p );
|
|
// discretize timing info
|
|
p->pAig->pManTime = Tim_ManDup( p->pManTime, 1 );
|
|
pAig = p->pAig; p->pAig = NULL;
|
|
return pAig;
|
|
}
|
|
|
|
/**Function*************************************************************
|
|
|
|
Synopsis [Collects the nodes in a topological order.]
|
|
|
|
Description []
|
|
|
|
SideEffects []
|
|
|
|
SeeAlso []
|
|
|
|
***********************************************************************/
|
|
int Ntl_ManBuildModelAig( Ntl_Man_t * p, Ntl_Obj_t * pBox )
|
|
{
|
|
extern int Ntl_ManCollapse_rec( Ntl_Man_t * p, Ntl_Net_t * pNet );
|
|
Ntl_Mod_t * pModel = pBox->pImplem;
|
|
Ntl_Obj_t * pObj;
|
|
Ntl_Net_t * pNet, * pNetBox;
|
|
int i;
|
|
assert( Ntl_ObjFaninNum(pBox) == Ntl_ModelPiNum(pModel) );
|
|
assert( Ntl_ObjFanoutNum(pBox) == Ntl_ModelPoNum(pModel) );
|
|
// clear net visited flags
|
|
Ntl_ModelForEachNet( pModel, pNet, i )
|
|
pNet->nVisits = 0;
|
|
// transfer from the box to the PIs of the model
|
|
Ntl_ModelForEachPi( pModel, pObj, i )
|
|
{
|
|
pNet = Ntl_ObjFanout0(pObj);
|
|
pNetBox = Ntl_ObjFanin( pBox, i );
|
|
pNet->pCopy = pNetBox->pCopy;
|
|
pNet->nVisits = 2;
|
|
}
|
|
// compute AIG for the internal nodes
|
|
Ntl_ModelForEachPo( pModel, pObj, i )
|
|
if ( !Ntl_ManCollapse_rec( p, Ntl_ObjFanin0(pObj) ) )
|
|
return 0;
|
|
// transfer from the POs of the model to the box
|
|
Ntl_ModelForEachPo( pModel, pObj, i )
|
|
{
|
|
pNet = Ntl_ObjFanin0(pObj);
|
|
pNetBox = Ntl_ObjFanout( pBox, i );
|
|
pNetBox->pCopy = pNet->pCopy;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
/**Function*************************************************************
|
|
|
|
Synopsis [Collects the nodes in a topological order.]
|
|
|
|
Description []
|
|
|
|
SideEffects []
|
|
|
|
SeeAlso []
|
|
|
|
***********************************************************************/
|
|
int Ntl_ManCollapse_rec( Ntl_Man_t * p, Ntl_Net_t * pNet )
|
|
{
|
|
Ntl_Obj_t * pObj;
|
|
Ntl_Net_t * pNetFanin;
|
|
int i;
|
|
// skip visited
|
|
if ( pNet->nVisits == 2 )
|
|
return 1;
|
|
// if the node is on the path, this is a combinational loop
|
|
if ( pNet->nVisits == 1 )
|
|
return 0;
|
|
// mark the node as the one on the path
|
|
pNet->nVisits = 1;
|
|
// derive the box
|
|
pObj = pNet->pDriver;
|
|
assert( Ntl_ObjIsNode(pObj) || Ntl_ObjIsBox(pObj) );
|
|
// visit the input nets of the box
|
|
Ntl_ObjForEachFanin( pObj, pNetFanin, i )
|
|
if ( !Ntl_ManCollapse_rec( p, pNetFanin ) )
|
|
return 0;
|
|
// add box inputs/outputs to COs/CIs
|
|
if ( Ntl_ObjIsBox(pObj) )
|
|
{
|
|
if ( !Ntl_ManBuildModelAig( p, pObj ) )
|
|
return 0;
|
|
}
|
|
if ( Ntl_ObjIsNode(pObj) )
|
|
pNet->pCopy = Ntl_ManBuildNodeAig( pObj );
|
|
pNet->nVisits = 2;
|
|
return 1;
|
|
}
|
|
|
|
/**Function*************************************************************
|
|
|
|
Synopsis [Performs DFS.]
|
|
|
|
Description [Checks for combinational loops. Collects PI/PO nets.
|
|
Collects nodes in the topological order.]
|
|
|
|
SideEffects []
|
|
|
|
SeeAlso []
|
|
|
|
***********************************************************************/
|
|
Aig_Man_t * Ntl_ManCollapse( Ntl_Man_t * p )
|
|
{
|
|
Aig_Man_t * pAig;
|
|
Ntl_Mod_t * pRoot;
|
|
Ntl_Obj_t * pObj;
|
|
Ntl_Net_t * pNet;
|
|
int i;
|
|
// start the AIG manager
|
|
assert( p->pAig == NULL );
|
|
p->pAig = Aig_ManStart( 10000 );
|
|
p->pAig->pName = Aig_UtilStrsav( p->pName );
|
|
p->pAig->pSpec = Aig_UtilStrsav( p->pSpec );
|
|
// get the root model
|
|
pRoot = Ntl_ManRootModel( p );
|
|
// clear net visited flags
|
|
Ntl_ModelForEachNet( pRoot, pNet, i )
|
|
pNet->nVisits = 0;
|
|
// collect primary inputs
|
|
Ntl_ModelForEachPi( pRoot, pObj, i )
|
|
{
|
|
assert( Ntl_ObjFanoutNum(pObj) == 1 );
|
|
pNet = Ntl_ObjFanout0(pObj);
|
|
pNet->pCopy = Aig_ObjCreatePi( p->pAig );
|
|
if ( pNet->nVisits )
|
|
{
|
|
printf( "Ntl_ManCollapse(): Primary input appears twice in the list.\n" );
|
|
return 0;
|
|
}
|
|
pNet->nVisits = 2;
|
|
}
|
|
// collect latch outputs
|
|
Ntl_ModelForEachLatch( pRoot, pObj, i )
|
|
{
|
|
assert( Ntl_ObjFanoutNum(pObj) == 1 );
|
|
pNet = Ntl_ObjFanout0(pObj);
|
|
pNet->pCopy = Aig_ObjCreatePi( p->pAig );
|
|
if ( pNet->nVisits )
|
|
{
|
|
printf( "Ntl_ManCollapse(): Latch output is duplicated or defined as a primary input.\n" );
|
|
return 0;
|
|
}
|
|
pNet->nVisits = 2;
|
|
}
|
|
// visit the nodes starting from primary outputs
|
|
Ntl_ModelForEachPo( pRoot, pObj, i )
|
|
{
|
|
pNet = Ntl_ObjFanin0(pObj);
|
|
if ( !Ntl_ManCollapse_rec( p, pNet ) )
|
|
{
|
|
printf( "Ntl_ManCollapse(): Error: Combinational loop is detected.\n" );
|
|
return 0;
|
|
}
|
|
Aig_ObjCreatePo( p->pAig, pNet->pCopy );
|
|
}
|
|
// visit the nodes starting from latch inputs outputs
|
|
Ntl_ModelForEachLatch( pRoot, pObj, i )
|
|
{
|
|
pNet = Ntl_ObjFanin0(pObj);
|
|
if ( !Ntl_ManCollapse_rec( p, pNet ) )
|
|
{
|
|
printf( "Ntl_ManCollapse(): Error: Combinational loop is detected.\n" );
|
|
return 0;
|
|
}
|
|
Aig_ObjCreatePo( p->pAig, pNet->pCopy );
|
|
}
|
|
// cleanup the AIG
|
|
Aig_ManCleanup( p->pAig );
|
|
pAig = p->pAig; p->pAig = NULL;
|
|
return pAig;
|
|
}
|
|
|
|
/**Function*************************************************************
|
|
|
|
Synopsis [Derives AIG for CEC.]
|
|
|
|
Description [Uses CIs/COs collected in the internal arrays.]
|
|
|
|
SideEffects []
|
|
|
|
SeeAlso []
|
|
|
|
***********************************************************************/
|
|
Aig_Man_t * Ntl_ManCollapseForCec( Ntl_Man_t * p )
|
|
{
|
|
Aig_Man_t * pAig;
|
|
Ntl_Obj_t * pObj;
|
|
Ntl_Net_t * pNet;
|
|
int i;
|
|
// create the manager
|
|
p->pAig = Aig_ManStart( 10000 );
|
|
p->pAig->pName = Aig_UtilStrsav( p->pName );
|
|
p->pAig->pSpec = Aig_UtilStrsav( p->pSpec );
|
|
// set the inputs
|
|
Ntl_ManForEachCiNet( p, pObj, i )
|
|
{
|
|
assert( Ntl_ObjFanoutNum(pObj) == 1 );
|
|
pNet = Ntl_ObjFanout0(pObj);
|
|
pNet->pCopy = Aig_ObjCreatePi( p->pAig );
|
|
if ( pNet->nVisits )
|
|
{
|
|
printf( "Ntl_ManCollapseForCec(): Primary input appears twice in the list.\n" );
|
|
return 0;
|
|
}
|
|
pNet->nVisits = 2;
|
|
}
|
|
// derive the outputs
|
|
Ntl_ManForEachCoNet( p, pObj, i )
|
|
{
|
|
pNet = Ntl_ObjFanin0(pObj);
|
|
if ( !Ntl_ManCollapse_rec( p, pNet ) )
|
|
{
|
|
printf( "Ntl_ManCollapseForCec(): Error: Combinational loop is detected.\n" );
|
|
return 0;
|
|
}
|
|
Aig_ObjCreatePo( p->pAig, pNet->pCopy );
|
|
}
|
|
// cleanup the AIG
|
|
Aig_ManCleanup( p->pAig );
|
|
pAig = p->pAig; p->pAig = NULL;
|
|
return pAig;
|
|
}
|
|
|
|
/**Function*************************************************************
|
|
|
|
Synopsis [Derives AIG for SEC.]
|
|
|
|
Description [Uses CIs/COs collected in the internal arrays.]
|
|
|
|
SideEffects []
|
|
|
|
SeeAlso []
|
|
|
|
***********************************************************************/
|
|
Aig_Man_t * Ntl_ManCollapseForSec( Ntl_Man_t * p1, Ntl_Man_t * p2 )
|
|
{
|
|
Aig_Man_t * pAig;
|
|
Aig_Obj_t * pMiter;
|
|
Ntl_Mod_t * pRoot1, * pRoot2;
|
|
Ntl_Obj_t * pObj;
|
|
Ntl_Net_t * pNet;
|
|
Vec_Ptr_t * vPairs;
|
|
int i;
|
|
assert( Vec_PtrSize(p1->vCis) > 0 );
|
|
assert( Vec_PtrSize(p1->vCos) > 0 );
|
|
assert( Vec_PtrSize(p1->vCis) == Vec_PtrSize(p2->vCis) );
|
|
assert( Vec_PtrSize(p1->vCos) == Vec_PtrSize(p2->vCos) );
|
|
|
|
// create the manager
|
|
pAig = p1->pAig = p2->pAig = Aig_ManStart( 10000 );
|
|
pAig->pName = Aig_UtilStrsav( p1->pName );
|
|
pAig->pSpec = Aig_UtilStrsav( p1->pSpec );
|
|
vPairs = Vec_PtrStart( 2 * Vec_PtrSize(p1->vCos) );
|
|
// placehoder output to be used later for the miter output
|
|
Aig_ObjCreatePo( pAig, Aig_ManConst1(pAig) );
|
|
|
|
/////////////////////////////////////////////////////
|
|
// primary inputs
|
|
Ntl_ManForEachCiNet( p1, pObj, i )
|
|
{
|
|
assert( Ntl_ObjFanoutNum(pObj) == 1 );
|
|
pNet = Ntl_ObjFanout0(pObj);
|
|
pNet->pCopy = Aig_ObjCreatePi( pAig );
|
|
if ( pNet->nVisits )
|
|
{
|
|
printf( "Ntl_ManCollapseForSec(): Primary input appears twice in the list.\n" );
|
|
return 0;
|
|
}
|
|
pNet->nVisits = 2;
|
|
}
|
|
// latch outputs
|
|
pRoot1 = Ntl_ManRootModel(p1);
|
|
Ntl_ModelForEachLatch( pRoot1, pObj, i )
|
|
{
|
|
assert( Ntl_ObjFanoutNum(pObj) == 1 );
|
|
pNet = Ntl_ObjFanout0(pObj);
|
|
pNet->pCopy = Aig_ObjCreatePi( pAig );
|
|
if ( (pObj->LatchId & 3) == 1 )
|
|
pNet->pCopy = Aig_Not(pNet->pCopy);
|
|
if ( pNet->nVisits )
|
|
{
|
|
printf( "Ntl_ManCollapseForSec(): Latch output is duplicated or defined as a primary input.\n" );
|
|
return 0;
|
|
}
|
|
pNet->nVisits = 2;
|
|
}
|
|
// derive the outputs
|
|
Ntl_ManForEachCoNet( p1, pObj, i )
|
|
{
|
|
pNet = Ntl_ObjFanin0(pObj);
|
|
if ( !Ntl_ManCollapse_rec( p1, pNet ) )
|
|
{
|
|
printf( "Ntl_ManCollapseForSec(): Error: Combinational loop is detected.\n" );
|
|
return 0;
|
|
}
|
|
// Aig_ObjCreatePo( pAig, pNet->pCopy );
|
|
Vec_PtrWriteEntry( vPairs, 2 * i, pNet->pCopy );
|
|
}
|
|
// visit the nodes starting from latch inputs outputs
|
|
Ntl_ModelForEachLatch( pRoot1, pObj, i )
|
|
{
|
|
pNet = Ntl_ObjFanin0(pObj);
|
|
if ( !Ntl_ManCollapse_rec( p1, pNet ) )
|
|
{
|
|
printf( "Ntl_ManCollapseForSec(): Error: Combinational loop is detected.\n" );
|
|
return 0;
|
|
}
|
|
if ( (pObj->LatchId & 3) == 1 )
|
|
Aig_ObjCreatePo( pAig, Aig_Not(pNet->pCopy) );
|
|
else
|
|
Aig_ObjCreatePo( pAig, pNet->pCopy );
|
|
}
|
|
|
|
/////////////////////////////////////////////////////
|
|
// primary inputs
|
|
Ntl_ManForEachCiNet( p2, pObj, i )
|
|
{
|
|
assert( Ntl_ObjFanoutNum(pObj) == 1 );
|
|
pNet = Ntl_ObjFanout0(pObj);
|
|
pNet->pCopy = Aig_ManPi( pAig, i );
|
|
if ( pNet->nVisits )
|
|
{
|
|
printf( "Ntl_ManCollapseForSec(): Primary input appears twice in the list.\n" );
|
|
return 0;
|
|
}
|
|
pNet->nVisits = 2;
|
|
}
|
|
// latch outputs
|
|
pRoot2 = Ntl_ManRootModel(p2);
|
|
Ntl_ModelForEachLatch( pRoot2, pObj, i )
|
|
{
|
|
assert( Ntl_ObjFanoutNum(pObj) == 1 );
|
|
pNet = Ntl_ObjFanout0(pObj);
|
|
pNet->pCopy = Aig_ObjCreatePi( pAig );
|
|
if ( (pObj->LatchId & 3) == 1 )
|
|
pNet->pCopy = Aig_Not(pNet->pCopy);
|
|
if ( pNet->nVisits )
|
|
{
|
|
printf( "Ntl_ManCollapseForSec(): Latch output is duplicated or defined as a primary input.\n" );
|
|
return 0;
|
|
}
|
|
pNet->nVisits = 2;
|
|
}
|
|
// derive the outputs
|
|
Ntl_ManForEachCoNet( p2, pObj, i )
|
|
{
|
|
pNet = Ntl_ObjFanin0(pObj);
|
|
if ( !Ntl_ManCollapse_rec( p2, pNet ) )
|
|
{
|
|
printf( "Ntl_ManCollapseForSec(): Error: Combinational loop is detected.\n" );
|
|
return 0;
|
|
}
|
|
// Aig_ObjCreatePo( pAig, pNet->pCopy );
|
|
Vec_PtrWriteEntry( vPairs, 2 * i + 1, pNet->pCopy );
|
|
}
|
|
// visit the nodes starting from latch inputs outputs
|
|
Ntl_ModelForEachLatch( pRoot2, pObj, i )
|
|
{
|
|
pNet = Ntl_ObjFanin0(pObj);
|
|
if ( !Ntl_ManCollapse_rec( p2, pNet ) )
|
|
{
|
|
printf( "Ntl_ManCollapseForSec(): Error: Combinational loop is detected.\n" );
|
|
return 0;
|
|
}
|
|
if ( (pObj->LatchId & 3) == 1 )
|
|
Aig_ObjCreatePo( pAig, Aig_Not(pNet->pCopy) );
|
|
else
|
|
Aig_ObjCreatePo( pAig, pNet->pCopy );
|
|
}
|
|
|
|
/////////////////////////////////////////////////////
|
|
pMiter = Aig_Miter(pAig, vPairs);
|
|
Vec_PtrFree( vPairs );
|
|
Aig_ObjPatchFanin0( pAig, Aig_ManPo(pAig,0), pMiter );
|
|
pAig->nRegs = Ntl_ModelLatchNum( pRoot1 ) + Ntl_ModelLatchNum( pRoot2 );
|
|
Aig_ManCleanup( pAig );
|
|
return pAig;
|
|
}
|
|
|
|
|
|
/**Function*************************************************************
|
|
|
|
Synopsis [Increments reference counter of the net.]
|
|
|
|
Description []
|
|
|
|
SideEffects []
|
|
|
|
SeeAlso []
|
|
|
|
***********************************************************************/
|
|
static inline void Ntl_NetIncrementRefs( Ntl_Net_t * pNet )
|
|
{
|
|
int nRefs = (int)(long)pNet->pCopy;
|
|
pNet->pCopy = (void *)(long)(nRefs + 1);
|
|
}
|
|
|
|
/**Function*************************************************************
|
|
|
|
Synopsis [Extracts logic newtork out of the netlist.]
|
|
|
|
Description []
|
|
|
|
SideEffects []
|
|
|
|
SeeAlso []
|
|
|
|
***********************************************************************/
|
|
Nwk_Obj_t * Ntl_ManExtractNwk_rec( Ntl_Man_t * p, Ntl_Net_t * pNet, Nwk_Man_t * pNtk, Vec_Int_t * vCover, Vec_Int_t * vMemory )
|
|
{
|
|
extern Hop_Obj_t * Kit_CoverToHop( Hop_Man_t * pMan, Vec_Int_t * vCover, int nVars, Vec_Int_t * vMemory );
|
|
Ntl_Net_t * pFaninNet;
|
|
Nwk_Obj_t * pNode;
|
|
int i;
|
|
if ( pNet->fMark )
|
|
return pNet->pCopy;
|
|
pNet->fMark = 1;
|
|
pNode = Nwk_ManCreateNode( pNtk, Ntl_ObjFaninNum(pNet->pDriver), (int)(long)pNet->pCopy );
|
|
Ntl_ObjForEachFanin( pNet->pDriver, pFaninNet, i )
|
|
{
|
|
Ntl_ManExtractNwk_rec( p, pFaninNet, pNtk, vCover, vMemory );
|
|
Nwk_ObjAddFanin( pNode, pFaninNet->pCopy );
|
|
}
|
|
if ( Ntl_ObjFaninNum(pNet->pDriver) == 0 )
|
|
pNode->pFunc = Hop_NotCond( Hop_ManConst1(pNtk->pManHop), Kit_PlaIsConst0(pNet->pDriver->pSop) );
|
|
else
|
|
{
|
|
Kit_PlaToIsop( pNet->pDriver->pSop, vCover );
|
|
pNode->pFunc = Kit_CoverToHop( pNtk->pManHop, vCover, Ntl_ObjFaninNum(pNet->pDriver), vMemory );
|
|
if ( Kit_PlaIsComplement(pNet->pDriver->pSop) )
|
|
pNode->pFunc = Hop_Not(pNode->pFunc);
|
|
}
|
|
return pNet->pCopy = pNode;
|
|
}
|
|
|
|
/**Function*************************************************************
|
|
|
|
Synopsis [Extracts logic network out of the netlist.]
|
|
|
|
Description []
|
|
|
|
SideEffects []
|
|
|
|
SeeAlso []
|
|
|
|
***********************************************************************/
|
|
Nwk_Man_t * Ntl_ManExtractNwk( Ntl_Man_t * p, Aig_Man_t * pAig )
|
|
{
|
|
Nwk_Man_t * pNtk;
|
|
Nwk_Obj_t * pNode;
|
|
Ntl_Mod_t * pRoot;
|
|
Ntl_Net_t * pNet;
|
|
Ntl_Obj_t * pObj;
|
|
Aig_Obj_t * pAnd;
|
|
Vec_Int_t * vCover, * vMemory;
|
|
int i, k;
|
|
pRoot = Ntl_ManRootModel( p );
|
|
assert( Ntl_ModelBoxNum(pRoot) == 0 );
|
|
assert( Ntl_ModelLatchNum(pRoot) == 0 );
|
|
assert( Ntl_ModelPiNum(pRoot) == Aig_ManPiNum(pAig) );
|
|
assert( Ntl_ModelPoNum(pRoot) == Aig_ManPoNum(pAig) );
|
|
vCover = Vec_IntAlloc( 100 );
|
|
vMemory = Vec_IntAlloc( 1 << 16 );
|
|
// count the number of fanouts of each net
|
|
Ntl_ModelForEachNet( pRoot, pNet, i )
|
|
{
|
|
pNet->pCopy = NULL;
|
|
pNet->fMark = 0;
|
|
}
|
|
Ntl_ModelForEachObj( pRoot, pObj, i )
|
|
Ntl_ObjForEachFanin( pObj, pNet, k )
|
|
Ntl_NetIncrementRefs( pNet );
|
|
// construct the network
|
|
pNtk = Nwk_ManAlloc();
|
|
pNtk->pName = Aig_UtilStrsav( pAig->pName );
|
|
pNtk->pSpec = Aig_UtilStrsav( pAig->pSpec );
|
|
Aig_ManSetPioNumbers( pAig );
|
|
Aig_ManForEachObj( pAig, pAnd, i )
|
|
{
|
|
if ( Aig_ObjIsPi(pAnd) )
|
|
{
|
|
pObj = Ntl_ModelPi( pRoot, Aig_ObjPioNum(pAnd) );
|
|
pNet = Ntl_ObjFanout0(pObj);
|
|
pNet->fMark = 1;
|
|
pNet->pCopy = Nwk_ManCreateCi( pNtk, (int)(long)pNet->pCopy );
|
|
}
|
|
else if ( Aig_ObjIsPo(pAnd) )
|
|
{
|
|
pObj = Ntl_ModelPo( pRoot, Aig_ObjPioNum(pAnd) );
|
|
pNet = Ntl_ObjFanin0(pObj);
|
|
pNet->pCopy = Ntl_ManExtractNwk_rec( p, pNet, pNtk, vCover, vMemory );
|
|
pNode = Nwk_ManCreateCo( pNtk );
|
|
Nwk_ObjAddFanin( pNode, pNet->pCopy );
|
|
}
|
|
}
|
|
Aig_ManCleanPioNumbers( pAig );
|
|
Ntl_ModelForEachNet( pRoot, pNet, i )
|
|
{
|
|
pNet->pCopy = NULL;
|
|
pNet->fMark = 0;
|
|
}
|
|
Vec_IntFree( vCover );
|
|
Vec_IntFree( vMemory );
|
|
// create timing manager from the current design
|
|
return pNtk;
|
|
}
|
|
|
|
/**Function*************************************************************
|
|
|
|
Synopsis [Extracts logic newtork out of the netlist.]
|
|
|
|
Description []
|
|
|
|
SideEffects []
|
|
|
|
SeeAlso []
|
|
|
|
***********************************************************************/
|
|
Nwk_Man_t * Ntl_ManReadNwk( char * pFileName, Aig_Man_t * pAig, Tim_Man_t * pManTime )
|
|
{
|
|
Nwk_Man_t * pNtk;
|
|
Ntl_Man_t * pNtl;
|
|
Ntl_Mod_t * pRoot;
|
|
pNtl = Ioa_ReadBlif( pFileName, 1 );
|
|
if ( pNtl == NULL )
|
|
{
|
|
printf( "Ntl_ManReadNwk(): Reading BLIF has failed.\n" );
|
|
return NULL;
|
|
}
|
|
pRoot = Ntl_ManRootModel( pNtl );
|
|
if ( Ntl_ModelPiNum(pRoot) != Aig_ManPiNum(pAig) )
|
|
{
|
|
printf( "Ntl_ManReadNwk(): The number of primary inputs does not match (%d and %d).\n",
|
|
Ntl_ModelPiNum(pRoot), Aig_ManPiNum(pAig) );
|
|
return NULL;
|
|
}
|
|
if ( Ntl_ModelPoNum(pRoot) != Aig_ManPoNum(pAig) )
|
|
{
|
|
printf( "Ntl_ManReadNwk(): The number of primary outputs does not match (%d and %d).\n",
|
|
Ntl_ModelPoNum(pRoot), Aig_ManPoNum(pAig) );
|
|
return NULL;
|
|
}
|
|
pNtk = Ntl_ManExtractNwk( pNtl, pAig );
|
|
Ntl_ManFree( pNtl );
|
|
if ( pManTime )
|
|
pNtk->pManTime = Tim_ManDup( pManTime, 0 );
|
|
return pNtk;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
/// END OF FILE ///
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
|