summaryrefslogtreecommitdiffstats
path: root/src/map/if
diff options
context:
space:
mode:
authorAlan Mishchenko <alanmi@berkeley.edu>2011-10-03 13:34:17 +0700
committerAlan Mishchenko <alanmi@berkeley.edu>2011-10-03 13:34:17 +0700
commit8c302870f44d718261962d24e034ee19a8b6add8 (patch)
treeed60253334a5c9e0c095a8208bb5ab0678ab1c47 /src/map/if
parent0f9dacb7bec32bc19afb068b4bcf53a781ab2a0e (diff)
downloadabc-8c302870f44d718261962d24e034ee19a8b6add8.tar.gz
abc-8c302870f44d718261962d24e034ee19a8b6add8.tar.bz2
abc-8c302870f44d718261962d24e034ee19a8b6add8.zip
Changes to the matching procedure.
Diffstat (limited to 'src/map/if')
-rw-r--r--src/map/if/ifDec16.c216
-rw-r--r--src/map/if/ifMan.c8
2 files changed, 214 insertions, 10 deletions
diff --git a/src/map/if/ifDec16.c b/src/map/if/ifDec16.c
index a6d0edee..39881875 100644
--- a/src/map/if/ifDec16.c
+++ b/src/map/if/ifDec16.c
@@ -92,6 +92,15 @@ static inline int If_CluWordNum( int nVars )
{
return nVars <= 6 ? 1 : 1 << (nVars-6);
}
+static inline int If_CluCountOnes( word t )
+{
+ t = (t & 0x5555555555555555) + ((t>> 1) & 0x5555555555555555);
+ t = (t & 0x3333333333333333) + ((t>> 2) & 0x3333333333333333);
+ t = (t & 0x0F0F0F0F0F0F0F0F) + ((t>> 4) & 0x0F0F0F0F0F0F0F0F);
+ t = (t & 0x00FF00FF00FF00FF) + ((t>> 8) & 0x00FF00FF00FF00FF);
+ t = (t & 0x0000FFFF0000FFFF) + ((t>>16) & 0x0000FFFF0000FFFF);
+ return (t & 0x00000000FFFFFFFF) + (t>>32);
+}
int If_CluHashKey( word * pTruth, int nWords, int Size )
{
static unsigned BigPrimes[8] = {12582917, 25165843, 50331653, 100663319, 201326611, 402653189, 805306457, 1610612741};
@@ -130,6 +139,22 @@ If_Grp_t * If_CluHashLookup( If_Man_t * p, word * pTruth )
((If_Hte_t **)p->pHashTable)[HashKey] = pEntry;
return &pEntry->Group;
}
+void If_CluHashPrintStats( If_Man_t * p )
+{
+ If_Hte_t * pEntry;
+ int i, Counter;
+ for ( i = 0; i < p->nTableSize; i++ )
+ {
+ Counter = 0;
+ for ( pEntry = ((If_Hte_t **)p->pHashTable)[i]; pEntry; pEntry = pEntry->pNext )
+ Counter++;
+ if ( Counter == 0 )
+ continue;
+ if ( Counter < 10 )
+ continue;
+ printf( "%d=%d ", i, Counter );
+ }
+}
// variable permutation for large functions
static inline void If_CluClear( word * pIn, int nVars )
@@ -194,6 +219,18 @@ static inline word If_CluAdjust( word t, int nVars )
t |= t << (1<<nVars++);
return t;
}
+static inline void If_CluAdjustBig( word * pF, int nVarsCur, int nVarsMax )
+{
+ int v, nWords;
+ if ( nVarsCur == nVarsMax )
+ return;
+ assert( nVarsCur < nVarsMax );
+ for ( v = Abc_MaxInt( nVarsCur, 6 ); v < nVarsMax; v++ )
+ {
+ nWords = If_CluWordNum( v );
+ If_CluCopy( pF + nWords, pF, v );
+ }
+}
static inline void If_CluSwapAdjacent( word * pOut, word * pIn, int iVar, int nVars )
{
int i, k, nWords = If_CluWordNum( nVars );
@@ -231,6 +268,147 @@ static inline void If_CluSwapAdjacent( word * pOut, word * pIn, int iVar, int nV
}
}
+void If_CluChangePhase( word * pF, int nVars, int iVar )
+{
+ int nWords = If_CluWordNum( nVars );
+ assert( iVar < nVars );
+ if ( iVar < 6 )
+ {
+ int i, Shift = (1 << iVar);
+ for ( i = 0; i < nWords; i++ )
+ pF[i] = ((pF[i] & ~Truth6[iVar]) << Shift) | ((pF[i] & Truth6[iVar]) >> Shift);
+ }
+ else
+ {
+ word Temp;
+ int i, k, Step = (1 << (iVar - 6));
+ for ( k = 0; k < nWords; k += 2*Step )
+ {
+ for ( i = 0; i < Step; i++ )
+ {
+ Temp = pF[i];
+ pF[i] = pF[Step+i];
+ pF[Step+i] = Temp;
+ }
+ pF += 2*Step;
+ }
+ }
+}
+void If_CluCountOnesInCofs( word * pTruth, int nVars, int * pStore )
+{
+ int nWords = If_CluWordNum( nVars );
+ int i, k, nOnes = 0, Limit = Abc_MinInt( nVars, 6 );
+ memset( pStore, 0, sizeof(int) * 2 * nVars );
+ // compute positive cofactors
+ for ( k = 0; k < nWords; k++ )
+ for ( i = 0; i < Limit; i++ )
+ pStore[2*i+1] += If_CluCountOnes( pTruth[k] & Truth6[i] );
+ if ( nVars > 6 )
+ for ( k = 0; k < nWords; k++ )
+ for ( i = 6; i < nVars; i++ )
+ if ( k & (1 << (i-6)) )
+ pStore[2*i+1] += If_CluCountOnes( pTruth[k] );
+ // compute negative cofactors
+ for ( k = 0; k < nWords; k++ )
+ nOnes += If_CluCountOnes( pTruth[k] );
+ for ( i = 0; i < nVars; i++ )
+ pStore[2*i] = nOnes - pStore[2*i+1];
+}
+unsigned If_CluSemiCanonicize( word * pTruth, int nVars, int * pCanonPerm )
+{
+ word pFunc[CLU_WRD_MAX], * pIn = pTruth, * pOut = pFunc, * pTemp;
+ int pStore[CLU_VAR_MAX*2];
+ unsigned uCanonPhase = 0;
+ int i, Temp, fChange, Counter = 0;
+//Kit_DsdPrintFromTruth( (unsigned*)pTruth, nVars ); printf( "\n" );
+
+ // collect signatures
+ If_CluCountOnesInCofs( pTruth, nVars, pStore );
+ // canonicize phase
+ for ( i = 0; i < nVars; i++ )
+ {
+ if ( pStore[2*i+0] <= pStore[2*i+1] )
+ continue;
+ uCanonPhase |= (1 << i);
+ Temp = pStore[2*i+0];
+ pStore[2*i+0] = pStore[2*i+1];
+ pStore[2*i+1] = Temp;
+ If_CluChangePhase( pIn, nVars, i );
+ }
+ // compute permutation
+ for ( i = 0; i < nVars; i++ )
+ pCanonPerm[i] = i;
+ do {
+ fChange = 0;
+ for ( i = 0; i < nVars-1; i++ )
+ {
+ if ( pStore[2*i] <= pStore[2*(i+1)] )
+ continue;
+ Counter++;
+ fChange = 1;
+
+ Temp = pCanonPerm[i];
+ pCanonPerm[i] = pCanonPerm[i+1];
+ pCanonPerm[i+1] = Temp;
+
+ Temp = pStore[2*i];
+ pStore[2*i] = pStore[2*(i+1)];
+ pStore[2*(i+1)] = Temp;
+
+ Temp = pStore[2*i+1];
+ pStore[2*i+1] = pStore[2*(i+1)+1];
+ pStore[2*(i+1)+1] = Temp;
+
+ If_CluSwapAdjacent( pOut, pIn, i, nVars );
+ pTemp = pIn; pIn = pOut; pOut = pTemp;
+ }
+ } while ( fChange );
+ // swap if it was moved an odd number of times
+ if ( Counter & 1 )
+ If_CluCopy( pOut, pIn, nVars );
+ return uCanonPhase;
+}
+void If_CluSemiCanonicizeVerify( word * pTruth, word * pTruth0, int nVars, int * pCanonPerm, unsigned uCanonPhase )
+{
+ word pFunc[CLU_WRD_MAX], pGunc[CLU_WRD_MAX], * pIn = pTruth, * pOut = pFunc, * pTemp;
+ int i, Temp, fChange, Counter = 0;
+ If_CluCopy( pGunc, pTruth, nVars );
+ // undo permutation
+ do {
+ fChange = 0;
+ for ( i = 0; i < nVars-1; i++ )
+ {
+ if ( pCanonPerm[i] < pCanonPerm[i+1] )
+ continue;
+
+ Counter++;
+ fChange = 1;
+
+ Temp = pCanonPerm[i];
+ pCanonPerm[i] = pCanonPerm[i+1];
+ pCanonPerm[i+1] = Temp;
+
+ If_CluSwapAdjacent( pOut, pIn, i, nVars );
+ pTemp = pIn; pIn = pOut; pOut = pTemp;
+ }
+ } while ( fChange );
+ if ( Counter & 1 )
+ If_CluCopy( pOut, pIn, nVars );
+ // undo phase
+ for ( i = 0; i < nVars; i++ )
+ if ( (uCanonPhase >> i) & 1 )
+ If_CluChangePhase( pTruth, nVars, i );
+ // compare
+ if ( !If_CluEqual(pTruth0, pTruth, nVars) )
+ {
+ Kit_DsdPrintFromTruth( (unsigned*)pTruth0, nVars ); printf( "\n" );
+ Kit_DsdPrintFromTruth( (unsigned*)pGunc, nVars ); printf( "\n" );
+ Kit_DsdPrintFromTruth( (unsigned*)pTruth, nVars ); printf( "\n" );
+ printf( "SemiCanonical verification FAILED!\n" );
+ }
+}
+
+
void If_CluPrintGroup( If_Grp_t * g )
{
int i;
@@ -898,15 +1076,28 @@ static inline int If_CluSupport( word * t, int nVars )
}
// returns the best group found
-If_Grp_t If_CluCheck( If_Man_t * p, word * pTruth, int nVars, int nLutLeaf, int nLutRoot )
+If_Grp_t If_CluCheck( If_Man_t * p, word * pTruth0, int nVars, int nLutLeaf, int nLutRoot )
{
If_Grp_t G1 = {0}, R = {0}, * pHashed = NULL;
- word Truth, pF[CLU_WRD_MAX];//, pG[CLU_WRD_MAX];
- int V2P[CLU_VAR_MAX+2], P2V[CLU_VAR_MAX+2];
- int i, nSupp;
+ word Truth, pTruth[CLU_WRD_MAX], pF[CLU_WRD_MAX];//, pG[CLU_WRD_MAX];
+ int V2P[CLU_VAR_MAX+2], P2V[CLU_VAR_MAX+2], pCanonPerm[CLU_VAR_MAX];
+ int i, nSupp, uCanonPhase;
assert( nVars <= CLU_VAR_MAX );
assert( nVars <= nLutLeaf + nLutRoot - 1 );
+ // canonicize truth table
+ If_CluCopy( pTruth, pTruth0, p->pPars->nLutSize );
+
+ if ( 0 )
+ {
+ uCanonPhase = If_CluSemiCanonicize( pTruth, nVars, pCanonPerm );
+ If_CluAdjustBig( pTruth, nVars, p->pPars->nLutSize );
+ }
+
+// If_CluSemiCanonicizeVerify( pTruth, pTruth0, nVars, pCanonPerm, uCanonPhase );
+// If_CluCopy( pTruth, pTruth0, p->pPars->nLutSize );
+
+
/*
{
int pCanonPerm[32];
@@ -929,6 +1120,8 @@ If_Grp_t If_CluCheck( If_Man_t * p, word * pTruth, int nVars, int nLutLeaf, int
if ( !nSupp || !If_CluSuppIsMinBase(nSupp) )
return G1;
+
+
// check hash table
pHashed = If_CluHashLookup( p, pTruth );
if ( pHashed && pHashed->nVars != CLU_UNUSED )
@@ -1161,14 +1354,25 @@ void If_CluTest()
// word t = 0x0100200000000001;
// word t2[4] = { 0x0000800080008000, 0x8000000000008000, 0x8000000080008000, 0x0000000080008000 };
// word t = 0x07770FFF07770FFF;
-// word t = 0x002000D000D00020;
+ word t = 0x002000D000D00020;
// word t = 0x000F000E000F000F;
- word t = 0xF7FFF7F7F7F7F7F7;
+// word t = 0xF7FFF7F7F7F7F7F7;
+ word s = t;
int nVars = 6;
int nLutLeaf = 4;
int nLutRoot = 4;
If_Grp_t G;
+/*
+ int pCanonPerm[CLU_VAR_MAX];
+ int uCanonPhase;
+
+ Kit_DsdPrintFromTruth( (unsigned*)&s, nVars ); printf( "\n" );
+ uCanonPhase = If_CluSemiCanonicize( &s, nVars, pCanonPerm );
+ Kit_DsdPrintFromTruth( (unsigned*)&s, nVars ); printf( "\n" );
+
+ If_CluSemiCanonicizeVerify( &s, &t, nVars, pCanonPerm, uCanonPhase );
+*/
return;
If_CutPerformCheck07( NULL, (unsigned *)&t, 6, 6, NULL );
diff --git a/src/map/if/ifMan.c b/src/map/if/ifMan.c
index 8ea579c4..c6aaa9c4 100644
--- a/src/map/if/ifMan.c
+++ b/src/map/if/ifMan.c
@@ -133,8 +133,8 @@ void If_ManStop( If_Man_t * p )
int i;
for ( i = 0; i <= 16; i++ )
if ( p->nCutsUseless[i] )
- Abc_Print( 1, "Useless cuts %2d = %7d (out of %7d) (%6.2f %%)\n", i, p->nCutsUseless[i], p->nCutsCount[i], 100.0*p->nCutsUseless[i]/(p->nCutsCount[i]+1) );
- Abc_Print( 1, "Useless cuts all = %7d (out of %7d) (%6.2f %%)\n", p->nCutsUselessAll, p->nCutsCountAll, 100.0*p->nCutsUselessAll/(p->nCutsCountAll+1) );
+ Abc_Print( 1, "Useless cuts %2d = %9d (out of %9d) (%6.2f %%)\n", i, p->nCutsUseless[i], p->nCutsCount[i], 100.0*p->nCutsUseless[i]/(p->nCutsCount[i]+1) );
+ Abc_Print( 1, "Useless cuts all = %9d (out of %9d) (%6.2f %%)\n", p->nCutsUselessAll, p->nCutsCountAll, 100.0*p->nCutsUselessAll/(p->nCutsCountAll+1) );
}
// Abc_PrintTime( 1, "Truth", p->timeTruth );
// Abc_Print( 1, "Small support = %d.\n", p->nSmallSupp );
@@ -161,8 +161,8 @@ void If_ManStop( If_Man_t * p )
if ( p->vSwitching )
Vec_IntFree( p->vSwitching );
// hash table
-// if ( p->nTableEntries )
-// printf( "Entries = %d. Size = %d.\n", p->nTableEntries, p->nTableSize );
+ if ( p->nTableEntries )
+ printf( "Entries = %d. Size = %d.\n", p->nTableEntries, p->nTableSize );
ABC_FREE( p->pHashTable );
if ( p->pMemEntries )
Mem_FixedStop( p->pMemEntries, 0 );