mirror of https://github.com/YosysHQ/abc.git
481 lines
16 KiB
C
481 lines
16 KiB
C
/**CFile****************************************************************
|
|
|
|
FileName [sswPairs.c]
|
|
|
|
SystemName [ABC: Logic synthesis and verification system.]
|
|
|
|
PackageName [Inductive prover with constraints.]
|
|
|
|
Synopsis [Calls to the SAT solver.]
|
|
|
|
Author [Alan Mishchenko]
|
|
|
|
Affiliation [UC Berkeley]
|
|
|
|
Date [Ver. 1.0. Started - September 1, 2008.]
|
|
|
|
Revision [$Id: sswPairs.c,v 1.00 2008/09/01 00:00:00 alanmi Exp $]
|
|
|
|
***********************************************************************/
|
|
|
|
#include "sswInt.h"
|
|
|
|
ABC_NAMESPACE_IMPL_START
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
/// DECLARATIONS ///
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
/// FUNCTION DEFINITIONS ///
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
/**Function*************************************************************
|
|
|
|
Synopsis [Reports the status of the miter.]
|
|
|
|
Description []
|
|
|
|
SideEffects []
|
|
|
|
SeeAlso []
|
|
|
|
***********************************************************************/
|
|
int Ssw_MiterStatus( Aig_Man_t * p, int fVerbose )
|
|
{
|
|
Aig_Obj_t * pObj, * pChild;
|
|
int i, CountConst0 = 0, CountNonConst0 = 0, CountUndecided = 0;
|
|
// if ( p->pData )
|
|
// return 0;
|
|
Saig_ManForEachPo( p, pObj, i )
|
|
{
|
|
pChild = Aig_ObjChild0(pObj);
|
|
// check if the output is constant 0
|
|
if ( pChild == Aig_ManConst0(p) )
|
|
{
|
|
CountConst0++;
|
|
continue;
|
|
}
|
|
// check if the output is constant 1
|
|
if ( pChild == Aig_ManConst1(p) )
|
|
{
|
|
CountNonConst0++;
|
|
continue;
|
|
}
|
|
// check if the output is a primary input
|
|
if ( p->nRegs == 0 && Aig_ObjIsCi(Aig_Regular(pChild)) )
|
|
{
|
|
CountNonConst0++;
|
|
continue;
|
|
}
|
|
// check if the output can be not constant 0
|
|
if ( Aig_Regular(pChild)->fPhase != (unsigned)Aig_IsComplement(pChild) )
|
|
{
|
|
CountNonConst0++;
|
|
continue;
|
|
}
|
|
CountUndecided++;
|
|
}
|
|
|
|
if ( fVerbose )
|
|
{
|
|
Abc_Print( 1, "Miter has %d outputs. ", Saig_ManPoNum(p) );
|
|
Abc_Print( 1, "Const0 = %d. ", CountConst0 );
|
|
Abc_Print( 1, "NonConst0 = %d. ", CountNonConst0 );
|
|
Abc_Print( 1, "Undecided = %d. ", CountUndecided );
|
|
Abc_Print( 1, "\n" );
|
|
}
|
|
|
|
if ( CountNonConst0 )
|
|
return 0;
|
|
if ( CountUndecided )
|
|
return -1;
|
|
return 1;
|
|
}
|
|
|
|
/**Function*************************************************************
|
|
|
|
Synopsis [Transfer equivalent pairs to the miter.]
|
|
|
|
Description []
|
|
|
|
SideEffects []
|
|
|
|
SeeAlso []
|
|
|
|
***********************************************************************/
|
|
Vec_Int_t * Ssw_TransferSignalPairs( Aig_Man_t * pMiter, Aig_Man_t * pAig1, Aig_Man_t * pAig2, Vec_Int_t * vIds1, Vec_Int_t * vIds2 )
|
|
{
|
|
Vec_Int_t * vIds;
|
|
Aig_Obj_t * pObj1, * pObj2;
|
|
Aig_Obj_t * pObj1m, * pObj2m;
|
|
int i;
|
|
vIds = Vec_IntAlloc( 2 * Vec_IntSize(vIds1) );
|
|
for ( i = 0; i < Vec_IntSize(vIds1); i++ )
|
|
{
|
|
pObj1 = Aig_ManObj( pAig1, Vec_IntEntry(vIds1, i) );
|
|
pObj2 = Aig_ManObj( pAig2, Vec_IntEntry(vIds2, i) );
|
|
pObj1m = Aig_Regular((Aig_Obj_t *)pObj1->pData);
|
|
pObj2m = Aig_Regular((Aig_Obj_t *)pObj2->pData);
|
|
assert( pObj1m && pObj2m );
|
|
if ( pObj1m == pObj2m )
|
|
continue;
|
|
if ( pObj1m->Id < pObj2m->Id )
|
|
{
|
|
Vec_IntPush( vIds, pObj1m->Id );
|
|
Vec_IntPush( vIds, pObj2m->Id );
|
|
}
|
|
else
|
|
{
|
|
Vec_IntPush( vIds, pObj2m->Id );
|
|
Vec_IntPush( vIds, pObj1m->Id );
|
|
}
|
|
}
|
|
return vIds;
|
|
}
|
|
|
|
/**Function*************************************************************
|
|
|
|
Synopsis [Transform pairs into class representation.]
|
|
|
|
Description []
|
|
|
|
SideEffects []
|
|
|
|
SeeAlso []
|
|
|
|
***********************************************************************/
|
|
Vec_Int_t ** Ssw_TransformPairsIntoTempClasses( Vec_Int_t * vPairs, int nObjNumMax )
|
|
{
|
|
Vec_Int_t ** pvClasses; // vector of classes
|
|
int * pReprs; // mapping nodes into their representatives
|
|
int Entry, idObj, idRepr, idReprObj, idReprRepr, i;
|
|
// allocate data-structures
|
|
pvClasses = ABC_CALLOC( Vec_Int_t *, nObjNumMax );
|
|
pReprs = ABC_ALLOC( int, nObjNumMax );
|
|
for ( i = 0; i < nObjNumMax; i++ )
|
|
pReprs[i] = -1;
|
|
// consider pairs
|
|
for ( i = 0; i < Vec_IntSize(vPairs); i += 2 )
|
|
{
|
|
// get both objects
|
|
idRepr = Vec_IntEntry( vPairs, i );
|
|
idObj = Vec_IntEntry( vPairs, i+1 );
|
|
assert( idObj > 0 );
|
|
assert( (pReprs[idRepr] == -1) || (pvClasses[pReprs[idRepr]] != NULL) );
|
|
assert( (pReprs[idObj] == -1) || (pvClasses[pReprs[idObj] ] != NULL) );
|
|
// get representatives of both objects
|
|
idReprRepr = pReprs[idRepr];
|
|
idReprObj = pReprs[idObj];
|
|
// check different situations
|
|
if ( idReprRepr == -1 && idReprObj == -1 )
|
|
{ // they do not have classes
|
|
// create a class
|
|
pvClasses[idRepr] = Vec_IntAlloc( 4 );
|
|
Vec_IntPush( pvClasses[idRepr], idRepr );
|
|
Vec_IntPush( pvClasses[idRepr], idObj );
|
|
pReprs[ idRepr ] = idRepr;
|
|
pReprs[ idObj ] = idRepr;
|
|
}
|
|
else if ( idReprRepr >= 0 && idReprObj == -1 )
|
|
{ // representative has a class
|
|
// add iObj to the same class
|
|
Vec_IntPushUniqueOrder( pvClasses[idReprRepr], idObj );
|
|
pReprs[ idObj ] = idReprRepr;
|
|
}
|
|
else if ( idReprRepr == -1 && idReprObj >= 0 )
|
|
{ // object has a class
|
|
assert( idReprObj != idRepr );
|
|
if ( idReprObj < idRepr )
|
|
{ // add idRepr to the same class
|
|
Vec_IntPushUniqueOrder( pvClasses[idReprObj], idRepr );
|
|
pReprs[ idRepr ] = idReprObj;
|
|
}
|
|
else // if ( idReprObj > idRepr )
|
|
{ // make idRepr new representative
|
|
Vec_IntPushFirst( pvClasses[idReprObj], idRepr );
|
|
pvClasses[idRepr] = pvClasses[idReprObj];
|
|
pvClasses[idReprObj] = NULL;
|
|
// set correct representatives of each node
|
|
Vec_IntForEachEntry( pvClasses[idRepr], Entry, i )
|
|
pReprs[ Entry ] = idRepr;
|
|
}
|
|
}
|
|
else // if ( idReprRepr >= 0 && idReprObj >= 0 )
|
|
{ // both have classes
|
|
if ( idReprRepr == idReprObj )
|
|
{ // the classes are the same
|
|
// nothing to do
|
|
}
|
|
else
|
|
{ // the classes are different
|
|
// find the repr of the new class
|
|
if ( idReprRepr < idReprObj )
|
|
{
|
|
Vec_IntForEachEntry( pvClasses[idReprObj], Entry, i )
|
|
{
|
|
Vec_IntPushUniqueOrder( pvClasses[idReprRepr], Entry );
|
|
pReprs[ Entry ] = idReprRepr;
|
|
}
|
|
Vec_IntFree( pvClasses[idReprObj] );
|
|
pvClasses[idReprObj] = NULL;
|
|
}
|
|
else // if ( idReprRepr > idReprObj )
|
|
{
|
|
Vec_IntForEachEntry( pvClasses[idReprRepr], Entry, i )
|
|
{
|
|
Vec_IntPushUniqueOrder( pvClasses[idReprObj], Entry );
|
|
pReprs[ Entry ] = idReprObj;
|
|
}
|
|
Vec_IntFree( pvClasses[idReprRepr] );
|
|
pvClasses[idReprRepr] = NULL;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
ABC_FREE( pReprs );
|
|
return pvClasses;
|
|
}
|
|
|
|
/**Function*************************************************************
|
|
|
|
Synopsis []
|
|
|
|
Description []
|
|
|
|
SideEffects []
|
|
|
|
SeeAlso []
|
|
|
|
***********************************************************************/
|
|
void Ssw_FreeTempClasses( Vec_Int_t ** pvClasses, int nObjNumMax )
|
|
{
|
|
int i;
|
|
for ( i = 0; i < nObjNumMax; i++ )
|
|
if ( pvClasses[i] )
|
|
Vec_IntFree( pvClasses[i] );
|
|
ABC_FREE( pvClasses );
|
|
}
|
|
|
|
/**Function*************************************************************
|
|
|
|
Synopsis [Performs signal correspondence for the miter of two AIGs with node pairs defined.]
|
|
|
|
Description []
|
|
|
|
SideEffects []
|
|
|
|
SeeAlso []
|
|
|
|
***********************************************************************/
|
|
Aig_Man_t * Ssw_SignalCorrespondenceWithPairs( Aig_Man_t * pAig1, Aig_Man_t * pAig2, Vec_Int_t * vIds1, Vec_Int_t * vIds2, Ssw_Pars_t * pPars )
|
|
{
|
|
Ssw_Man_t * p;
|
|
Aig_Man_t * pAigNew, * pMiter;
|
|
Ssw_Pars_t Pars;
|
|
Vec_Int_t * vPairs;
|
|
Vec_Int_t ** pvClasses;
|
|
assert( Vec_IntSize(vIds1) == Vec_IntSize(vIds2) );
|
|
// create sequential miter
|
|
pMiter = Saig_ManCreateMiter( pAig1, pAig2, 0 );
|
|
Aig_ManCleanup( pMiter );
|
|
// transfer information to the miter
|
|
vPairs = Ssw_TransferSignalPairs( pMiter, pAig1, pAig2, vIds1, vIds2 );
|
|
// create representation of the classes
|
|
pvClasses = Ssw_TransformPairsIntoTempClasses( vPairs, Aig_ManObjNumMax(pMiter) );
|
|
Vec_IntFree( vPairs );
|
|
// if parameters are not given, create them
|
|
if ( pPars == NULL )
|
|
Ssw_ManSetDefaultParams( pPars = &Pars );
|
|
// start the induction manager
|
|
p = Ssw_ManCreate( pMiter, pPars );
|
|
// create equivalence classes using these IDs
|
|
p->ppClasses = Ssw_ClassesPreparePairs( pMiter, pvClasses );
|
|
p->pSml = Ssw_SmlStart( pMiter, 0, p->nFrames + p->pPars->nFramesAddSim, 1 );
|
|
Ssw_ClassesSetData( p->ppClasses, p->pSml, (unsigned(*)(void *,Aig_Obj_t *))Ssw_SmlObjHashWord, (int(*)(void *,Aig_Obj_t *))Ssw_SmlObjIsConstWord, (int(*)(void *,Aig_Obj_t *,Aig_Obj_t *))Ssw_SmlObjsAreEqualWord );
|
|
// perform refinement of classes
|
|
pAigNew = Ssw_SignalCorrespondenceRefine( p );
|
|
// cleanup
|
|
Ssw_FreeTempClasses( pvClasses, Aig_ManObjNumMax(pMiter) );
|
|
Ssw_ManStop( p );
|
|
Aig_ManStop( pMiter );
|
|
return pAigNew;
|
|
}
|
|
|
|
/**Function*************************************************************
|
|
|
|
Synopsis [Runs inductive SEC for the miter of two AIGs with node pairs defined.]
|
|
|
|
Description []
|
|
|
|
SideEffects []
|
|
|
|
SeeAlso []
|
|
|
|
***********************************************************************/
|
|
Aig_Man_t * Ssw_SignalCorrespondeceTestPairs( Aig_Man_t * pAig )
|
|
{
|
|
Aig_Man_t * pAigNew, * pAigRes;
|
|
Ssw_Pars_t Pars, * pPars = &Pars;
|
|
Vec_Int_t * vIds1, * vIds2;
|
|
Aig_Obj_t * pObj, * pRepr;
|
|
int RetValue, i;
|
|
abctime clk = Abc_Clock();
|
|
Ssw_ManSetDefaultParams( pPars );
|
|
pPars->fVerbose = 1;
|
|
pAigNew = Ssw_SignalCorrespondence( pAig, pPars );
|
|
// record pairs of equivalent nodes
|
|
vIds1 = Vec_IntAlloc( Aig_ManObjNumMax(pAig) );
|
|
vIds2 = Vec_IntAlloc( Aig_ManObjNumMax(pAig) );
|
|
Aig_ManForEachObj( pAig, pObj, i )
|
|
{
|
|
pRepr = Aig_Regular((Aig_Obj_t *)pObj->pData);
|
|
if ( pRepr == NULL )
|
|
continue;
|
|
if ( Aig_ManObj(pAigNew, pRepr->Id) == NULL )
|
|
continue;
|
|
/*
|
|
if ( Aig_ObjIsNode(pObj) )
|
|
Abc_Print( 1, "n " );
|
|
else if ( Saig_ObjIsPi(pAig, pObj) )
|
|
Abc_Print( 1, "pi " );
|
|
else if ( Saig_ObjIsLo(pAig, pObj) )
|
|
Abc_Print( 1, "lo " );
|
|
*/
|
|
Vec_IntPush( vIds1, Aig_ObjId(pObj) );
|
|
Vec_IntPush( vIds2, Aig_ObjId(pRepr) );
|
|
}
|
|
Abc_Print( 1, "Recorded %d pairs (before: %d after: %d).\n", Vec_IntSize(vIds1), Aig_ManObjNumMax(pAig), Aig_ManObjNumMax(pAigNew) );
|
|
// try the new AIGs
|
|
pAigRes = Ssw_SignalCorrespondenceWithPairs( pAig, pAigNew, vIds1, vIds2, pPars );
|
|
Vec_IntFree( vIds1 );
|
|
Vec_IntFree( vIds2 );
|
|
// report the results
|
|
RetValue = Ssw_MiterStatus( pAigRes, 1 );
|
|
if ( RetValue == 1 )
|
|
Abc_Print( 1, "Verification successful. " );
|
|
else if ( RetValue == 0 )
|
|
Abc_Print( 1, "Verification failed with the counter-example. " );
|
|
else
|
|
Abc_Print( 1, "Verification UNDECIDED. Remaining registers %d (total %d). ",
|
|
Aig_ManRegNum(pAigRes), Aig_ManRegNum(pAig) + Aig_ManRegNum(pAigNew) );
|
|
ABC_PRT( "Time", Abc_Clock() - clk );
|
|
// cleanup
|
|
Aig_ManStop( pAigNew );
|
|
return pAigRes;
|
|
}
|
|
|
|
/**Function*************************************************************
|
|
|
|
Synopsis [Runs inductive SEC for the miter of two AIGs with node pairs defined.]
|
|
|
|
Description []
|
|
|
|
SideEffects []
|
|
|
|
SeeAlso []
|
|
|
|
***********************************************************************/
|
|
int Ssw_SecWithPairs( Aig_Man_t * pAig1, Aig_Man_t * pAig2, Vec_Int_t * vIds1, Vec_Int_t * vIds2, Ssw_Pars_t * pPars )
|
|
{
|
|
Aig_Man_t * pAigRes;
|
|
int RetValue;
|
|
abctime clk = Abc_Clock();
|
|
assert( vIds1 != NULL && vIds2 != NULL );
|
|
// try the new AIGs
|
|
Abc_Print( 1, "Performing specialized verification with node pairs.\n" );
|
|
pAigRes = Ssw_SignalCorrespondenceWithPairs( pAig1, pAig2, vIds1, vIds2, pPars );
|
|
// report the results
|
|
RetValue = Ssw_MiterStatus( pAigRes, 1 );
|
|
if ( RetValue == 1 )
|
|
Abc_Print( 1, "Verification successful. " );
|
|
else if ( RetValue == 0 )
|
|
Abc_Print( 1, "Verification failed with a counter-example. " );
|
|
else
|
|
Abc_Print( 1, "Verification UNDECIDED. The number of remaining regs = %d (total = %d). ",
|
|
Aig_ManRegNum(pAigRes), Aig_ManRegNum(pAig1) + Aig_ManRegNum(pAig2) );
|
|
ABC_PRT( "Time", Abc_Clock() - clk );
|
|
// cleanup
|
|
Aig_ManStop( pAigRes );
|
|
return RetValue;
|
|
}
|
|
|
|
/**Function*************************************************************
|
|
|
|
Synopsis [Runs inductive SEC for the miter of two AIGs.]
|
|
|
|
Description []
|
|
|
|
SideEffects []
|
|
|
|
SeeAlso []
|
|
|
|
***********************************************************************/
|
|
int Ssw_SecGeneral( Aig_Man_t * pAig1, Aig_Man_t * pAig2, Ssw_Pars_t * pPars )
|
|
{
|
|
Aig_Man_t * pAigRes, * pMiter;
|
|
int RetValue;
|
|
abctime clk = Abc_Clock();
|
|
// try the new AIGs
|
|
Abc_Print( 1, "Performing general verification without node pairs.\n" );
|
|
pMiter = Saig_ManCreateMiter( pAig1, pAig2, 0 );
|
|
Aig_ManCleanup( pMiter );
|
|
pAigRes = Ssw_SignalCorrespondence( pMiter, pPars );
|
|
Aig_ManStop( pMiter );
|
|
// report the results
|
|
RetValue = Ssw_MiterStatus( pAigRes, 1 );
|
|
if ( RetValue == 1 )
|
|
Abc_Print( 1, "Verification successful. " );
|
|
else if ( RetValue == 0 )
|
|
Abc_Print( 1, "Verification failed with a counter-example. " );
|
|
else
|
|
Abc_Print( 1, "Verification UNDECIDED. The number of remaining regs = %d (total = %d). ",
|
|
Aig_ManRegNum(pAigRes), Aig_ManRegNum(pAig1) + Aig_ManRegNum(pAig2) );
|
|
ABC_PRT( "Time", Abc_Clock() - clk );
|
|
// cleanup
|
|
Aig_ManStop( pAigRes );
|
|
return RetValue;
|
|
}
|
|
|
|
/**Function*************************************************************
|
|
|
|
Synopsis [Runs inductive SEC for the miter of two AIGs.]
|
|
|
|
Description []
|
|
|
|
SideEffects []
|
|
|
|
SeeAlso []
|
|
|
|
***********************************************************************/
|
|
int Ssw_SecGeneralMiter( Aig_Man_t * pMiter, Ssw_Pars_t * pPars )
|
|
{
|
|
Aig_Man_t * pAigRes;
|
|
int RetValue;
|
|
abctime clk = Abc_Clock();
|
|
// try the new AIGs
|
|
// Abc_Print( 1, "Performing general verification without node pairs.\n" );
|
|
pAigRes = Ssw_SignalCorrespondence( pMiter, pPars );
|
|
// report the results
|
|
RetValue = Ssw_MiterStatus( pAigRes, 1 );
|
|
if ( RetValue == 1 )
|
|
Abc_Print( 1, "Verification successful. " );
|
|
else if ( RetValue == 0 )
|
|
Abc_Print( 1, "Verification failed with a counter-example. " );
|
|
else
|
|
Abc_Print( 1, "Verification UNDECIDED. The number of remaining regs = %d (total = %d). ",
|
|
Aig_ManRegNum(pAigRes), Aig_ManRegNum(pMiter) );
|
|
ABC_PRT( "Time", Abc_Clock() - clk );
|
|
// cleanup
|
|
Aig_ManStop( pAigRes );
|
|
return RetValue;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
/// END OF FILE ///
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
ABC_NAMESPACE_IMPL_END
|