// -*- C++ -*- // // 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 : "GraphicAnabaticEngine.cpp" | // +-----------------------------------------------------------------+ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "anabatic/GCell.h" #include "anabatic/GraphicAnabaticEngine.h" #include "anabatic/Dijkstra.h" namespace Anabatic { using namespace std; using Hurricane::Error; using Hurricane::Warning; using Hurricane::Exception; using Hurricane::Breakpoint; using Hurricane::DebugSession; using Hurricane::UpdateSession; using Hurricane::Point; using Hurricane::Entity; using Hurricane::Net; using Hurricane::Graphics; using Hurricane::ColorScale; using Hurricane::DisplayStyle; using Hurricane::DrawingStyle; using Hurricane::ControllerWidget; using Hurricane::ExceptionWidget; using CRL::Catalog; using CRL::AllianceFramework; // ------------------------------------------------------------------- // Test functions. void anabaticTest_1 ( AnabaticEngine* engine ) { engine->getSouthWestGCell()->doGrid(); Point position ( DbU::fromLambda(100.0), DbU::fromLambda(100.0) ); GCell* gcell = engine->getGCellUnder( position ); cerr << "Gcell under:" << position << " is " << gcell << endl; } void anabaticTest_2 ( AnabaticEngine* engine ) { UpdateSession::open(); GCell* row0 = engine->getSouthWestGCell(); DbU::Unit xcorner = engine->getCell()->getAbutmentBox().getXMin(); DbU::Unit ycorner = engine->getCell()->getAbutmentBox().getYMin(); cdebug.log(119,1) << "row0: " << row0 << endl; GCell* row1 = row0->hcut( ycorner+DbU::fromLambda(50.0) ); cdebug.tabw(119,-1); cdebug.log(119,1) << "row1: " << row1 << endl; GCell* row2 = row1->hcut( ycorner+DbU::fromLambda(2*50.0) ); cdebug.tabw(119,-1); cdebug.log(119,1) << "row2: " << row2 << endl; row0 = row0->vcut( xcorner+DbU::fromLambda(50.0) ); cdebug.tabw(119,-1); cdebug.log(119,1) << "row0+1: " << row0 << endl; row0 = row0->vcut( xcorner+DbU::fromLambda(3*50.0) ); cdebug.tabw(119,-1); cdebug.log(119,1) << "row0+2: " << row0 << endl; row0 = row0->vcut( xcorner+DbU::fromLambda(5*50.0) ); cdebug.tabw(119,-1); cdebug.log(119,1) << "row0+3: " << row0 << endl; row1 = row1->vcut( xcorner+DbU::fromLambda(2*50.0) ); cdebug.tabw(119,-1); cdebug.log(119,1) << "row1+1: " << row1 << endl; cdebug.tabw(119,-1); UpdateSession::close(); } void anabaticTest_3 ( AnabaticEngine* engine ) { for ( int i=0 ; i<4 ; ++i ) { cdebug.log(110,1) << "Running test 3, loop:" << i << " ..." << endl; if ( i%2) anabaticTest_1( engine ); else anabaticTest_2( engine ); engine->reset(); cdebug.log(110,1) << "Test 3, loop:" << i << " completed." << endl; cdebug.tabw(110,-1); } } void anabaticTest_4 ( AnabaticEngine* engine ) { Cell* cell = engine->getCell(); cell->flattenNets( Cell::Flags::BuildRings ); cell->createRoutingPadRings( Cell::Flags::BuildRings ); engine->getSouthWestGCell()->doGrid(); vector nets; nets.push_back( cell->getNet( "ialu.inv_x2_3_sig" ) ); nets.push_back( cell->getNet( "iram.na3_x1_13_sig" ) ); nets.push_back( cell->getNet( "iram.ram_idx_7(1)" ) ); Dijkstra* dijkstra = new Dijkstra ( engine ); for ( Net* net : nets ) { dijkstra->load( net ); dijkstra->run(); } delete dijkstra; } void anabaticTest_5 ( AnabaticEngine* engine ) { Cell* cell = engine->getCell(); cell->flattenNets( Cell::Flags::BuildRings ); cell->createRoutingPadRings( Cell::Flags::BuildRings ); //DebugSession::addToTrace( cell->getNet("alu_out(3)") ); DebugSession::addToTrace( cell->getNet("imuxe.not_aux2") ); UpdateSession::open(); engine->getSouthWestGCell()->doGrid(); Dijkstra* dijkstra = new Dijkstra ( engine ); for ( Net* net : cell->getNets() ) { if (net->isPower() or net->isClock()) continue; dijkstra->load( net ); dijkstra->run(); } delete dijkstra; UpdateSession::close(); } // ------------------------------------------------------------------- // Class : "Anabatic::GraphicAnabaticEngine". size_t GraphicAnabaticEngine::_references = 0; GraphicAnabaticEngine* GraphicAnabaticEngine::_singleton = NULL; void GraphicAnabaticEngine::initGCell ( CellWidget* widget ) { widget->getDrawingPlanes().setPen( Qt::NoPen ); } void GraphicAnabaticEngine::drawGCell ( CellWidget* widget , const Go* go , const BasicLayer* basicLayer , const Box& box , const Transformation& transformation ) { const GCell* gcell = static_cast(go); QPainter& painter = widget->getPainter(); QPen pen = Graphics::getPen ("Anabatic::GCell",widget->getDarkening()); Box bb = gcell->getBoundingBox(); painter.setPen ( pen ); painter.setBrush( Graphics::getBrush("Anabatic::GCell",widget->getDarkening()) ); painter.drawRect( widget->dbuToScreenRect(bb) ); if (gcell->isFlat()) return; QString text = QString("id:%1").arg(gcell->getId()); QFont font = Graphics::getFixedFont( QFont::Bold ); painter.setFont(font); pen.setWidth( 1 ); painter.setPen( pen ); painter.save (); painter.translate( widget->dbuToScreenPoint(bb.getCenter().getX(), bb.getCenter().getY()) ); painter.drawRect (QRect( -75, -25, 150, 50 )); painter.drawText (QRect( -75, -25, 150, 50 ) , text , QTextOption(Qt::AlignCenter) ); painter.restore (); } void GraphicAnabaticEngine::initEdge ( CellWidget* widget ) { widget->getDrawingPlanes().setPen( Qt::NoPen ); } void GraphicAnabaticEngine::drawEdge ( CellWidget* widget , const Go* go , const BasicLayer* basicLayer , const Box& box , const Transformation& transformation ) { const Edge* edge = static_cast(go); if (edge) { Box bb = edge->getBoundingBox(); unsigned int occupancy = 255; if (edge->getRealOccupancy() < edge->getCapacity()) occupancy = (unsigned int)( 255.0 * ( (float)edge->getRealOccupancy() / (float)edge->getCapacity() ) ); QPainter& painter = widget->getPainter(); if (edge->getRealOccupancy() > edge->getCapacity()) { QColor color ( Qt::cyan ); painter.setPen( DisplayStyle::darken(color,widget->getDarkening()) ); } QBrush brush = QBrush( Qt::white, Qt::DiagCrossPattern ); if (edge->getCapacity() > 0.0) brush = Graphics::getColorScale( ColorScale::Fire ).getBrush( occupancy, widget->getDarkening() ); painter.setPen( Qt::NoPen ); painter.setBrush( brush ); painter.drawRect( widget->dbuToScreenRect( bb, false) ); QString text = QString("%1/%2").arg(edge->getRealOccupancy()).arg(edge->getCapacity()); QColor color ( Qt::white ); QFont font = Graphics::getFixedFont( QFont::Bold ); painter.setPen (DisplayStyle::darken(color,widget->getDarkening())); painter.setFont(font); if (edge->isVertical()) { painter.save (); painter.translate( widget->dbuToScreenPoint(bb.getXMin(), bb.getYMin()) ); painter.rotate ( -90 ); painter.drawText (QRect( 0 , 0 , widget->dbuToScreenLength(bb.getHeight()) , widget->dbuToScreenLength(bb.getWidth ())) , text , QTextOption(Qt::AlignCenter) ); painter.restore (); } else painter.drawText( widget->dbuToScreenRect(bb,false ), text, QTextOption(Qt::AlignCenter) ); painter.setPen( Qt::NoPen ); } } AnabaticEngine* GraphicAnabaticEngine::createEngine () { Cell* cell = getCell(); AnabaticEngine* engine = AnabaticEngine::get( cell ); if (not engine) { engine = AnabaticEngine::create( cell ); } else cerr << Warning( "%s already has a Anabatic engine.", getString(cell).c_str() ) << endl; return engine; } AnabaticEngine* GraphicAnabaticEngine::getForFramework ( unsigned int flags ) { // Currently, only one framework is avalaible: Alliance. AnabaticEngine* engine = AnabaticEngine::get( getCell() ); if (engine) return engine; if (flags & CreateEngine) { engine = createEngine(); if (not engine) throw Error( "Failed to create Anabatic engine on %s.", getString(getCell()).c_str() ); } else { throw Error( "AnabaticEngine not created yet, run the global router first." ); } return engine; } void GraphicAnabaticEngine::_runTest () { if (_viewer) _viewer->emitCellAboutToChange(); AnabaticEngine* engine = getForFramework( CreateEngine ); //anabaticTest_1( engine ); //anabaticTest_2( engine ); //anabaticTest_3( engine ); //anabaticTest_4( engine ); anabaticTest_5( engine ); if (_viewer) _viewer->emitCellChanged(); } void GraphicAnabaticEngine::addToMenu ( CellViewer* viewer ) { assert( _viewer == NULL ); _viewer = viewer; if (_viewer->hasMenuAction("placeAndRoute.anabatic")) { cerr << Warning( "GraphicAnabaticEngine::addToMenu() - Anabatic engine already hooked in." ) << endl; return; } _viewer->addToMenu( "placeAndRoute.anabatic" , "Anabatic - &Test Run" , "Perform a test run of Anabatic on the design" , std::bind(&GraphicAnabaticEngine::_runTest,this) ); } const Name& GraphicAnabaticEngine::getName () const { return AnabaticEngine::staticGetName(); } Cell* GraphicAnabaticEngine::getCell () { if (_viewer == NULL) { throw Error( "Anabatic: GraphicAnabaticEngine not bound to any Viewer." ); return NULL; } if (_viewer->getCell() == NULL) { throw Error( "Anabatic: No Cell is loaded into the Viewer." ); return NULL; } return _viewer->getCell(); } GraphicAnabaticEngine* GraphicAnabaticEngine::grab () { if (not _references) _singleton = new GraphicAnabaticEngine (); _references++; return _singleton; } size_t GraphicAnabaticEngine::release () { --_references; if (not _references) { delete _singleton; _singleton = NULL; } return _references; } GraphicAnabaticEngine::GraphicAnabaticEngine () : GraphicTool() , _viewer (NULL) { addDrawGo( "Anabatic::GCell", initGCell, drawGCell ); addDrawGo( "Anabatic::Edge" , initEdge , drawEdge ); } GraphicAnabaticEngine::~GraphicAnabaticEngine () { } } // Anabatic namespace.