// -*- C++ -*- // // This file is part of the Coriolis Software. // Copyright (c) UPMC/LIP6 2008-2010, All Rights Reserved // // =================================================================== // // $Id$ // // x-----------------------------------------------------------------x // | | // | 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@asim.lip6.fr | // | =============================================================== | // | C++ Module : "./ChipTools.cpp" | // | *************************************************************** | // | U p d a t e s | // | | // x-----------------------------------------------------------------x #include #include #include "hurricane/Warning.h" #include "hurricane/Bug.h" #include "hurricane/DataBase.h" #include "hurricane/Technology.h" #include "hurricane/Horizontal.h" #include "hurricane/Vertical.h" #include "hurricane/RoutingPad.h" #include "crlcore/RoutingGauge.h" #include "crlcore/AllianceFramework.h" #include "katabatic/Session.h" #include "katabatic/AutoContact.h" #include "katabatic/AutoSegment.h" #include "katabatic/AutoHorizontal.h" #include "katabatic/AutoVertical.h" #include "katabatic/GCell.h" #include "katabatic/GCellGrid.h" #include "katabatic/KatabaticEngine.h" namespace { using namespace std; using namespace CRL; using namespace Hurricane; using namespace Katabatic; enum SegmentType { LocalSegments=0x10, GlobalSegments=0x20 }; bool isChip ( Cell* cell, Instance*& core ) { AllianceFramework* af = AllianceFramework::get(); int pads = 0; int cores = 0; forEach ( Instance*, iinstance, cell->getInstances() ) { if ( af->isPad(iinstance->getMasterCell()) ) ++pads; else { ++cores; core = *iinstance; } } return (pads > 0) and (cores == 1); } void breakSegments ( GCell* begin, GCell* end, Layer::Mask mask, unsigned int flags ) { for ( GCell* gcell=begin ; gcell != NULL ; ) { ltrace(200) << "Pre-break in " << gcell << endl; if ( (flags & Constant::Horizontal) and (flags & GlobalSegments) ) { vector* hsegments = gcell->getHSegments(); for ( size_t i=0 ; i<(*hsegments).size() ; ++i ) { if ( not ((*hsegments)[i]->getLayer()->getMask() & mask) ) continue; ltrace(200) << "Pre-break: " << (*hsegments)[i] << " @" << gcell << endl; (*hsegments)[i]->makeDogLeg ( gcell, true ); } if ( gcell == end ) break; gcell = gcell->getUp(); } if ( (flags & Constant::Vertical) and (flags & GlobalSegments) ) { vector* vsegments = gcell->getVSegments(); for ( size_t i=0 ; i<(*vsegments).size() ; ++i ) { if ( not ((*vsegments)[i]->getLayer()->getMask() & mask) ) continue; ltrace(200) << "Pre-break: " << (*vsegments)[i] << " @" << gcell << endl; (*vsegments)[i]->makeDogLeg ( gcell, true ); } if ( gcell == end ) break; gcell = gcell->getUp(); } if ( flags & GlobalSegments ) { vector* contacts = gcell->getContacts(); vector segments; for ( size_t i=0 ; isize() ; i++ ) { forEach ( Component*, component, (*contacts)[i]->getSlaveComponents() ) { Segment* segment = dynamic_cast(*component); AutoSegment* autoSegment = Session::lookup ( segment ); if ( autoSegment and (autoSegment->getDirection() & flags) ) segments.push_back ( autoSegment ); } } for ( size_t i=0 ; igetDirection() << "&" << flags << " " << segments[i] << endl; segments[i]->makeDogLeg ( gcell, true ); } } } Session::revalidate (); } void reselectPadRp ( Cell* cell ) { AllianceFramework* af = AllianceFramework::get(); forEach ( Net*, inet, cell->getNets() ) { if ( inet->getType() == Net::Type::GROUND ) continue; if ( inet->getType() == Net::Type::POWER ) continue; if ( inet->getType() == Net::Type::CLOCK ) continue; forEach ( RoutingPad*, irp, inet->getRoutingPads() ) { Instance* instance = irp->getOccurrence().getPath().getTailInstance(); if ( instance ) { Cell* masterCell = instance->getMasterCell(); if ( af->isPad(masterCell) ) irp->setOnBestComponent(RoutingPad::LowestLayer); } } // RoutingPad*. } // Net*. } } // End of anonymous namespace. namespace Katabatic { void KatabaticEngine::slackenBorder ( Box bb, Layer::Mask mask, unsigned int flags ) { GCell* begin = getGCellGrid()->getGCell ( Point(bb.getXMin(),bb.getYMin()) ); GCell* end = getGCellGrid()->getGCell ( Point(bb.getXMin(),bb.getYMax()) ); breakSegments ( begin, end, mask, flags ); begin = getGCellGrid()->getGCell ( Point(bb.getXMax(),bb.getYMin()) ); end = getGCellGrid()->getGCell ( Point(bb.getXMax(),bb.getYMax()) ); breakSegments ( begin, end, mask, flags ); Session::revalidate (); } void KatabaticEngine::slackenBlockIos ( Instance* core ) { cmess1 << " o Slackening IOs of <" << core->getName() << ">." << endl; Layer::Mask mask = Session::getRoutingLayer(1)->getMask(); slackenBorder ( core->getBoundingBox().inflate(DbU::lambda(50.0)) , mask , GlobalSegments|Constant::Horizontal ); } void KatabaticEngine::chipPrep () { if ( isChip() ) { reselectPadRp ( getCell() ); // slackenBlockIos ( _core ); // cmess1 << " o Slackening Pads-connected segments." << endl; // slackenBorder ( _cell->getBoundingBox().inflate(DbU::lambda(-425.0)) // , Session::getRoutingLayer(3)->getMask() // , GlobalSegments|LocalSegments|Constant::Horizontal // ); // slackenBorder ( _cell->getBoundingBox().inflate(DbU::lambda(-425.0)) // , Session::getRoutingLayer(1)->getMask() // , GlobalSegments|Constant::Horizontal // ); } } ChipTools::ChipTools ( Cell* cell ) : _cell (cell) , _core (NULL) , _isChip (false) , _chipBb (cell->getBoundingBox()) , _leftPadsBb () , _rightPadsBb () , _topPadsBb () , _bottomPadsBb() , _chipCorona () { _isChip = ::isChip ( _cell, _core ); if ( _isChip ) { // Ugly: hard-coded pads height. const DbU::Unit padHeight = DbU::lambda(400.0); Box outer = _cell->getBoundingBox().inflate ( -padHeight ); _chipCorona = Torus ( outer, _core->getBoundingBox() ); _leftPadsBb = Box ( _chipBb.getXMin() , _chipBb.getYMin(), _chipCorona.getOuterBox().getXMin(), _chipBb.getYMax() ); _rightPadsBb = Box ( _chipCorona.getOuterBox().getXMax(), _chipBb.getYMin(), _chipBb.getXMax(), _chipBb.getYMax() ); _bottomPadsBb = Box ( _chipBb.getXMin() , _chipBb.getYMin(), _chipBb.getXMax(), _chipCorona.getOuterBox().getYMin() ); _topPadsBb = Box ( _chipBb.getXMin(), _chipCorona.getOuterBox().getYMax(), _chipBb.getXMax(), _chipBb.getYMax() ); cmess1 << " o Design is a full chip." << endl; cmess1 << " - Core: <" << _core->getName() << ">/<" << _core->getMasterCell()->getName() << ">." << endl; cmess1 << " - Corona: " << _chipCorona << "." << endl; } else { _chipCorona = Torus ( _cell->getBoundingBox(), _cell->getBoundingBox() ); } } string ChipTools::_getString () const { ostringstream s; s << "<" << _getTypeName() << " " << _cell->getName() << " core:" << getString(((_core) ? _core->getName() : "NULL")) << ">"; return s.str(); } Record* ChipTools::_getRecord () const { Record* record = new Record ( _getString() ); record->add ( getSlot ( "_cell" , _cell ) ); record->add ( getSlot ( "_core" , _core ) ); record->add ( getSlot ( "_isChip" , &_isChip ) ); record->add ( getSlot ( "_chipBb" , &_chipBb ) ); record->add ( getSlot ( "_leftPadsBb" , &_leftPadsBb ) ); record->add ( getSlot ( "_rightPadsBb" , &_rightPadsBb ) ); record->add ( getSlot ( "_topPadsBb" , &_topPadsBb ) ); record->add ( getSlot ( "_bottomPadsBb", &_bottomPadsBb ) ); record->add ( getSlot ( "_chipCorona" , &_chipCorona ) ); return record; } } // End of Katabatic namespace.