summaryrefslogtreecommitdiffstats
path: root/src/sat/bmc
diff options
context:
space:
mode:
authorAlan Mishchenko <alanmi@berkeley.edu>2015-09-04 15:40:53 -0700
committerAlan Mishchenko <alanmi@berkeley.edu>2015-09-04 15:40:53 -0700
commitb11344b454f2df60de7b303b1b102ec62a96d01d (patch)
tree990c16f3f5feb6f145b17b5852749f3997105b05 /src/sat/bmc
parenta207f6c07117fc577076f924984a0cbad1c0b0b0 (diff)
downloadabc-b11344b454f2df60de7b303b1b102ec62a96d01d.tar.gz
abc-b11344b454f2df60de7b303b1b102ec62a96d01d.tar.bz2
abc-b11344b454f2df60de7b303b1b102ec62a96d01d.zip
Experiments with SAT-based collapsing.
Diffstat (limited to 'src/sat/bmc')
-rw-r--r--src/sat/bmc/bmcClp.c172
1 files changed, 122 insertions, 50 deletions
diff --git a/src/sat/bmc/bmcClp.c b/src/sat/bmc/bmcClp.c
index d0948176..15cac4bb 100644
--- a/src/sat/bmc/bmcClp.c
+++ b/src/sat/bmc/bmcClp.c
@@ -38,7 +38,7 @@ extern Cnf_Dat_t * Mf_ManGenerateCnf( Gia_Man_t * pGia, int nLutSize, int fCnfOb
/**Function*************************************************************
- Synopsis []
+ Synopsis [Performs one round of removing literals.]
Description []
@@ -47,7 +47,7 @@ extern Cnf_Dat_t * Mf_ManGenerateCnf( Gia_Man_t * pGia, int nLutSize, int fCnfOb
SeeAlso []
***********************************************************************/
-int Bmc_CollapseExpandCanon( sat_solver * pSat, Vec_Int_t * vLits, Vec_Int_t * vNums, Vec_Int_t * vTemp, int nBTLimit )
+int Bmc_CollapseExpandRound( sat_solver * pSat, sat_solver * pSatOn, Vec_Int_t * vLits, Vec_Int_t * vNums, Vec_Int_t * vTemp, int nBTLimit, int fCanon )
{
int k, n, iLit, status;
// try removing one literal at a time
@@ -56,6 +56,26 @@ int Bmc_CollapseExpandCanon( sat_solver * pSat, Vec_Int_t * vLits, Vec_Int_t * v
int Save = Vec_IntEntry( vLits, k );
if ( Save == -1 )
continue;
+ // check if this literal when expanded overlaps with the on-set
+ if ( pSatOn )
+ {
+ // it is ok to skip the first round if the literal is positive
+ if ( fCanon && !Abc_LitIsCompl(Save) )
+ continue;
+ // put into new array
+ Vec_IntClear( vTemp );
+ Vec_IntForEachEntry( vLits, iLit, n )
+ if ( iLit != -1 )
+ Vec_IntPush( vTemp, Abc_LitNotCond(iLit, k==n) );
+ // check against onset
+ status = sat_solver_solve( pSatOn, Vec_IntArray(vTemp), Vec_IntLimit(vTemp), nBTLimit, 0, 0, 0 );
+ if ( status == l_Undef )
+ return -1;
+ //printf( "%d", status == l_True );
+ if ( status == l_False )
+ continue;
+ // otherwise keep trying to remove it
+ }
Vec_IntWriteEntry( vLits, k, -1 );
// put into new array
Vec_IntClear( vTemp );
@@ -69,6 +89,8 @@ int Bmc_CollapseExpandCanon( sat_solver * pSat, Vec_Int_t * vLits, Vec_Int_t * v
if ( status == l_True )
Vec_IntWriteEntry( vLits, k, Save );
}
+// if ( pSatOn )
+// printf( "\n" );
// put into new array
Vec_IntClear( vNums );
Vec_IntForEachEntry( vLits, iLit, n )
@@ -79,7 +101,7 @@ int Bmc_CollapseExpandCanon( sat_solver * pSat, Vec_Int_t * vLits, Vec_Int_t * v
/**Function*************************************************************
- Synopsis []
+ Synopsis [Expends minterm into a cube.]
Description []
@@ -88,44 +110,73 @@ int Bmc_CollapseExpandCanon( sat_solver * pSat, Vec_Int_t * vLits, Vec_Int_t * v
SeeAlso []
***********************************************************************/
-int Bmc_CollapseExpand( sat_solver * pSat, Vec_Int_t * vLits, Vec_Int_t * vNums, Vec_Int_t * vTemp, int nBTLimit )
+int Bmc_CollapseExpand( sat_solver * pSat, sat_solver * pSatOn, Vec_Int_t * vLits, Vec_Int_t * vNums, Vec_Int_t * vTemp, int nBTLimit, int fCanon )
{
- int i, k, iLit, status, nFinal, * pFinal;
- // check against offset
- status = sat_solver_solve( pSat, Vec_IntArray(vLits), Vec_IntLimit(vLits), nBTLimit, 0, 0, 0 );
- if ( status == l_Undef )
- return -1;
- assert( status == l_False );
- // get subset of literals
- nFinal = sat_solver_final( pSat, &pFinal );
-/*
- // collect literals
- Vec_IntClear( vNums );
- for ( i = 0; i < nFinal; i++ )
+ // perform one quick reduction if it is non-canonical
+ if ( !fCanon )
{
- iLit = Vec_IntFind( vLits, Abc_LitNot(pFinal[i]) );
- assert( iLit >= 0 );
- Vec_IntPush( vNums, iLit );
- }
-*/
- // mark unused literals
- Vec_IntForEachEntry( vLits, iLit, i )
- {
- for ( k = 0; k < nFinal; k++ )
- if ( iLit == Abc_LitNot(pFinal[k]) )
- break;
- if ( k == nFinal )
- Vec_IntWriteEntry( vLits, i, -1 );
+ int i, k, iLit, status, nFinal, * pFinal;
+ // check against offset
+ status = sat_solver_solve( pSat, Vec_IntArray(vLits), Vec_IntLimit(vLits), nBTLimit, 0, 0, 0 );
+ if ( status == l_Undef )
+ return -1;
+ assert( status == l_False );
+ // get subset of literals
+ nFinal = sat_solver_final( pSat, &pFinal );
+ // mark unused literals
+ Vec_IntForEachEntry( vLits, iLit, i )
+ {
+ for ( k = 0; k < nFinal; k++ )
+ if ( iLit == Abc_LitNot(pFinal[k]) )
+ break;
+ if ( k == nFinal )
+ Vec_IntWriteEntry( vLits, i, -1 );
+ }
}
- Bmc_CollapseExpandCanon( pSat, vLits, vNums, vTemp, nBTLimit );
+ Bmc_CollapseExpandRound( pSat, pSatOn, vLits, vNums, vTemp, nBTLimit, fCanon );
+ Bmc_CollapseExpandRound( pSat, NULL, vLits, vNums, vTemp, nBTLimit, fCanon );
+ return 0;
+}
-/*
- int i;
- Vec_IntClear( vNums );
+/**Function*************************************************************
+
+ Synopsis [Returns SAT solver in the 'sat' state with the given assignment.]
+
+ Description []
+
+ SideEffects []
+
+ SeeAlso []
+
+***********************************************************************/
+int Bmc_ComputeCanonical( sat_solver * pSat, Vec_Int_t * vLits, Vec_Int_t * vTemp, int nBTLimit )
+{
+ int i, k, iLit, status = l_Undef;
for ( i = 0; i < Vec_IntSize(vLits); i++ )
- Vec_IntPush( vNums, i );
-*/
- return 0;
+ {
+ // copy the first i+1 literals
+ Vec_IntClear( vTemp );
+ Vec_IntForEachEntryStop( vLits, iLit, k, i+1 )
+ Vec_IntPush( vTemp, iLit );
+ // check if it satisfies the on-set
+ status = sat_solver_solve( pSat, Vec_IntArray(vTemp), Vec_IntLimit(vTemp), nBTLimit, 0, 0, 0 );
+ if ( status == l_Undef )
+ return l_Undef;
+ if ( status == l_True )
+ continue;
+ // if it is UNSAT, try the opposite literal
+ iLit = Vec_IntEntry( vLits, i );
+ // if literal is positive, there is no way to flip it
+ if ( !Abc_LitIsCompl(iLit) )
+ return l_False;
+ Vec_IntWriteEntry( vLits, i, Abc_LitNot(iLit) );
+ Vec_IntForEachEntryStart( vLits, iLit, k, i+1 )
+ Vec_IntWriteEntry( vLits, k, Abc_LitNot(Abc_LitRegular(iLit)) );
+ // recheck
+ i--;
+ }
+ assert( status == l_True );
+ return status;
}
/**Function*************************************************************
@@ -145,6 +196,7 @@ Vec_Str_t * Bmc_CollapseOneInt( Gia_Man_t * p, int nCubeLim, int nBTLimit, int f
int nVars = Gia_ManCiNum(p);
Vec_Int_t * vVars = Vec_IntAlloc( nVars );
Vec_Int_t * vLits = Vec_IntAlloc( nVars );
+ Vec_Int_t * vLitsC= Vec_IntAlloc( nVars );
Vec_Int_t * vNums = Vec_IntAlloc( nVars );
Vec_Int_t * vCube = Vec_IntAlloc( nVars );
Vec_Str_t * vSop = Vec_StrAlloc( 100 );
@@ -152,9 +204,10 @@ Vec_Str_t * Bmc_CollapseOneInt( Gia_Man_t * p, int nCubeLim, int nBTLimit, int f
// create SAT solver
Cnf_Dat_t * pCnf = Mf_ManGenerateCnf( p, 8, 0, 0, 0 );
- sat_solver * pSat[2] = {
+ sat_solver * pSat[3] = {
(sat_solver *)Cnf_DataWriteIntoSolver(pCnf, 1, 0),
- (sat_solver *)Cnf_DataWriteIntoSolver(pCnf, 1, 0)
+ (sat_solver *)Cnf_DataWriteIntoSolver(pCnf, 1, 0),
+ fCanon ? (sat_solver *)Cnf_DataWriteIntoSolver(pCnf, 1, 0) : NULL
};
// collect CI variables
@@ -162,14 +215,18 @@ Vec_Str_t * Bmc_CollapseOneInt( Gia_Man_t * p, int nCubeLim, int nBTLimit, int f
for ( n = 0; n < nVars; n++ )
Vec_IntPush( vVars, iCiVarBeg + n );
+ // start with all negative literals
+ Vec_IntForEachEntry( vVars, iVar, n )
+ Vec_IntPush( vLitsC, Abc_Var2Lit(iVar, 1) );
+
// check that on-set/off-set is sat
- for ( n = 0; n < 2; n++ )
+ for ( n = 0; n < 2 + fCanon; n++ )
{
- iLit = Abc_Var2Lit( iOut + 1, n ); // n=0 => F=1 n=1 => F=0
+ iLit = Abc_Var2Lit( iOut + 1, n&1 ); // n=0 => F=1 n=1 => F=0
status = sat_solver_addclause( pSat[n], &iLit, &iLit + 1 );
if ( status == 0 )
{
- Vec_StrPrintStr( vSop, (n ^ fCompl) ? " 1\n" : " 0\n" );
+ Vec_StrPrintStr( vSop, ((n&1) ^ fCompl) ? " 1\n" : " 0\n" );
Vec_StrPush( vSop, '\0' );
goto cleanup; // const0/1
}
@@ -181,7 +238,7 @@ Vec_Str_t * Bmc_CollapseOneInt( Gia_Man_t * p, int nCubeLim, int nBTLimit, int f
}
if ( status == l_False )
{
- Vec_StrPrintStr( vSop, (n ^ fCompl) ? " 1\n" : " 0\n" );
+ Vec_StrPrintStr( vSop, ((n&1) ^ fCompl) ? " 1\n" : " 0\n" );
Vec_StrPush( vSop, '\0' );
goto cleanup; // const0/1
}
@@ -189,13 +246,19 @@ Vec_Str_t * Bmc_CollapseOneInt( Gia_Man_t * p, int nCubeLim, int nBTLimit, int f
Vec_StrPush( vSop, '\0' );
// prepare on-set for solving
- if ( fCanon )
- sat_solver_prepare_enum( pSat[0], Vec_IntArray(vVars), Vec_IntSize(vVars) );
+// if ( fCanon )
+// sat_solver_prepare_enum( pSat[0], Vec_IntArray(vVars), Vec_IntSize(vVars) );
Count = 0;
while ( 1 )
{
// get the assignment
- status = sat_solver_solve( pSat[0], NULL, NULL, 0, 0, 0, 0 );
+ if ( fCanon )
+ status = Bmc_ComputeCanonical( pSat[0], vLitsC, vCube, nBTLimit );
+ else
+ {
+ sat_solver_clean_polarity( pSat[0], Vec_IntArray(vVars), Vec_IntSize(vVars) );
+ status = sat_solver_solve( pSat[0], NULL, NULL, 0, 0, 0, 0 );
+ }
if ( status == l_Undef )
{
Vec_StrFreeP( &vSop );
@@ -212,8 +275,13 @@ Vec_Str_t * Bmc_CollapseOneInt( Gia_Man_t * p, int nCubeLim, int nBTLimit, int f
}
// collect values
Vec_IntClear( vLits );
+ Vec_IntClear( vLitsC );
Vec_IntForEachEntry( vVars, iVar, n )
- Vec_IntPush( vLits, Abc_Var2Lit(iVar, !sat_solver_var_value(pSat[0], iVar)) );
+ {
+ iLit = Abc_Var2Lit(iVar, !sat_solver_var_value(pSat[0], iVar));
+ Vec_IntPush( vLits, iLit );
+ Vec_IntPush( vLitsC, iLit );
+ }
// print minterm
if ( fPrintMinterm )
{
@@ -223,10 +291,7 @@ Vec_Str_t * Bmc_CollapseOneInt( Gia_Man_t * p, int nCubeLim, int nBTLimit, int f
printf( "\n" );
}
// expand the values
- if ( fCanon )
- status = Bmc_CollapseExpandCanon( pSat[1], vLits, vNums, vCube, nBTLimit );
- else
- status = Bmc_CollapseExpand( pSat[1], vLits, vNums, vCube, nBTLimit );
+ status = Bmc_CollapseExpand( pSat[1], fCanon ? pSat[2] : pSat[0], vLits, vNums, vCube, nBTLimit, fCanon );
if ( status < 0 )
{
Vec_StrFreeP( &vSop );
@@ -254,15 +319,22 @@ Vec_Str_t * Bmc_CollapseOneInt( Gia_Man_t * p, int nCubeLim, int nBTLimit, int f
status = sat_solver_addclause( pSat[0], Vec_IntArray(vCube), Vec_IntLimit(vCube) );
if ( status == 0 )
break;
+ // add cube
+ if ( fCanon )
+ status = sat_solver_addclause( pSat[2], Vec_IntArray(vCube), Vec_IntLimit(vCube) );
+ assert( status == 1 );
}
//printf( "Finished enumerating %d assignments.\n", Count );
cleanup:
Vec_IntFree( vVars );
Vec_IntFree( vLits );
+ Vec_IntFree( vLitsC );
Vec_IntFree( vNums );
Vec_IntFree( vCube );
sat_solver_delete( pSat[0] );
sat_solver_delete( pSat[1] );
+ if ( fCanon )
+ sat_solver_delete( pSat[2] );
Cnf_DataFree( pCnf );
return vSop;
}