O correct bug : don't always create an oaScalarInst, when the instance already exist (saved to disk and reading), just find it.
O integration of old parser O and more, more cleaning.
This commit is contained in:
parent
2ac25af062
commit
2b78ecef1a
|
@ -42,6 +42,7 @@
|
|||
crlcore/COptions.h
|
||||
crlcore/XmlParser.h
|
||||
crlcore/GdsDriver.h
|
||||
crlcore/OAParser.h
|
||||
crlcore/OADriver.h
|
||||
crlcore/CifDriver.h
|
||||
crlcore/SearchPath.h
|
||||
|
@ -75,7 +76,7 @@
|
|||
COptions.cpp
|
||||
XmlParser.cpp
|
||||
GdsDriver.cpp
|
||||
OADriver.cpp
|
||||
OAParserDriver.cpp
|
||||
CifDriver.cpp
|
||||
SearchPath.cpp
|
||||
Environment.cpp
|
||||
|
@ -133,7 +134,8 @@
|
|||
if ( OA_FOUND )
|
||||
include_directories ( ${OA_INCLUDE_DIR} )
|
||||
endif ( OA_FOUND )
|
||||
set (openaccess_cpps openaccess/OpenAccessDriver.cpp )
|
||||
set (openaccess_cpps openaccess/OpenAccessDriver.cpp
|
||||
openaccess/OpenAccessParser.cpp )
|
||||
|
||||
set ( VstParserSourceDir ${CRLCORE_SOURCE_DIR}/src/ccore/alliance/vst )
|
||||
set ( VstParserBinaryDir ${CRLCORE_BINARY_DIR}/src/ccore/alliance/vst )
|
||||
|
|
|
@ -1,15 +0,0 @@
|
|||
// -*- C++ -*-
|
||||
|
||||
#include "hurricane/Cell.h"
|
||||
using namespace Hurricane;
|
||||
|
||||
#include "crlcore/OADriver.h"
|
||||
#include "openaccess/OpenAccessDriver.h"
|
||||
|
||||
namespace CRL {
|
||||
OADriver::OADriver(Cell* cell) : _cell(cell) {}
|
||||
|
||||
void OADriver::save(const string& filePath) {
|
||||
CRL::OpenAccessWrapper::oaDriver(filePath, _cell);
|
||||
}
|
||||
}// namespace CRL
|
|
@ -0,0 +1,25 @@
|
|||
// -*- C++ -*-
|
||||
|
||||
#include "hurricane/Cell.h"
|
||||
using namespace Hurricane;
|
||||
|
||||
#include "crlcore/OADriver.h"
|
||||
#include "crlcore/OAParser.h"
|
||||
#include "openaccess/OpenAccess.h"
|
||||
|
||||
namespace CRL {
|
||||
OADriver::OADriver(Cell* cell) : _cell(cell) {}
|
||||
|
||||
void OADriver::save(const string& filePath) {
|
||||
CRL::OpenAccess::oaDriver(filePath, _cell);
|
||||
}
|
||||
|
||||
OAParser::OAParser(const string& libPath,const string& cellName)
|
||||
: _libPath(libPath), _cellName(cellName) {}
|
||||
|
||||
Cell* OAParser::open() {
|
||||
//dummy for now
|
||||
return NULL;
|
||||
}
|
||||
|
||||
}// namespace CRL
|
|
@ -62,7 +62,7 @@
|
|||
# include "Spice.h"
|
||||
# include "Bookshelf.h"
|
||||
# include "LefDef.h"
|
||||
# include "openaccess/OpenAccessDriver.h"
|
||||
# include "openaccess/OpenAccess.h"
|
||||
|
||||
|
||||
namespace {
|
||||
|
@ -219,8 +219,8 @@ namespace CRL {
|
|||
registerSlot ( "spi", (CellParser_t*)spiceParser , "spi" );
|
||||
registerSlot ( "def", (CellParser_t*)defParser , "def" );
|
||||
registerSlot ( "def", (LibraryParser_t*)lefParser , "lef" );
|
||||
// registerSlot ( "oa", (CellParser_t*)OpenAccessWrapper::oaDesignLoader , "oa" );
|
||||
// registerSlot ( "oa", (LibraryParser_t*)OpenAccessWrapper::oaLibLoader , "oa" );
|
||||
registerSlot ( "oa", (CellParser_t*)OpenAccess::oaCellParser , "oa" );
|
||||
registerSlot ( "oa", (LibraryParser_t*)OpenAccess::oaLibParser , "oa" );
|
||||
registerSlot ( "aux", (CellParser_t*)bookshelfParser, "aux" );
|
||||
}
|
||||
|
||||
|
@ -334,7 +334,7 @@ namespace CRL {
|
|||
registerSlot ( "def", (CellDriver_t*)defDriver , "def" );
|
||||
registerSlot ( "aux", (CellDriver_t*)bookshelfDriver, "test.aux" );
|
||||
registerSlot ( "spi", (CellDriver_t*)spiceDriver , "spi" );
|
||||
registerSlot ( "oa" , (CellDriver_t*)OpenAccessWrapper::oaDriver, "");
|
||||
registerSlot ( "oa" , (CellDriver_t*)OpenAccess::oaDriver, "oa");
|
||||
}
|
||||
|
||||
|
||||
|
|
|
@ -11,12 +11,11 @@ namespace Hurricane {
|
|||
|
||||
namespace CRL {
|
||||
class OADriver {
|
||||
private:
|
||||
Hurricane::Cell* _cell;
|
||||
public:
|
||||
OADriver(Hurricane::Cell*);
|
||||
void save(const std::string& path);
|
||||
|
||||
private:
|
||||
Hurricane::Cell* _cell;
|
||||
};
|
||||
} // End of CRL namespace.
|
||||
|
||||
|
|
|
@ -0,0 +1,23 @@
|
|||
// -*- C++ -*-
|
||||
|
||||
#ifndef __OA_PARSER_H__
|
||||
#define __OA_PARSER_H__
|
||||
|
||||
#include <string>
|
||||
|
||||
namespace Hurricane {
|
||||
class Cell;
|
||||
}
|
||||
|
||||
namespace CRL {
|
||||
class OAParser {
|
||||
private:
|
||||
std::string _libPath;
|
||||
std::string _cellName;
|
||||
public:
|
||||
OAParser(const string& libPath,const string& cellName);
|
||||
Hurricane::Cell* open();
|
||||
};
|
||||
} // End of CRL namespace.
|
||||
|
||||
#endif//__OA_PARSER_H__
|
|
@ -1,5 +1,5 @@
|
|||
// -*-compile-command:"cd ../../../../.. && make"-*-
|
||||
// Time-stamp: "2010-07-07 13:37:14" - OpenAccessWrapper.h
|
||||
// Time-stamp: "2010-07-26 16:16:41" - OpenAccess.h
|
||||
// x-----------------------------------------------------------------x
|
||||
// | This file is part of the hurricaneAMS Software. |
|
||||
// | Copyright (c) UPMC/LIP6 2008-2010, All Rights Reserved |
|
||||
|
@ -11,8 +11,8 @@
|
|||
// | E-mail : Jean-Manuel.Caba@asim.lip6.fr |
|
||||
// x-----------------------------------------------------------------x
|
||||
|
||||
#ifndef __OPENACCESSWRAPPER_H__
|
||||
#define __OPENACCESSWRAPPER_H__
|
||||
#ifndef __OPENACCESS_H__
|
||||
#define __OPENACCESS_H__
|
||||
|
||||
#include <string>
|
||||
|
||||
|
@ -21,11 +21,23 @@ namespace Hurricane {
|
|||
class Cell;
|
||||
}
|
||||
|
||||
namespace CRL {
|
||||
class OpenAccessWrapper {
|
||||
public:
|
||||
static void oaDriver(const std::string& cellPath, Cell* cell);
|
||||
};
|
||||
#ifdef HAVE_OPENACCESS
|
||||
#include "oa/oaDesignDB.h"
|
||||
using namespace oa;
|
||||
#else
|
||||
namespace oa {
|
||||
class oaCell;
|
||||
class oaLib;
|
||||
}
|
||||
#endif
|
||||
|
||||
#endif /*__OPENACCESSWRAPPER_H__ */
|
||||
namespace CRL {
|
||||
class OpenAccess {
|
||||
public:
|
||||
static Hurricane::Cell* oaCellParser(oa::oaCell* cell);
|
||||
static Hurricane::Library* oaLibParser(oa::oaLib* lib);
|
||||
static oa::oaCell* oaDriver(const std::string& libPath, Hurricane::Cell* cell);
|
||||
};
|
||||
}//namespace CRL
|
||||
|
||||
#endif /*__OPENACCESS_H__ */
|
|
@ -0,0 +1,624 @@
|
|||
// -*-compile-command:"cd ../../../../.. && make"-*-
|
||||
// Time-stamp: "2010-07-26 16:09:36" - OpenAccessCommon.h
|
||||
// x-----------------------------------------------------------------x
|
||||
// | This file is part of the hurricaneAMS Software. |
|
||||
// | Copyright (c) UPMC/LIP6 2008-2010, All Rights Reserved |
|
||||
// | =============================================================== |
|
||||
// | Author : Chistophe Alexandre |
|
||||
// | E-mail : Christophe.Alexandre@asim.lip6.fr |
|
||||
// x-----------------------------------------------------------------x
|
||||
// | Author : Jean-Manuel Caba |
|
||||
// | E-mail : Jean-Manuel.Caba@asim.lip6.fr |
|
||||
// x-----------------------------------------------------------------x
|
||||
#ifndef __OPENACCESSCOMMON_H__
|
||||
#define __OPENACCESSCOMMON_H__
|
||||
|
||||
#include <iostream>
|
||||
using namespace std;
|
||||
|
||||
#ifdef HAVE_OPENACCESS
|
||||
#include "hurricane/Cell.h"
|
||||
#include "hurricane/BasicLayer.h"
|
||||
using namespace Hurricane;
|
||||
|
||||
#include "oa/oaDesignDB.h"
|
||||
using namespace oa;
|
||||
|
||||
#undef assert
|
||||
#define assert(cond) if (! (cond) ) throw Error("assertion failed : " + string( #cond ) )
|
||||
//#define assert(cond)
|
||||
|
||||
namespace {
|
||||
|
||||
/**
|
||||
giving a oaDesign pointer that should be not NULL
|
||||
return the oaString representing its name
|
||||
*/
|
||||
inline oaString getDesignName(oaDesign* design) {
|
||||
cerr << "getDesignName" << endl;
|
||||
assert(design);
|
||||
oaNativeNS ns;
|
||||
oaString libName, cellName, viewName;
|
||||
design->getLibName(ns, libName);
|
||||
design->getCellName(ns, cellName);
|
||||
design->getViewName(ns, viewName);
|
||||
oaString designName = "<" + libName + "," + cellName + "," + viewName + ">";
|
||||
return designName;
|
||||
}
|
||||
|
||||
/**
|
||||
giving a oaBlock
|
||||
print the connectivity, mainly used for debug purpose ...
|
||||
@todo remove when not needed anymore
|
||||
*/
|
||||
inline void printBlockTerms(oaBlock* block) {
|
||||
cerr << "printBlockTerms" << endl;
|
||||
assert(block);
|
||||
oaNativeNS ns;
|
||||
oaDesign* design = block->getDesign();
|
||||
cerr << " o Printing " << getDesignName(design) << " terms" << endl;
|
||||
oaIter<oaTerm> termIter(block->getTerms());
|
||||
while (oaTerm* term = termIter.getNext()) {
|
||||
oaString termName;
|
||||
term->getName(ns, termName);
|
||||
cerr << " - " << termName << endl;
|
||||
}
|
||||
cerr << endl;
|
||||
}
|
||||
|
||||
/**
|
||||
Convert material from Hurricane to OA ...
|
||||
@todo verify
|
||||
*/
|
||||
inline oaMaterial getOAMaterial(const BasicLayer::Material& material) {
|
||||
cerr << "getOAMaterial" << endl;
|
||||
switch ( material.getCode() ) {
|
||||
case BasicLayer::Material::nWell: return oacNWellMaterial;
|
||||
case BasicLayer::Material::pWell: return oacPWellMaterial;
|
||||
case BasicLayer::Material::nImplant: return oacNImplantMaterial;
|
||||
case BasicLayer::Material::pImplant: return oacPImplantMaterial;
|
||||
case BasicLayer::Material::active: return oacOtherMaterial;//is it OK?
|
||||
case BasicLayer::Material::poly: return oacPolyMaterial;
|
||||
case BasicLayer::Material::cut: return oacCutMaterial;
|
||||
case BasicLayer::Material::metal: return oacMetalMaterial;
|
||||
case BasicLayer::Material::blockage:
|
||||
//there is no blockage type but a specific oaLayerBlockage class
|
||||
return oacOtherMaterial;
|
||||
case BasicLayer::Material::other: return oacOtherMaterial;
|
||||
default:
|
||||
throw Error("Unrecognized material");
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
Convertion helper for Net convertion ...
|
||||
@todo verify
|
||||
*/
|
||||
inline oaTermType getOATermType(const Net::Direction& direction) {
|
||||
cerr << "getOATermType" << endl;
|
||||
switch (direction) {
|
||||
case Net::Direction::IN:
|
||||
return oacInputTermType;
|
||||
case Net::Direction::OUT:
|
||||
return oacOutputTermType;
|
||||
case Net::Direction::INOUT:
|
||||
return oacInputOutputTermType;
|
||||
case Net::Direction::TRISTATE:
|
||||
return oacTristateTermType;
|
||||
case Net::Direction::UNDEFINED:
|
||||
return oacUnusedTermType;// is it OK ?
|
||||
default:
|
||||
throw Error("Unrecognized direction");
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
Convertion helper for Net convertion ...
|
||||
@todo verify
|
||||
*/
|
||||
inline oaSigType getOASigType(const Net::Type& type) {
|
||||
cerr << "getOASigType" << endl;
|
||||
switch (type.getCode()) {
|
||||
case Net::Type::LOGICAL:
|
||||
return oacSignalSigType;
|
||||
case Net::Type::CLOCK:
|
||||
return oacClockSigType;
|
||||
case Net::Type::POWER:
|
||||
return oacPowerSigType;
|
||||
case Net::Type::GROUND:
|
||||
return oacGroundSigType;
|
||||
case Net::Type::UNDEFINED:
|
||||
return oacAnalogSigType;// is it OK ?
|
||||
default:
|
||||
throw Error("Unrecognized net type");
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
Convertion helper ...
|
||||
*/
|
||||
inline oaOrient getOAOrientFromOrientation(const Transformation::Orientation& orientation) {
|
||||
cerr << "getOAOrientFromOrientation" << endl;
|
||||
switch (orientation) {
|
||||
case Transformation::Orientation::ID:
|
||||
return oacR0;
|
||||
case Transformation::Orientation::R1:
|
||||
return oacR90;
|
||||
case Transformation::Orientation::R2:
|
||||
return oacR180;
|
||||
case Transformation::Orientation::R3:
|
||||
return oacR270;
|
||||
case Transformation::Orientation::MX:
|
||||
return oacMX;
|
||||
case Transformation::Orientation::XR:
|
||||
return oacMXR90;
|
||||
case Transformation::Orientation::MY:
|
||||
return oacMY;
|
||||
case Transformation::Orientation::YR:
|
||||
return oacMYR90;
|
||||
default:
|
||||
throw Error("Unrecognized orientation");
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
Convertion helper ...
|
||||
*/
|
||||
inline void getOATransformFromTransformation(oaTransform& transform, const Transformation& transformation) {
|
||||
cerr << "getOATransformFromTransformation" << endl;
|
||||
transform.set(transformation.getTx(),
|
||||
transformation.getTy(),
|
||||
getOAOrientFromOrientation(transformation.getOrientation()));
|
||||
}
|
||||
|
||||
/**
|
||||
Convertion helper ...
|
||||
*/
|
||||
inline void getOABoxForBox(oaBox& box, const Box& hbox) {
|
||||
cerr << "getOABoxForBox" << endl;
|
||||
box.set(hbox.getXMin(), hbox.getYMin(), hbox.getXMax(), hbox.getYMax());
|
||||
}
|
||||
|
||||
/**
|
||||
Create InstTerm representing connection of nets ...
|
||||
always return a non NULL value
|
||||
*/
|
||||
inline oaInstTerm* getInstTerm(oaInst* inst, Plug* plug,oaNet* net) {
|
||||
cerr << "getInstTerm" << endl;
|
||||
assert(inst);
|
||||
assert(plug);
|
||||
oaNativeNS ns;
|
||||
oaScalarName scPlugName(ns, getString(plug->getMasterNet()->getName()).c_str());
|
||||
oaName instTermName(scPlugName);
|
||||
oaInstTerm* instTerm = oaInstTerm::find(inst, instTermName);
|
||||
if (instTerm) {
|
||||
return instTerm;
|
||||
}
|
||||
oaDesign* design = inst->getMaster();
|
||||
assert(design);
|
||||
oaBlock* masterBlock = design->getTopBlock();
|
||||
oaTerm* term = oaTerm::find(masterBlock, instTermName);
|
||||
assert(term);
|
||||
cerr << "looking for " << plug->getName() << endl;
|
||||
printBlockTerms(masterBlock);
|
||||
cerr << "oaInstTerm::create" << endl;
|
||||
instTerm = oaInstTerm::create(net, inst, term);
|
||||
assert(instTerm);
|
||||
return instTerm;
|
||||
}
|
||||
|
||||
/**
|
||||
save design stored in a map
|
||||
*/
|
||||
inline void saveDesignsInMap(map<const Cell*, oaDesign*> cell2OAdesign){
|
||||
for (map<const Cell*, oaDesign*>::iterator it = cell2OAdesign.begin();
|
||||
it != cell2OAdesign.end();
|
||||
++it) {
|
||||
cerr << it->first << endl;
|
||||
oaDesign* design = it->second;
|
||||
design->save();
|
||||
design->close();
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
print the oaLayera in a oaTech ...
|
||||
*/
|
||||
inline void printOALayers(oaTech* theOATech){
|
||||
cerr << "printOALayers" << endl;
|
||||
assert(theOATech);
|
||||
oaIter<oaLayer> lIter(theOATech->getLayers());
|
||||
while(oaLayer* l = lIter.getNext()){
|
||||
oaString layerName;
|
||||
l->getName(layerName);
|
||||
cerr << " o created oaLayer " << layerName << endl;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
@todo complete,verify ...
|
||||
*/
|
||||
inline BasicLayer::Material::Code oaMaterialToBasicLayerType(const oaMaterial& material) {
|
||||
switch(material) {
|
||||
case oacNWellMaterial:
|
||||
return BasicLayer::Material::nWell;
|
||||
case oacPWellMaterial:
|
||||
return BasicLayer::Material::pWell;
|
||||
case oacNImplantMaterial:
|
||||
return BasicLayer::Material::nImplant;
|
||||
case oacPImplantMaterial:
|
||||
return BasicLayer::Material::pImplant;
|
||||
case oacPolyMaterial:
|
||||
return BasicLayer::Material::poly;
|
||||
case oacCutMaterial:
|
||||
return BasicLayer::Material::cut;
|
||||
case oacMetalMaterial:
|
||||
case oacContactlessMetalMaterial:
|
||||
return BasicLayer::Material::metal;
|
||||
default:
|
||||
return BasicLayer::Material::other;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
generate info from library name
|
||||
*/
|
||||
inline pair<oaScalarName,string> libInfos(const string& path,
|
||||
const string& libName){
|
||||
oaNativeNS ns;
|
||||
const char* strNameLib = libName.c_str();
|
||||
oaScalarName scNameLib(ns, strNameLib);
|
||||
string strPathLib = path + '/' + strNameLib;
|
||||
return make_pair<oaScalarName,string>(scNameLib,strPathLib);
|
||||
}
|
||||
|
||||
/**
|
||||
open oaLib with the info gathered by libPath function
|
||||
*/
|
||||
inline oaLib* openOALib(const pair<oaScalarName,string>& infos){
|
||||
oaLib *lib = oaLib::find(infos.first);
|
||||
const char* pathLib = infos.second.c_str();
|
||||
if (!lib) {
|
||||
if (oaLib::exists(pathLib)){
|
||||
lib = oaLib::open(infos.first, pathLib);
|
||||
}else{
|
||||
string cmd = "mkdir -p "+ infos.second;
|
||||
system(cmd.c_str());
|
||||
}
|
||||
if(!lib){
|
||||
lib = oaLib::create(infos.first, pathLib);
|
||||
}
|
||||
}
|
||||
return lib;
|
||||
}
|
||||
|
||||
inline void createCDS(const pair<oaScalarName,string>& infos){
|
||||
try{
|
||||
cerr << "Overwriting cds.lib file begin" << endl;
|
||||
string cdsPath = infos.second + "/cds.lib";
|
||||
oaLibDefList* ldl = oaLibDefList::get( cdsPath.c_str(), 'a');
|
||||
assert(ldl);
|
||||
assert(ldl->isValid());
|
||||
if(!oaLibDef::find(ldl, infos.first))
|
||||
oaLibDef::create(ldl, infos.first, infos.second.c_str());
|
||||
ldl->save();
|
||||
ldl->destroy();//claim memory
|
||||
ldl = NULL;
|
||||
cerr << "Overwrited cds.lib file end" << endl;
|
||||
}catch(oaException& e){
|
||||
cerr << "ERROR cds: " << e.getMsg() << endl;
|
||||
exit(-2);
|
||||
}
|
||||
}
|
||||
|
||||
inline oaCell* OADesignToOACell(oaDesign* design){
|
||||
assert(design);
|
||||
oaScalarName cellName;
|
||||
design->getCellName(cellName);
|
||||
oaLib* lib = design->getLib();
|
||||
oaBoolean gotAccess = false;
|
||||
gotAccess = lib->getAccess(oacReadLibAccess);
|
||||
oaCell* cell = oaCell::find(lib,cellName);
|
||||
if(gotAccess)
|
||||
lib->releaseAccess();
|
||||
assert(cell);
|
||||
return cell;
|
||||
}
|
||||
|
||||
//multiplicator of BBOX coordinates
|
||||
static int convertFactor = 1;
|
||||
//for namespaces in Hurricane
|
||||
static const Name OACellLibrariesName("OACellLibraries");
|
||||
static const Name OADesignLibrariesName("OADesignLibraries");
|
||||
|
||||
inline Library* findLibraryByNameInLibrary(const Library* rootLibrary, const Name& libraryName) {
|
||||
for_each_library(library, rootLibrary->getLibraries()) {
|
||||
if (library->getName() == libraryName) {
|
||||
return library;
|
||||
}
|
||||
Library* foundLibrary = findLibraryByNameInLibrary(library, libraryName);
|
||||
if (foundLibrary) {
|
||||
return foundLibrary;
|
||||
}
|
||||
end_for;
|
||||
}
|
||||
return NULL;
|
||||
}
|
||||
|
||||
inline Library* findLibraryByNameInDB(const DataBase* db, const Name& libraryName) {
|
||||
Library* rootLibrary = db->getRootLibrary();
|
||||
if (rootLibrary->getName() == libraryName) {
|
||||
return rootLibrary;
|
||||
}
|
||||
return findLibraryByNameInLibrary(rootLibrary, libraryName);
|
||||
}
|
||||
|
||||
inline void getAllCells(Library* rootLibrary, set<Cell*>& cellSet) {
|
||||
for_each_cell(cell, rootLibrary->getCells()) {
|
||||
cellSet.insert(cell);
|
||||
end_for;
|
||||
}
|
||||
for_each_library(library, rootLibrary->getLibraries()) {
|
||||
getAllCells(library, cellSet);
|
||||
end_for;
|
||||
}
|
||||
}
|
||||
|
||||
inline Cell* findCellInLibraries(const Library* rootLibrary, const Name& cellName) {
|
||||
for_each_cell(cell, rootLibrary->getCells()) {
|
||||
if (cell->getName() == cellName) {
|
||||
return cell;
|
||||
}
|
||||
end_for;
|
||||
}
|
||||
for_each_library(library, rootLibrary->getLibraries()) {
|
||||
Cell* cell = findCellInLibraries(library, cellName);
|
||||
if (cell) {
|
||||
return cell;
|
||||
}
|
||||
end_for;
|
||||
}
|
||||
return NULL;
|
||||
}
|
||||
|
||||
inline Library* getRootLibrary() {
|
||||
DataBase* db = DataBase::getDB();
|
||||
if (!db) {
|
||||
return NULL;
|
||||
}
|
||||
return db->getRootLibrary();
|
||||
}
|
||||
|
||||
/**
|
||||
function helper
|
||||
*/
|
||||
inline Library* getOACellLibraries() {
|
||||
Library* rootLibrary = getRootLibrary();
|
||||
if (!rootLibrary) {
|
||||
return NULL;
|
||||
}
|
||||
for_each_library(library, rootLibrary->getLibraries()) {
|
||||
if (library->getName() == OACellLibrariesName) {
|
||||
return library;
|
||||
}
|
||||
end_for;
|
||||
}
|
||||
return NULL;
|
||||
}
|
||||
|
||||
/**
|
||||
helper function
|
||||
*/
|
||||
inline Library* getOADesignLibraries() {
|
||||
Library* rootLibrary = getRootLibrary();
|
||||
if (!rootLibrary) {
|
||||
return NULL;
|
||||
}
|
||||
for_each_library(library, rootLibrary->getLibraries()) {
|
||||
if (library->getName() == OADesignLibrariesName) {
|
||||
return library;
|
||||
}
|
||||
end_for;
|
||||
}
|
||||
return NULL;
|
||||
}
|
||||
|
||||
/**
|
||||
utility to open a design by name
|
||||
*/
|
||||
inline oaView* pickView(oaCell* oa_Cell) {
|
||||
//oacMaskLayout Type is first
|
||||
oaView* toReturnView = NULL;
|
||||
|
||||
oaCollection<oaCellView, oaCell> cellViews = oa_Cell->getCellViews();
|
||||
oaIter<oaCellView> cellViewIter(cellViews);
|
||||
|
||||
while (oaCellView* cellView = cellViewIter.getNext()) {
|
||||
oaView* view = cellView->getView();
|
||||
oaViewType* viewType = view->getViewType();
|
||||
oaScalarName viewName;
|
||||
view->getName(viewName);
|
||||
if (viewType == oaViewType::get(oacMaskLayout)) {
|
||||
return view;
|
||||
}
|
||||
if (viewType == oaViewType::get(oacSchematic) ||
|
||||
viewType == oaViewType::get(oacNetlist) ||
|
||||
viewType == oaViewType::get(oacSchematicSymbol)) {
|
||||
toReturnView = view;
|
||||
}
|
||||
}
|
||||
return toReturnView;
|
||||
}
|
||||
|
||||
/**
|
||||
utility to open a design by name
|
||||
*/
|
||||
inline oaDesign* openDesign(const oaNameSpace& oaNS, oaCell* oa_Cell) {
|
||||
oaView* view = pickView(oa_Cell);
|
||||
if (view != NULL) {
|
||||
oaScalarName libName;
|
||||
oaLib* cellLib = oa_Cell->getLib();
|
||||
cellLib->getName(libName);
|
||||
oaScalarName cellName;
|
||||
oa_Cell->getName(cellName);
|
||||
oaViewType* viewType = view->getViewType();
|
||||
oaScalarName viewName;
|
||||
view->getName(viewName);
|
||||
return oaDesign::open(libName, cellName, viewName, viewType, 'r');
|
||||
}
|
||||
return NULL;
|
||||
}
|
||||
|
||||
void loadOACellInCell(oaCell* oa_Cell, Cell* cell) {
|
||||
oaNativeNS oaNS;
|
||||
oaDesign* cellDesign = openDesign(oaNS, oa_Cell);
|
||||
|
||||
if (cellDesign != NULL) {
|
||||
oaModule* module = cellDesign->getTopModule();
|
||||
oaCollection<oaModInst, oaModule> oaModInsts = module->getInsts();
|
||||
oaIter<oaModInst> modInstIter(oaModInsts);
|
||||
while (oaModInst* modInst = modInstIter.getNext()) {
|
||||
oaString oaModInstStr;
|
||||
modInst->getName(oaNS, oaModInstStr);
|
||||
//cerr << "inst : " << oaModInstStr << endl;
|
||||
oaModule* masterModule = modInst->getMasterModule();
|
||||
if (masterModule) {
|
||||
oaString oaModuleStr;
|
||||
masterModule->getName(oaNS, oaModuleStr);
|
||||
//cerr << "master : " << oaModuleStr << endl;
|
||||
//find hurricane Cell
|
||||
Cell* masterCell = findCellInLibraries(getRootLibrary(), Name(oaModuleStr));
|
||||
if (!masterCell) {
|
||||
cout << "\n***Quitting. Cannot get MasterCell :" ;
|
||||
cout << oaModuleStr << endl;
|
||||
exit(8);
|
||||
}
|
||||
Instance* instance = Instance::create(cell, Name(oaModInstStr) ,masterCell);
|
||||
//cerr << instance << endl;
|
||||
} else {
|
||||
cerr << "master : NULL" << endl;
|
||||
}
|
||||
}//end while
|
||||
|
||||
//now treat nets
|
||||
oaCollection<oaModNet, oaModule> oaModNets = module->getNets();
|
||||
oaIter<oaModNet> oaModNetIter(oaModNets);
|
||||
while (oaModNet* oa_ModNet = oaModNetIter.getNext()) {
|
||||
oaString oaModNetStr;
|
||||
oa_ModNet->getName(oaNS, oaModNetStr);
|
||||
//cerr << oaModNetStr << endl;
|
||||
Net* net = cell->getNet(Name(oaModNetStr));
|
||||
if (!net) {
|
||||
net = Net::create(cell, Name(oaModNetStr));
|
||||
}
|
||||
|
||||
oaCollection<oaModInstTerm, oaModNet> oaModInstTerms = oa_ModNet->getInstTerms();
|
||||
oaIter<oaModInstTerm> oaModInstTermIter(oaModInstTerms);
|
||||
while (oaModInstTerm* oa_ModInstTerm = oaModInstTermIter.getNext()) {
|
||||
oaModInst* modInst = oa_ModInstTerm->getInst();
|
||||
oaString oaModInstStr;
|
||||
modInst->getName(oaNS, oaModInstStr);
|
||||
//find hurricane instance
|
||||
Instance* instance = cell->getInstance(Name(oaModInstStr));
|
||||
if (!instance) {
|
||||
cout << "\n***Quitting. Cannot get Instance :" ;
|
||||
cout << oaModInstStr << endl;
|
||||
exit(8);
|
||||
}
|
||||
oaModTerm* oa_ModTerm = oa_ModInstTerm->getTerm();
|
||||
oaString oaModTermStr;
|
||||
oa_ModTerm->getName(oaNS, oaModTermStr);
|
||||
Net* masterNet = instance->getMasterCell()->getNet(Name(oaModTermStr));
|
||||
if (!masterNet) {
|
||||
cout << "\n***Quitting. Cannot get Master Net :" ;
|
||||
cout << oaModTermStr << endl;
|
||||
exit(8);
|
||||
}
|
||||
Plug* plug = instance->getPlug(masterNet);
|
||||
plug->setNet(net);
|
||||
//cerr << plug << endl;
|
||||
}
|
||||
//cerr << net << endl;
|
||||
}
|
||||
|
||||
|
||||
oaScalarName cellName;
|
||||
oa_Cell->getName(cellName);
|
||||
oaString cellNameString;
|
||||
cellName.get(cellNameString);
|
||||
cell->setName(Name(cellNameString));
|
||||
cell->setTerminal(false);
|
||||
|
||||
//physical part
|
||||
oaBlock* block = cellDesign->getTopBlock();
|
||||
if (block) {
|
||||
oaBox oa_box;
|
||||
block->getBBox(oa_box);
|
||||
Point lowerLeft(DbU::db(oa_box.lowerLeft().x()), DbU::db(oa_box.lowerLeft().y()));
|
||||
Point upperRight(DbU::db(oa_box.upperRight().x()), DbU::db(oa_box.upperRight().y()));
|
||||
cell->setAbutmentBox(Box(lowerLeft, upperRight));
|
||||
|
||||
oaCollection<oaInst, oaBlock> oaInsts = block->getInsts();
|
||||
oaIter<oaInst> oaInstIter(oaInsts);
|
||||
while (oaInst* oa_Inst = oaInstIter.getNext()) {
|
||||
oaString oaInstStr;
|
||||
oa_Inst->getName(oaNS, oaInstStr);
|
||||
Instance* instance = cell->getInstance(Name(oaInstStr));
|
||||
if (instance) {
|
||||
//cerr << "found " << instance << endl;
|
||||
oaPlacementStatus placementStatus= oa_Inst->getPlacementStatus();
|
||||
//switch (placementStatus) {
|
||||
// case oacNonePlacementStatus :
|
||||
// cerr << " none" << endl;
|
||||
// break;
|
||||
// case oacUnplacedPlacementStatus :
|
||||
// cerr << " unplaced" << endl;
|
||||
// break;
|
||||
// case oacPlacedPlacementStatus :
|
||||
// cerr << " placed" << endl;
|
||||
// break;
|
||||
// case oacFixedPlacementStatus :
|
||||
// cerr << " fixed" << endl;
|
||||
// break;
|
||||
// default :
|
||||
// cerr << "other" << endl;
|
||||
//}
|
||||
oaPoint instOrigin;
|
||||
oa_Inst->getOrigin(instOrigin);
|
||||
//cerr << instOrigin.x() << " " << instOrigin.y() << endl;
|
||||
} else {
|
||||
cerr << "cannot find " << oaInstStr << endl;
|
||||
}
|
||||
}
|
||||
oaSitePattern sitePattern;
|
||||
block->getSitePattern(sitePattern);
|
||||
|
||||
for (int i = 0; i < sitePattern.getNumElements(); i++) {
|
||||
const oaSiteRef& siteRef = sitePattern.get(i);
|
||||
cerr << "site : " << siteRef.siteName() << endl;
|
||||
}
|
||||
|
||||
oaCollection<oaRow, oaBlock> oaRows = block->getRows();
|
||||
oaIter<oaRow> oaRowIter(oaRows);
|
||||
while (oaRow* oa_Row = oaRowIter.getNext()) {
|
||||
cerr << "row" << endl;
|
||||
oaString siteName;
|
||||
oa_Row->getSiteDefName(siteName);
|
||||
cerr << siteName << endl;
|
||||
}
|
||||
|
||||
oaCollection<oaAreaBoundary, oaBlock> oaAreaBoundaries = block->getBoundaries();
|
||||
oaIter<oaBoundary> oaBoundaryIter(oaAreaBoundaries);
|
||||
while (oaBoundary* oa_Boundary = oaBoundaryIter.getNext()) {
|
||||
cerr << "boundary" << endl;
|
||||
}
|
||||
} else {
|
||||
cerr << "no block view " << endl;
|
||||
}
|
||||
}//end if (cellDesign != NULL)
|
||||
}
|
||||
}//end anonymous namespace
|
||||
|
||||
#endif//HAVE_OPENACCESS
|
||||
|
||||
#endif//__OPENACCESSCOMMON_H__
|
|
@ -1,658 +0,0 @@
|
|||
// -*-compile-command:"cd ../../../../.. && make"-*-
|
||||
// Time-stamp: "2010-07-23 16:05:52" - OpenAccessWrapper.cpp
|
||||
// x-----------------------------------------------------------------x
|
||||
// | This file is part of the hurricaneAMS Software. |
|
||||
// | Copyright (c) UPMC/LIP6 2008-2010, All Rights Reserved |
|
||||
// | =============================================================== |
|
||||
// | Author : Chistophe Alexandre |
|
||||
// | E-mail : Christophe.Alexandre@asim.lip6.fr |
|
||||
// x-----------------------------------------------------------------x
|
||||
// | Author : Jean-Manuel Caba |
|
||||
// | E-mail : Jean-Manuel.Caba@asim.lip6.fr |
|
||||
// x-----------------------------------------------------------------x
|
||||
|
||||
#include <iostream>
|
||||
#include <algorithm>
|
||||
#include <vector>
|
||||
using namespace std;
|
||||
|
||||
#ifdef HAVE_OPENACCESS
|
||||
#include "oa/oaDesignDB.h"
|
||||
using namespace oa;
|
||||
#endif
|
||||
|
||||
#include "hurricane/DataBase.h"
|
||||
#include "hurricane/Technology.h"
|
||||
#include "hurricane/Library.h"
|
||||
#include "hurricane/Cell.h"
|
||||
#include "hurricane/NetExternalComponents.h"
|
||||
#include "hurricane/Segment.h"
|
||||
#include "hurricane/Pad.h"
|
||||
#include "hurricane/BasicLayer.h"
|
||||
#include "hurricane/Slice.h"
|
||||
using namespace Hurricane;
|
||||
|
||||
#include "OpenAccessDriver.h"
|
||||
#include "OpenAccessUtils.h"
|
||||
|
||||
namespace {
|
||||
|
||||
#ifdef HAVE_OPENACCESS
|
||||
|
||||
/**
|
||||
Class to drive OA to Hurricane
|
||||
*/
|
||||
class OADriver {
|
||||
private:
|
||||
typedef map<const Library*, oaLib*> Library2OALibMap;
|
||||
typedef map<const Cell*, oaDesign*> Cell2OADesignMap;
|
||||
typedef map<Instance*, oaInst*> Instance2OAInstsMap;
|
||||
typedef map<Layer*, oaPhysicalLayer*> Layer2OAPhysicalLayerMap;
|
||||
|
||||
string _path;
|
||||
oaTech* _oaTech;
|
||||
Library2OALibMap _library2OALib;
|
||||
Cell2OADesignMap _cell2OADesign4Netlist;
|
||||
Cell2OADesignMap _cell2OADesign4Schematic;
|
||||
Cell2OADesignMap _cell2OADesign4Symbolic;
|
||||
Cell2OADesignMap _cell2OADesign4Layout;
|
||||
Instance2OAInstsMap _instance2OAInst;
|
||||
Layer2OAPhysicalLayerMap _layer2OAPhysicalLayer;
|
||||
DataBase* _db;
|
||||
Technology* _technology;
|
||||
int _layerID;
|
||||
public:
|
||||
OADriver(const string& path):
|
||||
_path(path),
|
||||
_oaTech(NULL),
|
||||
_library2OALib(),
|
||||
_cell2OADesign4Netlist(),
|
||||
_cell2OADesign4Schematic(),
|
||||
_cell2OADesign4Symbolic(),
|
||||
_cell2OADesign4Layout(),
|
||||
_instance2OAInst(),
|
||||
_layer2OAPhysicalLayer(),
|
||||
_db(NULL),
|
||||
_technology(NULL),
|
||||
_layerID(0) {
|
||||
_db = DataBase::getDB();
|
||||
if (!_db) {
|
||||
throw Error("no database");
|
||||
}
|
||||
_technology = _db->getTechnology();
|
||||
if (!_technology) {
|
||||
throw Error("no technology");
|
||||
}
|
||||
}
|
||||
|
||||
~OADriver() {
|
||||
cerr << "SAVING ALL" << endl;
|
||||
_oaTech->save();
|
||||
_oaTech->close();
|
||||
saveDesignsInMap(_cell2OADesign4Netlist);
|
||||
saveDesignsInMap(_cell2OADesign4Schematic);
|
||||
saveDesignsInMap(_cell2OADesign4Symbolic);
|
||||
saveDesignsInMap(_cell2OADesign4Layout);
|
||||
for (Library2OALibMap::iterator it = _library2OALib.begin();
|
||||
it != _library2OALib.end();
|
||||
++it) {
|
||||
oaLib* lib = it->second;
|
||||
lib->close();
|
||||
}
|
||||
cerr << "ALL SAVED" << endl;
|
||||
}
|
||||
|
||||
/**
|
||||
Create an empty oaLib from a Library
|
||||
no cells are added in this oaLib
|
||||
all sub Library are also converted.
|
||||
*/
|
||||
oaLib* getOALibForLibrary(const Library* library) {
|
||||
cerr << "getOALibForLibrary" << endl;
|
||||
assert(library);
|
||||
Library2OALibMap::iterator it = _library2OALib.find(library);
|
||||
if (it != _library2OALib.end()) {
|
||||
return it->second;
|
||||
}
|
||||
|
||||
// 1) create or open library
|
||||
pair<oaScalarName,string> infos=libInfos(_path,
|
||||
getString(library->getName()));
|
||||
oaLib *lib = openOALib(infos);
|
||||
_library2OALib[library] = lib;
|
||||
|
||||
#if 0
|
||||
// 2) for each cell convert them too : if it's a standard cell library for example
|
||||
for_each_cell(c ,library->getCells()){
|
||||
getOADesignForCell(c);
|
||||
end_for;
|
||||
}
|
||||
|
||||
// 3) also convert each contained library if any
|
||||
for_each_library(l ,library->getLibraries()){
|
||||
getOALibForLibrary(l);
|
||||
end_for;
|
||||
}
|
||||
#endif
|
||||
// 4) create, update library list file
|
||||
createCDS(infos);
|
||||
|
||||
return lib;
|
||||
}
|
||||
|
||||
/**
|
||||
convert oaLayer from a Layer ...
|
||||
*/
|
||||
oaLayer* getOALayerFromLayer(Layer* layer,oaTech* theOATech) {
|
||||
cerr << "getOALayerFromLayer" << endl;
|
||||
assert(layer);
|
||||
Layer2OAPhysicalLayerMap::iterator it = _layer2OAPhysicalLayer.find(layer);
|
||||
if (it != _layer2OAPhysicalLayer.end()) {
|
||||
return it->second;
|
||||
}
|
||||
assert(theOATech);
|
||||
|
||||
// 1) get or create layer
|
||||
oaString layerName = getString(layer->getName()).c_str();
|
||||
oaPhysicalLayer* aOALayer = NULL;
|
||||
aOALayer = oaPhysicalLayer::find(theOATech, layerName, true);
|
||||
if(aOALayer){
|
||||
_layer2OAPhysicalLayer[layer] = aOALayer;
|
||||
return aOALayer;
|
||||
}
|
||||
BasicLayer* bLayer = dynamic_cast<BasicLayer*>(layer);
|
||||
if(bLayer)
|
||||
aOALayer = oaPhysicalLayer::create(theOATech, layerName, _layerID++,getOAMaterial(bLayer->getMaterial()));
|
||||
else
|
||||
aOALayer = oaPhysicalLayer::create(theOATech, layerName, _layerID++);
|
||||
assert(aOALayer);
|
||||
|
||||
_layer2OAPhysicalLayer[layer] = aOALayer;
|
||||
|
||||
//create and add layer constraint for Layer specific manufacturing rules
|
||||
cerr << " o get value for constraint" << endl;
|
||||
long minSize = Hurricane::DbU::getDb(layer->getMinimalSize());
|
||||
long minSpace = Hurricane::DbU::getDb(layer->getMinimalSpacing());
|
||||
long pitch = Hurricane::DbU::getDb(layer->getPitch());
|
||||
|
||||
#if 0
|
||||
cerr << " o create constraint for min size : " << pitch << endl;
|
||||
oaLayerConstraint* cMinSize = NULL;
|
||||
try{
|
||||
cMinSize = oaLayerConstraint::create(aOALayer->getNumber(),
|
||||
oaLayerConstraintDef::get(oacMinSize),
|
||||
oaIntValue::create(theOATech,500))
|
||||
}catch(oaException& e){
|
||||
cerr << "ERROR oaLayer: " << e.getMsg() << endl;
|
||||
exit(-2);
|
||||
}catch(...){
|
||||
cerr << "ERROR oaLayer: [UNKNOWN]" << endl;
|
||||
exit(-1);
|
||||
}
|
||||
assert(cMinSize);
|
||||
cerr << " o create constraint for min space" << endl;
|
||||
oaLayerConstraint* cMinSpace = oaLayerConstraint::create(aOALayer->getNumber(),
|
||||
oaLayerConstraintDef::get(oacMinSpacing),
|
||||
oaIntValue::create(theOATech->getLib(),minSpace));
|
||||
assert(cMinSpace);
|
||||
cerr << " o create constraint for pitchH" << endl;
|
||||
oaLayerConstraint* cPitchH = oaLayerConstraint::create(aOALayer->getNumber(),
|
||||
oaLayerConstraintDef::get(oacHorizontalRouteGridPitch),
|
||||
oaIntValue::create(theOATech->getLib(),pitch));
|
||||
assert(cPitchH);
|
||||
|
||||
cerr << " o create constraint for pitchV" << endl;
|
||||
oaLayerConstraint* cPitchV = oaLayerConstraint::create(aOALayer->getNumber(),
|
||||
oaLayerConstraintDef::get(oacVerticalRouteGridPitch),
|
||||
oaIntValue::create(theOATech->getLib(),pitch));
|
||||
assert(cPitchV);
|
||||
#endif
|
||||
if(bLayer){
|
||||
unsigned gdsIInumber = bLayer->getExtractNumber();
|
||||
}
|
||||
return aOALayer;
|
||||
}
|
||||
|
||||
/**
|
||||
create a oaTech from a Hurricane::Technology in it's Library
|
||||
also create the oaLib corresponding to the Hurricane::Library
|
||||
containing the Hurricane::Technology
|
||||
@todo complete with technology info for layers
|
||||
@see getOALibForLibrary
|
||||
*/
|
||||
oaTech* getOATechForTechnology(const Technology* technology,const Library* lib) {
|
||||
cerr << "createOATechForTechnology" << endl;
|
||||
assert(technology);
|
||||
|
||||
// 1) get or create Library for the techno
|
||||
assert(lib);
|
||||
oaLib* techOAlib = getOALibForLibrary(lib);
|
||||
assert(techOAlib);
|
||||
|
||||
// 2) get or create oaTech container
|
||||
bool created = false;
|
||||
cerr << "oaTech::find" << endl;
|
||||
oaTech* theOATech = oaTech::find(techOAlib);
|
||||
if(!theOATech){
|
||||
if (oaTech::exists(techOAlib)){
|
||||
cerr << "oaTech::open" << endl;
|
||||
theOATech = oaTech::open(techOAlib,'a');
|
||||
}
|
||||
if(!theOATech){
|
||||
cerr << "oaTech::create" << endl;
|
||||
theOATech = oaTech::create(techOAlib);
|
||||
created = true;
|
||||
}
|
||||
}
|
||||
assert(techOAlib);
|
||||
if(created){
|
||||
theOATech->setDefaultManufacturingGrid(10);
|
||||
theOATech->setDBUPerUU(oaViewType::get(oacMaskLayout), 2000);
|
||||
|
||||
//create and add foundry constraint group for General manufacturing rules
|
||||
//and add oaSimpleConstraintType too
|
||||
oaConstraintGroup *cgFoundry = theOATech->getFoundryRules();
|
||||
|
||||
/*
|
||||
add the constraint group LEFDefaultRouteSpec for oa2lef
|
||||
*/
|
||||
|
||||
//TODO: add this layer
|
||||
//first create "utility" layers
|
||||
//strLayerDev ( "device" );
|
||||
//strLayerText( "text" );
|
||||
//strLayerPin ( "pin" );
|
||||
//strLayerWire( "wire" );
|
||||
}
|
||||
|
||||
// get or create physical layer
|
||||
|
||||
for_each_layer(layer, technology->getLayers()) {
|
||||
getOALayerFromLayer(layer,theOATech);
|
||||
end_for;
|
||||
}
|
||||
printOALayers(theOATech);
|
||||
|
||||
return theOATech;
|
||||
}
|
||||
|
||||
/**
|
||||
convert Hurricane::Instance to oaInst ...
|
||||
*/
|
||||
oaInst* getOAInstForInstance(Instance* instance,oaBlock* topBlock) {
|
||||
cerr << "getOAInstForInstance " << instance << endl;
|
||||
assert(instance);
|
||||
Instance2OAInstsMap::iterator it = _instance2OAInst.find(instance);
|
||||
if (it != _instance2OAInst.end()) {
|
||||
return it->second;
|
||||
}
|
||||
assert(topBlock);
|
||||
|
||||
// 1) get the master cell for the instance
|
||||
Cell* masterCell = instance->getMasterCell();
|
||||
assert(masterCell);
|
||||
oaDesign* masterDesign = getOADesignForCell(masterCell);
|
||||
assert(masterDesign);
|
||||
|
||||
oaNativeNS ns;
|
||||
oaScalarName scMasterName;
|
||||
masterDesign->getCellName(scMasterName);
|
||||
oaString strMasterName;
|
||||
scMasterName.get(strMasterName);
|
||||
oaScalarName scInstName(ns, getString(instance->getName()).c_str());
|
||||
|
||||
oaTransform transform;
|
||||
getOATransformFromTransformation(transform, instance->getTransformation());
|
||||
oaScalarInst* blockInst = oaScalarInst::create(topBlock, masterDesign, scInstName, transform);
|
||||
_instance2OAInst[instance] = blockInst;
|
||||
return blockInst;
|
||||
}
|
||||
|
||||
/**
|
||||
convert Hurricane::plug to oaInstTerm
|
||||
and add it to if connected.
|
||||
always return a non NULL value
|
||||
*/
|
||||
oaInstTerm* getOAInstTermFromPlug(Plug* plug,oaNet* net){
|
||||
cerr << "getOAInstTermFromPlug " << plug << endl;
|
||||
assert(plug);
|
||||
Instance* instance = plug->getInstance();
|
||||
Instance2OAInstsMap::iterator it = _instance2OAInst.find(instance);
|
||||
assert(it != _instance2OAInst.end());
|
||||
oaInst* blockInst = it->second;
|
||||
oaInstTerm* instTerm = getInstTerm(blockInst, plug,net);
|
||||
assert(instTerm);
|
||||
return instTerm;
|
||||
}
|
||||
|
||||
oaPin* getOAPinFromNet(Net* net,oaNet* blockNet){
|
||||
cerr << "getOAPinFromNet" << endl;
|
||||
assert(net);
|
||||
assert(net->isExternal());
|
||||
assert(blockNet);
|
||||
oaNativeNS ns;
|
||||
oaScalarName scNetName(ns, getString(net->getName()).c_str());
|
||||
oaTerm* term = oaTerm::create(blockNet, scNetName, getOATermType(net->getDirection()));
|
||||
assert(term);
|
||||
oaPin* pin = oaPin::create(term);
|
||||
assert(pin);
|
||||
|
||||
#if 0
|
||||
oaLayerNum layer;
|
||||
oaPurposeNum purpose;
|
||||
getLayerPurpose(block->getDesign(), "pin", "drawing", layer, purpose);
|
||||
oaPinFig *fig = oaPolygon::create(block, layer, purpose, points);
|
||||
#endif
|
||||
return pin;
|
||||
}
|
||||
|
||||
oaRect* getOARectFromComponent(Component* component,oaBlock* topBlock){
|
||||
cerr << "getOARectFromComponent" << endl;
|
||||
assert(component);
|
||||
assert(topBlock);
|
||||
oaBox box;
|
||||
getOABoxForBox(box, component->getBoundingBox());
|
||||
Layer* layer = (Layer*) component->getLayer();
|
||||
assert(layer);
|
||||
oaPhysicalLayer* physLayer = NULL;
|
||||
Layer2OAPhysicalLayerMap::iterator it = _layer2OAPhysicalLayer.find(layer);
|
||||
if (it != _layer2OAPhysicalLayer.end()) {
|
||||
physLayer = it->second;
|
||||
}
|
||||
assert(physLayer);
|
||||
oaLayerNum layerNum = physLayer->getNumber();
|
||||
oaRect* rect = oaRect::create(topBlock, layerNum,
|
||||
oaPurpose::get(_oaTech, oacDrawingPurposeType)->getNumber(),
|
||||
box);
|
||||
return rect;
|
||||
}
|
||||
|
||||
/**
|
||||
convert Hurricane::Net to oaNet only at
|
||||
the logical abstraction point of view
|
||||
always return a non NULL value
|
||||
*/
|
||||
oaNet* getOANetFromNet(Net* net,oaBlock* topBlock) {
|
||||
cerr << "getOANetFromNet " << net << endl;
|
||||
assert(net);
|
||||
oaNativeNS ns;
|
||||
oaScalarName scNetName(ns, getString(net->getName()).c_str());
|
||||
oaScalarNet* blockNet = NULL;
|
||||
blockNet = oaScalarNet::find(topBlock, scNetName);
|
||||
if(blockNet)
|
||||
return blockNet;
|
||||
assert(!blockNet);
|
||||
blockNet = oaScalarNet::create(topBlock, scNetName, getOASigType(net->getType()));
|
||||
assert(blockNet);
|
||||
if (net->isExternal()) {
|
||||
oaPin* pin = getOAPinFromNet(net,blockNet);
|
||||
Components externalComponents = NetExternalComponents::get(net);
|
||||
for_each_component(component, externalComponents) {
|
||||
oaRect* rect = getOARectFromComponent(component,topBlock);
|
||||
rect->addToPin(pin);
|
||||
end_for;
|
||||
}
|
||||
}
|
||||
cerr << " o transformation of plugs" << endl;
|
||||
for_each_plug(plug, net->getPlugs()) {
|
||||
getOAInstTermFromPlug(plug,blockNet);
|
||||
end_for;
|
||||
}
|
||||
return blockNet;
|
||||
}
|
||||
|
||||
/**
|
||||
create oaRect for slice ...
|
||||
*/
|
||||
void getOARectFromSlice(Slice* slice,oaBlock* topBlock){
|
||||
cerr << "getOARectFromSlice" << endl;
|
||||
assert(slice);
|
||||
assert(topBlock);
|
||||
for_each_component(component, slice->getComponents()) {
|
||||
oaRect* rect = getOARectFromComponent(component,topBlock);
|
||||
end_for;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
Create initial oaDesign from the cell and save it as a netlist view
|
||||
*/
|
||||
oaDesign* createOAasNetlist(const Cell* cell) {
|
||||
cerr << "createNetlist " << cell << endl;
|
||||
assert(cell);
|
||||
Cell2OADesignMap::iterator it = _cell2OADesign4Netlist.find(cell);
|
||||
if (it != _cell2OADesign4Netlist.end()) {
|
||||
return it->second;
|
||||
}
|
||||
|
||||
// 1) get the lib containing the cell
|
||||
oaNativeNS ns;
|
||||
oaLib* lib = getOALibForLibrary(cell->getLibrary());
|
||||
assert(lib);
|
||||
|
||||
// 2) create a netlist CellView of the cell
|
||||
oaScalarName scNameDesign(ns, getString(cell->getName()).c_str());
|
||||
oaScalarName scNameView(ns, "netlist");
|
||||
oaScalarName scNameLib;
|
||||
lib->getName(scNameLib);
|
||||
cerr << "oaDesign::open for netlist view" << endl;
|
||||
oaDesign* designCellView = oaDesign::open(scNameLib, scNameDesign, scNameView, oaViewType::get(oacNetlist), 'a');
|
||||
_cell2OADesign4Netlist[cell] = designCellView;
|
||||
|
||||
// 3) create oaBlock singleton where we will do all the work
|
||||
oaBlock* topBlock = designCellView->getTopBlock();
|
||||
if(!topBlock){
|
||||
cerr << "oaBlock::create for netlist view" << endl;
|
||||
topBlock = oaBlock::create(designCellView);
|
||||
}
|
||||
assert(topBlock);
|
||||
|
||||
// 4) convert each OA object
|
||||
cerr << "transformation of slices" << endl;
|
||||
for_each_slice(slice, cell->getSlices()){
|
||||
getOARectFromSlice(slice,topBlock);
|
||||
end_for;
|
||||
}
|
||||
cerr << "transformation of instances" << endl;
|
||||
for_each_instance(instance, cell->getInstances()){
|
||||
getOAInstForInstance(instance,topBlock);
|
||||
end_for;
|
||||
}
|
||||
cerr << "transformation of nets" << endl;
|
||||
for_each_net(net, cell->getNets()){
|
||||
getOANetFromNet(net,topBlock);
|
||||
end_for;
|
||||
}
|
||||
return designCellView;
|
||||
}
|
||||
|
||||
/**
|
||||
Add symbol view to previous view ...
|
||||
*/
|
||||
oaDesign* addSymbol(const Cell* cell,oaDesign* previous) {
|
||||
cerr << "addSymbol" << cell << endl;
|
||||
assert(cell);
|
||||
assert(previous);
|
||||
Cell2OADesignMap::iterator it = _cell2OADesign4Symbolic.find(cell);
|
||||
if (it != _cell2OADesign4Symbolic.end()) {
|
||||
return it->second;
|
||||
}
|
||||
oaNativeNS ns;
|
||||
oaLib* lib = getOALibForLibrary(cell->getLibrary());
|
||||
assert(lib);
|
||||
oaScalarName scNameDesign(ns, getString(cell->getName()).c_str());
|
||||
oaScalarName scNameView(ns, "symbolic");
|
||||
oaScalarName scNameLib;
|
||||
lib->getName(scNameLib);
|
||||
|
||||
// create a symbolic CellView of the cell
|
||||
oaViewType* vType = oaViewType::get(oacSchematicSymbol);
|
||||
cerr << "oaDesign::open for symbolic view" << endl;
|
||||
oaDesign* designCellView = oaDesign::open(scNameLib, scNameDesign, scNameView, vType, 'a');
|
||||
_cell2OADesign4Symbolic[cell] = designCellView;
|
||||
|
||||
// get module or embed previous module
|
||||
oaModule* topMod = NULL;
|
||||
topMod = designCellView->getTopModule();
|
||||
if(!topMod){
|
||||
topMod = oaModule::embed(designCellView, previous);
|
||||
designCellView->setTopModule(topMod);
|
||||
}
|
||||
oaBlock *topBlock = designCellView->getTopBlock();
|
||||
assert(topBlock);
|
||||
|
||||
return designCellView;
|
||||
}
|
||||
|
||||
/**
|
||||
Add schematic view to previous view ...
|
||||
*/
|
||||
oaDesign* addSchematic(const Cell* cell,oaDesign* previous) {
|
||||
cerr << "addSchematic" << cell << endl;
|
||||
assert(cell);
|
||||
assert(previous);
|
||||
Cell2OADesignMap::iterator it = _cell2OADesign4Schematic.find(cell);
|
||||
if (it != _cell2OADesign4Schematic.end()) {
|
||||
return it->second;
|
||||
}
|
||||
|
||||
oaNativeNS ns;
|
||||
oaLib* lib = getOALibForLibrary(cell->getLibrary());
|
||||
assert(lib);
|
||||
oaScalarName scNameDesign(ns, getString(cell->getName()).c_str());
|
||||
oaScalarName scNameView(ns, "schematic");
|
||||
oaScalarName scNameLib;
|
||||
lib->getName(scNameLib);
|
||||
|
||||
// create a schematic CellView of the cell
|
||||
oaViewType* vType = oaViewType::get(oacSchematic);
|
||||
cerr << "oaDesign::open for schematic view" << endl;
|
||||
oaDesign* designCellView = oaDesign::open(scNameLib, scNameDesign, scNameView, vType, 'a');
|
||||
_cell2OADesign4Schematic[cell] = designCellView;
|
||||
|
||||
// get module or embed previous module
|
||||
oaModule* topMod = NULL;
|
||||
topMod = designCellView->getTopModule();
|
||||
if(!topMod){
|
||||
topMod = oaModule::embed(designCellView, previous);
|
||||
designCellView->setTopModule(topMod);
|
||||
}
|
||||
oaBlock *topBlock = designCellView->getTopBlock();
|
||||
assert(topBlock);
|
||||
|
||||
return designCellView;
|
||||
}
|
||||
|
||||
/**
|
||||
Add layout view to previous view ...
|
||||
*/
|
||||
oaDesign* addLayout(const Cell* cell,oaDesign* previous) {
|
||||
cerr << "addLayout" << cell << endl;
|
||||
assert(cell);
|
||||
assert(previous);
|
||||
Cell2OADesignMap::iterator it = _cell2OADesign4Layout.find(cell);
|
||||
if (it != _cell2OADesign4Layout.end()) {
|
||||
return it->second;
|
||||
}
|
||||
|
||||
oaNativeNS ns;
|
||||
oaLib* lib = getOALibForLibrary(cell->getLibrary());
|
||||
assert(lib);
|
||||
oaScalarName scNameDesign(ns, getString(cell->getName()).c_str());
|
||||
oaScalarName scNameView(ns, "layout");
|
||||
oaScalarName scNameLib;
|
||||
lib->getName(scNameLib);
|
||||
|
||||
// create a layout CellView of the cell
|
||||
oaViewType* vType = oaViewType::get(oacMaskLayout);
|
||||
cerr << "oaDesign::open for layout view" << endl;
|
||||
oaDesign* designCellView = oaDesign::open(scNameLib, scNameDesign, scNameView, vType, 'a');
|
||||
_cell2OADesign4Layout[cell] = designCellView;
|
||||
|
||||
// get module or embed previous module
|
||||
oaModule* topMod = NULL;
|
||||
topMod = designCellView->getTopModule();
|
||||
if(!topMod){
|
||||
topMod = oaModule::embed(designCellView, previous);
|
||||
designCellView->setTopModule(topMod);
|
||||
}
|
||||
oaBlock *topBlock = designCellView->getTopBlock();
|
||||
assert(topBlock);
|
||||
|
||||
return designCellView;
|
||||
}
|
||||
|
||||
/**
|
||||
Convert a Cell to OA designs ...
|
||||
*/
|
||||
oaDesign* getOADesignForCell(const Cell* cell) {
|
||||
cerr << "getOADesignForCell " << cell << endl;
|
||||
assert(cell);
|
||||
|
||||
// 1) get technology
|
||||
if(!_oaTech)
|
||||
_oaTech = getOATechForTechnology(_technology,cell->getLibrary());
|
||||
|
||||
// 2) create OA structure ...
|
||||
oaDesign* netlistView = createOAasNetlist(cell);
|
||||
assert(netlistView);
|
||||
|
||||
oaCell* c1 = OADesignToOACell(netlistView);
|
||||
assert(c1);
|
||||
|
||||
oaDesign* symbolicView = addSymbol(cell,netlistView);
|
||||
assert(symbolicView);
|
||||
|
||||
oaCell* c2 = OADesignToOACell(symbolicView);
|
||||
assert(c2);
|
||||
|
||||
oaDesign* schematicView = addSchematic(cell,symbolicView);
|
||||
assert(schematicView);
|
||||
|
||||
oaCell* c3 = OADesignToOACell(schematicView);
|
||||
assert(c3);
|
||||
|
||||
oaDesign* layoutView = addLayout(cell,schematicView);
|
||||
assert(layoutView);
|
||||
|
||||
oaCell* c4 = OADesignToOACell(layoutView);
|
||||
assert(c4);
|
||||
|
||||
assert(c1 == c2 && c2 == c3 && c3 == c4);
|
||||
|
||||
return netlistView;
|
||||
}
|
||||
|
||||
oaCell* getOACellForCell(const Cell* cell) {
|
||||
return OADesignToOACell( getOADesignForCell(cell) );
|
||||
}
|
||||
};
|
||||
#endif
|
||||
}
|
||||
|
||||
|
||||
namespace CRL {
|
||||
void OpenAccessWrapper::oaDriver(const string& path, Cell* cell) {
|
||||
#ifdef HAVE_OPENACCESS
|
||||
//for the moment a driver for hurricaneAMS
|
||||
//save the Cell only and all used Cells
|
||||
cerr << "Saving " << cell << " in " << path << endl;
|
||||
try {
|
||||
oaDesignInit(oacAPIMajorRevNumber,
|
||||
oacAPIMinorRevNumber,
|
||||
oacDataModelRevNumber);
|
||||
|
||||
OADriver oaDriver(path);
|
||||
oaCell* convertedCell = oaDriver.getOACellForCell(cell);
|
||||
}catch (oaException &e) {
|
||||
cerr << "OA::ERROR => " << e.getMsg() << endl;
|
||||
exit(1);
|
||||
}catch(std::exception& e){
|
||||
cerr << "STD::ERROR => " << e.what() << endl;
|
||||
exit(2);
|
||||
}
|
||||
#else
|
||||
cerr << "\nDummy OpenAccess driver call for " << path << endl;
|
||||
#endif
|
||||
}
|
||||
}
|
|
@ -0,0 +1,374 @@
|
|||
// -*-compile-command:"cd ../../../../.. && make"-*-
|
||||
// Time-stamp: "2010-07-26 16:16:18" - OpenAccessParser.cpp
|
||||
// x-----------------------------------------------------------------x
|
||||
// | This file is part of the hurricaneAMS Software. |
|
||||
// | Copyright (c) UPMC/LIP6 2008-2010, All Rights Reserved |
|
||||
// | =============================================================== |
|
||||
// | Author : Chistophe Alexandre |
|
||||
// | E-mail : Christophe.Alexandre@asim.lip6.fr |
|
||||
// x-----------------------------------------------------------------x
|
||||
// | Author : Jean-Manuel Caba |
|
||||
// | E-mail : Jean-Manuel.Caba@asim.lip6.fr |
|
||||
// x-----------------------------------------------------------------x
|
||||
|
||||
#include <iostream>
|
||||
#include <set>
|
||||
using namespace std;
|
||||
|
||||
#include "hurricane/DataBase.h"
|
||||
#include "hurricane/Technology.h"
|
||||
#include "hurricane/BasicLayer.h"
|
||||
#include "hurricane/Library.h"
|
||||
#include "hurricane/Horizontal.h"
|
||||
#include "hurricane/Vertical.h"
|
||||
#include "hurricane/NetExternalComponents.h"
|
||||
using namespace Hurricane;
|
||||
|
||||
#include "OpenAccess.h"
|
||||
#include "OpenAccessCommon.h"
|
||||
|
||||
namespace {
|
||||
|
||||
#ifdef HAVE_OPENACCESS
|
||||
class OAParser {
|
||||
typedef map<oaLib*, Library*> OALib2LibMap;
|
||||
typedef map<Name, Library*> Name2LibMap;
|
||||
typedef map<oaLayerNum, Layer*> OALayerNum2LayerMap;
|
||||
typedef map<Cell*, oaCell*> Cell2OACellMap;
|
||||
|
||||
oaTech* _oaTechnology; // only one technology;
|
||||
OALayerNum2LayerMap _oaLayerNum2LayerMap;
|
||||
OALib2LibMap _oaLib2LibMap;
|
||||
Name2LibMap _name2LibMap;
|
||||
Cell2OACellMap _cell2OACellMap;
|
||||
set<Cell*> _loadedCells;
|
||||
|
||||
OAParser():
|
||||
_oaTechnology(NULL),
|
||||
_oaLayerNum2LayerMap(),
|
||||
_oaLib2LibMap(),
|
||||
_name2LibMap(),
|
||||
_cell2OACellMap(),
|
||||
_loadedCells(){
|
||||
try {
|
||||
oaDesignInit(oacAPIMajorRevNumber,
|
||||
oacAPIMinorRevNumber,
|
||||
oacDataModelRevNumber);
|
||||
} catch (oaException &excp) {
|
||||
cout << "ERROR: " << excp.getMsg() << endl;
|
||||
exit(1);
|
||||
}
|
||||
|
||||
|
||||
DataBase* db = DataBase::getDB();
|
||||
if (!db) {
|
||||
db = DataBase::create();
|
||||
}
|
||||
Library* rootLibrary = db->getRootLibrary();
|
||||
if (!rootLibrary) {
|
||||
rootLibrary = Library::create ( db, "RootLibrary" );
|
||||
}
|
||||
Library::create(rootLibrary, OACellLibrariesName);
|
||||
Library::create(rootLibrary, OADesignLibrariesName);
|
||||
}
|
||||
|
||||
~OAParser() {
|
||||
for (OALib2LibMap::iterator lit = _oaLib2LibMap.begin();
|
||||
lit != _oaLib2LibMap.end();
|
||||
lit++) {
|
||||
lit->first->close();
|
||||
}
|
||||
}
|
||||
|
||||
Cell* getCell(const string& cellNameStr) {
|
||||
DataBase* db = DataBase::getDB();
|
||||
if (!db || !db->getRootLibrary() ) {
|
||||
return NULL;
|
||||
}
|
||||
Name cellName(cellNameStr);
|
||||
Cell* cell = findCellInLibraries(getOADesignLibraries(), cellName);
|
||||
if (!cell) {
|
||||
return NULL;
|
||||
}
|
||||
|
||||
set<Cell*>::const_iterator csit = _loadedCells.find(cell);
|
||||
if (csit == _loadedCells.end()) {
|
||||
Cell2OACellMap::const_iterator cit = _cell2OACellMap.find(cell);
|
||||
if (cit == _cell2OACellMap.end()) {
|
||||
cerr << "error : cannot find cell" << endl;
|
||||
exit(8);
|
||||
}
|
||||
oaCell* oa_Cell = cit->second;
|
||||
loadOACellInCell(oa_Cell, cell);
|
||||
_loadedCells.insert(cell);
|
||||
}
|
||||
return cell;
|
||||
}
|
||||
|
||||
void loadOALib(const string& libNameStr, const string& libPathStr, bool asDesignLibrary) {
|
||||
Name libName(libNameStr);
|
||||
Name2LibMap::const_iterator nit = _name2LibMap.find(libName);
|
||||
if (nit != _name2LibMap.end()) {
|
||||
Library* library = nit->second;
|
||||
//verify that it's the same library : name and path
|
||||
cerr << "already loaded" << endl;
|
||||
return;
|
||||
}
|
||||
|
||||
oaNativeNS oaNS;
|
||||
oaString libNameOAStr(libNameStr.c_str());
|
||||
oaScalarName libOAName(oaNS, libNameOAStr);
|
||||
|
||||
try {
|
||||
oaLib* oaLibrary = oaLib::open(libOAName, libPathStr.c_str());
|
||||
if (oaLibrary->isReadable()) {
|
||||
if (!oaLibrary->getAccess(oacReadLibAccess)) {
|
||||
cout << "\n***Quitting. Cannot get LibAccess\n" ;
|
||||
exit(8);
|
||||
}
|
||||
|
||||
oaTechnology2Technology(oaLibrary);
|
||||
|
||||
//create Hurricane library
|
||||
Name libraryName = Name(libNameStr);
|
||||
DataBase* db = DataBase::getDB();
|
||||
if (!db) {
|
||||
cerr << "No DataBase" << endl;
|
||||
exit(8);
|
||||
}
|
||||
if (findLibraryByNameInDB(db, libraryName)) {
|
||||
cerr << "ERROR" << endl;
|
||||
exit(8);
|
||||
}
|
||||
|
||||
Library* library;
|
||||
if (asDesignLibrary) {
|
||||
library = Library::create(getOADesignLibraries(), Name(libNameStr));
|
||||
} else {
|
||||
library = Library::create(getOACellLibraries(), Name(libNameStr));
|
||||
}
|
||||
cerr << library << endl;
|
||||
|
||||
oaCollection<oaCell, oaLib> cells = oaLibrary->getCells();
|
||||
oaIter<oaCell> cellsIter(cells);
|
||||
|
||||
while (oaCell* cell = cellsIter.getNext()) {
|
||||
oaScalarName cellOAName;
|
||||
oaString cellNameString;
|
||||
cell->getName(cellOAName);
|
||||
cellOAName.get(cellNameString);
|
||||
|
||||
cerr << cellNameString << endl;
|
||||
|
||||
oaDesign* cellDesign = openDesign(oaNS, cell);
|
||||
if (cellDesign != NULL) {
|
||||
Cell* hCell = NULL;
|
||||
//logic part
|
||||
oaModule* module = cellDesign->getTopModule();
|
||||
if (module) {
|
||||
hCell = Cell::create(library, Name(cellNameString));
|
||||
hCell->setTerminal(false);
|
||||
if (!asDesignLibrary) {
|
||||
hCell->setFlattenLeaf(true);
|
||||
}
|
||||
cerr << hCell << endl;
|
||||
if (asDesignLibrary) {
|
||||
_cell2OACellMap[hCell] = cell;
|
||||
}
|
||||
oaCollection<oaModTerm, oaModule> oaModTerms = module->getTerms();
|
||||
oaIter<oaModTerm> oaModTermIter(oaModTerms);
|
||||
while (oaModTerm* modTerm = oaModTermIter.getNext()) {
|
||||
oaString oaModTermStr;
|
||||
modTerm->getName(oaNS, oaModTermStr);
|
||||
Net* net = Net::create(hCell, Name(oaModTermStr));
|
||||
net->setExternal(true);
|
||||
switch (modTerm->getTermType()) {
|
||||
case oacInputTermType:
|
||||
net->setDirection(Net::Direction::IN);
|
||||
break;
|
||||
case oacOutputTermType:
|
||||
net->setDirection(Net::Direction::OUT);
|
||||
break;
|
||||
case oacInputOutputTermType:
|
||||
net->setDirection(Net::Direction::INOUT);
|
||||
break;
|
||||
case oacTristateTermType:
|
||||
net->setDirection(Net::Direction::TRISTATE);
|
||||
break;
|
||||
default:
|
||||
net->setDirection(Net::Direction::UNDEFINED);
|
||||
break;
|
||||
}
|
||||
cerr << net << endl;
|
||||
oaModNet* oaNet = modTerm->getNet();
|
||||
if (oaNet->isGlobal()) {
|
||||
net->setGlobal(true);
|
||||
cerr << "global" << endl;
|
||||
}
|
||||
if (oaNet->getSigType() == oacClockSigType) {
|
||||
net->setType(Net::Type::CLOCK);
|
||||
} else if (oaNet->getSigType() == oacPowerSigType) {
|
||||
net->setGlobal(true);
|
||||
net->setType(Net::Type::POWER);
|
||||
} else if (oaNet->getSigType() == oacGroundSigType) {
|
||||
net->setGlobal(true);
|
||||
net->setType(Net::Type::GROUND);
|
||||
} else {
|
||||
net->setType(Net::Type::LOGICAL);
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
//physical part
|
||||
oaBlock* block = cellDesign->getTopBlock();
|
||||
if (block && hCell) {
|
||||
oaBox oa_box;
|
||||
block->getBBox(oa_box);
|
||||
cerr << "box values" << endl;
|
||||
cerr << oa_box.lowerLeft().x() << " , " << oa_box.lowerLeft().y() << endl;
|
||||
cerr << oa_box.upperRight().x() << " , " << oa_box.upperRight().y() << endl;
|
||||
cerr << endl;
|
||||
Point lowerLeft(DbU::db(oa_box.lowerLeft().x()), DbU::db(oa_box.lowerLeft().y()));
|
||||
Point upperRight(DbU::db(oa_box.upperRight().x()), DbU::db(oa_box.upperRight().y()));
|
||||
hCell->setAbutmentBox(Box(lowerLeft, upperRight));
|
||||
oaCollection<oaTerm, oaBlock> oaTerms = block->getTerms();
|
||||
oaIter<oaTerm> oaTermIter(oaTerms);
|
||||
while (oaTerm* term = oaTermIter.getNext()) {
|
||||
oaString oaTermStr;
|
||||
term->getName(oaNS, oaTermStr);
|
||||
Net* net = hCell->getNet(Name(oaTermStr));
|
||||
if (net) {
|
||||
cerr << "found net : " << net << endl;
|
||||
oaCollection<oaPin, oaTerm> oaPins(term->getPins());
|
||||
oaIter<oaPin> oaPinIter(oaPins);
|
||||
while (oaPin* pin = oaPinIter.getNext()) {
|
||||
oaString oaPinStr;
|
||||
pin->getName(oaPinStr);
|
||||
cerr << "pin : " << oaPinStr << endl;
|
||||
|
||||
oaCollection<oaPinFig, oaPin> oaPinFigs(pin->getFigs());
|
||||
oaIter<oaPinFig> oaPinFigIter(oaPinFigs);
|
||||
while (oaPinFig* pinFig = oaPinFigIter.getNext()) {
|
||||
cerr << "pinFig" << endl;
|
||||
if (pinFig->getType() == oacRectType) {
|
||||
oaRect* rect = static_cast<oaRect*>(pinFig);
|
||||
OALayerNum2LayerMap::iterator layerMapIt = _oaLayerNum2LayerMap.find(rect->getLayerNum());
|
||||
if (layerMapIt != _oaLayerNum2LayerMap.end()) {
|
||||
Layer* layer = layerMapIt->second;
|
||||
oaBox pinBBox;
|
||||
rect->getBBox(pinBBox);
|
||||
DbU::Unit width = DbU::db( (pinBBox.right() - pinBBox.left()) * (double)convertFactor);
|
||||
DbU::Unit height = DbU::db( (pinBBox.top() - pinBBox.bottom()) * (double)convertFactor);
|
||||
DbU::Unit cx = DbU::db(pinBBox.right() * (double)convertFactor) + width/2;
|
||||
DbU::Unit cy = DbU::db(pinBBox.top() * (double)convertFactor) + height/2;
|
||||
if (width > height) {
|
||||
Horizontal* horizontal=Horizontal::create(net, layer, cy, height, DbU::db(pinBBox.right() * (double)convertFactor), DbU::db(pinBBox.left() * (double)convertFactor));
|
||||
NetExternalComponents::setExternal(horizontal);
|
||||
} else {
|
||||
Vertical* vertical= Vertical::create(net, layer, cx, width, DbU::db(pinBBox.bottom() * (double)convertFactor), DbU::db(pinBBox.top() * (double)convertFactor));
|
||||
NetExternalComponents::setExternal(vertical);
|
||||
}
|
||||
} else {
|
||||
cerr << "cannot find layer..." << endl;
|
||||
}
|
||||
|
||||
cerr << "rect " << rect->getLayerNum() << endl;
|
||||
} else {
|
||||
cerr << "no rect" << endl;
|
||||
}
|
||||
}
|
||||
}
|
||||
} else {
|
||||
//TODO
|
||||
}
|
||||
}
|
||||
} else {
|
||||
cerr << "no block" << endl;
|
||||
}
|
||||
}
|
||||
}
|
||||
oaLibrary->releaseAccess();
|
||||
//oaLibrary->close(); do not close !!
|
||||
//FIXME save opened libraries ??
|
||||
}
|
||||
} catch (oaException &excp) {
|
||||
cout << "ERROR: " << excp.getMsg() << endl;
|
||||
exit(1);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
void oaTechnology2Technology(oaLib* oaLibrary) {
|
||||
try {
|
||||
oaTech* tech = oaTech::open(oaLibrary);
|
||||
|
||||
if (!tech) {
|
||||
cout << "ERROR" << endl;
|
||||
exit(1);
|
||||
}
|
||||
if (_oaTechnology && tech != _oaTechnology) {
|
||||
cout << "ERROR" << endl;
|
||||
exit(1);
|
||||
}
|
||||
if (_oaTechnology) {
|
||||
return;
|
||||
}
|
||||
_oaTechnology = tech;
|
||||
|
||||
DataBase* db = DataBase::getDB();
|
||||
if (!db) {
|
||||
cerr << "No DataBase" << endl;
|
||||
exit(8);
|
||||
}
|
||||
Technology* technology = db->getTechnology();
|
||||
if (!technology) {
|
||||
technology = Technology::create(db, "OpenAccess");
|
||||
}
|
||||
|
||||
oaCollection<oaLayer, oaTech> oaLayers(_oaTechnology->getLayers());
|
||||
oaIter<oaLayer> oaLayerIt(oaLayers);
|
||||
while (oaLayer* oaLayer = oaLayerIt.getNext()) {
|
||||
oaString oaLayerStr;
|
||||
oaLayer->getName(oaLayerStr);
|
||||
Name layerName(oaLayerStr);
|
||||
Layer* layer = technology->getLayer(layerName);
|
||||
if (!layer) {
|
||||
layer = BasicLayer::create(technology, layerName, BasicLayer::Material::other, oaLayer->getNumber());
|
||||
}
|
||||
cerr << layer << endl;
|
||||
_oaLayerNum2LayerMap[oaLayer->getNumber()] = layer;
|
||||
}
|
||||
} catch (oaException &excp) {
|
||||
cout << "ERROR: " << excp.getMsg() << endl;
|
||||
exit(1);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
void getDesigns(set<Cell*>& designCellSet) {
|
||||
getAllCells(getOADesignLibraries(), designCellSet);
|
||||
}
|
||||
};//OAParser class
|
||||
#endif
|
||||
}//namespace
|
||||
|
||||
namespace CRL {
|
||||
Cell* OpenAccess::oaCellParser(oaCell* cell){
|
||||
#ifdef HAVE_OPENACCESS
|
||||
cerr << "\nto implement ... " << endl;
|
||||
#else
|
||||
cerr << "\nDummy OpenAccess driver call for " << endl;
|
||||
#endif
|
||||
}
|
||||
|
||||
Library* OpenAccess::oaLibParser(oaLib* lib){
|
||||
#ifdef HAVE_OPENACCESS
|
||||
cerr << "\nto implement ... " << endl;
|
||||
#else
|
||||
cerr << "\nDummy OpenAccess driver call for " << path << endl;
|
||||
#endif
|
||||
}
|
||||
|
||||
}
|
|
@ -1,321 +0,0 @@
|
|||
|
||||
#ifndef __OPENACCESSUTILS_H__
|
||||
#define __OPENACCESSUTILS_H__
|
||||
|
||||
#include <iostream>
|
||||
using namespace std;
|
||||
|
||||
#ifdef HAVE_OPENACCESS
|
||||
#include "oa/oaDesignDB.h"
|
||||
using namespace oa;
|
||||
|
||||
#include "hurricane/Cell.h"
|
||||
#include "hurricane/BasicLayer.h"
|
||||
using namespace Hurricane;
|
||||
|
||||
#undef assert
|
||||
#define assert(cond) if (! (cond) ) throw Error("assertion failed : " + string( #cond ) )
|
||||
//#define assert(cond)
|
||||
|
||||
namespace {
|
||||
|
||||
/**
|
||||
giving a oaDesign pointer that should be not NULL
|
||||
return the oaString representing its name
|
||||
*/
|
||||
inline oaString getDesignName(oaDesign* design) {
|
||||
cerr << "getDesignName" << endl;
|
||||
assert(design);
|
||||
oaNativeNS ns;
|
||||
oaString libName, cellName, viewName;
|
||||
design->getLibName(ns, libName);
|
||||
design->getCellName(ns, cellName);
|
||||
design->getViewName(ns, viewName);
|
||||
oaString designName = "<" + libName + "," + cellName + "," + viewName + ">";
|
||||
return designName;
|
||||
}
|
||||
|
||||
/**
|
||||
giving a oaBlock
|
||||
print the connectivity, mainly used for debug purpose ...
|
||||
@todo remove when not needed anymore
|
||||
*/
|
||||
inline void printBlockTerms(oaBlock* block) {
|
||||
cerr << "printBlockTerms" << endl;
|
||||
assert(block);
|
||||
oaNativeNS ns;
|
||||
oaDesign* design = block->getDesign();
|
||||
cerr << " o Printing " << getDesignName(design) << " terms" << endl;
|
||||
oaIter<oaTerm> termIter(block->getTerms());
|
||||
while (oaTerm* term = termIter.getNext()) {
|
||||
oaString termName;
|
||||
term->getName(ns, termName);
|
||||
cerr << " - " << termName << endl;
|
||||
}
|
||||
cerr << endl;
|
||||
}
|
||||
|
||||
/**
|
||||
Convert material from Hurricane to OA ...
|
||||
@todo verify
|
||||
*/
|
||||
inline oaMaterial getOAMaterial(const BasicLayer::Material& material) {
|
||||
cerr << "getOAMaterial" << endl;
|
||||
switch ( material.getCode() ) {
|
||||
case BasicLayer::Material::nWell: return oacNWellMaterial;
|
||||
case BasicLayer::Material::pWell: return oacPWellMaterial;
|
||||
case BasicLayer::Material::nImplant: return oacNImplantMaterial;
|
||||
case BasicLayer::Material::pImplant: return oacPImplantMaterial;
|
||||
case BasicLayer::Material::active: return oacOtherMaterial;//is it OK?
|
||||
case BasicLayer::Material::poly: return oacPolyMaterial;
|
||||
case BasicLayer::Material::cut: return oacCutMaterial;
|
||||
case BasicLayer::Material::metal: return oacMetalMaterial;
|
||||
case BasicLayer::Material::blockage:
|
||||
//there is no blockage type but a specific oaLayerBlockage class
|
||||
return oacOtherMaterial;
|
||||
case BasicLayer::Material::other: return oacOtherMaterial;
|
||||
default:
|
||||
throw Error("Unrecognized material");
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
Convertion helper for Net convertion ...
|
||||
@todo verify
|
||||
*/
|
||||
inline oaTermType getOATermType(const Net::Direction& direction) {
|
||||
cerr << "getOATermType" << endl;
|
||||
switch (direction) {
|
||||
case Net::Direction::IN:
|
||||
return oacInputTermType;
|
||||
case Net::Direction::OUT:
|
||||
return oacOutputTermType;
|
||||
case Net::Direction::INOUT:
|
||||
return oacInputOutputTermType;
|
||||
case Net::Direction::TRISTATE:
|
||||
return oacTristateTermType;
|
||||
case Net::Direction::UNDEFINED:
|
||||
return oacUnusedTermType;// is it OK ?
|
||||
default:
|
||||
throw Error("Unrecognized direction");
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
Convertion helper for Net convertion ...
|
||||
@todo verify
|
||||
*/
|
||||
inline oaSigType getOASigType(const Net::Type& type) {
|
||||
cerr << "getOASigType" << endl;
|
||||
switch (type.getCode()) {
|
||||
case Net::Type::LOGICAL:
|
||||
return oacSignalSigType;
|
||||
case Net::Type::CLOCK:
|
||||
return oacClockSigType;
|
||||
case Net::Type::POWER:
|
||||
return oacPowerSigType;
|
||||
case Net::Type::GROUND:
|
||||
return oacGroundSigType;
|
||||
case Net::Type::UNDEFINED:
|
||||
return oacAnalogSigType;// is it OK ?
|
||||
default:
|
||||
throw Error("Unrecognized net type");
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
Convertion helper ...
|
||||
*/
|
||||
inline oaOrient getOAOrientFromOrientation(const Transformation::Orientation& orientation) {
|
||||
cerr << "getOAOrientFromOrientation" << endl;
|
||||
switch (orientation) {
|
||||
case Transformation::Orientation::ID:
|
||||
return oacR0;
|
||||
case Transformation::Orientation::R1:
|
||||
return oacR90;
|
||||
case Transformation::Orientation::R2:
|
||||
return oacR180;
|
||||
case Transformation::Orientation::R3:
|
||||
return oacR270;
|
||||
case Transformation::Orientation::MX:
|
||||
return oacMX;
|
||||
case Transformation::Orientation::XR:
|
||||
return oacMXR90;
|
||||
case Transformation::Orientation::MY:
|
||||
return oacMY;
|
||||
case Transformation::Orientation::YR:
|
||||
return oacMYR90;
|
||||
default:
|
||||
throw Error("Unrecognized orientation");
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
Convertion helper ...
|
||||
*/
|
||||
inline void getOATransformFromTransformation(oaTransform& transform, const Transformation& transformation) {
|
||||
cerr << "getOATransformFromTransformation" << endl;
|
||||
transform.set(transformation.getTx(),
|
||||
transformation.getTy(),
|
||||
getOAOrientFromOrientation(transformation.getOrientation()));
|
||||
}
|
||||
|
||||
/**
|
||||
Convertion helper ...
|
||||
*/
|
||||
inline void getOABoxForBox(oaBox& box, const Box& hbox) {
|
||||
cerr << "getOABoxForBox" << endl;
|
||||
box.set(hbox.getXMin(), hbox.getYMin(), hbox.getXMax(), hbox.getYMax());
|
||||
}
|
||||
|
||||
/**
|
||||
Create InstTerm representing connection of nets ...
|
||||
always return a non NULL value
|
||||
*/
|
||||
inline oaInstTerm* getInstTerm(oaInst* inst, Plug* plug,oaNet* net) {
|
||||
cerr << "getInstTerm" << endl;
|
||||
assert(inst);
|
||||
assert(plug);
|
||||
oaNativeNS ns;
|
||||
oaScalarName scPlugName(ns, getString(plug->getMasterNet()->getName()).c_str());
|
||||
oaName instTermName(scPlugName);
|
||||
oaInstTerm* instTerm = oaInstTerm::find(inst, instTermName);
|
||||
if (instTerm) {
|
||||
return instTerm;
|
||||
}
|
||||
oaDesign* design = inst->getMaster();
|
||||
assert(design);
|
||||
oaBlock* masterBlock = design->getTopBlock();
|
||||
oaTerm* term = oaTerm::find(masterBlock, instTermName);
|
||||
assert(term);
|
||||
cerr << "looking for " << plug->getName() << endl;
|
||||
printBlockTerms(masterBlock);
|
||||
cerr << "oaInstTerm::create" << endl;
|
||||
instTerm = oaInstTerm::create(net, inst, term);
|
||||
assert(instTerm);
|
||||
return instTerm;
|
||||
}
|
||||
|
||||
/**
|
||||
save design stored in a map
|
||||
*/
|
||||
inline void saveDesignsInMap(map<const Cell*, oaDesign*> cell2OAdesign){
|
||||
for (map<const Cell*, oaDesign*>::iterator it = cell2OAdesign.begin();
|
||||
it != cell2OAdesign.end();
|
||||
++it) {
|
||||
cerr << it->first << endl;
|
||||
oaDesign* design = it->second;
|
||||
design->save();
|
||||
design->close();
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
print the oaLayera in a oaTech ...
|
||||
*/
|
||||
inline void printOALayers(oaTech* theOATech){
|
||||
cerr << "printOALayers" << endl;
|
||||
assert(theOATech);
|
||||
oaIter<oaLayer> lIter(theOATech->getLayers());
|
||||
while(oaLayer* l = lIter.getNext()){
|
||||
oaString layerName;
|
||||
l->getName(layerName);
|
||||
cerr << " o created oaLayer " << layerName << endl;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
@todo complete,verify ...
|
||||
*/
|
||||
inline BasicLayer::Material::Code oaMaterialToBasicLayerType(const oaMaterial& material) {
|
||||
switch(material) {
|
||||
case oacNWellMaterial:
|
||||
return BasicLayer::Material::nWell;
|
||||
case oacPWellMaterial:
|
||||
return BasicLayer::Material::pWell;
|
||||
case oacNImplantMaterial:
|
||||
return BasicLayer::Material::nImplant;
|
||||
case oacPImplantMaterial:
|
||||
return BasicLayer::Material::pImplant;
|
||||
case oacPolyMaterial:
|
||||
return BasicLayer::Material::poly;
|
||||
case oacCutMaterial:
|
||||
return BasicLayer::Material::cut;
|
||||
case oacMetalMaterial:
|
||||
case oacContactlessMetalMaterial:
|
||||
return BasicLayer::Material::metal;
|
||||
default:
|
||||
return BasicLayer::Material::other;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
generate info from library name
|
||||
*/
|
||||
inline pair<oaScalarName,string> libInfos(const string& path,
|
||||
const string& libName){
|
||||
oaNativeNS ns;
|
||||
const char* strNameLib = libName.c_str();
|
||||
oaScalarName scNameLib(ns, strNameLib);
|
||||
string strPathLib = path + '/' + strNameLib;
|
||||
return make_pair<oaScalarName,string>(scNameLib,strPathLib);
|
||||
}
|
||||
|
||||
/**
|
||||
open oaLib with the info gathered by libPath function
|
||||
*/
|
||||
inline oaLib* openOALib(const pair<oaScalarName,string>& infos){
|
||||
oaLib *lib = oaLib::find(infos.first);
|
||||
const char* pathLib = infos.second.c_str();
|
||||
if (!lib) {
|
||||
if (oaLib::exists(pathLib)){
|
||||
lib = oaLib::open(infos.first, pathLib);
|
||||
}else{
|
||||
string cmd = "mkdir -p "+ infos.second;
|
||||
system(cmd.c_str());
|
||||
}
|
||||
if(!lib){
|
||||
lib = oaLib::create(infos.first, pathLib);
|
||||
}
|
||||
}
|
||||
return lib;
|
||||
}
|
||||
|
||||
inline void createCDS(const pair<oaScalarName,string>& infos){
|
||||
try{
|
||||
cerr << "Overwriting cds.lib file begin" << endl;
|
||||
string cdsPath = infos.second + "/cds.lib";
|
||||
oaLibDefList* ldl = oaLibDefList::get( cdsPath.c_str(), 'a');
|
||||
assert(ldl);
|
||||
assert(ldl->isValid());
|
||||
if(!oaLibDef::find(ldl, infos.first))
|
||||
oaLibDef::create(ldl, infos.first, infos.second.c_str());
|
||||
ldl->save();
|
||||
ldl->destroy();//claim memory
|
||||
ldl = NULL;
|
||||
cerr << "Overwrited cds.lib file end" << endl;
|
||||
}catch(oaException& e){
|
||||
cerr << "ERROR cds: " << e.getMsg() << endl;
|
||||
exit(-2);
|
||||
}
|
||||
}
|
||||
|
||||
inline oaCell* OADesignToOACell(oaDesign* design){
|
||||
assert(design);
|
||||
oaScalarName cellName;
|
||||
design->getCellName(cellName);
|
||||
oaLib* lib = design->getLib();
|
||||
oaBoolean gotAccess = false;
|
||||
gotAccess = lib->getAccess(oacReadLibAccess);
|
||||
oaCell* cell = oaCell::find(lib,cellName);
|
||||
if(gotAccess)
|
||||
lib->releaseAccess();
|
||||
assert(cell);
|
||||
return cell;
|
||||
}
|
||||
|
||||
}//end anonymous namespace
|
||||
|
||||
#endif//HAVE_OPENACCESS
|
||||
|
||||
#endif//__OPENACCESSUTILS_H__
|
|
@ -1,644 +0,0 @@
|
|||
// x-----------------------------------------------------------------x
|
||||
// | This file is part of the hurricaneAMS Software. |
|
||||
// | Copyright (c) UPMC/LIP6 2008-2010, All Rights Reserved |
|
||||
// | =============================================================== |
|
||||
// | Author : Chistophe Alexandre |
|
||||
// | E-mail : Christophe.Alexandre@asim.lip6.fr |
|
||||
// x-----------------------------------------------------------------x
|
||||
#include <iostream>
|
||||
using namespace std;
|
||||
|
||||
#include "hurricane/DataBase.h"
|
||||
#include "hurricane/Technology.h"
|
||||
#include "hurricane/BasicLayer.h"
|
||||
#include "hurricane/Library.h"
|
||||
#include "hurricane/Horizontal.h"
|
||||
#include "hurricane/Vertical.h"
|
||||
#include "hurricane/NetExternalComponents.h"
|
||||
using namespace Hurricane;
|
||||
|
||||
#include "OAWrapper.h"
|
||||
|
||||
//open library
|
||||
|
||||
namespace {
|
||||
static int convertFactor = 1;
|
||||
|
||||
void cantLock() {
|
||||
cout << "\n***Quitting. Cannot get LibAccess\n" ;
|
||||
exit(8);
|
||||
}
|
||||
|
||||
Library* findLibraryByNameInLibrary(const Library* rootLibrary, const Name& libraryName) {
|
||||
for_each_library(library, rootLibrary->getLibraries()) {
|
||||
if (library->getName() == libraryName) {
|
||||
return library;
|
||||
}
|
||||
Library* foundLibrary = findLibraryByNameInLibrary(library, libraryName);
|
||||
if (foundLibrary) {
|
||||
return foundLibrary;
|
||||
}
|
||||
end_for;
|
||||
}
|
||||
return NULL;
|
||||
}
|
||||
|
||||
Library* findLibraryByNameInDB(const DataBase* db, const Name& libraryName) {
|
||||
Library* rootLibrary = db->getRootLibrary();
|
||||
if (rootLibrary->getName() == libraryName) {
|
||||
return rootLibrary;
|
||||
}
|
||||
return findLibraryByNameInLibrary(rootLibrary, libraryName);
|
||||
}
|
||||
|
||||
void getAllCells(Library* rootLibrary, set<Cell*>& cellSet) {
|
||||
for_each_cell(cell, rootLibrary->getCells()) {
|
||||
cellSet.insert(cell);
|
||||
end_for;
|
||||
}
|
||||
for_each_library(library, rootLibrary->getLibraries()) {
|
||||
getAllCells(library, cellSet);
|
||||
end_for;
|
||||
}
|
||||
}
|
||||
|
||||
Cell* findCellInLibraries(const Library* rootLibrary, const Name& cellName) {
|
||||
for_each_cell(cell, rootLibrary->getCells()) {
|
||||
if (cell->getName() == cellName) {
|
||||
return cell;
|
||||
}
|
||||
end_for;
|
||||
}
|
||||
for_each_library(library, rootLibrary->getLibraries()) {
|
||||
Cell* cell = findCellInLibraries(library, cellName);
|
||||
if (cell) {
|
||||
return cell;
|
||||
}
|
||||
end_for;
|
||||
}
|
||||
return NULL;
|
||||
}
|
||||
|
||||
Library* getRootLibrary() {
|
||||
DataBase* db = DataBase::getDB();
|
||||
if (!db) {
|
||||
return NULL;
|
||||
}
|
||||
return db->getRootLibrary();
|
||||
}
|
||||
|
||||
static const Name OACellLibrariesName("OACellLibraries");
|
||||
static const Name OADesignLibrariesName("OADesignLibraries");
|
||||
|
||||
Library* getOACellLibraries() {
|
||||
Library* rootLibrary = getRootLibrary();
|
||||
if (!rootLibrary) {
|
||||
return NULL;
|
||||
}
|
||||
for_each_library(library, rootLibrary->getLibraries()) {
|
||||
if (library->getName() == OACellLibrariesName) {
|
||||
return library;
|
||||
}
|
||||
end_for;
|
||||
}
|
||||
return NULL;
|
||||
}
|
||||
|
||||
Library* getOADesignLibraries() {
|
||||
Library* rootLibrary = getRootLibrary();
|
||||
if (!rootLibrary) {
|
||||
return NULL;
|
||||
}
|
||||
for_each_library(library, rootLibrary->getLibraries()) {
|
||||
if (library->getName() == OADesignLibrariesName) {
|
||||
return library;
|
||||
}
|
||||
end_for;
|
||||
}
|
||||
return NULL;
|
||||
}
|
||||
|
||||
oaView* pickView(oaCell* oa_Cell) {
|
||||
//oacMaskLayout Type is first
|
||||
oaView* toReturnView = NULL;
|
||||
|
||||
oaCollection<oaCellView, oaCell> cellViews = oa_Cell->getCellViews();
|
||||
oaIter<oaCellView> cellViewIter(cellViews);
|
||||
|
||||
while (oaCellView* cellView = cellViewIter.getNext()) {
|
||||
oaView* view = cellView->getView();
|
||||
oaViewType* viewType = view->getViewType();
|
||||
oaScalarName viewName;
|
||||
view->getName(viewName);
|
||||
if (viewType == oaViewType::get(oacMaskLayout)) {
|
||||
return view;
|
||||
}
|
||||
if (viewType == oaViewType::get(oacSchematic) ||
|
||||
viewType == oaViewType::get(oacNetlist) ||
|
||||
viewType == oaViewType::get(oacSchematicSymbol)) {
|
||||
toReturnView = view;
|
||||
}
|
||||
}
|
||||
return toReturnView;
|
||||
}
|
||||
|
||||
oaDesign* openDesign(const oaNameSpace& oaNS, oaCell* oa_Cell) {
|
||||
oaView* view = pickView(oa_Cell);
|
||||
if (view != NULL) {
|
||||
oaScalarName libName;
|
||||
oaLib* cellLib = oa_Cell->getLib();
|
||||
cellLib->getName(libName);
|
||||
oaScalarName cellName;
|
||||
oa_Cell->getName(cellName);
|
||||
oaViewType* viewType = view->getViewType();
|
||||
oaScalarName viewName;
|
||||
view->getName(viewName);
|
||||
return oaDesign::open(libName, cellName, viewName, viewType, 'r');
|
||||
}
|
||||
return NULL;
|
||||
}
|
||||
|
||||
|
||||
void loadOACellInCell(oaCell* oa_Cell, Cell* cell) {
|
||||
oaNativeNS oaNS;
|
||||
oaDesign* cellDesign = openDesign(oaNS, oa_Cell);
|
||||
|
||||
if (cellDesign != NULL) {
|
||||
oaModule* module = cellDesign->getTopModule();
|
||||
oaCollection<oaModInst, oaModule> oaModInsts = module->getInsts();
|
||||
oaIter<oaModInst> modInstIter(oaModInsts);
|
||||
while (oaModInst* modInst = modInstIter.getNext()) {
|
||||
oaString oaModInstStr;
|
||||
modInst->getName(oaNS, oaModInstStr);
|
||||
//cerr << "inst : " << oaModInstStr << endl;
|
||||
oaModule* masterModule = modInst->getMasterModule();
|
||||
if (masterModule) {
|
||||
oaString oaModuleStr;
|
||||
masterModule->getName(oaNS, oaModuleStr);
|
||||
//cerr << "master : " << oaModuleStr << endl;
|
||||
//find hurricane Cell
|
||||
Cell* masterCell = findCellInLibraries(getRootLibrary(), Name(oaModuleStr));
|
||||
if (!masterCell) {
|
||||
cout << "\n***Quitting. Cannot get MasterCell :" ;
|
||||
cout << oaModuleStr << endl;
|
||||
exit(8);
|
||||
}
|
||||
Instance* instance = Instance::create(cell, Name(oaModInstStr) ,masterCell);
|
||||
//cerr << instance << endl;
|
||||
} else {
|
||||
cerr << "master : NULL" << endl;
|
||||
}
|
||||
}
|
||||
|
||||
//now treat nets
|
||||
oaCollection<oaModNet, oaModule> oaModNets = module->getNets();
|
||||
oaIter<oaModNet> oaModNetIter(oaModNets);
|
||||
while (oaModNet* oa_ModNet = oaModNetIter.getNext()) {
|
||||
oaString oaModNetStr;
|
||||
oa_ModNet->getName(oaNS, oaModNetStr);
|
||||
//cerr << oaModNetStr << endl;
|
||||
Net* net = cell->getNet(Name(oaModNetStr));
|
||||
if (!net) {
|
||||
net = Net::create(cell, Name(oaModNetStr));
|
||||
}
|
||||
|
||||
oaCollection<oaModInstTerm, oaModNet> oaModInstTerms = oa_ModNet->getInstTerms();
|
||||
oaIter<oaModInstTerm> oaModInstTermIter(oaModInstTerms);
|
||||
while (oaModInstTerm* oa_ModInstTerm = oaModInstTermIter.getNext()) {
|
||||
oaModInst* modInst = oa_ModInstTerm->getInst();
|
||||
oaString oaModInstStr;
|
||||
modInst->getName(oaNS, oaModInstStr);
|
||||
//find hurricane instance
|
||||
Instance* instance = cell->getInstance(Name(oaModInstStr));
|
||||
if (!instance) {
|
||||
cout << "\n***Quitting. Cannot get Instance :" ;
|
||||
cout << oaModInstStr << endl;
|
||||
exit(8);
|
||||
}
|
||||
oaModTerm* oa_ModTerm = oa_ModInstTerm->getTerm();
|
||||
oaString oaModTermStr;
|
||||
oa_ModTerm->getName(oaNS, oaModTermStr);
|
||||
Net* masterNet = instance->getMasterCell()->getNet(Name(oaModTermStr));
|
||||
if (!masterNet) {
|
||||
cout << "\n***Quitting. Cannot get Master Net :" ;
|
||||
cout << oaModTermStr << endl;
|
||||
exit(8);
|
||||
}
|
||||
Plug* plug = instance->getPlug(masterNet);
|
||||
plug->setNet(net);
|
||||
//cerr << plug << endl;
|
||||
}
|
||||
//cerr << net << endl;
|
||||
}
|
||||
|
||||
|
||||
oaScalarName cellName;
|
||||
oa_Cell->getName(cellName);
|
||||
oaString cellNameString;
|
||||
cellName.get(cellNameString);
|
||||
cell->setName(Name(cellNameString));
|
||||
cell->setTerminal(false);
|
||||
|
||||
//physical part
|
||||
oaBlock* block = cellDesign->getTopBlock();
|
||||
if (block) {
|
||||
oaBox oa_box;
|
||||
block->getBBox(oa_box);
|
||||
Point lowerLeft(DbU::db(oa_box.lowerLeft().x()), DbU::db(oa_box.lowerLeft().y()));
|
||||
Point upperRight(DbU::db(oa_box.upperRight().x()), DbU::db(oa_box.upperRight().y()));
|
||||
cell->setAbutmentBox(Box(lowerLeft, upperRight));
|
||||
|
||||
oaCollection<oaInst, oaBlock> oaInsts = block->getInsts();
|
||||
oaIter<oaInst> oaInstIter(oaInsts);
|
||||
while (oaInst* oa_Inst = oaInstIter.getNext()) {
|
||||
oaString oaInstStr;
|
||||
oa_Inst->getName(oaNS, oaInstStr);
|
||||
Instance* instance = cell->getInstance(Name(oaInstStr));
|
||||
if (instance) {
|
||||
//cerr << "found " << instance << endl;
|
||||
oaPlacementStatus placementStatus= oa_Inst->getPlacementStatus();
|
||||
//switch (placementStatus) {
|
||||
// case oacNonePlacementStatus :
|
||||
// cerr << " none" << endl;
|
||||
// break;
|
||||
// case oacUnplacedPlacementStatus :
|
||||
// cerr << " unplaced" << endl;
|
||||
// break;
|
||||
// case oacPlacedPlacementStatus :
|
||||
// cerr << " placed" << endl;
|
||||
// break;
|
||||
// case oacFixedPlacementStatus :
|
||||
// cerr << " fixed" << endl;
|
||||
// break;
|
||||
// default :
|
||||
// cerr << "other" << endl;
|
||||
//}
|
||||
oaPoint instOrigin;
|
||||
oa_Inst->getOrigin(instOrigin);
|
||||
//cerr << instOrigin.x() << " " << instOrigin.y() << endl;
|
||||
} else {
|
||||
cerr << "cannot find " << oaInstStr << endl;
|
||||
}
|
||||
}
|
||||
oaSitePattern sitePattern;
|
||||
block->getSitePattern(sitePattern);
|
||||
|
||||
for (int i = 0; i < sitePattern.getNumElements(); i++) {
|
||||
const oaSiteRef& siteRef = sitePattern.get(i);
|
||||
cerr << "site : " << siteRef.siteName() << endl;
|
||||
}
|
||||
|
||||
oaCollection<oaRow, oaBlock> oaRows = block->getRows();
|
||||
oaIter<oaRow> oaRowIter(oaRows);
|
||||
while (oaRow* oa_Row = oaRowIter.getNext()) {
|
||||
cerr << "row" << endl;
|
||||
oaString siteName;
|
||||
oa_Row->getSiteDefName(siteName);
|
||||
cerr << siteName << endl;
|
||||
}
|
||||
|
||||
oaCollection<oaAreaBoundary, oaBlock> oaAreaBoundaries = block->getBoundaries();
|
||||
oaIter<oaBoundary> oaBoundaryIter(oaAreaBoundaries);
|
||||
while (oaBoundary* oa_Boundary = oaBoundaryIter.getNext()) {
|
||||
cerr << "boundary" << endl;
|
||||
}
|
||||
|
||||
|
||||
} else {
|
||||
cerr << "no block view " << endl;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
namespace CRL {
|
||||
|
||||
OAWrapper* OAWrapper::_singleton = NULL;
|
||||
|
||||
OAWrapper::OAWrapper():
|
||||
_oaTechnology(NULL),
|
||||
_oaLayerNum2LayerMap(),
|
||||
_oaLib2LibMap(),
|
||||
_name2LibMap(),
|
||||
_cell2OACellMap(),
|
||||
_loadedCells()
|
||||
{
|
||||
try {
|
||||
oaDesignInit(oacAPIMajorRevNumber,
|
||||
oacAPIMinorRevNumber,
|
||||
oacDataModelRevNumber);
|
||||
} catch (oaException &excp) {
|
||||
cout << "ERROR: " << excp.getMsg() << endl;
|
||||
exit(1);
|
||||
}
|
||||
|
||||
|
||||
DataBase* db = DataBase::getDB();
|
||||
if (!db) {
|
||||
db = DataBase::create();
|
||||
}
|
||||
Library* rootLibrary = db->getRootLibrary();
|
||||
if (!rootLibrary) {
|
||||
rootLibrary = Library::create ( db, "RootLibrary" );
|
||||
}
|
||||
Library::create(rootLibrary, OACellLibrariesName);
|
||||
Library::create(rootLibrary, OADesignLibrariesName);
|
||||
}
|
||||
|
||||
OAWrapper* OAWrapper::getOAWrapper() {
|
||||
if (!_singleton) {
|
||||
_singleton = new OAWrapper();
|
||||
}
|
||||
return _singleton;
|
||||
}
|
||||
|
||||
OAWrapper::~OAWrapper() {
|
||||
for (OALib2LibMap::iterator lit = _oaLib2LibMap.begin();
|
||||
lit != _oaLib2LibMap.end();
|
||||
lit++) {
|
||||
lit->first->close();
|
||||
}
|
||||
}
|
||||
|
||||
Cell* OAWrapper::getCell(const string& cellNameStr) {
|
||||
DataBase* db = DataBase::getDB();
|
||||
if (!db || !db->getRootLibrary() ) {
|
||||
return NULL;
|
||||
}
|
||||
Name cellName(cellNameStr);
|
||||
Cell* cell = findCellInLibraries(getOADesignLibraries(), cellName);
|
||||
if (!cell) {
|
||||
return NULL;
|
||||
}
|
||||
|
||||
set<Cell*>::const_iterator csit = _loadedCells.find(cell);
|
||||
if (csit == _loadedCells.end()) {
|
||||
Cell2OACellMap::const_iterator cit = _cell2OACellMap.find(cell);
|
||||
if (cit == _cell2OACellMap.end()) {
|
||||
cerr << "error : cannot find cell" << endl;
|
||||
exit(8);
|
||||
}
|
||||
oaCell* oa_Cell = cit->second;
|
||||
loadOACellInCell(oa_Cell, cell);
|
||||
_loadedCells.insert(cell);
|
||||
}
|
||||
return cell;
|
||||
}
|
||||
|
||||
oaLib* OAWrapper::openOALib(const oaScalarName& libOAName, const string& libPathStr) {
|
||||
|
||||
try {
|
||||
} catch (oaException &excp) {
|
||||
cout << "ERROR: " << excp.getMsg() << endl;
|
||||
exit(1);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
void OAWrapper::loadOALib(const string& libNameStr, const string& libPathStr, bool asDesignLibrary) {
|
||||
Name libName(libNameStr);
|
||||
Name2LibMap::const_iterator nit = _name2LibMap.find(libName);
|
||||
if (nit != _name2LibMap.end()) {
|
||||
Library* library = nit->second;
|
||||
//verify that it's the same library : name and path
|
||||
cerr << "already loaded" << endl;
|
||||
return;
|
||||
}
|
||||
|
||||
oaNativeNS oaNS;
|
||||
oaString libNameOAStr(libNameStr.c_str());
|
||||
oaScalarName libOAName(oaNS, libNameOAStr);
|
||||
|
||||
try {
|
||||
oaLib* oaLibrary = oaLib::open(libOAName, libPathStr.c_str());
|
||||
if (oaLibrary->isReadable()) {
|
||||
if (!oaLibrary->getAccess(oacReadLibAccess)) {
|
||||
cantLock();
|
||||
}
|
||||
|
||||
oaTechnology2Technology(oaLibrary);
|
||||
|
||||
//create Hurricane library
|
||||
Name libraryName = Name(libNameStr);
|
||||
DataBase* db = DataBase::getDB();
|
||||
if (!db) {
|
||||
cerr << "No DataBase" << endl;
|
||||
exit(8);
|
||||
}
|
||||
if (findLibraryByNameInDB(db, libraryName)) {
|
||||
cerr << "ERROR" << endl;
|
||||
exit(8);
|
||||
}
|
||||
|
||||
Library* library;
|
||||
if (asDesignLibrary) {
|
||||
library = Library::create(getOADesignLibraries(), Name(libNameStr));
|
||||
} else {
|
||||
library = Library::create(getOACellLibraries(), Name(libNameStr));
|
||||
}
|
||||
cerr << library << endl;
|
||||
|
||||
oaCollection<oaCell, oaLib> cells = oaLibrary->getCells();
|
||||
oaIter<oaCell> cellsIter(cells);
|
||||
|
||||
while (oaCell* cell = cellsIter.getNext()) {
|
||||
oaScalarName cellOAName;
|
||||
oaString cellNameString;
|
||||
cell->getName(cellOAName);
|
||||
cellOAName.get(cellNameString);
|
||||
|
||||
cerr << cellNameString << endl;
|
||||
|
||||
oaDesign* cellDesign = openDesign(oaNS, cell);
|
||||
if (cellDesign != NULL) {
|
||||
Cell* hCell = NULL;
|
||||
//logic part
|
||||
oaModule* module = cellDesign->getTopModule();
|
||||
if (module) {
|
||||
hCell = Cell::create(library, Name(cellNameString));
|
||||
hCell->setTerminal(false);
|
||||
if (!asDesignLibrary) {
|
||||
hCell->setFlattenLeaf(true);
|
||||
}
|
||||
cerr << hCell << endl;
|
||||
if (asDesignLibrary) {
|
||||
_cell2OACellMap[hCell] = cell;
|
||||
}
|
||||
oaCollection<oaModTerm, oaModule> oaModTerms = module->getTerms();
|
||||
oaIter<oaModTerm> oaModTermIter(oaModTerms);
|
||||
while (oaModTerm* modTerm = oaModTermIter.getNext()) {
|
||||
oaString oaModTermStr;
|
||||
modTerm->getName(oaNS, oaModTermStr);
|
||||
Net* net = Net::create(hCell, Name(oaModTermStr));
|
||||
net->setExternal(true);
|
||||
switch (modTerm->getTermType()) {
|
||||
case oacInputTermType:
|
||||
net->setDirection(Net::Direction::IN);
|
||||
break;
|
||||
case oacOutputTermType:
|
||||
net->setDirection(Net::Direction::OUT);
|
||||
break;
|
||||
case oacInputOutputTermType:
|
||||
net->setDirection(Net::Direction::INOUT);
|
||||
break;
|
||||
case oacTristateTermType:
|
||||
net->setDirection(Net::Direction::TRISTATE);
|
||||
break;
|
||||
default:
|
||||
net->setDirection(Net::Direction::UNDEFINED);
|
||||
break;
|
||||
}
|
||||
cerr << net << endl;
|
||||
oaModNet* oaNet = modTerm->getNet();
|
||||
if (oaNet->isGlobal()) {
|
||||
net->setGlobal(true);
|
||||
cerr << "global" << endl;
|
||||
}
|
||||
if (oaNet->getSigType() == oacClockSigType) {
|
||||
net->setType(Net::Type::CLOCK);
|
||||
} else if (oaNet->getSigType() == oacPowerSigType) {
|
||||
net->setGlobal(true);
|
||||
net->setType(Net::Type::POWER);
|
||||
} else if (oaNet->getSigType() == oacGroundSigType) {
|
||||
net->setGlobal(true);
|
||||
net->setType(Net::Type::GROUND);
|
||||
} else {
|
||||
net->setType(Net::Type::LOGICAL);
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
//physical part
|
||||
oaBlock* block = cellDesign->getTopBlock();
|
||||
if (block && hCell) {
|
||||
oaBox oa_box;
|
||||
block->getBBox(oa_box);
|
||||
cerr << "box values" << endl;
|
||||
cerr << oa_box.lowerLeft().x() << " , " << oa_box.lowerLeft().y() << endl;
|
||||
cerr << oa_box.upperRight().x() << " , " << oa_box.upperRight().y() << endl;
|
||||
cerr << endl;
|
||||
Point lowerLeft(DbU::db(oa_box.lowerLeft().x()), DbU::db(oa_box.lowerLeft().y()));
|
||||
Point upperRight(DbU::db(oa_box.upperRight().x()), DbU::db(oa_box.upperRight().y()));
|
||||
hCell->setAbutmentBox(Box(lowerLeft, upperRight));
|
||||
oaCollection<oaTerm, oaBlock> oaTerms = block->getTerms();
|
||||
oaIter<oaTerm> oaTermIter(oaTerms);
|
||||
while (oaTerm* term = oaTermIter.getNext()) {
|
||||
oaString oaTermStr;
|
||||
term->getName(oaNS, oaTermStr);
|
||||
Net* net = hCell->getNet(Name(oaTermStr));
|
||||
if (net) {
|
||||
cerr << "found net : " << net << endl;
|
||||
oaCollection<oaPin, oaTerm> oaPins(term->getPins());
|
||||
oaIter<oaPin> oaPinIter(oaPins);
|
||||
while (oaPin* pin = oaPinIter.getNext()) {
|
||||
oaString oaPinStr;
|
||||
pin->getName(oaPinStr);
|
||||
cerr << "pin : " << oaPinStr << endl;
|
||||
|
||||
oaCollection<oaPinFig, oaPin> oaPinFigs(pin->getFigs());
|
||||
oaIter<oaPinFig> oaPinFigIter(oaPinFigs);
|
||||
while (oaPinFig* pinFig = oaPinFigIter.getNext()) {
|
||||
cerr << "pinFig" << endl;
|
||||
if (pinFig->getType() == oacRectType) {
|
||||
oaRect* rect = static_cast<oaRect*>(pinFig);
|
||||
OALayerNum2LayerMap::iterator layerMapIt = _oaLayerNum2LayerMap.find(rect->getLayerNum());
|
||||
if (layerMapIt != _oaLayerNum2LayerMap.end()) {
|
||||
Layer* layer = layerMapIt->second;
|
||||
oaBox pinBBox;
|
||||
rect->getBBox(pinBBox);
|
||||
DbU::Unit width = DbU::db( (pinBBox.right() - pinBBox.left()) * (double)convertFactor);
|
||||
DbU::Unit height = DbU::db( (pinBBox.top() - pinBBox.bottom()) * (double)convertFactor);
|
||||
DbU::Unit cx = DbU::db(pinBBox.right() * (double)convertFactor) + width/2;
|
||||
DbU::Unit cy = DbU::db(pinBBox.top() * (double)convertFactor) + height/2;
|
||||
if (width > height) {
|
||||
Horizontal* horizontal=Horizontal::create(net, layer, cy, height, DbU::db(pinBBox.right() * (double)convertFactor), DbU::db(pinBBox.left() * (double)convertFactor));
|
||||
NetExternalComponents::setExternal(horizontal);
|
||||
} else {
|
||||
Vertical* vertical= Vertical::create(net, layer, cx, width, DbU::db(pinBBox.bottom() * (double)convertFactor), DbU::db(pinBBox.top() * (double)convertFactor));
|
||||
NetExternalComponents::setExternal(vertical);
|
||||
}
|
||||
} else {
|
||||
cerr << "cannot find layer..." << endl;
|
||||
}
|
||||
|
||||
cerr << "rect " << rect->getLayerNum() << endl;
|
||||
} else {
|
||||
cerr << "no rect" << endl;
|
||||
}
|
||||
}
|
||||
}
|
||||
} else {
|
||||
//TODO
|
||||
}
|
||||
}
|
||||
} else {
|
||||
cerr << "no block" << endl;
|
||||
}
|
||||
}
|
||||
}
|
||||
oaLibrary->releaseAccess();
|
||||
//oaLibrary->close(); do not close !!
|
||||
//FIXME save opened libraries ??
|
||||
}
|
||||
} catch (oaException &excp) {
|
||||
cout << "ERROR: " << excp.getMsg() << endl;
|
||||
exit(1);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
void OAWrapper::oaTechnology2Technology(oaLib* oaLibrary) {
|
||||
try {
|
||||
oaTech* tech = oaTech::open(oaLibrary);
|
||||
|
||||
if (!tech) {
|
||||
cout << "ERROR" << endl;
|
||||
exit(1);
|
||||
}
|
||||
if (_oaTechnology && tech != _oaTechnology) {
|
||||
cout << "ERROR" << endl;
|
||||
exit(1);
|
||||
}
|
||||
if (_oaTechnology) {
|
||||
return;
|
||||
}
|
||||
_oaTechnology = tech;
|
||||
|
||||
DataBase* db = DataBase::getDB();
|
||||
if (!db) {
|
||||
cerr << "No DataBase" << endl;
|
||||
exit(8);
|
||||
}
|
||||
Technology* technology = db->getTechnology();
|
||||
if (!technology) {
|
||||
technology = Technology::create(db, "OpenAccess");
|
||||
}
|
||||
|
||||
oaCollection<oaLayer, oaTech> oaLayers(_oaTechnology->getLayers());
|
||||
oaIter<oaLayer> oaLayerIt(oaLayers);
|
||||
while (oaLayer* oaLayer = oaLayerIt.getNext()) {
|
||||
oaString oaLayerStr;
|
||||
oaLayer->getName(oaLayerStr);
|
||||
Name layerName(oaLayerStr);
|
||||
Layer* layer = technology->getLayer(layerName);
|
||||
if (!layer) {
|
||||
layer = BasicLayer::create(technology, layerName, BasicLayer::Material::other, oaLayer->getNumber());
|
||||
}
|
||||
cerr << layer << endl;
|
||||
_oaLayerNum2LayerMap[oaLayer->getNumber()] = layer;
|
||||
}
|
||||
} catch (oaException &excp) {
|
||||
cout << "ERROR: " << excp.getMsg() << endl;
|
||||
exit(1);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
void OAWrapper::getDesigns(set<Cell*>& designCellSet) {
|
||||
getAllCells(getOADesignLibraries(), designCellSet);
|
||||
}
|
||||
|
||||
}
|
|
@ -1,49 +0,0 @@
|
|||
// x-----------------------------------------------------------------x
|
||||
// | This file is part of the hurricaneAMS Software. |
|
||||
// | Copyright (c) UPMC/LIP6 2008-2010, All Rights Reserved |
|
||||
// | =============================================================== |
|
||||
// | Author : Chistophe Alexandre |
|
||||
// | E-mail : Christophe.Alexandre@asim.lip6.fr |
|
||||
// x-----------------------------------------------------------------x
|
||||
#ifndef __OA_WRAPPER_H__
|
||||
#define __OA_WRAPPER_H__
|
||||
|
||||
#include <set>
|
||||
using namespace std;
|
||||
|
||||
#include "oa/oaDesignDB.h"
|
||||
using namespace oa;
|
||||
|
||||
namespace CRL {
|
||||
|
||||
class OAWrapper {
|
||||
public:
|
||||
static OAWrapper* getOAWrapper();
|
||||
void loadOALib(const string& libNameStr, const string& libPath, bool asDesignLibrary = false);
|
||||
Cell* getCell(const string& cellName);
|
||||
void getDesigns(set<Cell*>& designCellSet);
|
||||
|
||||
private:
|
||||
OAWrapper();
|
||||
~OAWrapper();
|
||||
void openLib(string& libPath);
|
||||
void oaTechnology2Technology(oaLib* lib);
|
||||
oaLib* openOALib(const oaScalarName& libOAName, const string& libPath);
|
||||
|
||||
typedef map<oaLib*, Library*> OALib2LibMap;
|
||||
typedef map<Name, Library*> Name2LibMap;
|
||||
typedef map<oaLayerNum, Layer*> OALayerNum2LayerMap;
|
||||
typedef map<Cell*, oaCell*> Cell2OACellMap;
|
||||
|
||||
static OAWrapper* _singleton;
|
||||
oaTech* _oaTechnology; // only one technology;
|
||||
OALayerNum2LayerMap _oaLayerNum2LayerMap;
|
||||
OALib2LibMap _oaLib2LibMap;
|
||||
Name2LibMap _name2LibMap;
|
||||
Cell2OACellMap _cell2OACellMap;
|
||||
set<Cell*> _loadedCells;
|
||||
};
|
||||
|
||||
}
|
||||
|
||||
#endif /* __OA_WRAPPER_H__ */
|
|
@ -1,397 +0,0 @@
|
|||
// x-----------------------------------------------------------------x
|
||||
// | This file is part of the hurricaneAMS Software. |
|
||||
// | Copyright (c) UPMC/LIP6 2008-2010, All Rights Reserved |
|
||||
// | =============================================================== |
|
||||
// | Author : Chistophe Alexandre |
|
||||
// | E-mail : Christophe.Alexandre@asim.lip6.fr |
|
||||
// x-----------------------------------------------------------------x
|
||||
#ifdef HAVE_OPENACCESS
|
||||
#include "oa/oaDesignDB.h"
|
||||
using namespace oa;
|
||||
#endif
|
||||
|
||||
#include "hurricane/DataBase.h"
|
||||
#include "hurricane/Library.h"
|
||||
#include "hurricane/Cell.h"
|
||||
#include "hurricane/NetExternalComponents.h"
|
||||
#include "hurricane/Segment.h"
|
||||
#include "hurricane/Pad.h"
|
||||
using namespace Hurricane;
|
||||
|
||||
#include "OpenAccessWrapper.h"
|
||||
|
||||
namespace {
|
||||
|
||||
#ifdef HAVE_OPENACCESS
|
||||
|
||||
#if 0
|
||||
oaModInstTerm* getModInstTerm(oaModInst* modInst, Plug* plug) {
|
||||
oaNativeNS ns;
|
||||
oaScalarName scPlugName(ns, getString(plug->getName()).c_str());
|
||||
oaName instTermName(scPlugName);
|
||||
oaModInstTerm* instTerm = oaModInstTerm::find(modInst, instTermName);
|
||||
if (instTerm) {
|
||||
return instTerm;
|
||||
}
|
||||
oaModule* masterModule = modInst->getMasterModule();
|
||||
oaModTerm* modTerm = oaModTerm::find(masterModule, instTermName);
|
||||
assert(modTerm);
|
||||
oaModNet* modNet = oaModNet::find(masterModule, instTermName);
|
||||
assert(modNet);
|
||||
instTerm = oaModInstTerm::create(modNet, modInst, instTermName);
|
||||
return instTerm;
|
||||
}
|
||||
#endif
|
||||
|
||||
oaString getDesignName(oaDesign* design) {
|
||||
oaNativeNS ns;
|
||||
oaString libName, cellName, viewName;
|
||||
design->getLibName(ns, libName);
|
||||
design->getCellName(ns, cellName);
|
||||
design->getViewName(ns, viewName);
|
||||
oaString designName = "<" + libName + "," + cellName + "," + viewName + ">";
|
||||
return designName;
|
||||
}
|
||||
|
||||
void printBlockTerms(oaBlock* block) {
|
||||
oaNativeNS ns;
|
||||
oaDesign* design = block->getDesign();
|
||||
cerr << " o Printing " << getDesignName(design) << " terms" << endl;
|
||||
oaIter<oaTerm> termIter(block->getTerms());
|
||||
while (oaTerm* term = termIter.getNext()) {
|
||||
oaString termName;
|
||||
term->getName(ns, termName);
|
||||
cerr << " - " << termName << endl;
|
||||
}
|
||||
cerr << endl;
|
||||
}
|
||||
|
||||
oaTermType getOATermType(const Net::Direction& direction) {
|
||||
switch (direction) {
|
||||
case Net::Direction::IN:
|
||||
return oacInputTermType;
|
||||
case Net::Direction::OUT:
|
||||
return oacOutputTermType;
|
||||
case Net::Direction::INOUT:
|
||||
return oacInputOutputTermType;
|
||||
case Net::Direction::TRISTATE:
|
||||
return oacTristateTermType;
|
||||
default:
|
||||
throw Error("unrecognized direction");
|
||||
}
|
||||
}
|
||||
|
||||
oaOrient getOAOrientFromOrientation(const Transformation::Orientation& orientation) {
|
||||
cerr << orientation << endl;
|
||||
switch (orientation) {
|
||||
case Transformation::Orientation::ID:
|
||||
return oacR0;
|
||||
case Transformation::Orientation::R1:
|
||||
return oacR90;
|
||||
case Transformation::Orientation::R2:
|
||||
return oacR180;
|
||||
case Transformation::Orientation::R3:
|
||||
return oacR270;
|
||||
case Transformation::Orientation::MX:
|
||||
return oacMX;
|
||||
case Transformation::Orientation::XR:
|
||||
return oacMXR90;
|
||||
case Transformation::Orientation::MY:
|
||||
return oacMY;
|
||||
case Transformation::Orientation::YR:
|
||||
return oacMYR90;
|
||||
default:
|
||||
throw Error("Unrecognized orientation");
|
||||
}
|
||||
}
|
||||
|
||||
void getOATransformFromTransformation(oaTransform& transform, const Transformation& transformation) {
|
||||
transform.set(transformation.getTx(),
|
||||
transformation.getTy(),
|
||||
getOAOrientFromOrientation(transformation.getOrientation()));
|
||||
}
|
||||
|
||||
void getOABoxForBox(oaBox& box, const Box& hbox) {
|
||||
box.set(hbox.getXMin(), hbox.getYMin(), hbox.getXMax(), hbox.getYMax());
|
||||
}
|
||||
|
||||
oaInstTerm* getInstTerm(oaInst* inst, Plug* plug) {
|
||||
oaNativeNS ns;
|
||||
oaScalarName scPlugName(ns, getString(plug->getMasterNet()->getName()).c_str());
|
||||
oaName instTermName(scPlugName);
|
||||
oaInstTerm* instTerm = oaInstTerm::find(inst, instTermName);
|
||||
if (instTerm) {
|
||||
return instTerm;
|
||||
}
|
||||
oaDesign* design = inst->getMaster();
|
||||
assert(design);
|
||||
oaBlock* masterBlock = design->getTopBlock();
|
||||
oaTerm* term = oaTerm::find(masterBlock, instTermName);
|
||||
if (!term) {
|
||||
cerr << "looking for " << plug->getName() << endl;
|
||||
printBlockTerms(masterBlock);
|
||||
}
|
||||
assert(term);
|
||||
instTerm = oaInstTerm::create(NULL, inst, term);
|
||||
return instTerm;
|
||||
}
|
||||
|
||||
class OADriver {
|
||||
private:
|
||||
typedef map<const Library*, oaLib*> Library2OALibMap;
|
||||
typedef map<const Cell*, oaDesign*> Cell2OADesignMap;
|
||||
typedef pair<oaModInst*, oaInst*> OAInstPair;
|
||||
typedef map<Instance*, oaInst*> Instance2OAInstsMap;
|
||||
|
||||
string _path;
|
||||
oaTech* _oaTech;
|
||||
oaBlock* _oaBlock;
|
||||
Library2OALibMap _library2OALibMap;
|
||||
Cell2OADesignMap _cell2OADesignMap;
|
||||
Instance2OAInstsMap _instance2OAInstMap;
|
||||
|
||||
public:
|
||||
|
||||
OADriver(const string& path):
|
||||
_path(path),
|
||||
_oaTech(NULL),
|
||||
_oaBlock(NULL),
|
||||
_library2OALibMap(),
|
||||
_cell2OADesignMap(),
|
||||
_instance2OAInstMap() {
|
||||
DataBase* db = DataBase::getDB();
|
||||
if (!db) {
|
||||
throw Error("no database");
|
||||
}
|
||||
Technology* technology = db->getTechnology();
|
||||
//createOATechForTechnology(technology);
|
||||
}
|
||||
|
||||
~OADriver() {
|
||||
cerr << "SAVING ALL" << endl;
|
||||
|
||||
#if 0
|
||||
oaNativeNS ns;
|
||||
string fullPath = _path + "/lib.defs";
|
||||
oaScalarName scPath(ns, fullPath.c_str());
|
||||
oaString strPath;
|
||||
scPath.get(strPath);
|
||||
oaLibDefList* libDefList = oaLibDefList::get(strPath, 'w');
|
||||
#endif
|
||||
|
||||
for (Cell2OADesignMap::iterator it = _cell2OADesignMap.begin();
|
||||
it != _cell2OADesignMap.end();
|
||||
++it) {
|
||||
cerr << it->first << endl;
|
||||
oaDesign* design = it->second;
|
||||
design->save();
|
||||
design->close();
|
||||
}
|
||||
|
||||
for (Library2OALibMap::iterator it = _library2OALibMap.begin();
|
||||
it != _library2OALibMap.end();
|
||||
++it) {
|
||||
oaLib* lib = it->second;
|
||||
#if 0
|
||||
oaScalarName libName;
|
||||
lib->getName(libName);
|
||||
oaString libPath;
|
||||
lib->getFullPath(libPath);
|
||||
oaLibDef* libDef = oaLibDef::create(libDefList, libName, libPath);
|
||||
#endif
|
||||
lib->close();
|
||||
}
|
||||
|
||||
// libDefList->save();
|
||||
cerr << "ALL SAVED" << endl;
|
||||
}
|
||||
|
||||
|
||||
oaLib* getOALibForLibrary(const Library* library) {
|
||||
Library2OALibMap::iterator it = _library2OALibMap.find(library);
|
||||
if (it != _library2OALibMap.end()) {
|
||||
return it->second;
|
||||
}
|
||||
oaNativeNS ns;
|
||||
const char* strNameLib = getString(library->getName()).c_str();
|
||||
if (!strcmp(strNameLib, ".")) {
|
||||
strNameLib = "worklib";
|
||||
}
|
||||
string strPathLib = _path + '/' + strNameLib;
|
||||
//cerr << "Creating new Library for Lib name="
|
||||
// << strNameLib << " to filesystem path=" << strPathLib << endl;
|
||||
oaScalarName scNameLib(ns, strNameLib);
|
||||
cerr << "Creating new Library for Lib name="
|
||||
<< strNameLib << " to filesystem path=" << strPathLib << endl;
|
||||
oaLib* lib = oaLib::create(scNameLib, strPathLib.c_str(), oacSharedLibMode, "oaDMFileSys");
|
||||
|
||||
//oaScalarName libName;
|
||||
//_oaTech->getLibName(libName);
|
||||
//oaTech::attach(lib, libName);
|
||||
//assert(oaTech::find(lib));
|
||||
|
||||
oaTech* oaTech = oaTech::create(lib);
|
||||
oaTech->setDefaultManufacturingGrid(10);
|
||||
oaTech->setDBUPerUU(oaViewType::get(oacMaskLayout), 1000);
|
||||
|
||||
oaLayer* drawingLayer = oaPhysicalLayer::create(oaTech, "drawing", 0);
|
||||
|
||||
_library2OALibMap[library] = lib;
|
||||
|
||||
return lib;
|
||||
}
|
||||
|
||||
void createOATechForTechnology(const Technology* technology) {
|
||||
string strPathLib = _path + "/techlib";
|
||||
oaNativeNS ns;
|
||||
oaScalarName scNameLib(ns, "techlib");
|
||||
oaLib* lib = oaLib::create(scNameLib, strPathLib.c_str(), oacSharedLibMode, "oaDMFileSys");
|
||||
_oaTech = oaTech::create(lib);
|
||||
|
||||
oaLayer* drawingLayer = oaPhysicalLayer::create(_oaTech, "drawing", 0);
|
||||
|
||||
#if 0
|
||||
for_each_layer(hlayer, technology->getLayers()) {
|
||||
oaString strLayerName = getString(hlayer->getName()).c_str();
|
||||
oaPhysicalLayer* layer = oaPhysicalLayer::create(techno, strLayerName, layerNum++);
|
||||
end_for;
|
||||
}
|
||||
#endif
|
||||
lib->close();
|
||||
}
|
||||
|
||||
oaDesign* getOADesignForCell(const Cell* cell) {
|
||||
cerr << "getOADesignForCell " << cell << endl;
|
||||
Cell2OADesignMap::iterator it = _cell2OADesignMap.find(cell);
|
||||
if (it != _cell2OADesignMap.end()) {
|
||||
return it->second;
|
||||
}
|
||||
|
||||
oaNativeNS ns;
|
||||
oaLib* lib = getOALibForLibrary(cell->getLibrary());
|
||||
oaScalarName scNameDesign(ns, getString(cell->getName()).c_str());
|
||||
oaScalarName scNameView(ns, "layout");
|
||||
oaScalarName scNameLib;
|
||||
lib->getName(scNameLib);
|
||||
|
||||
oaDesign* design = oaDesign::open(scNameLib, scNameDesign, scNameView, oaViewType::get(oacSchematic), 'w');
|
||||
//oaModule* module = oaModule::create(design);
|
||||
oaBlock* block = oaBlock::create(design);
|
||||
oaTech* tech = oaTech::find(lib);
|
||||
assert(tech);
|
||||
oaBox box;
|
||||
getOABoxForBox(box, cell->getAbutmentBox());
|
||||
cerr << cell << " " << cell->getAbutmentBox() << endl;
|
||||
oaRect::create(block, 0, oaPurpose::get(tech, oacDrawingPurposeType)->getNumber(), box);
|
||||
|
||||
for_each_instance(instance, cell->getInstances()) {
|
||||
Cell* masterCell = instance->getMasterCell();
|
||||
oaDesign* masterDesign = getOADesignForCell(masterCell);
|
||||
oaScalarName scDesignName;
|
||||
masterDesign->getCellName(scDesignName);
|
||||
oaString strDesignName;
|
||||
scDesignName.get(strDesignName);
|
||||
//oaModule* masterModule = masterDesign->getTopModule();
|
||||
//assert(masterModule);
|
||||
oaScalarName scInstName(ns, getString(instance->getName()).c_str());
|
||||
//oaModScalarInst* modInst = oaModScalarInst::create(masterModule, masterDesign, scInstName);
|
||||
|
||||
oaTransform transform;
|
||||
getOATransformFromTransformation(transform, instance->getTransformation());
|
||||
oaScalarInst* blockInst = oaScalarInst::create(block, masterDesign, scInstName, transform);
|
||||
_instance2OAInstMap[instance] = blockInst;
|
||||
end_for;
|
||||
}
|
||||
|
||||
for_each_net(net, cell->getNets()) {
|
||||
oaScalarName scNetName(ns, getString(net->getName()).c_str());
|
||||
//oaModNet* modNet = oaModScalarNet::create(module, scNetName, oacAnalogSigType);
|
||||
oaScalarNet* blockNet = oaScalarNet::create(block, scNetName, oacAnalogSigType);
|
||||
if (net->isExternal()) {
|
||||
oaTerm* term = oaTerm::create(blockNet, scNetName, getOATermType(net->getDirection()));
|
||||
oaPin* pin = oaPin::create(term);
|
||||
oaTech* tech = oaTech::find(lib);
|
||||
assert(tech);
|
||||
Components externalComponents = NetExternalComponents::get(net);
|
||||
for_each_component(component, externalComponents) {
|
||||
oaBox box;
|
||||
getOABoxForBox(box, component->getBoundingBox());
|
||||
oaRect* rect = oaRect::create(block, 0,
|
||||
oaPurpose::get(tech, oacDrawingPurposeType)->getNumber(),
|
||||
box);
|
||||
rect->addToPin(pin);
|
||||
end_for;
|
||||
}
|
||||
}
|
||||
|
||||
for_each_plug(plug, net->getPlugs()) {
|
||||
Instance* instance = plug->getInstance();
|
||||
Instance2OAInstsMap::iterator it = _instance2OAInstMap.find(instance);
|
||||
assert(it != _instance2OAInstMap.end());
|
||||
//oaModInst* modInst = it->second.first;
|
||||
oaInst* blockInst = it->second;
|
||||
//oaModInstTerm* modInstTerm = getModInstTerm(modInst, plug);
|
||||
oaInstTerm* instTerm = getInstTerm(blockInst, plug);
|
||||
//modInstTerm->addToNet(modNet);
|
||||
instTerm->addToNet(blockNet);
|
||||
end_for;
|
||||
}
|
||||
|
||||
for_each_component(component, net->getComponents()) {
|
||||
end_for;
|
||||
}
|
||||
end_for;
|
||||
}
|
||||
|
||||
_cell2OADesignMap[cell] = design;
|
||||
return design;
|
||||
}
|
||||
};
|
||||
#endif
|
||||
|
||||
}
|
||||
|
||||
namespace CRL {
|
||||
|
||||
void OpenAccessWrapper::oaLibLoader(const string& libPath, Library *lib, Catalog &catalog) {
|
||||
cerr << "\nDummy OpenAccess lib loader call for " << libPath << endl;
|
||||
}
|
||||
|
||||
void OpenAccessWrapper::oaDesignLoader(const string& path, Cell* cell) {
|
||||
cerr << "\nDummy OpenAccess design loader call for " << path << endl;
|
||||
}
|
||||
|
||||
void OpenAccessWrapper::oaDriver(const string& path, Cell* cell, unsigned int& saveState) {
|
||||
#ifdef HAVE_OPENACCESS
|
||||
//for the moment a driver for hurricaneAMS
|
||||
//save the Cell and all used Cells
|
||||
cerr << "Saving " << cell << " in " << path << endl;
|
||||
try {
|
||||
|
||||
oaDesignInit(
|
||||
oacAPIMajorRevNumber,
|
||||
oacAPIMinorRevNumber,
|
||||
oacDataModelRevNumber);
|
||||
|
||||
Library* library = cell->getLibrary();
|
||||
string pathLib = path + "/" + getString(library->getName());
|
||||
|
||||
const char* strPathLib = pathLib.c_str();
|
||||
|
||||
// if (oaLib::exists(strPathLib)) {
|
||||
// throw Error("impossible to drive Cell, there is an existing OpenAccess db");
|
||||
// }
|
||||
|
||||
OADriver oaDriver(pathLib);
|
||||
oaDesign* design = oaDriver.getOADesignForCell(cell);
|
||||
|
||||
} catch (oaException &excp) {
|
||||
cerr << "ERROR: " << excp.getMsg() << endl;
|
||||
exit(1);
|
||||
}
|
||||
#else
|
||||
cerr << "\nDummy OpenAccess driver call for " << path << endl;
|
||||
#endif
|
||||
}
|
||||
|
||||
}
|
|
@ -1,32 +0,0 @@
|
|||
// x-----------------------------------------------------------------x
|
||||
// | This file is part of the hurricaneAMS Software. |
|
||||
// | Copyright (c) UPMC/LIP6 2008-2010, All Rights Reserved |
|
||||
// | =============================================================== |
|
||||
// | Author : Chistophe Alexandre |
|
||||
// | E-mail : Christophe.Alexandre@asim.lip6.fr |
|
||||
// x-----------------------------------------------------------------x
|
||||
#ifndef __OPENACCESSWRAPPER_H
|
||||
#define __OPENACCESSWRAPPER_H
|
||||
|
||||
#include <string>
|
||||
using namespace std;
|
||||
|
||||
namespace Hurricane {
|
||||
class Library;
|
||||
class Cell;
|
||||
}
|
||||
|
||||
namespace CRL {
|
||||
|
||||
class Catalog;
|
||||
|
||||
class OpenAccessWrapper {
|
||||
public:
|
||||
static void oaLibLoader(const string& libPath, Hurricane::Library* lib, Catalog& catalog);
|
||||
static void oaDesignLoader(const string& path, Hurricane::Cell* cell);
|
||||
static void oaDriver(const string& cellPath, Cell* cell, unsigned int& saveState);
|
||||
};
|
||||
|
||||
}
|
||||
|
||||
#endif /*__OPENACCESSWRAPPER_H */
|
|
@ -44,9 +44,9 @@ void testCell(Cell* dev,char* pathToTest){
|
|||
return;
|
||||
}
|
||||
system((string("mkdir -p ") + string(pathToTest)).c_str());
|
||||
/* cerr << "driving GDS" << endl;
|
||||
GdsDriver(dev).save(string(pathToTest) + "/GDSdrive");
|
||||
*/
|
||||
cerr << "driving GDS" << endl;
|
||||
GdsDriver(dev).save(string(pathToTest) + "/" + getString(dev->getName()) + ".gds");
|
||||
|
||||
cerr << "driving OA" << endl;
|
||||
OADriver(dev).save(string(pathToTest) + "/OAdrive");
|
||||
}
|
||||
|
|
Loading…
Reference in New Issue