2016-05-23 09:15:25 -05:00
|
|
|
// -*- 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 <boost/bind.hpp>
|
|
|
|
#include <QAction>
|
|
|
|
#include <QMenu>
|
|
|
|
#include <QMenuBar>
|
|
|
|
#include <QApplication>
|
|
|
|
#include <hurricane/Warning.h>
|
|
|
|
#include <hurricane/Error.h>
|
|
|
|
#include <hurricane/Breakpoint.h>
|
|
|
|
#include <hurricane/DebugSession.h>
|
2016-05-30 11:52:38 -05:00
|
|
|
#include <hurricane/UpdateSession.h>
|
2016-05-23 09:15:25 -05:00
|
|
|
#include <hurricane/Go.h>
|
|
|
|
#include <hurricane/Net.h>
|
|
|
|
#include <hurricane/Cell.h>
|
|
|
|
#include <hurricane/viewer/Graphics.h>
|
|
|
|
#include <hurricane/viewer/CellWidget.h>
|
|
|
|
#include <hurricane/viewer/CellViewer.h>
|
|
|
|
#include <hurricane/viewer/ControllerWidget.h>
|
|
|
|
#include <hurricane/viewer/ExceptionWidget.h>
|
|
|
|
#include <crlcore/Utilities.h>
|
|
|
|
#include <crlcore/AllianceFramework.h>
|
|
|
|
#include "anabatic/GCell.h"
|
|
|
|
#include "anabatic/GraphicAnabaticEngine.h"
|
2016-05-30 04:30:29 -05:00
|
|
|
#include "anabatic/Dijkstra.h"
|
2016-05-23 09:15:25 -05:00
|
|
|
|
|
|
|
|
|
|
|
namespace Anabatic {
|
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
using Hurricane::Error;
|
|
|
|
using Hurricane::Warning;
|
|
|
|
using Hurricane::Exception;
|
|
|
|
using Hurricane::Breakpoint;
|
|
|
|
using Hurricane::DebugSession;
|
2016-05-30 11:52:38 -05:00
|
|
|
using Hurricane::UpdateSession;
|
2016-05-23 09:15:25 -05:00
|
|
|
using Hurricane::Point;
|
|
|
|
using Hurricane::Entity;
|
|
|
|
using Hurricane::Net;
|
|
|
|
using Hurricane::Graphics;
|
|
|
|
using Hurricane::ColorScale;
|
|
|
|
using Hurricane::DisplayStyle;
|
2016-05-26 06:56:16 -05:00
|
|
|
using Hurricane::DrawingStyle;
|
2016-05-23 09:15:25 -05:00
|
|
|
using Hurricane::ControllerWidget;
|
|
|
|
using Hurricane::ExceptionWidget;
|
|
|
|
using CRL::Catalog;
|
|
|
|
using CRL::AllianceFramework;
|
2016-05-26 11:30:03 -05:00
|
|
|
|
|
|
|
|
|
|
|
// -------------------------------------------------------------------
|
|
|
|
// 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 )
|
|
|
|
{
|
2016-05-30 11:52:38 -05:00
|
|
|
UpdateSession::open();
|
|
|
|
|
2016-05-26 11:30:03 -05:00
|
|
|
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);
|
2016-05-30 11:52:38 -05:00
|
|
|
|
|
|
|
UpdateSession::close();
|
2016-05-26 11:30:03 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void anabaticTest_3 ( AnabaticEngine* engine )
|
|
|
|
{
|
|
|
|
for ( int i=0 ; i<4 ; ++i ) {
|
|
|
|
cdebug.log(110,1) << "Running test 3, loop:" << i << " ..." << endl;
|
2016-05-23 09:15:25 -05:00
|
|
|
|
2016-05-26 11:30:03 -05:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-05-30 04:30:29 -05:00
|
|
|
void anabaticTest_4 ( AnabaticEngine* engine )
|
|
|
|
{
|
|
|
|
Cell* cell = engine->getCell();
|
|
|
|
cell->flattenNets( Cell::Flags::BuildRings );
|
|
|
|
cell->createRoutingPadRings( Cell::Flags::BuildRings );
|
|
|
|
|
|
|
|
engine->getSouthWestGCell()->doGrid();
|
|
|
|
|
2016-05-30 11:52:38 -05:00
|
|
|
vector<Net*> 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 ) {
|
2016-05-30 04:30:29 -05:00
|
|
|
dijkstra->load( net );
|
|
|
|
dijkstra->run();
|
|
|
|
}
|
2016-05-30 11:52:38 -05:00
|
|
|
delete dijkstra;
|
2016-05-30 04:30:29 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-05-26 11:30:03 -05:00
|
|
|
// -------------------------------------------------------------------
|
|
|
|
// Class : "Anabatic::GraphicAnabaticEngine".
|
2016-05-23 09:15:25 -05:00
|
|
|
|
|
|
|
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<const GCell*>(go);
|
|
|
|
|
|
|
|
QPainter& painter = widget->getPainter();
|
Anabatic transient commit 6 "Somatoline, ca marche!".
* Bug: In Anabatic:
- In Dijktra::load(), do not create Contact while looping over the Net's
components (for RoutingPads).
- In Dijkstra::propagate(), reset the connexId of reached Vertexes
on updating the stamp to avoid using connexId from previous runs.
Push the vertexes on the queue while backtracking (with a distance
of 0.0). Do not reset the "_from" as we need it in Dijkstra::toWires()
to know how were the routing is.
- In Edge::getDistance(), convert to float more early so the normalisation
do not always end up in zero.
- In GCell::_add(), when the axis of the new edge is equal to one already
on the given side, adds it *after* the existing edge and not before.
(to be coherent with the way GCells are split in two)
- In GCell::hcut() and GCell::vcut(), create an Edge if is equal to X/Y Max,
*but* the new chunk is flat. The new chunk is then expected to expand
"below".
- In GraphicsAnabaticEngine::drawGCell(), display the id of the GCell in
a small box at the center of it's bounding box. Nothing displayed for
flat GCells, to avoid cluttering.
2016-06-05 11:38:09 -05:00
|
|
|
QPen pen = Graphics::getPen ("Anabatic::GCell",widget->getDarkening());
|
|
|
|
Box bb = gcell->getBoundingBox();
|
2016-05-23 09:15:25 -05:00
|
|
|
|
Anabatic transient commit 6 "Somatoline, ca marche!".
* Bug: In Anabatic:
- In Dijktra::load(), do not create Contact while looping over the Net's
components (for RoutingPads).
- In Dijkstra::propagate(), reset the connexId of reached Vertexes
on updating the stamp to avoid using connexId from previous runs.
Push the vertexes on the queue while backtracking (with a distance
of 0.0). Do not reset the "_from" as we need it in Dijkstra::toWires()
to know how were the routing is.
- In Edge::getDistance(), convert to float more early so the normalisation
do not always end up in zero.
- In GCell::_add(), when the axis of the new edge is equal to one already
on the given side, adds it *after* the existing edge and not before.
(to be coherent with the way GCells are split in two)
- In GCell::hcut() and GCell::vcut(), create an Edge if is equal to X/Y Max,
*but* the new chunk is flat. The new chunk is then expected to expand
"below".
- In GraphicsAnabaticEngine::drawGCell(), display the id of the GCell in
a small box at the center of it's bounding box. Nothing displayed for
flat GCells, to avoid cluttering.
2016-06-05 11:38:09 -05:00
|
|
|
painter.setPen ( pen );
|
2016-05-26 06:56:16 -05:00
|
|
|
painter.setBrush( Graphics::getBrush("Anabatic::GCell",widget->getDarkening()) );
|
Anabatic transient commit 6 "Somatoline, ca marche!".
* Bug: In Anabatic:
- In Dijktra::load(), do not create Contact while looping over the Net's
components (for RoutingPads).
- In Dijkstra::propagate(), reset the connexId of reached Vertexes
on updating the stamp to avoid using connexId from previous runs.
Push the vertexes on the queue while backtracking (with a distance
of 0.0). Do not reset the "_from" as we need it in Dijkstra::toWires()
to know how were the routing is.
- In Edge::getDistance(), convert to float more early so the normalisation
do not always end up in zero.
- In GCell::_add(), when the axis of the new edge is equal to one already
on the given side, adds it *after* the existing edge and not before.
(to be coherent with the way GCells are split in two)
- In GCell::hcut() and GCell::vcut(), create an Edge if is equal to X/Y Max,
*but* the new chunk is flat. The new chunk is then expected to expand
"below".
- In GraphicsAnabaticEngine::drawGCell(), display the id of the GCell in
a small box at the center of it's bounding box. Nothing displayed for
flat GCells, to avoid cluttering.
2016-06-05 11:38:09 -05:00
|
|
|
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 ();
|
2016-05-23 09:15:25 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
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<const Edge*>(go);
|
|
|
|
|
|
|
|
if (edge) {
|
2016-05-26 06:56:16 -05:00
|
|
|
Box bb = edge->getBoundingBox();
|
2016-05-23 09:15:25 -05:00
|
|
|
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 );
|
2016-05-26 06:56:16 -05:00
|
|
|
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 );
|
2016-05-23 09:15:25 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
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 );
|
|
|
|
|
2016-05-26 11:30:03 -05:00
|
|
|
//anabaticTest_1( engine );
|
|
|
|
//anabaticTest_2( engine );
|
2016-05-30 04:30:29 -05:00
|
|
|
//anabaticTest_3( engine );
|
|
|
|
anabaticTest_4( engine );
|
2016-05-23 09:15:25 -05:00
|
|
|
|
2016-05-26 06:56:16 -05:00
|
|
|
if (_viewer) _viewer->emitCellChanged();
|
2016-05-23 09:15:25 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
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( "<b>Anabatic:</b> GraphicAnabaticEngine not bound to any Viewer." );
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_viewer->getCell() == NULL) {
|
|
|
|
throw Error( "<b>Anabatic:</b> 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.
|