summaryrefslogtreecommitdiffstats
path: root/src/map/amap
diff options
context:
space:
mode:
authorAlan Mishchenko <alanmi@berkeley.edu>2009-01-18 08:01:00 -0800
committerAlan Mishchenko <alanmi@berkeley.edu>2009-01-18 08:01:00 -0800
commitf936cc0680c98ffe51b3a1716c996072d5dbf76c (patch)
tree784a2a809fb6b972ec6a8e2758ab758ca590d01a /src/map/amap
parentc9ad5880cc61787dec6d018111b63023407ce0e6 (diff)
downloadabc-f936cc0680c98ffe51b3a1716c996072d5dbf76c.tar.gz
abc-f936cc0680c98ffe51b3a1716c996072d5dbf76c.tar.bz2
abc-f936cc0680c98ffe51b3a1716c996072d5dbf76c.zip
Version abc90118
Diffstat (limited to 'src/map/amap')
-rw-r--r--src/map/amap/amap.h83
-rw-r--r--src/map/amap/amapCore.c103
-rw-r--r--src/map/amap/amapGraph.c389
-rw-r--r--src/map/amap/amapInt.h380
-rw-r--r--src/map/amap/amapLib.c361
-rw-r--r--src/map/amap/amapMan.c99
-rw-r--r--src/map/amap/amapMatch.c538
-rw-r--r--src/map/amap/amapMerge.c521
-rw-r--r--src/map/amap/amapOutput.c181
-rw-r--r--src/map/amap/amapParse.c457
-rw-r--r--src/map/amap/amapPerm.c344
-rw-r--r--src/map/amap/amapRead.c412
-rw-r--r--src/map/amap/amapRule.c368
-rw-r--r--src/map/amap/amapUniq.c312
-rw-r--r--src/map/amap/module.make12
15 files changed, 4560 insertions, 0 deletions
diff --git a/src/map/amap/amap.h b/src/map/amap/amap.h
new file mode 100644
index 00000000..ee845e7f
--- /dev/null
+++ b/src/map/amap/amap.h
@@ -0,0 +1,83 @@
+/**CFile****************************************************************
+
+ FileName [amap.h]
+
+ SystemName [ABC: Logic synthesis and verification system.]
+
+ PackageName [Technology mapper for standard cells.]
+
+ Synopsis [External declarations.]
+
+ Author [Alan Mishchenko]
+
+ Affiliation [UC Berkeley]
+
+ Date [Ver. 1.0. Started - June 20, 2005.]
+
+ Revision [$Id: amap.h,v 1.00 2005/06/20 00:00:00 alanmi Exp $]
+
+***********************************************************************/
+
+#ifndef __AMAP_H__
+#define __AMAP_H__
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+////////////////////////////////////////////////////////////////////////
+/// INCLUDES ///
+////////////////////////////////////////////////////////////////////////
+
+////////////////////////////////////////////////////////////////////////
+/// PARAMETERS ///
+////////////////////////////////////////////////////////////////////////
+
+////////////////////////////////////////////////////////////////////////
+/// BASIC TYPES ///
+////////////////////////////////////////////////////////////////////////
+
+
+typedef struct Amap_Par_t_ Amap_Par_t;
+struct Amap_Par_t_
+{
+ int nIterFlow; // iterations of area flow
+ int nIterArea; // iteratoins of exact area
+ int fUseMuxes; // enables the use of MUXes
+ int fUseXors; // enables the use of XORs
+ int fFreeInvs; // assume inverters are free (area = 0)
+ float fEpsilon; // used to compare floating point numbers
+ int fVerbose; // verbosity flag
+};
+
+typedef struct Amap_Out_t_ Amap_Out_t;
+struct Amap_Out_t_
+{
+ char * pName; // gate name
+ short Type; // node type (-1=input; 0=internal; 1=output)
+ short nFans; // number of fanins
+ int pFans[0]; // fanin
+};
+
+////////////////////////////////////////////////////////////////////////
+/// MACRO DEFINITIONS ///
+////////////////////////////////////////////////////////////////////////
+
+////////////////////////////////////////////////////////////////////////
+/// FUNCTION DECLARATIONS ///
+////////////////////////////////////////////////////////////////////////
+
+/*=== amapCore.c ==========================================================*/
+extern void Amap_ManSetDefaultParams( Amap_Par_t * pPars );
+extern Vec_Ptr_t * Amap_ManTest( Aig_Man_t * pAig, Amap_Par_t * pPars );
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif
+
+////////////////////////////////////////////////////////////////////////
+/// END OF FILE ///
+////////////////////////////////////////////////////////////////////////
+
diff --git a/src/map/amap/amapCore.c b/src/map/amap/amapCore.c
new file mode 100644
index 00000000..f1554862
--- /dev/null
+++ b/src/map/amap/amapCore.c
@@ -0,0 +1,103 @@
+/**CFile****************************************************************
+
+ FileName [amapCore.c]
+
+ SystemName [ABC: Logic synthesis and verification system.]
+
+ PackageName [Technology mapper for standard cells.]
+
+ Synopsis [Core mapping procedures.]
+
+ Author [Alan Mishchenko]
+
+ Affiliation [UC Berkeley]
+
+ Date [Ver. 1.0. Started - June 20, 2005.]
+
+ Revision [$Id: amapCore.c,v 1.00 2005/06/20 00:00:00 alanmi Exp $]
+
+***********************************************************************/
+
+#include "amapInt.h"
+
+////////////////////////////////////////////////////////////////////////
+/// DECLARATIONS ///
+////////////////////////////////////////////////////////////////////////
+
+////////////////////////////////////////////////////////////////////////
+/// FUNCTION DEFINITIONS ///
+////////////////////////////////////////////////////////////////////////
+
+/**Function*************************************************************
+
+ Synopsis [This procedure sets default parameters.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+void Amap_ManSetDefaultParams( Amap_Par_t * p )
+{
+ memset( p, 0, sizeof(Amap_Par_t) );
+ p->nIterFlow = 1; // iterations of area flow
+ p->nIterArea = 4; // iteratoins of exact area
+ p->fUseMuxes = 0; // enables the use of MUXes
+ p->fUseXors = 1; // enables the use of XORs
+ p->fFreeInvs = 0; // assume inverters are free (area = 0)
+ p->fEpsilon = (float)0.001; // used to compare floating point numbers
+ p->fVerbose = 0; // verbosity flag
+}
+
+/**Function*************************************************************
+
+ Synopsis []
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Vec_Ptr_t * Amap_ManTest( Aig_Man_t * pAig, Amap_Par_t * pPars )
+{
+ extern void * Abc_FrameReadLibGen2();
+ Vec_Ptr_t * vRes;
+ Amap_Man_t * p;
+ Amap_Lib_t * pLib;
+ int clkTotal = clock();
+ pLib = Abc_FrameReadLibGen2();
+ if ( pLib == NULL )
+ {
+ printf( "Library is not available.\n" );
+ return NULL;
+ }
+ p = Amap_ManStart( Aig_ManNodeNum(pAig) );
+ p->pPars = pPars;
+ p->pLib = pLib;
+ p->fAreaInv = pPars->fFreeInvs? 0.0 : pLib->pGateInv->dArea;
+ p->fUseMux = pPars->fUseMuxes && pLib->fHasMux;
+ p->fUseXor = pPars->fUseXors && pLib->fHasXor;
+ p->ppCutsTemp = CALLOC( Amap_Cut_t *, 2 * pLib->nNodes );
+ p->pMatsTemp = CALLOC( int, 2 * pLib->nNodes );
+ Amap_ManCreate( p, pAig );
+ Amap_ManMap( p );
+ vRes = NULL;
+ vRes = Amap_ManProduceMapped( p );
+ Amap_ManStop( p );
+if ( pPars->fVerbose )
+{
+PRT( "Total runtime", clock() - clkTotal );
+}
+ return vRes;
+}
+
+
+////////////////////////////////////////////////////////////////////////
+/// END OF FILE ///
+////////////////////////////////////////////////////////////////////////
+
+
diff --git a/src/map/amap/amapGraph.c b/src/map/amap/amapGraph.c
new file mode 100644
index 00000000..83cadc2c
--- /dev/null
+++ b/src/map/amap/amapGraph.c
@@ -0,0 +1,389 @@
+/**CFile****************************************************************
+
+ FileName [amapGraph.c]
+
+ SystemName [ABC: Logic synthesis and verification system.]
+
+ PackageName [Technology mapper for standard cells.]
+
+ Synopsis [Internal AIG manager.]
+
+ Author [Alan Mishchenko]
+
+ Affiliation [UC Berkeley]
+
+ Date [Ver. 1.0. Started - June 20, 2005.]
+
+ Revision [$Id: amapGraph.c,v 1.00 2005/06/20 00:00:00 alanmi Exp $]
+
+***********************************************************************/
+
+#include "amapInt.h"
+
+////////////////////////////////////////////////////////////////////////
+/// DECLARATIONS ///
+////////////////////////////////////////////////////////////////////////
+
+////////////////////////////////////////////////////////////////////////
+/// FUNCTION DEFINITIONS ///
+////////////////////////////////////////////////////////////////////////
+
+/**Function*************************************************************
+
+ Synopsis [Creates object.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Amap_Obj_t * Amap_ManSetupObj( Amap_Man_t * p )
+{
+ Amap_Obj_t * pObj;
+ pObj = (Amap_Obj_t *)Aig_MmFixedEntryFetch( p->pMemObj );
+ memset( pObj, 0, sizeof(Amap_Obj_t) );
+ pObj->nFouts[0] = 1; // needed for flow to work in the first pass
+ pObj->Id = Vec_PtrSize(p->vObjs);
+ Vec_PtrPush( p->vObjs, pObj );
+ return pObj;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Creates constant 1 node.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Amap_Obj_t * Amap_ManCreateConst1( Amap_Man_t * p )
+{
+ Amap_Obj_t * pObj;
+ pObj = Amap_ManSetupObj( p );
+ pObj->Type = AMAP_OBJ_CONST1;
+ pObj->fPhase = 1;
+ p->nObjs[AMAP_OBJ_CONST1]++;
+ return pObj;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Creates primary input.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Amap_Obj_t * Amap_ManCreatePi( Amap_Man_t * p )
+{
+ Amap_Obj_t * pObj;
+ pObj = Amap_ManSetupObj( p );
+ pObj->Type = AMAP_OBJ_PI;
+ pObj->IdPio = Vec_PtrSize( p->vPis );
+ Vec_PtrPush( p->vPis, pObj );
+ p->nObjs[AMAP_OBJ_PI]++;
+ return pObj;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Creates primary output with the given driver.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Amap_Obj_t * Amap_ManCreatePo( Amap_Man_t * p, Amap_Obj_t * pFan0 )
+{
+ Amap_Obj_t * pObj;
+ pObj = Amap_ManSetupObj( p );
+ pObj->IdPio = Vec_PtrSize( p->vPos );
+ Vec_PtrPush( p->vPos, pObj );
+ pObj->Type = AMAP_OBJ_PO;
+ pObj->Fan[0] = Amap_ObjToLit(pFan0); Amap_Regular(pFan0)->nRefs++;
+ pObj->Level = Amap_Regular(pFan0)->Level;
+ if ( p->nLevelMax < (int)pObj->Level )
+ p->nLevelMax = (int)pObj->Level;
+ p->nObjs[AMAP_OBJ_PO]++;
+ return pObj;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Create the new node assuming it does not exist.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Amap_Obj_t * Amap_ManCreateAnd( Amap_Man_t * p, Amap_Obj_t * pFan0, Amap_Obj_t * pFan1 )
+{
+ Amap_Obj_t * pObj;
+ pObj = Amap_ManSetupObj( p );
+ pObj->Type = AMAP_OBJ_AND;
+ pObj->Fan[0] = Amap_ObjToLit(pFan0); Amap_Regular(pFan0)->nRefs++;
+ pObj->Fan[1] = Amap_ObjToLit(pFan1); Amap_Regular(pFan1)->nRefs++;
+ assert( Amap_Lit2Var(pObj->Fan[0]) != Amap_Lit2Var(pObj->Fan[1]) );
+ pObj->fPhase = Amap_ObjPhaseReal(pFan0) & Amap_ObjPhaseReal(pFan1);
+ pObj->Level = 1 + AIG_MAX( Amap_Regular(pFan0)->Level, Amap_Regular(pFan1)->Level );
+ if ( p->nLevelMax < (int)pObj->Level )
+ p->nLevelMax = (int)pObj->Level;
+ p->nObjs[AMAP_OBJ_AND]++;
+ return pObj;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Create the new node assuming it does not exist.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Amap_Obj_t * Amap_ManCreateXor( Amap_Man_t * p, Amap_Obj_t * pFan0, Amap_Obj_t * pFan1 )
+{
+ Amap_Obj_t * pObj;
+ pObj = Amap_ManSetupObj( p );
+ pObj->Type = AMAP_OBJ_XOR;
+ pObj->Fan[0] = Amap_ObjToLit(pFan0); Amap_Regular(pFan0)->nRefs++;
+ pObj->Fan[1] = Amap_ObjToLit(pFan1); Amap_Regular(pFan1)->nRefs++;
+ pObj->fPhase = Amap_ObjPhaseReal(pFan0) ^ Amap_ObjPhaseReal(pFan1);
+ pObj->Level = 2 + AIG_MAX( Amap_Regular(pFan0)->Level, Amap_Regular(pFan1)->Level );
+ if ( p->nLevelMax < (int)pObj->Level )
+ p->nLevelMax = (int)pObj->Level;
+ p->nObjs[AMAP_OBJ_XOR]++;
+ return pObj;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Create the new node assuming it does not exist.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Amap_Obj_t * Amap_ManCreateMux( Amap_Man_t * p, Amap_Obj_t * pFan0, Amap_Obj_t * pFan1, Amap_Obj_t * pFanC )
+{
+ Amap_Obj_t * pObj;
+ pObj = Amap_ManSetupObj( p );
+ pObj->Type = AMAP_OBJ_MUX;
+ pObj->Fan[0] = Amap_ObjToLit(pFan0); Amap_Regular(pFan0)->nRefs++;
+ pObj->Fan[1] = Amap_ObjToLit(pFan1); Amap_Regular(pFan1)->nRefs++;
+ pObj->Fan[2] = Amap_ObjToLit(pFanC); Amap_Regular(pFanC)->nRefs++;
+ pObj->fPhase = (Amap_ObjPhaseReal(pFan1) & Amap_ObjPhaseReal(pFanC)) |
+ (Amap_ObjPhaseReal(pFan0) & ~Amap_ObjPhaseReal(pFanC));
+ pObj->Level = AIG_MAX( Amap_Regular(pFan0)->Level, Amap_Regular(pFan1)->Level );
+ pObj->Level = 2 + AIG_MAX( pObj->Level, Amap_Regular(pFanC)->Level );
+ if ( p->nLevelMax < (int)pObj->Level )
+ p->nLevelMax = (int)pObj->Level;
+ p->nObjs[AMAP_OBJ_MUX]++;
+ return pObj;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Creates the choice node.]
+
+ Description [Should be called after the equivalence class nodes are linked.]
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+void Amap_ManCreateChoice( Amap_Man_t * p, Amap_Obj_t * pObj )
+{
+ Amap_Obj_t * pTemp;
+ // mark the node as a representative if its class
+// assert( pObj->fRepr == 0 );
+ pObj->fRepr = 1;
+ // update the level of this node (needed for correct required time computation)
+ for ( pTemp = pObj; pTemp; pTemp = Amap_ObjChoice(p, pTemp) )
+ {
+ pObj->Level = AIG_MAX( pObj->Level, pTemp->Level );
+// pTemp->nVisits++; pTemp->nVisitsCopy++;
+ }
+ // mark the largest level
+ if ( p->nLevelMax < (int)pObj->Level )
+ p->nLevelMax = (int)pObj->Level;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Creates XOR/MUX choices for the node.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+void Amap_ManCreateXorChoices( Amap_Man_t * p, Amap_Obj_t * pFan0, Amap_Obj_t * pFan1, Amap_Obj_t * pChoices[] )
+{
+ pChoices[0] = Amap_ManCreateXor( p, pFan0, pFan1 );
+ pChoices[1] = Amap_ManCreateXor( p, Amap_Not(pFan0), pFan1 );
+ pChoices[2] = Amap_ManCreateXor( p, pFan0, Amap_Not(pFan1) );
+ pChoices[3] = Amap_ManCreateXor( p, Amap_Not(pFan0), Amap_Not(pFan1) );
+}
+
+/**Function*************************************************************
+
+ Synopsis [Creates XOR/MUX choices for the node.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+void Amap_ManCreateMuxChoices( Amap_Man_t * p, Amap_Obj_t * pFan0, Amap_Obj_t * pFan1, Amap_Obj_t * pFanC, Amap_Obj_t * pChoices[] )
+{
+ pChoices[0] = Amap_ManCreateMux( p, pFan0, pFan1, pFanC );
+ pChoices[1] = Amap_ManCreateMux( p, Amap_Not(pFan0), Amap_Not(pFan1), pFanC );
+ pChoices[2] = Amap_ManCreateMux( p, pFan1, pFan0, Amap_Not(pFanC) );
+ pChoices[3] = Amap_ManCreateMux( p, Amap_Not(pFan1), Amap_Not(pFan0), Amap_Not(pFanC) );
+}
+
+/**Function*************************************************************
+
+ Synopsis [Drags pointer out through the copy.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+static inline Amap_Obj_t * Amap_AndToObj( Aig_Obj_t * pObj )
+{
+ return Amap_NotCond( (Amap_Obj_t *)Aig_Regular(pObj)->pData, Aig_IsComplement(pObj) );
+}
+
+/**Function*************************************************************
+
+ Synopsis [Starts the AIG manager.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Amap_Obj_t * Amap_ManGetLast_rec( Amap_Man_t * p, Amap_Obj_t * pObj )
+{
+ if ( pObj->Equiv == 0 )
+ return pObj;
+ return Amap_ManGetLast_rec( p, Amap_ObjChoice(p, pObj) );
+}
+
+/**Function*************************************************************
+
+ Synopsis [Starts the AIG manager.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+void Amap_ManCreate( Amap_Man_t * p, Aig_Man_t * pAig )
+{
+ Vec_Ptr_t * vNodes;
+ Amap_Obj_t * pChoices[4];
+ Aig_Obj_t * pObj, * pFanin, * pPrev, * pFan0, * pFan1, * pFanC;
+ int i, fChoices;
+ if ( pAig->pEquivs )
+ vNodes = Aig_ManDfsChoices( pAig );
+ else
+ vNodes = Aig_ManDfs( pAig, 1 );
+ p->pConst1 = Amap_ManCreateConst1( p );
+ // print warning about excessive memory usage
+ if ( p->pPars->fVerbose )
+ {
+ if ( 1.0 * Aig_ManObjNum(pAig) * sizeof(Amap_Obj_t) / (1<<30) > 0.1 )
+ printf( "Warning: Mapper allocates %.3f Gb for subject graph with %d objects.\n",
+ 1.0 * Aig_ManObjNum(pAig) * sizeof(Amap_Obj_t) / (1<<30), Aig_ManObjNum(pAig) );
+ }
+ // create PIs and remember them in the old nodes
+ Aig_ManCleanData(pAig);
+ Aig_ManConst1(pAig)->pData = Amap_ManConst1( p );
+ Aig_ManForEachPi( pAig, pObj, i )
+ pObj->pData = Amap_ManCreatePi( p );
+ // load the AIG into the mapper
+ Vec_PtrForEachEntry( vNodes, pObj, i )
+ {
+ fChoices = 0;
+ if ( p->fUseXor && Aig_ObjRecognizeExor(pObj, &pFan0, &pFan1 ) )
+ {
+ Amap_ManCreateXorChoices( p, Amap_AndToObj(pFan0), Amap_AndToObj(pFan1), pChoices );
+ fChoices = 1;
+ }
+ else if ( p->fUseMux && Aig_ObjIsMuxType(pObj) )
+ {
+ pFanC = Aig_ObjRecognizeMux( pObj, &pFan1, &pFan0 );
+ Amap_ManCreateMuxChoices( p, Amap_AndToObj(pFan0), Amap_AndToObj(pFan1), Amap_AndToObj(pFanC), pChoices );
+ fChoices = 1;
+ }
+ pObj->pData = Amap_ManCreateAnd( p, (Amap_Obj_t *)Aig_ObjChild0Copy(pObj), (Amap_Obj_t *)Aig_ObjChild1Copy(pObj) );
+ if ( fChoices )
+ {
+ p->nChoicesAdded++;
+ Amap_ObjSetChoice( (Amap_Obj_t *)pObj->pData, pChoices[0] );
+ Amap_ObjSetChoice( pChoices[0], pChoices[1] );
+ Amap_ObjSetChoice( pChoices[1], pChoices[2] );
+ Amap_ObjSetChoice( pChoices[2], pChoices[3] );
+ Amap_ManCreateChoice( p, (Amap_Obj_t *)pObj->pData );
+ }
+ if ( Aig_ObjIsChoice( pAig, pObj ) )
+ {
+// assert( !fChoices );
+ p->nChoicesGiven++;
+ for ( pPrev = pObj, pFanin = Aig_ObjEquiv(pAig, pObj); pFanin; pPrev = pFanin, pFanin = Aig_ObjEquiv(pAig, pFanin) )
+ {
+ ((Amap_Obj_t *)pFanin->pData)->fRepr = 0;
+ Amap_ObjSetChoice( Amap_ManGetLast_rec(p, (Amap_Obj_t *)pPrev->pData),
+ (Amap_Obj_t *)pFanin->pData );
+ }
+ Amap_ManCreateChoice( p, (Amap_Obj_t *)pObj->pData );
+ }
+ }
+ Vec_PtrFree( vNodes );
+ // set the primary outputs without copying the phase
+ Aig_ManForEachPo( pAig, pObj, i )
+ pObj->pData = Amap_ManCreatePo( p, (Amap_Obj_t *)Aig_ObjChild0Copy(pObj) );
+ if ( p->pPars->fVerbose )
+ printf( "Performing mapping with %d given and %d created choices.\n",
+ p->nChoicesGiven, p->nChoicesAdded );
+}
+
+////////////////////////////////////////////////////////////////////////
+/// END OF FILE ///
+////////////////////////////////////////////////////////////////////////
+
+
diff --git a/src/map/amap/amapInt.h b/src/map/amap/amapInt.h
new file mode 100644
index 00000000..954790c9
--- /dev/null
+++ b/src/map/amap/amapInt.h
@@ -0,0 +1,380 @@
+/**CFile****************************************************************
+
+ FileName [amapInt.h]
+
+ SystemName [ABC: Logic synthesis and verification system.]
+
+ PackageName [Technology mapper for standard cells.]
+
+ Synopsis [Internal declarations.]
+
+ Author [Alan Mishchenko]
+
+ Affiliation [UC Berkeley]
+
+ Date [Ver. 1.0. Started - June 20, 2005.]
+
+ Revision [$Id: amapInt.h,v 1.00 2005/06/20 00:00:00 alanmi Exp $]
+
+***********************************************************************/
+
+#ifndef __AMAP_INT_H__
+#define __AMAP_INT_H__
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+////////////////////////////////////////////////////////////////////////
+/// INCLUDES ///
+////////////////////////////////////////////////////////////////////////
+
+#include "aig.h"
+#include "amap.h"
+
+////////////////////////////////////////////////////////////////////////
+/// PARAMETERS ///
+////////////////////////////////////////////////////////////////////////
+
+// the largest gate size in the library
+// (gates above this size will be ignored)
+#define AMAP_MAXINS 15
+
+#define AMAP_STRING_CONST0 "CONST0"
+#define AMAP_STRING_CONST1 "CONST1"
+
+// object types
+typedef enum {
+ AMAP_OBJ_NONE, // 0: non-existent object
+ AMAP_OBJ_CONST1, // 1: constant 1
+ AMAP_OBJ_PI, // 2: primary input
+ AMAP_OBJ_PO, // 3: primary output
+ AMAP_OBJ_AND, // 4: AND node
+ AMAP_OBJ_XOR, // 5: XOR node
+ AMAP_OBJ_MUX, // 6: MUX node
+ AMAP_OBJ_VOID // 7: unused object
+} Amap_Type_t;
+
+////////////////////////////////////////////////////////////////////////
+/// BASIC TYPES ///
+////////////////////////////////////////////////////////////////////////
+
+typedef struct Amap_Lib_t_ Amap_Lib_t;
+typedef struct Amap_Pin_t_ Amap_Pin_t;
+typedef struct Amap_Gat_t_ Amap_Gat_t;
+typedef struct Amap_Nod_t_ Amap_Nod_t;
+typedef struct Amap_Set_t_ Amap_Set_t;
+
+typedef struct Amap_Man_t_ Amap_Man_t;
+typedef struct Amap_Obj_t_ Amap_Obj_t;
+typedef struct Amap_Cut_t_ Amap_Cut_t;
+typedef struct Amap_Mat_t_ Amap_Mat_t;
+
+struct Amap_Man_t_
+{
+ // user data
+ Amap_Par_t * pPars;
+ Amap_Lib_t * pLib;
+ // internal parameters
+ float fEpsilonInternal;
+ float fAreaInv;
+ int fUseXor;
+ int fUseMux;
+ // internal AIG with choices
+ Vec_Ptr_t * vPis;
+ Vec_Ptr_t * vPos;
+ Vec_Ptr_t * vObjs;
+ Aig_MmFixed_t * pMemObj;
+ Aig_MmFlex_t * pMemCuts;
+ Aig_MmFlex_t * pMemCutBest;
+ Aig_MmFlex_t * pMemTemp;
+ Amap_Obj_t * pConst1;
+ int nObjs[AMAP_OBJ_VOID];
+ int nLevelMax;
+ int nChoicesGiven;
+ int nChoicesAdded;
+ // mapping data-structures
+ Vec_Int_t * vTemp;
+ int * pMatsTemp;
+ Amap_Cut_t ** ppCutsTemp;
+ Amap_Cut_t * pCutsPi;
+ Vec_Ptr_t * vCuts0;
+ Vec_Ptr_t * vCuts1;
+ Vec_Ptr_t * vCuts2;
+ // statistics
+ int nCutsUsed;
+ int nCutsTried;
+ int nCutsTried3;
+ int nBytesUsed;
+};
+struct Amap_Lib_t_
+{
+ char * pName; // library name
+ Vec_Ptr_t * vGates; // represenation of gates
+ Vec_Ptr_t * vSorted; // gates sorted for area-only mapping
+ Vec_Ptr_t * vSelect; // gates selected for area-only mapping
+ Amap_Gat_t * pGate0; // the constant zero gate
+ Amap_Gat_t * pGate1; // the constant one gate
+ Amap_Gat_t * pGateBuf; // the buffer
+ Amap_Gat_t * pGateInv; // the inverter
+ Aig_MmFlex_t * pMemGates; // memory manager for objects
+ int fHasXor; // XOR/NXOR gates are present
+ int fHasMux; // MUX/NMUX gates are present
+ // structural representation
+ int fVerbose; // enable detailed statistics
+ Amap_Nod_t * pNodes; // representation nodes
+ int nNodes; // the number of nodes used
+ int nNodesAlloc; // the number of nodes allocated
+ Vec_Ptr_t * vRules; // the rule of AND gate
+ Vec_Ptr_t * vRulesX; // the rule of XOR gate
+ Vec_Int_t * vRules3; // the rule of MUX gate
+ int ** pRules; // simplified representation
+ int ** pRulesX; // simplified representation
+ Aig_MmFlex_t * pMemSet; // memory manager for sets
+ int nSets; // the number of sets created
+};
+struct Amap_Pin_t_
+{
+ char * pName;
+ int Phase;
+ double dLoadInput;
+ double dLoadMax;
+ double dDelayBlockRise;
+ double dDelayFanoutRise;
+ double dDelayBlockFall;
+ double dDelayFanoutFall;
+ double dDelayBlockMax;
+};
+struct Amap_Gat_t_
+{
+ Amap_Lib_t * pLib; // library
+ char * pName; // the name of the gate
+ char * pOutName; // name of the output
+ double dArea; // the area of the gate
+ char * pForm; // the formula describing functionality
+ unsigned * pFunc; // truth table
+ unsigned Id : 23; // unique number of the gate
+ unsigned fMux : 1; // denotes MUX-gates
+ unsigned nPins : 8; // number of inputs
+ Amap_Pin_t Pins[0]; // description of inputs
+};
+struct Amap_Set_t_
+{
+ Amap_Set_t * pNext;
+ unsigned iGate : 16;
+ unsigned fInv : 1;
+ unsigned nIns : 15;
+ char Ins[AMAP_MAXINS];// mapping from gate inputs into fanins
+};
+struct Amap_Nod_t_
+{
+ unsigned Id : 16; // ID of the node
+ unsigned nSuppSize: 8; // support size
+ unsigned Type : 8; // the type of node
+ short iFan0; // fanin0
+ short iFan1; // fanin1
+ short iFan2; // fanin2
+ short Unused; //
+ Amap_Set_t * pSets; // implementable gates
+};
+struct Amap_Cut_t_
+{
+ unsigned iMat : 16;
+ unsigned fInv : 1;
+ unsigned nFans : 15;
+ int Fans[0];
+};
+struct Amap_Mat_t_
+{
+ Amap_Cut_t * pCut; // the cut
+ Amap_Set_t * pSet; // the set
+ float Area; // area flow / exact area of the node
+ float AveFan; // edge flow of the node
+ float Delay; // delay of the node
+};
+struct Amap_Obj_t_
+{
+ unsigned Type : 3;
+ unsigned Id : 29;
+ unsigned IdPio : 29;
+ unsigned fPhase : 1;
+ unsigned fRepr : 1;
+ unsigned fPolar : 1; // pCutBest->fInv ^ pSetBest->fInv
+ unsigned Level : 20;
+ unsigned nCuts : 12;
+ int nRefs;
+ int Equiv;
+ int Fan[3];
+ union {
+ void * pData;
+ int iData;
+ };
+ // match of the node
+ float EstRefs; // the number of estimated fanouts
+ int nFouts[2]; // the number of refs in each polarity
+ Amap_Mat_t Best; // the best match of the node
+};
+
+static inline int Amap_Var2Lit( int Var, int fCompl ) { return Var + Var + fCompl; }
+static inline int Amap_Lit2Var( int Lit ) { return Lit >> 1; }
+static inline int Amap_LitIsCompl( int Lit ) { return Lit & 1; }
+static inline int Amap_LitNot( int Lit ) { return Lit ^ 1; }
+static inline int Amap_LitNotCond( int Lit, int c ) { return Lit ^ (int)(c > 0); }
+static inline int Amap_LitRegular( int Lit ) { return Lit & ~01; }
+
+static inline Amap_Obj_t * Amap_Regular( Amap_Obj_t * p ) { return (Amap_Obj_t *)((PORT_PTRUINT_T)(p) & ~01); }
+static inline Amap_Obj_t * Amap_Not( Amap_Obj_t * p ) { return (Amap_Obj_t *)((PORT_PTRUINT_T)(p) ^ 01); }
+static inline Amap_Obj_t * Amap_NotCond( Amap_Obj_t * p, int c ) { return (Amap_Obj_t *)((PORT_PTRUINT_T)(p) ^ (c)); }
+static inline int Amap_IsComplement( Amap_Obj_t * p ) { return (int )(((PORT_PTRUINT_T)p) & 01); }
+
+static inline int Amap_ManPiNum( Amap_Man_t * p ) { return p->nObjs[AMAP_OBJ_PI]; }
+static inline int Amap_ManPoNum( Amap_Man_t * p ) { return p->nObjs[AMAP_OBJ_PO]; }
+static inline int Amap_ManAndNum( Amap_Man_t * p ) { return p->nObjs[AMAP_OBJ_AND]; }
+static inline int Amap_ManXorNum( Amap_Man_t * p ) { return p->nObjs[AMAP_OBJ_XOR]; }
+static inline int Amap_ManMuxNum( Amap_Man_t * p ) { return p->nObjs[AMAP_OBJ_MUX]; }
+static inline int Amap_ManObjNum( Amap_Man_t * p ) { return Vec_PtrSize(p->vObjs); }
+static inline int Amap_ManNodeNum( Amap_Man_t * p ) { return p->nObjs[AMAP_OBJ_AND] + p->nObjs[AMAP_OBJ_XOR] + p->nObjs[AMAP_OBJ_MUX]; }
+
+static inline Amap_Obj_t * Amap_ManConst1( Amap_Man_t * p ) { return p->pConst1; }
+static inline Amap_Obj_t * Amap_ManPi( Amap_Man_t * p, int i ) { return (Amap_Obj_t *)Vec_PtrEntry( p->vPis, i ); }
+static inline Amap_Obj_t * Amap_ManPo( Amap_Man_t * p, int i ) { return (Amap_Obj_t *)Vec_PtrEntry( p->vPos, i ); }
+static inline Amap_Obj_t * Amap_ManObj( Amap_Man_t * p, int i ) { return (Amap_Obj_t *)Vec_PtrEntry( p->vObjs, i ); }
+
+static inline int Amap_ObjIsConst1( Amap_Obj_t * pObj ) { return pObj->Type == AMAP_OBJ_CONST1; }
+static inline int Amap_ObjIsPi( Amap_Obj_t * pObj ) { return pObj->Type == AMAP_OBJ_PI; }
+static inline int Amap_ObjIsPo( Amap_Obj_t * pObj ) { return pObj->Type == AMAP_OBJ_PO; }
+static inline int Amap_ObjIsAnd( Amap_Obj_t * pObj ) { return pObj->Type == AMAP_OBJ_AND; }
+static inline int Amap_ObjIsXor( Amap_Obj_t * pObj ) { return pObj->Type == AMAP_OBJ_XOR; }
+static inline int Amap_ObjIsMux( Amap_Obj_t * pObj ) { return pObj->Type == AMAP_OBJ_MUX; }
+static inline int Amap_ObjIsNode( Amap_Obj_t * pObj ) { return pObj->Type == AMAP_OBJ_AND || pObj->Type == AMAP_OBJ_XOR || pObj->Type == AMAP_OBJ_MUX; }
+
+static inline int Amap_ObjToLit( Amap_Obj_t * pObj ) { return Amap_Var2Lit( Amap_Regular(pObj)->Id, Amap_IsComplement(pObj) ); }
+static inline Amap_Obj_t * Amap_ObjFanin0( Amap_Man_t * p, Amap_Obj_t * pObj ) { return Amap_ManObj(p, Amap_Lit2Var(pObj->Fan[0])); }
+static inline Amap_Obj_t * Amap_ObjFanin1( Amap_Man_t * p, Amap_Obj_t * pObj ) { return Amap_ManObj(p, Amap_Lit2Var(pObj->Fan[1])); }
+static inline Amap_Obj_t * Amap_ObjFanin2( Amap_Man_t * p, Amap_Obj_t * pObj ) { return Amap_ManObj(p, Amap_Lit2Var(pObj->Fan[2])); }
+static inline int Amap_ObjFaninC0( Amap_Obj_t * pObj ) { return Amap_LitIsCompl(pObj->Fan[0]); }
+static inline int Amap_ObjFaninC1( Amap_Obj_t * pObj ) { return Amap_LitIsCompl(pObj->Fan[1]); }
+static inline int Amap_ObjFaninC2( Amap_Obj_t * pObj ) { return Amap_LitIsCompl(pObj->Fan[2]); }
+static inline void * Amap_ObjCopy( Amap_Obj_t * pObj ) { return pObj->pData; }
+static inline int Amap_ObjLevel( Amap_Obj_t * pObj ) { return pObj->Level; }
+static inline void Amap_ObjSetLevel( Amap_Obj_t * pObj, int Level ) { pObj->Level = Level; }
+static inline void Amap_ObjSetCopy( Amap_Obj_t * pObj, void * pCopy ) { pObj->pData = pCopy; }
+static inline Amap_Obj_t * Amap_ObjChoice( Amap_Man_t * p, Amap_Obj_t * pObj ) { return pObj->Equiv? Amap_ManObj(p, pObj->Equiv) : NULL; }
+static inline void Amap_ObjSetChoice( Amap_Obj_t * pObj, Amap_Obj_t * pEqu){ assert(pObj->Equiv==0); pObj->Equiv = pEqu->Id; }
+static inline int Amap_ObjPhaseReal( Amap_Obj_t * pObj ) { return Amap_Regular(pObj)->fPhase ^ Amap_IsComplement(pObj); }
+static inline int Amap_ObjRefsTotal( Amap_Obj_t * pObj ) { return pObj->nFouts[0] + pObj->nFouts[1]; }
+
+static inline Amap_Gat_t * Amap_LibGate( Amap_Lib_t * p, int i ) { return Vec_PtrEntry(p->vGates, i); }
+static inline Amap_Nod_t * Amap_LibNod( Amap_Lib_t * p, int i ) { return p->pNodes + i; }
+
+// returns pointer to the next cut (internal cuts only)
+static inline Amap_Cut_t * Amap_ManCutNext( Amap_Cut_t * pCut )
+{ return (Amap_Cut_t *)(((int *)pCut)+pCut->nFans+1); }
+// returns pointer to the place of the next cut (temporary cuts only)
+static inline Amap_Cut_t ** Amap_ManCutNextP( Amap_Cut_t * pCut )
+{ return (Amap_Cut_t **)(((int *)pCut)+pCut->nFans+1); }
+
+extern void Kit_DsdPrintFromTruth( unsigned * pTruth, int nVars );
+
+////////////////////////////////////////////////////////////////////////
+/// MACRO DEFINITIONS ///
+////////////////////////////////////////////////////////////////////////
+
+// iterator over the primary inputs
+#define Amap_ManForEachPi( p, pObj, i ) \
+ Vec_PtrForEachEntry( p->vPis, pObj, i )
+// iterator over the primary outputs
+#define Amap_ManForEachPo( p, pObj, i ) \
+ Vec_PtrForEachEntry( p->vPos, pObj, i )
+// iterator over all objects, including those currently not used
+#define Amap_ManForEachObj( p, pObj, i ) \
+ Vec_PtrForEachEntry( p->vObjs, pObj, i ) if ( (pObj) == NULL ) {} else
+// iterator over all nodes
+#define Amap_ManForEachNode( p, pObj, i ) \
+ Vec_PtrForEachEntry( p->vObjs, pObj, i ) if ( (pObj) == NULL || !Amap_ObjIsNode(pObj) ) {} else
+
+// iterator through all gates of the library
+#define Amap_LibForEachGate( pLib, pGate, i ) \
+ Vec_PtrForEachEntry( pLib->vGates, pGate, i )
+// iterator through all pins of the gate
+#define Amap_GateForEachPin( pGate, pPin ) \
+ for ( pPin = pGate->Pins; pPin < pGate->Pins + pGate->nPins; pPin++ )
+
+// iterator through all cuts of the node
+#define Amap_NodeForEachCut( pNode, pCut, i ) \
+ for ( i = 0, pCut = (Amap_Cut_t *)pNode->pData; i < (int)pNode->nCuts; \
+ i++, pCut = Amap_ManCutNext(pCut) )
+
+// iterator through all sets of one library node
+#define Amap_LibNodeForEachSet( pNod, pSet ) \
+ for ( pSet = pNod->pSets; pSet; pSet = pSet->pNext )
+
+// iterates through each fanin of the match
+#define Amap_MatchForEachFaninCompl( p, pM, pFanin, fCompl, i ) \
+ for ( i = 0; i < (int)(pM)->pCut->nFans && \
+ ((pFanin = Amap_ManObj((p), Amap_Lit2Var((pM)->pCut->Fans[Amap_Lit2Var((pM)->pSet->Ins[i])]))), 1) && \
+ ((fCompl = Amap_LitIsCompl((pM)->pSet->Ins[i]) ^ Amap_LitIsCompl((pM)->pCut->Fans[Amap_Lit2Var((pM)->pSet->Ins[i])])), 1); \
+ i++ )
+
+// iterates through each fanin of the match
+#define Amap_MatchForEachFanin( p, pM, pFanin, i ) \
+ for ( i = 0; i < (int)(pM)->pCut->nFans && \
+ ((pFanin = Amap_ManObj((p), Amap_Lit2Var((pM)->pCut->Fans[Amap_Lit2Var((pM)->pSet->Ins[i])]))), 1); \
+ i++ )
+
+////////////////////////////////////////////////////////////////////////
+/// FUNCTION DECLARATIONS ///
+////////////////////////////////////////////////////////////////////////
+
+/*=== amapCore.c ==========================================================*/
+/*=== amapGraph.c ==========================================================*/
+extern Amap_Obj_t * Amap_ManCreatePi( Amap_Man_t * p );
+extern Amap_Obj_t * Amap_ManCreatePo( Amap_Man_t * p, Amap_Obj_t * pFan0 );
+extern Amap_Obj_t * Amap_ManCreateAnd( Amap_Man_t * p, Amap_Obj_t * pFan0, Amap_Obj_t * pFan1 );
+extern Amap_Obj_t * Amap_ManCreateXor( Amap_Man_t * p, Amap_Obj_t * pFan0, Amap_Obj_t * pFan1 );
+extern Amap_Obj_t * Amap_ManCreateMux( Amap_Man_t * p, Amap_Obj_t * pFanC, Amap_Obj_t * pFan1, Amap_Obj_t * pFan0 );
+extern void Amap_ManCreateChoice( Amap_Man_t * p, Amap_Obj_t * pObj );
+extern void Amap_ManCreate( Amap_Man_t * p, Aig_Man_t * pAig );
+/*=== amapLib.c ==========================================================*/
+extern Amap_Lib_t * Amap_LibAlloc();
+extern void Amap_LibFree( Amap_Lib_t * p );
+extern int Amap_LibNumPinsMax( Amap_Lib_t * p );
+extern void Amap_LibWrite( FILE * pFile, Amap_Lib_t * pLib, int fPrintDsd );
+extern Vec_Ptr_t * Amap_LibSelectGates( Amap_Lib_t * p, int fVerbose );
+extern void Amap_LibPrintSelectedGates( Amap_Lib_t * p, int fAllGates );
+extern Amap_Lib_t * Amap_LibReadAndPrepare( char * pFileName, int fVerbose, int fVeryVerbose );
+/*=== amapMan.c ==========================================================*/
+extern Amap_Man_t * Amap_ManStart( int nNodes );
+extern void Amap_ManStop( Amap_Man_t * p );
+/*=== amapMatch.c ==========================================================*/
+extern void Amap_ManMap( Amap_Man_t * p );
+/*=== amapMerge.c ==========================================================*/
+extern void Amap_ManMerge( Amap_Man_t * p );
+/*=== amapOutput.c ==========================================================*/
+extern Vec_Ptr_t * Amap_ManProduceMapped( Amap_Man_t * p );
+/*=== amapParse.c ==========================================================*/
+extern int Amap_LibParseEquations( Amap_Lib_t * p, int fVerbose );
+/*=== amapPerm.c ==========================================================*/
+/*=== amapRead.c ==========================================================*/
+extern Amap_Lib_t * Amap_LibReadFile( char * pFileName, int fVerbose );
+/*=== amapRule.c ==========================================================*/
+extern short * Amap_LibTableFindNode( Amap_Lib_t * p, int iFan0, int iFan1, int fXor );
+extern void Amap_LibCreateRules( Amap_Lib_t * p, int fVeryVerbose );
+/*=== amapUniq.c ==========================================================*/
+extern int Amap_LibFindNode( Amap_Lib_t * pLib, int iFan0, int iFan1, int fXor );
+extern int Amap_LibFindMux( Amap_Lib_t * p, int iFan0, int iFan1, int iFan2 );
+extern int Amap_LibCreateVar( Amap_Lib_t * p );
+extern int Amap_LibCreateNode( Amap_Lib_t * p, int iFan0, int iFan1, int fXor );
+extern int Amap_LibCreateMux( Amap_Lib_t * p, int iFan0, int iFan1, int iFan2 );
+extern int ** Amap_LibLookupTableAlloc( Vec_Ptr_t * vVec, int fVerbose );
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif
+
+////////////////////////////////////////////////////////////////////////
+/// END OF FILE ///
+////////////////////////////////////////////////////////////////////////
+
diff --git a/src/map/amap/amapLib.c b/src/map/amap/amapLib.c
new file mode 100644
index 00000000..816f0703
--- /dev/null
+++ b/src/map/amap/amapLib.c
@@ -0,0 +1,361 @@
+/**CFile****************************************************************
+
+ FileName [amapLib.c]
+
+ SystemName [ABC: Logic synthesis and verification system.]
+
+ PackageName [Technology mapper for standard cells.]
+
+ Synopsis [Standard-cell library.]
+
+ Author [Alan Mishchenko]
+
+ Affiliation [UC Berkeley]
+
+ Date [Ver. 1.0. Started - June 20, 2005.]
+
+ Revision [$Id: amapLib.c,v 1.00 2005/06/20 00:00:00 alanmi Exp $]
+
+***********************************************************************/
+
+#include "amapInt.h"
+
+////////////////////////////////////////////////////////////////////////
+/// DECLARATIONS ///
+////////////////////////////////////////////////////////////////////////
+
+////////////////////////////////////////////////////////////////////////
+/// FUNCTION DEFINITIONS ///
+////////////////////////////////////////////////////////////////////////
+
+/**Function*************************************************************
+
+ Synopsis [Allocs a library.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Amap_Lib_t * Amap_LibAlloc()
+{
+ Amap_Lib_t * p;
+ p = (Amap_Lib_t *)ALLOC( Amap_Lib_t, 1 );
+ memset( p, 0, sizeof(Amap_Lib_t) );
+ p->vGates = Vec_PtrAlloc( 100 );
+ p->pMemGates = Aig_MmFlexStart();
+ p->pMemSet = Aig_MmFlexStart();
+ return p;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Deallocs a library.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+void Amap_LibFree( Amap_Lib_t * p )
+{
+ if ( p == NULL )
+ return;
+ if ( p->vSelect )
+ Vec_PtrFree( p->vSelect );
+ if ( p->vSorted )
+ Vec_PtrFree( p->vSorted );
+ if ( p->vGates )
+ Vec_PtrFree( p->vGates );
+ if ( p->vRules )
+ Vec_VecFree( (Vec_Vec_t *)p->vRules );
+ if ( p->vRulesX )
+ Vec_VecFree( (Vec_Vec_t *)p->vRulesX );
+ if ( p->vRules3 )
+ Vec_IntFree( p->vRules3 );
+ Aig_MmFlexStop( p->pMemGates, 0 );
+ Aig_MmFlexStop( p->pMemSet, 0 );
+ FREE( p->pRules );
+ FREE( p->pRulesX );
+ FREE( p->pNodes );
+ free( p );
+}
+
+/**Function*************************************************************
+
+ Synopsis [Returns the largest gate size.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+int Amap_LibNumPinsMax( Amap_Lib_t * p )
+{
+ Amap_Gat_t * pGate;
+ int i, Counter = 0;
+ Amap_LibForEachGate( p, pGate, i )
+ if ( Counter < (int)pGate->nPins )
+ Counter = pGate->nPins;
+ return Counter;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Writes one pin.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+void Amap_LibWritePin( FILE * pFile, Amap_Pin_t * pPin )
+{
+ char * pPhaseNames[10] = { "UNKNOWN", "INV", "NONINV" };
+ fprintf( pFile, " PIN " );
+ fprintf( pFile, "%9s ", pPin->pName );
+ fprintf( pFile, "%10s ", pPhaseNames[pPin->Phase] );
+ fprintf( pFile, "%6d ", (int)pPin->dLoadInput );
+ fprintf( pFile, "%6d ", (int)pPin->dLoadMax );
+ fprintf( pFile, "%6.2f ", pPin->dDelayBlockRise );
+ fprintf( pFile, "%6.2f ", pPin->dDelayFanoutRise );
+ fprintf( pFile, "%6.2f ", pPin->dDelayBlockFall );
+ fprintf( pFile, "%6.2f", pPin->dDelayFanoutFall );
+ fprintf( pFile, "\n" );
+}
+
+/**Function*************************************************************
+
+ Synopsis [Writes one gate.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+void Amap_LibWriteGate( FILE * pFile, Amap_Gat_t * pGate, int fPrintDsd )
+{
+ Amap_Pin_t * pPin;
+ fprintf( pFile, "GATE " );
+ fprintf( pFile, "%12s ", pGate->pName );
+ fprintf( pFile, "%10.2f ", pGate->dArea );
+ fprintf( pFile, "%s=%s;\n", pGate->pOutName, pGate->pForm );
+ if ( fPrintDsd )
+ {
+ if ( pGate->pFunc == NULL )
+ printf( "Truth table is not available.\n" );
+ else
+ Kit_DsdPrintFromTruth( pGate->pFunc, pGate->nPins );
+ }
+ Amap_GateForEachPin( pGate, pPin )
+ Amap_LibWritePin( pFile, pPin );
+}
+
+/**Function*************************************************************
+
+ Synopsis [Writes library.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+void Amap_LibWrite( FILE * pFile, Amap_Lib_t * pLib, int fPrintDsd )
+{
+ Amap_Gat_t * pGate;
+ int i;
+ fprintf( pFile, "# The genlib library \"%s\".\n", pLib->pName );
+ Amap_LibForEachGate( pLib, pGate, i )
+ Amap_LibWriteGate( pFile, pGate, fPrintDsd );
+}
+
+/**Function*************************************************************
+
+ Synopsis [Compares two gates by area.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+int Amap_LibCompareGatesByArea( Amap_Gat_t ** pp1, Amap_Gat_t ** pp2 )
+{
+ double Diff = (*pp1)->dArea - (*pp2)->dArea;
+ if ( Diff < 0.0 )
+ return -1;
+ if ( Diff > 0.0 )
+ return 1;
+ return 0;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Compares gates by area.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Vec_Ptr_t * Amap_LibSortGatesByArea( Amap_Lib_t * pLib )
+{
+ Vec_Ptr_t * vSorted;
+ vSorted = Vec_PtrDup( pLib->vGates );
+ qsort( (void *)Vec_PtrArray(vSorted), Vec_PtrSize(vSorted), sizeof(void *),
+ (int (*)(const void *, const void *)) Amap_LibCompareGatesByArea );
+ return vSorted;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Finds min-area gate with the given function.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Amap_Gat_t * Amap_LibFindGate( Amap_Lib_t * p, unsigned uTruth )
+{
+ Amap_Gat_t * pGate;
+ int i;
+ Vec_PtrForEachEntry( p->vSorted, pGate, i )
+ if ( pGate->nPins <= 5 && pGate->pFunc[0] == uTruth )
+ return pGate;
+ return NULL;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Selects gates useful for area-only mapping.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Vec_Ptr_t * Amap_LibSelectGates( Amap_Lib_t * p, int fVerbose )
+{
+ Vec_Ptr_t * vSelect;
+ Amap_Gat_t * pGate, * pGate2;
+ int i, k, clk = clock();
+ p->pGate0 = Amap_LibFindGate( p, 0 );
+ p->pGate1 = Amap_LibFindGate( p, ~0 );
+ p->pGateBuf = Amap_LibFindGate( p, 0xAAAAAAAA );
+ p->pGateInv = Amap_LibFindGate( p, ~0xAAAAAAAA );
+ vSelect = Vec_PtrAlloc( 100 );
+ Vec_PtrForEachEntry( p->vSorted, pGate, i )
+ {
+ if ( pGate->pFunc == NULL )
+ continue;
+ Vec_PtrForEachEntryStop( p->vSorted, pGate2, k, i )
+ {
+ if ( pGate2->pFunc == NULL )
+ continue;
+ if ( pGate2->nPins != pGate->nPins )
+ continue;
+ if ( !memcmp( pGate2->pFunc, pGate->pFunc, sizeof(unsigned) * Aig_TruthWordNum(pGate->nPins) ) )
+ break;
+ }
+ if ( k < i )
+ continue;
+ Vec_PtrPush( vSelect, pGate );
+ }
+ return vSelect;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Selects gates useful for area-only mapping.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+void Amap_LibPrintSelectedGates( Amap_Lib_t * p, int fAllGates )
+{
+ Vec_Ptr_t * vArray;
+ Amap_Gat_t * pGate;
+ int i;
+ vArray = fAllGates? p->vGates : p->vSelect;
+ Vec_PtrForEachEntry( vArray, pGate, i )
+ {
+ printf( "Gate %4d : %15s Area = %9.2f\n", pGate->Id, pGate->pName, pGate->dArea );
+ printf( " Formula: %s=%s\n", pGate->pOutName, pGate->pForm );
+ printf( " DSD: " );
+ Kit_DsdPrintFromTruth( pGate->pFunc, pGate->nPins );
+ }
+}
+
+/**Function*************************************************************
+
+ Synopsis [Parses equations for the gates.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Amap_Lib_t * Amap_LibReadAndPrepare( char * pFileName, int fVerbose, int fVeryVerbose )
+{
+ Amap_Lib_t * p;
+ int clk = clock();
+ p = Amap_LibReadFile( pFileName, fVerbose );
+ if ( fVerbose )
+ printf( "Read %d gates from file \"%s\".\n", Vec_PtrSize(p->vGates), pFileName );
+ if ( p == NULL )
+ return NULL;
+ if ( !Amap_LibParseEquations( p, fVerbose ) )
+ {
+ Amap_LibFree( p );
+ return NULL;
+ }
+ p->vSorted = Amap_LibSortGatesByArea( p );
+ p->vSelect = Amap_LibSelectGates( p, fVerbose );
+ if ( fVerbose )
+ {
+ printf( "Selected %d functionally unique gates. ",
+ Vec_PtrSize(p->vSelect), Vec_PtrSize(p->vSorted) );
+ PRT( "Time", clock() - clk );
+ }
+ clk = clock();
+ Amap_LibCreateRules( p, fVeryVerbose );
+ if ( fVerbose )
+ {
+ printf( "Created %d rules and %d matches. ",
+ p->nNodes, p->nSets );
+ PRT( "Time", clock() - clk );
+ }
+ return p;
+}
+
+////////////////////////////////////////////////////////////////////////
+/// END OF FILE ///
+////////////////////////////////////////////////////////////////////////
+
+
diff --git a/src/map/amap/amapMan.c b/src/map/amap/amapMan.c
new file mode 100644
index 00000000..521b4ffd
--- /dev/null
+++ b/src/map/amap/amapMan.c
@@ -0,0 +1,99 @@
+/**CFile****************************************************************
+
+ FileName [amapMan.c]
+
+ SystemName [ABC: Logic synthesis and verification system.]
+
+ PackageName [Technology mapper for standard cells.]
+
+ Synopsis [Mapping manager.]
+
+ Author [Alan Mishchenko]
+
+ Affiliation [UC Berkeley]
+
+ Date [Ver. 1.0. Started - June 20, 2005.]
+
+ Revision [$Id: amapMan.c,v 1.00 2005/06/20 00:00:00 alanmi Exp $]
+
+***********************************************************************/
+
+#include "amapInt.h"
+
+////////////////////////////////////////////////////////////////////////
+/// DECLARATIONS ///
+////////////////////////////////////////////////////////////////////////
+
+////////////////////////////////////////////////////////////////////////
+/// FUNCTION DEFINITIONS ///
+////////////////////////////////////////////////////////////////////////
+
+/**Function*************************************************************
+
+ Synopsis [Starts the AIG manager.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Amap_Man_t * Amap_ManStart( int nNodes )
+{
+ Amap_Man_t * p;
+ // start the manager
+ p = ALLOC( Amap_Man_t, 1 );
+ memset( p, 0, sizeof(Amap_Man_t) );
+ p->fEpsilonInternal = (float)0.01;
+ // allocate arrays for nodes
+ p->vPis = Vec_PtrAlloc( 100 );
+ p->vPos = Vec_PtrAlloc( 100 );
+ p->vObjs = Vec_PtrAlloc( 100 );
+ p->vTemp = Vec_IntAlloc( 100 );
+ p->vCuts0 = Vec_PtrAlloc( 100 );
+ p->vCuts1 = Vec_PtrAlloc( 100 );
+ p->vCuts2 = Vec_PtrAlloc( 100 );
+ // prepare the memory manager
+ p->pMemObj = Aig_MmFixedStart( sizeof(Amap_Obj_t), nNodes );
+ p->pMemCuts = Aig_MmFlexStart();
+ p->pMemCutBest = Aig_MmFlexStart();
+ p->pMemTemp = Aig_MmFlexStart();
+ return p;
+}
+
+/**Function*************************************************************
+
+ Synopsis []
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+void Amap_ManStop( Amap_Man_t * p )
+{
+ Vec_PtrFree( p->vPis );
+ Vec_PtrFree( p->vPos );
+ Vec_PtrFree( p->vObjs );
+ Vec_PtrFree( p->vCuts0 );
+ Vec_PtrFree( p->vCuts1 );
+ Vec_PtrFree( p->vCuts2 );
+ Vec_IntFree( p->vTemp );
+ Aig_MmFixedStop( p->pMemObj, 0 );
+ Aig_MmFlexStop( p->pMemCuts, 0 );
+ Aig_MmFlexStop( p->pMemCutBest, 0 );
+ Aig_MmFlexStop( p->pMemTemp, 0 );
+ FREE( p->pMatsTemp );
+ FREE( p->ppCutsTemp );
+ FREE( p->pCutsPi );
+ free( p );
+}
+
+////////////////////////////////////////////////////////////////////////
+/// END OF FILE ///
+////////////////////////////////////////////////////////////////////////
+
+
diff --git a/src/map/amap/amapMatch.c b/src/map/amap/amapMatch.c
new file mode 100644
index 00000000..0b15a931
--- /dev/null
+++ b/src/map/amap/amapMatch.c
@@ -0,0 +1,538 @@
+/**CFile****************************************************************
+
+ FileName [amapMatch.c]
+
+ SystemName [ABC: Logic synthesis and verification system.]
+
+ PackageName [Technology mapper for standard cells.]
+
+ Synopsis []
+
+ Author [Alan Mishchenko]
+
+ Affiliation [UC Berkeley]
+
+ Date [Ver. 1.0. Started - June 20, 2005.]
+
+ Revision [$Id: amapMatch.c,v 1.00 2005/06/20 00:00:00 alanmi Exp $]
+
+***********************************************************************/
+
+#include "amapInt.h"
+
+////////////////////////////////////////////////////////////////////////
+/// DECLARATIONS ///
+////////////////////////////////////////////////////////////////////////
+
+////////////////////////////////////////////////////////////////////////
+/// FUNCTION DEFINITIONS ///
+////////////////////////////////////////////////////////////////////////
+
+/**Function*************************************************************
+
+ Synopsis [Duplicates the cut using new memory manager.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Amap_Cut_t * Amap_ManDupCut( Amap_Man_t * p, Amap_Cut_t * pCut )
+{
+ Amap_Cut_t * pNew;
+ int nBytes = sizeof(Amap_Cut_t) + sizeof(int) * pCut->nFans;
+ pNew = (Amap_Cut_t *)Aig_MmFlexEntryFetch( p->pMemCutBest, nBytes );
+ memcpy( pNew, pCut, nBytes );
+ return pNew;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Starts the match with cut and set.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+static inline void Amap_ManMatchStart( Amap_Mat_t * p, Amap_Cut_t * pCut, Amap_Set_t * pSet )
+{
+ memset( p, 0, sizeof(Amap_Mat_t) );
+ p->pCut = pCut;
+ p->pSet = pSet;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Cleans reference counters.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+void Amap_ManCleanRefs( Amap_Man_t * p )
+{
+ Amap_Obj_t * pObj;
+ int i;
+ Amap_ManForEachObj( p, pObj, i )
+ pObj->nFouts[0] = pObj->nFouts[1] = 0;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Computes delay.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+float Amap_ManMaxDelay( Amap_Man_t * p )
+{
+ Amap_Obj_t * pObj;
+ float Delay = 0.0;
+ int i;
+ Amap_ManForEachPo( p, pObj, i )
+ Delay = AIG_MAX( Delay, Amap_ObjFanin0(p,pObj)->Best.Delay );
+ return Delay;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Cleans reference counters.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+void Amap_ManCleanData( Amap_Man_t * p )
+{
+ Amap_Obj_t * pObj;
+ int i;
+// Amap_ManForEachNode( p, pObj, i )
+// FREE( pObj->pData );
+ Amap_ManForEachObj( p, pObj, i )
+ pObj->pData = NULL;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Compute nodes used in the mapping.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+float Amap_ManComputeMapping_rec( Amap_Man_t * p, Amap_Obj_t * pObj, int fCompl )
+{
+ Amap_Mat_t * pM = &pObj->Best;
+ Amap_Obj_t * pFanin;
+ Amap_Gat_t * pGate;
+ int i, iFanin, fComplFanin;
+ float Area;
+ if ( pObj->nFouts[fCompl]++ + pObj->nFouts[!fCompl] > 0 )
+ return 0.0;
+ if ( Amap_ObjIsPi(pObj) || Amap_ObjIsConst1(pObj) )
+ return 0.0;
+ pGate = Amap_LibGate( p->pLib, pM->pSet->iGate );
+ assert( pGate->nPins == pM->pCut->nFans );
+ Area = pGate->dArea;
+ for ( i = 0; i < (int)pGate->nPins; i++ )
+ {
+ iFanin = Amap_Lit2Var( pM->pSet->Ins[i] );
+ pFanin = Amap_ManObj( p, Amap_Lit2Var(pM->pCut->Fans[iFanin]) );
+ fComplFanin = Amap_LitIsCompl( pM->pSet->Ins[i] ) ^ Amap_LitIsCompl( pM->pCut->Fans[iFanin] );
+ Area += Amap_ManComputeMapping_rec( p, pFanin, fComplFanin );
+ }
+ return Area;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Compute nodes used in the mapping.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+float Amap_ManComputeMapping( Amap_Man_t * p )
+{
+ Amap_Obj_t * pObj;
+ float Area = 0.0;
+ int i;
+ Amap_ManCleanRefs( p );
+ Amap_ManForEachPo( p, pObj, i )
+ Area += Amap_ManComputeMapping_rec( p, Amap_ObjFanin0(p, pObj), Amap_ObjFaninC0(pObj) );
+ return Area;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Counts the number of inverters to be added.]
+
+ Description [Should be called after mapping has been set.]
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+int Amap_ManCountInverters( Amap_Man_t * p )
+{
+ Amap_Obj_t * pObj;
+ int i, Counter = 0;
+ Amap_ManForEachObj( p, pObj, i )
+ Counter += (int)(pObj->nFouts[!pObj->fPolar] > 0);
+ return Counter;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Compare two matches.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+static inline int Amap_CutCompare( Amap_Man_t * p, Amap_Mat_t * pM0, Amap_Mat_t * pM1 )
+{
+ // compare area flows
+ if ( pM0->Area < pM1->Area - p->pPars->fEpsilon )
+ return -1;
+ if ( pM0->Area > pM1->Area + p->pPars->fEpsilon )
+ return 1;
+
+ // compare average fanouts
+ if ( pM0->AveFan > pM1->AveFan - p->pPars->fEpsilon )
+ return -1;
+ if ( pM0->AveFan < pM1->AveFan + p->pPars->fEpsilon )
+ return 1;
+
+ // compare delay
+ if ( pM0->Delay < pM1->Delay - p->pPars->fEpsilon )
+ return -1;
+ if ( pM0->Delay > pM1->Delay + p->pPars->fEpsilon )
+ return 1;
+ return 1;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Counts area while dereferencing the match.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+static inline float Amap_CutAreaDeref( Amap_Man_t * p, Amap_Mat_t * pM )
+{
+ Amap_Obj_t * pFanin;
+ int i, fCompl;
+ float Area = Amap_LibGate( p->pLib, pM->pSet->iGate )->dArea;
+ Amap_MatchForEachFaninCompl( p, pM, pFanin, fCompl, i )
+ {
+ assert( Amap_ObjRefsTotal(pFanin) > 0 );
+ if ( (int)pFanin->fPolar != fCompl && pFanin->nFouts[fCompl] == 1 )
+ Area += p->fAreaInv;
+ if ( --pFanin->nFouts[fCompl] + pFanin->nFouts[!fCompl] == 0 && Amap_ObjIsNode(pFanin) )
+ Area += Amap_CutAreaDeref( p, &pFanin->Best );
+ }
+ return Area;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Counts area while referencing the match.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+static inline float Amap_CutAreaRef( Amap_Man_t * p, Amap_Mat_t * pM )
+{
+ Amap_Obj_t * pFanin;
+ int i, fCompl;
+ float Area = Amap_LibGate( p->pLib, pM->pSet->iGate )->dArea;
+ Amap_MatchForEachFaninCompl( p, pM, pFanin, fCompl, i )
+ {
+ assert( Amap_ObjRefsTotal(pFanin) >= 0 );
+ if ( (int)pFanin->fPolar != fCompl && pFanin->nFouts[fCompl] == 0 )
+ Area += p->fAreaInv;
+ if ( pFanin->nFouts[fCompl]++ + pFanin->nFouts[!fCompl] == 0 && Amap_ObjIsNode(pFanin) )
+ Area += Amap_CutAreaRef( p, &pFanin->Best );
+ }
+ return Area;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Derives area of the match for a non-referenced node.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+static inline float Amap_CutAreaDerefed( Amap_Man_t * p, Amap_Obj_t * pNode, Amap_Mat_t * pM )
+{
+ float aResult, aResult2;
+ int fComplNew;
+ aResult2 = Amap_CutAreaRef( p, pM );
+ aResult = Amap_CutAreaDeref( p, pM );
+ assert( aResult > aResult2 - p->fEpsilonInternal );
+ assert( aResult < aResult2 + p->fEpsilonInternal );
+ // if node is needed in another polarity, add inverter
+ fComplNew = pM->pCut->fInv ^ pM->pSet->fInv;
+ if ( pNode->nFouts[fComplNew] == 0 && pNode->nFouts[!fComplNew] > 0 )
+ aResult += p->fAreaInv;
+ return aResult;
+}
+
+/**Function*************************************************************
+
+ Synopsis []
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+static inline void Amap_CutAreaTest( Amap_Man_t * p, Amap_Obj_t * pNode )
+{
+ float aResult, aResult2;
+ if ( Amap_ObjRefsTotal(pNode) == 0 )
+ {
+ aResult2 = Amap_CutAreaRef( p, &pNode->Best );
+ aResult = Amap_CutAreaDeref( p, &pNode->Best );
+ assert( aResult > aResult2 - p->fEpsilonInternal );
+ assert( aResult < aResult2 + p->fEpsilonInternal );
+ }
+ else
+ {
+ aResult = Amap_CutAreaDeref( p, &pNode->Best );
+ aResult2 = Amap_CutAreaRef( p, &pNode->Best );
+ assert( aResult > aResult2 - p->fEpsilonInternal );
+ assert( aResult < aResult2 + p->fEpsilonInternal );
+ }
+}
+
+/**Function*************************************************************
+
+ Synopsis [Derives parameters for the match.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+static inline void Amap_ManMatchGetFlows( Amap_Man_t * p, Amap_Mat_t * pM )
+{
+ Amap_Mat_t * pMFanin;
+ Amap_Obj_t * pFanin;
+ Amap_Gat_t * pGate;
+ int i;
+ pGate = Amap_LibGate( p->pLib, pM->pSet->iGate );
+ assert( pGate->nPins == pM->pCut->nFans );
+ assert( pM->Area == 0.0 );
+ pM->Area = pGate->dArea;
+ pM->AveFan = 0.0;
+ pM->Delay = 0.0;
+ Amap_MatchForEachFanin( p, pM, pFanin, i )
+ {
+ pMFanin = &pFanin->Best;
+ pM->Delay = AIG_MAX( pM->Delay, pMFanin->Delay );
+ pM->AveFan += Amap_ObjRefsTotal(pFanin);
+ if ( Amap_ObjRefsTotal(pFanin) == 0 )
+ pM->Area += pMFanin->Area;
+ else
+ pM->Area += pMFanin->Area / pFanin->EstRefs;
+ }
+ pM->AveFan /= pGate->nPins;
+ pM->Delay += 1.0;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Derives parameters for the match.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+static inline void Amap_ManMatchGetExacts( Amap_Man_t * p, Amap_Obj_t * pNode, Amap_Mat_t * pM )
+{
+ Amap_Mat_t * pMFanin;
+ Amap_Obj_t * pFanin;
+ Amap_Gat_t * pGate;
+ int i;
+ pGate = Amap_LibGate( p->pLib, pM->pSet->iGate );
+ assert( pGate->nPins == pM->pCut->nFans );
+ assert( pM->Area == 0.0 );
+ pM->AveFan = 0.0;
+ pM->Delay = 0.0;
+ Amap_MatchForEachFanin( p, pM, pFanin, i )
+ {
+ pMFanin = &pFanin->Best;
+ pM->Delay = AIG_MAX( pM->Delay, pMFanin->Delay );
+ pM->AveFan += Amap_ObjRefsTotal(pFanin);
+ }
+ pM->AveFan /= pGate->nPins;
+ pM->Delay += 1.0;
+ pM->Area = Amap_CutAreaDerefed( p, pNode, pM );
+}
+
+/**Function*************************************************************
+
+ Synopsis [Computes the best match at each node.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+void Amap_ManMatchNode( Amap_Man_t * p, Amap_Obj_t * pNode, int fFlow, int fRefs )
+{
+ Amap_Mat_t M1, M2, * pMBest = &M1, * pMThis = &M2;
+ Amap_Cut_t * pCut;
+ Amap_Set_t * pSet;
+ Amap_Nod_t * pNod;
+ int i;
+ if ( fRefs )
+ pNode->EstRefs = (float)((2.0 * pNode->EstRefs + Amap_ObjRefsTotal(pNode)) / 3.0);
+ else
+ pNode->EstRefs = (float)pNode->nRefs;
+ if ( fRefs && Amap_ObjRefsTotal(pNode) > 0 )
+ Amap_CutAreaDeref( p, &pNode->Best );
+ pMBest->pCut = NULL;
+ Amap_NodeForEachCut( pNode, pCut, i )
+ {
+ if ( pCut->iMat == 0 )
+ continue;
+ pNod = Amap_LibNod( p->pLib, pCut->iMat );
+ Amap_LibNodeForEachSet( pNod, pSet )
+ {
+ Amap_ManMatchStart( pMThis, pCut, pSet );
+ if ( fFlow )
+ Amap_ManMatchGetFlows( p, pMThis );
+ else
+ Amap_ManMatchGetExacts( p, pNode, pMThis );
+ if ( pMBest->pCut == NULL || Amap_CutCompare(p, pMBest, pMThis) == 1 )
+ *pMBest = *pMThis;
+ }
+ }
+ pNode->fPolar = pMBest->pCut->fInv ^ pMBest->pSet->fInv;
+ pNode->Best = *pMBest;
+ pNode->Best.pCut = Amap_ManDupCut( p, pNode->Best.pCut );
+ if ( fRefs && Amap_ObjRefsTotal(pNode) > 0 )
+ Amap_CutAreaRef( p, &pNode->Best );
+}
+
+/**Function*************************************************************
+
+ Synopsis [Performs one round of mapping.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+void Amap_ManMatch( Amap_Man_t * p, int fFlow, int fRefs )
+{
+ Aig_MmFlex_t * pMemOld;
+ Amap_Obj_t * pObj;
+ float Area;
+ int i, nInvs, clk = clock();
+ pMemOld = p->pMemCutBest;
+ p->pMemCutBest = Aig_MmFlexStart();
+ Amap_ManForEachNode( p, pObj, i )
+ if ( pObj->pData )
+ Amap_ManMatchNode( p, pObj, fFlow, fRefs );
+ Aig_MmFlexStop( pMemOld, 0 );
+ Area = Amap_ManComputeMapping( p );
+ nInvs = Amap_ManCountInverters( p );
+if ( p->pPars->fVerbose )
+{
+ printf( "Area =%9.2f. Gate =%9.2f. Inv =%9.2f. (%6d.) Delay =%6.2f. ",
+ Area + nInvs * p->fAreaInv,
+ Area, nInvs * p->fAreaInv, nInvs,
+ Amap_ManMaxDelay(p) );
+PRT( "Time ", clock() - clk );
+}
+ // test procedures
+// Amap_ManForEachNode( p, pObj, i )
+// Amap_CutAreaTest( p, pObj );
+}
+
+/**Function*************************************************************
+
+ Synopsis [Performs mapping.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+void Amap_ManMap( Amap_Man_t * p )
+{
+ int i;
+ Amap_ManMerge( p );
+ for ( i = 0; i < p->pPars->nIterFlow; i++ )
+ Amap_ManMatch( p, 1, i>0 );
+ for ( i = 0; i < p->pPars->nIterArea; i++ )
+ Amap_ManMatch( p, 0, p->pPars->nIterFlow>0||i>0 );
+/*
+ for ( i = 0; i < p->pPars->nIterFlow; i++ )
+ Amap_ManMatch( p, 1, 1 );
+ for ( i = 0; i < p->pPars->nIterArea; i++ )
+ Amap_ManMatch( p, 0, 1 );
+*/
+ Amap_ManCleanData( p );
+}
+
+////////////////////////////////////////////////////////////////////////
+/// END OF FILE ///
+////////////////////////////////////////////////////////////////////////
+
+
diff --git a/src/map/amap/amapMerge.c b/src/map/amap/amapMerge.c
new file mode 100644
index 00000000..b024616e
--- /dev/null
+++ b/src/map/amap/amapMerge.c
@@ -0,0 +1,521 @@
+/**CFile****************************************************************
+
+ FileName [amapMerge.c]
+
+ SystemName [ABC: Logic synthesis and verification system.]
+
+ PackageName [Technology mapper for standard cells.]
+
+ Synopsis [Computing cuts for the node.]
+
+ Author [Alan Mishchenko]
+
+ Affiliation [UC Berkeley]
+
+ Date [Ver. 1.0. Started - June 20, 2005.]
+
+ Revision [$Id: amapMerge.c,v 1.00 2005/06/20 00:00:00 alanmi Exp $]
+
+***********************************************************************/
+
+#include "amapInt.h"
+
+////////////////////////////////////////////////////////////////////////
+/// DECLARATIONS ///
+////////////////////////////////////////////////////////////////////////
+
+////////////////////////////////////////////////////////////////////////
+/// FUNCTION DEFINITIONS ///
+////////////////////////////////////////////////////////////////////////
+
+/**Function*************************************************************
+
+ Synopsis [Creates new cut and adds it to storage.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Amap_Cut_t * Amap_ManSetupPis( Amap_Man_t * p )
+{
+ Amap_Obj_t * pObj;
+ Amap_Cut_t * pCut;
+ int i, nBytes = sizeof(Amap_Cut_t) + sizeof(int);
+ char * pBuffer = ALLOC( char, Amap_ManPiNum(p) * nBytes );
+ Amap_ManForEachPi( p, pObj, i )
+ {
+ pCut = (Amap_Cut_t *)( pBuffer + i*nBytes );
+ pCut->iMat = 0;
+ pCut->fInv = 0;
+ pCut->nFans = 1;
+ pCut->Fans[0] = Amap_Var2Lit( pObj->Id, 0 );
+ pObj->pData = pCut;
+ pObj->nCuts = 1;
+ pObj->EstRefs = (float)1.0;
+ }
+ return (Amap_Cut_t *)pBuffer;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Creates new cut and adds it to storage.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Amap_Cut_t * Amap_ManCutStore( Amap_Man_t * p, Amap_Cut_t * pCut, int fCompl )
+{
+ Amap_Cut_t * pNew;
+ int iFan, nBytes = sizeof(Amap_Cut_t) + sizeof(int) * pCut->nFans + sizeof(Amap_Cut_t *);
+ pNew = (Amap_Cut_t *)Aig_MmFlexEntryFetch( p->pMemTemp, nBytes );
+ pNew->iMat = pCut->iMat;
+ pNew->fInv = pCut->fInv ^ fCompl;
+ pNew->nFans = pCut->nFans;
+ memcpy( pNew->Fans, pCut->Fans, sizeof(int) * pCut->nFans );
+ // add it to storage
+ iFan = Amap_Var2Lit( pNew->iMat, pNew->fInv );
+ if ( p->ppCutsTemp[ iFan ] == NULL )
+ Vec_IntPushOrder( p->vTemp, iFan );
+ *Amap_ManCutNextP( pNew ) = p->ppCutsTemp[ iFan ];
+ p->ppCutsTemp[ iFan ] = pNew;
+ return pNew;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Creates new cut and adds it to storage.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Amap_Cut_t * Amap_ManCutCreate( Amap_Man_t * p,
+ Amap_Cut_t * pCut0, Amap_Cut_t * pCut1, int iMat )
+{
+ Amap_Cut_t * pCut;
+ int i, nSize = pCut0->nFans + pCut1->nFans;
+ int nBytes = sizeof(Amap_Cut_t) + sizeof(int) * nSize + sizeof(Amap_Cut_t *);
+ assert( pCut0->iMat >= pCut1->iMat );
+ pCut = (Amap_Cut_t *)Aig_MmFlexEntryFetch( p->pMemTemp, nBytes );
+ pCut->iMat = iMat;
+ pCut->fInv = 0;
+ pCut->nFans = nSize;
+ for ( i = 0; i < (int)pCut0->nFans; i++ )
+ pCut->Fans[i] = pCut0->Fans[i];
+ for ( i = 0; i < (int)pCut1->nFans; i++ )
+ pCut->Fans[pCut0->nFans+i] = pCut1->Fans[i];
+ // add it to storage
+ if ( p->ppCutsTemp[ pCut->iMat ] == NULL )
+ Vec_IntPushOrder( p->vTemp, pCut->iMat );
+ *Amap_ManCutNextP( pCut ) = p->ppCutsTemp[ pCut->iMat ];
+ p->ppCutsTemp[ pCut->iMat ] = pCut;
+ return pCut;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Creates new cut and adds it to storage.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Amap_Cut_t * Amap_ManCutCreate3( Amap_Man_t * p,
+ Amap_Cut_t * pCut0, Amap_Cut_t * pCut1, Amap_Cut_t * pCut2, int iMat )
+{
+ Amap_Cut_t * pCut;
+ int i, nSize = pCut0->nFans + pCut1->nFans + pCut2->nFans;
+ int nBytes = sizeof(Amap_Cut_t) + sizeof(int) * nSize + sizeof(Amap_Cut_t *);
+ pCut = (Amap_Cut_t *)Aig_MmFlexEntryFetch( p->pMemTemp, nBytes );
+ pCut->iMat = iMat;
+ pCut->fInv = 0;
+ pCut->nFans = nSize;
+ for ( i = 0; i < (int)pCut0->nFans; i++ )
+ pCut->Fans[i] = pCut0->Fans[i];
+ for ( i = 0; i < (int)pCut1->nFans; i++ )
+ pCut->Fans[pCut0->nFans+i] = pCut1->Fans[i];
+ for ( i = 0; i < (int)pCut2->nFans; i++ )
+ pCut->Fans[pCut0->nFans+pCut1->nFans+i] = pCut2->Fans[i];
+ // add it to storage
+ if ( p->ppCutsTemp[ pCut->iMat ] == NULL )
+ Vec_IntPushOrder( p->vTemp, pCut->iMat );
+ *Amap_ManCutNextP( pCut ) = p->ppCutsTemp[ pCut->iMat ];
+ p->ppCutsTemp[ pCut->iMat ] = pCut;
+ return pCut;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Removes cuts from the temporary storage.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+void Amap_ManCutSaveStored( Amap_Man_t * p, Amap_Obj_t * pNode )
+{
+ int * pBuffer;
+ Amap_Cut_t * pNext, * pCut;
+ int i, nWords, Entry, nCuts;
+ assert( pNode->pData == NULL );
+ // count memory needed
+ nCuts = 1;
+ nWords = 2;
+ Vec_IntForEachEntry( p->vTemp, Entry, i )
+ {
+ for ( pCut = p->ppCutsTemp[Entry]; pCut; pCut = *Amap_ManCutNextP(pCut) )
+ {
+ nCuts++;
+ nWords += pCut->nFans + 1;
+ }
+ }
+ p->nBytesUsed += 4*nWords;
+ // allocate memory
+ pBuffer = (int *)Aig_MmFlexEntryFetch( p->pMemCuts, 4*nWords );
+ pNext = (Amap_Cut_t *)pBuffer;
+ // add the first cut
+ pNext->iMat = 0;
+ pNext->fInv = 0;
+ pNext->nFans = 1;
+ pNext->Fans[0] = Amap_Var2Lit(pNode->Id, 0);
+ pNext = (Amap_Cut_t *)(pBuffer + 2);
+ // add other cuts
+ Vec_IntForEachEntry( p->vTemp, Entry, i )
+ {
+ for ( pCut = p->ppCutsTemp[Entry]; pCut; pCut = *Amap_ManCutNextP(pCut) )
+ {
+ memcpy( pNext, pCut, sizeof(int) * (pCut->nFans + 1) );
+ pNext = (Amap_Cut_t *)((int *)pNext + pCut->nFans + 1);
+ }
+ p->ppCutsTemp[Entry] = NULL;
+ }
+ assert( (int *)pNext - pBuffer == nWords );
+ // restore the storage
+ Vec_IntClear( p->vTemp );
+ Aig_MmFlexRestart( p->pMemTemp );
+ for ( i = 0; i < 2*p->pLib->nNodes; i++ )
+ if ( p->ppCutsTemp[i] != NULL )
+ printf( "Amap_ManCutSaveStored(): Error!\n" );
+ pNode->pData = (Amap_Cut_t *)pBuffer;
+ pNode->nCuts = nCuts;
+// printf("%d ", nCuts );
+ // verify cuts
+ pCut = NULL;
+ Amap_NodeForEachCut( pNode, pNext, i )
+// for ( i = 0, pNext = (Amap_Cut_t *)pNode->pData; i < (int)pNode->nCuts;
+// i++, pNext = Amap_ManCutNext(pNext) )
+ {
+ assert( pCut == NULL || pCut->iMat <= pNext->iMat );
+ pCut = pNext;
+ }
+}
+
+/**Function*************************************************************
+
+ Synopsis [Returns the number of possible new cuts.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+int Amap_ManMergeCountCuts( Amap_Man_t * p, Amap_Obj_t * pNode )
+{
+ Amap_Obj_t * pFanin0 = Amap_ObjFanin0( p, pNode );
+ Amap_Obj_t * pFanin1 = Amap_ObjFanin1( p, pNode );
+ Amap_Cut_t * pCut0, * pCut1;
+ int Entry, c0, c1, iCompl0, iCompl1, iFan0, iFan1;
+ int Counter = 1;
+ Amap_NodeForEachCut( pFanin0, pCut0, c0 )
+ Amap_NodeForEachCut( pFanin1, pCut1, c1 )
+ {
+ iCompl0 = pCut0->fInv ^ Amap_ObjFaninC0(pNode);
+ iCompl1 = pCut1->fInv ^ Amap_ObjFaninC1(pNode);
+ iFan0 = !pCut0->iMat? 0: Amap_Var2Lit( pCut0->iMat, iCompl0 );
+ iFan1 = !pCut1->iMat? 0: Amap_Var2Lit( pCut1->iMat, iCompl1 );
+ Entry = Amap_LibFindNode( p->pLib, iFan0, iFan1, pNode->Type == AMAP_OBJ_XOR );
+ Counter += ( Entry >=0 );
+// if ( Entry >=0 )
+// printf( "Full: %d + %d = %d\n", iFan0, iFan1, Entry );
+ }
+ return Counter;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Print cuts.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+void Amap_ManPrintCuts( Amap_Obj_t * pNode )
+{
+ Amap_Cut_t * pCut;
+ int c, i;
+ printf( "NODE %5d : Type = ", pNode->Id );
+ if ( pNode->Type == AMAP_OBJ_AND )
+ printf( "AND" );
+ else if ( pNode->Type == AMAP_OBJ_XOR )
+ printf( "XOR" );
+ else if ( pNode->Type == AMAP_OBJ_MUX )
+ printf( "MUX" );
+ printf( " Cuts = %d\n", pNode->nCuts );
+ Amap_NodeForEachCut( pNode, pCut, c )
+ {
+ printf( "%3d : Mat= %3d Inv=%d ", c, pCut->iMat, pCut->fInv );
+ for ( i = 0; i < (int)pCut->nFans; i++ )
+ printf( "%d%c ", Amap_Lit2Var(pCut->Fans[i]), Amap_LitIsCompl(pCut->Fans[i])?'-':'+' );
+ printf( "\n" );
+ }
+}
+
+/**Function*************************************************************
+
+ Synopsis [Derives cuts for one node.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+void Amap_ManMergeNodeChoice( Amap_Man_t * p, Amap_Obj_t * pNode )
+{
+ Amap_Obj_t * pTemp;
+ Amap_Cut_t * pCut;
+ int c;
+ // go through the nodes of the choice node
+ for ( pTemp = pNode; pTemp; pTemp = Amap_ObjChoice(p, pTemp) )
+ {
+ Amap_NodeForEachCut( pTemp, pCut, c )
+ if ( pCut->iMat )
+ Amap_ManCutStore( p, pCut, pNode->fPhase ^ pTemp->fPhase );
+ pTemp->pData = NULL;
+ }
+ Amap_ManCutSaveStored( p, pNode );
+
+// Amap_ManPrintCuts( pNode );
+}
+
+/**Function*************************************************************
+
+ Synopsis [Derives cuts for one node.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+int Amap_ManFindCut( Amap_Obj_t * pNode, Amap_Obj_t * pFanin, int fComplFanin, int Val, Vec_Ptr_t * vCuts )
+{
+ Amap_Cut_t * pCut;
+ int c, iCompl, iFan;
+ Vec_PtrClear( vCuts );
+ Amap_NodeForEachCut( pFanin, pCut, c )
+ {
+ iCompl = pCut->fInv ^ fComplFanin;
+ iFan = !pCut->iMat? 0: Amap_Var2Lit( pCut->iMat, iCompl );
+ if ( iFan == Val )
+ Vec_PtrPush( vCuts, pCut );
+ }
+ return Vec_PtrSize(vCuts) == 0;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Derives cuts for one node.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+void Amap_ManMergeNodeCutsMux( Amap_Man_t * p, Amap_Obj_t * pNode )
+{
+ Vec_Int_t * vRules = p->pLib->vRules3;
+ Amap_Obj_t * pFanin0 = Amap_ObjFanin0( p, pNode );
+ Amap_Obj_t * pFanin1 = Amap_ObjFanin1( p, pNode );
+ Amap_Obj_t * pFanin2 = Amap_ObjFanin2( p, pNode );
+ int fComplFanin0 = Amap_ObjFaninC0( pNode );
+ int fComplFanin1 = Amap_ObjFaninC1( pNode );
+ int fComplFanin2 = Amap_ObjFaninC2( pNode );
+ Amap_Cut_t * pCut0, * pCut1, * pCut2;
+ int x, c0, c1, c2;
+ assert( pNode->pData == NULL );
+ assert( pNode->Type == AMAP_OBJ_MUX );
+ assert( pNode->fRepr == 0 );
+ // go through the rules
+ for ( x = 0; x < Vec_IntSize(vRules); x += 4 )
+ {
+ if ( Amap_ManFindCut( pNode, pFanin0, fComplFanin0, Vec_IntEntry(vRules, x), p->vCuts0 ) )
+ continue;
+ if ( Amap_ManFindCut( pNode, pFanin1, fComplFanin1, Vec_IntEntry(vRules, x+1), p->vCuts1 ) )
+ continue;
+ if ( Amap_ManFindCut( pNode, pFanin2, fComplFanin2, Vec_IntEntry(vRules, x+2), p->vCuts2 ) )
+ continue;
+ Vec_PtrForEachEntry( p->vCuts0, pCut0, c0 )
+ Vec_PtrForEachEntry( p->vCuts1, pCut1, c1 )
+ Vec_PtrForEachEntry( p->vCuts2, pCut2, c2 )
+ {
+ Amap_Nod_t * pNod = Amap_LibNod( p->pLib, Vec_IntEntry(vRules, x+3) );
+ if ( pNod->pSets == NULL )
+ continue;
+ // complement literals
+ if ( pCut0->nFans == 1 && (pCut0->fInv ^ fComplFanin0) )
+ pCut0->Fans[0] = Amap_LitNot(pCut0->Fans[0]);
+ if ( pCut1->nFans == 1 && (pCut1->fInv ^ fComplFanin1) )
+ pCut1->Fans[0] = Amap_LitNot(pCut1->Fans[0]);
+ if ( pCut2->nFans == 1 && (pCut2->fInv ^ fComplFanin2) )
+ pCut2->Fans[0] = Amap_LitNot(pCut2->Fans[0]);
+ // create new cut
+ Amap_ManCutCreate3( p, pCut0, pCut1, pCut2, Vec_IntEntry(vRules, x+3) );
+ // uncomplement literals
+ if ( pCut0->nFans == 1 && (pCut0->fInv ^ fComplFanin0) )
+ pCut0->Fans[0] = Amap_LitNot(pCut0->Fans[0]);
+ if ( pCut1->nFans == 1 && (pCut1->fInv ^ fComplFanin1) )
+ pCut1->Fans[0] = Amap_LitNot(pCut1->Fans[0]);
+ if ( pCut2->nFans == 1 && (pCut2->fInv ^ fComplFanin2) )
+ pCut2->Fans[0] = Amap_LitNot(pCut2->Fans[0]);
+ }
+ }
+ Amap_ManCutSaveStored( p, pNode );
+ p->nCutsUsed += pNode->nCuts;
+ p->nCutsTried3 += pFanin0->nCuts * pFanin1->nCuts * pFanin2->nCuts;
+
+// Amap_ManPrintCuts( pNode );
+}
+
+/**Function*************************************************************
+
+ Synopsis [Derives cuts for one node.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+void Amap_ManMergeNodeCuts( Amap_Man_t * p, Amap_Obj_t * pNode )
+{
+ Amap_Obj_t * pFanin0 = Amap_ObjFanin0( p, pNode );
+ Amap_Obj_t * pFanin1 = Amap_ObjFanin1( p, pNode );
+ Amap_Cut_t * pCut0, * pCut1;
+ int ** pRules, Entry, i, k, c, iCompl0, iCompl1, iFan0, iFan1;
+ assert( pNode->pData == NULL );
+ if ( pNode->Type == AMAP_OBJ_MUX )
+ {
+ Amap_ManMergeNodeCutsMux( p, pNode );
+ return;
+ }
+ assert( pNode->Type != AMAP_OBJ_MUX );
+ pRules = (pNode->Type == AMAP_OBJ_AND)? p->pLib->pRules: p->pLib->pRulesX;
+ Amap_NodeForEachCut( pFanin0, pCut0, c )
+ {
+ iCompl0 = pCut0->fInv ^ Amap_ObjFaninC0(pNode);
+ iFan0 = !pCut0->iMat? 0: Amap_Var2Lit( pCut0->iMat, iCompl0 );
+ // complement literals
+ if ( pCut0->nFans == 1 && iCompl0 )
+ pCut0->Fans[0] = Amap_LitNot(pCut0->Fans[0]);
+ // label resulting sets
+ for ( i = 0; (Entry = pRules[iFan0][i]); i++ )
+ p->pMatsTemp[Entry & 0xffff] = (Entry >> 16);
+ // iterate through the cuts
+ Amap_NodeForEachCut( pFanin1, pCut1, k )
+ {
+ iCompl1 = pCut1->fInv ^ Amap_ObjFaninC1(pNode);
+ iFan1 = !pCut1->iMat? 0: Amap_Var2Lit( pCut1->iMat, iCompl1 );
+ if ( p->pMatsTemp[iFan1] == 0 )
+ continue;
+ // complement literals
+ if ( pCut1->nFans == 1 && iCompl1 )
+ pCut1->Fans[0] = Amap_LitNot(pCut1->Fans[0]);
+ // create new cut
+ if ( iFan0 >= iFan1 )
+ Amap_ManCutCreate( p, pCut0, pCut1, p->pMatsTemp[iFan1] );
+ else
+ Amap_ManCutCreate( p, pCut1, pCut0, p->pMatsTemp[iFan1] );
+ // uncomplement literals
+ if ( pCut1->nFans == 1 && iCompl1 )
+ pCut1->Fans[0] = Amap_LitNot(pCut1->Fans[0]);
+ }
+ // uncomplement literals
+ if ( pCut0->nFans == 1 && iCompl0 )
+ pCut0->Fans[0] = Amap_LitNot(pCut0->Fans[0]);
+ // label resulting sets
+ for ( i = 0; (Entry = pRules[iFan0][i]); i++ )
+ p->pMatsTemp[Entry & 0xffff] = 0;
+ }
+ Amap_ManCutSaveStored( p, pNode );
+ p->nCutsUsed += pNode->nCuts;
+ p->nCutsTried += pFanin0->nCuts * pFanin1->nCuts;
+// assert( (int)pNode->nCuts == Amap_ManMergeCountCuts(p, pNode) );
+ if ( pNode->fRepr )
+ Amap_ManMergeNodeChoice( p, pNode );
+
+// Amap_ManPrintCuts( pNode );
+}
+
+/**Function*************************************************************
+
+ Synopsis [Derives cuts for all nodes.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+void Amap_ManMerge( Amap_Man_t * p )
+{
+ Amap_Obj_t * pObj;
+ int i, clk = clock();
+ p->pCutsPi = Amap_ManSetupPis( p );
+ Amap_ManForEachNode( p, pObj, i )
+ Amap_ManMergeNodeCuts( p, pObj );
+ if ( p->pPars->fVerbose )
+ {
+ printf( "AIG object is %d bytes. ", sizeof(Amap_Obj_t) );
+ printf( "Internal AIG = %5.2f Mb. Cuts = %5.2f Mb.\n",
+ 1.0*Amap_ManObjNum(p)*sizeof(Amap_Obj_t)/(1<<20), 1.0*p->nBytesUsed/(1<<20) );
+ printf( "Node =%6d. Try =%9d. Try3 =%10d. Used =%7d. R =%6.2f. ",
+ Amap_ManNodeNum(p), p->nCutsTried, p->nCutsTried3, p->nCutsUsed,
+ 1.0*p->nCutsUsed/Amap_ManNodeNum(p) );
+PRT( "Time ", clock() - clk );
+ }
+}
+
+
+////////////////////////////////////////////////////////////////////////
+/// END OF FILE ///
+////////////////////////////////////////////////////////////////////////
+
+
diff --git a/src/map/amap/amapOutput.c b/src/map/amap/amapOutput.c
new file mode 100644
index 00000000..1decc52e
--- /dev/null
+++ b/src/map/amap/amapOutput.c
@@ -0,0 +1,181 @@
+/**CFile****************************************************************
+
+ FileName [amapOutput.c]
+
+ SystemName [ABC: Logic synthesis and verification system.]
+
+ PackageName [Technology mapper for standard cells.]
+
+ Synopsis [Core mapping procedures.]
+
+ Author [Alan Mishchenko]
+
+ Affiliation [UC Berkeley]
+
+ Date [Ver. 1.0. Started - June 20, 2005.]
+
+ Revision [$Id: amapOutput.c,v 1.00 2005/06/20 00:00:00 alanmi Exp $]
+
+***********************************************************************/
+
+#include "amapInt.h"
+
+////////////////////////////////////////////////////////////////////////
+/// DECLARATIONS ///
+////////////////////////////////////////////////////////////////////////
+
+static inline char * Amap_OuputStrsav( Aig_MmFlex_t * p, char * pStr )
+{ return pStr ? strcpy(Aig_MmFlexEntryFetch(p, strlen(pStr)+1), pStr) : NULL; }
+
+////////////////////////////////////////////////////////////////////////
+/// FUNCTION DEFINITIONS ///
+////////////////////////////////////////////////////////////////////////
+
+/**Function*************************************************************
+
+ Synopsis [Allocates structure for storing one gate.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Amap_Out_t * Amap_OutputStructAlloc( Aig_MmFlex_t * pMem, Amap_Gat_t * pGate )
+{
+ Amap_Out_t * pRes;
+ int nFans = pGate? pGate->nPins : 1;
+ pRes = (Amap_Out_t *)Aig_MmFlexEntryFetch( pMem, sizeof(Amap_Out_t)+sizeof(int)*nFans );
+ memset( pRes, 0, sizeof(Amap_Out_t) );
+ memset( pRes->pFans, 0xff, sizeof(int)*nFans );
+ pRes->pName = pGate? Amap_OuputStrsav( pMem, pGate->pName ) : NULL;
+ pRes->nFans = nFans;
+ return pRes;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Returns mapped network as an array of structures.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Vec_Ptr_t * Amap_ManProduceMapped( Amap_Man_t * p )
+{
+ Vec_Ptr_t * vNodes;
+ Aig_MmFlex_t * pMem;
+ Amap_Obj_t * pObj, * pFanin;
+ Amap_Gat_t * pGate;
+ Amap_Out_t * pRes;
+ int i, k, iFanin, fCompl;
+ float TotalArea = 0.0;
+ pMem = Aig_MmFlexStart();
+ // create mapping object for each node used in the mapping
+ vNodes = Vec_PtrAlloc( 10 );
+ Amap_ManForEachObj( p, pObj, i )
+ {
+ if ( Amap_ObjIsPi(pObj) )
+ {
+ assert( pObj->fPolar == 0 );
+ pRes = Amap_OutputStructAlloc( pMem, NULL );
+ pRes->Type = -1;
+ pRes->nFans = 0;
+ // save this structure
+ pObj->iData = Vec_PtrSize( vNodes );
+ Vec_PtrPush( vNodes, pRes );
+ // create invertor if needed
+ if ( pObj->nFouts[1] ) // this PI is used in the neg polarity
+ {
+ pRes = Amap_OutputStructAlloc( pMem, p->pLib->pGateInv );
+ pRes->pFans[0] = pObj->iData;
+ // save this structure
+ Vec_PtrPush( vNodes, pRes );
+ TotalArea += p->pLib->pGateInv->dArea;
+ }
+ continue;
+ }
+ if ( Amap_ObjIsNode(pObj) )
+ {
+ // skip the node that is not used in the mapping
+ if ( Amap_ObjRefsTotal(pObj) == 0 )
+ continue;
+ // get the gate
+ pGate = Amap_LibGate( p->pLib, pObj->Best.pSet->iGate );
+ assert( pGate->nPins == pObj->Best.pCut->nFans );
+ // allocate structure
+ pRes = Amap_OutputStructAlloc( pMem, pGate );
+ Amap_MatchForEachFaninCompl( p, &pObj->Best, pFanin, fCompl, k )
+ {
+ assert( Amap_ObjRefsTotal(pFanin) );
+ if ( (int)pFanin->fPolar == fCompl )
+ pRes->pFans[k] = pFanin->iData;
+ else
+ pRes->pFans[k] = pFanin->iData + 1;
+ }
+ // save this structure
+ pObj->iData = Vec_PtrSize( vNodes );
+ Vec_PtrPush( vNodes, pRes );
+ TotalArea += pGate->dArea;
+ // create invertor if needed
+ if ( pObj->nFouts[!pObj->fPolar] ) // needed in the opposite polarity
+ {
+ pRes = Amap_OutputStructAlloc( pMem, p->pLib->pGateInv );
+ pRes->pFans[0] = pObj->iData;
+ // save this structure
+ Vec_PtrPush( vNodes, pRes );
+ TotalArea += p->pLib->pGateInv->dArea;
+ }
+ continue;
+ }
+ if ( Amap_ObjIsPo(pObj) )
+ {
+ assert( pObj->fPolar == 0 );
+ pFanin = Amap_ObjFanin0(p, pObj);
+ assert( Amap_ObjRefsTotal(pFanin) );
+ if ( Amap_ObjIsConst1(pFanin) )
+ { // create constant node
+ if ( Amap_ObjFaninC0(pObj) )
+ {
+ pRes = Amap_OutputStructAlloc( pMem, p->pLib->pGate0 );
+ TotalArea += p->pLib->pGate0->dArea;
+ }
+ else
+ {
+ pRes = Amap_OutputStructAlloc( pMem, p->pLib->pGate1 );
+ TotalArea += p->pLib->pGate1->dArea;
+ }
+ // save this structure
+ iFanin = Vec_PtrSize( vNodes );
+ Vec_PtrPush( vNodes, pRes );
+ }
+ else
+ {
+ if ( (int)pFanin->fPolar == Amap_ObjFaninC0(pObj) )
+ iFanin = pFanin->iData;
+ else
+ iFanin = pFanin->iData + 1;
+ }
+ // create PO node
+ pRes = Amap_OutputStructAlloc( pMem, NULL );
+ pRes->Type = 1;
+ pRes->pFans[0] = iFanin;
+ // save this structure
+ Vec_PtrPush( vNodes, pRes );
+ }
+ }
+ // return memory manager in the last entry of the array
+ Vec_PtrPush( vNodes, pMem );
+ return vNodes;
+}
+
+
+////////////////////////////////////////////////////////////////////////
+/// END OF FILE ///
+////////////////////////////////////////////////////////////////////////
+
+
diff --git a/src/map/amap/amapParse.c b/src/map/amap/amapParse.c
new file mode 100644
index 00000000..262fbd5a
--- /dev/null
+++ b/src/map/amap/amapParse.c
@@ -0,0 +1,457 @@
+/**CFile****************************************************************
+
+ FileName [amapParse.c]
+
+ SystemName [ABC: Logic synthesis and verification system.]
+
+ PackageName [Technology mapper for standard cells.]
+
+ Synopsis [Parses representations of gates.]
+
+ Author [Alan Mishchenko]
+
+ Affiliation [UC Berkeley]
+
+ Date [Ver. 1.0. Started - June 20, 2005.]
+
+ Revision [$Id: amapParse.c,v 1.00 2005/06/20 00:00:00 alanmi Exp $]
+
+***********************************************************************/
+
+#include "amapInt.h"
+#include "hop.h"
+
+////////////////////////////////////////////////////////////////////////
+/// DECLARATIONS ///
+////////////////////////////////////////////////////////////////////////
+
+// the list of operation symbols to be used in expressions
+#define AMAP_EQN_SYM_OPEN '(' // opening paranthesis
+#define AMAP_EQN_SYM_CLOSE ')' // closing paranthesis
+#define AMAP_EQN_SYM_CONST0 '0' // constant 0
+#define AMAP_EQN_SYM_CONST1 '1' // constant 1
+#define AMAP_EQN_SYM_NEG '!' // negation before the variable
+#define AMAP_EQN_SYM_NEGAFT '\'' // negation after the variable
+#define AMAP_EQN_SYM_AND '*' // logic AND
+#define AMAP_EQN_SYM_XOR '^' // logic XOR
+#define AMAP_EQN_SYM_OR '+' // logic OR
+
+// the list of opcodes (also specifying operation precedence)
+#define AMAP_EQN_OPER_NEG 10 // negation
+#define AMAP_EQN_OPER_AND 9 // logic AND
+#define AMAP_EQN_OPER_XOR 8 // logic XOR
+#define AMAP_EQN_OPER_OR 7 // logic OR
+#define AMAP_EQN_OPER_MARK 1 // OpStack token standing for an opening paranthesis
+
+// these are values of the internal Flag
+#define AMAP_EQN_FLAG_START 1 // after the opening parenthesis
+#define AMAP_EQN_FLAG_VAR 2 // after operation is received
+#define AMAP_EQN_FLAG_OPER 3 // after operation symbol is received
+#define AMAP_EQN_FLAG_ERROR 4 // when error is detected
+
+////////////////////////////////////////////////////////////////////////
+/// FUNCTION DEFINITIONS ///
+////////////////////////////////////////////////////////////////////////
+
+/**Function*************************************************************
+
+ Synopsis [Performs the operation on the top entries in the stack.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Hop_Obj_t * Amap_ParseFormulaOper( Hop_Man_t * pMan, Vec_Ptr_t * pStackFn, int Oper )
+{
+ Hop_Obj_t * gArg1, * gArg2, * gFunc;
+ // perform the given operation
+ gArg2 = Vec_PtrPop( pStackFn );
+ gArg1 = Vec_PtrPop( pStackFn );
+ if ( Oper == AMAP_EQN_OPER_AND )
+ gFunc = Hop_And( pMan, gArg1, gArg2 );
+ else if ( Oper == AMAP_EQN_OPER_OR )
+ gFunc = Hop_Or( pMan, gArg1, gArg2 );
+ else if ( Oper == AMAP_EQN_OPER_XOR )
+ gFunc = Hop_Exor( pMan, gArg1, gArg2 );
+ else
+ return NULL;
+// Cudd_Ref( gFunc );
+// Cudd_RecursiveDeref( dd, gArg1 );
+// Cudd_RecursiveDeref( dd, gArg2 );
+ Vec_PtrPush( pStackFn, gFunc );
+ return gFunc;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Derives the AIG corresponding to the equation.]
+
+ Description [Takes the stream to output messages, the formula, the vector
+ of variable names and the AIG manager.]
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Hop_Obj_t * Amap_ParseFormula( FILE * pOutput, char * pFormInit, Vec_Ptr_t * vVarNames, Hop_Man_t * pMan )
+{
+ char * pFormula;
+ Vec_Ptr_t * pStackFn;
+ Vec_Int_t * pStackOp;
+ Hop_Obj_t * gFunc;
+ char * pTemp, * pName;
+ int nParans, fFound, Flag;
+ int Oper, Oper1, Oper2;
+ int i, v;
+
+ // make sure that the number of opening and closing parantheses is the same
+ nParans = 0;
+ for ( pTemp = pFormInit; *pTemp; pTemp++ )
+ if ( *pTemp == '(' )
+ nParans++;
+ else if ( *pTemp == ')' )
+ nParans--;
+ if ( nParans != 0 )
+ {
+ fprintf( pOutput, "Amap_ParseFormula(): Different number of opening and closing parantheses ().\n" );
+ return NULL;
+ }
+
+ // copy the formula
+ pFormula = ALLOC( char, strlen(pFormInit) + 3 );
+ sprintf( pFormula, "(%s)", pFormInit );
+
+ // start the stacks
+ pStackFn = Vec_PtrAlloc( 100 );
+ pStackOp = Vec_IntAlloc( 100 );
+
+ Flag = AMAP_EQN_FLAG_START;
+ for ( pTemp = pFormula; *pTemp; pTemp++ )
+ {
+ switch ( *pTemp )
+ {
+ // skip all spaces, tabs, and end-of-lines
+ case ' ':
+ case '\t':
+ case '\r':
+ case '\n':
+ continue;
+ case AMAP_EQN_SYM_CONST0:
+ Vec_PtrPush( pStackFn, Hop_ManConst0(pMan) ); // Cudd_Ref( b0 );
+ if ( Flag == AMAP_EQN_FLAG_VAR )
+ {
+ fprintf( pOutput, "Amap_ParseFormula(): No operation symbol before constant 0.\n" );
+ Flag = AMAP_EQN_FLAG_ERROR;
+ break;
+ }
+ Flag = AMAP_EQN_FLAG_VAR;
+ break;
+ case AMAP_EQN_SYM_CONST1:
+ Vec_PtrPush( pStackFn, Hop_ManConst1(pMan) ); // Cudd_Ref( b1 );
+ if ( Flag == AMAP_EQN_FLAG_VAR )
+ {
+ fprintf( pOutput, "Amap_ParseFormula(): No operation symbol before constant 1.\n" );
+ Flag = AMAP_EQN_FLAG_ERROR;
+ break;
+ }
+ Flag = AMAP_EQN_FLAG_VAR;
+ break;
+ case AMAP_EQN_SYM_NEG:
+ if ( Flag == AMAP_EQN_FLAG_VAR )
+ {// if NEGBEF follows a variable, AND is assumed
+ Vec_IntPush( pStackOp, AMAP_EQN_OPER_AND );
+ Flag = AMAP_EQN_FLAG_OPER;
+ }
+ Vec_IntPush( pStackOp, AMAP_EQN_OPER_NEG );
+ break;
+ case AMAP_EQN_SYM_NEGAFT:
+ if ( Flag != AMAP_EQN_FLAG_VAR )
+ {// if there is no variable before NEGAFT, it is an error
+ fprintf( pOutput, "Amap_ParseFormula(): No variable is specified before the negation suffix.\n" );
+ Flag = AMAP_EQN_FLAG_ERROR;
+ break;
+ }
+ else // if ( Flag == PARSE_FLAG_VAR )
+ Vec_PtrPush( pStackFn, Hop_Not( Vec_PtrPop(pStackFn) ) );
+ break;
+ case AMAP_EQN_SYM_AND:
+ case AMAP_EQN_SYM_OR:
+ case AMAP_EQN_SYM_XOR:
+ if ( Flag != AMAP_EQN_FLAG_VAR )
+ {
+ fprintf( pOutput, "Amap_ParseFormula(): There is no variable before AND, EXOR, or OR.\n" );
+ Flag = AMAP_EQN_FLAG_ERROR;
+ break;
+ }
+ if ( *pTemp == AMAP_EQN_SYM_AND )
+ Vec_IntPush( pStackOp, AMAP_EQN_OPER_AND );
+ else if ( *pTemp == AMAP_EQN_SYM_OR )
+ Vec_IntPush( pStackOp, AMAP_EQN_OPER_OR );
+ else //if ( *pTemp == AMAP_EQN_SYM_XOR )
+ Vec_IntPush( pStackOp, AMAP_EQN_OPER_XOR );
+ Flag = AMAP_EQN_FLAG_OPER;
+ break;
+ case AMAP_EQN_SYM_OPEN:
+ if ( Flag == AMAP_EQN_FLAG_VAR )
+ {
+// Vec_IntPush( pStackOp, AMAP_EQN_OPER_AND );
+ fprintf( pOutput, "Amap_ParseFormula(): An opening paranthesis follows a var without operation sign.\n" );
+ Flag = AMAP_EQN_FLAG_ERROR;
+ break;
+ }
+ Vec_IntPush( pStackOp, AMAP_EQN_OPER_MARK );
+ // after an opening bracket, it feels like starting over again
+ Flag = AMAP_EQN_FLAG_START;
+ break;
+ case AMAP_EQN_SYM_CLOSE:
+ if ( Vec_IntSize( pStackOp ) != 0 )
+ {
+ while ( 1 )
+ {
+ if ( Vec_IntSize( pStackOp ) == 0 )
+ {
+ fprintf( pOutput, "Amap_ParseFormula(): There is no opening paranthesis\n" );
+ Flag = AMAP_EQN_FLAG_ERROR;
+ break;
+ }
+ Oper = Vec_IntPop( pStackOp );
+ if ( Oper == AMAP_EQN_OPER_MARK )
+ break;
+
+ // perform the given operation
+ if ( Amap_ParseFormulaOper( pMan, pStackFn, Oper ) == NULL )
+ {
+ fprintf( pOutput, "Amap_ParseFormula(): Unknown operation\n" );
+ free( pFormula );
+ return NULL;
+ }
+ }
+ }
+ else
+ {
+ fprintf( pOutput, "Amap_ParseFormula(): There is no opening paranthesis\n" );
+ Flag = AMAP_EQN_FLAG_ERROR;
+ break;
+ }
+ if ( Flag != AMAP_EQN_FLAG_ERROR )
+ Flag = AMAP_EQN_FLAG_VAR;
+ break;
+
+
+ default:
+ // scan the next name
+ for ( i = 0; pTemp[i] &&
+ pTemp[i] != ' ' && pTemp[i] != '\t' && pTemp[i] != '\r' && pTemp[i] != '\n' &&
+ pTemp[i] != AMAP_EQN_SYM_AND && pTemp[i] != AMAP_EQN_SYM_OR &&
+ pTemp[i] != AMAP_EQN_SYM_XOR && pTemp[i] != AMAP_EQN_SYM_CLOSE; i++ )
+ {
+ if ( pTemp[i] == AMAP_EQN_SYM_NEG || pTemp[i] == AMAP_EQN_SYM_OPEN )
+ {
+ fprintf( pOutput, "Amap_ParseFormula(): The negation sign or an opening paranthesis inside the variable name.\n" );
+ Flag = AMAP_EQN_FLAG_ERROR;
+ break;
+ }
+ }
+ // variable name is found
+ fFound = 0;
+ Vec_PtrForEachEntry( vVarNames, pName, v )
+ if ( strncmp(pTemp, pName, i) == 0 && strlen(pName) == (unsigned)i )
+ {
+ pTemp += i-1;
+ fFound = 1;
+ break;
+ }
+ if ( !fFound )
+ {
+ fprintf( pOutput, "Amap_ParseFormula(): The parser cannot find var \"%s\" in the input var list.\n", pTemp );
+ Flag = AMAP_EQN_FLAG_ERROR;
+ break;
+ }
+ if ( Flag == AMAP_EQN_FLAG_VAR )
+ {
+ fprintf( pOutput, "Amap_ParseFormula(): The variable name \"%s\" follows another var without operation sign.\n", pTemp );
+ Flag = AMAP_EQN_FLAG_ERROR;
+ break;
+ }
+ Vec_PtrPush( pStackFn, Hop_IthVar( pMan, v ) ); // Cudd_Ref( pbVars[v] );
+ Flag = AMAP_EQN_FLAG_VAR;
+ break;
+ }
+
+ if ( Flag == AMAP_EQN_FLAG_ERROR )
+ break; // error exit
+ else if ( Flag == AMAP_EQN_FLAG_START )
+ continue; // go on parsing
+ else if ( Flag == AMAP_EQN_FLAG_VAR )
+ while ( 1 )
+ { // check if there are negations in the OpStack
+ if ( Vec_IntSize( pStackOp ) == 0 )
+ break;
+ Oper = Vec_IntPop( pStackOp );
+ if ( Oper != AMAP_EQN_OPER_NEG )
+ {
+ Vec_IntPush( pStackOp, Oper );
+ break;
+ }
+ else
+ {
+ Vec_PtrPush( pStackFn, Hop_Not(Vec_PtrPop(pStackFn)) );
+ }
+ }
+ else // if ( Flag == AMAP_EQN_FLAG_OPER )
+ while ( 1 )
+ { // execute all the operations in the OpStack
+ // with precedence higher or equal than the last one
+ Oper1 = Vec_IntPop( pStackOp ); // the last operation
+ if ( Vec_IntSize( pStackOp ) == 0 )
+ { // if it is the only operation, push it back
+ Vec_IntPush( pStackOp, Oper1 );
+ break;
+ }
+ Oper2 = Vec_IntPop( pStackOp ); // the operation before the last one
+ if ( Oper2 >= Oper1 )
+ { // if Oper2 precedence is higher or equal, execute it
+ if ( Amap_ParseFormulaOper( pMan, pStackFn, Oper2 ) == NULL )
+ {
+ fprintf( pOutput, "Amap_ParseFormula(): Unknown operation\n" );
+ free( pFormula );
+ return NULL;
+ }
+ Vec_IntPush( pStackOp, Oper1 ); // push the last operation back
+ }
+ else
+ { // if Oper2 precedence is lower, push them back and done
+ Vec_IntPush( pStackOp, Oper2 );
+ Vec_IntPush( pStackOp, Oper1 );
+ break;
+ }
+ }
+ }
+
+ if ( Flag != AMAP_EQN_FLAG_ERROR )
+ {
+ if ( Vec_PtrSize(pStackFn) != 0 )
+ {
+ gFunc = Vec_PtrPop(pStackFn);
+ if ( Vec_PtrSize(pStackFn) == 0 )
+ if ( Vec_IntSize( pStackOp ) == 0 )
+ {
+ Vec_PtrFree(pStackFn);
+ Vec_IntFree(pStackOp);
+// Cudd_Deref( gFunc );
+ free( pFormula );
+ return gFunc;
+ }
+ else
+ fprintf( pOutput, "Amap_ParseFormula(): Something is left in the operation stack\n" );
+ else
+ fprintf( pOutput, "Amap_ParseFormula(): Something is left in the function stack\n" );
+ }
+ else
+ fprintf( pOutput, "Amap_ParseFormula(): The input string is empty\n" );
+ }
+ free( pFormula );
+ return NULL;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Parses equations for the gates.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+int Amap_LibParseEquations( Amap_Lib_t * p, int fVerbose )
+{
+ extern int Kit_TruthSupportSize( unsigned * pTruth, int nVars );
+ Hop_Man_t * pMan;
+ Hop_Obj_t * pObj;
+ Vec_Ptr_t * vNames;
+ Vec_Int_t * vTruth;
+ Amap_Gat_t * pGate;
+ Amap_Pin_t * pPin;
+ unsigned * pTruth;
+ int i, nPinMax;
+ nPinMax = Amap_LibNumPinsMax(p);
+ if ( nPinMax > AMAP_MAXINS )
+ printf( "Gates with more than %d inputs will be ignored.\n", AMAP_MAXINS );
+ vTruth = Vec_IntAlloc( 1 << 16 );
+ vNames = Vec_PtrAlloc( 100 );
+ pMan = Hop_ManStart();
+ Hop_IthVar( pMan, nPinMax - 1 );
+ Vec_PtrForEachEntry( p->vGates, pGate, i )
+ {
+ if ( pGate->nPins == 0 )
+ {
+ pGate->pFunc = (unsigned *)Aig_MmFlexEntryFetch( p->pMemGates, 4 );
+ if ( strcmp( pGate->pForm, AMAP_STRING_CONST0 ) == 0 )
+ pGate->pFunc[0] = 0;
+ else if ( strcmp( pGate->pForm, AMAP_STRING_CONST1 ) == 0 )
+ pGate->pFunc[0] = ~0;
+ else
+ {
+ printf( "Cannot parse formula \"%s\" of gate \"%s\" with no pins.\n", pGate->pForm, pGate->pName );
+ break;
+ }
+ continue;
+ }
+ if ( pGate->nPins > AMAP_MAXINS )
+ continue;
+ Vec_PtrClear( vNames );
+ Amap_GateForEachPin( pGate, pPin )
+ Vec_PtrPush( vNames, pPin->pName );
+ pObj = Amap_ParseFormula( stdout, pGate->pForm, vNames, pMan );
+ if ( pObj == NULL )
+ break;
+ pTruth = Hop_ManConvertAigToTruth( pMan, pObj, pGate->nPins, vTruth, 0 );
+ if ( Kit_TruthSupportSize(pTruth, pGate->nPins) < (int)pGate->nPins )
+ {
+ printf( "Skipping gate \"%s\" because its formula \"%s\" does not depend on some pin variables.\n", pGate->pName, pGate->pForm );
+ continue;
+ }
+ pGate->pFunc = (unsigned *)Aig_MmFlexEntryFetch( p->pMemGates, sizeof(unsigned)*Aig_TruthWordNum(pGate->nPins) );
+ memcpy( pGate->pFunc, pTruth, sizeof(unsigned)*Aig_TruthWordNum(pGate->nPins) );
+ }
+ Vec_PtrFree( vNames );
+ Vec_IntFree( vTruth );
+ Hop_ManStop( pMan );
+ return i == Vec_PtrSize(p->vGates);
+}
+
+/**Function*************************************************************
+
+ Synopsis [Parses equations for the gates.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+void Amap_LibParseTest( char * pFileName )
+{
+ int fVerbose = 1;
+ Amap_Lib_t * p;
+ int clk = clock();
+ p = Amap_LibReadFile( pFileName, fVerbose );
+ if ( p == NULL )
+ return;
+ Amap_LibParseEquations( p, fVerbose );
+ Amap_LibFree( p );
+ PRT( "Total time", clock() - clk );
+}
+
+////////////////////////////////////////////////////////////////////////
+/// END OF FILE ///
+////////////////////////////////////////////////////////////////////////
+
+
diff --git a/src/map/amap/amapPerm.c b/src/map/amap/amapPerm.c
new file mode 100644
index 00000000..17fb57e2
--- /dev/null
+++ b/src/map/amap/amapPerm.c
@@ -0,0 +1,344 @@
+/**CFile****************************************************************
+
+ FileName [amapPerm.c]
+
+ SystemName [ABC: Logic synthesis and verification system.]
+
+ PackageName [Technology mapper for standard cells.]
+
+ Synopsis [Deriving permutation for the gate.]
+
+ Author [Alan Mishchenko]
+
+ Affiliation [UC Berkeley]
+
+ Date [Ver. 1.0. Started - June 20, 2005.]
+
+ Revision [$Id: amapPerm.c,v 1.00 2005/06/20 00:00:00 alanmi Exp $]
+
+***********************************************************************/
+
+#include "amapInt.h"
+#include "kit.h"
+
+////////////////////////////////////////////////////////////////////////
+/// DECLARATIONS ///
+////////////////////////////////////////////////////////////////////////
+
+////////////////////////////////////////////////////////////////////////
+/// FUNCTION DEFINITIONS ///
+////////////////////////////////////////////////////////////////////////
+
+/**Function*************************************************************
+
+ Synopsis [Collects fanins of the node.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+void Amap_LibCollectFanins_rec( Amap_Lib_t * pLib, Amap_Nod_t * pNod, Vec_Int_t * vFanins )
+{
+ Amap_Nod_t * pFan0, * pFan1;
+ if ( pNod->Id == 0 )
+ {
+ Vec_IntPush( vFanins, 0 );
+ return;
+ }
+ pFan0 = Amap_LibNod( pLib, Amap_Lit2Var(pNod->iFan0) );
+ if ( Amap_LitIsCompl(pNod->iFan0) || pFan0->Type != pNod->Type )
+ Vec_IntPush( vFanins, pNod->iFan0 );
+ else
+ Amap_LibCollectFanins_rec( pLib, pFan0, vFanins );
+ pFan1 = Amap_LibNod( pLib, Amap_Lit2Var(pNod->iFan1) );
+ if ( Amap_LitIsCompl(pNod->iFan1) || pFan1->Type != pNod->Type )
+ Vec_IntPush( vFanins, pNod->iFan1 );
+ else
+ Amap_LibCollectFanins_rec( pLib, pFan1, vFanins );
+}
+
+/**Function*************************************************************
+
+ Synopsis [Collects fanins of the node.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Vec_Int_t * Amap_LibCollectFanins( Amap_Lib_t * pLib, Amap_Nod_t * pNod )
+{
+ Vec_Int_t * vFanins = Vec_IntAlloc( 10 );
+ Amap_LibCollectFanins_rec( pLib, pNod, vFanins );
+ return vFanins;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Matches the node with the DSD node.]
+
+ Description [Returns perm if the node can be matched.]
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Vec_Int_t * Amap_LibDeriveGatePerm_rec( Amap_Lib_t * pLib, Kit_DsdNtk_t * pNtk, int iLit, Amap_Nod_t * pNod )
+{
+ Vec_Int_t * vPerm, * vPermFanin, * vNodFanin, * vDsdLits;
+ Kit_DsdObj_t * pDsdObj, * pDsdFanin;
+ Amap_Nod_t * pNodFanin;
+ int iDsdFanin, iNodFanin, Value, iDsdLit, i, k, j;
+ assert( !Kit_DsdLitIsCompl(iLit) );
+ pDsdObj = Kit_DsdNtkObj( pNtk, Kit_DsdLit2Var(iLit) );
+ if ( pDsdObj == NULL )
+ {
+ vPerm = Vec_IntAlloc( 1 );
+ Vec_IntPush( vPerm, iLit );
+ return vPerm;
+ }
+ if ( pDsdObj->Type == KIT_DSD_PRIME && pNod->Type == AMAP_OBJ_MUX )
+ {
+ vPerm = Vec_IntAlloc( 10 );
+
+ iDsdFanin = Kit_DsdLitRegular(pDsdObj->pFans[0]);
+ pNodFanin = Amap_LibNod( pLib, Amap_Lit2Var(pNod->iFan0) );
+ vPermFanin = Amap_LibDeriveGatePerm_rec( pLib, pNtk, iDsdFanin, pNodFanin );
+ Vec_IntForEachEntry( vPermFanin, Value, k )
+ Vec_IntPush( vPerm, Value );
+ Vec_IntFree( vPermFanin );
+
+ iDsdFanin = Kit_DsdLitRegular(pDsdObj->pFans[1]);
+ pNodFanin = Amap_LibNod( pLib, Amap_Lit2Var(pNod->iFan1) );
+ vPermFanin = Amap_LibDeriveGatePerm_rec( pLib, pNtk, iDsdFanin, pNodFanin );
+ Vec_IntForEachEntry( vPermFanin, Value, k )
+ Vec_IntPush( vPerm, Value );
+ Vec_IntFree( vPermFanin );
+
+ iDsdFanin = Kit_DsdLitRegular(pDsdObj->pFans[2]);
+ pNodFanin = Amap_LibNod( pLib, Amap_Lit2Var(pNod->iFan2) );
+ vPermFanin = Amap_LibDeriveGatePerm_rec( pLib, pNtk, iDsdFanin, pNodFanin );
+ Vec_IntForEachEntry( vPermFanin, Value, k )
+ Vec_IntPush( vPerm, Value );
+ Vec_IntFree( vPermFanin );
+
+ return vPerm;
+ }
+ // return if wrong types
+ if ( pDsdObj->Type == KIT_DSD_PRIME || pNod->Type == AMAP_OBJ_MUX )
+ return NULL;
+ // return if sizes do not agree
+ vNodFanin = Amap_LibCollectFanins( pLib, pNod );
+ if ( Vec_IntSize(vNodFanin) != (int)pDsdObj->nFans )
+ {
+ Vec_IntFree( vNodFanin );
+ return NULL;
+ }
+ // match fanins of DSD with fanins of nodes
+ // clean the mark and save variable literals
+ vPerm = Vec_IntAlloc( 10 );
+ vDsdLits = Vec_IntAlloc( 10 );
+ Kit_DsdObjForEachFaninReverse( pNtk, pDsdObj, iDsdFanin, i )
+ {
+ pDsdFanin = Kit_DsdNtkObj( pNtk, Kit_DsdLit2Var(iDsdFanin) );
+ if ( pDsdFanin )
+ pDsdFanin->fMark = 0;
+ else
+ Vec_IntPush( vDsdLits, iDsdFanin );
+ }
+ // match each fanins of the node
+ iDsdLit = 0;
+ Vec_IntForEachEntry( vNodFanin, iNodFanin, k )
+ {
+ if ( iNodFanin == 0 )
+ {
+ iDsdFanin = Vec_IntEntry( vDsdLits, iDsdLit++ );
+ Vec_IntPush( vPerm, iDsdFanin );
+ continue;
+ }
+ // find a matching component
+ pNodFanin = Amap_LibNod( pLib, Amap_Lit2Var(iNodFanin) );
+ Kit_DsdObjForEachFaninReverse( pNtk, pDsdObj, iDsdFanin, i )
+ {
+ pDsdFanin = Kit_DsdNtkObj( pNtk, Kit_DsdLit2Var(iDsdFanin) );
+ if ( pDsdFanin == NULL )
+ continue;
+ if ( pDsdFanin->fMark == 1 )
+ continue;
+ if ( !((pDsdFanin->Type == KIT_DSD_AND && pNodFanin->Type == AMAP_OBJ_AND) ||
+ (pDsdFanin->Type == KIT_DSD_XOR && pNodFanin->Type == AMAP_OBJ_XOR) ||
+ (pDsdFanin->Type == KIT_DSD_PRIME && pNodFanin->Type == AMAP_OBJ_MUX)) )
+ continue;
+ vPermFanin = Amap_LibDeriveGatePerm_rec( pLib, pNtk, Kit_DsdLitRegular(iDsdFanin), pNodFanin );
+ if ( vPermFanin == NULL )
+ continue;
+ pDsdFanin->fMark = 1;
+ Vec_IntForEachEntry( vPermFanin, Value, j )
+ Vec_IntPush( vPerm, Value );
+ Vec_IntFree( vPermFanin );
+ break;
+ }
+ }
+ assert( iDsdLit == Vec_IntSize(vDsdLits) );
+ Vec_IntFree( vNodFanin );
+ Vec_IntFree( vDsdLits );
+ return vPerm;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Performs verification of one gate and one node.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+unsigned * Amap_LibVerifyPerm_rec( Amap_Lib_t * pLib, Amap_Nod_t * pNod,
+ Vec_Ptr_t * vTtElems, Vec_Int_t * vTruth, int nWords, int * piInput )
+{
+ Amap_Nod_t * pFan0, * pFan1;
+ unsigned * pTruth0, * pTruth1, * pTruth;
+ int i;
+ assert( pNod->Type != AMAP_OBJ_MUX );
+ if ( pNod->Id == 0 )
+ return Vec_PtrEntry( vTtElems, (*piInput)++ );
+ pFan0 = Amap_LibNod( pLib, Amap_Lit2Var(pNod->iFan0) );
+ pTruth0 = Amap_LibVerifyPerm_rec( pLib, pFan0, vTtElems, vTruth, nWords, piInput );
+ pFan1 = Amap_LibNod( pLib, Amap_Lit2Var(pNod->iFan1) );
+ pTruth1 = Amap_LibVerifyPerm_rec( pLib, pFan1, vTtElems, vTruth, nWords, piInput );
+ pTruth = Vec_IntFetch( vTruth, nWords );
+ if ( pNod->Type == AMAP_OBJ_XOR )
+ for ( i = 0; i < nWords; i++ )
+ pTruth[i] = pTruth0[i] ^ pTruth1[i];
+ else if ( !Amap_LitIsCompl(pNod->iFan0) && !Amap_LitIsCompl(pNod->iFan1) )
+ for ( i = 0; i < nWords; i++ )
+ pTruth[i] = pTruth0[i] & pTruth1[i];
+ else if ( !Amap_LitIsCompl(pNod->iFan0) && Amap_LitIsCompl(pNod->iFan1) )
+ for ( i = 0; i < nWords; i++ )
+ pTruth[i] = pTruth0[i] & ~pTruth1[i];
+ else if ( Amap_LitIsCompl(pNod->iFan0) && !Amap_LitIsCompl(pNod->iFan1) )
+ for ( i = 0; i < nWords; i++ )
+ pTruth[i] = ~pTruth0[i] & pTruth1[i];
+ else // if ( Amap_LitIsCompl(pNod->iFan0) && Hop_ObjFaninC1(pObj) )
+ for ( i = 0; i < nWords; i++ )
+ pTruth[i] = ~pTruth0[i] & ~pTruth1[i];
+ return pTruth;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Performs verification of one gate and one node.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+void Amap_LibVerifyPerm( Amap_Lib_t * pLib, Amap_Gat_t * pGate, Kit_DsdNtk_t * pNtk, Amap_Nod_t * pNod, int * pArray )
+{
+ Vec_Ptr_t * vTtElems;
+ Vec_Ptr_t * vTtElemsPol;
+ Vec_Int_t * vTruth;
+ unsigned * pTruth;
+ int i, nWords;
+ int iInput = 0;
+
+ // allocate storage for truth tables
+ assert( pGate->nPins > 1 );
+ nWords = Kit_TruthWordNum( pGate->nPins );
+ vTruth = Vec_IntAlloc( nWords * AMAP_MAXINS );
+ vTtElems = Vec_PtrAllocTruthTables( pGate->nPins );
+ vTtElemsPol = Vec_PtrAlloc( pGate->nPins );
+ for ( i = 0; i < (int)pGate->nPins; i++ )
+ {
+ pTruth = Vec_PtrEntry( vTtElems, Amap_Lit2Var(pArray[i]) );
+ if ( Amap_LitIsCompl( pArray[i] ) )
+ Kit_TruthNot( pTruth, pTruth, pGate->nPins );
+ Vec_PtrPush( vTtElemsPol, pTruth );
+ }
+//Extra_PrintBinary( stdout, Vec_PtrEntry(vTtElemsPol, 0), 4 ); printf("\n" );
+//Extra_PrintBinary( stdout, Vec_PtrEntry(vTtElemsPol, 1), 4 ); printf("\n" );
+ // compute the truth table recursively
+ pTruth = Amap_LibVerifyPerm_rec( pLib, pNod, vTtElemsPol, vTruth, nWords, &iInput );
+ assert( iInput == (int)pGate->nPins );
+ if ( Kit_DsdLitIsCompl(pNtk->Root) )
+ Kit_TruthNot( pTruth, pTruth, pGate->nPins );
+//Extra_PrintBinary( stdout, pTruth, 4 ); printf("\n" );
+//Extra_PrintBinary( stdout, pGate->pFunc, 4 ); printf("\n" );
+ // compare
+ if ( !Kit_TruthIsEqual(pGate->pFunc, pTruth, pGate->nPins) )
+ printf( "Verification failed for gate %d (%s) and node %d.\n",
+ pGate->Id, pGate->pForm, pNod->Id );
+ Vec_IntFree( vTruth );
+ Vec_PtrFree( vTtElems );
+ Vec_PtrFree( vTtElemsPol );
+}
+
+/**Function*************************************************************
+
+ Synopsis [Matches the node with the DSD of a gate.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+int Amap_LibDeriveGatePerm( Amap_Lib_t * pLib, Amap_Gat_t * pGate, Kit_DsdNtk_t * pNtk, Amap_Nod_t * pNod, char * pArray )
+{
+ int fVerbose = 0;
+ Vec_Int_t * vPerm;
+ int Entry, Entry2, i, k;
+ vPerm = Amap_LibDeriveGatePerm_rec( pLib, pNtk, Kit_DsdLitRegular(pNtk->Root), pNod );
+ if ( vPerm == NULL )
+ return 0;
+ // check that the permutation is valid
+ assert( Vec_IntSize(vPerm) == (int)pNod->nSuppSize );
+ Vec_IntForEachEntry( vPerm, Entry, i )
+ Vec_IntForEachEntryStart( vPerm, Entry2, k, i+1 )
+ if ( Amap_Lit2Var(Entry) == Amap_Lit2Var(Entry2) )
+ {
+ Vec_IntFree( vPerm );
+ return 0;
+ }
+
+ // reverse the permutation
+ Vec_IntForEachEntry( vPerm, Entry, i )
+ {
+ assert( Entry < 2 * (int)pNod->nSuppSize );
+ pArray[Kit_DsdLit2Var(Entry)] = Amap_Var2Lit( i, Kit_DsdLitIsCompl(Entry) );
+// pArray[i] = Entry;
+//printf( "%d=%d%c ", Kit_DsdLit2Var(Entry), i, Kit_DsdLitIsCompl(Entry)?'-':'+' );
+ }
+//printf( "\n" );
+// if ( Kit_DsdNonDsdSizeMax(pNtk) < 3 )
+// Amap_LibVerifyPerm( pLib, pGate, pNtk, pNod, Vec_IntArray(vPerm) );
+ Vec_IntFree( vPerm );
+ // print the result
+ if ( fVerbose )
+ {
+ printf( "node %4d : ", pNod->Id );
+ for ( i = 0; i < (int)pNod->nSuppSize; i++ )
+ printf( "%d=%d%c ", i, Amap_Lit2Var(pArray[i]), Amap_LitIsCompl(pArray[i])?'-':'+' );
+ printf( "\n" );
+ }
+ return 1;
+}
+
+////////////////////////////////////////////////////////////////////////
+/// END OF FILE ///
+////////////////////////////////////////////////////////////////////////
+
+
diff --git a/src/map/amap/amapRead.c b/src/map/amap/amapRead.c
new file mode 100644
index 00000000..182de5d1
--- /dev/null
+++ b/src/map/amap/amapRead.c
@@ -0,0 +1,412 @@
+/**CFile****************************************************************
+
+ FileName [amapRead.c]
+
+ SystemName [ABC: Logic synthesis and verification system.]
+
+ PackageName [Technology mapper for standard cells.]
+
+ Synopsis []
+
+ Author [Alan Mishchenko]
+
+ Affiliation [UC Berkeley]
+
+ Date [Ver. 1.0. Started - June 20, 2005.]
+
+ Revision [$Id: amapRead.c,v 1.00 2005/06/20 00:00:00 alanmi Exp $]
+
+***********************************************************************/
+
+#include "amapInt.h"
+
+////////////////////////////////////////////////////////////////////////
+/// DECLARATIONS ///
+////////////////////////////////////////////////////////////////////////
+
+#define AMAP_STRING_GATE "GATE"
+#define AMAP_STRING_PIN "PIN"
+#define AMAP_STRING_NONINV "NONINV"
+#define AMAP_STRING_INV "INV"
+#define AMAP_STRING_UNKNOWN "UNKNOWN"
+
+// these symbols (and no other) can appear in the formulas
+#define AMAP_SYMB_AND '*'
+#define AMAP_SYMB_OR1 '+'
+#define AMAP_SYMB_OR2 '|'
+#define AMAP_SYMB_XOR '^'
+#define AMAP_SYMB_NOT '!'
+#define AMAP_SYMB_AFTNOT '\''
+#define AMAP_SYMB_OPEN '('
+#define AMAP_SYMB_CLOSE ')'
+
+typedef enum {
+ AMAP_PHASE_UNKNOWN,
+ AMAP_PHASE_INV,
+ AMAP_PHASE_NONINV
+} Amap_PinPhase_t;
+
+static inline Amap_Gat_t * Amap_ParseGateAlloc( Aig_MmFlex_t * p, int nPins )
+{ return (Amap_Gat_t *)Aig_MmFlexEntryFetch( p, sizeof(Amap_Gat_t)+sizeof(Amap_Pin_t)*nPins ); }
+static inline char * Amap_ParseStrsav( Aig_MmFlex_t * p, char * pStr )
+{ return pStr ? strcpy(Aig_MmFlexEntryFetch(p, strlen(pStr)+1), pStr) : NULL; }
+
+////////////////////////////////////////////////////////////////////////
+/// FUNCTION DEFINITIONS ///
+////////////////////////////////////////////////////////////////////////
+
+/**Function*************************************************************
+
+ Synopsis [Loads the file into temporary buffer.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+char * Amap_LoadFile( char * pFileName )
+{
+ extern FILE * Io_FileOpen( const char * FileName, const char * PathVar, const char * Mode, int fVerbose );
+ FILE * pFile;
+ char * pBuffer;
+ int nFileSize;
+ // open the BLIF file for binary reading
+ pFile = Io_FileOpen( pFileName, "open_path", "rb", 1 );
+// pFile = fopen( FileName, "rb" );
+ // if we got this far, file should be okay otherwise would
+ // have been detected by caller
+ if ( pFile == NULL )
+ {
+ printf( "Cannot open file \"%s\".\n", pFileName );
+ return NULL;
+ }
+ assert ( pFile != NULL );
+ // get the file size, in bytes
+ fseek( pFile, 0, SEEK_END );
+ nFileSize = ftell( pFile );
+ // move the file current reading position to the beginning
+ rewind( pFile );
+ // load the contents of the file into memory
+ pBuffer = ALLOC( char, nFileSize + 10 );
+ fread( pBuffer, nFileSize, 1, pFile );
+ // terminate the string with '\0'
+ pBuffer[ nFileSize ] = '\0';
+ strcat( pBuffer, "\n.end\n" );
+ // close file
+ fclose( pFile );
+ return pBuffer;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Eliminates comments from the input file.]
+
+ Description [As a byproduct, this procedure also counts the number
+ lines and dot-statements in the input file. This also joins non-comment
+ lines that are joined with a backspace '\']
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+void Amap_RemoveComments( char * pBuffer, int * pnDots, int * pnLines )
+{
+ char * pCur;
+ int nDots, nLines;
+ // scan through the buffer and eliminate comments
+ // (in the BLIF file, comments are lines starting with "#")
+ nDots = nLines = 0;
+ for ( pCur = pBuffer; *pCur; pCur++ )
+ {
+ // if this is the beginning of comment
+ // clean it with spaces until the new line statement
+ if ( *pCur == '#' )
+ while ( *pCur != '\n' )
+ *pCur++ = ' ';
+
+ // count the number of new lines and dots
+ if ( *pCur == '\n' ) {
+ if (*(pCur-1)=='\r') {
+ // DOS(R) file support
+ if (*(pCur-2)!='\\') nLines++;
+ else {
+ // rewind to backslash and overwrite with a space
+ *(pCur-2) = ' ';
+ *(pCur-1) = ' ';
+ *pCur = ' ';
+ }
+ } else {
+ // UNIX(TM) file support
+ if (*(pCur-1)!='\\') nLines++;
+ else {
+ // rewind to backslash and overwrite with a space
+ *(pCur-1) = ' ';
+ *pCur = ' ';
+ }
+ }
+ }
+ else if ( *pCur == '.' )
+ nDots++;
+ }
+ if ( pnDots )
+ *pnDots = nDots;
+ if ( pnLines )
+ *pnLines = nLines;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Splits the stream into tokens.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Vec_Ptr_t * Amap_DeriveTokens( char * pBuffer )
+{
+ Vec_Ptr_t * vTokens;
+ char * pToken;
+ vTokens = Vec_PtrAlloc( 1000 );
+ pToken = strtok( pBuffer, " ;=\t\r\n" );
+ while ( pToken )
+ {
+ Vec_PtrPush( vTokens, pToken );
+ pToken = strtok( NULL, " ;=\t\r\n" );
+ }
+ return vTokens;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Finds the number of pins.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+int Amap_ParseCountPins( Vec_Ptr_t * vTokens, int iPos )
+{
+ char * pToken;
+ int i, Counter = 0;
+ Vec_PtrForEachEntryStart( vTokens, pToken, i, iPos )
+ if ( !strcmp( pToken, AMAP_STRING_PIN ) )
+ Counter++;
+ else if ( !strcmp( pToken, AMAP_STRING_GATE ) )
+ return Counter;
+ return Counter;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Collect the pin names used in the formula.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+int Amap_GateCollectNames( Aig_MmFlex_t * pMem, char * pForm, char * pPinNames[] )
+{
+ char Buffer[1000];
+ char * pTemp;
+ int nPins, i;
+ // save the formula as it was
+ strcpy( Buffer, pForm );
+ // remove the non-name symbols
+ for ( pTemp = Buffer; *pTemp; pTemp++ )
+ if ( *pTemp == AMAP_SYMB_AND || *pTemp == AMAP_SYMB_OR1 || *pTemp == AMAP_SYMB_OR2
+ || *pTemp == AMAP_SYMB_XOR || *pTemp == AMAP_SYMB_NOT || *pTemp == AMAP_SYMB_OPEN
+ || *pTemp == AMAP_SYMB_CLOSE || *pTemp == AMAP_SYMB_AFTNOT )
+ *pTemp = ' ';
+ // save the names
+ nPins = 0;
+ pTemp = strtok( Buffer, " " );
+ while ( pTemp )
+ {
+ for ( i = 0; i < nPins; i++ )
+ if ( strcmp( pTemp, pPinNames[i] ) == 0 )
+ break;
+ if ( i == nPins )
+ { // cannot find this name; save it
+ pPinNames[nPins++] = Amap_ParseStrsav( pMem, pTemp );
+ }
+ // get the next name
+ pTemp = strtok( NULL, " " );
+ }
+ return nPins;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Creates a duplicate gate with pins specified.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Amap_Gat_t * Amap_ParseGateWithSamePins( Amap_Gat_t * p )
+{
+ Amap_Gat_t * pGate;
+ Amap_Pin_t * pPin;
+ char * pPinNames[128];
+ int nPinNames;
+ assert( p->nPins == 1 && !strcmp( p->Pins->pName, "*" ) );
+ nPinNames = Amap_GateCollectNames( p->pLib->pMemGates, p->pForm, pPinNames );
+ pGate = Amap_ParseGateAlloc( p->pLib->pMemGates, nPinNames );
+ *pGate = *p;
+ pGate->nPins = nPinNames;
+ Amap_GateForEachPin( pGate, pPin )
+ {
+ *pPin = *p->Pins;
+ pPin->pName = pPinNames[pPin - pGate->Pins];
+ }
+ return pGate;
+}
+
+/**Function*************************************************************
+
+ Synopsis []
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Amap_Lib_t * Amap_ParseTokens( Vec_Ptr_t * vTokens, int fVerbose )
+{
+ Amap_Lib_t * p;
+ Amap_Gat_t * pGate;
+ Amap_Pin_t * pPin;
+ char * pToken;
+ int nPins, iPos = 0;
+ p = Amap_LibAlloc();
+ pToken = Vec_PtrEntry(vTokens, iPos++);
+ do
+ {
+ if ( strcmp( pToken, AMAP_STRING_GATE ) )
+ {
+ printf( "The first line should begin with %s.\n", AMAP_STRING_GATE );
+ return NULL;
+ }
+ // start gate
+ nPins = Amap_ParseCountPins( vTokens, iPos );
+ pGate = Amap_ParseGateAlloc( p->pMemGates, nPins );
+ memset( pGate, 0, sizeof(Amap_Gat_t) );
+ pGate->Id = Vec_PtrSize( p->vGates );
+ Vec_PtrPush( p->vGates, pGate );
+ pGate->pLib = p;
+ pGate->nPins = nPins;
+ // read gate
+ pToken = Vec_PtrEntry(vTokens, iPos++);
+ pGate->pName = Amap_ParseStrsav( p->pMemGates, pToken );
+ pToken = Vec_PtrEntry(vTokens, iPos++);
+ pGate->dArea = atof( pToken );
+ pToken = Vec_PtrEntry(vTokens, iPos++);
+ pGate->pOutName = Amap_ParseStrsav( p->pMemGates, pToken );
+ pToken = Vec_PtrEntry(vTokens, iPos++);
+ pGate->pForm = Amap_ParseStrsav( p->pMemGates, pToken );
+ // read pins
+ Amap_GateForEachPin( pGate, pPin )
+ {
+ pToken = Vec_PtrEntry(vTokens, iPos++);
+ if ( strcmp( pToken, AMAP_STRING_PIN ) )
+ {
+ printf( "Cannot parse gate %s.\n", pGate->pName );
+ return NULL;
+ }
+ // read pin
+ pToken = Vec_PtrEntry(vTokens, iPos++);
+ pPin->pName = Amap_ParseStrsav( p->pMemGates, pToken );
+ pToken = Vec_PtrEntry(vTokens, iPos++);
+ if ( strcmp( pToken, AMAP_STRING_UNKNOWN ) == 0 )
+ pPin->Phase = AMAP_PHASE_UNKNOWN;
+ else if ( strcmp( pToken, AMAP_STRING_INV ) == 0 )
+ pPin->Phase = AMAP_PHASE_INV;
+ else if ( strcmp( pToken, AMAP_STRING_NONINV ) == 0 )
+ pPin->Phase = AMAP_PHASE_NONINV;
+ else
+ {
+ printf( "Cannot read phase of pin %s of gate %s\n", pPin->pName, pGate->pName );
+ return NULL;
+ }
+ pToken = Vec_PtrEntry(vTokens, iPos++);
+ pPin->dLoadInput = atof( pToken );
+ pToken = Vec_PtrEntry(vTokens, iPos++);
+ pPin->dLoadMax = atof( pToken );
+ pToken = Vec_PtrEntry(vTokens, iPos++);
+ pPin->dDelayBlockRise = atof( pToken );
+ pToken = Vec_PtrEntry(vTokens, iPos++);
+ pPin->dDelayFanoutRise = atof( pToken );
+ pToken = Vec_PtrEntry(vTokens, iPos++);
+ pPin->dDelayBlockFall = atof( pToken );
+ pToken = Vec_PtrEntry(vTokens, iPos++);
+ pPin->dDelayFanoutFall = atof( pToken );
+ if ( pPin->dDelayBlockRise > pPin->dDelayBlockFall )
+ pPin->dDelayBlockMax = pPin->dDelayBlockRise;
+ else
+ pPin->dDelayBlockMax = pPin->dDelayBlockFall;
+ }
+ // fix the situation when all pins are represented as one
+ if ( pGate->nPins == 1 && !strcmp( pGate->Pins->pName, "*" ) )
+ {
+ pGate = Amap_ParseGateWithSamePins( pGate );
+ Vec_PtrPop( p->vGates );
+ Vec_PtrPush( p->vGates, pGate );
+ }
+ pToken = Vec_PtrEntry(vTokens, iPos++);
+ }
+ while ( strcmp( pToken, ".end" ) );
+ return p;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Reads the library from the input file.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Amap_Lib_t * Amap_LibReadFile( char * pFileName, int fVerbose )
+{
+ Amap_Lib_t * pLib;
+ Vec_Ptr_t * vTokens;
+ char * pBuffer;
+ pBuffer = Amap_LoadFile( pFileName );
+ if ( pBuffer == NULL )
+ return NULL;
+ Amap_RemoveComments( pBuffer, NULL, NULL );
+ vTokens = Amap_DeriveTokens( pBuffer );
+ pLib = Amap_ParseTokens( vTokens, fVerbose );
+ if ( pLib == NULL )
+ return NULL;
+ pLib->pName = Amap_ParseStrsav( pLib->pMemGates, pFileName );
+ Vec_PtrFree( vTokens );
+ free( pBuffer );
+ return pLib;
+}
+
+////////////////////////////////////////////////////////////////////////
+/// END OF FILE ///
+////////////////////////////////////////////////////////////////////////
+
+
diff --git a/src/map/amap/amapRule.c b/src/map/amap/amapRule.c
new file mode 100644
index 00000000..212f7631
--- /dev/null
+++ b/src/map/amap/amapRule.c
@@ -0,0 +1,368 @@
+/**CFile****************************************************************
+
+ FileName [amapRule.c]
+
+ SystemName [ABC: Logic synthesis and verification system.]
+
+ PackageName [Technology mapper for standard cells.]
+
+ Synopsis [Matching rules.]
+
+ Author [Alan Mishchenko]
+
+ Affiliation [UC Berkeley]
+
+ Date [Ver. 1.0. Started - June 20, 2005.]
+
+ Revision [$Id: amapRule.c,v 1.00 2005/06/20 00:00:00 alanmi Exp $]
+
+***********************************************************************/
+
+#include "amapInt.h"
+#include "kit.h"
+
+////////////////////////////////////////////////////////////////////////
+/// DECLARATIONS ///
+////////////////////////////////////////////////////////////////////////
+
+extern int Amap_LibDeriveGatePerm( Amap_Lib_t * pLib, Amap_Gat_t * pGate, Kit_DsdNtk_t * pNtk, Amap_Nod_t * pNod, char * pArray );
+
+////////////////////////////////////////////////////////////////////////
+/// FUNCTION DEFINITIONS ///
+////////////////////////////////////////////////////////////////////////
+
+/**Function*************************************************************
+
+ Synopsis [Checks if the three-argument rule exist.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Vec_Int_t * Amap_CreateRulesPrime( Amap_Lib_t * p, Vec_Int_t * vNods0, Vec_Int_t * vNods1, Vec_Int_t * vNods2 )
+{
+ Vec_Int_t * vRes;
+ int i, k, j, iNod, iNod0, iNod1, iNod2;
+ if ( p->vRules3 == NULL )
+ p->vRules3 = Vec_IntAlloc( 100 );
+ vRes = Vec_IntAlloc( 10 );
+ Vec_IntForEachEntry( vNods0, iNod0, i )
+ Vec_IntForEachEntry( vNods1, iNod1, k )
+ Vec_IntForEachEntry( vNods2, iNod2, j )
+ {
+ iNod = Amap_LibFindMux( p, iNod0, iNod1, iNod2 );
+ if ( iNod == -1 )
+ iNod = Amap_LibCreateMux( p, iNod0, iNod1, iNod2 );
+ Vec_IntPush( vRes, Amap_Var2Lit(iNod, 0) );
+ }
+ return vRes;
+}
+
+/**Function*************************************************************
+
+ Synopsis []
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+void Amap_CreateRulesTwo( Amap_Lib_t * p, Vec_Int_t * vNods, Vec_Int_t * vNods0, Vec_Int_t * vNods1, int fXor )
+{
+ int i, k, iNod, iNod0, iNod1;
+ Vec_IntForEachEntry( vNods0, iNod0, i )
+ Vec_IntForEachEntry( vNods1, iNod1, k )
+ {
+ iNod = Amap_LibFindNode( p, iNod0, iNod1, fXor );
+ if ( iNod == -1 )
+ iNod = Amap_LibCreateNode( p, iNod0, iNod1, fXor );
+ Vec_IntPushUnique( vNods, Amap_Var2Lit(iNod, 0) );
+ }
+}
+
+/**Function*************************************************************
+
+ Synopsis []
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+int Amap_CreateCheckAllZero( Vec_Ptr_t * vVecNods )
+{
+ Vec_Int_t * vNods;
+ int i;
+ Vec_PtrForEachEntryReverse( vVecNods, vNods, i )
+ if ( Vec_IntSize(vNods) != 1 || Vec_IntEntry(vNods,0) != 0 )
+ return 0;
+ return 1;
+}
+
+/**Function*************************************************************
+
+ Synopsis []
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Vec_Int_t * Amap_CreateRulesVector_rec( Amap_Lib_t * p, Vec_Ptr_t * vVecNods, int fXor )
+{
+ Vec_Ptr_t * vVecNods0, * vVecNods1;
+ Vec_Int_t * vRes, * vNods, * vNods0, * vNods1;
+ int i, k;
+ if ( Vec_PtrSize(vVecNods) == 1 )
+ return Vec_IntDup( Vec_PtrEntry(vVecNods, 0) );
+ vRes = Vec_IntAlloc( 10 );
+ vVecNods0 = Vec_PtrAlloc( Vec_PtrSize(vVecNods) );
+ vVecNods1 = Vec_PtrAlloc( Vec_PtrSize(vVecNods) );
+ if ( Amap_CreateCheckAllZero(vVecNods) )
+ {
+ for ( i = Vec_PtrSize(vVecNods)-1; i > 0; i-- )
+ {
+ Vec_PtrClear( vVecNods0 );
+ Vec_PtrClear( vVecNods1 );
+ Vec_PtrForEachEntryStop( vVecNods, vNods, k, i )
+ Vec_PtrPush( vVecNods0, vNods );
+ Vec_PtrForEachEntryStart( vVecNods, vNods, k, i )
+ Vec_PtrPush( vVecNods1, vNods );
+ vNods0 = Amap_CreateRulesVector_rec( p, vVecNods0, fXor );
+ vNods1 = Amap_CreateRulesVector_rec( p, vVecNods1, fXor );
+ Amap_CreateRulesTwo( p, vRes, vNods0, vNods1, fXor );
+ Vec_IntFree( vNods0 );
+ Vec_IntFree( vNods1 );
+ }
+ }
+ else
+ {
+ int Limit = (1 << Vec_PtrSize(vVecNods))-2;
+ for ( i = 1; i < Limit; i++ )
+ {
+ Vec_PtrClear( vVecNods0 );
+ Vec_PtrClear( vVecNods1 );
+ Vec_PtrForEachEntryReverse( vVecNods, vNods, k )
+ {
+ if ( i & (1 << k) )
+ Vec_PtrPush( vVecNods1, vNods );
+ else
+ Vec_PtrPush( vVecNods0, vNods );
+ }
+ assert( Vec_PtrSize(vVecNods0) > 0 );
+ assert( Vec_PtrSize(vVecNods1) > 0 );
+ assert( Vec_PtrSize(vVecNods0) < Vec_PtrSize(vVecNods) );
+ assert( Vec_PtrSize(vVecNods1) < Vec_PtrSize(vVecNods) );
+ vNods0 = Amap_CreateRulesVector_rec( p, vVecNods0, fXor );
+ vNods1 = Amap_CreateRulesVector_rec( p, vVecNods1, fXor );
+ Amap_CreateRulesTwo( p, vRes, vNods0, vNods1, fXor );
+ Vec_IntFree( vNods0 );
+ Vec_IntFree( vNods1 );
+ }
+ }
+ Vec_PtrFree( vVecNods0 );
+ Vec_PtrFree( vVecNods1 );
+ return vRes;
+}
+
+/**Function*************************************************************
+
+ Synopsis []
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Vec_Int_t * Amap_CreateRulesFromDsd_rec( Amap_Lib_t * pLib, Kit_DsdNtk_t * p, int iLit )
+{
+ Vec_Int_t * vRes;
+ Vec_Ptr_t * vVecNods;
+ Vec_Int_t * vNodsFanin;
+ Kit_DsdObj_t * pObj;
+ unsigned i;
+ int iFanin, iNod, k;
+ assert( !Kit_DsdLitIsCompl(iLit) );
+ pObj = Kit_DsdNtkObj( p, Kit_DsdLit2Var(iLit) );
+ if ( pObj == NULL )
+ return Vec_IntStartNatural( 1 );
+ // solve for the inputs
+ vVecNods = Vec_PtrAlloc( pObj->nFans );
+ Kit_DsdObjForEachFanin( p, pObj, iFanin, i )
+ {
+ vNodsFanin = Amap_CreateRulesFromDsd_rec( pLib, p, Kit_DsdLitRegular(iFanin) );
+ if ( Kit_DsdLitIsCompl(iFanin) )
+ {
+ Vec_IntForEachEntry( vNodsFanin, iNod, k )
+ if ( iNod > 0 )
+ Vec_IntWriteEntry( vNodsFanin, k, Amap_LitNot(iNod) );
+ }
+ Vec_PtrPush( vVecNods, vNodsFanin );
+ }
+ if ( pObj->Type == KIT_DSD_AND )
+ vRes = Amap_CreateRulesVector_rec( pLib, vVecNods, 0 );
+ else if ( pObj->Type == KIT_DSD_XOR )
+ vRes = Amap_CreateRulesVector_rec( pLib, vVecNods, 1 );
+ else if ( pObj->Type == KIT_DSD_PRIME )
+ {
+ assert( pObj->nFans == 3 );
+ assert( Kit_DsdObjTruth(pObj)[0] == 0xCACACACA );
+ vRes = Amap_CreateRulesPrime( pLib, Vec_PtrEntry(vVecNods, 0),
+ Vec_PtrEntry(vVecNods, 1), Vec_PtrEntry(vVecNods, 2) );
+ }
+ else assert( 0 );
+ Vec_PtrForEachEntry( vVecNods, vNodsFanin, k )
+ Vec_IntFree( vNodsFanin );
+ Vec_PtrFree( vVecNods );
+ return vRes;
+}
+
+/**Function*************************************************************
+
+ Synopsis []
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Vec_Int_t * Amap_CreateRulesFromDsd( Amap_Lib_t * pLib, Kit_DsdNtk_t * p )
+{
+ Vec_Int_t * vNods;
+ int iNod, i;
+ assert( p->nVars >= 2 );
+ vNods = Amap_CreateRulesFromDsd_rec( pLib, p, Kit_DsdLitRegular(p->Root) );
+ if ( vNods == NULL )
+ return NULL;
+ if ( Kit_DsdLitIsCompl(p->Root) )
+ {
+ Vec_IntForEachEntry( vNods, iNod, i )
+ Vec_IntWriteEntry( vNods, i, Amap_LitNot(iNod) );
+ }
+ return vNods;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Creates rules for the given gate]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+void Amap_CreateRulesForGate( Amap_Lib_t * pLib, Amap_Gat_t * pGate )
+{
+ Kit_DsdNtk_t * pNtk, * pTemp;
+ Vec_Int_t * vNods;
+ Amap_Nod_t * pNod;
+ Amap_Set_t * pSet;
+ int iNod, i;
+// if ( pGate->nPins > 4 )
+// return;
+ pNtk = Kit_DsdDecomposeMux( pGate->pFunc, pGate->nPins, 2 );
+ if ( pGate->nPins == 2 && (pGate->pFunc[0] == 0x66666666 || pGate->pFunc[0] == ~0x66666666) )
+ pLib->fHasXor = 1;
+ if ( Kit_DsdNonDsdSizeMax(pNtk) == 3 )
+ pLib->fHasMux = pGate->fMux = 1;
+ pNtk = Kit_DsdExpand( pTemp = pNtk );
+ Kit_DsdNtkFree( pTemp );
+ Kit_DsdVerify( pNtk, pGate->pFunc, pGate->nPins );
+if ( pLib->fVerbose )
+//if ( pGate->fMux )
+{
+printf( "\nProcessing library gate %4d: %10s ", pGate->Id, pGate->pName );
+Kit_DsdPrint( stdout, pNtk );
+}
+ vNods = Amap_CreateRulesFromDsd( pLib, pNtk );
+ if ( vNods )
+ {
+ Vec_IntForEachEntry( vNods, iNod, i )
+ {
+ assert( iNod > 1 );
+ pNod = Amap_LibNod( pLib, Amap_Lit2Var(iNod) );
+ assert( pNod->Type == AMAP_OBJ_MUX || pNod->nSuppSize == pGate->nPins );
+ pSet = (Amap_Set_t *)Aig_MmFlexEntryFetch( pLib->pMemSet, sizeof(Amap_Set_t) );
+ memset( pSet, 0, sizeof(Amap_Set_t) );
+ pSet->iGate = pGate->Id;
+ pSet->fInv = Amap_LitIsCompl(iNod);
+ pSet->nIns = pGate->nPins;
+ if ( Amap_LibDeriveGatePerm( pLib, pGate, pNtk, pNod, pSet->Ins ) == 0 )
+ {
+if ( pLib->fVerbose )
+{
+ printf( "Cound not prepare gate \"%s\": ", pGate->pName );
+ Kit_DsdPrint( stdout, pNtk );
+}
+ continue;
+ }
+ pSet->pNext = pNod->pSets;
+ pNod->pSets = pSet;
+ pLib->nSets++;
+ }
+ Vec_IntFree( vNods );
+ }
+ Kit_DsdNtkFree( pNtk );
+}
+
+/**Function*************************************************************
+
+ Synopsis [Creates rules for the given gate]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+void Amap_LibCreateRules( Amap_Lib_t * pLib, int fVeryVerbose )
+{
+ Amap_Gat_t * pGate;
+ int i, nGates = 0;
+ int clk = clock();
+ pLib->fVerbose = fVeryVerbose;
+ pLib->vRules = Vec_PtrAlloc( 100 );
+ pLib->vRulesX = Vec_PtrAlloc( 100 );
+ pLib->vRules3 = Vec_IntAlloc( 100 );
+ Amap_LibCreateVar( pLib );
+ Vec_PtrForEachEntry( pLib->vSelect, pGate, i )
+ {
+ if ( pGate->nPins < 2 )
+ continue;
+ if ( pGate->pFunc == NULL )
+ {
+ printf( "Amap_LibCreateRules(): Skipping gate %s (%s).\n", pGate->pName, pGate->pForm );
+ continue;
+ }
+ Amap_CreateRulesForGate( pLib, pGate );
+ nGates++;
+ }
+ assert( Vec_PtrSize(pLib->vRules) == 2*pLib->nNodes );
+ assert( Vec_PtrSize(pLib->vRulesX) == 2*pLib->nNodes );
+ pLib->pRules = Amap_LibLookupTableAlloc( pLib->vRules, 0 );
+ pLib->pRulesX = Amap_LibLookupTableAlloc( pLib->vRulesX, 0 );
+ Vec_VecFree( (Vec_Vec_t *)pLib->vRules ); pLib->vRules = NULL;
+ Vec_VecFree( (Vec_Vec_t *)pLib->vRulesX ); pLib->vRulesX = NULL;
+}
+
+////////////////////////////////////////////////////////////////////////
+/// END OF FILE ///
+////////////////////////////////////////////////////////////////////////
+
+
diff --git a/src/map/amap/amapUniq.c b/src/map/amap/amapUniq.c
new file mode 100644
index 00000000..83fce6a2
--- /dev/null
+++ b/src/map/amap/amapUniq.c
@@ -0,0 +1,312 @@
+/**CFile****************************************************************
+
+ FileName [amapUniq.c]
+
+ SystemName [ABC: Logic synthesis and verification system.]
+
+ PackageName [Technology mapper for standard cells.]
+
+ Synopsis [Checks if the structural node already exists.]
+
+ Author [Alan Mishchenko]
+
+ Affiliation [UC Berkeley]
+
+ Date [Ver. 1.0. Started - June 20, 2005.]
+
+ Revision [$Id: amapUniq.c,v 1.00 2005/06/20 00:00:00 alanmi Exp $]
+
+***********************************************************************/
+
+#include "amapInt.h"
+
+////////////////////////////////////////////////////////////////////////
+/// DECLARATIONS ///
+////////////////////////////////////////////////////////////////////////
+
+////////////////////////////////////////////////////////////////////////
+/// FUNCTION DEFINITIONS ///
+////////////////////////////////////////////////////////////////////////
+
+/**Function*************************************************************
+
+ Synopsis [Checks if the entry exists and returns value.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+static inline int Vec_IntCheckWithMask( Vec_Int_t * p, int Entry )
+{
+ int i;
+ for ( i = 0; i < p->nSize; i++ )
+ if ( (0xffff & p->pArray[i]) == (0xffff & Entry) )
+ return p->pArray[i] >> 16;
+ return -1;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Pushes entry in the natural order.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+static inline void Vec_IntPushOrderWithMask( Vec_Int_t * p, int Entry )
+{
+ int i;
+ if ( p->nSize == p->nCap )
+ Vec_IntGrow( p, 2 * p->nCap );
+ p->nSize++;
+ for ( i = p->nSize-2; i >= 0; i-- )
+ if ( (0xffff & p->pArray[i]) > (0xffff & Entry) )
+ p->pArray[i+1] = p->pArray[i];
+ else
+ break;
+ p->pArray[i+1] = Entry;
+}
+
+/**Function*************************************************************
+
+ Synopsis []
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+int Amap_LibFindNode( Amap_Lib_t * pLib, int iFan0, int iFan1, int fXor )
+{
+ if ( fXor )
+ return Vec_IntCheckWithMask( Vec_PtrEntry(pLib->vRulesX, iFan0), iFan1 );
+ else
+ return Vec_IntCheckWithMask( Vec_PtrEntry(pLib->vRules, iFan0), iFan1 );
+}
+
+/**Function*************************************************************
+
+ Synopsis [Checks if the three-argument rule exist.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+int Amap_LibFindMux( Amap_Lib_t * p, int iFan0, int iFan1, int iFan2 )
+{
+ int x;
+ for ( x = 0; x < Vec_IntSize(p->vRules3); x += 4 )
+ {
+ if ( Vec_IntEntry(p->vRules3, x) == iFan0 &&
+ Vec_IntEntry(p->vRules3, x+1) == iFan1 &&
+ Vec_IntEntry(p->vRules3, x+2) == iFan2 )
+ {
+ return Vec_IntEntry(p->vRules3, x+3);
+ }
+ }
+ return -1;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Creates a new node.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+Amap_Nod_t * Amap_LibCreateObj( Amap_Lib_t * p )
+{
+ Amap_Nod_t * pNode;
+ if ( p->nNodes == p->nNodesAlloc )
+ {
+ p->pNodes = REALLOC( Amap_Nod_t, p->pNodes, 2*p->nNodesAlloc );
+ p->nNodesAlloc *= 2;
+ }
+ pNode = Amap_LibNod( p, p->nNodes );
+ memset( pNode, 0, sizeof(Amap_Nod_t) );
+ pNode->Id = p->nNodes++;
+ Vec_PtrPush( p->vRules, Vec_IntAlloc(8) );
+ Vec_PtrPush( p->vRules, Vec_IntAlloc(8) );
+ Vec_PtrPush( p->vRulesX, Vec_IntAlloc(8) );
+ Vec_PtrPush( p->vRulesX, Vec_IntAlloc(8) );
+ return pNode;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Creates a new node.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+int Amap_LibCreateVar( Amap_Lib_t * p )
+{
+ Amap_Nod_t * pNode;
+ // start the manager
+ assert( p->pNodes == NULL );
+ p->nNodesAlloc = 256;
+ p->pNodes = ALLOC( Amap_Nod_t, p->nNodesAlloc );
+ // create the first node
+ pNode = Amap_LibCreateObj( p );
+ p->pNodes->Type = AMAP_OBJ_PI;
+ p->pNodes->nSuppSize = 1;
+ return 0;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Creates a new node.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+int Amap_LibCreateNode( Amap_Lib_t * p, int iFan0, int iFan1, int fXor )
+{
+ Amap_Nod_t * pNode;
+ int iFan;
+ if ( iFan0 < iFan1 )
+ {
+ iFan = iFan0;
+ iFan0 = iFan1;
+ iFan1 = iFan;
+ }
+ pNode = Amap_LibCreateObj( p );
+ pNode->Type = fXor? AMAP_OBJ_XOR : AMAP_OBJ_AND;
+ pNode->nSuppSize = p->pNodes[Amap_Lit2Var(iFan0)].nSuppSize + p->pNodes[Amap_Lit2Var(iFan1)].nSuppSize;
+ pNode->iFan0 = iFan0;
+ pNode->iFan1 = iFan1;
+if ( p->fVerbose )
+printf( "Creating node %5d %c : iFan0 = %5d%c iFan1 = %5d%c\n",
+pNode->Id, (fXor?'x':' '),
+Amap_Lit2Var(iFan0), (Amap_LitIsCompl(iFan0)?'-':'+'),
+Amap_Lit2Var(iFan1), (Amap_LitIsCompl(iFan1)?'-':'+') );
+
+ if ( fXor )
+ {
+ if ( iFan0 == iFan1 )
+ Vec_IntPushOrderWithMask( Vec_PtrEntry(p->vRulesX, iFan0), (pNode->Id << 16) | iFan1 );
+ else
+ {
+ Vec_IntPushOrderWithMask( Vec_PtrEntry(p->vRulesX, iFan0), (pNode->Id << 16) | iFan1 );
+ Vec_IntPushOrderWithMask( Vec_PtrEntry(p->vRulesX, iFan1), (pNode->Id << 16) | iFan0 );
+ }
+ }
+ else
+ {
+ if ( iFan0 == iFan1 )
+ Vec_IntPushOrderWithMask( Vec_PtrEntry(p->vRules, iFan0), (pNode->Id << 16) | iFan1 );
+ else
+ {
+ Vec_IntPushOrderWithMask( Vec_PtrEntry(p->vRules, iFan0), (pNode->Id << 16) | iFan1 );
+ Vec_IntPushOrderWithMask( Vec_PtrEntry(p->vRules, iFan1), (pNode->Id << 16) | iFan0 );
+ }
+ }
+ return pNode->Id;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Creates a new node.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+int Amap_LibCreateMux( Amap_Lib_t * p, int iFan0, int iFan1, int iFan2 )
+{
+ Amap_Nod_t * pNode;
+ pNode = Amap_LibCreateObj( p );
+ pNode->Type = AMAP_OBJ_MUX;
+ pNode->nSuppSize = p->pNodes[Amap_Lit2Var(iFan0)].nSuppSize + p->pNodes[Amap_Lit2Var(iFan1)].nSuppSize + p->pNodes[Amap_Lit2Var(iFan2)].nSuppSize;
+ pNode->iFan0 = iFan0;
+ pNode->iFan1 = iFan1;
+ pNode->iFan2 = iFan2;
+if ( p->fVerbose )
+printf( "Creating node %5d %c : iFan0 = %5d%c iFan1 = %5d%c iFan2 = %5d%c\n",
+pNode->Id, 'm',
+Amap_Lit2Var(iFan0), (Amap_LitIsCompl(iFan0)?'-':'+'),
+Amap_Lit2Var(iFan1), (Amap_LitIsCompl(iFan1)?'-':'+'),
+Amap_Lit2Var(iFan2), (Amap_LitIsCompl(iFan2)?'-':'+') );
+
+ Vec_IntPush( p->vRules3, iFan0 );
+ Vec_IntPush( p->vRules3, iFan1 );
+ Vec_IntPush( p->vRules3, iFan2 );
+ Vec_IntPush( p->vRules3, pNode->Id );
+ return pNode->Id;
+}
+
+/**Function*************************************************************
+
+ Synopsis [Allocates triangular lookup table.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+int ** Amap_LibLookupTableAlloc( Vec_Ptr_t * vVec, int fVerbose )
+{
+ Vec_Int_t * vOne;
+ int ** pRes, * pBuffer;
+ int i, k, nTotal, nSize, nEntries, Value;
+ // count the total size
+ nEntries = nSize = Vec_PtrSize( vVec );
+ Vec_PtrForEachEntry( vVec, vOne, i )
+ nEntries += Vec_IntSize(vOne);
+ pBuffer = ALLOC( int, nSize * sizeof(void *) + nEntries );
+ pRes = (int **)pBuffer;
+ pRes[0] = pBuffer + nSize * sizeof(void *);
+ nTotal = 0;
+ Vec_PtrForEachEntry( vVec, vOne, i )
+ {
+ pRes[i] = pRes[0] + nTotal;
+ nTotal += Vec_IntSize(vOne) + 1;
+ if ( fVerbose )
+ printf( "%d : ", i );
+ Vec_IntForEachEntry( vOne, Value, k )
+ {
+ pRes[i][k] = Value;
+ if ( fVerbose )
+ printf( "%d(%d) ", Value&0xffff, Value>>16 );
+ }
+ if ( fVerbose )
+ printf( "\n" );
+ pRes[i][k] = 0;
+ }
+ assert( nTotal == nEntries );
+ return pRes;
+}
+
+////////////////////////////////////////////////////////////////////////
+/// END OF FILE ///
+////////////////////////////////////////////////////////////////////////
+
+
diff --git a/src/map/amap/module.make b/src/map/amap/module.make
new file mode 100644
index 00000000..d5edadb9
--- /dev/null
+++ b/src/map/amap/module.make
@@ -0,0 +1,12 @@
+SRC += src/map/amap/amapCore.c \
+ src/map/amap/amapGraph.c \
+ src/map/amap/amapLib.c \
+ src/map/amap/amapMan.c \
+ src/map/amap/amapMatch.c \
+ src/map/amap/amapMerge.c \
+ src/map/amap/amapOutput.c \
+ src/map/amap/amapParse.c \
+ src/map/amap/amapPerm.c \
+ src/map/amap/amapRead.c \
+ src/map/amap/amapRule.c \
+ src/map/amap/amapUniq.c