2010-03-09 09:24:29 -06:00
|
|
|
|
|
|
|
// -*- C++ -*-
|
|
|
|
//
|
|
|
|
// This file is part of the Coriolis Software.
|
2013-12-03 18:58:58 -06:00
|
|
|
// Copyright (c) UPMC/LIP6 2008-2013, All Rights Reserved
|
2010-03-09 09:24:29 -06:00
|
|
|
//
|
2013-12-03 18:58:58 -06:00
|
|
|
// +-----------------------------------------------------------------+
|
2010-03-09 09:24:29 -06:00
|
|
|
// | C O R I O L I S |
|
|
|
|
// | K a t a b a t i c - Routing Toolbox |
|
|
|
|
// | |
|
|
|
|
// | Author : Jean-Paul CHAPUT |
|
2013-12-03 18:58:58 -06:00
|
|
|
// | E-mail : Jean-Paul.Chaput@lip6.fr |
|
2010-03-09 09:24:29 -06:00
|
|
|
// | =============================================================== |
|
|
|
|
// | C++ Module : "./GCellGrid.cpp" |
|
2013-12-03 18:58:58 -06:00
|
|
|
// +-----------------------------------------------------------------+
|
2010-03-09 09:24:29 -06:00
|
|
|
|
|
|
|
|
|
|
|
#include "hurricane/Error.h"
|
|
|
|
#include "hurricane/Cell.h"
|
|
|
|
#include "knik/KnikEngine.h"
|
|
|
|
#include "katabatic/Session.h"
|
|
|
|
#include "katabatic/GCellGrid.h"
|
|
|
|
#include "katabatic/KatabaticEngine.h"
|
|
|
|
|
|
|
|
|
|
|
|
namespace Katabatic {
|
|
|
|
|
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
using Hurricane::tab;
|
|
|
|
using Hurricane::ltracein;
|
|
|
|
using Hurricane::ltraceout;
|
|
|
|
using Hurricane::inltrace;
|
|
|
|
using Hurricane::Error;
|
|
|
|
using Hurricane::ForEachIterator;
|
|
|
|
using Knik::KnikEngine;
|
|
|
|
|
|
|
|
|
|
|
|
const char* missingKnikEngine =
|
|
|
|
"%s :\n\n"
|
|
|
|
" Hey, Banana! You forgot to run the Knik global router on %s.\n";
|
|
|
|
|
|
|
|
|
|
|
|
// -------------------------------------------------------------------
|
|
|
|
// Class : "Katabatic::GCellGrid".
|
|
|
|
|
|
|
|
|
|
|
|
GCellGrid::GCellGrid ( KatabaticEngine* ktbt )
|
2010-08-22 07:37:48 -05:00
|
|
|
: Grid<GCell> (ktbt->getCell()->getAbutmentBox())
|
|
|
|
, _katabatic (ktbt)
|
2010-12-12 15:43:25 -06:00
|
|
|
, _densityMode (MaxDensity)
|
2010-08-22 07:37:48 -05:00
|
|
|
, _hEdgeCapacity(ktbt->getConfiguration()->getHEdgeCapacity())
|
|
|
|
, _vEdgeCapacity(ktbt->getConfiguration()->getVEdgeCapacity())
|
2010-03-09 09:24:29 -06:00
|
|
|
{ }
|
|
|
|
|
|
|
|
|
|
|
|
void GCellGrid::_postCreate ()
|
|
|
|
{
|
2014-03-15 04:47:37 -05:00
|
|
|
KnikEngine* knik;
|
2010-03-09 09:24:29 -06:00
|
|
|
knik = KnikEngine::get ( getCell() );
|
|
|
|
if ( !knik )
|
|
|
|
throw Error ( missingKnikEngine, "GCellGrid::_postCreate()", getString(getCell()).c_str() );
|
|
|
|
|
|
|
|
vector<DbU::Unit> knikGraduations;
|
|
|
|
|
|
|
|
knik->getHorizontalCutLines ( knikGraduations );
|
|
|
|
for ( size_t i=0 ; i<knikGraduations.size() ; i++ )
|
|
|
|
_yGraduations.addGraduation ( knikGraduations[i] );
|
|
|
|
_yGraduations.sort ();
|
|
|
|
_rows = _yGraduations.getSize() - 1;
|
|
|
|
|
|
|
|
knik->getVerticalCutLines ( knikGraduations );
|
|
|
|
for ( size_t i=0 ; i<knikGraduations.size() ; i++ )
|
|
|
|
_xGraduations.addGraduation ( knikGraduations[i] );
|
|
|
|
_xGraduations.sort ();
|
|
|
|
_columns = _xGraduations.getSize() - 1;
|
|
|
|
|
|
|
|
_rawSize = _columns * _rows;
|
|
|
|
|
|
|
|
ltrace(80) << "Katabatic GCell Matrix [" << getColumns() << "x" << getRows() << "]" << endl;
|
|
|
|
ltracein(80);
|
|
|
|
ltrace(80) << "_xGraduations := " << _xGraduations._print() << endl;
|
|
|
|
ltrace(80) << "_yGraduations := " << _yGraduations._print() << endl;
|
|
|
|
|
|
|
|
for ( unsigned int row=0 ; row < getRows() ; row++ ) {
|
|
|
|
for ( unsigned int column=0; column<getColumns() ; column++ ) {
|
|
|
|
Box boundingBox = Box ( _xGraduations[column ]
|
|
|
|
, _yGraduations[row ]
|
|
|
|
, _xGraduations[column+1]
|
|
|
|
, _yGraduations[row +1]
|
|
|
|
);
|
|
|
|
|
|
|
|
_gcells.push_back ( GCell::create ( this, _gcells.size(), boundingBox ) );
|
|
|
|
|
|
|
|
ltrace(80) << "GCell [" << column << "," << row << "] "
|
|
|
|
<< _gcells[_gcells.size()-1] << endl;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cmess1 << " o Reading GCell grid from Knik ["
|
2010-05-03 04:16:28 -05:00
|
|
|
<< getColumns() << "x" << getRows() << "] "
|
|
|
|
<< "(" << (getColumns()*getRows()) << ")." << endl;
|
2010-03-09 09:24:29 -06:00
|
|
|
|
|
|
|
ltraceout(80);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
GCellGrid* GCellGrid::create ( KatabaticEngine* ktbt )
|
|
|
|
{
|
|
|
|
GCellGrid* grid = new GCellGrid ( ktbt );
|
|
|
|
grid->_postCreate ();
|
|
|
|
return grid;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
GCellGrid::~GCellGrid ()
|
|
|
|
{ }
|
|
|
|
|
|
|
|
|
|
|
|
void GCellGrid::_preDestroy ()
|
|
|
|
{
|
|
|
|
ltrace(90) << "GCellGrid::_preDestroy()" << endl;
|
|
|
|
ltracein(90);
|
|
|
|
|
|
|
|
vector<GCell*>::iterator it = _gcells.begin();
|
|
|
|
vector<GCell*>::iterator end = _gcells.end ();
|
|
|
|
for ( ; it != end ; it++ ) (*it)->destroy ();
|
|
|
|
|
|
|
|
ltraceout(90);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Cell* GCellGrid::getCell () const
|
|
|
|
{
|
|
|
|
return _katabatic->getCell();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-04-23 08:13:54 -05:00
|
|
|
Interval GCellGrid::getUSide ( unsigned int direction ) const
|
|
|
|
{
|
|
|
|
Interval side;
|
|
|
|
switch ( direction ) {
|
|
|
|
default:
|
|
|
|
case Constant::Horizontal: side = Interval(_boundingBox.getXMin(),_boundingBox.getXMax()); break;
|
|
|
|
case Constant::Vertical: side = Interval(_boundingBox.getYMin(),_boundingBox.getYMax()); break;
|
|
|
|
}
|
|
|
|
return side;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
void GCellGrid::updateContacts ( unsigned int flags )
|
2010-03-09 09:24:29 -06:00
|
|
|
{
|
2013-12-03 18:58:58 -06:00
|
|
|
if (flags & KbOpenSession) Session::open( _katabatic );
|
2010-03-09 09:24:29 -06:00
|
|
|
|
|
|
|
forEach ( GCell*, gcell, getGCells() )
|
2013-12-03 18:58:58 -06:00
|
|
|
gcell->updateContacts();
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
if (flags & KbOpenSession) Session::close();
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
size_t GCellGrid::updateDensity ()
|
|
|
|
{
|
|
|
|
size_t saturateds = 0;
|
|
|
|
forEach ( GCell*, gcell, getGCells() ) {
|
|
|
|
saturateds += gcell->updateDensity ();
|
|
|
|
}
|
|
|
|
return saturateds;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
size_t GCellGrid::checkDensity () const
|
|
|
|
{
|
|
|
|
size_t saturateds = 0;
|
|
|
|
forEach ( GCell*, gcell, const_cast<GCellGrid*>(this)->getGCells() ) {
|
|
|
|
saturateds += gcell->checkDensity ();
|
|
|
|
}
|
|
|
|
return saturateds;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool GCellGrid::checkEdgeSaturation ( float threshold ) const
|
|
|
|
{
|
|
|
|
bool overload = false;
|
|
|
|
forEach ( GCell*, gcell, const_cast<GCellGrid*>(this)->getGCells() ) {
|
|
|
|
overload = gcell->checkEdgeSaturation(threshold) or overload;
|
|
|
|
}
|
|
|
|
return overload;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
string GCellGrid::_getTypeName () const
|
|
|
|
{
|
|
|
|
return _TName("GCellGrid");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
string GCellGrid::_getString () const
|
|
|
|
{
|
|
|
|
return "<" + _getTypeName() + " "
|
|
|
|
+ getString(getRows()) + "x" + getString(getColumns()) + ">";
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void GCellGrid::_xmlWrite ( ostream& o )
|
|
|
|
{
|
|
|
|
updateDensity ();
|
|
|
|
|
|
|
|
o << "<GCellGrid Cell=\"" << getCell()->getName() << "\">" << endl;
|
|
|
|
o << "<!-- GRID Lower Left Corner Density -->" << endl;
|
|
|
|
o << "<!-- column row XMin YMin hor ver con -->" << endl;
|
|
|
|
|
|
|
|
unsigned int row = 0;
|
|
|
|
forEach ( GCell*, gcell, getGCells() ) {
|
|
|
|
if ( gcell->getRow() > row ) {
|
|
|
|
o << endl;
|
|
|
|
o << "<!-- column row XMin YMin hor ver con -->" << endl;
|
|
|
|
row = gcell->getRow();
|
|
|
|
}
|
|
|
|
|
|
|
|
o << " "; gcell->_xmlWrite ( o ); o << endl;
|
|
|
|
}
|
|
|
|
|
|
|
|
o << "</GCellGrid>" << endl;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Record* GCellGrid::_getRecord () const
|
|
|
|
{
|
|
|
|
Record* record = Grid<GCell>::_getRecord ();
|
|
|
|
record->add ( getSlot ( "_katabatic", _katabatic ) );
|
|
|
|
return record;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
} // End of Katabatic namespace.
|