diff options
Diffstat (limited to 'src/sat/bsat/satProof.c')
-rw-r--r-- | src/sat/bsat/satProof.c | 476 |
1 files changed, 476 insertions, 0 deletions
diff --git a/src/sat/bsat/satProof.c b/src/sat/bsat/satProof.c new file mode 100644 index 00000000..b3cf2b6c --- /dev/null +++ b/src/sat/bsat/satProof.c @@ -0,0 +1,476 @@ +/**CFile**************************************************************** + + FileName [satProof.c] + + SystemName [ABC: Logic synthesis and verification system.] + + PackageName [SAT solver.] + + Synopsis [Proof manipulation procedures.] + + Author [Alan Mishchenko] + + Affiliation [UC Berkeley] + + Date [Ver. 1.0. Started - June 20, 2005.] + + Revision [$Id: satProof.c,v 1.4 2005/09/16 22:55:03 casem Exp $] + +***********************************************************************/ + +#include "satSolver.h" +#include "vec.h" +#include "aig.h" + +ABC_NAMESPACE_IMPL_START + + +/* + Proof is represented as a vector of integers. + The first entry is -1. + The clause is represented as an offset in this array. + One clause's entry is <size><label><ant1><ant2>...<antN> + Label is initialized to 0. + Root clauses are 1-based. They are marked by prepending bit 1; +*/ + +//////////////////////////////////////////////////////////////////////// +/// DECLARATIONS /// +//////////////////////////////////////////////////////////////////////// + +typedef struct Sat_Set_t_ Sat_Set_t; +struct Sat_Set_t_ +{ + int nEnts; + int Label; + int pEnts[0]; +}; + +static inline int Sat_SetCheck( Vec_Int_t * p, Sat_Set_t * pNode ) { return (int *)pNode > Vec_IntArray(p) && (int *)pNode < Vec_IntLimit(p); } +static inline int Sat_SetId( Vec_Int_t * p, Sat_Set_t * pNode ) { return (int *)pNode - Vec_IntArray(p); } +static inline Sat_Set_t * Sat_SetFromId( Vec_Int_t * p, int i ) { return (Sat_Set_t *)(Vec_IntArray(p) + i); } +static inline int Sat_SetSize( Sat_Set_t * pNode ) { return pNode->nEnts + 2; } + +#define Sat_PoolForEachSet( p, pNode, i ) \ + for ( i = 1; (i < Vec_IntSize(p)) && ((pNode) = Sat_SetFromId(p, Vec_IntEntry(p,i))); i += Sat_SetSize(pNode) ) +#define Sat_SetForEachSet( p, pSet, pNode, i ) \ + for ( i = 0; (i < pSet->nEnts) && (((pNode) = ((pSet->pEnts[i] & 1) ? NULL : Sat_SetFromId(p, pSet->pEnts[i]))), 1); i++ ) +#define Sat_VecForEachSet( pVec, p, pNode, i ) \ + for ( i = 0; (i < Vec_IntSize(pVec)) && ((pNode) = Sat_SetFromId(p, Vec_IntEntry(pVec,i))); i++ ) + +//////////////////////////////////////////////////////////////////////// +/// FUNCTION DEFINITIONS /// +//////////////////////////////////////////////////////////////////////// + +/**Function************************************************************* + + Synopsis [Recursively visits useful proof nodes.] + + Description [] + + SideEffects [] + + SeeAlso [] + +***********************************************************************/ +int Sat_ProofReduceOne( Vec_Int_t * p, Sat_Set_t * pNode, int * pnSize, Vec_Int_t * vStack ) +{ + Sat_Set_t * pNext; + int i, NodeId; + if ( pNode->Label ) + return pNode->Label; + // start with node + pNode->Label = 1; + Vec_IntPush( vStack, Sat_SetId(p, pNode) ); + // perform DFS search + while ( Vec_IntSize(vStack) ) + { + NodeId = Vec_IntPop( vStack ); + if ( NodeId & 1 ) // extrated second time + { + pNode = Sat_SetFromId( p, NodeId ^ 1 ); + pNode->Label = *pnSize; + *pnSize += Sat_SetSize(pNode); + // update fanins + Sat_SetForEachSet( p, pNode, pNext, i ) + if ( pNext ) + pNode->pEnts[i] = pNext->Label; + continue; + } + // extracted first time + // add second time + Vec_IntPush( vStack, NodeId ^ 1 ); + // add its anticedents + pNode = Sat_SetFromId( p, NodeId ); + Sat_SetForEachSet( p, pNode, pNext, i ) + if ( pNext && !pNext->Label ) + { + pNext->Label = 1; + Vec_IntPush( vStack, Sat_SetId(p, pNode) ); // add first time + } + } + return pNode->Label; +} + +/**Function************************************************************* + + Synopsis [Recursively visits useful proof nodes.] + + Description [] + + SideEffects [] + + SeeAlso [] + +***********************************************************************/ +/* +int Sat_ProofReduce_rec( Vec_Int_t * p, Sat_Set_t * pNode, int * pnSize ) +{ + int * pBeg; + assert( Sat_SetCheck(p, pNode) ); + if ( pNode->Label ) + return pNode->Label; + for ( pBeg = pNode->pEnts; pBeg < pNode->pEnts + pNode->nEnts; pBeg++ ) + if ( !(*pBeg & 1) ) + *pBeg = Sat_ProofReduce_rec( p, Sat_SetFromId(p, *pBeg), pnSize ); + pNode->Label = *pnSize; + *pnSize += Sat_SetSize(pNode); + return pNode->Label; +} +*/ + +/**Function************************************************************* + + Synopsis [Reduces the proof to contain only roots and their children.] + + Description [The result is updated proof and updated roots.] + + SideEffects [] + + SeeAlso [] + +***********************************************************************/ +void Sat_ProofReduce( Vec_Int_t * p, Vec_Int_t * vRoots ) +{ + int i, nSize = 1; + int * pBeg, * pEnd, * pNew; + Vec_Int_t * vStack; + Sat_Set_t * pNode; + // mark used nodes + vStack = Vec_IntAlloc( 1000 ); + Sat_VecForEachSet( vRoots, p, pNode, i ) + vRoots->pArray[i] = Sat_ProofReduceOne( p, pNode, &nSize, vStack ); + Vec_IntFree( vStack ); + // compact proof + pNew = Vec_IntArray(p) + 1; + Sat_PoolForEachSet( p, pNode, i ) + { + if ( !pNode->Label ) + continue; + assert( pNew - Vec_IntArray(p) == pNode->Label ); + pNode->Label = 0; + pBeg = (int *)pNode; + pEnd = pBeg + Sat_SetSize(pNode); + while ( pBeg < pEnd ) + *pNew++ = *pBeg++; + } + // report the result + printf( "The proof was reduced from %d to %d (by %6.2f %%)\n", + Vec_IntSize(p), nSize, 100.0 * (Vec_IntSize(p) - nSize) / Vec_IntSize(p) ); + assert( pNew - Vec_IntArray(p) == nSize ); + Vec_IntShrink( p, nSize ); +} + +/**Function************************************************************* + + Synopsis [Recursively visits useful proof nodes.] + + Description [] + + SideEffects [] + + SeeAlso [] + +***********************************************************************/ +void Sat_ProofLabel( Vec_Int_t * p, Sat_Set_t * pNode, Vec_Int_t * vUsed, Vec_Int_t * vStack ) +{ + Sat_Set_t * pNext; + int i, NodeId; + if ( pNode->Label ) + return; + // start with node + pNode->Label = 1; + Vec_IntPush( vStack, Sat_SetId(p, pNode) ); + // perform DFS search + while ( Vec_IntSize(vStack) ) + { + NodeId = Vec_IntPop( vStack ); + if ( NodeId & 1 ) // extrated second time + { + Vec_IntPush( vUsed, NodeId ^ 1 ); + continue; + } + // extracted first time + // add second time + Vec_IntPush( vStack, NodeId ^ 1 ); + // add its anticedents + pNode = Sat_SetFromId( p, NodeId ); + Sat_SetForEachSet( p, pNode, pNext, i ) + if ( pNext && !pNext->Label ) + { + pNext->Label = 1; + Vec_IntPush( vStack, Sat_SetId(p, pNode) ); // add first time + } + } +} + +/**Function************************************************************* + + Synopsis [Computes UNSAT core.] + + Description [The result is the array of root clause indexes.] + + SideEffects [] + + SeeAlso [] + +***********************************************************************/ +Vec_Int_t * Sat_ProofCore( Vec_Int_t * p, int nRoots, int * pBeg, int * pEnd ) +{ + unsigned * pSeen; + Vec_Int_t * vCore, * vUsed, * vStack; + Sat_Set_t * pNode; + int i; + // collect visited clauses + vUsed = Vec_IntAlloc( 1000 ); + vStack = Vec_IntAlloc( 1000 ); + while ( pBeg < pEnd ) + Sat_ProofLabel( p, Sat_SetFromId(p, *pBeg++), vUsed, vStack ); + Vec_IntFree( vStack ); + // find the core + vCore = Vec_IntAlloc( 1000 ); + pSeen = ABC_CALLOC( unsigned, Aig_BitWordNum(nRoots) ); + Sat_VecForEachSet( vUsed, p, pNode, i ) + { + pNode->Label = 0; + for ( pBeg = pNode->pEnts; pBeg < pNode->pEnts + pNode->nEnts; pBeg++ ) + if ( (*pBeg & 1) && !Aig_InfoHasBit(pBeg, *pBeg>>1) ) + { + Aig_InfoSetBit( pBeg, *pBeg>>1 ); + Vec_IntPush( vCore, (*pBeg>>1)-1 ); + } + } + Vec_IntFree( vUsed ); + ABC_FREE( pSeen ); + return vCore; +} + + +/**Function************************************************************* + + Synopsis [Performs one resultion step.] + + Description [Returns ID of the resolvent if success, and -1 if failure.] + + SideEffects [] + + SeeAlso [] + +***********************************************************************/ +Sat_Set_t * Sat_ProofResolve( Vec_Int_t * p, Sat_Set_t * c1, Sat_Set_t * c2 ) +{ + int i, k, Id, Var = -1, Count = 0; + // find resolution variable + for ( i = 0; i < c1->nEnts; i++ ) + for ( k = 0; k < c2->nEnts; k++ ) + if ( (c1->pEnts[i] ^ c2->pEnts[k]) == 1 ) + { + Var = (c1->pEnts[i] >> 1); + Count++; + } + if ( Count == 0 ) + { + printf( "Cannot find resolution variable\n" ); + return NULL; + } + if ( Count > 1 ) + { + printf( "Found more than 1 resolution variables\n" ); + return NULL; + } + // perform resolution + Id = Vec_IntSize( p ); + Vec_IntPush( p, 0 ); // placeholder + Vec_IntPush( p, 0 ); + for ( i = 0; i < c1->nEnts; i++ ) + { + if ( (c1->pEnts[i] >> 1) == Var ) + continue; + for ( k = Id + 2; k < Vec_IntSize(p); k++ ) + if ( p->pArray[k] == c1->pEnts[i] ) + break; + if ( k == Vec_IntSize(p) ) + Vec_IntPush( p, c1->pEnts[i] ); + } + for ( i = 0; i < c2->nEnts; i++ ) + { + if ( (c2->pEnts[i] >> 1) == Var ) + continue; + for ( k = Id + 2; k < Vec_IntSize(p); k++ ) + if ( p->pArray[k] == c2->pEnts[i] ) + break; + if ( k == Vec_IntSize(p) ) + Vec_IntPush( p, c2->pEnts[i] ); + } + Vec_IntWriteEntry( p, Id, Vec_IntSize(p) - Id - 2 ); + return Sat_SetFromId( p, Id ); +} + +/**Function************************************************************* + + Synopsis [Checks the proof for consitency.] + + Description [] + + SideEffects [] + + SeeAlso [] + +***********************************************************************/ +Sat_Set_t * Sat_ProofCheckGetOne( Vec_Int_t * vClauses, Vec_Int_t * vProof, Vec_Int_t * vResolves, int iAnt ) +{ + Sat_Set_t * pAnt; + if ( iAnt & 1 ) + return Sat_SetFromId( vClauses, iAnt >> 1 ); + pAnt = Sat_SetFromId( vProof, iAnt ); + return Sat_SetFromId( vResolves, pAnt->Label ); +} + +/**Function************************************************************* + + Synopsis [Checks the proof for consitency.] + + Description [] + + SideEffects [] + + SeeAlso [] + +***********************************************************************/ +void Sat_ProofCheck( Vec_Int_t * vClauses, Vec_Int_t * vProof, int iRoot ) +{ + Vec_Int_t * vOrigs, * vStack, * vUsed, * vResolves; + Sat_Set_t * pSet, * pSet0, * pSet1; + int i, k; + // collect original clauses + vOrigs = Vec_IntAlloc( 1000 ); + Vec_IntPush( vOrigs, -1 ); + Sat_PoolForEachSet( vClauses, pSet, i ) + Vec_IntPush( vOrigs, Sat_SetId(vClauses, pSet) ); + // collect visited clauses + vUsed = Vec_IntAlloc( 1000 ); + vStack = Vec_IntAlloc( 1000 ); + Sat_ProofLabel( vProof, Sat_SetFromId(vProof, iRoot), vUsed, vStack ); + Vec_IntFree( vStack ); + // perform resolution steps + vResolves = Vec_IntAlloc( 1000 ); + Vec_IntPush( vResolves, -1 ); + Sat_VecForEachSet( vUsed, vProof, pSet, i ) + { + pSet0 = Sat_ProofCheckGetOne( vOrigs, vProof, vResolves, pSet->pEnts[0] ); + for ( k = 1; k < pSet->nEnts; k++ ) + { + pSet1 = Sat_ProofCheckGetOne( vOrigs, vProof, vResolves, pSet->pEnts[k] ); + pSet0 = Sat_ProofResolve( vResolves, pSet0, pSet1 ); + } + pSet->Label = Sat_SetId( vResolves, pSet0 ); + } + // clean the proof + Sat_VecForEachSet( vUsed, vProof, pSet, i ) + pSet->Label = 0; + // compare the final clause + if ( pSet0->nEnts > 0 ) + printf( "Cound not derive the empty clause\n" ); + Vec_IntFree( vResolves ); + Vec_IntFree( vUsed ); + Vec_IntFree( vOrigs ); +} + + +/**Function************************************************************* + + Synopsis [Creates variable map.] + + Description [1=A, 2=B, neg=global] + + SideEffects [] + + SeeAlso [] + +***********************************************************************/ +void Sat_ProofVarMapCheck( Vec_Int_t * vClauses, Vec_Int_t * vVarMap ) +{ + Sat_Set_t * pSet; + int i, k, fSeeA, fSeeB; + // make sure all clauses are either A or B + Sat_PoolForEachSet( vClauses, pSet, i ) + { + fSeeA = fSeeB = 0; + for ( k = 0; k < pSet->nEnts; k++ ) + { + fSeeA += (Vec_IntEntry(vVarMap, pSet->pEnts[k]) == 1); + fSeeB += (Vec_IntEntry(vVarMap, pSet->pEnts[k]) == 2); + } + if ( fSeeA && fSeeB ) + printf( "VarMap error!\n" ); + } +} + +/**Function************************************************************* + + Synopsis [Computes interpolant of the proof.] + + Description [Assuming that res vars are recorded, too...] + + SideEffects [] + + SeeAlso [] + +***********************************************************************/ +Aig_Man_t * Sat_ProofInterpolant( Vec_Int_t * vProof, int iRoot, Vec_Int_t * vClauses, Vec_Int_t * vVarMap ) +{ + Vec_Int_t * vOrigs, * vUsed, * vStack; + Aig_Man_t * pAig; + Sat_Set_t * pSet; + int i; + + Sat_ProofVarMapCheck( vClauses, vVarMap ); + + // collect original clauses + vOrigs = Vec_IntAlloc( 1000 ); + Vec_IntPush( vOrigs, -1 ); + Sat_PoolForEachSet( vClauses, pSet, i ) + Vec_IntPush( vOrigs, Sat_SetId(vClauses, pSet) ); + + // collect visited clauses + vUsed = Vec_IntAlloc( 1000 ); + vStack = Vec_IntAlloc( 1000 ); + Sat_ProofLabel( vProof, Sat_SetFromId(vProof, iRoot), vUsed, vStack ); + Vec_IntFree( vStack ); + + // start the AIG + pAig = Aig_ManStart( 10000 ); + pAig->pName = Aig_UtilStrsav( "interpol" ); + for ( i = 0; i < Vec_IntSize(vVarMap); i++ ) + if ( Vec_IntEntry(vVarMap, i) < 0 ) + Aig_ObjCreatePi( pAig ); + + return pAig; +} + +//////////////////////////////////////////////////////////////////////// +/// END OF FILE /// +//////////////////////////////////////////////////////////////////////// + +ABC_NAMESPACE_IMPL_END + |