1200 lines
30 KiB
C++
1200 lines
30 KiB
C++
|
|
// This file is part of the Coriolis Project.
|
|
// Copyright (C) Laboratoire LIP6 - Departement ASIM
|
|
// Universite Pierre et Marie Curie
|
|
//
|
|
// Date : 29/01/2004
|
|
// Author : Hugo Clément <Hugo.Clement@lip6.fr>
|
|
|
|
|
|
#include <math.h>
|
|
|
|
#include "hurricane/DataBase.h"
|
|
#include "hurricane/Technology.h"
|
|
#include "hurricane/UpdateSession.h"
|
|
#include "crlcore/Utilities.h"
|
|
#include "crlcore/ToolBox.h"
|
|
#include "crlcore/CellGauge.h"
|
|
#include "crlcore/LefDefExtension.h"
|
|
#include "crlcore/AllianceFramework.h"
|
|
#include "nimbus/GCell.h"
|
|
#include "nimbus/Fence.h"
|
|
#include "nimbus/VFence.h"
|
|
#include "nimbus/HFence.h"
|
|
#include "nimbus/StepProperty.h"
|
|
#include "nimbus/Grid.h"
|
|
#include "nimbus/NimbusEngine.h"
|
|
|
|
namespace Nimbus {
|
|
|
|
#define RG_HORIZONTAL 0
|
|
#define RG_VERTICAL 1
|
|
|
|
|
|
|
|
/*
|
|
* ********************************************************************
|
|
* GCell_GCells declaration
|
|
* ********************************************************************
|
|
*/
|
|
class Grid_GCells : public Collection<GCell*> {
|
|
|
|
public: typedef Collection<GCell*> Inherit;
|
|
|
|
public: class Locator : public Hurricane::Locator<GCell*> {
|
|
|
|
public: typedef Hurricane::Locator<GCell*> Inherit;
|
|
|
|
private: GCell* _nb;
|
|
private: unsigned _direction;
|
|
|
|
public: Locator(GCell* gcell = NULL);
|
|
public: Locator(const Locator& locator);
|
|
|
|
public: Locator& operator=(const Locator& locator);
|
|
|
|
public: virtual GCell* getElement() const;
|
|
public: virtual Hurricane::Locator<GCell*>* getClone() const;
|
|
|
|
public: virtual bool isValid() const;
|
|
|
|
public: virtual void progress();
|
|
|
|
public: virtual string _getString() const;
|
|
};
|
|
|
|
|
|
private: GCell* _nb;
|
|
private: unsigned _step;
|
|
|
|
public: Grid_GCells(const Grid* grid, unsigned step);
|
|
public: Grid_GCells(const Grid_GCells& gcelles);
|
|
|
|
public: Grid_GCells& operator=(const Grid_GCells& gcelles);
|
|
|
|
public: virtual Collection<GCell*>* getClone() const;
|
|
public: virtual Hurricane::Locator<GCell*>* getLocator() const;
|
|
|
|
public: virtual string _getString() const;
|
|
};
|
|
|
|
/*
|
|
* ********************************************************************
|
|
* GCell_GCells declaration
|
|
* ********************************************************************
|
|
*/
|
|
class Grid_FastGCells : public Collection<GCell*> {
|
|
|
|
public: typedef Collection<GCell*> Inherit;
|
|
|
|
public: class Locator : public Hurricane::Locator<GCell*> {
|
|
|
|
public: typedef Hurricane::Locator<GCell*> Inherit;
|
|
|
|
private: GCell* _nb;
|
|
private: unsigned _direction;
|
|
|
|
public: Locator(GCell* gcell = NULL);
|
|
public: Locator(const Locator& locator);
|
|
|
|
public: Locator& operator=(const Locator& locator);
|
|
|
|
public: virtual GCell* getElement() const;
|
|
public: virtual Hurricane::Locator<GCell*>* getClone() const;
|
|
|
|
public: virtual bool isValid() const;
|
|
|
|
public: virtual void progress();
|
|
|
|
public: virtual string _getString() const;
|
|
};
|
|
|
|
|
|
private: GCell* _nb;
|
|
private: unsigned _step;
|
|
|
|
public: Grid_FastGCells(const Grid* grid, unsigned step);
|
|
public: Grid_FastGCells(const Grid_FastGCells& gcelles);
|
|
|
|
public: Grid_FastGCells& operator=(const Grid_FastGCells& gcelles);
|
|
|
|
public: virtual Collection<GCell*>* getClone() const;
|
|
public: virtual Hurricane::Locator<GCell*>* getLocator() const;
|
|
|
|
public: virtual string _getString() const;
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
* ********************************************************************
|
|
* Grid
|
|
*
|
|
* ********************************************************************
|
|
*/
|
|
Grid::Grid (NimbusEngine* nimbus)
|
|
//*******************************
|
|
: _nimbus (nimbus)
|
|
, _cell (nimbus->getCell())
|
|
, _rootGCell(NULL)
|
|
, _layer (NULL)
|
|
, _fences ()
|
|
{
|
|
_layer = DataBase::getDB()->getTechnology()->getLayer (Name ("SPL1"));
|
|
if ( _layer == NULL )
|
|
throw Error("NimbusEngine needs <SPL1> layer to be defined");
|
|
|
|
Box defBox = LefDefExtension::getRowsBox(_cell);
|
|
|
|
if ( not _nimbus->getWorkZone().isEmpty() ) {
|
|
_rootGCell = GCell::create(_nimbus, 0, _nimbus->getWorkZone());
|
|
} else if ( not defBox.isEmpty() ) {
|
|
UpdateSession::open();
|
|
|
|
if ( _cell->getAbutmentBox().isEmpty() )
|
|
_cell->setAbutmentBox ( defBox );
|
|
|
|
cmess1 << " - DEF abutment box found: " << defBox << endl;
|
|
_rootGCell = GCell::create(_nimbus, 0, defBox);
|
|
|
|
UpdateSession::close();
|
|
} else if ( _cell->getAbutmentBox().isEmpty() ) {
|
|
if ( isNoInstancePlacedOrFixed(_cell) ) {
|
|
DbU::Unit minWidth = DbU::Max;
|
|
DbU::Unit sumWidth = 0;
|
|
forEach ( Instance*, iinstance, _cell->getInstances() ) {
|
|
Cell* model = iinstance->getMasterCell();
|
|
DbU::Unit width = model->getAbutmentBox().getWidth();
|
|
|
|
if ( width < minWidth ) minWidth = width;
|
|
sumWidth += width;
|
|
}
|
|
//rectangularShape(_margin, sumWidth, minWidth, nrows);
|
|
rectangularShape(_nimbus->getMargin(), nimbus->getAspectRatio(), sumWidth, minWidth );
|
|
|
|
UpdateSession::open();
|
|
|
|
_cell->setAbutmentBox( Box ( _rootGCell->getXMin(), _rootGCell->getYMin()
|
|
, _rootGCell->getXMax(), _rootGCell->getYMax()) );
|
|
#if 0
|
|
Point center = box.getCenter();
|
|
forEach ( Instance*, iinstance, cell->getInstances() ) {
|
|
iinstance->setPlacementStatus ( Instance::PlacementStatus::PLACED );
|
|
setStep(**iinstance, 0);
|
|
|
|
Box insABox = iinstance->getAbutmentBox();
|
|
Box masterABox = iinstance->getMasterCell()->getAbutmentBox();
|
|
DbU::Unit xPos = center.getX();
|
|
DbU::Unit yPos = center.getY();
|
|
|
|
instance->setTransformation
|
|
( getTransformation ( masterABox
|
|
, xPos - insABox.getHalfWidth()
|
|
, yPos - insABox.getHalfHeight()
|
|
, Transformation::Orientation::ID
|
|
) );
|
|
}
|
|
#endif
|
|
UpdateSession::close();
|
|
} else {
|
|
Box box = _cell->getBoundingBox();
|
|
|
|
UpdateSession::open();
|
|
_cell->setAbutmentBox ( box );
|
|
_rootGCell = GCell::create(_nimbus, 0, box);
|
|
UpdateSession::close();
|
|
}
|
|
} else {
|
|
cmess1 << " - Design abutment box found: " << _cell->getAbutmentBox() << endl;
|
|
_rootGCell = GCell::create(_nimbus, 0, _cell->getAbutmentBox());
|
|
}
|
|
_rootGCell->setAsPlacementLeaf();
|
|
_rootGCell->setAsRoutingLeaf();
|
|
}
|
|
|
|
Grid::~Grid ()
|
|
//************
|
|
{
|
|
_rootGCell->destroy();
|
|
}
|
|
|
|
#if 0
|
|
void Grid::Clear()
|
|
// **************
|
|
{
|
|
_oldBoxes.clear();
|
|
_newBoxes.clear();
|
|
|
|
return;
|
|
}
|
|
#endif
|
|
|
|
void Grid::rectangularShape(double margin, double aspectRatio, DbU::Unit sumWidth, DbU::Unit minWidth)
|
|
// ***************************************************************************************************
|
|
{
|
|
cmess1 << " o Creating abutment box: (margin: " << margin
|
|
<< ", aspect ratio:" << aspectRatio << ")" << endl;
|
|
|
|
CellGauge* cg = AllianceFramework::get()->getCellGauge();
|
|
DbU::Unit rowHeight = cg->getSliceHeight();
|
|
DbU::Unit pitch = cg->getPitch();
|
|
DbU::Unit marginWidth = (DbU::Unit)( (1.0+margin) * (double)sumWidth );
|
|
|
|
// AR = x/y S = x*y = marginWidth*SH x=S/y AR = S/y^2
|
|
// y = sqrt(S/AR)
|
|
|
|
DbU::Unit y = (DbU::Unit)sqrt ( ((double)marginWidth*(double)rowHeight) / aspectRatio );
|
|
unsigned int rows = y / rowHeight;
|
|
|
|
if ( (rows == 0) or (rows % rowHeight) ) ++rows;
|
|
DbU::Unit rowWidth = marginWidth / rows;
|
|
|
|
if ( rowWidth < minWidth ) {
|
|
rowWidth = minWidth;
|
|
rows = marginWidth / rowWidth;
|
|
|
|
if ( rows == 0 ) ++rows;
|
|
}
|
|
|
|
DbU::Unit adjustWidth = rowWidth % pitch;
|
|
if ( adjustWidth != 0 )
|
|
rowWidth += pitch - adjustWidth;
|
|
|
|
Box ab = Box ( 0, 0, rowWidth, rows * rowHeight );
|
|
|
|
#if DEPRECATED
|
|
if ( nrows == 0 ) {
|
|
nrows = (int)(sqrt((1.0 + margin) * sumwidth / rowHeight) + 0.5);
|
|
}
|
|
|
|
DbU::Unit rowWidth
|
|
= DbU::lambda
|
|
( (int)((double)((int)((1.0 + margin) * DbU::getLambda(sumwidth)
|
|
/ ((double)nrows * DbU::getLambda(pitch)) ))
|
|
* DbU::getLambda(pitch) + 0.5)
|
|
);
|
|
|
|
Box ab = Box(0, 0, rowWidth, nrows * rowHeight);
|
|
#endif
|
|
|
|
cmess1 << " - Abutment box: " << ab << endl;
|
|
|
|
_rootGCell = GCell::create(_nimbus, (unsigned) 0, ab);
|
|
}
|
|
|
|
void Grid::horizontalLongSplit (unsigned step, DbU::Unit& Y) const
|
|
//***************************************************************
|
|
{
|
|
//Clear();
|
|
|
|
recLS (step, Y, RG_HORIZONTAL, _rootGCell);
|
|
return;
|
|
}
|
|
|
|
void Grid::verticalLongSplit (unsigned step, DbU::Unit& X) const
|
|
//*************************************************************
|
|
{
|
|
//Clear();
|
|
|
|
recLS (step, X, RG_VERTICAL, _rootGCell);
|
|
return;
|
|
}
|
|
|
|
|
|
void Grid::recLS (unsigned step, DbU::Unit& coord, unsigned direction, GCell* gcell) const
|
|
//*************************************************************************************
|
|
{
|
|
//cout << " recLS : " << step << "-" << gcell->getStep() << endl;
|
|
|
|
if (gcell->getStep() >= step) {
|
|
|
|
return;
|
|
}
|
|
|
|
if (gcell->getStep() == step - 1) {
|
|
|
|
if (!(gcell->hasSubGCells())) {
|
|
gcell->makeSub();
|
|
}
|
|
|
|
if (direction == RG_HORIZONTAL) {
|
|
//cout << "subHSplit" << endl;
|
|
gcell->subHSplit (coord);
|
|
//cout << "subHSplitted" << endl;
|
|
} else if (direction == RG_VERTICAL) {
|
|
//cout << "subVSplit" << endl;
|
|
gcell->subVSplit (coord);
|
|
//cout << "subVSplitted" << endl;
|
|
} else {
|
|
cerr << "Nothing to do here" << endl;
|
|
abort();
|
|
}
|
|
|
|
return;
|
|
|
|
} else { // exploration récursive
|
|
|
|
if (!(gcell->hasSubGCells())) {
|
|
gcell->makeSub();
|
|
}
|
|
|
|
for_each_gcell (nb, gcell->getSubGCells()) {
|
|
if (direction == RG_HORIZONTAL) {
|
|
if (
|
|
nb->horizontalIsCrossed (coord)
|
|
)
|
|
{
|
|
recLS (step, coord, direction, nb);
|
|
}
|
|
} else if (direction == RG_VERTICAL) {
|
|
if (
|
|
nb->verticalIsCrossed (coord)
|
|
)
|
|
{
|
|
recLS (step, coord, direction, nb);
|
|
}
|
|
} else {
|
|
cerr << "Nothing to do here" << endl;
|
|
abort();
|
|
}
|
|
|
|
end_for;
|
|
}
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
Layer* Grid::getLayer () const {
|
|
|
|
return _layer;
|
|
}
|
|
|
|
GCell* Grid::recGetGCell (unsigned step, Point& p, GCell* gcell) const {
|
|
GCell* res = NULL;
|
|
|
|
if (! (gcell->contains(p))) {
|
|
return NULL;
|
|
}
|
|
|
|
if (gcell->getStep() == step) {
|
|
return gcell;
|
|
|
|
} else {
|
|
for_each_gcell (subBox, gcell->getSubGCells()) {
|
|
|
|
res = recGetGCell (step, p, subBox);
|
|
|
|
if (res)
|
|
break;
|
|
//return res;
|
|
end_for;
|
|
}
|
|
}
|
|
|
|
assert (res);
|
|
|
|
return res;
|
|
} // recGetGCell
|
|
|
|
GCell* Grid::getGCell (unsigned step, Point& p) const
|
|
{
|
|
|
|
return recGetGCell (step, p, _rootGCell);
|
|
}
|
|
|
|
const Box Grid::getBox (unsigned step, Point& p) const
|
|
{
|
|
|
|
GCell* gcell = getGCell (step, p);
|
|
return Box (gcell->getXMin(), gcell->getYMin(), gcell->getXMax(), gcell->getYMax());
|
|
}
|
|
|
|
Point Grid::getPosition (unsigned step, Point& pos) const
|
|
{
|
|
|
|
return (getGCell(step,pos)->getCenter());
|
|
}
|
|
|
|
DbU::Unit Grid::getX (unsigned step, DbU::Unit& X) const
|
|
{
|
|
Point p = Point (X, DbU::lambda (0));
|
|
return getGCell (step, p)->getXCenter();
|
|
}
|
|
|
|
DbU::Unit Grid::getY (unsigned step, DbU::Unit& Y) const
|
|
{
|
|
Point p = Point (DbU::lambda (0), Y);
|
|
return getGCell (step, p)->getYCenter();
|
|
}
|
|
|
|
|
|
void Grid::recDumpGrid (GCell* gcell) const{
|
|
|
|
cout << " Step : " << gcell->getStep() << endl;
|
|
gcell->dumpPavement();
|
|
|
|
for_each_gcell (subBox, gcell->getSubGCells()) {
|
|
recDumpGrid(subBox);
|
|
end_for;
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
void Grid::dumpGrid () const{
|
|
|
|
cout << " o dumping grid..." << endl;
|
|
recDumpGrid (_rootGCell);
|
|
cout << " - grid dumped." << endl;
|
|
|
|
return;
|
|
}
|
|
|
|
/*
|
|
* ********************************************************************
|
|
* Collection des boites du niveau step
|
|
*/
|
|
GCells Grid::getGCells (unsigned step) const
|
|
{
|
|
return Grid_GCells(this, step);
|
|
}
|
|
GCells Grid::getFastGCells (unsigned step) const
|
|
{
|
|
return Grid_FastGCells(this, step);
|
|
}
|
|
|
|
/*
|
|
* ********************************************************************
|
|
*
|
|
* Création des Frontières
|
|
*/
|
|
void Grid::createFences (unsigned step)
|
|
{
|
|
_fences[step] = new FenceSet();
|
|
|
|
for_each_gcell (gcell, getGCells(step)) {
|
|
|
|
//cmess1 << gcell << endl;
|
|
|
|
if (!gcell->getLeftFence()) {
|
|
GCell* nbleft = gcell->computeLeftOfMe();
|
|
if (nbleft) {
|
|
Fence* fence = static_cast<Fence*> (VFence::create (this, nbleft, gcell));
|
|
gcell->getContainer()->addInnerFence(fence);
|
|
fence->setStep(step);
|
|
_fences[step]->_insert(fence);
|
|
}
|
|
|
|
}
|
|
if (!gcell->getRightFence()) {
|
|
GCell* nbright = gcell->computeRightOfMe();
|
|
if (nbright) {
|
|
Fence* fence = static_cast<Fence*> (VFence::create (this, gcell, nbright));
|
|
gcell->getContainer()->addInnerFence(fence);
|
|
fence->setStep(step);
|
|
_fences[step]->_insert(fence);
|
|
}
|
|
}
|
|
if (!gcell->getUpFence()) {
|
|
GCell* nbup = gcell->computeUpOfMe();
|
|
if (nbup) {
|
|
Fence* fence = static_cast<Fence*> (HFence::create (this, nbup, gcell));
|
|
gcell->getContainer()->addInnerFence(fence);
|
|
fence->setStep(step);
|
|
_fences[step]->_insert(fence);
|
|
}
|
|
}
|
|
if (!gcell->getDownFence()) {
|
|
GCell* nbdown = gcell->computeDownOfMe();
|
|
if (nbdown) {
|
|
Fence* fence = static_cast<Fence*> (HFence::create (this, gcell, nbdown));
|
|
gcell->getContainer()->addInnerFence(fence);
|
|
fence->setStep(step);
|
|
_fences[step]->_insert(fence);
|
|
}
|
|
}
|
|
end_for;
|
|
}
|
|
return;
|
|
}
|
|
|
|
/*
|
|
* ********************************************************************
|
|
*
|
|
* Collection des frontieres pour un niveau
|
|
*/
|
|
Fences Grid::getFences (unsigned step)
|
|
{
|
|
return _fences[step]->getElements();
|
|
}
|
|
|
|
Fence* Grid::getFenceBetween (GCell* nb1, GCell* nb2) const
|
|
{
|
|
Fence* fence = testFenceBetween (nb1, nb2);
|
|
if (!fence)
|
|
{
|
|
cerr << "no fence between " << nb1 << " and " << nb2 << "." << endl;
|
|
abort();
|
|
throw Error ("getFenceBetween : gcelles are not linked together");
|
|
}
|
|
|
|
return fence;
|
|
}
|
|
|
|
Fence* Grid::testFenceBetween (GCell* nb1, GCell* nb2) const
|
|
{
|
|
if (nb1 == nb2)
|
|
{
|
|
return NULL;
|
|
#if 0
|
|
abort();
|
|
throw Error ("getFenceBetween : same gcelles");
|
|
#endif
|
|
}
|
|
if (nb1->getStep() != nb2->getStep())
|
|
{
|
|
return NULL;
|
|
#if 0
|
|
abort();
|
|
throw Error ("getFenceBetween : gcelles from different grid levels");
|
|
#endif
|
|
}
|
|
for_each_fence (fence, nb1->getSurroundingFences())
|
|
{
|
|
if (fence->getOppositeGCell (nb1) == nb2)
|
|
return fence;
|
|
|
|
end_for;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
|
|
// Compat
|
|
|
|
Grid* Grid::create (unsigned n, DbU::Unit& width, DbU::Unit& height) {
|
|
|
|
throw Error ("Nimbus: Grid: entering an outdated and deactivated function");
|
|
return NULL;
|
|
}
|
|
|
|
|
|
Box Grid::getGRBox (unsigned step, Point& p) {
|
|
|
|
return (getBox (step, p));
|
|
}
|
|
|
|
Contact* Grid::getContact (Net& net, Point& position) {
|
|
|
|
for_each_contact (contact, net.getContacts()) {
|
|
|
|
if (contact->getPosition() == position) {
|
|
return contact;
|
|
} else {
|
|
// vide
|
|
}
|
|
|
|
end_for;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
Contact* Grid::getOrCreateContact (Net& net, Point& position, DbU::Unit width, DbU::Unit height) {
|
|
|
|
Contact* c = NULL;
|
|
|
|
c = getContact (net, position);
|
|
|
|
if (!c)
|
|
c = Contact::create (&net, getLayer(), position.getX(), position.getY(), width, height);
|
|
|
|
return c;
|
|
}
|
|
#if 0
|
|
Contact* Grid::getOrCreateContact (Net& net, Point& position) {
|
|
|
|
Contact* c = NULL;
|
|
|
|
c = getContact (net, position);
|
|
|
|
if (!c)
|
|
c = Contact::create (&net, getLayer(), position.getX(), position.getY(), DbU::lambda(5), DbU::lambda(5));
|
|
|
|
return c;
|
|
}
|
|
#endif
|
|
|
|
DbU::Unit Grid::getVSplitterHalfLength (unsigned step) {
|
|
|
|
return DbU::lambda (2);
|
|
}
|
|
|
|
DbU::Unit Grid::getHSplitterHalfLength (unsigned step) {
|
|
|
|
return DbU::lambda (2);
|
|
}
|
|
|
|
Point Grid::getSubUpLeft (unsigned step, Point& pos) {
|
|
|
|
GCell* gcell = getGCell (step, pos);
|
|
return gcell->getSubUpperLeft()->getCenter();
|
|
}
|
|
Point Grid::getSubUpRight (unsigned step, Point& pos) {
|
|
|
|
GCell* gcell = getGCell (step, pos);
|
|
assert (gcell);
|
|
assert (gcell->getSubUpperRight());
|
|
return gcell->getSubUpperRight()->getCenter();
|
|
}
|
|
Point Grid::getSubDownLeft (unsigned step, Point& pos) {
|
|
|
|
GCell* gcell = getGCell (step, pos);
|
|
return gcell->getSubBottomLeft()->getCenter();
|
|
}
|
|
Point Grid::getSubDownRight (unsigned step, Point& pos) {
|
|
|
|
GCell* gcell = getGCell (step, pos);
|
|
return gcell->getSubBottomRight()->getCenter();
|
|
}
|
|
|
|
GCell* Grid::getLeftOf (const GCell* gcell) const
|
|
{
|
|
GCell* nbl = gcell->getLeftOfMe();
|
|
if (!nbl) nbl = gcell->computeLeftOfMe();
|
|
|
|
return nbl;
|
|
}
|
|
GCell* Grid::getRightOf (const GCell* gcell) const
|
|
{
|
|
GCell* nbr = gcell->getRightOfMe();
|
|
if (!nbr) nbr = gcell->computeRightOfMe();
|
|
|
|
return nbr;
|
|
}
|
|
GCell* Grid::getUpOf (const GCell* gcell) const
|
|
{
|
|
GCell* nbu = gcell->getUpOfMe();
|
|
if (!nbu) nbu = gcell->computeUpOfMe();
|
|
|
|
return nbu;
|
|
}
|
|
GCell* Grid::getDownOf (const GCell* gcell) const
|
|
{
|
|
GCell* nbd = gcell->getDownOfMe();
|
|
if (!nbd) nbd = gcell->computeDownOfMe();
|
|
|
|
return nbd;
|
|
}
|
|
|
|
GCell* Grid::getUpperLeftCorner (unsigned step) const
|
|
{
|
|
GCell* gcell = getRoot();
|
|
|
|
for (unsigned level = 0 ; level < step ; level++) {
|
|
gcell = gcell->getSubUpperLeft();
|
|
if (!gcell) throw Error ("Nimbus crushed when diving too deep.");
|
|
}
|
|
return gcell;
|
|
}
|
|
GCell* Grid::getUpperRightCorner (unsigned step) const
|
|
{
|
|
GCell* gcell = getRoot();
|
|
|
|
for (unsigned level = 0 ; level < step ; level++) {
|
|
gcell = gcell->getSubUpperRight();
|
|
if (!gcell) throw Error ("Nimbus crushed when diving too deep.");
|
|
}
|
|
return gcell;
|
|
}
|
|
GCell* Grid::getLowerLeftCorner (unsigned step) const
|
|
{
|
|
GCell* gcell = getRoot();
|
|
|
|
for (unsigned level = 0 ; level < step ; level++) {
|
|
gcell = gcell->getSubLowerLeft();
|
|
if (!gcell) throw Error ("Nimbus crushed when diving too deep.");
|
|
}
|
|
return gcell;
|
|
}
|
|
GCell* Grid::getLowerRightCorner (unsigned step) const
|
|
{
|
|
GCell* gcell = getRoot();
|
|
|
|
for (unsigned level = 0 ; level < step ; level++) {
|
|
gcell = gcell->getSubLowerRight();
|
|
if (!gcell) throw Error ("Nimbus crushed when diving too deep.");
|
|
}
|
|
return gcell;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
* ********************************************************************
|
|
*
|
|
* Grid_GCells implementation
|
|
*/
|
|
|
|
Grid_GCells::Grid_GCells(const Grid* grid, unsigned step)
|
|
// ************************************************************************
|
|
: Inherit()
|
|
, _nb(NULL)
|
|
, _step(step)
|
|
{
|
|
_nb = grid->getRoot();
|
|
|
|
for (unsigned level = 0 ; level < step ; level++) {
|
|
_nb = _nb->getSubUpperLeft();
|
|
if (!_nb) throw Error ("Nimbus crushed when diving too deep.");
|
|
}
|
|
}
|
|
|
|
Grid_GCells::Grid_GCells(const Grid_GCells& gcelles)
|
|
// ************************************************************************
|
|
: Inherit()
|
|
, _nb(gcelles._nb)
|
|
, _step(gcelles._step)
|
|
{
|
|
}
|
|
|
|
Grid_GCells& Grid_GCells::operator=(const Grid_GCells& gcelles)
|
|
// ***********************************************************************************
|
|
{
|
|
_nb = gcelles._nb;
|
|
_step = gcelles._step;
|
|
return *this;
|
|
}
|
|
|
|
Collection<GCell*>* Grid_GCells::getClone() const
|
|
// ***********************************************
|
|
{
|
|
return new Grid_GCells(*this);
|
|
}
|
|
|
|
Locator<GCell*>* Grid_GCells::getLocator() const
|
|
// ******************************************************
|
|
{
|
|
return new Locator(_nb);
|
|
}
|
|
|
|
string Grid_GCells::_getString() const
|
|
// *************************************
|
|
{
|
|
string s = "<" + _TName("GCell::GCells");
|
|
if (_nb) s += " " + getString(_nb);
|
|
s += ">";
|
|
return s;
|
|
}
|
|
|
|
/*
|
|
* ********************************************************************
|
|
* Grid_GCells::Locator implementation
|
|
*/
|
|
Grid_GCells::Locator::Locator(GCell* gcell)
|
|
// ********************************************************
|
|
: Inherit(),
|
|
_nb(gcell),
|
|
_direction(0)
|
|
{
|
|
}
|
|
|
|
Grid_GCells::Locator::Locator(const Locator& locator)
|
|
// ****************************************************
|
|
: Inherit(),
|
|
_nb(locator._nb),
|
|
_direction(locator._direction)
|
|
{
|
|
}
|
|
|
|
Grid_GCells::Locator& Grid_GCells::Locator::operator=(const Locator& locator)
|
|
// ******************************************************************************
|
|
{
|
|
_nb = locator._nb;
|
|
_direction = locator._direction;
|
|
return *this;
|
|
}
|
|
|
|
GCell* Grid_GCells::Locator::getElement() const
|
|
// *********************************************
|
|
{
|
|
return _nb;
|
|
}
|
|
|
|
Locator<GCell*>* Grid_GCells::Locator::getClone() const
|
|
// *****************************************************
|
|
{
|
|
return new Locator(*this);
|
|
}
|
|
|
|
bool Grid_GCells::Locator::isValid() const
|
|
// *****************************************
|
|
{
|
|
return (_nb != NULL);
|
|
}
|
|
|
|
void Grid_GCells::Locator::progress()
|
|
// ************************************
|
|
{
|
|
GCell* gcell = NULL;
|
|
|
|
switch (_direction) {
|
|
|
|
case 0:
|
|
gcell = _nb->computeRightOfMe();
|
|
if (!gcell) {
|
|
gcell = _nb->computeDownOfMe();
|
|
_direction = 1;
|
|
}
|
|
break;
|
|
case 1:
|
|
gcell = _nb->computeLeftOfMe();
|
|
if (!gcell) {
|
|
gcell = _nb->computeDownOfMe();
|
|
_direction = 0;
|
|
}
|
|
break;
|
|
default:
|
|
gcell = NULL;
|
|
break;
|
|
}
|
|
|
|
_nb = gcell;
|
|
}
|
|
|
|
string Grid_GCells::Locator::_getString() const
|
|
// **********************************************
|
|
{
|
|
string s = "<" + _TName("GCells::GCells::Locator");
|
|
if (_nb) s += " " + getString(_nb);
|
|
s += ">";
|
|
return s;
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
* ********************************************************************
|
|
* Grid_FastGCells implementation
|
|
*/
|
|
Grid_FastGCells::Grid_FastGCells(const Grid* grid, unsigned step)
|
|
// **************************************************************************************
|
|
: Inherit()
|
|
, _nb(NULL)
|
|
, _step(step)
|
|
{
|
|
_nb = grid->getRoot();
|
|
|
|
for (unsigned level = 0 ; level < step ; level++) {
|
|
_nb = _nb->getSubUpperLeft();
|
|
if (!_nb) throw Error ("Nimbus crushed when diving too deep.");
|
|
}
|
|
}
|
|
|
|
Grid_FastGCells::Grid_FastGCells(const Grid_FastGCells& gcelles)
|
|
// ************************************************************************************
|
|
: Inherit()
|
|
, _nb(gcelles._nb)
|
|
, _step(gcelles._step)
|
|
{
|
|
}
|
|
|
|
Grid_FastGCells& Grid_FastGCells::operator=(const Grid_FastGCells& gcelles)
|
|
// ****************************************************************
|
|
{
|
|
_nb = gcelles._nb;
|
|
_step = gcelles._step;
|
|
return *this;
|
|
}
|
|
|
|
Collection<GCell*>* Grid_FastGCells::getClone() const
|
|
// ***********************************************
|
|
{
|
|
return new Grid_FastGCells(*this);
|
|
}
|
|
|
|
Locator<GCell*>* Grid_FastGCells::getLocator() const
|
|
// **********************************************
|
|
{
|
|
return new Locator(_nb);
|
|
}
|
|
|
|
string Grid_FastGCells::_getString() const
|
|
// *************************************
|
|
{
|
|
string s = "<" + _TName("GCell::GCells");
|
|
if (_nb) s += " " + getString(_nb);
|
|
s += ">";
|
|
return s;
|
|
}
|
|
|
|
/*
|
|
* ********************************************************************
|
|
* Grid_FastGCells::Locator implementation
|
|
*/
|
|
Grid_FastGCells::Locator::Locator(GCell* gcell)
|
|
// ********************************************************
|
|
: Inherit(),
|
|
_nb(gcell),
|
|
_direction(0)
|
|
{
|
|
}
|
|
|
|
Grid_FastGCells::Locator::Locator(const Locator& locator)
|
|
// ****************************************************
|
|
: Inherit(),
|
|
_nb(locator._nb),
|
|
_direction(locator._direction)
|
|
{
|
|
}
|
|
|
|
Grid_FastGCells::Locator& Grid_FastGCells::Locator::operator=(const Locator& locator)
|
|
// ******************************************************************************
|
|
{
|
|
_nb = locator._nb;
|
|
_direction = locator._direction;
|
|
return *this;
|
|
}
|
|
|
|
GCell* Grid_FastGCells::Locator::getElement() const
|
|
// *********************************************************
|
|
{
|
|
return _nb;
|
|
}
|
|
|
|
Locator<GCell*>* Grid_FastGCells::Locator::getClone() const
|
|
// *****************************************************************
|
|
{
|
|
return new Locator(*this);
|
|
}
|
|
|
|
bool Grid_FastGCells::Locator::isValid() const
|
|
// **************************************************
|
|
{
|
|
return (_nb != NULL);
|
|
}
|
|
|
|
void Grid_FastGCells::Locator::progress()
|
|
// *********************************************
|
|
{
|
|
GCell* gcell = NULL;
|
|
|
|
switch (_direction) {
|
|
|
|
case 0:
|
|
gcell = _nb->getRightOfMe();
|
|
if (!gcell) {
|
|
gcell = _nb->getDownOfMe();
|
|
_direction = 1;
|
|
}
|
|
break;
|
|
case 1:
|
|
gcell = _nb->getLeftOfMe();
|
|
if (!gcell) {
|
|
gcell = _nb->getDownOfMe();
|
|
_direction = 0;
|
|
}
|
|
break;
|
|
default:
|
|
gcell = NULL;
|
|
break;
|
|
}
|
|
|
|
_nb = gcell;
|
|
}
|
|
|
|
string Grid_FastGCells::Locator::_getString() const
|
|
// *******************************************************
|
|
{
|
|
string s = "<" + _TName("Grid::FastGCells::Locator");
|
|
if (_nb) s += " " + getString(_nb);
|
|
s += ">";
|
|
return s;
|
|
}
|
|
|
|
|
|
#if 0
|
|
|
|
|
|
|
|
|
|
/*
|
|
* ********************************************************************
|
|
* Grid_Fences implementation
|
|
*/
|
|
Grid_Fences::Grid_Fences(const Grid* grid, unsigned step)
|
|
// **************************************************************************************
|
|
: Inherit()
|
|
, _nb(NULL)
|
|
, _fence(NULL)
|
|
, _step(step)
|
|
{
|
|
_nb = grid->getRoot();
|
|
|
|
for (unsigned level = 0 ; level < step ; level++) {
|
|
_nb = _nb->getSubUpperLeft();
|
|
if (!_nb) throw Error ("Nimbus crushed while diving too deep.");
|
|
}
|
|
}
|
|
|
|
Grid_Fences::Grid_Fences(const Grid_Fences& fences)
|
|
// ************************************************************************************
|
|
: Inherit()
|
|
, _nb(fences._nb)
|
|
, _fence(fences._nb)
|
|
, _step(fences._step)
|
|
{
|
|
}
|
|
|
|
Grid_Fences& Grid_Fences::operator=(const Grid_Fences& fences)
|
|
// ****************************************************************
|
|
{
|
|
_nb = ._nb;
|
|
_step = ._step;
|
|
return *this;
|
|
}
|
|
|
|
Collection<Fence*>* Grid_Fences::getClone() const
|
|
// ***********************************************
|
|
{
|
|
return new Grid_Fences(*this);
|
|
}
|
|
|
|
Locator<Fence*>* Grid_Fences::getLocator() const
|
|
// **********************************************
|
|
{
|
|
return new Locator(_nb);
|
|
}
|
|
|
|
string Grid_Fences::_getString() const
|
|
// *************************************
|
|
{
|
|
string s = "<" + _TName("Fence::Fencees");
|
|
if (_nb) s += " " + getString(_nb);
|
|
s += ">";
|
|
return s;
|
|
}
|
|
|
|
/*
|
|
* ********************************************************************
|
|
* Grid_Fences::Locator implementation
|
|
*/
|
|
Grid_Fences::Locator::Locator(Fence* gcell)
|
|
// ********************************************************
|
|
: Inherit(),
|
|
_nb(gcell),
|
|
_direction(0)
|
|
{
|
|
}
|
|
|
|
Grid_Fences::Locator::Locator(const Locator& locator)
|
|
// ****************************************************
|
|
: Inherit(),
|
|
_nb(locator._nb),
|
|
_direction(locator._direction)
|
|
{
|
|
}
|
|
|
|
Grid_Fences::Locator& Grid_Fences::Locator::operator=(const Locator& locator)
|
|
// ******************************************************************************
|
|
{
|
|
_nb = locator._nb;
|
|
_direction = locator._direction;
|
|
return *this;
|
|
}
|
|
|
|
Fence* Grid_Fences::Locator::getElement() const
|
|
// *********************************************************
|
|
{
|
|
return _nb;
|
|
}
|
|
|
|
Locator<Fence*>* Grid_Fences::Locator::getClone() const
|
|
// *****************************************************************
|
|
{
|
|
return new Locator(*this);
|
|
}
|
|
|
|
bool Grid_Fences::Locator::isValid() const
|
|
// **************************************************
|
|
{
|
|
return (_nb != NULL);
|
|
}
|
|
|
|
void Grid_Fences::Locator::progress()
|
|
// *********************************************
|
|
{
|
|
Fence* gcell = NULL;
|
|
|
|
switch (_direction) {
|
|
|
|
case 0:
|
|
gcell = _nb->getLeftOfMe();
|
|
if (!gcell) {
|
|
gcell = _nb->getDownOfMe();
|
|
_direction = 1;
|
|
}
|
|
break;
|
|
case 1:
|
|
gcell = _nb->getRightOfMe();
|
|
if (!gcell) {
|
|
gcell = _nb->getDownOfMe();
|
|
_direction = 0;
|
|
}
|
|
break;
|
|
default:
|
|
gcell = NULL;
|
|
break;
|
|
}
|
|
|
|
_nb = gcell;
|
|
}
|
|
|
|
string Grid_Fences::Locator::_getString() const
|
|
// *******************************************************
|
|
{
|
|
string s = "<" + _TName("Grid::FastFencees::Locator");
|
|
if (_nb) s += " " + getString(_nb);
|
|
s += ">";
|
|
return s;
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/*
|
|
* FenceSet implementation
|
|
*
|
|
*/
|
|
Grid::FenceSet::FenceSet()
|
|
// **********************
|
|
: Inherit()
|
|
{
|
|
}
|
|
|
|
unsigned Grid::FenceSet::_getHashValue(Fence* fence) const
|
|
{
|
|
return ( (unsigned int)( (unsigned long)fence ) ) / 2;
|
|
}
|
|
|
|
Fence* Grid::FenceSet::_getNextElement(Fence* fence) const
|
|
{
|
|
return fence->_getNextOfGridFenceSet();
|
|
}
|
|
|
|
void Grid::FenceSet::_setNextElement(Fence* fence, Fence* nextFence) const
|
|
{
|
|
fence->_setNextOfGridFenceSet(nextFence);
|
|
return;
|
|
}
|
|
|
|
|
|
}
|