2010-03-09 09:24:55 -06:00
|
|
|
// -*- C++ -*-
|
|
|
|
//
|
|
|
|
// This file is part of the Coriolis Software.
|
Support of RoutingGauge, part 2.
In Katabatic & Kite, remove all hard-coded values related to track pitches.
* New: In <Session>, add more convenience function to access RoutingGauge
characteristics.
* New: In <AutoSegment>, <AutoContact>, add support for the "depth spin",
that is, if the source/target contacts are going "top" or "down".
Used to compute the perpandicular pitch. Need a small modification
of the revalidation mechanism. The observers of <AutoSegment> are
notified when the spin changes.
* New: In <AutoSegment>, the getPPitch() method allow to compute the
"perpandicular pitch". For now it is simply the greatest from the
source perpandicular pitch and the target perpandicular pitch.
Make uses of the "depth spin".
* New: In <TrackElement>, <TrackSegment>, cache the perpandicular pitch.
Updated through the notification from the observable.
2014-05-19 10:58:38 -05:00
|
|
|
// Copyright (c) UPMC/LIP6 2008-2014, 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.
|