// -*- C++ -*- // // This file is part of the Coriolis Software. // Copyright (c) UPMC 2012-2016, All Rights Reserved // // +-----------------------------------------------------------------+ // | C O R I O L I S | // | A n a b a t i c - Routing Toolbox | // | | // | Author : Jean-Paul CHAPUT | // | E-mail : Jean-Paul.Chaput@lip6.fr | // | =============================================================== | // | C++ Module : "./AutoContactTerminal.cpp" | // +-----------------------------------------------------------------+ #include #include #include #include "hurricane/Bug.h" #include "hurricane/Error.h" #include "hurricane/Warning.h" #include "hurricane/Layer.h" #include "hurricane/ViaLayer.h" #include "hurricane/BasicLayer.h" #include "hurricane/Technology.h" #include "hurricane/Net.h" #include "hurricane/Plug.h" #include "hurricane/RoutingPad.h" #include "hurricane/Vertical.h" #include "hurricane/Horizontal.h" #include "hurricane/DebugSession.h" #include "crlcore/RoutingGauge.h" #include "anabatic/AutoContactTerminal.h" #include "anabatic/AutoContactTurn.h" #include "anabatic/AutoVertical.h" #include "anabatic/AutoHorizontal.h" #include "anabatic/Session.h" namespace Anabatic { using std::ostringstream; using Hurricane::Bug; using Hurricane::Error; using Hurricane::DebugSession; using Hurricane::Transformation; using Hurricane::Entity; // ------------------------------------------------------------------- // Class : "Anabatic::AutoContactTerminal". AutoContactTerminal* AutoContactTerminal::create ( GCell* gcell , Component* anchor , const Layer* layer , Point point , DbU::Unit width , DbU::Unit height ) { cdebug_log(145,1) << "AutoContactTerminal::create(... Point, ...)" << endl; cdebug_log(145,0) << "@" << point << endl; anchor->getBodyHook()->detach(); AutoContactTerminal* autoContact = AutoContactTerminal::create( gcell , anchor , layer , point.getX(), point.getY() , width, height ); cdebug_tabw(145,-1); return autoContact; } AutoContactTerminal* AutoContactTerminal::create ( GCell* gcell , Component* anchor , const Layer* layer , const DbU::Unit x , const DbU::Unit y , const DbU::Unit width , const DbU::Unit height ) { cdebug_log(145,0) << "AutoContactTerminal::create(... x, y, ...)" << endl; cdebug_log(145,0) << "@ x:" << DbU::getValueString(x) << " y:" << DbU::getValueString(y) << endl; _preCreate( gcell, anchor->getNet(), layer ); Point anchorPosition = anchor->getPosition(); Contact* contact = Contact::create( anchor , layer , x - anchorPosition.getX() , y - anchorPosition.getY() , width , height ); AutoContactTerminal* autoContact = new AutoContactTerminal( gcell, contact ); autoContact->_postCreate(); autoContact->unsetFlags( CntInCreationStage ); cdebug_log(145,0) << "create(Component*) " << autoContact << endl; return autoContact; } AutoContactTerminal::AutoContactTerminal ( GCell* gcell, Contact* contact ) : AutoContact(gcell,contact) , _segment (NULL) { setFlags( CntTerminal ); } AutoContactTerminal::~AutoContactTerminal () { } bool AutoContactTerminal::isEndPoint () const { RoutingPad* rp = dynamic_cast( getAnchor() ); return (rp->getBodyHook()->getSlaveHooks().getSize() == 1); } AutoSegment* AutoContactTerminal::getOpposite ( const AutoSegment* ) const { return NULL; } AutoSegment* AutoContactTerminal::getPerpandicular ( const AutoSegment* ) const { return NULL; } AutoSegment* AutoContactTerminal::getSegment ( unsigned int index ) const { if (_segment) { switch ( index ) { case 0: return (_segment->isHorizontal()) ? _segment : NULL; case 2: return (_segment->isVertical ()) ? _segment : NULL; } } return NULL; } RoutingPad* AutoContactTerminal::getRoutingPad () const { return dynamic_cast(getAnchor()); } AutoSegments AutoContactTerminal::getRpConnecteds () const { return AutoSegments_OnRoutingPad(this); } Box AutoContactTerminal::getNativeConstraintBox () const { cdebug_log(145,1) << "AutoContactTerminal::getNativeConstraintBox()" << endl; if (isUserNativeConstraints()) { cdebug_log(145,1) << " Native constraints sets by user:" << getConstraintBox() << endl; cdebug_tabw(145,-1); cdebug_tabw(145,-1); return getConstraintBox(); } Component* component = getAnchor(); if (component == NULL) { cerr << Error( "%s is not anchored.", getString(this).c_str() ) << endl; cdebug_tabw(145,-1); return _gcell->getBoundingBox (); } DbU::Unit xMin; DbU::Unit xMax; DbU::Unit yMin; DbU::Unit yMax; Vertical* vertical; Horizontal* horizontal; RoutingPad* routingPad; if ( (horizontal = dynamic_cast(component)) ) { cdebug_log(145,0) << "Anchor: " << horizontal << "@" << horizontal->getSourcePosition() << endl; xMin = horizontal->getSourcePosition().getX(); xMax = horizontal->getTargetPosition().getX(); yMin = yMax = horizontal->getTargetPosition().getY(); } else if ( (vertical = dynamic_cast(component)) ) { cdebug_log(145,0) << "Anchor: " << vertical << "@" << vertical->getSourcePosition() << endl; yMin = vertical->getSourcePosition().getY(); yMax = vertical->getTargetPosition().getY(); xMin = xMax = vertical->getTargetPosition().getX(); } else if ( (routingPad = dynamic_cast(component)) ) { Entity* entity = routingPad->getOccurrence().getEntity(); Transformation transf = routingPad->getOccurrence().getPath().getTransformation(); cdebug_log(145,0) << "Anchor: " << routingPad << endl; int rpOrient = 1; switch ( transf.getOrientation() ) { case Transformation::Orientation::R1: case Transformation::Orientation::R3: case Transformation::Orientation::XR: case Transformation::Orientation::YR: rpOrient = 2; break; default: break; } if (dynamic_cast(entity)) { // rpOrient *is* the rotation. } else if ( dynamic_cast(entity) ) { // rpOrient is the inverse rotation. rpOrient = (rpOrient == 1) ? 2 : 1; } else { rpOrient = 0; } switch ( rpOrient ) { case 1: xMin = routingPad->getSourcePosition().getX(); xMax = routingPad->getTargetPosition().getX(); yMin = yMax = routingPad->getTargetPosition().getY(); break; case 2: yMin = routingPad->getSourcePosition().getY(); yMax = routingPad->getTargetPosition().getY(); xMin = xMax = routingPad->getTargetPosition().getX(); break; default: xMin = xMax = routingPad->getPosition().getX(); yMin = yMax = routingPad->getPosition().getY(); break; } } else { xMin = xMax = component->getPosition().getX(); yMin = yMax = component->getPosition().getY(); } order( xMin, xMax ); order( yMin, yMax ); Box bb ( xMin, yMin, xMax, yMax ); if (_segment and _segment->isWide()) { if (dynamic_cast(_segment)) bb.inflate( 0, 0, 0, -_segment->getWidth() ); else bb.inflate( 0, 0, -_segment->getWidth(), 0 ); } cdebug_log(145,0) << "| Using (y): " << DbU::getValueString(bb.getYMin()) << " " << DbU::getValueString(bb.getYMax()) << endl; cdebug_tabw(145,-1); return bb; } void AutoContactTerminal::_invalidate ( Flags flags ) { if (_segment) _segment->invalidate(); } void AutoContactTerminal::cacheDetach ( AutoSegment* segment ) { if (_segment == segment) { _segment = NULL; setFlags( CntInvalidatedCache ); } } void AutoContactTerminal::cacheAttach ( AutoSegment* segment ) { if (_segment) { cerr << Bug( "%s::cacheAttach() On %s,\n" " cache has not been cleared first, cancelled." , _getTypeName().c_str(), getString(this).c_str() ) << endl; return; } _segment = segment; unsetFlags( CntInvalidatedCache ); } void AutoContactTerminal::updateCache () { DebugSession::open( getNet(), 140, 150 ); cdebug_log(145,1) << _getTypeName() << "::updateCache() " << this << endl; Component* anchor; Horizontal** horizontals = new Horizontal* [2]; Vertical** verticals = new Vertical* [2]; _getTopology( base(), anchor, horizontals, verticals, 2 ); if (anchor == NULL) showTopologyError( "Terminal is missing an anchor (RoutingPad or Component)." ); size_t count = 0; if (horizontals[0] != NULL) ++count; if (horizontals[1] != NULL) ++count; if (verticals [0] != NULL) ++count; if (verticals [1] != NULL) ++count; if (count > 1) { showTopologyError( "Terminal has more than one segment." ); } if (horizontals[0] != NULL ) { _segment = Session::lookup( horizontals[0] ); } else { _segment = Session::lookup( verticals[0] ); } if (_segment == NULL) { ostringstream os; os << this << ", AutoSegment lookup failed for:" << "\n h1: " << horizontals[0] << "\n v1: " << verticals[0]; delete [] horizontals; delete [] verticals; showTopologyError( os.str() ); throw Error( os.str() ); } unsetFlags( CntInvalidatedCache ); cdebug_log(145,0) << "seg:" << _segment << endl; delete [] horizontals; delete [] verticals; cdebug_tabw(145,-1); DebugSession::close(); } void AutoContactTerminal::updateGeometry () { DebugSession::open( getNet(), 140, 150 ); cdebug_log(145,1) << _getTypeName() << "::updateGeometry() " << this << endl; if (isInvalidatedCache()) updateCache(); if (isInvalidatedCache()) { cerr << Error( "%s::updateGeometry() %s: Unable to restore cache." , _getTypeName().c_str(), getString(this).c_str() ) << endl; cdebug_tabw(145,-1); return; } base()->invalidate( false ); unsetFlags( CntInvalidated ); ostringstream message; if (not hasBadTopology()) { Box anchorBb = getAnchor()->getBoundingBox(); anchorBb.inflate( Session::getViaWidth (getAnchor()->getLayer()) - Session::getWireWidth(getAnchor()->getLayer()) ); if (_segment->isHorizontal()) { DbU::Unit axis = _segment->getY(); if (_segment->isWide()) { axis += (- _segment->getWidth() + Session::getWireWidth(_segment->getLayer())) / 2; setHeight( _segment->getContactWidth() ); } if (not getUConstraints(Flags::Vertical).contains(axis)) { cdebug_log(145,0) << "Cached: " << _segment << endl; message << "Terminal horizontal segment Y " << DbU::getValueString(axis) << " axis is outside RoutingPad " << getUConstraints(Flags::Vertical) << "."; Interval intv; _segment->getConstraints( intv ); message << "\n Segment constraints: " << intv << endl; Flags flags = Flags::NoFlags; if (_segment->isCreated()) flags |= Flags::CParanoid; showTopologyError( message.str(), flags ); } else setY( _segment->getY() ); } else { DbU::Unit axis = _segment->getX(); if (_segment->isWide()) { axis += (- _segment->getWidth() + Session::getWireWidth(_segment->getLayer())) / 2; setWidth ( _segment->getContactWidth() ); setHeight( anchorBb.getHeight() ); cdebug_log(145,0) << "Contact for wide segment." << endl; } if (not getUConstraints(Flags::Horizontal).contains(axis)) { cdebug_log(145,0) << "Cached: " << _segment << endl; message << "Terminal vertical segment X" << DbU::getValueString(axis) << " axis is outside RoutingPad " << getUConstraints(Flags::Horizontal) << "."; Flags flags = Flags::NoFlags; if (_segment->isCreated()) flags |= Flags::CParanoid; showTopologyError( message.str(), flags ); } else setX( _segment->getX() ); } } cdebug_tabw(145,-1); DebugSession::close(); } void AutoContactTerminal::updateTopology () { DebugSession::open( getNet(), 140, 150 ); cdebug_log(145,1) << _getTypeName() << "::updateTopology() " << this << endl; if (isInvalidatedCache()) updateCache(); if (isInvalidatedCache()) { cerr << Error( "%s::updateGeometry() %s: Unable to restore cache." , _getTypeName().c_str(), getString(this).c_str() ) << endl; cdebug_tabw(145,-1); return; } RoutingGauge* rg = Session::getRoutingGauge(); size_t anchorDepth = rg->getLayerDepth( (_flags & CntIgnoreAnchor) ? getLayer() : getAnchor()->getLayer() ); size_t segmentDepth = rg->getLayerDepth( _segment->getLayer() ); size_t delta = abssub( anchorDepth, segmentDepth ); if (delta > 3) { showTopologyError( "Sheared Terminal, layer delta exceed 3." ); setFlags( CntBadTopology ); } else { if (delta > 1) { //_segment = _segment->makeDogleg( this ); _segment->makeDogleg( this ); cdebug_log(145,0) << "Update seg: " << _segment << endl; delta = abssub( anchorDepth, rg->getLayerDepth( _segment->getLayer() ) ); } else if (delta == 0) setLayer( rg->getRoutingLayer(anchorDepth) ); else if (delta == 1) setLayer( rg->getContactLayer(std::min(anchorDepth,segmentDepth)) ); } _segment->invalidate( this ); cdebug_tabw(145,-1); DebugSession::close(); } void AutoContactTerminal::forceOnGrid ( Point gridPoint ) { setFlags( CntUserNativeConstraints ); setConstraintBox( Box(gridPoint) ); } string AutoContactTerminal::_getTypeName () const { return "ContactTerminal"; } } // Anabatic namespace.