Anabatic transient commit 4.

* New: In Anabatic:
    - A Dijkstra skeleton, not debugged yet.
This commit is contained in:
Jean-Paul Chaput 2016-05-30 11:30:29 +02:00
parent 6ef5819310
commit 1f4d6b7e83
15 changed files with 849 additions and 80 deletions

View File

@ -64,8 +64,10 @@ namespace Anabatic {
, _gcells ()
, _viewer (NULL)
, _flags (Flags::NoFlags)
, _stamp (-1)
{
_matrix.setCell( cell, _configuration->getSliceHeight()*2 );
_matrix.setCell( cell, _configuration->getSliceHeight() );
Edge::unity = _configuration->getSliceHeight();
}

View File

@ -15,10 +15,11 @@ endif ( CHECK_DETERMINISM )
set( includes anabatic/Constants.h
anabatic/Configuration.h
anabatic/Matrix.h
anabatic/Edge.h
anabatic/Edge.h anabatic/Edges.h
anabatic/GCell.h #anabatic/GCells.h
anabatic/AnabaticEngine.h
anabatic/GraphicAnabaticEngine.h
anabatic/Dijkstra.h
)
set( mocIncludes anabatic/GraphicAnabaticEngine.h )
set( pyIncludes anabatic/PyAnabaticEngine.h
@ -28,9 +29,11 @@ endif ( CHECK_DETERMINISM )
Configuration.cpp
Matrix.cpp
Edge.cpp
Edges.cpp
GCell.cpp
AnabaticEngine.cpp
GraphicAnabaticEngine.cpp
Dijkstra.cpp
)
set( pyCpps PyAnabaticEngine.cpp
PyGraphicAnabaticEngine.cpp

View File

@ -118,7 +118,7 @@ namespace Anabatic {
, _gcontact (other._gcontact)
, _cg (NULL)
, _rg (NULL)
, _extensionCaps (other._extensionCaps)
, _extensionCaps(other._extensionCaps)
, _allowedDepth (other._allowedDepth)
{
if (other._cg) _cg = other._cg->getClone();
@ -145,6 +145,14 @@ namespace Anabatic {
bool ConfigurationConcrete::isGContact ( const Layer* layer ) const
{ return (layer and (layer == _gcontact)); }
const Layer* ConfigurationConcrete::getGContactLayer () const
{ return _gcontact; }
const Layer* ConfigurationConcrete::getGHorizontalLayer () const
{ return _gmetalh; }
const Layer* ConfigurationConcrete::getGVerticalLayer () const
{ return _gmetalv; }
size_t ConfigurationConcrete::getDepth () const
{ return _rg->getDepth(); }

238
anabatic/src/Dijkstra.cpp Normal file
View File

@ -0,0 +1,238 @@
// -*- mode: C++; explicit-buffer-name: "Dijkstra.cpp<anabatic>" -*-
//
// This file is part of the Coriolis Software.
// Copyright (c) UPMC 2016-2016, All Rights Reserved
//
// +-----------------------------------------------------------------+
// | C O R I O L I S |
// | A n a b a t i c - Global Routing Toolbox |
// | |
// | Author : Jean-Paul CHAPUT |
// | E-mail : Jean-Paul.Chaput@lip6.fr |
// | =============================================================== |
// | C++ Module : "./anabatic/Dijkstra.cpp" |
// +-----------------------------------------------------------------+
#include <limits>
#include "hurricane/Error.h"
#include "hurricane/Net.h"
#include "hurricane/RoutingPad.h"
#include "hurricane/Horizontal.h"
#include "hurricane/Vertical.h"
#include "hurricane/UpdateSession.h"
#include "anabatic/AnabaticEngine.h"
#include "anabatic/Dijkstra.h"
namespace Anabatic {
using std::cerr;
using std::endl;
using std::numeric_limits;
using Hurricane::Error;
using Hurricane::Component;
using Hurricane::Horizontal;
using Hurricane::Vertical;
using Hurricane::RoutingPad;
using Hurricane::UpdateSession;
// -------------------------------------------------------------------
// Class : "Anabatic::Vertex".
float Vertex::unreached = numeric_limits<float>::max();
bool Vertex::hasValidStamp () const
{ return _stamp == getAnabatic()->getStamp(); }
// -------------------------------------------------------------------
// Class : "Anabatic::Dijkstra".
Dijkstra::Dijkstra ( AnabaticEngine* anabatic )
: _anabatic (anabatic)
, _vertexes ()
, _net (NULL)
, _stamp (-1)
, _sources ()
, _targets ()
, _searchArea ()
, _connectedsId(-1)
, _queue ()
{
const vector<GCell*>& gcells = _anabatic->getGCells();
for ( GCell* gcell : gcells ) {
_vertexes.push_back( new Vertex (gcell) );
}
}
Dijkstra::~Dijkstra ()
{
for ( Vertex* vertex : _vertexes ) delete vertex;
}
void Dijkstra::load ( Net* net )
{
UpdateSession::open();
_searchArea.makeEmpty();
_stamp = _anabatic->incStamp();
for ( Component* component : net->getRoutingPads() ) {
RoutingPad* rp = dynamic_cast<RoutingPad*>( component );
if (rp) {
Box rpBb = rp->getBoundingBox();
Point center = rpBb.getCenter();
GCell* gcell = _anabatic->getGCellUnder( center );
if (not gcell) {
cerr << Error( "Dijkstra::load(): %s of %s is not under any GCell.\n"
" It will be ignored ans the routing will be incomplete."
, getString(rp ).c_str()
, getString(net).c_str()
) << endl;
continue;
}
_searchArea.merge( rpBb );
Vertex* vertex = gcell->lookup<Vertex>();
if (vertex->getConnexId() < 0) {
vertex->setStamp ( _stamp );
vertex->setConnexId( _targets.size() );
vertex->getGContact( _net );
_targets.insert( vertex );
}
}
}
UpdateSession::close();
}
void Dijkstra::selectFirstSource ()
{
if (_targets.empty()) {
cerr << Error( "Dijkstra::selectFirstSource(): %s has no vertexes to route, ignored.\n"
, getString(_net).c_str()
) << endl;
return;
}
Point areaCenter = _searchArea.getCenter();
auto ivertex = _targets.begin();
Vertex* firstSource = *ivertex++;
DbU::Unit minDistance = areaCenter.manhattanDistance( firstSource->getCenter() );
for ( ; ivertex != _targets.end() ; ++ivertex ) {
DbU::Unit distance = areaCenter.manhattanDistance( (*ivertex)->getCenter() );
if (distance < minDistance) {
minDistance = distance;
firstSource = *ivertex;
}
}
_targets.erase( firstSource );
}
void Dijkstra::propagate ()
{
while ( not _queue.empty() ) {
Vertex* current = _queue.top();
_queue.pop();
if ((current->getConnexId() == _connectedsId) or (current->getConnexId() < 0)) {
for ( Edge* edge : current->getGCell()->getEdges() ) {
if (edge == current->getFrom()) continue;
GCell* gneighbor = edge->getOpposite(current->getGCell());
Vertex* vneighbor = gneighbor->lookup<Vertex>();
if (vneighbor->getConnexId() == _connectedsId) continue;
float distance = current->getDistance() + edge->getDistance();
if (distance < current->getDistance()) {
if (vneighbor->getDistance() != Vertex::unreached) _queue.erase( vneighbor );
else vneighbor->setStamp( _stamp );
vneighbor->setDistance( distance );
_queue.push( vneighbor );
}
}
}
// We did reach another target (different <connexId>).
// Tag back the path.
_targets.erase( current );
while ( current ) {
_sources.insert( current );
current->setDistance( 0.0 );
current->setConnexId( _connectedsId );
current = current->getPredecessor();
}
break;
}
}
void Dijkstra::run ()
{
if (_sources.empty()) return;
UpdateSession::open();
_queue.clear();
_queue.push( *_sources.begin() );
_connectedsId = (*_sources.begin())->getConnexId();
while ( not _targets.empty() ) propagate();
toWires();
UpdateSession::close();
}
void Dijkstra::toWires ()
{
for ( Vertex* vertex : _sources ) {
Edge* from = vertex->getFrom();
if (not from) continue;
Vertex* source = vertex;
Vertex* target = source->getPredecessor();
if ( (source->getGCell()->getXMin() > target->getGCell()->getXMin())
or (source->getGCell()->getYMin() > target->getGCell()->getYMin()) )
std::swap( source, target );
Contact* sourceContact = source->getGContact( _net );
Contact* targetContact = source->getGContact( _net );
if (from->isHorizontal()) {
Horizontal::create( sourceContact
, targetContact
, _anabatic->getConfiguration()->getGHorizontalLayer()
, from->getAxis()
, DbU::fromLambda(2.0)
);
} else {
Vertical::create( sourceContact
, targetContact
, _anabatic->getConfiguration()->getGVerticalLayer()
, from->getAxis()
, DbU::fromLambda(2.0)
);
}
}
}
} // Anabatic namespace.

View File

@ -29,6 +29,7 @@ namespace Anabatic {
Name Edge::_extensionName = "Anabatic::Edge";
DbU::Unit Edge::unity = 1;
Edge::Edge ( GCell* source, GCell* target, Flags flags )
@ -155,6 +156,16 @@ namespace Anabatic {
}
float Edge::getDistance () const
{
Point sourceCenter = getSource()->getBoundingBox().getCenter();
Point targetCenter = getTarget()->getBoundingBox().getCenter();
return ( (targetCenter.getX() - sourceCenter.getX())
+ (targetCenter.getY() - sourceCenter.getY()) ) / unity;
}
void Edge::_setSource ( GCell* source )
{
if (source == _target)

117
anabatic/src/Edges.cpp Normal file
View File

@ -0,0 +1,117 @@
// -*- mode: C++; explicit-buffer-name: "Edges.cpp<anabatic>" -*-
//
// This file is part of the Coriolis Software.
// Copyright (c) UPMC 2016-2016, All Rights Reserved
//
// +-----------------------------------------------------------------+
// | C O R I O L I S |
// | A n a b a t i c - Global Routing Toolbox |
// | |
// | Author : Jean-Paul CHAPUT |
// | E-mail : Jean-Paul.Chaput@lip6.fr |
// | =============================================================== |
// | C++ Module : "./anabatic/Edges.cpp" |
// +-----------------------------------------------------------------+
#include "anabatic/Edges.h"
#include "anabatic/GCell.h"
namespace Anabatic {
using std::endl;
// -------------------------------------------------------------------
// Class : "Anabatic::GCell_Edges".
GCell_Edges::Locator::Locator ( const GCell* gcell )
: EdgesHL()
, _gcell(gcell)
, _flags(Flags::EastSide)
, _iedge(0)
{
if (_gcell->getEastEdges().empty()) progress();
}
EdgesHL* GCell_Edges::Locator::getClone () const
{ return new Locator (*this); }
Edge* GCell_Edges::Locator::getElement () const
{
if (_flags & Flags::EastSide ) return _gcell->getEastEdges ()[_iedge];
if (_flags & Flags::NorthSide) return _gcell->getNorthEdges()[_iedge];
if (_flags & Flags::WestSide ) return _gcell->getWestEdges ()[_iedge];
if (_flags & Flags::SouthSide) return _gcell->getSouthEdges()[_iedge];
return NULL;
}
bool GCell_Edges::Locator::isValid () const
{ return not _flags; }
void GCell_Edges::Locator::progress ()
{
cdebug.log(110) << "GCell_Edges::Locator::progress()" << endl;
++_iedge;
while (_flags) {
if (_flags & Flags::EastSide) {
if (_iedge < _gcell->getEastEdges().size()) break;
_flags = Flags::NorthSide;
_iedge = 0;
}
if (_flags & Flags::NorthSide) {
if (_iedge < _gcell->getNorthEdges().size()) break;
_flags = Flags::WestSide;
_iedge = 0;
}
if (_flags & Flags::WestSide) {
if (_iedge < _gcell->getWestEdges().size()) break;
_flags = Flags::SouthSide;
_iedge = 0;
}
if (_flags & Flags::SouthSide) {
if (_iedge < _gcell->getSouthEdges().size()) break;
_flags = 0;
_iedge = 0;
}
}
}
string GCell_Edges::Locator::_getString () const
{
string s = "<GCell_Edges::Locator";
if (_flags & Flags::EastSide ) s += "East[" + getString(_iedge) + "]";
if (_flags & Flags::NorthSide) s += "North[" + getString(_iedge) + "]";
if (_flags & Flags::WestSide ) s += "West[" + getString(_iedge) + "]";
if (_flags & Flags::SouthSide) s += "South[" + getString(_iedge) + "]";
if (_flags == 0) s += "invalid";
s += ">";
return s;
}
EdgesHC* GCell_Edges::getClone () const
{ return new GCell_Edges (*this); }
EdgesHL* GCell_Edges::getLocator () const
{ return new Locator (_gcell); }
string GCell_Edges::_getString () const
{
string s = "<GCell_Edges "
+ getString(_gcell)
+ ">";
return s;
}
} // Anabatic namespace.

View File

@ -15,6 +15,7 @@
#include <iostream>
#include "hurricane/Contact.h"
#include "anabatic/GCell.h"
#include "anabatic/AnabaticEngine.h"
@ -39,6 +40,8 @@ namespace Anabatic {
, _northEdges()
, _xmin (xmin)
, _ymin (ymin)
, _contacts ()
, _lookup (NULL)
{ }
@ -506,6 +509,23 @@ namespace Anabatic {
}
Contact* GCell::getGContact ( Net* net )
{
for ( Contact* contact : _contacts ) {
if (contact->getNet() == net) return contact;
}
Point center = getBoundingBox().getCenter();
return Contact::create( net
, _anabatic->getConfiguration()->getGContactLayer()
, center.getX()
, center.getY()
, DbU::fromLambda(2.0)
, DbU::fromLambda(2.0)
);
}
const Name& GCell::getName () const
{ return _extensionName; }

View File

@ -35,6 +35,7 @@
#include <crlcore/AllianceFramework.h>
#include "anabatic/GCell.h"
#include "anabatic/GraphicAnabaticEngine.h"
#include "anabatic/Dijkstra.h"
namespace Anabatic {
@ -123,6 +124,24 @@ namespace Anabatic {
}
void anabaticTest_4 ( AnabaticEngine* engine )
{
Cell* cell = engine->getCell();
cell->flattenNets( Cell::Flags::BuildRings );
cell->createRoutingPadRings( Cell::Flags::BuildRings );
engine->getSouthWestGCell()->doGrid();
Net* net = cell->getNet( "ialu.inv_x2_3_sig" );
if (net) {
Dijkstra* dijkstra = new Dijkstra ( engine );
dijkstra->load( net );
dijkstra->run();
delete dijkstra;
}
}
// -------------------------------------------------------------------
// Class : "Anabatic::GraphicAnabaticEngine".
@ -254,7 +273,8 @@ namespace Anabatic {
//anabaticTest_1( engine );
//anabaticTest_2( engine );
anabaticTest_3( engine );
//anabaticTest_3( engine );
anabaticTest_4( engine );
if (_viewer) _viewer->emitCellChanged();
}

View File

@ -56,10 +56,15 @@ namespace Anabatic {
virtual Configuration* getConfiguration ();
inline CellViewer* getViewer () const;
inline void setViewer ( CellViewer* );
inline const vector<GCell*>& getGCells () const;
inline GCell* getSouthWestGCell () const;
inline GCell* getGCellUnder ( DbU::Unit x, DbU::Unit y ) const;
inline GCell* getGCellUnder ( Point ) const;
int getCapacity ( Interval, Flags ) const;
// Dijkstra related functions.
inline int getStamp () const;
inline int incStamp ();
// Misc. functions.
inline const Flags& flags () const;
inline Flags& flags ();
void reset ();
@ -87,11 +92,13 @@ namespace Anabatic {
vector<GCell*> _gcells;
CellViewer* _viewer;
Flags _flags;
int _stamp;
};
inline CellViewer* AnabaticEngine::getViewer () const { return _viewer; }
inline void AnabaticEngine::setViewer ( CellViewer* viewer ) { _viewer=viewer; }
inline const vector<GCell*>& AnabaticEngine::getGCells () const { return _gcells; }
inline GCell* AnabaticEngine::getSouthWestGCell () const { return _gcells[0]; }
inline GCell* AnabaticEngine::getGCellUnder ( DbU::Unit x, DbU::Unit y ) const { return _matrix.getUnder(x,y); }
inline GCell* AnabaticEngine::getGCellUnder ( Point p ) const { return _matrix.getUnder(p); }
@ -116,6 +123,9 @@ namespace Anabatic {
}
}
inline int AnabaticEngine::getStamp () const { return _stamp; }
inline int AnabaticEngine::incStamp () { return ++_stamp; }
} // Anabatic namespace.

View File

@ -60,6 +60,9 @@ namespace Anabatic {
// Methods.
virtual bool isGMetal ( const Layer* ) const = 0;
virtual bool isGContact ( const Layer* ) const = 0;
virtual const Layer* getGContactLayer () const = 0;
virtual const Layer* getGHorizontalLayer () const = 0;
virtual const Layer* getGVerticalLayer () const = 0;
virtual size_t getDepth () const = 0;
virtual size_t getAllowedDepth () const = 0;
virtual size_t getLayerDepth ( const Layer* ) const = 0;
@ -110,6 +113,9 @@ namespace Anabatic {
// Methods.
virtual bool isGMetal ( const Layer* ) const;
virtual bool isGContact ( const Layer* ) const;
virtual const Layer* getGContactLayer () const;
virtual const Layer* getGHorizontalLayer () const;
virtual const Layer* getGVerticalLayer () const;
virtual size_t getDepth () const;
virtual size_t getAllowedDepth () const;
virtual size_t getLayerDepth ( const Layer* ) const;

View File

@ -0,0 +1,191 @@
// -*- mode: C++; explicit-buffer-name: "Dijkstra.h<anabatic>" -*-
//
// This file is part of the Coriolis Software.
// Copyright (c) UPMC 2016-2016, All Rights Reserved
//
// +-----------------------------------------------------------------+
// | C O R I O L I S |
// | A n a b a t i c - Global Routing Toolbox |
// | |
// | Author : Jean-Paul CHAPUT |
// | E-mail : Jean-Paul.Chaput@lip6.fr |
// | =============================================================== |
// | C++ Header : "./anabatic/Dijkstra.h" |
// +-----------------------------------------------------------------+
#ifndef ANABATIC_DIJKSTRA_H
#define ANABATIC_DIJKSTRA_H
#include <set>
namespace Hurricane {
class Net;
}
#include "anabatic/GCell.h"
namespace Anabatic {
using std::set;
using Hurricane::Net;
class AnabaticEngine;
// -------------------------------------------------------------------
// Class : "Anabatic::Vertex".
class Vertex {
public:
class CompareById {
public:
inline bool operator() ( const Vertex* lhs, const Vertex* rhs );
};
public:
static float unreached;
public:
inline Vertex ( GCell* );
inline Vertex ( size_t id );
inline ~Vertex ();
inline unsigned int getId () const;
inline GCell* getGCell () const;
inline AnabaticEngine* getAnabatic () const;
inline Contact* getGContact ( Net* );
bool hasValidStamp () const;
inline Point getCenter () const;
inline float getDistance () const;
inline int getStamp () const;
inline int getConnexId () const;
inline Edge* getFrom () const;
inline Vertex* getPredecessor () const;
inline void setDistance ( float );
inline void setStamp ( int );
inline void setConnexId ( int );
inline void setFrom ( Edge* );
private:
Vertex ( const Vertex& );
Vertex& operator= ( const Vertex& );
private:
size_t _id;
GCell* _gcell;
int _connexId;
int _stamp;
float _distance;
Edge* _from;
};
inline Vertex::Vertex ( GCell* gcell )
: _id (gcell->getId())
, _gcell (gcell)
, _connexId(-1)
, _stamp (-1)
, _distance(unreached)
, _from (NULL)
{
gcell->setLookup<Vertex>( this );
}
inline Vertex::Vertex ( size_t id )
: _id (id)
, _gcell (NULL)
, _connexId(-1)
, _stamp (-1)
, _distance(unreached)
, _from (NULL)
{ }
inline Vertex::~Vertex () { }
inline unsigned int Vertex::getId () const { return _id; }
inline GCell* Vertex::getGCell () const { return _gcell; }
inline AnabaticEngine* Vertex::getAnabatic () const { return _gcell->getAnabatic(); }
inline Contact* Vertex::getGContact ( Net* net ) { return _gcell->getGContact(net); }
inline Point Vertex::getCenter () const { return _gcell->getBoundingBox().getCenter(); }
inline float Vertex::getDistance () const { return hasValidStamp() ? _distance : unreached; }
inline int Vertex::getStamp () const { return _stamp; }
inline int Vertex::getConnexId () const { return hasValidStamp() ? _connexId : -1; }
inline Edge* Vertex::getFrom () const { return _from; }
inline void Vertex::setDistance ( float distance ) { _distance=distance; }
inline void Vertex::setFrom ( Edge* from ) { _from=from; }
inline void Vertex::setStamp ( int stamp ) { _stamp=stamp; }
inline void Vertex::setConnexId ( int id ) { _connexId=id; }
inline Vertex* Vertex::getPredecessor () const
{ return (hasValidStamp() and _from) ? _from->getOpposite(_gcell)->lookup<Vertex>() : NULL; }
inline bool Vertex::CompareById::operator() ( const Vertex* lhs, const Vertex* rhs )
{ return lhs->getId() < rhs->getId(); }
typedef set<Vertex*,Vertex::CompareById> VertexSet;
// -------------------------------------------------------------------
// Class : "Anabatic::PriorityQueue".
class PriorityQueue {
public:
inline PriorityQueue ();
inline ~PriorityQueue ();
inline bool empty () const;
inline size_t size () const;
inline void push ( Vertex* );
inline void erase ( Vertex* );
inline Vertex* top ();
inline void pop ();
inline void clear ();
private:
class CompareByDistance {
public:
inline bool operator() ( const Vertex* lhs, const Vertex* rhs );
};
private:
set<Vertex*,CompareByDistance> _queue;
};
inline bool PriorityQueue::CompareByDistance::operator() ( const Vertex* lhs, const Vertex* rhs )
{ return lhs->getDistance() < rhs->getDistance(); }
inline PriorityQueue::PriorityQueue () : _queue() { }
inline PriorityQueue::~PriorityQueue () { }
inline bool PriorityQueue::empty () const { return _queue.empty(); }
inline size_t PriorityQueue::size () const { return _queue.size(); }
inline void PriorityQueue::push ( Vertex* v ) { _queue.insert(v); }
inline void PriorityQueue::erase ( Vertex* v ) { _queue.erase(v); }
inline Vertex* PriorityQueue::top () { return _queue.empty() ? NULL : *_queue.begin(); }
inline void PriorityQueue::pop () { _queue.erase(_queue.begin()); }
inline void PriorityQueue::clear () { _queue.clear(); }
// -------------------------------------------------------------------
// Class : "Anabatic::Dijkstra".
class Dijkstra {
public:
Dijkstra ( AnabaticEngine* );
~Dijkstra ();
public:
void load ( Net* );
void run ();
void propagate ();
void selectFirstSource ();
void toWires ();
private:
Dijkstra ( const Dijkstra& );
Dijkstra& operator= ( const Dijkstra& );
private:
AnabaticEngine* _anabatic;
vector<Vertex*> _vertexes;
Net* _net;
int _stamp;
VertexSet _sources;
VertexSet _targets;
Box _searchArea;
int _connectedsId;
PriorityQueue _queue;
};
} // Anabatic namespace.
#endif // ANABATIC_DIJKSTRA_H

View File

@ -23,6 +23,7 @@
#include "hurricane/Box.h"
#include "hurricane/ExtensionGo.h"
#include "anabatic/Constants.h"
#include "anabatic/Edges.h"
namespace Anabatic {
@ -43,6 +44,8 @@ namespace Anabatic {
class Edge : public ExtensionGo {
public:
typedef ExtensionGo Super;
public:
static DbU::Unit unity;
public:
static Edge* create ( GCell* source, GCell* target, Flags flags=Flags::NoFlags );
virtual void destroy ();
@ -52,6 +55,7 @@ namespace Anabatic {
inline unsigned int getCapacity () const;
inline unsigned int getRealOccupancy () const;
inline unsigned int getEstimateOccupancy () const;
float getDistance () const;
inline GCell* getSource () const;
inline GCell* getTarget () const;
GCell* getOpposite ( const GCell* ) const;

View File

@ -0,0 +1,108 @@
// -*- mode: C++; explicit-buffer-name: "Edges.h<anabatic>" -*-
//
// This file is part of the Coriolis Software.
// Copyright (c) UPMC 2016-2016, All Rights Reserved
//
// +-----------------------------------------------------------------+
// | C O R I O L I S |
// | A n a b a t i c - Global Routing Toolbox |
// | |
// | Author : Jean-Paul CHAPUT |
// | E-mail : Jean-Paul.Chaput@lip6.fr |
// | =============================================================== |
// | C++ Header : "./anabatic/Edges.h" |
// +-----------------------------------------------------------------+
#ifndef ANABATIC_EDGES_H
#define ANABATIC_EDGES_H
#include <string>
#include <vector>
#include "hurricane/Collection.h"
#include "anabatic/Constants.h"
namespace Anabatic {
using std::string;
using std::vector;
using Hurricane::Record;
using Hurricane::Filter;
using Hurricane::Locator;
using Hurricane::Collection;
using Hurricane::GenericFilter;
using Hurricane::GenericLocator;
using Hurricane::GenericCollection;
class Edge;
class GCell;
// -------------------------------------------------------------------
// Collections.
typedef Hurricane::Filter<Edge*> EdgesHF;
typedef Hurricane::Locator<Edge*> EdgesHL;
typedef Hurricane::Collection<Edge*> EdgesHC;
typedef GenericCollection<Edge*> Edges;
typedef GenericLocator<Edge*> EdgesLocator;
typedef GenericFilter<Edge*> EdgesFilter;
class GCell;
// -------------------------------------------------------------------
// Class : "GCell_Edges".
class GCell_Edges : public EdgesHC {
public:
// Sub-Class: Locator.
class Locator : public EdgesHL {
public:
Locator ( const GCell* gcell );
inline Locator ( const Locator& );
virtual Edge* getElement () const;
virtual EdgesHL* getClone () const;
virtual bool isValid () const;
virtual void progress ();
virtual string _getString () const;
protected:
const GCell* _gcell;
Flags _flags;
size_t _iedge;
};
// GCell_Edges.
public:
inline GCell_Edges ( const GCell* gcell );
inline GCell_Edges ( const GCell_Edges& );
virtual EdgesHC* getClone () const;
virtual EdgesHL* getLocator () const;
virtual string _getString () const;
protected:
const GCell* _gcell;
};
inline GCell_Edges::Locator::Locator ( const Locator &locator )
: EdgesHL()
, _gcell(locator._gcell)
, _flags(locator._flags)
, _iedge(locator._iedge)
{ }
inline GCell_Edges::GCell_Edges ( const GCell* gcell )
: EdgesHC()
, _gcell(gcell)
{ }
inline GCell_Edges::GCell_Edges ( const GCell_Edges& edges )
: EdgesHC()
, _gcell(edges._gcell)
{ }
} // Anabatic namespace.
#endif // ANABATIC_EDGES_H

View File

@ -24,6 +24,9 @@
#include "hurricane/Box.h"
#include "hurricane/Cell.h"
#include "hurricane/ExtensionGo.h"
namespace Hurricane {
class Contact;
}
#include "anabatic/Edge.h"
@ -37,7 +40,9 @@ namespace Anabatic {
using Hurricane::Point;
using Hurricane::Interval;
using Hurricane::Box;
using Hurricane::Net;
using Hurricane::Entity;
using Hurricane::Contact;
using Hurricane::Cell;
class AnabaticEngine;
@ -64,6 +69,11 @@ namespace Anabatic {
inline DbU::Unit getYMax ( int shrink=0 ) const;
inline Interval getSide ( Flags direction ) const;
inline Point getCenter () const;
inline const vector<Edge*>& getWestEdges () const;
inline const vector<Edge*>& getEastEdges () const;
inline const vector<Edge*>& getNorthEdges () const;
inline const vector<Edge*>& getSouthEdges () const;
inline Edges getEdges () const;
inline GCell* getWest () const;
inline GCell* getEast () const;
inline GCell* getSouth () const;
@ -73,9 +83,16 @@ namespace Anabatic {
GCell* getSouth ( DbU::Unit x ) const;
GCell* getNorth ( DbU::Unit x ) const;
GCell* getUnder ( DbU::Unit x, DbU::Unit y ) const;
inline GCell* getUnder ( Point p ) const;
GCell* hcut ( DbU::Unit y );
GCell* vcut ( DbU::Unit x );
bool doGrid ();
Contact* getGContact ( Net* );
template<typename Type>
inline void setLookup ( Type* decorator );
template<typename Type>
inline Type* lookup () const;
// Misc. functions.
inline const Flags& flags () const;
inline Flags& flags ();
void _add ( Edge* edge, Flags side );
@ -114,6 +131,8 @@ namespace Anabatic {
vector<Edge*> _northEdges;
DbU::Unit _xmin;
DbU::Unit _ymin;
vector<Contact*> _contacts;
void* _lookup;
};
@ -123,10 +142,16 @@ namespace Anabatic {
inline AnabaticEngine* GCell::getAnabatic () const { return _anabatic; }
inline DbU::Unit GCell::getXMin () const { return _xmin; }
inline DbU::Unit GCell::getYMin () const { return _ymin; }
inline Edges GCell::getEdges () const { return new GCell_Edges(this); }
inline const vector<Edge*>& GCell::getWestEdges () const { return _westEdges; }
inline const vector<Edge*>& GCell::getEastEdges () const { return _eastEdges; }
inline const vector<Edge*>& GCell::getNorthEdges () const { return _northEdges; }
inline const vector<Edge*>& GCell::getSouthEdges () const { return _southEdges; }
inline GCell* GCell::getWest () const { return _westEdges.empty() ? NULL : _westEdges[0]->getOpposite(this); }
inline GCell* GCell::getEast () const { return _eastEdges.empty() ? NULL : _eastEdges[0]->getOpposite(this); }
inline GCell* GCell::getSouth () const { return _southEdges.empty() ? NULL : _southEdges[0]->getOpposite(this); }
inline GCell* GCell::getNorth () const { return _northEdges.empty() ? NULL : _northEdges[0]->getOpposite(this); }
inline GCell* GCell::getUnder ( Point p ) const { return getUnder(p.getX(),p.getY()); }
inline const Flags& GCell::flags () const { return _flags; }
inline Flags& GCell::flags () { return _flags; }
@ -147,6 +172,12 @@ namespace Anabatic {
return Interval( getXMin(), getXMax() );
}
template<typename Type>
inline void GCell::setLookup ( Type* decorator ) { _lookup=reinterpret_cast<void*>(decorator); }
template<typename Type>
inline Type* GCell::lookup () const { return reinterpret_cast<Type*>( _lookup ); }
// -------------------------------------------------------------------
// Class : "GCellSet".

View File

@ -98,8 +98,8 @@ namespace Hurricane {
inline void Observable::notify ( unsigned int flags )
{
for ( auto iobserver : _observers ) {
iobserver->notify( flags );
for ( BaseObserver* iobserver : _observers ) {
if (iobserver) iobserver->notify( flags );
}
}