// -*- C++ -*- // // This file is part of the Coriolis Software. // Copyright (c) UPMC 2008-2014, All Rights Reserved // // +-----------------------------------------------------------------+ // | 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 : "./TrackElement.cpp" | // +-----------------------------------------------------------------+ #include #include #include "hurricane/Bug.h" #include "hurricane/Warning.h" #include "hurricane/Net.h" #include "hurricane/Name.h" #include "katabatic/AutoContact.h" #include "katabatic/GCell.h" #include "crlcore/RoutingGauge.h" #include "kite/DataNegociate.h" #include "kite/TrackElement.h" #include "kite/TrackCost.h" #include "kite/Track.h" #include "kite/Session.h" #include "kite/RoutingEvent.h" #include "kite/NegociateWindow.h" namespace { using namespace std; using namespace Hurricane; using namespace CRL; using namespace Kite; void dummyOverlapCost ( const TrackElement* segment, TrackCost& cost ) { cerr << Warning("No overlapCost callback has been set (%s)." ,getString(segment).c_str()) << endl; } } // Anonymous namespace. namespace Kite { using Hurricane::inltrace; using Hurricane::ltracein; using Hurricane::ltraceout; using Hurricane::tab; using Hurricane::Bug; using Hurricane::Net; using Hurricane::Name; using Katabatic::GCell; // ------------------------------------------------------------------- // Comparison Classes. // // Return: lhs < rhs. bool Compare::operator() ( TrackElement* lhs, TrackElement* rhs ) { return lhs->getFreedomDegree() > rhs->getFreedomDegree(); } bool CompareByPosition::operator() ( const TrackElement* lhs, const TrackElement* rhs ) const { if (lhs == rhs) return false; if (lhs->isBlockage() xor rhs->isBlockage()) return lhs->isBlockage(); if (lhs->getLength() < rhs->getLength()) return true; if (lhs->getLength() > rhs->getLength()) return false; if (lhs->isHorizontal() xor rhs->isHorizontal()) return rhs->isHorizontal(); if (lhs->getAxis() > rhs->getAxis()) return true; if (lhs->getAxis() < rhs->getAxis()) return false; if (lhs->getSourceU() > rhs->getSourceU()) return true; if (lhs->getSourceU() < rhs->getSourceU()) return false; if (lhs->isBlockage() and rhs->isBlockage()) return false; return lhs->getId() < rhs->getId(); } // ------------------------------------------------------------------- // Class : "SegmentObserver". void SegmentObserver::notify ( unsigned int flags ) { TrackElement* segment = getOwner(); if (flags & AutoSegment::Invalidate) { if (not segment->isInvalidated()) { ltrace(200) << "::notify() on " << segment << endl; segment->invalidate(); } } if (flags & AutoSegment::Revalidate) { // Revalidation must be delayed until *all* the AutoSegments have been revalidated. // if (segment->isInvalidated()) { // ltrace(200) << "::notify() on " << segment << endl; // segment->revalidate( true ); // } } if (flags & AutoSegment::RevalidatePPitch) { segment->updatePPitch(); } } // ------------------------------------------------------------------- // Class : "TrackElement". SegmentOverlapCostCB* TrackElement::_overlapCostCallback = dummyOverlapCost; SegmentOverlapCostCB* TrackElement::setOverlapCostCB ( SegmentOverlapCostCB* cb ) { SegmentOverlapCostCB* oldCb = _overlapCostCallback; _overlapCostCallback = cb; return oldCb; } // Wrapped AutoSegment Functions. AutoSegment* TrackElement::base () const { return NULL; } bool TrackElement::isFixed () const { return false; } bool TrackElement::isLocal () const { return true; } bool TrackElement::isGlobal () const { return not isLocal(); } bool TrackElement::isBipoint () const { return false; } bool TrackElement::isTerminal () const { return false; } bool TrackElement::isStrongTerminal ( unsigned int ) const { return false; } bool TrackElement::isStrap () const { return false; } bool TrackElement::isSlackened () const { return false; } bool TrackElement::isDogleg () const { return false; } bool TrackElement::isSameLayerDogleg () const { return false; } bool TrackElement::isUserDefined () const { return false; } // Predicates. bool TrackElement::canSlacken () const { return false; } bool TrackElement::canPivotUp ( float ) const { return false; }; bool TrackElement::canPivotDown ( float ) const { return false; }; bool TrackElement::canMoveUp ( float, unsigned int ) const { return false; }; bool TrackElement::canDogleg () { return false; }; bool TrackElement::canDogleg ( Interval ) { return false; }; bool TrackElement::canDogleg ( Katabatic::GCell*, unsigned int ) { return false; }; // Accessors. unsigned long TrackElement::getId () const { return 0; } unsigned long TrackElement::getFreedomDegree () const { return 0; } DbU::Unit TrackElement::getPitch () const { return 0; } DbU::Unit TrackElement::getPPitch () const { return 0; } float TrackElement::getMaxUnderDensity ( unsigned int ) const { return 0.0; }; unsigned int TrackElement::getDoglegLevel () const { return 0; } TrackElement* TrackElement::getParent () const { return NULL; } Interval TrackElement::getSourceConstraints () const { return Interval(); } Interval TrackElement::getTargetConstraints () const { return Interval(); } DataNegociate* TrackElement::getDataNegociate ( unsigned int ) const { return NULL; } TrackElements TrackElement::getPerpandiculars () { return new TrackElements_Perpandiculars(NULL); } void TrackElement::invalidate () { } TrackElement* TrackElement::getCanonical ( Interval& i ) { i=Interval(getSourceU(),getTargetU()); return this; } TrackElement* TrackElement::getSourceDogleg () { return NULL; } TrackElement* TrackElement::getTargetDogleg () { return NULL; } // Mutators. void TrackElement::setTrack ( Track* track ) { _track = track; } void TrackElement::updateFreedomDegree () { } void TrackElement::setDoglegLevel ( unsigned int ) { } void TrackElement::swapTrack ( TrackElement* ) { } void TrackElement::reschedule ( unsigned int ) { } void TrackElement::detach () { } void TrackElement::revalidate () { } void TrackElement::updatePPitch () { } void TrackElement::setAxis ( DbU::Unit, unsigned int flags ) { } TrackElement* TrackElement::makeDogleg () { return NULL; } TrackElement* TrackElement::makeDogleg ( Interval, unsigned int& ) { return NULL; } TrackElement* TrackElement::makeDogleg ( Katabatic::GCell*, TrackElement*&, TrackElement*& ) { return NULL; } void TrackElement::_postDoglegs ( TrackElement*&, TrackElement*& ) { } bool TrackElement::moveAside ( unsigned int ) { return false; } bool TrackElement::slacken ( unsigned int ) { return false; } bool TrackElement::moveUp ( unsigned int ) { return false; } bool TrackElement::moveDown ( unsigned int ) { return false; } #if THIS_IS_DISABLED void TrackElement::desalignate () { } #endif bool TrackElement::_check () const { return true; } TrackElement::TrackElement ( Track* track ) : _flags (0) , _track (track) , _index ((size_t)-1) , _sourceU (0) , _targetU (0) , _observer(this) { } void TrackElement::_postCreate () { } TrackElement::~TrackElement () { } void TrackElement::_preDestroy () { } void TrackElement::destroy () { _preDestroy (); delete this; } TrackElement* TrackElement::getNext () const { size_t dummy = _index; return _track->getNext( dummy, getNet() ); } TrackElement* TrackElement::getPrevious () const { size_t dummy = _index; return _track->getPrevious( dummy, getNet() ); } Interval TrackElement::getFreeInterval () const { if (not _track) return Interval(false); size_t begin = _index; size_t end = _index; return _track->expandFreeInterval( begin, end, Track::InsideElement, getNet() ); } size_t TrackElement::getGCells ( Katabatic::GCellVector& gcells ) const { vector().swap( gcells ); return gcells.size(); } void TrackElement::incOverlapCost ( Net* net, TrackCost& cost ) const { if (not _track or (getNet() == net)) return; _overlapCostCallback( this, cost ); } string TrackElement::_getTypeName () const { return "TrackElement"; } string TrackElement::_getString () const { return "<"+_getTypeName()+">"; } Record* TrackElement::_getRecord () const { Record* record = new Record( _getString() ); record->add( getSlot( "_flags", _track ) ); record->add( getSlot( "_track", _track ) ); record->add( getSlot( "_index", _index ) ); record->add( DbU::getValueSlot( "_sourceU", &_sourceU ) ); record->add( DbU::getValueSlot( "_targetU", &_targetU ) ); return record; } } // Kite namespace.