// This file is part of the Coriolis Project. // Copyright (C) Laboratoire LIP6 - Departement ASIM // Universite Pierre et Marie Curie // // Main contributors : // Christophe Alexandre // Sophie Belloeil // Hugo Clément // Jean-Paul Chaput // Damien Dupuis // Christian Masson // Marek Sroka // // The Coriolis Project is free software; you can redistribute it and/or // modify it under the terms of the GNU General Public License as // published by the Free Software Foundation; either version 2 of the // License, or (at your option) any later version. // // The Coriolis Project is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU // General Public License for more details. // // You should have received a copy of the GNU General Public License // along with the Coriolis Project; if not, write to the Free Software // Foundation, inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA // // // License-Tag // // Date : 29/01/2004 // Author : Christophe Alexandre // // Authors-Tag #include "hurricane/Warning.h" #include "hurricane/Cell.h" #include "crlcore/CellGauge.h" #include "crlcore/AllianceFramework.h" #include "nimbus/NimbusEngine.h" #include "mauka/MaukaEngine.h" #include "mauka/SimAnnealingPlacer.h" #include "mauka/SubRow.h" #include "mauka/Row.h" #include "mauka/Surface.h" namespace { using namespace std; using namespace Hurricane; using namespace Mauka; void VerifyInstanceQuadtreeInsertion(Cell *cell) { cerr << "VerifyInstanceQuadtreeInsertion" << endl; for_each_instance(instance, cell->getInstances()) { if (!instance->isMaterialized()) { Cell* masterCell = instance->getMasterCell(); cerr << tab << instance << " of " << masterCell << " is not materialized..." << endl; tab++; VerifyInstanceQuadtreeInsertion(masterCell); tab--; } end_for; } } void DisplayInstanceOccurrence(Occurrence& instanceOccurrence) { cerr << instanceOccurrence << " " << instanceOccurrence.getBoundingBox() << endl; Instance* instance = static_cast(instanceOccurrence.getEntity()); Path path = instanceOccurrence.getPath(); cerr << path.getTransformation() << endl; cerr << instance->getTransformation() << endl; while (!path.isEmpty()) { cerr << path.getTailInstance() << " " << path.getTailInstance()->getTransformation() << endl; cerr << path.getTailInstance()->getMasterCell() << endl; cerr << path.getTailInstance()->getMasterCell()->getAbutmentBox() << endl; cerr << path.getTailInstance()->getMasterCell()->getBoundingBox() << endl; cerr << path.getTailInstance()->getAbutmentBox() << endl; cerr << path.getTailInstance()->getBoundingBox() << endl; path = path.getHeadPath(); } } void DisplayNonLeafInstances(Cell* cell, Box area) { ltrace(100) << "display of " << cell << " in " << area << endl; for_each_instance(instance, cell->getNonLeafInstancesUnder(area)) { ltrace(100) << instance << " " << instance->getBoundingBox() << endl; Cell* masterCell = instance->getMasterCell(); ltracein(20); DisplayNonLeafInstances(masterCell, area); ltraceout(20); end_for; } } } // End of anonymous namespace. namespace Mauka { using CRL::AllianceFramework; using Nimbus::NimbusEngine; using Nimbus::GCellLocator; Surface::Surface(MaukaEngine* mauka, const Box& placementbox) : Inherit(mauka->getCell(), placementbox), _mauka(mauka), _rowVector(), _rowYMax(), _rowYMinInv(), _rowZeroOrientation(false), _margin(0.0), _binWidthMax(0), _binWidthMin(0), _searchWidth(0), _searchHeight(0) {} Surface* Surface::create(MaukaEngine* mauka, const Box& placementbox) { if (!mauka) { throw Error("Can't create " + _TName("Surface") + " : empty mauka"); } if (!mauka->getCell()) { throw Error("Can't create " + _TName("Surface") + " : empty cell on mauka"); } Surface* surface = new Surface(mauka, placementbox); surface->_postCreate(); return surface; } namespace { void PlacementVerification(Cell* cell, Box& box) { set occurrenceSet; for_each_occurrence(occurrence, cell->getLeafInstanceOccurrencesUnder(box)) { occurrenceSet.insert(occurrence); end_for; } // for_each_occurrence(occurrence, cell->getLeafInstanceOccurrences()) { // if (occurrenceSet.find(occurrence) == occurrenceSet.end()) { // cerr << occurrence << ":" << occurrence.getBoundingBox() << endl; // throw Error("occurrence is badly placed"); // } // end_for; // } } typedef list InstanceOccurrencesList; class PlacementProblem { public: GCell* _gcell; public: InstanceOccurrencesList _fixedInstanceOccurrenceList; public: MaukaEngine::UVector _toPlaceInstanceOccurrencesUVector; public: Surface::SubRowList _subRowList; PlacementProblem(GCell* gcell) : _gcell(gcell) , _fixedInstanceOccurrenceList() , _toPlaceInstanceOccurrencesUVector() , _subRowList() {} }; typedef list PlacementProblemList; } void Surface::_postCreate () { typedef set OccurrenceSet; Inherit::_postCreate(); NimbusEngine* nimbus = NULL; if (_box.isEmpty()) { nimbus = NimbusEngine::get ( getCell() ); if (nimbus == NULL) { throw Error("Can't create Mauka::Surface, no Box and no Nimbus"); } _box = nimbus->getGrid()->getRoot()->getBox(); } PlacementVerification ( getCell(), _box ); DbU::Unit sliceHeight = DbU::lambda(1); DbU::Unit pitch = DbU::lambda(1); AllianceFramework* af = AllianceFramework::get(); if (af != NULL) { sliceHeight = af->getCellGauge()->getSliceHeight(); pitch = af->getCellGauge()->getPitch(); } else { // For the moment very stupid method: take the first instantiated Cell. forEach ( Instance*, iinstance, getCell()->getInstances() ) { Cell* masterCell = iinstance->getMasterCell(); sliceHeight = masterCell->getAbutmentBox().getHeight(); break; } } if ( _box.isEmpty() or _box.isPonctual() or _box.isFlat()) throw Error("Wrong Box for Area: %s",getString(_box).c_str()); if ( getHeight() % sliceHeight ) throw Error("Box Height %s must be a multiple of Slice Height %s" ,DbU::getValueString(getHeight()).c_str() ,DbU::getValueString(sliceHeight).c_str()); DbU::Unit instanceToPlaceWidthMax = 0; DbU::Unit instanceToPlaceWidthSum = 0; for ( unsigned int id = 0; id < _mauka->_instanceWidths.size(); ++id ) { DbU::Unit instanceWidth = _mauka->_instanceWidths[id]; instanceToPlaceWidthSum += instanceWidth; if ( instanceWidth % pitch ) throw Error("Width of %s (%s) is not a multiple of pitch (%s)." ,getString(_mauka->_instanceOccurrencesVector[id]).c_str() ,DbU::getValueString(instanceWidth).c_str() ,DbU::getValueString(pitch).c_str()); if (instanceWidth > instanceToPlaceWidthMax) instanceToPlaceWidthMax = instanceWidth; } _binWidthMax = DbU::lambda ((unsigned)( 2.0 * DbU::getLambda(instanceToPlaceWidthMax) / DbU::getLambda(pitch)) * DbU::getLambda(pitch)); _binWidthMin = DbU::lambda ((unsigned)(DbU::getLambda(_binWidthMax) / (DbU::getLambda(pitch) * 2)) * DbU::getLambda(pitch)); DbU::Unit surfaceTotalWidth = 0; PlacementProblemList placementProblemList; OccurrenceSet verifyInstanceOccurrencesSet; if ( nimbus ) { forEach ( GCell*, igcell, nimbus->getPlacementLeaves() ) { PlacementProblem* placementProblem = new PlacementProblem(*igcell); placementProblemList.push_back(placementProblem); InstanceOccurrencesList toPlaceInstanceOccurrencesList; // Search for preplaced leaf instances forEach ( Occurrence , ioccurrence, _mauka->getCell()->getLeafInstanceOccurrencesUnder(igcell->getBox()) ) { Instance* instance = static_cast((*ioccurrence).getEntity()); if ( instance->isFixed() ) { placementProblem->_fixedInstanceOccurrenceList.push_back(*ioccurrence); } else { MaukaEngine::InstanceOccurrencesMap::const_iterator iomit = _mauka->_instanceOccurrencesMap.find(*ioccurrence); if ( iomit == _mauka->_instanceOccurrencesMap.end() ) throw Error("Instance occurrence unexpectedly appeared:\n" " %s",getString(*ioccurrence).c_str()); placementProblem->_toPlaceInstanceOccurrencesUVector.push_back(iomit->second); verifyInstanceOccurrencesSet.insert(*ioccurrence); } } forEach(Occurrence, ioccurrence, _mauka->getCell()->getLeafInstanceOccurrences()) { Instance* instance = static_cast((*ioccurrence).getEntity()); if ( instance->isFixed() ) continue; MaukaEngine::InstanceOccurrencesMap::const_iterator iomit = _mauka->_instanceOccurrencesMap.find(*ioccurrence); if (iomit == _mauka->_instanceOccurrencesMap.end()) throw Error("Instance occurrence unexpectedly appeared:\n" " %s",getString(*ioccurrence).c_str()); placementProblem->_toPlaceInstanceOccurrencesUVector.push_back(iomit->second); verifyInstanceOccurrencesSet.insert(*ioccurrence); } DbU::Unit searchWidth = DbU::lambda(_mauka->getSearchRatio() * DbU::getLambda(igcell->getWidth())); if (_searchWidth < searchWidth) _searchWidth = searchWidth; DbU::Unit searchHeight = DbU::lambda(_mauka->getSearchRatio() * DbU::getLambda(igcell->getHeight())); if (_searchHeight < searchHeight) _searchHeight = searchHeight; } for ( PlacementProblemList::iterator pplit = placementProblemList.begin(); pplit != placementProblemList.end(); pplit++) { PlacementProblem* placementProblem = *pplit; GCell* gcell = placementProblem->_gcell; if ( placementProblem->_fixedInstanceOccurrenceList.size() != 0 ) { bool rowZeroOrientation = false; int width = (unsigned)(gcell->getWidth () / pitch); // Number of X pitch int height = (unsigned)(gcell->getHeight() / sliceHeight); // Number of Y slices MaukaEngine::PrePlaceTab prePlaceTab(height, MaukaEngine::PrePlaceRow(width, false)); InstanceOccurrencesList::const_iterator iolit = placementProblem->_fixedInstanceOccurrenceList.begin(); // Find the orientation of the first row. Instance* refInstance = static_cast(iolit->getEntity()); Box refAbutmentBox = refInstance->getAbutmentBox(); Transformation refTransformation = refInstance->getTransformation(); iolit->getPath().getTransformation().applyOn(refAbutmentBox); iolit->getPath().getTransformation().applyOn(refTransformation); Box preplacedBox = gcell->getIntersection(refAbutmentBox); // Position en y ramene au slice int refInsYPos = (int)((preplacedBox.getYMin() - gcell->getYMin()) / sliceHeight); // Hauteur ramene a la hauteur du slice int refInsSliceHeight = (int)(preplacedBox.getHeight() / sliceHeight); Transformation::Orientation orientation = refTransformation.getOrientation(); if ( refInsYPos % 2 == 0 ) { if ( (refInsSliceHeight % 2 == 0) or (orientation == Transformation::Orientation::ID) or (orientation == Transformation::Orientation::MX)) rowZeroOrientation = true; else rowZeroOrientation = false; } else { if ( (refInsSliceHeight % 2 == 0) or (orientation == Transformation::Orientation::ID) or (orientation == Transformation::Orientation::MX)) rowZeroOrientation = false; else rowZeroOrientation = true; } // Tests for each Instance. for ( InstanceOccurrencesList::iterator iolit = placementProblem->_fixedInstanceOccurrenceList.begin(); iolit != placementProblem->_fixedInstanceOccurrenceList.end(); iolit++ ) { Instance* instance = static_cast(iolit->getEntity()); Box instanceAbutmentBox = instance->getAbutmentBox(); iolit->getPath().getTransformation().applyOn(instanceAbutmentBox); Box preplacedBox = gcell->getIntersection(instanceAbutmentBox); DbU::Unit insWidth = preplacedBox.getWidth(); DbU::Unit insHeight = preplacedBox.getHeight(); int insPitchWidth = (int)(insWidth / pitch); // Largeur ramene au pitch int insSliceHeight = (int)(insHeight / sliceHeight); // Hauteur ramene a la hauteur du slice int ypos = (int)((preplacedBox.getYMin() - gcell->getYMin()) / sliceHeight); // Position en Y ramene au slice int xpos = (int)((preplacedBox.getXMin() - gcell->getXMin()) / pitch); // Position en X ramene au pitch for ( int yit = ypos; yit < ypos + insSliceHeight; ++yit ) { for ( int xit = xpos; xit < xpos + insPitchWidth; ++xit ) { if ( (xit > width - 1) or (yit > height - 1) or (xit < 0) or (yit < 0) ) throw Error("%s is out of the abutment box" ,getString(*iolit).c_str()); if ( prePlaceTab[yit][xit] == false ) prePlaceTab[yit][xit] = true; else throw Error("%s is badly placed. There is another instance at it's position" ,getString(*iolit).c_str()); } } } bool rowOrientation = rowZeroOrientation; for ( int y = 0; y < (int)prePlaceTab.size(); ++y ) { int x = 0; while ( x < (int)prePlaceTab[y].size() ) { while ( (x < (int)prePlaceTab[y].size()) and (prePlaceTab[y][x] == true) ) ++x; DbU::Unit subRowXMin = gcell->getXMin() + x * pitch; if (x >= (int)prePlaceTab[y].size()) break; while ( (x < (int)prePlaceTab[y].size()) and (prePlaceTab[y][x] == false) ) ++x; DbU::Unit subRowXMax = gcell->getXMin() + x * pitch; if ( subRowXMax - subRowXMin > _binWidthMin ) { SubRow* subRow = SubRow::create ( getCell() , this , Box(subRowXMin, gcell->getYMin() + y * sliceHeight ,subRowXMax, gcell->getYMin() + (y+1) * sliceHeight) , rowOrientation ); placementProblem->_subRowList.push_back(subRow); } } rowOrientation = not rowOrientation; } } else { bool rowOrientation = false; for ( DbU::Unit ymin = gcell->getYMin() ; ymin <= gcell->getYMax() - sliceHeight ; ymin += sliceHeight ) { SubRow* subRow = SubRow::create ( getCell() , this , Box(gcell->getXMin(), ymin ,gcell->getXMax(), ymin + sliceHeight) , rowOrientation ); placementProblem->_subRowList.push_back(subRow); rowOrientation = not rowOrientation; } } } } else { PlacementProblem* placementProblem = new PlacementProblem(NULL); _searchHeight = getHeight(); _searchWidth = getWidth(); bool rowOrientation = false; unsigned nRows = getHeight() / sliceHeight; surfaceTotalWidth = getWidth() * nRows; for ( DbU::Unit ymin = getYMin(); ymin <= getYMax() - sliceHeight; ymin += sliceHeight ) { SubRow* subRow = SubRow::create ( getCell() , this , Box(getXMin(), ymin,getXMax(), ymin + sliceHeight) , rowOrientation ); placementProblem->_subRowList.push_back(subRow); rowOrientation = not rowOrientation; } for ( unsigned int id=0; id < _mauka->_instanceOccurrencesVector.size(); ++id ) { placementProblem->_toPlaceInstanceOccurrencesUVector.push_back(id); } throw Error("MaukaEngine needs NimbusEngine"); } for (PlacementProblemList::iterator pplit = placementProblemList.begin(); pplit != placementProblemList.end(); pplit++) { _DisplayInstances((*pplit)->_toPlaceInstanceOccurrencesUVector, (*pplit)->_subRowList); delete *pplit; } placementProblemList.clear (); if ( verifyInstanceOccurrencesSet.size() != _mauka->_instanceOccurrencesVector.size() ) { cerr << Error("There are %d Instances not taken into account" ,(_mauka->_instanceOccurrencesVector.size() - verifyInstanceOccurrencesSet.size())) << endl; VerifyInstanceQuadtreeInsertion(_mauka->getCell()); for ( MaukaEngine::InstanceOccurrencesVector::iterator iovit = _mauka->_instanceOccurrencesVector.begin(); iovit != _mauka->_instanceOccurrencesVector.end(); iovit++ ) { OccurrenceSet::iterator osit = verifyInstanceOccurrencesSet.find(*iovit); if ( osit == verifyInstanceOccurrencesSet.end() ) { cerr << " o Problem with " << *iovit << endl; DisplayInstanceOccurrence ( *iovit ); DisplayNonLeafInstances ( _mauka->getCell(), iovit->getBoundingBox() ); return; } } throw Error("There are %d Instances not taken into account" ,(_mauka->_instanceOccurrencesVector.size() - verifyInstanceOccurrencesSet.size())); } for ( RowVector::const_iterator rvit = _rowVector.begin(); rvit != _rowVector.end(); rvit++ ) { surfaceTotalWidth += (*rvit)->getSubRowsWidth(); } _computeRowsAndSubRows(); _margin = 1.0 - DbU::getLambda(instanceToPlaceWidthSum) / DbU::getLambda(surfaceTotalWidth); if ( _margin < 0 ) { throw Error("There is not enough free space to place the circuit %s < %s" ,DbU::getValueString(surfaceTotalWidth).c_str() ,DbU::getValueString(instanceToPlaceWidthSum).c_str() ); } if (_margin < 0.15) cerr << Warning("Low margin %d, may not be able to complete successfully",_margin) << endl; _computeCapacity(); } void Surface::_preDestroy() { Inherit::_preDestroy(); for (RowVector::const_iterator rvit = _rowVector.begin(); rvit != _rowVector.end(); rvit++) (*rvit)->destroy(); _rowVector.clear(); } void Surface::_computeCapacity() { for (RowVector::const_iterator rvit = _rowVector.begin(); rvit != _rowVector.end(); rvit++) (*rvit)->_computeCapacity(_margin); } namespace { struct RowSortByY { bool operator()(const Container* row1, const Container* row2) const { return row1->getYMin() < row2->getYMin(); } }; } Row* Surface::InsertSubRowInRow(SubRow* subrow, bool orientation) { //cerr << subrow << endl; Row* row = NULL; if (_rowVector.size() == 0) { row = Row::create(getCell(), this, subrow->getBox(), orientation); _rowVector.push_back(row); } else { RowVector::iterator rvit = lower_bound(_rowVector.begin(), _rowVector.end(), subrow, RowSortByY()); if (rvit == _rowVector.end()) { row = Row::create(getCell(), this, subrow->getBox(), orientation); _rowVector.push_back(row); } else if ((*rvit)->getYMin() == subrow->getYMin()) { row = *rvit; if (row->_orientation != orientation) { cerr << "PROBLEM in " << row << " orientation with " << subrow << endl; } //assert(row->_orientation == orientation); } else { row = Row::create(getCell(), this, subrow->getBox(), orientation); _rowVector.insert(rvit, row); } } row->_InsertSubRow(subrow); //cerr << row << endl; return row; } void Surface::_computeRowsAndSubRows() { _rowYMax.clear(); _rowYMinInv.clear(); for (unsigned i = 0; i < _rowVector.size(); i++) { Row* row = _rowVector[i]; _rowYMax[row->getYMax()] = i; _rowYMinInv[row->getYMin()] = i; row->_computeSubRows(); } } Bin* Surface::getBinInSurface(Bin* srcbin, double dist) { Point srcPos = srcbin->getCenter(); DbU::Unit lowerY = srcPos.getY() - DbU::lambda((dist * DbU::getLambda(_searchHeight))); if (lowerY < getYMin()) lowerY = getYMin(); DbU::Unit upperY = srcPos.getY() + DbU::lambda((dist * DbU::getLambda(_searchHeight))); if (upperY > getYMax()) upperY = getYMax(); RowYMax::iterator rinf = _rowYMax.upper_bound(lowerY); RowYMax::iterator rsup = _rowYMinInv.upper_bound(upperY); unsigned randidx = rinf->second + (unsigned)((rsup->second - rinf->second + 1) * (rand() / (RAND_MAX+1.0))); Row* searchRow = _rowVector[randidx]; DbU::Unit lowerX = srcPos.getX() - DbU::lambda((dist * DbU::getLambda(_searchWidth))); if ((lowerX < searchRow->getXMin()) || (lowerX > searchRow->getXMax())) lowerX = searchRow->getXMin(); DbU::Unit upperX = srcPos.getX() + DbU::lambda((dist * DbU::getLambda(_searchWidth))); if ((upperX > searchRow->getXMax()) || (upperX < searchRow->getXMin())) upperX = searchRow->getXMax(); SubRow* subRow = searchRow->getSubRowBetween(lowerX, upperX); if ((lowerX < subRow->getXMin()) || (lowerX > subRow->getXMax())) lowerX = subRow->getXMin(); if ((upperX > subRow->getXMax()) || (upperX < subRow->getXMin())) upperX = subRow->getXMax(); Bin* dstBin = subRow->getBinBetween(lowerX, upperX, srcbin); return dstBin; } double Surface::getBinCost() const { double binCost = 0.0; for (RowVector::const_iterator rvit = _rowVector.begin(); rvit != _rowVector.end(); rvit++) binCost += (*rvit)->getBinCost(); return binCost; } double Surface::getRowCost() const { double rowCost = 0.0; for (RowVector::const_iterator rvit = _rowVector.begin(); rvit != _rowVector.end(); rvit++) rowCost += (*rvit)->getCost(); return rowCost; } DbU::Unit Surface::getBinsSize() const { DbU::Unit totalBinsSize = 0; for (RowVector::const_iterator rvit = _rowVector.begin(); rvit != _rowVector.end(); rvit++) totalBinsSize += (*rvit)->getBinsSize(); return totalBinsSize; } DbU::Unit Surface::getBinsCapa() const { DbU::Unit totalBinsCapa = 0; for (RowVector::const_iterator rvit = _rowVector.begin(); rvit != _rowVector.end(); rvit++) totalBinsCapa += (*rvit)->getBinsCapa(); return totalBinsCapa; } DbU::Unit Surface::getSubRowsCapa() const { DbU::Unit totalSubRowsCapa = 0; for (RowVector::const_iterator rvit = _rowVector.begin(); rvit != _rowVector.end(); rvit++) { totalSubRowsCapa += (*rvit)->getSubRowsCapa(); } return totalSubRowsCapa; } namespace { class sortInstanceOccurrencesIdsByWidth { private: MaukaEngine* _mauka; public: sortInstanceOccurrencesIdsByWidth(MaukaEngine* mauka): _mauka(mauka) {} public: bool operator()(unsigned instanceoccurrenceid1, unsigned instanceoccurrenceid2) const { DbU::Unit width1 = _mauka->getInstanceIdWidth(instanceoccurrenceid1); DbU::Unit width2 = _mauka->getInstanceIdWidth(instanceoccurrenceid2); return width1 > width2; } }; } void Surface::_DisplayInstances(MaukaEngine::UVector& instanceids, SubRowList& subrowlist) { // Insert instances in subrows sort(instanceids.begin(), instanceids.end(), sortInstanceOccurrencesIdsByWidth(_mauka)); SubRowList::iterator srlit = subrowlist.begin(); MaukaEngine::UVector::const_iterator insIterator = instanceids.begin(); MaukaEngine::UVector::const_iterator lastLoopInsertedInsIterator = insIterator; while(true) { if (insIterator == instanceids.end()) { cmess1 << " o Initial placement computing ... done. " << endl; break; // end of insertion } if (srlit == subrowlist.end()) { srlit = subrowlist.begin(); if (lastLoopInsertedInsIterator != insIterator) lastLoopInsertedInsIterator = insIterator; else { // insertion of instances with respect of Bins margin // did not succed, inserting what's left. while (insIterator != instanceids.end()) { for (SubRow::BinVector::iterator bvit = (*srlit)->_binVector.begin(); bvit != (*srlit)->_binVector.end(); bvit++) { Bin* bin = *bvit; unsigned instanceId = *insIterator; DbU::Unit instanceWidth = _mauka->_instanceWidths[instanceId]; if (instanceWidth > (*srlit)->getWidthVsSize()) break; bin->addInstance(instanceId); ++insIterator; if (insIterator == instanceids.end()) break; } if (insIterator == instanceids.end()) break; ++srlit; if (srlit == subrowlist.end()) { if (lastLoopInsertedInsIterator == insIterator) { //impossible to succeed // try to merge all bins in subrows bool merged = false; for (srlit = subrowlist.begin(); srlit != subrowlist.end(); srlit++) { merged = (*srlit)->_MergeBins(); } if (merged) { insIterator = instanceids.begin(); } else { //count instances to place unsigned nbInstancesToPlace = 0; while (insIterator++ != instanceids.end()) ++nbInstancesToPlace; throw Error("Not enough free space to place all the instances.\n" "Please increase the abutment box: %s (%d instances remains to place)" ,getString(getBoundingBox()).c_str() ,nbInstancesToPlace ); } } srlit = subrowlist.begin(); lastLoopInsertedInsIterator = insIterator; } } } } if (insIterator != instanceids.end()) { SubRow* subRow = *srlit; for (SubRow::BinVector::iterator bvit = (*srlit)->_binVector.begin(); bvit != (*srlit)->_binVector.end(); bvit++) { Bin* bin = *bvit; if (insIterator == instanceids.end()) break; unsigned instanceId = *insIterator; DbU::Unit instanceWidth = _mauka->_instanceWidths[instanceId]; if (instanceWidth > subRow->getCapaVsSize()) break; if (bin->TryAddInstance(instanceId)) ++insIterator; } ++srlit; } } } Record* Surface::_getRecord() const { Record* record = Inherit::_getRecord(); if (record) { record->add(getSlot("MaukaEngine", _mauka)); } return record; } void Surface::DisplayBinHits() const { for (RowVector::const_iterator rvit = _rowVector.begin(); rvit != _rowVector.end(); rvit++) (*rvit)->DisplayBinHits(); } void Surface::PlotBinsStats(ofstream& out) const { out << "plot [:] '-' w l 2" << endl ; } void Surface::Plot(ofstream& out) const { out << getXMin() << " " << getYMin() << endl << getXMin() << " " << getYMax() << endl << getXMax() << " " << getYMax() << endl << getXMax() << " " << getYMin() << endl << getXMin() << " " << getYMin() << endl << endl; out << "EOF" << endl; } }