/**CFile**************************************************************** FileName [sclLib.c] SystemName [ABC: Logic synthesis and verification system.] PackageName [Standard-cell library representation.] Synopsis [Standard cell library.] Author [Alan Mishchenko, Niklas Een] Affiliation [UC Berkeley] Date [Ver. 1.0. Started - August 24, 2012.] Revision [$Id: sclLib.c,v 1.0 2012/08/24 00:00:00 alanmi Exp $] ***********************************************************************/ #include "sclLib.h" #include "misc/st/st.h" #include "map/mio/mio.h" #include "bool/kit/kit.h" ABC_NAMESPACE_IMPL_START //////////////////////////////////////////////////////////////////////// /// DECLARATIONS /// //////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////// /// FUNCTION DEFINITIONS /// //////////////////////////////////////////////////////////////////////// /**Function************************************************************* Synopsis [Reading library from file.] Description [] SideEffects [] SeeAlso [] ***********************************************************************/ static void Abc_SclReadSurface( Vec_Str_t * vOut, int * pPos, SC_Surface * p ) { Vec_Flt_t * vVec; int i, j; for ( i = Vec_StrGetI(vOut, pPos); i != 0; i-- ) Vec_FltPush( p->vIndex0, Vec_StrGetF(vOut, pPos) ); for ( i = Vec_StrGetI(vOut, pPos); i != 0; i-- ) Vec_FltPush( p->vIndex1, Vec_StrGetF(vOut, pPos) ); for ( i = 0; i < Vec_FltSize(p->vIndex0); i++ ) { vVec = Vec_FltAlloc( Vec_FltSize(p->vIndex1) ); Vec_PtrPush( p->vData, vVec ); for ( j = 0; j < Vec_FltSize(p->vIndex1); j++ ) Vec_FltPush( vVec, Vec_StrGetF(vOut, pPos) ); } for ( i = 0; i < 3; i++ ) p->approx[0][i] = Vec_StrGetF( vOut, pPos ); for ( i = 0; i < 4; i++ ) p->approx[1][i] = Vec_StrGetF( vOut, pPos ); for ( i = 0; i < 6; i++ ) p->approx[2][i] = Vec_StrGetF( vOut, pPos ); } static void Abc_SclReadLibrary( Vec_Str_t * vOut, int * pPos, SC_Lib * p ) { int i, j, k, n; int version = Vec_StrGetI( vOut, pPos ); assert( version == 5 || version == ABC_SCL_CUR_VERSION ); // wrong version of the file // Read non-composite fields: p->pName = Vec_StrGetS(vOut, pPos); p->default_wire_load = Vec_StrGetS(vOut, pPos); p->default_wire_load_sel = Vec_StrGetS(vOut, pPos); p->default_max_out_slew = Vec_StrGetF(vOut, pPos); p->unit_time = Vec_StrGetI(vOut, pPos); p->unit_cap_fst = Vec_StrGetF(vOut, pPos); p->unit_cap_snd = Vec_StrGetI(vOut, pPos); // Read 'wire_load' vector: for ( i = Vec_StrGetI(vOut, pPos); i != 0; i-- ) { SC_WireLoad * pWL = Abc_SclWireLoadAlloc(); Vec_PtrPush( p->vWireLoads, pWL ); pWL->pName = Vec_StrGetS(vOut, pPos); pWL->res = Vec_StrGetF(vOut, pPos); pWL->cap = Vec_StrGetF(vOut, pPos); for ( j = Vec_StrGetI(vOut, pPos); j != 0; j-- ) { Vec_IntPush( pWL->vFanout, Vec_StrGetI(vOut, pPos) ); Vec_FltPush( pWL->vLen, Vec_StrGetF(vOut, pPos) ); } } // Read 'wire_load_sel' vector: for ( i = Vec_StrGetI(vOut, pPos); i != 0; i-- ) { SC_WireLoadSel * pWLS = Abc_SclWireLoadSelAlloc(); Vec_PtrPush( p->vWireLoadSels, pWLS ); pWLS->pName = Vec_StrGetS(vOut, pPos); for ( j = Vec_StrGetI(vOut, pPos); j != 0; j-- ) { Vec_FltPush( pWLS->vAreaFrom, Vec_StrGetF(vOut, pPos) ); Vec_FltPush( pWLS->vAreaTo, Vec_StrGetF(vOut, pPos) ); Vec_PtrPush( pWLS->vWireLoadModel, Vec_StrGetS(vOut, pPos) ); } } for ( i = Vec_StrGetI(vOut, pPos); i != 0; i-- ) { SC_Cell * pCell = Abc_SclCellAlloc(); pCell->Id = Vec_PtrSize(p->vCells); Vec_PtrPush( p->vCells, pCell ); pCell->pName = Vec_StrGetS(vOut, pPos); pCell->area = Vec_StrGetF(vOut, pPos); pCell->drive_strength = Vec_StrGetI(vOut, pPos); pCell->n_inputs = Vec_StrGetI(vOut, pPos); pCell->n_outputs = Vec_StrGetI(vOut, pPos); for ( j = 0; j < pCell->n_inputs; j++ ) { SC_Pin * pPin = Abc_SclPinAlloc(); Vec_PtrPush( pCell->vPins, pPin ); pPin->dir = sc_dir_Input; pPin->pName = Vec_StrGetS(vOut, pPos); pPin->rise_cap = Vec_StrGetF(vOut, pPos); pPin->fall_cap = Vec_StrGetF(vOut, pPos); } for ( j = 0; j < pCell->n_outputs; j++ ) { SC_Pin * pPin = Abc_SclPinAlloc(); Vec_PtrPush( pCell->vPins, pPin ); pPin->dir = sc_dir_Output; pPin->pName = Vec_StrGetS(vOut, pPos); pPin->max_out_cap = Vec_StrGetF(vOut, pPos); pPin->max_out_slew = Vec_StrGetF(vOut, pPos); k = Vec_StrGetI(vOut, pPos); assert( k == pCell->n_inputs ); // read function if ( version == 5 ) { // formula is not given assert( Vec_WrdSize(pPin->vFunc) == 0 ); Vec_WrdGrow( pPin->vFunc, Abc_Truth6WordNum(pCell->n_inputs) ); for ( k = 0; k < Vec_WrdCap(pPin->vFunc); k++ ) Vec_WrdPush( pPin->vFunc, Vec_StrGetW(vOut, pPos) ); } else { // (possibly empty) formula is always given assert( version == ABC_SCL_CUR_VERSION ); assert( pPin->func_text == NULL ); pPin->func_text = Vec_StrGetS(vOut, pPos); if ( pPin->func_text[0] == 0 ) { // formula is not given - read truth table ABC_FREE( pPin->func_text ); assert( Vec_WrdSize(pPin->vFunc) == 0 ); Vec_WrdGrow( pPin->vFunc, Abc_Truth6WordNum(pCell->n_inputs) ); for ( k = 0; k < Vec_WrdCap(pPin->vFunc); k++ ) Vec_WrdPush( pPin->vFunc, Vec_StrGetW(vOut, pPos) ); } else { // formula is given - derive truth table SC_Pin * pPin2; Vec_Ptr_t * vNames; // collect input names vNames = Vec_PtrAlloc( pCell->n_inputs ); SC_CellForEachPinIn( pCell, pPin2, n ) Vec_PtrPush( vNames, pPin2->pName ); // derive truth table assert( Vec_WrdSize(pPin->vFunc) == 0 ); Vec_WrdFree( pPin->vFunc ); pPin->vFunc = Mio_ParseFormulaTruth( pPin->func_text, (char **)Vec_PtrArray(vNames), pCell->n_inputs ); Vec_PtrFree( vNames ); } } // Read 'rtiming': (pin-to-pin timing tables for this particular output) for ( k = 0; k < pCell->n_inputs; k++ ) { SC_Timings * pRTime = Abc_SclTimingsAlloc(); Vec_PtrPush( pPin->vRTimings, pRTime ); pRTime->pName = Vec_StrGetS(vOut, pPos); n = Vec_StrGetI(vOut, pPos); assert( n <= 1 ); if ( n == 1 ) { SC_Timing * pTime = Abc_SclTimingAlloc(); Vec_PtrPush( pRTime->vTimings, pTime ); pTime->tsense = (SC_TSense)Vec_StrGetI(vOut, pPos); Abc_SclReadSurface( vOut, pPos, pTime->pCellRise ); Abc_SclReadSurface( vOut, pPos, pTime->pCellFall ); Abc_SclReadSurface( vOut, pPos, pTime->pRiseTrans ); Abc_SclReadSurface( vOut, pPos, pTime->pFallTrans ); } else assert( Vec_PtrSize(pPin->vRTimings) == 0 ); } } } } SC_Lib * Abc_SclRead( char * pFileName ) { SC_Lib * p; FILE * pFile; Vec_Str_t * vOut; int nFileSize, Pos = 0; pFile = fopen( pFileName, "rb" ); if ( pFile == NULL ) { printf( "Cannot open file \"%s\" for reading.\n", pFileName ); return NULL; } // get the file size, in bytes fseek( pFile, 0, SEEK_END ); nFileSize = ftell( pFile ); rewind( pFile ); // load the contents vOut = Vec_StrAlloc( nFileSize ); vOut->nSize = vOut->nCap; assert( nFileSize == Vec_StrSize(vOut) ); nFileSize = fread( Vec_StrArray(vOut), 1, Vec_StrSize(vOut), pFile ); assert( nFileSize == Vec_StrSize(vOut) ); fclose( pFile ); // read the library p = Abc_SclLibAlloc(); Abc_SclReadLibrary( vOut, &Pos, p ); assert( Pos == Vec_StrSize(vOut) ); Vec_StrFree( vOut ); // hash gates by name Abc_SclHashCells( p ); Abc_SclLinkCells( p ); return p; } void Abc_SclLoad( char * pFileName, SC_Lib ** ppScl ) { if ( *ppScl ) { Abc_SclLibFree( *ppScl ); *ppScl = NULL; } assert( *ppScl == NULL ); if ( pFileName ) *(SC_Lib **)ppScl = Abc_SclRead( pFileName ); } /**Function************************************************************* Synopsis [Writing library into file.] Description [] SideEffects [] SeeAlso [] ***********************************************************************/ static void Abc_SclWriteSurface( Vec_Str_t * vOut, SC_Surface * p ) { Vec_Flt_t * vVec; float Entry; int i, k; Vec_StrPutI( vOut, Vec_FltSize(p->vIndex0) ); Vec_FltForEachEntry( p->vIndex0, Entry, i ) Vec_StrPutF( vOut, Entry ); Vec_StrPutI( vOut, Vec_FltSize(p->vIndex1) ); Vec_FltForEachEntry( p->vIndex1, Entry, i ) Vec_StrPutF( vOut, Entry ); Vec_PtrForEachEntry( Vec_Flt_t *, p->vData, vVec, i ) Vec_FltForEachEntry( vVec, Entry, k ) Vec_StrPutF( vOut, Entry ); for ( i = 0; i < 3; i++ ) Vec_StrPutF( vOut, p->approx[0][i] ); for ( i = 0; i < 4; i++ ) Vec_StrPutF( vOut, p->approx[1][i] ); for ( i = 0; i < 6; i++ ) Vec_StrPutF( vOut, p->approx[2][i] ); } static void Abc_SclWriteLibrary( Vec_Str_t * vOut, SC_Lib * p ) { SC_WireLoad * pWL; SC_WireLoadSel * pWLS; SC_Cell * pCell; SC_Pin * pPin; int n_valid_cells; int i, j, k; Vec_StrPutI( vOut, ABC_SCL_CUR_VERSION ); // Write non-composite fields: Vec_StrPutS( vOut, p->pName ); Vec_StrPutS( vOut, p->default_wire_load ); Vec_StrPutS( vOut, p->default_wire_load_sel ); Vec_StrPutF( vOut, p->default_max_out_slew ); assert( p->unit_time >= 0 ); assert( p->unit_cap_snd >= 0 ); Vec_StrPutI( vOut, p->unit_time ); Vec_StrPutF( vOut, p->unit_cap_fst ); Vec_StrPutI( vOut, p->unit_cap_snd ); // Write 'wire_load' vector: Vec_StrPutI( vOut, Vec_PtrSize(p->vWireLoads) ); SC_LibForEachWireLoad( p, pWL, i ) { Vec_StrPutS( vOut, pWL->pName ); Vec_StrPutF( vOut, pWL->res ); Vec_StrPutF( vOut, pWL->cap ); Vec_StrPutI( vOut, Vec_IntSize(pWL->vFanout) ); for ( j = 0; j < Vec_IntSize(pWL->vFanout); j++ ) { Vec_StrPutI( vOut, Vec_IntEntry(pWL->vFanout, j) ); Vec_StrPutF( vOut, Vec_FltEntry(pWL->vLen, j) ); } } // Write 'wire_load_sel' vector: Vec_StrPutI( vOut, Vec_PtrSize(p->vWireLoadSels) ); SC_LibForEachWireLoadSel( p, pWLS, i ) { Vec_StrPutS( vOut, pWLS->pName ); Vec_StrPutI( vOut, Vec_FltSize(pWLS->vAreaFrom) ); for ( j = 0; j < Vec_FltSize(pWLS->vAreaFrom); j++) { Vec_StrPutF( vOut, Vec_FltEntry(pWLS->vAreaFrom, j) ); Vec_StrPutF( vOut, Vec_FltEntry(pWLS->vAreaTo, j) ); Vec_StrPutS( vOut, (char *)Vec_PtrEntry(pWLS->vWireLoadModel, j) ); } } // Write 'cells' vector: n_valid_cells = 0; SC_LibForEachCell( p, pCell, i ) if ( !(pCell->seq || pCell->unsupp) ) n_valid_cells++; Vec_StrPutI( vOut, n_valid_cells ); SC_LibForEachCell( p, pCell, i ) { if ( pCell->seq || pCell->unsupp ) continue; Vec_StrPutS( vOut, pCell->pName ); Vec_StrPutF( vOut, pCell->area ); Vec_StrPutI( vOut, pCell->drive_strength ); // Write 'pins': (sorted at this point; first inputs, then outputs) Vec_StrPutI( vOut, pCell->n_inputs); Vec_StrPutI( vOut, pCell->n_outputs); SC_CellForEachPinIn( pCell, pPin, j ) { assert(pPin->dir == sc_dir_Input); Vec_StrPutS( vOut, pPin->pName ); Vec_StrPutF( vOut, pPin->rise_cap ); Vec_StrPutF( vOut, pPin->fall_cap ); } SC_CellForEachPinOut( pCell, pPin, j ) { SC_Timings * pRTime; word uWord; assert(pPin->dir == sc_dir_Output); Vec_StrPutS( vOut, pPin->pName ); Vec_StrPutF( vOut, pPin->max_out_cap ); Vec_StrPutF( vOut, pPin->max_out_slew ); // write function if ( pPin->func_text == NULL ) { // formula is not given - write empty string Vec_StrPutS( vOut, "" ); // write truth table assert( Vec_WrdSize(pPin->vFunc) == Abc_Truth6WordNum(pCell->n_inputs) ); Vec_StrPutI( vOut, pCell->n_inputs ); Vec_WrdForEachEntry( pPin->vFunc, uWord, k ) // -- 'size = 1u << (n_vars - 6)' Vec_StrPutW( vOut, uWord ); // -- 64-bit number, written uncompressed (low-byte first) } else // formula is given Vec_StrPutS( vOut, pPin->func_text ); // Write 'rtiming': (pin-to-pin timing tables for this particular output) assert( Vec_PtrSize(pPin->vRTimings) == pCell->n_inputs ); SC_PinForEachRTiming( pPin, pRTime, k ) { Vec_StrPutS( vOut, pRTime->pName ); Vec_StrPutI( vOut, Vec_PtrSize(pRTime->vTimings) ); // -- NOTE! After post-processing, the size of the 'rtiming[k]' vector is either // 0 or 1 (in static timing, we have merged all tables to get the worst case). // The case with size 0 should only occur for multi-output gates. if ( Vec_PtrSize(pRTime->vTimings) == 1 ) { SC_Timing * pTime = (SC_Timing *)Vec_PtrEntry( pRTime->vTimings, 0 ); // -- NOTE! We don't need to save 'related_pin' string because we have sorted // the elements on input pins. Vec_StrPutI( vOut, (int)pTime->tsense); Abc_SclWriteSurface( vOut, pTime->pCellRise ); Abc_SclWriteSurface( vOut, pTime->pCellFall ); Abc_SclWriteSurface( vOut, pTime->pRiseTrans ); Abc_SclWriteSurface( vOut, pTime->pFallTrans ); } else assert( Vec_PtrSize(pRTime->vTimings) == 0 ); } } } } void Abc_SclWrite( char * pFileName, SC_Lib * p ) { Vec_Str_t * vOut; vOut = Vec_StrAlloc( 10000 ); Abc_SclWriteLibrary( vOut, p ); if ( Vec_StrSize(vOut) > 0 ) { FILE * pFile = fopen( pFileName, "wb" ); if ( pFile == NULL ) printf( "Cannot open file \"%s\" for writing.\n", pFileName ); else { fwrite( Vec_StrArray(vOut), 1, Vec_StrSize(vOut), pFile ); fclose( pFile ); } } Vec_StrFree( vOut ); } void Abc_SclSave( char * pFileName, SC_Lib * pScl ) { if ( pScl == NULL ) return; Abc_SclWrite( pFileName, pScl ); } /**Function************************************************************* Synopsis [Writing library into text file.] Description [] SideEffects [] SeeAlso [] ***********************************************************************/ static void Abc_SclWriteSurfaceText( FILE * s, SC_Surface * p ) { Vec_Flt_t * vVec; float Entry; int i, k; fprintf( s, " index_1(\"" ); Vec_FltForEachEntry( p->vIndex0, Entry, i ) fprintf( s, "%f%s", Entry, i == Vec_FltSize(p->vIndex0)-1 ? "":", " ); fprintf( s, "\");\n" ); fprintf( s, " index_2(\"" ); Vec_FltForEachEntry( p->vIndex1, Entry, i ) fprintf( s, "%f%s", Entry, i == Vec_FltSize(p->vIndex1)-1 ? "":", " ); fprintf( s, "\");\n" ); fprintf( s, " values (\"" ); Vec_PtrForEachEntry( Vec_Flt_t *, p->vData, vVec, i ) { Vec_FltForEachEntry( vVec, Entry, k ) fprintf( s, "%f%s", Entry, i == Vec_PtrSize(p->vData)-1 && k == Vec_FltSize(vVec)-1 ? "\");":", " ); if ( i == Vec_PtrSize(p->vData)-1 ) fprintf( s, "\n" ); else { fprintf( s, "\\\n" ); fprintf( s, " " ); } } /* fprintf( s, " approximations: \n" ); fprintf( s, " " ); for ( i = 0; i < 3; i++ ) fprintf( s, "%f ", p->approx[0][i] ); fprintf( s, "\n" ); fprintf( s, " " ); for ( i = 0; i < 4; i++ ) fprintf( s, "%f ", p->approx[1][i] ); fprintf( s, "\n" ); fprintf( s, " " ); for ( i = 0; i < 6; i++ ) fprintf( s, "%f ", p->approx[2][i] ); fprintf( s, "\n" ); fprintf( s, " \n" ); */ } static void Abc_SclWriteLibraryText( FILE * s, SC_Lib * p ) { SC_WireLoad * pWL; SC_WireLoadSel * pWLS; SC_Cell * pCell; SC_Pin * pPin; int n_valid_cells; int i, j, k; // fprintf( s, "%d", ABC_SCL_CUR_VERSION ); fprintf( s, "library(%s) {\n\n", p->pName ); if ( p->default_wire_load && strlen(p->default_wire_load) ) fprintf( s, " default_wire_load : \"%s\";\n", p->default_wire_load ); if ( p->default_wire_load_sel && strlen(p->default_wire_load_sel) ) fprintf( s, " default_wire_load_selection : \"%s\";\n", p->default_wire_load_sel ); if ( p->default_max_out_slew != -1 ) fprintf( s, " default_max_transition : %f;\n", p->default_max_out_slew ); if ( p->unit_time == 9 ) fprintf( s, " time_unit : \"1ns\";\n" ); else if ( p->unit_time == 10 ) fprintf( s, " time_unit : \"100ps\";\n" ); else if ( p->unit_time == 11 ) fprintf( s, " time_unit : \"10ps\";\n" ); else if ( p->unit_time == 12 ) fprintf( s, " time_unit : \"1ps\";\n" ); else assert( 0 ); fprintf( s, " capacitive_load_unit(%.1f,%s);\n", p->unit_cap_fst, p->unit_cap_snd == 12 ? "pf" : "ff" ); fprintf( s, "\n" ); // Write 'wire_load' vector: SC_LibForEachWireLoad( p, pWL, i ) { fprintf( s, " wire_load(\"%s\") {\n", pWL->pName ); fprintf( s, " capacitance : %f;\n", pWL->cap ); fprintf( s, " resistance : %f;\n", pWL->res ); for ( j = 0; j < Vec_IntSize(pWL->vFanout); j++ ) fprintf( s, " fanout_length( %d, %f );\n", Vec_IntEntry(pWL->vFanout, j), Vec_FltEntry(pWL->vLen, j) ); fprintf( s, " }\n\n" ); } // Write 'wire_load_sel' vector: SC_LibForEachWireLoadSel( p, pWLS, i ) { fprintf( s, " wire_load_selection(\"%s\") {\n", pWLS->pName ); for ( j = 0; j < Vec_FltSize(pWLS->vAreaFrom); j++) fprintf( s, " wire_load_from_area( %f, %f, \"%s\" );\n", Vec_FltEntry(pWLS->vAreaFrom, j), Vec_FltEntry(pWLS->vAreaTo, j), (char *)Vec_PtrEntry(pWLS->vWireLoadModel, j) ); fprintf( s, " }\n\n" ); } // Write 'cells' vector: n_valid_cells = 0; SC_LibForEachCell( p, pCell, i ) if ( !(pCell->seq || pCell->unsupp) ) n_valid_cells++; SC_LibForEachCell( p, pCell, i ) { if ( pCell->seq || pCell->unsupp ) continue; fprintf( s, "\n" ); fprintf( s, " cell(%s) {\n", pCell->pName ); fprintf( s, " /* n_inputs = %d n_outputs = %d */\n", pCell->n_inputs, pCell->n_outputs ); fprintf( s, " area : %f;\n", pCell->area ); fprintf( s, " drive_strength : %d;\n", pCell->drive_strength ); SC_CellForEachPinIn( pCell, pPin, j ) { assert(pPin->dir == sc_dir_Input); fprintf( s, " pin(%s) {\n", pPin->pName ); fprintf( s, " direction : %s;\n", "input" ); fprintf( s, " fall_capacitance : %f;\n", pPin->fall_cap ); fprintf( s, " rise_capacitance : %f;\n", pPin->rise_cap ); fprintf( s, " }\n" ); } SC_CellForEachPinOut( pCell, pPin, j ) { SC_Timings * pRTime; // word uWord; assert(pPin->dir == sc_dir_Output); fprintf( s, " pin(%s) {\n", pPin->pName ); fprintf( s, " direction : %s;\n", "output" ); fprintf( s, " max_capacitance : %f;\n", pPin->max_out_cap ); fprintf( s, " max_transition : %f;\n", pPin->max_out_slew ); fprintf( s, " function : \"%s\";\n", pPin->func_text ? pPin->func_text : "?" ); fprintf( s, " /* truth table = " ); Extra_PrintHex( s, (unsigned *)Vec_WrdArray(pPin->vFunc), pCell->n_inputs ); fprintf( s, " */\n" ); // Write 'rtiming': (pin-to-pin timing tables for this particular output) assert( Vec_PtrSize(pPin->vRTimings) == pCell->n_inputs ); SC_PinForEachRTiming( pPin, pRTime, k ) { if ( Vec_PtrSize(pRTime->vTimings) == 1 ) { SC_Timing * pTime = (SC_Timing *)Vec_PtrEntry( pRTime->vTimings, 0 ); fprintf( s, " timing() {\n" ); fprintf( s, " related_pin : \"%s\"\n", pRTime->pName ); if ( pTime->tsense == sc_ts_Pos ) fprintf( s, " timing_sense : positive_unate;\n" ); else if ( pTime->tsense == sc_ts_Neg ) fprintf( s, " timing_sense : negative_unate;\n" ); else if ( pTime->tsense == sc_ts_Non ) fprintf( s, " timing_sense : non_unate;\n" ); else assert( 0 ); fprintf( s, " cell_rise() {\n" ); Abc_SclWriteSurfaceText( s, pTime->pCellRise ); fprintf( s, " }\n" ); fprintf( s, " cell_fall() {\n" ); Abc_SclWriteSurfaceText( s, pTime->pCellFall ); fprintf( s, " }\n" ); fprintf( s, " rise_transition() {\n" ); Abc_SclWriteSurfaceText( s, pTime->pRiseTrans ); fprintf( s, " }\n" ); fprintf( s, " fall_transition() {\n" ); Abc_SclWriteSurfaceText( s, pTime->pFallTrans ); fprintf( s, " }\n" ); fprintf( s, " }\n" ); } else assert( Vec_PtrSize(pRTime->vTimings) == 0 ); } fprintf( s, " }\n" ); } fprintf( s, " }\n" ); } fprintf( s, "}\n\n" ); } void Abc_SclWriteText( char * pFileName, SC_Lib * p ) { FILE * pFile = fopen( pFileName, "wb" ); if ( pFile == NULL ) printf( "Cannot open text file \"%s\" for writing.\n", pFileName ); else { Abc_SclWriteLibraryText( pFile, p ); fclose( pFile ); } } /**Function************************************************************* Synopsis [Reading library from file.] Description [] SideEffects [] SeeAlso [] ***********************************************************************/ static unsigned Abc_SclHashString( char * pName, int TableSize ) { static int s_Primes[10] = { 1291, 1699, 2357, 4177, 5147, 5647, 6343, 7103, 7873, 8147 }; unsigned i, Key = 0; for ( i = 0; pName[i] != '\0'; i++ ) Key += s_Primes[i%10]*pName[i]*pName[i]; return Key % TableSize; } int * Abc_SclHashLookup( SC_Lib * p, char * pName ) { int i; for ( i = Abc_SclHashString(pName, p->nBins); i < p->nBins; i = (i + 1) % p->nBins ) if ( p->pBins[i] == -1 || !strcmp(pName, SC_LibCell(p, p->pBins[i])->pName) ) return p->pBins + i; assert( 0 ); return NULL; } void Abc_SclHashCells( SC_Lib * p ) { SC_Cell * pCell; int i, * pPlace; assert( p->nBins == 0 ); p->nBins = Abc_PrimeCudd( 5 * Vec_PtrSize(p->vCells) ); p->pBins = ABC_FALLOC( int, p->nBins ); SC_LibForEachCell( p, pCell, i ) { pPlace = Abc_SclHashLookup( p, pCell->pName ); assert( *pPlace == -1 ); *pPlace = i; } } int Abc_SclCellFind( SC_Lib * p, char * pName ) { int *pPlace = Abc_SclHashLookup( p, pName ); return pPlace ? *pPlace : -1; } int Abc_SclClassCellNum( SC_Cell * pClass ) { SC_Cell * pCell; int i, Count = 0; SC_RingForEachCell( pClass, pCell, i ) if ( !pCell->fSkip ) Count++; return Count; } /**Function************************************************************* Synopsis [Links equal gates into rings while sorting them by area.] Description [] SideEffects [] SeeAlso [] ***********************************************************************/ static int Abc_SclCompareCells( SC_Cell ** pp1, SC_Cell ** pp2 ) { if ( (*pp1)->n_inputs < (*pp2)->n_inputs ) return -1; if ( (*pp1)->n_inputs > (*pp2)->n_inputs ) return 1; if ( (*pp1)->area < (*pp2)->area ) return -1; if ( (*pp1)->area > (*pp2)->area ) return 1; return strcmp( (*pp1)->pName, (*pp2)->pName ); } void Abc_SclLinkCells( SC_Lib * p ) { SC_Cell * pCell, * pRepr = NULL; int i, k; assert( Vec_PtrSize(p->vCellClasses) == 0 ); SC_LibForEachCell( p, pCell, i ) { // find gate with the same function SC_LibForEachCellClass( p, pRepr, k ) if ( pCell->n_inputs == pRepr->n_inputs && pCell->n_outputs == pRepr->n_outputs && Vec_WrdEqual(SC_CellFunc(pCell), SC_CellFunc(pRepr)) ) break; if ( k == Vec_PtrSize(p->vCellClasses) ) { Vec_PtrPush( p->vCellClasses, pCell ); pCell->pNext = pCell->pPrev = pCell; continue; } // add it to the list before the cell pRepr->pPrev->pNext = pCell; pCell->pNext = pRepr; pCell->pPrev = pRepr->pPrev; pRepr->pPrev = pCell; } // sort cells by size the then by name qsort( (void *)Vec_PtrArray(p->vCellClasses), Vec_PtrSize(p->vCellClasses), sizeof(void *), (int(*)(const void *,const void *))Abc_SclCompareCells ); // sort cell lists SC_LibForEachCellClass( p, pRepr, k ) { Vec_Ptr_t * vList = Vec_PtrAlloc( 100 ); SC_RingForEachCell( pRepr, pCell, i ) Vec_PtrPush( vList, pCell ); qsort( (void *)Vec_PtrArray(vList), Vec_PtrSize(vList), sizeof(void *), (int(*)(const void *,const void *))Abc_SclCompareCells ); // create new representative pRepr = (SC_Cell *)Vec_PtrEntry( vList, 0 ); pRepr->pNext = pRepr->pPrev = pRepr; pRepr->Order = 0; // relink cells Vec_PtrForEachEntryStart( SC_Cell *, vList, pCell, i, 1 ) { pRepr->pPrev->pNext = pCell; pCell->pNext = pRepr; pCell->pPrev = pRepr->pPrev; pRepr->pPrev = pCell; pCell->Order = i; } // update list Vec_PtrWriteEntry( p->vCellClasses, k, pRepr ); Vec_PtrFree( vList ); } } /**Function************************************************************* Synopsis [Returns the wireload model for the given area.] Description [] SideEffects [] SeeAlso [] ***********************************************************************/ SC_WireLoad * Abc_SclFetchWireLoadModel( SC_Lib * p, char * pWLoadUsed ) { SC_WireLoad * pWL = NULL; int i; // Get the actual table and reformat it for 'wire_cap' output: assert( pWLoadUsed != NULL ); SC_LibForEachWireLoad( p, pWL, i ) if ( !strcmp(pWL->pName, pWLoadUsed) ) break; if ( i == Vec_PtrSize(p->vWireLoads) ) { Abc_Print( -1, "Cannot find wire load model \"%s\".\n", pWLoadUsed ); exit(1); } // printf( "Using wireload model \"%s\".\n", pWL->pName ); return pWL; } SC_WireLoad * Abc_SclFindWireLoadModel( SC_Lib * p, float Area ) { char * pWLoadUsed = NULL; int i; if ( p->default_wire_load_sel && strlen(p->default_wire_load_sel) ) { SC_WireLoadSel * pWLS = NULL; SC_LibForEachWireLoadSel( p, pWLS, i ) if ( !strcmp(pWLS->pName, p->default_wire_load_sel) ) break; if ( i == Vec_PtrSize(p->vWireLoadSels) ) { Abc_Print( -1, "Cannot find wire load selection model \"%s\".\n", p->default_wire_load_sel ); exit(1); } for ( i = 0; i < Vec_FltSize(pWLS->vAreaFrom); i++) if ( Area >= Vec_FltEntry(pWLS->vAreaFrom, i) && Area < Vec_FltEntry(pWLS->vAreaTo, i) ) { pWLoadUsed = (char *)Vec_PtrEntry(pWLS->vWireLoadModel, i); break; } if ( i == Vec_FltSize(pWLS->vAreaFrom) ) pWLoadUsed = (char *)Vec_PtrEntryLast(pWLS->vWireLoadModel); } else if ( p->default_wire_load && strlen(p->default_wire_load) ) pWLoadUsed = p->default_wire_load; else { Abc_Print( 0, "No wire model given.\n" ); return NULL; } return Abc_SclFetchWireLoadModel( p, pWLoadUsed ); } /**Function************************************************************* Synopsis [Compute delay parameters of pin/cell/class.] Description [] SideEffects [] SeeAlso [] ***********************************************************************/ void Abc_SclComputeParametersPin( SC_Lib * p, SC_Cell * pCell, int iPin, float Slew, float * pED, float * pPD ) { SC_Timings * pRTime; SC_Timing * pTime = NULL; SC_Pin * pPin; SC_Pair ArrIn = { 0.0, 0.0 }; SC_Pair SlewIn = { Slew, Slew }; SC_Pair Load0, Load1, Load2; SC_Pair ArrOut0 = { 0.0, 0.0 }; SC_Pair ArrOut1 = { 0.0, 0.0 }; SC_Pair ArrOut2 = { 0.0, 0.0 }; SC_Pair SlewOut = { 0.0, 0.0 }; Vec_Flt_t * vIndex; assert( iPin >= 0 && iPin < pCell->n_inputs ); pPin = SC_CellPin( pCell, pCell->n_inputs ); // find timing info for this pin assert( Vec_PtrSize(pPin->vRTimings) == pCell->n_inputs ); pRTime = (SC_Timings *)Vec_PtrEntry( pPin->vRTimings, iPin ); assert( Vec_PtrSize(pRTime->vTimings) == 1 ); pTime = (SC_Timing *)Vec_PtrEntry( pRTime->vTimings, 0 ); // get load points vIndex = pTime->pCellRise->vIndex1; // capacitance Load0.rise = Load0.fall = 0.0; Load1.rise = Load1.fall = Vec_FltEntry( vIndex, 0 ); Load2.rise = Load2.fall = Vec_FltEntry( vIndex, Vec_FltSize(vIndex) - 2 ); // compute delay Scl_LibPinArrival( pTime, &ArrIn, &SlewIn, &Load0, &ArrOut0, &SlewOut ); Scl_LibPinArrival( pTime, &ArrIn, &SlewIn, &Load1, &ArrOut1, &SlewOut ); Scl_LibPinArrival( pTime, &ArrIn, &SlewIn, &Load2, &ArrOut2, &SlewOut ); ArrOut0.rise = 0.5 * (ArrOut0.rise + ArrOut0.fall); ArrOut1.rise = 0.5 * (ArrOut1.rise + ArrOut1.fall); ArrOut2.rise = 0.5 * (ArrOut2.rise + ArrOut2.fall); // get tangent *pED = (ArrOut2.rise - ArrOut1.rise) / ((Load2.rise - Load1.rise) / SC_CellPinCap(pCell, iPin)); // get constant *pPD = ArrOut0.rise; } void Abc_SclComputeParametersCell( SC_Lib * p, SC_Cell * pCell, float Slew, float * pED, float * pPD ) { SC_Pin * pPin; float ED, PD, ed, pd; int i; ED = PD = ed = pd = 0; SC_CellForEachPinIn( pCell, pPin, i ) { Abc_SclComputeParametersPin( p, pCell, i, Slew, &ed, &pd ); ED += ed; PD += pd; } *pED = ED / pCell->n_inputs; *pPD = PD / pCell->n_inputs; } void Abc_SclComputeParametersClass( SC_Lib * p, SC_Cell * pRepr, float Slew, float * pED, float * pPD ) { SC_Cell * pCell; float ED, PD, ed, pd; int i, Count = 0; ED = PD = ed = pd = 0; SC_RingForEachCell( pRepr, pCell, i ) { Abc_SclComputeParametersCell( p, pCell, Slew, &ed, &pd ); ED += ed; PD += pd; Count++; } *pED = ED / Count; *pPD = PD / Count; } void Abc_SclComputeParametersClassPin( SC_Lib * p, SC_Cell * pRepr, int iPin, float Slew, float * pED, float * pPD ) { SC_Cell * pCell; float ED, PD, ed, pd; int i, Count = 0; ED = PD = ed = pd = 0; SC_RingForEachCell( pRepr, pCell, i ) { Abc_SclComputeParametersPin( p, pCell, Slew, iPin, &ed, &pd ); ED += ed; PD += pd; Count++; } *pED = ED / Count; *pPD = PD / Count; } float Abc_SclComputeDelayCellPin( SC_Lib * p, SC_Cell * pCell, int iPin, float Slew, float Gain ) { float ED = 0, PD = 0; Abc_SclComputeParametersPin( p, pCell, iPin, Slew, &ED, &PD ); return 0.01 * ED * Gain + PD; } float Abc_SclComputeDelayClassPin( SC_Lib * p, SC_Cell * pRepr, int iPin, float Slew, float Gain ) { SC_Cell * pCell; float Delay = 0; int i, Count = 0; SC_RingForEachCell( pRepr, pCell, i ) { if ( pCell->fSkip ) continue; // if ( pRepr == pCell ) // skip the first gate // continue; Delay += Abc_SclComputeDelayCellPin( p, pCell, iPin, Slew, Gain ); Count++; } return Delay / Abc_MaxInt(1, Count); } float Abc_SclComputeAreaClass( SC_Cell * pRepr ) { SC_Cell * pCell; float Area = 0; int i, Count = 0; SC_RingForEachCell( pRepr, pCell, i ) { if ( pCell->fSkip ) continue; Area += pCell->area; Count++; } return Area / Count; } /**Function************************************************************* Synopsis [Print cells] Description [] SideEffects [] SeeAlso [] ***********************************************************************/ void Abc_SclMarkSkippedCells( SC_Lib * p ) { char FileName[1000]; char Buffer[1000], * pName; SC_Cell * pCell; FILE * pFile; int CellId, nSkipped = 0; sprintf( FileName, "%s.skip", p->pName ); pFile = fopen( FileName, "rb" ); if ( pFile == NULL ) return; while ( fgets( Buffer, 999, pFile ) != NULL ) { pName = strtok( Buffer, "\r\n\t " ); if ( pName == NULL ) continue; CellId = Abc_SclCellFind( p, pName ); if ( CellId == -1 ) { printf( "Cannot find cell \"%s\" in the library \"%s\".\n", pName, p->pName ); continue; } pCell = SC_LibCell( p, CellId ); pCell->fSkip = 1; nSkipped++; } fclose( pFile ); printf( "Marked %d cells for skipping in the library \"%s\".\n", nSkipped, p->pName ); } void Abc_SclPrintCells( SC_Lib * p, float Slew, float Gain ) { SC_Cell * pCell, * pRepr; int i, k, nLength = 0; float ED = 0, PD = 0; assert( Vec_PtrSize(p->vCellClasses) > 0 ); printf( "Library \"%s\" ", p->pName ); printf( "has %d cells in %d classes. ", Vec_PtrSize(p->vCells), Vec_PtrSize(p->vCellClasses) ); printf( "Delay estimate is based on slew %.2f and gain %.2f.\n", Slew, Gain ); Abc_SclMarkSkippedCells( p ); // find the longest name SC_LibForEachCellClass( p, pRepr, k ) SC_RingForEachCell( pRepr, pCell, i ) nLength = Abc_MaxInt( nLength, strlen(pRepr->pName) ); // print cells SC_LibForEachCellClass( p, pRepr, k ) { printf( "Class%3d : ", k ); printf( "Ins = %d ", pRepr->n_inputs ); printf( "Outs = %d", pRepr->n_outputs ); for ( i = 0; i < pRepr->n_outputs; i++ ) { printf( " " ); Kit_DsdPrintFromTruth( (unsigned *)Vec_WrdArray(SC_CellPin(pRepr, pRepr->n_inputs+i)->vFunc), pRepr->n_inputs ); } printf( "\n" ); SC_RingForEachCell( pRepr, pCell, i ) { Abc_SclComputeParametersCell( p, pCell, Slew, &ED, &PD ); printf( " %3d ", i+1 ); printf( "%s", pCell->fSkip ? "s" : " " ); printf( " : " ); printf( "%-*s ", nLength, pCell->pName ); printf( "%2d ", pCell->drive_strength ); printf( "A =%8.2f ", pCell->area ); printf( "D =%6.0f ps ", 0.01 * ED * Gain + PD ); printf( "ED =%6.0f ps ", ED ); printf( "PD =%6.0f ps ", PD ); printf( "C =%5.1f ff ", Abc_SclGatePinCapAve(p, pCell) ); printf( "Lm =%5.1f ff ", 0.01 * Gain * Abc_SclGatePinCapAve(p, pCell) ); // printf( "MaxS =%5.1f ps ", SC_CellPin(pCell, pCell->n_inputs)->max_out_slew ); printf( "Lm2 =%5.0f ff ", SC_CellPin(pCell, pCell->n_inputs)->max_out_cap ); printf( "\n" ); } } } /**Function************************************************************* Synopsis [Derive GENLIB library.] Description [] SideEffects [] SeeAlso [] ***********************************************************************/ Vec_Str_t * Abc_SclDeriveGenlibStr( SC_Lib * p, float Slew, float Gain, int nGatesMin, int * pnCellCount ) { extern char * Abc_SclFindGateFormula( char * pGateName, char * pOutName ); char Buffer[200]; Vec_Str_t * vStr; SC_Cell * pRepr; SC_Pin * pPin; int i, k, Count = 2; Abc_SclMarkSkippedCells( p ); vStr = Vec_StrAlloc( 1000 ); Vec_StrPrintStr( vStr, "GATE _const0_ 0.00 z=CONST0;\n" ); Vec_StrPrintStr( vStr, "GATE _const1_ 0.00 z=CONST1;\n" ); SC_LibForEachCellClass( p, pRepr, i ) { if ( pRepr->n_outputs > 1 ) continue; if ( Abc_SclClassCellNum(pRepr) < nGatesMin ) continue; assert( strlen(pRepr->pName) < 200 ); Vec_StrPrintStr( vStr, "GATE " ); sprintf( Buffer, "%-16s", pRepr->pName ); Vec_StrPrintStr( vStr, Buffer ); Vec_StrPrintStr( vStr, " " ); // sprintf( Buffer, "%7.2f", Abc_SclComputeAreaClass(pRepr) ); sprintf( Buffer, "%7.2f", pRepr->area ); Vec_StrPrintStr( vStr, Buffer ); Vec_StrPrintStr( vStr, " " ); Vec_StrPrintStr( vStr, SC_CellPinName(pRepr, pRepr->n_inputs) ); Vec_StrPrintStr( vStr, "=" ); // Vec_StrPrintStr( vStr, SC_CellPinOutFunc(pRepr, 0) ); Vec_StrPrintStr( vStr, Abc_SclFindGateFormula(pRepr->pName, SC_CellPinName(pRepr, pRepr->n_inputs)) ); Vec_StrPrintStr( vStr, ";\n" ); SC_CellForEachPinIn( pRepr, pPin, k ) { float Delay = Abc_SclComputeDelayClassPin( p, pRepr, k, Slew, Gain ); assert( Delay > 0 ); Vec_StrPrintStr( vStr, " PIN " ); sprintf( Buffer, "%-4s", pPin->pName ); Vec_StrPrintStr( vStr, Buffer ); sprintf( Buffer, " UNKNOWN 1 999 %7.2f 0.00 %7.2f 0.00\n", Delay, Delay ); Vec_StrPrintStr( vStr, Buffer ); } Count++; } Vec_StrPrintStr( vStr, "\n.end\n" ); Vec_StrPush( vStr, '\0' ); // printf( "%s", Vec_StrArray(vStr) ); // printf( "GENLIB library with %d gates is produced.\n", Count ); if ( pnCellCount ) *pnCellCount = Count; return vStr; } void Abc_SclDumpGenlib( char * pFileName, SC_Lib * p, float Slew, float Gain, int nGatesMin ) { char FileName[1000]; int nCellCount = 0; Vec_Str_t * vStr; FILE * pFile; if ( pFileName == NULL ) sprintf( FileName, "%s_s%03d_g%03d_m%d.genlib", p->pName, (int)Slew, (int)Gain, nGatesMin ); else sprintf( FileName, "%s", pFileName ); pFile = fopen( FileName, "wb" ); if ( pFile == NULL ) { printf( "Cannot open file \"%s\" for writing.\n", FileName ); return; } vStr = Abc_SclDeriveGenlibStr( p, Slew, Gain, nGatesMin, &nCellCount ); fprintf( pFile, "%s", Vec_StrArray(vStr) ); Vec_StrFree( vStr ); fclose( pFile ); printf( "Written GENLIB library with %d gates into file \"%s\".\n", nCellCount, FileName ); } void Mio_SclDeriveGenlib( void * pScl, float Slew, float Gain, int nGatesMin ) { int nGateCount = 0; Vec_Str_t * vStr = Abc_SclDeriveGenlibStr( (SC_Lib *)pScl, Slew, Gain, nGatesMin, &nGateCount ); Vec_Str_t * vStr2 = Vec_StrDup( vStr ); int RetValue = Mio_UpdateGenlib2( vStr, vStr2, ((SC_Lib *)pScl)->pName, 0 ); Vec_StrFree( vStr ); Vec_StrFree( vStr2 ); if ( RetValue ) printf( "Internally derived GENLIB library \"%s\" with %d gates.\n", ((SC_Lib *)pScl)->pName, nGateCount ); else printf( "Reading library has filed.\n" ); } //////////////////////////////////////////////////////////////////////// /// END OF FILE /// //////////////////////////////////////////////////////////////////////// ABC_NAMESPACE_IMPL_END