abc/src/misc/util/utilBridge.c

475 lines
13 KiB
C
Raw Normal View History

2012-03-02 06:20:18 +01:00
/**CFile****************************************************************
FileName [utilBridge.c]
SystemName [ABC: Logic synthesis and verification system.]
PackageName []
Synopsis []
Author [Alan Mishchenko]
Affiliation [UC Berkeley]
Date [Ver. 1.0. Started - June 20, 2005.]
Revision [$Id: utilBridge.c,v 1.00 2005/06/20 00:00:00 alanmi Exp $]
***********************************************************************/
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <assert.h>
#include "abc_global.h"
#include "src/aig/gia/gia.h"
#include "src/misc/vec/vec.h"
ABC_NAMESPACE_IMPL_START
////////////////////////////////////////////////////////////////////////
/// DECLARATIONS ///
////////////////////////////////////////////////////////////////////////
extern void Gia_WriteAigerEncodeStr( Vec_Str_t * vStr, unsigned x );
extern unsigned Gia_ReadAigerDecode( unsigned char ** ppPos );
2012-03-02 07:36:34 +01:00
extern void Gia_WriteAigerEncodeStr( Vec_Str_t * vStr, unsigned x );
2012-03-02 06:20:18 +01:00
2012-03-02 07:36:34 +01:00
// this variable determines where the output goes
int in_bridge_mode = 0;
2012-03-02 06:20:18 +01:00
////////////////////////////////////////////////////////////////////////
/// FUNCTION DEFINITIONS ///
////////////////////////////////////////////////////////////////////////
/**Function*************************************************************
Synopsis []
Description []
SideEffects []
SeeAlso []
***********************************************************************/
2012-03-02 07:36:34 +01:00
Vec_Str_t * Gia_ManToBridgeVec( Gia_Man_t * p )
2012-03-02 06:20:18 +01:00
{
Vec_Str_t * vBuffer;
Gia_Obj_t * pObj;
int nNodes = 0, i, uLit, uLit0, uLit1;
// set the node numbers to be used in the output file
Gia_ManConst0(p)->Value = nNodes++;
Gia_ManForEachCi( p, pObj, i )
pObj->Value = nNodes++;
Gia_ManForEachAnd( p, pObj, i )
pObj->Value = nNodes++;
// write the header "M I L O A" where M = I + L + A
vBuffer = Vec_StrAlloc( 3*Gia_ManObjNum(p) );
Vec_StrPrintStr( vBuffer, "aig " );
Vec_StrPrintNum( vBuffer, Gia_ManCandNum(p) );
Vec_StrPrintStr( vBuffer, " " );
Vec_StrPrintNum( vBuffer, Gia_ManPiNum(p) );
Vec_StrPrintStr( vBuffer, " " );
Vec_StrPrintNum( vBuffer, Gia_ManRegNum(p) );
Vec_StrPrintStr( vBuffer, " " );
Vec_StrPrintNum( vBuffer, Gia_ManPoNum(p) );
Vec_StrPrintStr( vBuffer, " " );
Vec_StrPrintNum( vBuffer, Gia_ManAndNum(p) );
Vec_StrPrintStr( vBuffer, "\n" );
// write latch drivers
Gia_ManForEachRi( p, pObj, i )
{
uLit = Abc_Var2Lit( Gia_ObjValue(Gia_ObjFanin0(pObj)), Gia_ObjFaninC0(pObj) );
Vec_StrPrintNum( vBuffer, uLit );
Vec_StrPrintStr( vBuffer, "\n" );
}
// write PO drivers
Gia_ManForEachPo( p, pObj, i )
{
uLit = Abc_Var2Lit( Gia_ObjValue(Gia_ObjFanin0(pObj)), Gia_ObjFaninC0(pObj) );
Vec_StrPrintNum( vBuffer, uLit );
Vec_StrPrintStr( vBuffer, "\n" );
}
// write the nodes into the buffer
Gia_ManForEachAnd( p, pObj, i )
{
uLit = Abc_Var2Lit( Gia_ObjValue(pObj), 0 );
uLit0 = Abc_Var2Lit( Gia_ObjValue(Gia_ObjFanin0(pObj)), Gia_ObjFaninC0(pObj) );
uLit1 = Abc_Var2Lit( Gia_ObjValue(Gia_ObjFanin1(pObj)), Gia_ObjFaninC1(pObj) );
assert( uLit0 != uLit1 );
if ( uLit0 > uLit1 )
{
int Temp = uLit0;
uLit0 = uLit1;
uLit1 = Temp;
}
Gia_WriteAigerEncodeStr( vBuffer, uLit - uLit1 );
Gia_WriteAigerEncodeStr( vBuffer, uLit1 - uLit0 );
}
Vec_StrPrintStr( vBuffer, "c" );
return vBuffer;
}
/**Function*************************************************************
Synopsis []
Description []
SideEffects []
SeeAlso []
***********************************************************************/
2012-03-02 07:36:34 +01:00
void Gia_CreateHeader( FILE * pFile, int Type, int Size, unsigned char * pBuffer )
{
fprintf( pFile, "%.6d", Type );
fprintf( pFile, " " );
fprintf( pFile, "%.16d", Size );
fprintf( pFile, " " );
fwrite( pBuffer, Size, 1, pFile );
fflush( pFile );
}
/**Function*************************************************************
Synopsis []
Description []
SideEffects []
SeeAlso []
***********************************************************************/
#define BRIDGE_TEXT_MESSAGE 999996
#define BRIDGE_RESULTS 101
#define BRIDGE_ABS_NETLIST 107
#define BRIDGE_BAD_ABS 105
/**Function*************************************************************
Synopsis []
Description []
SideEffects []
SeeAlso []
***********************************************************************/
int Gia_ManToBridgeText( FILE * pFile, int Size, unsigned char * pBuffer )
{
Gia_CreateHeader( pFile, BRIDGE_TEXT_MESSAGE, Size, pBuffer );
return 1;
}
int Gia_ManToBridgeAbsNetlist( FILE * pFile, Gia_Man_t * p )
2012-03-02 06:20:18 +01:00
{
Vec_Str_t * vBuffer;
2012-03-02 07:36:34 +01:00
vBuffer = Gia_ManToBridgeVec( p );
Gia_CreateHeader( pFile, BRIDGE_ABS_NETLIST, Vec_StrSize(vBuffer), (unsigned char *)Vec_StrArray(vBuffer) );
2012-03-02 06:20:18 +01:00
Vec_StrFree( vBuffer );
2012-03-02 07:36:34 +01:00
return 1;
}
int Gia_ManToBridgeBadAbs( FILE * pFile )
{
Gia_CreateHeader( pFile, BRIDGE_BAD_ABS, 0, NULL );
return 1;
}
void Gia_ManFromBridgeHolds( FILE * pFile )
{
fputc( (char)3, pFile ); // true
fputc( (char)1, pFile ); // size of vector (Armin's encoding)
fputc( (char)0, pFile ); // number of the property (Armin's encoding)
fputc( (char)0, pFile ); // no invariant
}
void Gia_ManFromBridgeUnknown( FILE * pFile )
{
fputc( (char)0, pFile ); // undef
fputc( (char)1, pFile ); // size of vector (Armin's encoding)
fputc( (char)0, pFile ); // number of the property (Armin's encoding)
}
void Gia_ManFromBridgeCex( FILE * pFile, Abc_Cex_t * pCex )
{
int i, f, iBit;
Vec_Str_t * vStr = Vec_StrAlloc( 1000 );
Vec_StrPush( vStr, (char)2 ); // false
Vec_StrPush( vStr, (char)1 ); // size of vector (Armin's encoding)
Vec_StrPush( vStr, (char)0 ); // number of the property (Armin's encoding)
Vec_StrPush( vStr, (char)1 ); // size of vector (Armin's encoding)
Gia_WriteAigerEncodeStr( vStr, pCex->iFrame ); // depth
Gia_WriteAigerEncodeStr( vStr, 1 ); // concrete
Gia_WriteAigerEncodeStr( vStr, pCex->iFrame ); // n frames
iBit = pCex->nRegs;
for ( f = 0; f <= pCex->iFrame; f++ )
{
Gia_WriteAigerEncodeStr( vStr, pCex->nPis ); // num of inputs
for ( i = 0; i < pCex->nPis; i++ )
Vec_StrPush( vStr, (char)(Abc_InfoHasBit(pCex->pData, iBit++)?3:2) ); // value
}
assert( iBit == pCex->nBits );
Vec_StrPush( vStr, (char)1 ); // the number of frames (for a concrete counter-example)
for ( i = 0; i < pCex->nRegs; i++ )
Vec_StrPush( vStr, (char)2 ); // always zero ??????????????
fwrite( Vec_StrArray(vStr), Vec_StrSize(vStr), 1, pFile );
Vec_StrFree( vStr );
}
int Gia_ManToBridgeResult( FILE * pFile, int Result, Abc_Cex_t * pCex )
{
if ( Result == 0 ) // sat
Gia_ManFromBridgeCex( pFile, pCex );
else if ( Result == 1 ) // unsat
Gia_ManFromBridgeHolds( pFile );
else if ( Result == -1 ) // undef
Gia_ManFromBridgeUnknown( pFile );
else assert( 0 );
return 1;
2012-03-02 06:20:18 +01:00
}
/**Function*************************************************************
Synopsis []
Description []
SideEffects []
SeeAlso []
***********************************************************************/
Gia_Man_t * Gia_ManFromBridgeReadBody( int Size, unsigned char * pBuffer, Vec_Int_t ** pvInits )
{
int fHash = 0;
Vec_Int_t * vLits, * vInits;
Gia_Man_t * p = NULL;
unsigned char * pBufferPivot, * pBufferEnd = pBuffer + Size;
int i, nInputs, nFlops, nGates, nProps;
unsigned iFan0, iFan1;
nInputs = Gia_ReadAigerDecode( &pBuffer );
nFlops = Gia_ReadAigerDecode( &pBuffer );
nGates = Gia_ReadAigerDecode( &pBuffer );
vLits = Vec_IntAlloc( 1000 );
Vec_IntPush( vLits, -1 );
Vec_IntPush( vLits, 1 );
// start the AIG package
p = Gia_ManStart( nInputs + nFlops * 2 + nGates + 1 + 1 );
p->pName = Abc_UtilStrsav( "temp" );
// create PIs
for ( i = 0; i < nInputs; i++ )
Vec_IntPush( vLits, Gia_ManAppendCi( p ) );
// create flop outputs
for ( i = 0; i < nFlops; i++ )
Vec_IntPush( vLits, Gia_ManAppendCi( p ) );
// create nodes
if ( fHash )
Gia_ManHashAlloc( p );
for ( i = 0; i < nGates; i++ )
{
iFan0 = Gia_ReadAigerDecode( &pBuffer );
iFan1 = Gia_ReadAigerDecode( &pBuffer );
assert( (iFan0 & 1)==0 );
iFan0 >>= 1;
iFan0 = Abc_LitNotCond( Vec_IntEntry(vLits, iFan0 >> 1), iFan0 & 1 );
iFan1 = Abc_LitNotCond( Vec_IntEntry(vLits, iFan1 >> 1), iFan0 & 1 );
if ( fHash )
Vec_IntPush( vLits, Gia_ManHashAnd(p, iFan0, iFan1) );
else
Vec_IntPush( vLits, Gia_ManAppendAnd(p, iFan0, iFan1) );
}
if ( fHash )
Gia_ManHashStop( p );
// remember where flops begin
pBufferPivot = pBuffer;
// stroll through flops
for ( i = 0; i < nFlops; i++ )
Gia_ReadAigerDecode( &pBuffer );
// create POs
nProps = Gia_ReadAigerDecode( &pBuffer );
assert( nProps == 1 );
for ( i = 0; i < nProps; i++ )
{
iFan0 = Gia_ReadAigerDecode( &pBuffer );
iFan0 = Abc_LitNotCond( Vec_IntEntry(vLits, iFan0 >> 1), iFan0 & 1 );
Gia_ManAppendCo( p, iFan0 );
}
// make sure the end of buffer is reached
assert( pBufferEnd == pBuffer );
// resetting to flops
pBuffer = pBufferPivot;
vInits = Vec_IntAlloc( nFlops );
for ( i = 0; i < nFlops; i++ )
{
iFan0 = Gia_ReadAigerDecode( &pBuffer );
2012-03-02 07:36:34 +01:00
assert( (iFan0 & 3) == 2 );
2012-03-02 06:20:18 +01:00
Vec_IntPush( vInits, iFan0 & 3 ); // 0 = X value; 1 = not used; 2 = false; 3 = true
iFan0 >>= 2;
iFan0 = Abc_LitNotCond( Vec_IntEntry(vLits, iFan0 >> 1), iFan0 & 1 );
Gia_ManAppendCo( p, iFan0 );
}
Gia_ManSetRegNum( p, nFlops );
Vec_IntFree( vLits );
// remove wholes in the node list
if ( fHash )
{
Gia_Man_t * pTemp;
p = Gia_ManCleanup( pTemp = p );
Gia_ManStop( pTemp );
}
// return
if ( pvInits )
*pvInits = vInits;
else
Vec_IntFree( vInits );
return p;
}
/**Function*************************************************************
Synopsis []
Description []
SideEffects []
SeeAlso []
***********************************************************************/
int Gia_ManFromBridgeReadPackage( FILE * pFile, int * pType, int * pSize, unsigned char ** ppBuffer )
{
char Temp[24];
int RetValue;
RetValue = fread( Temp, 24, 1, pFile );
if ( RetValue != 1 )
{
printf( "Gia_ManFromBridgeReadPackage(); Error 1: Something is wrong!\n" );
return 0;
}
Temp[6] = 0;
Temp[23]= 0;
*pType = atoi( Temp );
*pSize = atoi( Temp + 7 );
*ppBuffer = ABC_ALLOC( char, *pSize );
RetValue = fread( *ppBuffer, *pSize, 1, pFile );
if ( RetValue != 1 && *pSize != 0 )
{
ABC_FREE( *ppBuffer );
printf( "Gia_ManFromBridgeReadPackage(); Error 2: Something is wrong!\n" );
return 0;
}
return 1;
}
/**Function*************************************************************
Synopsis []
Description []
SideEffects []
SeeAlso []
***********************************************************************/
Gia_Man_t * Gia_ManFromBridge( FILE * pFile, Vec_Int_t ** pvInit )
{
unsigned char * pBuffer;
int Type, Size, RetValue;
Gia_Man_t * p = NULL;
RetValue = Gia_ManFromBridgeReadPackage( pFile, &Type, &Size, &pBuffer );
ABC_FREE( pBuffer );
if ( !RetValue )
return NULL;
RetValue = Gia_ManFromBridgeReadPackage( pFile, &Type, &Size, &pBuffer );
if ( !RetValue )
return NULL;
p = Gia_ManFromBridgeReadBody( Size, pBuffer, pvInit );
ABC_FREE( pBuffer );
if ( p == NULL )
return NULL;
RetValue = Gia_ManFromBridgeReadPackage( pFile, &Type, &Size, &pBuffer );
ABC_FREE( pBuffer );
if ( !RetValue )
return NULL;
return p;
}
/*
{
extern void Gia_ManFromBridgeTest( char * pFileName );
Gia_ManFromBridgeTest( "C:\\_projects\\abc\\_TEST\\bug\\65\\par.dump" );
}
*/
/**Function*************************************************************
Synopsis []
Description []
SideEffects []
SeeAlso []
***********************************************************************/
void Gia_ManFromBridgeTest( char * pFileName )
{
Gia_Man_t * p;
FILE * pFile = fopen( pFileName, "rb" );
if ( pFile == NULL )
{
printf( "Cannot open input file \"%s\".\n", pFileName );
return;
}
p = Gia_ManFromBridge( pFile, NULL );
fclose ( pFile );
Gia_ManPrintStats( p, 0, 0 );
// Gia_WriteAiger( p, "temp.aig", 0, 0 );
Gia_ManStop( p );
}
2012-03-02 07:36:34 +01:00
2012-03-02 06:20:18 +01:00
////////////////////////////////////////////////////////////////////////
/// END OF FILE ///
////////////////////////////////////////////////////////////////////////
ABC_NAMESPACE_IMPL_END