2010-03-09 09:24:55 -06:00
|
|
|
// -*- C++ -*-
|
|
|
|
//
|
|
|
|
// This file is part of the Coriolis Software.
|
2016-03-06 05:36:18 -06:00
|
|
|
// Copyright (c) UPMC/LIP6 2008-2016, All Rights Reserved
|
2010-03-09 09:24:55 -06:00
|
|
|
//
|
2013-12-03 18:59:29 -06:00
|
|
|
// +-----------------------------------------------------------------+
|
2010-03-09 09:24:55 -06:00
|
|
|
// | C O R I O L I S |
|
|
|
|
// | K i t e - D e t a i l e d R o u t e r |
|
|
|
|
// | |
|
|
|
|
// | Author : Jean-Paul CHAPUT |
|
|
|
|
// | E-mail : Jean-Paul.Chaput@asim.lip6.fr |
|
|
|
|
// | =============================================================== |
|
|
|
|
// | C++ Module : "./TrackSegmentCost.cpp" |
|
2013-12-03 18:59:29 -06:00
|
|
|
// +-----------------------------------------------------------------+
|
2010-03-09 09:24:55 -06:00
|
|
|
|
|
|
|
|
|
|
|
#include <cstdlib>
|
|
|
|
#include <iostream>
|
|
|
|
#include <sstream>
|
|
|
|
#include <map>
|
|
|
|
|
|
|
|
#include "hurricane/Bug.h"
|
|
|
|
#include "hurricane/DebugSession.h"
|
|
|
|
#include "katabatic/AutoSegment.h"
|
|
|
|
#include "kite/TrackElement.h"
|
|
|
|
#include "kite/TrackSegmentCost.h"
|
|
|
|
|
|
|
|
|
|
|
|
namespace Kite {
|
|
|
|
|
|
|
|
|
|
|
|
using std::cerr;
|
|
|
|
using std::endl;
|
|
|
|
using std::map;
|
|
|
|
using std::multimap;
|
|
|
|
using std::make_pair;
|
|
|
|
using std::ostringstream;
|
|
|
|
using Hurricane::Bug;
|
|
|
|
using Hurricane::DebugSession;
|
|
|
|
using Hurricane::inltrace;
|
|
|
|
using Hurricane::ltracein;
|
|
|
|
using Hurricane::ltraceout;
|
|
|
|
using Hurricane::tab;
|
|
|
|
|
|
|
|
|
|
|
|
// -------------------------------------------------------------------
|
|
|
|
// Class : "TrackSegmentCost".
|
|
|
|
|
|
|
|
|
|
|
|
TrackSegmentCost::TrackSegmentCost ( TrackElement* trackSegment )
|
|
|
|
: _terminals (0)
|
|
|
|
, _ripupCount (0)
|
|
|
|
, _leftMinExtend (DbU::Max)
|
|
|
|
, _rightMinExtend(DbU::Min)
|
|
|
|
, _net (trackSegment->getNet())
|
|
|
|
, _attractors ()
|
|
|
|
{
|
|
|
|
//update ( trackSegment );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
TrackSegmentCost::~TrackSegmentCost ()
|
|
|
|
{ }
|
|
|
|
|
|
|
|
|
|
|
|
DbU::Unit TrackSegmentCost::getWiringDelta ( DbU::Unit axis ) const
|
|
|
|
{
|
|
|
|
DbU::Unit attraction = 0;
|
|
|
|
for ( size_t i=0 ; i < _attractors.size() ; i++ ) {
|
|
|
|
if ( _attractors[i] > axis ) attraction += _attractors[i] - axis;
|
|
|
|
else attraction += axis - _attractors[i];
|
|
|
|
}
|
|
|
|
return attraction;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void TrackSegmentCost::update ( TrackElement* trackSegment )
|
|
|
|
{
|
|
|
|
DebugSession::open ( trackSegment->getNet(), 148 );
|
|
|
|
|
|
|
|
ltrace(148) << "TrackSegmentCost::update() - " << trackSegment << endl;
|
|
|
|
ltracein(148);
|
|
|
|
|
|
|
|
vector<AutoSegment*> collapseds;
|
|
|
|
vector<AutoSegment*> perpandiculars;
|
|
|
|
map<DbU::Unit,int> attractorSpins;
|
|
|
|
|
|
|
|
AutoSegment::getTopologicalInfos ( trackSegment->base()
|
|
|
|
, collapseds
|
|
|
|
, perpandiculars
|
|
|
|
, _leftMinExtend
|
|
|
|
, _rightMinExtend
|
|
|
|
);
|
|
|
|
|
|
|
|
_terminals = AutoSegment::getTerminalCount ( trackSegment->base(), collapseds );
|
|
|
|
_attractors.clear ();
|
|
|
|
|
|
|
|
for ( size_t i=0 ; i < perpandiculars.size() ; i++ ) {
|
|
|
|
Interval interval;
|
|
|
|
TrackElement* perpandicular;
|
|
|
|
|
|
|
|
if ( perpandiculars[i]->isCanonical() ) {
|
|
|
|
perpandicular = Session::lookup ( perpandiculars[i]->base() );
|
|
|
|
if ( perpandicular )
|
|
|
|
perpandicular->getCanonical ( interval );
|
|
|
|
} else {
|
|
|
|
perpandicular = Session::lookup ( perpandiculars[i]->getCanonical(interval)->base() );
|
|
|
|
}
|
|
|
|
|
* ./kite:
- New: In RoutingEvent, while routing a full chip, transient full blockages
may happens due to the initial position of some perpandiculars on theirs
optimal positions. Check for it and perform a "ripupPerpandicular" with
the perpandiculars re-routed first (on normal operation, this is the
reverse).
- Change: In NegociateWindow::NegociateOverlapCost(), do not account terminals
if the layer is *above* METAL3 as it is unlikely that it's truly directly
connected to a terminal (true at least for our designs).
- Change: In RoutingEvent::State, add a "fullBlocked" flag to perform the
full blockage direction while building the state object.
- Change: In RoutingEvent::cacheAxisHint(), when the TrackSegment has parent,
uses the parent current axis position and not it's axis hint.
- Change: In Manipulator::insertInTrack(), when the overlapped segment is a
Local and is completly enclosed (shrinkLeft & shrinkRight), no longer
rip it up but force a shrink left/right instead.
- Bug: In RoutingEvent, the event sorting function was sorting in the *wrong*
order! The less prioritary first! With the correct sort, we won an
additionnal 30% in speed (total: 69%). From the "reference" time we have
a 3.2 speed-up. And we can successfully process denser designs...
2011-01-09 12:08:57 -06:00
|
|
|
if ( not perpandicular ) {
|
2010-03-09 09:24:55 -06:00
|
|
|
cerr << Bug("Not a TrackSegment: %s:%s\n (perpandicular: %s:%s)"
|
|
|
|
,getString((void*)perpandiculars[i]->getCanonical(interval)->base()).c_str()
|
|
|
|
,getString(perpandiculars[i]->getCanonical(interval)).c_str()
|
|
|
|
,getString((void*)perpandiculars[i]->base()).c_str()
|
|
|
|
,getString(perpandiculars[i]).c_str()
|
|
|
|
) << endl;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
interval.inflate ( DbU::lambda(-1.5) );
|
|
|
|
|
|
|
|
ltrace(148) << "| perpandicular: " << perpandiculars[i] << endl;
|
|
|
|
ltrace(148) << "| canonical: " << perpandicular << endl;
|
|
|
|
ltracein(148);
|
|
|
|
ltrace(148) << "interval: " << interval << endl;
|
|
|
|
|
|
|
|
if ( interval.isPonctual() ) {
|
|
|
|
ltrace(148) << "Punctual attractor @" << DbU::getValueString(interval.getVMin()) << endl;
|
|
|
|
_attractors.push_back ( interval.getVMin() );
|
|
|
|
ltraceout(148);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
* ./kite:
- New: In RoutingEvent, while routing a full chip, transient full blockages
may happens due to the initial position of some perpandiculars on theirs
optimal positions. Check for it and perform a "ripupPerpandicular" with
the perpandiculars re-routed first (on normal operation, this is the
reverse).
- Change: In NegociateWindow::NegociateOverlapCost(), do not account terminals
if the layer is *above* METAL3 as it is unlikely that it's truly directly
connected to a terminal (true at least for our designs).
- Change: In RoutingEvent::State, add a "fullBlocked" flag to perform the
full blockage direction while building the state object.
- Change: In RoutingEvent::cacheAxisHint(), when the TrackSegment has parent,
uses the parent current axis position and not it's axis hint.
- Change: In Manipulator::insertInTrack(), when the overlapped segment is a
Local and is completly enclosed (shrinkLeft & shrinkRight), no longer
rip it up but force a shrink left/right instead.
- Bug: In RoutingEvent, the event sorting function was sorting in the *wrong*
order! The less prioritary first! With the correct sort, we won an
additionnal 30% in speed (total: 69%). From the "reference" time we have
a 3.2 speed-up. And we can successfully process denser designs...
2011-01-09 12:08:57 -06:00
|
|
|
if ( ( interval.getVMin() != trackSegment->getAxis() )
|
|
|
|
or AutoSegment::isTopologicalBound(perpandiculars[i]
|
|
|
|
,false
|
|
|
|
,perpandicular->isHorizontal()
|
|
|
|
) ) {
|
2010-03-09 09:24:55 -06:00
|
|
|
map<DbU::Unit,int>::iterator iattractor = attractorSpins.find ( interval.getVMin() );
|
|
|
|
if ( iattractor == attractorSpins.end() ) {
|
|
|
|
attractorSpins.insert ( make_pair(interval.getVMin(),-1) );
|
|
|
|
} else {
|
|
|
|
iattractor->second -= 1;
|
|
|
|
}
|
|
|
|
ltrace(148) << "Left attractor @" << DbU::getValueString(interval.getVMin()) << endl;
|
|
|
|
}
|
|
|
|
|
* ./kite:
- New: In RoutingEvent, while routing a full chip, transient full blockages
may happens due to the initial position of some perpandiculars on theirs
optimal positions. Check for it and perform a "ripupPerpandicular" with
the perpandiculars re-routed first (on normal operation, this is the
reverse).
- Change: In NegociateWindow::NegociateOverlapCost(), do not account terminals
if the layer is *above* METAL3 as it is unlikely that it's truly directly
connected to a terminal (true at least for our designs).
- Change: In RoutingEvent::State, add a "fullBlocked" flag to perform the
full blockage direction while building the state object.
- Change: In RoutingEvent::cacheAxisHint(), when the TrackSegment has parent,
uses the parent current axis position and not it's axis hint.
- Change: In Manipulator::insertInTrack(), when the overlapped segment is a
Local and is completly enclosed (shrinkLeft & shrinkRight), no longer
rip it up but force a shrink left/right instead.
- Bug: In RoutingEvent, the event sorting function was sorting in the *wrong*
order! The less prioritary first! With the correct sort, we won an
additionnal 30% in speed (total: 69%). From the "reference" time we have
a 3.2 speed-up. And we can successfully process denser designs...
2011-01-09 12:08:57 -06:00
|
|
|
if ( ( interval.getVMax() != trackSegment->getAxis() )
|
|
|
|
or AutoSegment::isTopologicalBound(perpandiculars[i]
|
|
|
|
,true
|
|
|
|
,perpandicular->isHorizontal()
|
|
|
|
) ) {
|
2010-03-09 09:24:55 -06:00
|
|
|
map<DbU::Unit,int>::iterator iattractor = attractorSpins.find ( interval.getVMax() );
|
|
|
|
if ( iattractor == attractorSpins.end() ) {
|
|
|
|
attractorSpins.insert ( make_pair(interval.getVMax(),1) );
|
|
|
|
} else {
|
|
|
|
iattractor->second += 1;
|
|
|
|
}
|
|
|
|
ltrace(148) << "Right attractor @" << DbU::getValueString(interval.getVMax()) << endl;
|
|
|
|
}
|
|
|
|
|
|
|
|
ltraceout(148);
|
|
|
|
}
|
|
|
|
|
|
|
|
map<DbU::Unit,int>::iterator iattractor = attractorSpins.begin();
|
|
|
|
for ( ; iattractor != attractorSpins.end() ; iattractor++ ) {
|
|
|
|
if ( iattractor->second != 0 )
|
|
|
|
_attractors.push_back ( iattractor->first );
|
|
|
|
}
|
|
|
|
|
|
|
|
ostringstream s;
|
|
|
|
s << "Attractors [";
|
|
|
|
for ( size_t i=0 ; i<_attractors.size() ; i++ ) {
|
|
|
|
if ( i ) s << ", ";
|
|
|
|
s << DbU::getValueString(_attractors[i]);
|
|
|
|
}
|
|
|
|
s << "]";
|
|
|
|
ltrace(148) << s.str() << endl;
|
|
|
|
|
|
|
|
|
|
|
|
ltraceout(148);
|
|
|
|
DebugSession::close ();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
string TrackSegmentCost::_getString () const
|
|
|
|
{
|
|
|
|
return "<" + _getTypeName() + " "
|
|
|
|
+ getString(_terminals)
|
|
|
|
+ " [" + DbU::getValueString(_leftMinExtend)
|
|
|
|
+ ":" + DbU::getValueString(_rightMinExtend)
|
|
|
|
+ "]>";
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Record* TrackSegmentCost::_getRecord () const
|
|
|
|
{
|
|
|
|
Record* record = new Record ( getString(this) );
|
|
|
|
record->add ( getSlot ( "_terminals" , _terminals ) );
|
|
|
|
record->add ( getSlot ( "_ripupCount" , _ripupCount ) );
|
|
|
|
record->add ( getSlot ( "_leftMinExtend" , &_leftMinExtend ) );
|
|
|
|
record->add ( getSlot ( "_rightMinExtend", &_rightMinExtend ) );
|
|
|
|
record->add ( getSlot ( "_net" , _net ) );
|
|
|
|
|
|
|
|
return record;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
} // End of Kite namespace.
|