coriolis/katabatic/src/PowerRails.cpp

497 lines
15 KiB
C++

// -*- C++ -*-
//
// This file is part of the Coriolis Software.
// Copyright (c) UPMC/LIP6 2008-2013, All Rights Reserved
//
// +-----------------------------------------------------------------+
// | C O R I O L I S |
// | K a t a b a t i c - Routing Toolbox |
// | |
// | Author : Jean-Paul CHAPUT |
// | E-mail : Jean-Paul.Chaput@lip6.fr |
// | =============================================================== |
// | C++ Module : "./PowerRails.cpp" |
// +-----------------------------------------------------------------+
#include <algorithm>
#include "hurricane/Warning.h"
#include "hurricane/DataBase.h"
#include "hurricane/Technology.h"
#include "hurricane/Horizontal.h"
#include "hurricane/Vertical.h"
#include "hurricane/Net.h"
#include "hurricane/NetExternalComponents.h"
#include "hurricane/Cell.h"
#include "hurricane/Instance.h"
#include "crlcore/RoutingLayerGauge.h"
#include "crlcore/RoutingGauge.h"
#include "katabatic/KatabaticEngine.h"
namespace {
using namespace std;
using namespace Hurricane;
// -------------------------------------------------------------------
// Class : "::RailSegment".
class RailSegment {
public:
RailSegment ( DbU::Unit axis, DbU::Unit width );
inline DbU::Unit getAxis () const;
inline DbU::Unit getMin () const;
inline DbU::Unit getMax () const;
inline const Interval& getSpan () const;
inline DbU::Unit getWidth () const;
void merge ( DbU::Unit );
void doLayout ( Cell*, Net*, const Layer*, Constant::Direction ) const;
private:
DbU::Unit _axis;
DbU::Unit _width;
Interval _span;
};
RailSegment::RailSegment ( DbU::Unit axis, DbU::Unit width )
: _axis (axis)
, _width(width)
, _span ()
{ }
inline DbU::Unit RailSegment::getAxis () const { return _axis; }
inline DbU::Unit RailSegment::getMin () const { return _span.getVMin(); }
inline DbU::Unit RailSegment::getMax () const { return _span.getVMax(); }
inline const Interval& RailSegment::getSpan () const { return _span; }
inline DbU::Unit RailSegment::getWidth () const { return _width; }
inline void RailSegment::merge ( DbU::Unit bound ) { _span.merge(bound); }
void RailSegment::doLayout ( Cell* cell, Net* net, const Layer* layer, Constant::Direction direction ) const
{
Segment* segment = NULL;
switch ( direction ) {
case Constant::Horizontal:
segment = Horizontal::create( net
, layer
, _axis
, _width
, _span.getVMin()
, _span.getVMax() );
break;
case Constant::Vertical:
segment = Vertical::create( net
, layer
, _axis
, _width
, _span.getVMin()
, _span.getVMax() );
break;
}
if (segment)
NetExternalComponents::setExternal( segment );
}
// -------------------------------------------------------------------
// Class : "::PowerRail".
class PowerRail {
public:
struct CompareByAxis : binary_function<PowerRail*,PowerRail*,bool> {
bool operator() ( const PowerRail* lhs, const PowerRail* rhs );
};
public:
PowerRail ( Net::Type, const Layer*, Constant::Direction, DbU::Unit axis );
~PowerRail ();
inline Net::Type getType () const;
inline const Layer* getLayer () const;
inline DbU::Unit getAxis () const;
void merge ( DbU::Unit width, DbU::Unit min, DbU::Unit max );
void doLayout ( Cell*, Net* powerNet, Net* groundNet ) const;
private:
Net::Type _type;
const Layer* _layer;
Constant::Direction _direction;
DbU::Unit _axis;
list<RailSegment*> _segments;
};
PowerRail::PowerRail ( Net::Type type, const Layer* layer, Constant::Direction direction, DbU::Unit axis )
: _type (type)
, _layer (layer)
, _direction(direction)
, _axis (axis)
, _segments ()
{ }
PowerRail::~PowerRail ()
{
while (not _segments.empty()) {
delete _segments.front();
_segments.pop_front();
}
}
inline Net::Type PowerRail::getType () const { return _type; }
inline const Layer* PowerRail::getLayer () const { return _layer; }
inline DbU::Unit PowerRail::getAxis () const { return _axis; }
void PowerRail::merge ( DbU::Unit width, DbU::Unit min, DbU::Unit max )
{
RailSegment* inserted = NULL;
list<RailSegment*>::iterator isegment = _segments.begin();
for ( ; (isegment != _segments.end()) && !inserted ; isegment++ ) {
if ((*isegment)->getWidth() != width) continue;
if ((*isegment)->getMin() > max) {
inserted = new RailSegment ( _axis, width );
inserted->merge( min );
inserted->merge( max );
_segments.insert( isegment, inserted );
break;
}
if ((*isegment)->getMax() < min) continue;
inserted = *isegment;
(*isegment)->merge( min );
(*isegment)->merge( max );
list<RailSegment*>::iterator imerge = isegment;
if (imerge != _segments.end()) imerge++;
while (imerge != _segments.end()) {
if ((*imerge)->getMin() > (*isegment)->getMax()) break;
(*isegment)->merge( (*imerge)->getMax() );
delete *imerge;
_segments.erase( imerge );
imerge = isegment;
imerge++;
}
break;
}
if (not inserted) {
inserted = new RailSegment ( _axis, width );
inserted->merge( min );
inserted->merge( max );
_segments.insert( _segments.end(), inserted );
}
}
void PowerRail::doLayout ( Cell* cell, Net* powerNet, Net* groundNet ) const
{
//const Layer* layer = DataBase::getDB()->getTechnology()->getLayer("METAL1");
Net* railNet = (_type == Net::Type::POWER) ? powerNet : groundNet;
list<RailSegment*>::const_iterator isegment = _segments.begin();
for ( ; isegment != _segments.end() ; isegment++ )
(*isegment)->doLayout( cell, railNet, _layer, _direction );
}
bool PowerRail::CompareByAxis::operator() ( const PowerRail* lhs, const PowerRail* rhs )
{ return lhs->getAxis() < rhs->getAxis(); }
// -------------------------------------------------------------------
// Class : "::PowerPlane".
class PowerPlane {
public:
PowerPlane ( const Layer* );
~PowerPlane ();
inline const Layer* getLayer () const;
size_t find ( DbU::Unit axis, Constant::Direction ) const;
void merge ( Net::Type, Constant::Direction, DbU::Unit axis, DbU::Unit width, DbU::Unit min, DbU::Unit max );
void doLayout ( Cell* cell, Net* powerNet, Net* groundNet ) const;
private:
const Layer* _layer;
vector<PowerRail*> _hrails;
vector<PowerRail*> _vrails;
};
PowerPlane::PowerPlane ( const Layer* layer )
: _layer (layer)
, _hrails()
, _vrails()
{ }
PowerPlane::~PowerPlane ()
{
while (not _hrails.empty()) {
delete _hrails.back();
_hrails.pop_back();
}
while (not _vrails.empty()) {
delete _vrails.back();
_vrails.pop_back();
}
}
size_t PowerPlane::find ( DbU::Unit axis, Constant::Direction direction ) const
{
PowerRail bound (Net::Type::GROUND,NULL,Constant::Horizontal,axis);
if (direction == Constant::Horizontal) {
vector<PowerRail*>::const_iterator it
= lower_bound( _hrails.begin(), _hrails.end(), &bound,PowerRail::CompareByAxis() );
return it - _hrails.begin();
}
vector<PowerRail*>::const_iterator it
= lower_bound( _vrails.begin(), _vrails.end(), &bound,PowerRail::CompareByAxis() );
return it - _vrails.begin();
}
void PowerPlane::merge ( Net::Type type
, Constant::Direction direction
, DbU::Unit axis
, DbU::Unit width
, DbU::Unit rmin
, DbU::Unit rmax
)
{
vector<PowerRail*>* rails;
switch ( direction ) {
case Constant::Vertical : rails = &_vrails; break;
case Constant::Horizontal:
default:
rails = &_hrails; break;
}
size_t i = find( axis, direction );
if ( (i == rails->size()) or ((*rails)[i]->getAxis() != axis) ) {
PowerRail* rail = new PowerRail (type,_layer,direction,axis);
rail->merge( width, rmin, rmax );
rails->push_back( rail );
sort( rails->begin(), rails->end(), PowerRail::CompareByAxis() );
} else {
if ((*rails)[i]->getType() != type) {
cerr << Error("Short between power rails at %d.",DbU::getValueString(axis).c_str()) << endl;
return;
}
(*rails)[i]->merge( width, rmin, rmax );
}
}
void PowerPlane::doLayout ( Cell* cell, Net* powerNet, Net* groundNet ) const
{
for ( size_t i=0 ; i<_hrails.size() ; i++ )
_hrails[i]->doLayout( cell, powerNet, groundNet );
for ( size_t i=0 ; i<_vrails.size() ; i++ )
_vrails[i]->doLayout( cell, powerNet, groundNet );
}
// -------------------------------------------------------------------
// Class : "::PowerRail".
class PowerGrid {
public:
PowerGrid ( Cell* );
~PowerGrid ();
PowerPlane* getPlane ( const Layer* );
void merge ( const Transformation&, Horizontal* );
void merge ( const Transformation&, Vertical* );
void doLayout () const;
private:
Cell* _cell;
Net* _powerNet;
Net* _groundNet;
map<const Layer*,PowerPlane*> _planes;
};
PowerGrid::PowerGrid ( Cell* cell )
: _cell (cell)
, _powerNet (NULL)
, _groundNet(NULL)
, _planes ()
{
forEach ( Net*, inet, _cell->getNets() ) {
if (inet->getType() == Net::Type::POWER) {
if (not inet->isExternal()) {
cerr << Warning("Ignoring non-external power net %s."
,getString(*inet).c_str()) << endl;
continue;
}
_powerNet = *inet;
break;
}
}
if (not _powerNet)
cerr << Error("Missing POWER net in Cell %s.",getString(_cell->getName()).c_str()) << endl;
forEach ( Net*, inet, _cell->getNets() ) {
if (inet->getType() == Net::Type::GROUND) {
if (not inet->isExternal()) {
cerr << Warning("Ignoring non-external ground net %s."
,getString(*inet).c_str()) << endl;
continue;
}
_groundNet = *inet;
break;
}
}
if (not _groundNet)
cerr << Error("Missing GROUND net in Cell %s.",getString(_cell->getName()).c_str()) << endl;
}
PowerGrid::~PowerGrid ()
{
map<const Layer*,PowerPlane*>::iterator iplane = _planes.begin();
for ( ; iplane != _planes.end() ; iplane++ )
delete iplane->second;
}
PowerPlane* PowerGrid::getPlane ( const Layer* layer )
{
map<const Layer*,PowerPlane*>::iterator iplane = _planes.find(layer);
if (iplane != _planes.end())
return iplane->second;
PowerPlane* plane = new PowerPlane ( layer );
_planes.insert( make_pair(layer,plane) );
return plane;
}
void PowerGrid::merge ( const Transformation& transformation, Horizontal* horizontal )
{
PowerPlane* plane = getPlane ( horizontal->getLayer() );
Point source = horizontal->getSourcePosition();
Point target = horizontal->getTargetPosition();
transformation.applyOn( source );
transformation.applyOn( target );
if (source.getX() > target.getX()) swap( source, target );
plane->merge( horizontal->getNet()->getType()
, Constant::Horizontal
, source.getY()
, horizontal->getWidth()
, source.getX()
, target.getX()
);
}
void PowerGrid::merge ( const Transformation& transformation, Vertical* vertical )
{
PowerPlane* plane = getPlane( vertical->getLayer() );
Point source = vertical->getSourcePosition();
Point target = vertical->getTargetPosition();
transformation.applyOn( source );
transformation.applyOn( target );
if (source.getY() > target.getY()) swap( source, target );
plane->merge( vertical->getNet()->getType()
, Constant::Vertical
, source.getX()
, vertical->getWidth()
, source.getY()
, target.getY()
);
}
void PowerGrid::doLayout () const
{
map<const Layer*,PowerPlane*>::const_iterator iplane = _planes.begin ();
for ( ; iplane != _planes.end() ; iplane++ )
iplane->second->doLayout( _cell, _powerNet, _groundNet );
}
void copyUpPowerRails ( const Transformation& pathTransf
, Cell* instanceCell
, PowerGrid& powerGrid
)
{
forEach ( Net*, inet, instanceCell->getNets() ) {
switch ( inet->getType() ) {
case Net::Type::POWER:
case Net::Type::GROUND:
break;
default:
continue;
}
forEach ( Component*, icomponent, inet->getComponents() ) {
if (not NetExternalComponents::isExternal(*icomponent)) continue;
Horizontal* horizontal = dynamic_cast<Horizontal*>(*icomponent);
if (horizontal)
powerGrid.merge( pathTransf, horizontal );
else {
Vertical* vertical = dynamic_cast<Vertical*>(*icomponent);
if (vertical)
powerGrid.merge( pathTransf, vertical );
}
}
}
forEach ( Instance*, iinstance, instanceCell->getInstances() ) {
Transformation instanceTransf = iinstance->getTransformation();
pathTransf.applyOn( instanceTransf );
copyUpPowerRails( instanceTransf, iinstance->getMasterCell(), powerGrid );
}
}
} // Anonymous namespace.
namespace Katabatic {
using Hurricane::Point;
using Hurricane::Horizontal;
using Hurricane::Net;
using Hurricane::Cell;
using Hurricane::Instance;
void KatabaticEngine::makePowerRails ()
{
PowerGrid powerGrid (getCell());
Transformation topTransformation; // ID.
copyUpPowerRails( topTransformation, getCell(), powerGrid );
powerGrid.doLayout();
}
} // Katabatic namespace.