2010-03-09 09:24:29 -06:00
|
|
|
// -*- C++ -*-
|
|
|
|
//
|
|
|
|
// This file is part of the Coriolis Software.
|
Corrections in how to access pads terminals in Katabatic/Kite.
* Change: In Katabatic, in GCellTopology::doRp_AccessPad(), if the supporting
RoutingPad is big (more than two pitch), do not put the access
contact in the center but on the edge. This is to avoid cut violations
between the VIAs and the matrix of VIAs that may be generated under the
RoutingPad itself.
* Change: In Kite, in TrackSegment destructor, if the legnth of the wire,
without extensions is less than one picth, enlarge it so it encompass
it's source & target VIAs (to avoid notches).
* Bug: In Kite, in Track destructor, the TrackElements where detacheds,
but not deleted, causing a memory link.
2016-03-02 10:15:58 -06:00
|
|
|
// Copyright (c) UPMC 2008-2016, All Rights Reserved
|
2010-03-09 09:24:29 -06:00
|
|
|
//
|
2011-01-25 11:16:27 -06:00
|
|
|
// +-----------------------------------------------------------------+
|
2010-03-09 09:24:29 -06:00
|
|
|
// | C O R I O L I S |
|
|
|
|
// | K a t a b a t i c - Routing Toolbox |
|
|
|
|
// | |
|
|
|
|
// | Author : Jean-Paul CHAPUT |
|
2013-12-03 18:58:58 -06:00
|
|
|
// | E-mail : Jean-Paul.Chaput@lip6.fr |
|
2010-03-09 09:24:29 -06:00
|
|
|
// | =============================================================== |
|
|
|
|
// | C++ Module : "./LoadGrByNet.cpp" |
|
2011-01-25 11:16:27 -06:00
|
|
|
// +-----------------------------------------------------------------+
|
2010-03-09 09:24:29 -06:00
|
|
|
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
#include <cstdlib>
|
|
|
|
#include <sstream>
|
|
|
|
#include "hurricane/Bug.h"
|
|
|
|
#include "hurricane/Error.h"
|
|
|
|
#include "hurricane/Warning.h"
|
|
|
|
#include "hurricane/DebugSession.h"
|
|
|
|
#include "hurricane/Layer.h"
|
Added support for "same layer" dogleg. Big fix for pad routing.
* Change: In Knik, in Vertex, add a "blocked" flag to signal disabled
vertexes in the grid (must not be used by the global router).
Modificate the Graph::getVertex() method so that when a vertex
is geometrically queried, if is a blocked one, return a non-blocked
neighbor. This mechanism is introduced to, at last, prevent the
global router to go *under* the pad in case of a commplete chip.
* New: In Katabatic, in AutoSegment, a new state has been added: "reduced".
A reduced segment is in the same layer as it's perpandiculars.
To be reduced, a segments has to be connected on source & target to
AutoContactTurn, both of the perpandiculars must be of the same layer
(below or above) and it's length must not exceed one pitch in the
perpandicular direction.
To reduce an AutoSegment, call ::reduce() and to revert the state,
call ::raise(). Two associated predicates are associated:
::canReduce() and ::mustRaise().
Note: No two adjacent segments can be reduced at the same time.
* Bug: In Katabatic, in GCellTopology, add a new method ::doRp_AccessPad()
to connect to the pads. Create wiring, fixed and non managed by
Katabatic, to connect the pad connector layer to the lowest routing
layers (depth 1 & 2). The former implementation was sometimes leading
to gaps (sheared contact) that *must not* occurs during the building
stage.
Remark: This bug did put under the light the fact that the initial
wiring must be created without gaps. Gaps are closed by making doglegs
on contacts. But this mechanism could only work when the database if
fully initialised (the cache is up to date). Otherwise various problems
arise, in the canonization process for example.
* New: In Katabatic, in AutoContactTerminal::getNativeConstraintBox(),
when anchored on a RoutingPad, now take account the potential rotation
of the Path's transformation. Here again, for the chip's pads.
* New: In Kite, support for reduced AutoSegment. TrackSegment associateds
to reduced AutoSegment are *not* inserted into track to become
effectively invisibles. When a segment becomes reduced, a TrackEvent
is generated to remove it. Conversely when it is raised a RoutingEvent
is created/rescheduled to insert it. All this is mostly managed inside
the Session::revalidate() method.
* New: In Kite, in KiteEngine::createGlobalGraph(), in case of a chip,
mark all global routing vertexes (Knik) that are under a pad, as blockeds.
* Bug: In Cumulus, in PadsCorona.Side.getAxis(), inversion between X and
Y coordinate of the chip size. Did not show until a non-square chip
was routed (i.e. our MIPS R3000).
* Change: In Stratus1, in st_placement.py add the ClockBuffer class for
backward compatibility with the MIPS32 bench. Have to review this
functionnality coming from the deprecated placeAndroute.py.
In st_instance.py, no longer creates the Plug ring of a Net.
In my opinion it just clutter the display until the P&R is called.
Can re-enable later as an option (in Unicorn).
* Change: In Unicorn, in cgt.py, more reliable way of loading then running
user supplied scripts. Borrowed from alliance-checker-toolkit doChip.py .
2015-08-16 16:29:28 -05:00
|
|
|
#include "hurricane/BasicLayer.h"
|
|
|
|
#include "hurricane/RegularLayer.h"
|
2013-12-03 18:58:58 -06:00
|
|
|
#include "hurricane/Technology.h"
|
|
|
|
#include "hurricane/DataBase.h"
|
|
|
|
#include "hurricane/Net.h"
|
|
|
|
#include "hurricane/NetExternalComponents.h"
|
|
|
|
#include "hurricane/RoutingPad.h"
|
|
|
|
#include "hurricane/RoutingPads.h"
|
|
|
|
#include "hurricane/Pad.h"
|
|
|
|
#include "hurricane/Plug.h"
|
|
|
|
#include "hurricane/Cell.h"
|
|
|
|
#include "hurricane/Instance.h"
|
|
|
|
#include "hurricane/Vertical.h"
|
|
|
|
#include "hurricane/Horizontal.h"
|
|
|
|
#include "crlcore/AllianceFramework.h"
|
|
|
|
#include "crlcore/RoutingGauge.h"
|
|
|
|
#include "crlcore/Measures.h"
|
|
|
|
#include "katabatic/AutoContactTerminal.h"
|
|
|
|
#include "katabatic/AutoContactTurn.h"
|
|
|
|
#include "katabatic/AutoContactHTee.h"
|
|
|
|
#include "katabatic/AutoContactVTee.h"
|
|
|
|
#include "katabatic/AutoSegment.h"
|
|
|
|
#include "katabatic/GCellGrid.h"
|
|
|
|
#include "katabatic/KatabaticEngine.h"
|
2010-03-09 09:24:29 -06:00
|
|
|
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
Added support for "same layer" dogleg. Big fix for pad routing.
* Change: In Knik, in Vertex, add a "blocked" flag to signal disabled
vertexes in the grid (must not be used by the global router).
Modificate the Graph::getVertex() method so that when a vertex
is geometrically queried, if is a blocked one, return a non-blocked
neighbor. This mechanism is introduced to, at last, prevent the
global router to go *under* the pad in case of a commplete chip.
* New: In Katabatic, in AutoSegment, a new state has been added: "reduced".
A reduced segment is in the same layer as it's perpandiculars.
To be reduced, a segments has to be connected on source & target to
AutoContactTurn, both of the perpandiculars must be of the same layer
(below or above) and it's length must not exceed one pitch in the
perpandicular direction.
To reduce an AutoSegment, call ::reduce() and to revert the state,
call ::raise(). Two associated predicates are associated:
::canReduce() and ::mustRaise().
Note: No two adjacent segments can be reduced at the same time.
* Bug: In Katabatic, in GCellTopology, add a new method ::doRp_AccessPad()
to connect to the pads. Create wiring, fixed and non managed by
Katabatic, to connect the pad connector layer to the lowest routing
layers (depth 1 & 2). The former implementation was sometimes leading
to gaps (sheared contact) that *must not* occurs during the building
stage.
Remark: This bug did put under the light the fact that the initial
wiring must be created without gaps. Gaps are closed by making doglegs
on contacts. But this mechanism could only work when the database if
fully initialised (the cache is up to date). Otherwise various problems
arise, in the canonization process for example.
* New: In Katabatic, in AutoContactTerminal::getNativeConstraintBox(),
when anchored on a RoutingPad, now take account the potential rotation
of the Path's transformation. Here again, for the chip's pads.
* New: In Kite, support for reduced AutoSegment. TrackSegment associateds
to reduced AutoSegment are *not* inserted into track to become
effectively invisibles. When a segment becomes reduced, a TrackEvent
is generated to remove it. Conversely when it is raised a RoutingEvent
is created/rescheduled to insert it. All this is mostly managed inside
the Session::revalidate() method.
* New: In Kite, in KiteEngine::createGlobalGraph(), in case of a chip,
mark all global routing vertexes (Knik) that are under a pad, as blockeds.
* Bug: In Cumulus, in PadsCorona.Side.getAxis(), inversion between X and
Y coordinate of the chip size. Did not show until a non-square chip
was routed (i.e. our MIPS R3000).
* Change: In Stratus1, in st_placement.py add the ClockBuffer class for
backward compatibility with the MIPS32 bench. Have to review this
functionnality coming from the deprecated placeAndroute.py.
In st_instance.py, no longer creates the Plug ring of a Net.
In my opinion it just clutter the display until the P&R is called.
Can re-enable later as an option (in Unicorn).
* Change: In Unicorn, in cgt.py, more reliable way of loading then running
user supplied scripts. Borrowed from alliance-checker-toolkit doChip.py .
2015-08-16 16:29:28 -05:00
|
|
|
using Katabatic::AutoContactTerminal;
|
|
|
|
|
|
|
|
|
2014-05-13 09:30:41 -05:00
|
|
|
/*! \defgroup LoadGlobalRouting Global Routing Loading
|
|
|
|
* \brief Translation rules to build detailed routing from global
|
2013-12-03 18:58:58 -06:00
|
|
|
*
|
|
|
|
* This module documents how the global routing built by \c Knik is
|
|
|
|
* loaded into the \c Katabatic data-base. It is intented for developpers
|
|
|
|
* only.
|
|
|
|
*/
|
|
|
|
|
|
|
|
//! \addtogroup LoadGlobalRouting
|
|
|
|
//! \{
|
|
|
|
|
|
|
|
//! \enum LocalFunctionFlag
|
|
|
|
//! A set of flags for all functions of the LoadGrByNet module.
|
|
|
|
//! They can be combined to form the \e flags argument of functions.
|
|
|
|
//! the functions will ignore flags that are not intended to them.
|
|
|
|
//!
|
|
|
|
//! For \c HSmall, \c VSmall & \c Punctual see checkRoutingPadSize().
|
|
|
|
|
|
|
|
//! \var LocalFunctionFlag::NoFlags
|
|
|
|
//! A simple alias over zero to explicitly tell that no flag at all is
|
|
|
|
//! passed to the function.
|
|
|
|
|
|
|
|
//! \var LocalFunctionFlag::HAccess
|
|
|
|
//! The constructed topology will be accessed through an horizontal
|
|
|
|
//! segment. The absence of this flag tell that the access will be done
|
|
|
|
//! trough a vertical.
|
|
|
|
|
|
|
|
//! \var LocalFunctionFlag::VSmall
|
|
|
|
//! The RoutingPad vertically covers a very small number of access points,
|
|
|
|
//! so it is likely overconstrained for direct horizontal connexion.
|
|
|
|
|
|
|
|
//! \var LocalFunctionFlag::HSmall
|
|
|
|
//! The RoutingPad horizontally covers a very small number of access points,
|
|
|
|
//! so it is likely overconstrained for direct vertical connexion.
|
|
|
|
|
|
|
|
//! \var LocalFunctionFlag::Punctual
|
|
|
|
//! The RoutingPad covers only an access point in either direction.
|
|
|
|
|
|
|
|
//! \var LocalFunctionFlag::DoSourceContact
|
|
|
|
//! When creating Katabatic::AutoContactTerminal on non-punctual RoutingPad, this flag
|
|
|
|
//! request the creation of a contact <em>on the source point</em>.
|
|
|
|
|
|
|
|
//! \var LocalFunctionFlag::DoTargetContact
|
|
|
|
//! When creating Katabatic::AutoContactTerminal on non-punctual RoutingPad, this flag
|
|
|
|
//! request the creation of a contact <em>on the target point</em>.
|
|
|
|
|
|
|
|
|
|
|
|
//! \function unsigned int checkRoutingPadSize ( Component* rp );
|
|
|
|
//!
|
|
|
|
//! Look at the geometrical size of the Component and assess if
|
|
|
|
//! it's span is too narrow either horizontally or vertically.
|
|
|
|
//! Return a combination of flags indicating it's state:
|
|
|
|
//! - HSmall : less than 3 pitches in horizontal direction.
|
|
|
|
//! - VSmall : less than 3 pitches in vertical direction.
|
|
|
|
//! - Punctual : one pitch in either directions.
|
|
|
|
//!
|
|
|
|
//! The component can be a RoutingPad, a Vertical or an Horizontal.
|
|
|
|
//!
|
|
|
|
//! \image html checkRoutingPadSize.png "checkRoutingPadSize()"
|
|
|
|
|
|
|
|
/*! \class GCellTopology
|
|
|
|
*
|
|
|
|
* \brief Build the wiring for a Net inside a GCell (\b internal).
|
|
|
|
*
|
Added support for "same layer" dogleg. Big fix for pad routing.
* Change: In Knik, in Vertex, add a "blocked" flag to signal disabled
vertexes in the grid (must not be used by the global router).
Modificate the Graph::getVertex() method so that when a vertex
is geometrically queried, if is a blocked one, return a non-blocked
neighbor. This mechanism is introduced to, at last, prevent the
global router to go *under* the pad in case of a commplete chip.
* New: In Katabatic, in AutoSegment, a new state has been added: "reduced".
A reduced segment is in the same layer as it's perpandiculars.
To be reduced, a segments has to be connected on source & target to
AutoContactTurn, both of the perpandiculars must be of the same layer
(below or above) and it's length must not exceed one pitch in the
perpandicular direction.
To reduce an AutoSegment, call ::reduce() and to revert the state,
call ::raise(). Two associated predicates are associated:
::canReduce() and ::mustRaise().
Note: No two adjacent segments can be reduced at the same time.
* Bug: In Katabatic, in GCellTopology, add a new method ::doRp_AccessPad()
to connect to the pads. Create wiring, fixed and non managed by
Katabatic, to connect the pad connector layer to the lowest routing
layers (depth 1 & 2). The former implementation was sometimes leading
to gaps (sheared contact) that *must not* occurs during the building
stage.
Remark: This bug did put under the light the fact that the initial
wiring must be created without gaps. Gaps are closed by making doglegs
on contacts. But this mechanism could only work when the database if
fully initialised (the cache is up to date). Otherwise various problems
arise, in the canonization process for example.
* New: In Katabatic, in AutoContactTerminal::getNativeConstraintBox(),
when anchored on a RoutingPad, now take account the potential rotation
of the Path's transformation. Here again, for the chip's pads.
* New: In Kite, support for reduced AutoSegment. TrackSegment associateds
to reduced AutoSegment are *not* inserted into track to become
effectively invisibles. When a segment becomes reduced, a TrackEvent
is generated to remove it. Conversely when it is raised a RoutingEvent
is created/rescheduled to insert it. All this is mostly managed inside
the Session::revalidate() method.
* New: In Kite, in KiteEngine::createGlobalGraph(), in case of a chip,
mark all global routing vertexes (Knik) that are under a pad, as blockeds.
* Bug: In Cumulus, in PadsCorona.Side.getAxis(), inversion between X and
Y coordinate of the chip size. Did not show until a non-square chip
was routed (i.e. our MIPS R3000).
* Change: In Stratus1, in st_placement.py add the ClockBuffer class for
backward compatibility with the MIPS32 bench. Have to review this
functionnality coming from the deprecated placeAndroute.py.
In st_instance.py, no longer creates the Plug ring of a Net.
In my opinion it just clutter the display until the P&R is called.
Can re-enable later as an option (in Unicorn).
* Change: In Unicorn, in cgt.py, more reliable way of loading then running
user supplied scripts. Borrowed from alliance-checker-toolkit doChip.py .
2015-08-16 16:29:28 -05:00
|
|
|
* As this class is called to initially construct the Katabatic wiring,
|
|
|
|
* it must build a \b connex wiring. That is without gaps in layer depth,
|
|
|
|
* because the topology restauration mechanism (AutoContact::updateTopology())
|
|
|
|
* of the AutoContact cannot work until all AutoSegments are revalidated at
|
|
|
|
* least once. The topology restauration work by creating doglegs which in turn,
|
|
|
|
* call the canonization, which needs all the caches to be up to date.
|
2013-12-03 18:58:58 -06:00
|
|
|
*/
|
|
|
|
|
|
|
|
//! \function void GCellTopology::doRp_AutoContacts ( GCell* gcell, Component* rp, AutoContact*& source, AutoContact*& target, unsigned int flags );
|
|
|
|
//! \param gcell The GCell into which create the AutoContact.
|
|
|
|
//! \param rp The Component we want to access.
|
|
|
|
//! \param source The AutoContact created on the \c source (\e returned).
|
|
|
|
//! \param target The AutoContact created on the \c target (\e returned).
|
|
|
|
//! \param flags Managed by this function:
|
|
|
|
//! - LocalFunctionFlag::DoSourceContact
|
|
|
|
//! - LocalFunctionFlag::DoTargetContact
|
|
|
|
//!
|
|
|
|
//! Create the AutoContact directly anchored on the Component (terminal).
|
|
|
|
//! Three cases are manageds:
|
|
|
|
//! -# <b>Ordinary (non-punctual) \c METAL1 terminal</b>: an AutoContactTerminal
|
|
|
|
//! is anchored on the RoutingPad.
|
|
|
|
//! -# <b>Punctual \c METAL1 terminal</b>, the access must never be blocked
|
|
|
|
//! by other routing. To ensure it, we create a fixed AutoSegment (anchored
|
|
|
|
//! on two AutoContactTerminal) to cover it. The \e normal AutoContactTerminal
|
|
|
|
//! is also created.
|
|
|
|
//! -# <b>non \c METAL1 terminal</b>, as for the punctual \c METAL1, a
|
|
|
|
//! fixed protection is added over the RoutingPad. If we access
|
|
|
|
//! horizontally a vertical RoutingPad or vertically an horizontal
|
|
|
|
//! one, an extra AutoContactTerminal is added (to allow is displacement
|
|
|
|
//! along the RoutingPad).
|
|
|
|
//!
|
|
|
|
//! To avoid creating a fixed protection over a RoutingPad multiple times,
|
|
|
|
//! the RoutingPad and it's associated protection is stored in a static
|
|
|
|
//! \c map : \c __routingPadAutoSegments.
|
|
|
|
//!
|
|
|
|
//! Conversely, because an AutoContactTerminal can only be connected to one
|
|
|
|
//! segment, each time this function is called a new terminal will be created
|
|
|
|
//! (or maybe two in case of non-punctual terminals). If only one AutoContact
|
|
|
|
//! is requested, it is created centered on the RoutingPad. The initial
|
|
|
|
//! position of AutoContact <em>do not prevent them to move afterwards</em>,
|
|
|
|
//! even those created on source/target on a non-punctual RoutingPad.
|
|
|
|
//!
|
|
|
|
//! \remark For clarity we describe the layer management of this function in term
|
|
|
|
//! of \c METAL, but it is the RoutingGauge depth which is actually used.
|
|
|
|
//!
|
|
|
|
//! \image html doRp_AutoContacts.png "doRp_AutoContacts()"
|
|
|
|
|
|
|
|
//! \function AutoContact* GCellTopology::doRp_Access ( GCell* gcell, Component* rp, unsigned int flags );
|
|
|
|
//! \param gcell The GCell into which create the AutoContact.
|
|
|
|
//! \param rp The Component onto which anchor the access contact.
|
|
|
|
//! \param flags Relevant flags are:
|
|
|
|
//! - HAccess, the terminal is to be accessed through an horizontal
|
|
|
|
//! segment.
|
|
|
|
//! - VSmall, force the terminal to be considered as small in the
|
|
|
|
//! vertical direction.
|
|
|
|
//!
|
|
|
|
//! If \c HAccess is set, the Component is to be accessed trough an horizontal
|
|
|
|
//! segment. If unset, the access is done vertically.
|
|
|
|
//!
|
|
|
|
//! Create an AutoContact to access a Component (terminal). If the Component
|
|
|
|
//! is not to be accessed through an horizontal segment, and do not cover a
|
|
|
|
//! large span in the horizontal direction (flag \c VSmall), a local horizontal
|
|
|
|
//! AutoSegment is added to slacken the vertical constraints.
|
|
|
|
//!
|
|
|
|
//! \image html doRp_Access.png "doRp_Access()"
|
|
|
|
|
Corrections in how to access pads terminals in Katabatic/Kite.
* Change: In Katabatic, in GCellTopology::doRp_AccessPad(), if the supporting
RoutingPad is big (more than two pitch), do not put the access
contact in the center but on the edge. This is to avoid cut violations
between the VIAs and the matrix of VIAs that may be generated under the
RoutingPad itself.
* Change: In Kite, in TrackSegment destructor, if the legnth of the wire,
without extensions is less than one picth, enlarge it so it encompass
it's source & target VIAs (to avoid notches).
* Bug: In Kite, in Track destructor, the TrackElements where detacheds,
but not deleted, causing a memory link.
2016-03-02 10:15:58 -06:00
|
|
|
//! \function AutoContact* GCellTopology::doRp_AccessPad ( RoutingPad* rp, unsigned int flags );
|
Added support for "same layer" dogleg. Big fix for pad routing.
* Change: In Knik, in Vertex, add a "blocked" flag to signal disabled
vertexes in the grid (must not be used by the global router).
Modificate the Graph::getVertex() method so that when a vertex
is geometrically queried, if is a blocked one, return a non-blocked
neighbor. This mechanism is introduced to, at last, prevent the
global router to go *under* the pad in case of a commplete chip.
* New: In Katabatic, in AutoSegment, a new state has been added: "reduced".
A reduced segment is in the same layer as it's perpandiculars.
To be reduced, a segments has to be connected on source & target to
AutoContactTurn, both of the perpandiculars must be of the same layer
(below or above) and it's length must not exceed one pitch in the
perpandicular direction.
To reduce an AutoSegment, call ::reduce() and to revert the state,
call ::raise(). Two associated predicates are associated:
::canReduce() and ::mustRaise().
Note: No two adjacent segments can be reduced at the same time.
* Bug: In Katabatic, in GCellTopology, add a new method ::doRp_AccessPad()
to connect to the pads. Create wiring, fixed and non managed by
Katabatic, to connect the pad connector layer to the lowest routing
layers (depth 1 & 2). The former implementation was sometimes leading
to gaps (sheared contact) that *must not* occurs during the building
stage.
Remark: This bug did put under the light the fact that the initial
wiring must be created without gaps. Gaps are closed by making doglegs
on contacts. But this mechanism could only work when the database if
fully initialised (the cache is up to date). Otherwise various problems
arise, in the canonization process for example.
* New: In Katabatic, in AutoContactTerminal::getNativeConstraintBox(),
when anchored on a RoutingPad, now take account the potential rotation
of the Path's transformation. Here again, for the chip's pads.
* New: In Kite, support for reduced AutoSegment. TrackSegment associateds
to reduced AutoSegment are *not* inserted into track to become
effectively invisibles. When a segment becomes reduced, a TrackEvent
is generated to remove it. Conversely when it is raised a RoutingEvent
is created/rescheduled to insert it. All this is mostly managed inside
the Session::revalidate() method.
* New: In Kite, in KiteEngine::createGlobalGraph(), in case of a chip,
mark all global routing vertexes (Knik) that are under a pad, as blockeds.
* Bug: In Cumulus, in PadsCorona.Side.getAxis(), inversion between X and
Y coordinate of the chip size. Did not show until a non-square chip
was routed (i.e. our MIPS R3000).
* Change: In Stratus1, in st_placement.py add the ClockBuffer class for
backward compatibility with the MIPS32 bench. Have to review this
functionnality coming from the deprecated placeAndroute.py.
In st_instance.py, no longer creates the Plug ring of a Net.
In my opinion it just clutter the display until the P&R is called.
Can re-enable later as an option (in Unicorn).
* Change: In Unicorn, in cgt.py, more reliable way of loading then running
user supplied scripts. Borrowed from alliance-checker-toolkit doChip.py .
2015-08-16 16:29:28 -05:00
|
|
|
//! \param rp The Component onto which anchor the access contact.
|
|
|
|
//! \param flags Relevant flags are:
|
|
|
|
//! - HAccess, the terminal is to be accessed through an horizontal
|
|
|
|
//! segment.
|
|
|
|
//! - VSmall, force the terminal to be considered as small in the
|
|
|
|
//! vertical direction.
|
|
|
|
//! \return A Katabatic::AutoContactTerminal .
|
|
|
|
//!
|
|
|
|
//! The Component \c rp is a RoutingPad which belongs to a pad cell. This case
|
|
|
|
//! occurs when we are routing a complete chip. This method build, from the
|
|
|
|
//! \c rp a stack of articulated punctual segments and contacts to reach the
|
|
|
|
//! default H/V routing layers (usually \c METAL2 & \c METAL3). This may be
|
|
|
|
//! needed when the pad terminal is in \c METAL5, for instance.
|
|
|
|
//!
|
|
|
|
//! The returned AutoContactTerminal is anchored on the last punctual segment
|
|
|
|
//! build.
|
|
|
|
//!
|
|
|
|
//! The GCell into which the AutoContactTerminal is created may be under the
|
|
|
|
//! pads area. However, it will be right on the border of the GCell.
|
|
|
|
//! The global router vertexes of GCell under the pad area are marked as
|
|
|
|
//! blocked so will never be used for routing.
|
|
|
|
//!
|
|
|
|
//! \remark The segments and contacts added to ensure the layer connexity are not
|
|
|
|
//! put into the Katabatic database. They are plain Hurricane objects, invisibles
|
|
|
|
//! from it.
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
//! \function void GCellTopology::doRp_StairCaseH ( GCell* gcell, Component* rp1, Component* rp2 );
|
|
|
|
//!
|
|
|
|
//! Build the wiring to connect to horizontal Component. Two cases:
|
|
|
|
//! - The Component are aligneds, then only a straight wire is created.
|
|
|
|
//! - They are \e not aligned, then a complete dogleg is created.
|
|
|
|
//!
|
|
|
|
//! \image html doRp_StairCaseH.png "doRp_StairCaseH()"
|
|
|
|
|
|
|
|
//! \function void GCellTopology::doRp_StairCaseV ( GCell* gcell, Component* rp1, Component* rp2 );
|
|
|
|
//!
|
|
|
|
//! Build the wiring to connect to vertical Components. Two cases:
|
|
|
|
//! - The Components are aligneds, then only a straight wire is created.
|
|
|
|
//! - They are \e not aligned, then a complete dogleg is created.
|
|
|
|
//!
|
|
|
|
//! \image html doRp_StairCaseV.png "doRp_StairCaseV()"
|
|
|
|
|
Added support for "same layer" dogleg. Big fix for pad routing.
* Change: In Knik, in Vertex, add a "blocked" flag to signal disabled
vertexes in the grid (must not be used by the global router).
Modificate the Graph::getVertex() method so that when a vertex
is geometrically queried, if is a blocked one, return a non-blocked
neighbor. This mechanism is introduced to, at last, prevent the
global router to go *under* the pad in case of a commplete chip.
* New: In Katabatic, in AutoSegment, a new state has been added: "reduced".
A reduced segment is in the same layer as it's perpandiculars.
To be reduced, a segments has to be connected on source & target to
AutoContactTurn, both of the perpandiculars must be of the same layer
(below or above) and it's length must not exceed one pitch in the
perpandicular direction.
To reduce an AutoSegment, call ::reduce() and to revert the state,
call ::raise(). Two associated predicates are associated:
::canReduce() and ::mustRaise().
Note: No two adjacent segments can be reduced at the same time.
* Bug: In Katabatic, in GCellTopology, add a new method ::doRp_AccessPad()
to connect to the pads. Create wiring, fixed and non managed by
Katabatic, to connect the pad connector layer to the lowest routing
layers (depth 1 & 2). The former implementation was sometimes leading
to gaps (sheared contact) that *must not* occurs during the building
stage.
Remark: This bug did put under the light the fact that the initial
wiring must be created without gaps. Gaps are closed by making doglegs
on contacts. But this mechanism could only work when the database if
fully initialised (the cache is up to date). Otherwise various problems
arise, in the canonization process for example.
* New: In Katabatic, in AutoContactTerminal::getNativeConstraintBox(),
when anchored on a RoutingPad, now take account the potential rotation
of the Path's transformation. Here again, for the chip's pads.
* New: In Kite, support for reduced AutoSegment. TrackSegment associateds
to reduced AutoSegment are *not* inserted into track to become
effectively invisibles. When a segment becomes reduced, a TrackEvent
is generated to remove it. Conversely when it is raised a RoutingEvent
is created/rescheduled to insert it. All this is mostly managed inside
the Session::revalidate() method.
* New: In Kite, in KiteEngine::createGlobalGraph(), in case of a chip,
mark all global routing vertexes (Knik) that are under a pad, as blockeds.
* Bug: In Cumulus, in PadsCorona.Side.getAxis(), inversion between X and
Y coordinate of the chip size. Did not show until a non-square chip
was routed (i.e. our MIPS R3000).
* Change: In Stratus1, in st_placement.py add the ClockBuffer class for
backward compatibility with the MIPS32 bench. Have to review this
functionnality coming from the deprecated placeAndroute.py.
In st_instance.py, no longer creates the Plug ring of a Net.
In my opinion it just clutter the display until the P&R is called.
Can re-enable later as an option (in Unicorn).
* Change: In Unicorn, in cgt.py, more reliable way of loading then running
user supplied scripts. Borrowed from alliance-checker-toolkit doChip.py .
2015-08-16 16:29:28 -05:00
|
|
|
//! \function void GCellTopology::_do_xG_1Pad ();
|
|
|
|
//!
|
|
|
|
//! Construct the topology, when there is only global wires and one local
|
|
|
|
//! terminal, but coming from a Pad. As thoses connectors will always be
|
|
|
|
//! on one border of the GCell they can be considered as a kind of global.
|
|
|
|
//!
|
|
|
|
//! So this method mostly calls GCellTopology::doRp_AccessPad() to create
|
|
|
|
//! the AutoContactTerminal, then calls GCellTopology::_do_xG(), except
|
|
|
|
//! for straight lines which are managed directly.
|
|
|
|
|
|
|
|
//! \function void GCellTopology::_do_xG ();
|
2013-12-03 18:58:58 -06:00
|
|
|
//!
|
|
|
|
//! Construct the topology, when there is only global wires (no local terminals).
|
|
|
|
//!
|
|
|
|
//! Some topology are not handled because they must not be managed by this
|
|
|
|
//! function:
|
|
|
|
//! <ul>
|
|
|
|
//! <li>One global: nonsensical because there also must be a terminal.
|
|
|
|
//! <li>Two aligned globals: in that case we do a straight wire whithout
|
|
|
|
//! any AutoContact (handled by the source/target of the wire).
|
|
|
|
//! </ul>
|
|
|
|
//!
|
|
|
|
//! \image html _do_xG.png "_do_xG()"
|
|
|
|
|
|
|
|
//! \function void GCellTopology::_do_1G_1M1 ();
|
|
|
|
//!
|
|
|
|
//! Construct a topology where there is \e one global and one RoutingPad
|
|
|
|
//! in \c METAL1. The \c METAL1 is assumed to be vertical.
|
|
|
|
//!
|
|
|
|
//! \remark When accessing the RoutingPad through an horizontal global segment
|
|
|
|
//! and the vertical extension of the segment is small, the global is
|
|
|
|
//! still directly attached to the terminal, inducing a high constraint
|
|
|
|
//! on it. We left to job of slackening it to the router.
|
|
|
|
//!
|
|
|
|
//! \image html _do_1G_1M1.png "_do_1G_1M1()"
|
|
|
|
|
|
|
|
//! \function void GCellTopology::_do_1G_xM1 ();
|
|
|
|
//!
|
|
|
|
//! Construct a topology where there is \e one global and any number of
|
|
|
|
//! RoutingPad in \c METAL1. The \c METAL1 is assumed to be vertical.
|
|
|
|
//!
|
|
|
|
//! The RoutingPads are linked together two by two. If the horizontal
|
|
|
|
//! segments are not aligned by the router, part of the routage will be
|
|
|
|
//! done through the RoutingPad itself. The global incoming segment will
|
|
|
|
//! connected to the leftmost, rightmost or centermost RoutingPad according
|
|
|
|
//! from wich side it comes from.
|
|
|
|
//!
|
|
|
|
//! \image html _do_1G_xM1.png "_do_1G_xM1()"
|
|
|
|
|
|
|
|
//! \function void GCellTopology::_do_xG_1M1_1M2 ();
|
|
|
|
//!
|
|
|
|
//! Construct a topology where there is at least one global (and up to 4),
|
|
|
|
//! one \c METAL1 RoutingPad (assumed V) and one \c METAL2 RoutingPad (assumed H).
|
|
|
|
//!
|
|
|
|
//! In this topology, we want to try to reuse the \c METAL2 RoutingPad as a
|
|
|
|
//! feedtrough in the horizontal routage. Thus:
|
|
|
|
//! - The \c METAL1 and \c METAL2 RoutingPad are connected through a separate wiring.
|
|
|
|
//! - The south & west global wiring is attached to the leftmost contact of
|
|
|
|
//! the \c METAL2.
|
|
|
|
//! - The north & east global wiring is attached to the rightmost contact of
|
|
|
|
//! the \c METAL2.
|
|
|
|
//!
|
|
|
|
//! South/west and north/south can be build independantly. Depending on the number
|
|
|
|
//! of globals, they can consist of:
|
|
|
|
//! - Nothing (no south nor west).
|
|
|
|
//! - An AutoContact (west present).
|
|
|
|
//! - An horizontal plus a turn (south present).
|
|
|
|
//! - An horizontal plus a HTee (south & west present).
|
|
|
|
//!
|
|
|
|
//! \remark Not all configurations are represented below.
|
|
|
|
//!
|
|
|
|
//! \image html _do_xG_1M1_1M2.png "_do_xG_1M1_1M2()"
|
|
|
|
|
|
|
|
//! \function void GCellTopology::_do_xG_xM1_xM3 ();
|
|
|
|
//!
|
|
|
|
//! Construct a topology where there is at least one global (and up to 4),
|
|
|
|
//! at least one \c METAL1 RoutingPad (assumed V) and at least one \c METAL3
|
|
|
|
//! RoutingPad (assumed V).
|
|
|
|
//!
|
|
|
|
//! In this topology, we want to try to reuse the \c METAL3 RoutingPad as a
|
|
|
|
//! feedtrough in the vertical routage. Thus:
|
|
|
|
//! - The \c METAL1 and \c METAL3 RoutingPad are connected through a separate
|
|
|
|
//! wiring made of separate horizontals.
|
|
|
|
//! - The south-west global wiring is attached to the leftmost RoutingPad
|
|
|
|
//! if there isn't south or to the first \c METAL3 otherwise.
|
|
|
|
//! - The north-east global wiring is attached to the rightmost RoutingPad
|
|
|
|
//! if there isn't north or to the first \c METAL3 otherwise.
|
|
|
|
//!
|
|
|
|
//! South/west and north/south can be build independantly. Depending on the number
|
|
|
|
//! of globals, they can consist of:
|
|
|
|
//! - Nothing (no south nor west).
|
|
|
|
//! - An AutoContact on the leftmost RoutingPad (west present).
|
|
|
|
//! - An AutoContact on the first \c METAL3 (only south present).
|
|
|
|
//! - An AutoContact plus a vertical plus a VTee (south & west present).
|
|
|
|
//!
|
|
|
|
//! \image html _do_xG_xM1_xM3.png "_do_xG_xM1_xM3()"
|
|
|
|
|
|
|
|
//! \function void GCellTopology::_do_xG_xM2 ();
|
|
|
|
//!
|
|
|
|
//! Construct a topology where there is at least one global (and up to 4),
|
|
|
|
//! and any number of \c METAL2 RoutingPads (assumeds H).
|
|
|
|
//!
|
|
|
|
//! In this topology, we want to try to reuse the \c METAL2 RoutingPad as a
|
|
|
|
//! feedtrough in the horizontal routage. Thus:
|
|
|
|
//! - The RoutingPad are connecteds trough a separate staircase (or
|
|
|
|
//! straight wire if aligneds).
|
|
|
|
//! - The south-west global wiring is attached to the leftmost RoutingPad
|
|
|
|
//! if there isn't south or to the biggest horizontal RoutingPad otherwise.
|
|
|
|
//! - The north-east global wiring is attached to the rightmost RoutingPad
|
|
|
|
//! if there isn't south or to the biggest horizontal RoutingPad otherwise.
|
|
|
|
//!
|
|
|
|
//! \image html _do_xG_xM2.png "_do_xG_xM2()"
|
|
|
|
|
|
|
|
//! \function void GCellTopology::_do_1G_1M3 ();
|
|
|
|
//!
|
|
|
|
//! Construct a topology where there is one global and one \c METAL3 RoutingPad
|
|
|
|
//! (assumeds V).
|
|
|
|
//!
|
|
|
|
//! In this topology, we reuse the \c METAL3 RoutingPad as a feedtrough in the
|
|
|
|
//! vertical routage. Thus:
|
|
|
|
//! - If the global is either north or south, we directly connect to the
|
|
|
|
//! north end or south end of the RoutingPad. \red{The vertical global will}
|
|
|
|
//! \red{have no slack at all we assume that METAL3 terminals are only from}
|
|
|
|
//! \red{blocks and are aligneds vertically.}
|
|
|
|
//! - If the global is east or west \e and the RoutingPad is sufficiently
|
|
|
|
//! extended in the vertical direction, we connect an horizontal in the
|
|
|
|
//! normal way.
|
|
|
|
//! - If the global is not sufficiently extended, we add a turn to give some
|
|
|
|
//! slack to the global.
|
|
|
|
//!
|
|
|
|
//!
|
|
|
|
//! \image html _do_1G_1M3.png "_do_1G_1M3()"
|
|
|
|
|
|
|
|
//! \function void GCellTopology::_do_xG_xM3 ();
|
|
|
|
//!
|
|
|
|
//! Construct a topology where there at least one global and two \c METAL3 RoutingPad
|
|
|
|
//! (assumed V).
|
|
|
|
//!
|
|
|
|
//! In this topology, we reuse the \c METAL3 RoutingPad as a feedtrough in the
|
|
|
|
//! vertical routage. \red{We assume that the most likely relative position}
|
|
|
|
//! \red{of the RoutingPads is to be aligned vertically}.
|
|
|
|
//! Thus:
|
|
|
|
//! - All RoutingPads are linked two by two trough vertical staircases.
|
|
|
|
//! - The south-west global wiring is attached to the bottommost RoutingPad
|
|
|
|
//! (without vertical slack). If a misalignment is detected, then a
|
|
|
|
//! dogleg is added.
|
|
|
|
//! - The north-east global wiring is attached to the topmost RoutingPad
|
|
|
|
//! (without vertical slack).
|
|
|
|
//!
|
|
|
|
//! South/west and north/south can be build independantly. Depending on the number
|
|
|
|
//! of globals, they can consist of:
|
|
|
|
//! - Nothing (no south nor west).
|
|
|
|
//! - An sliding AutoContact on the bottommost RoutingPad (west present).
|
|
|
|
//! - An fixed AutoContact on the bottommost RoutingPad (only south present).
|
|
|
|
//! - An fixed AutoContact plus a vertical plus a VTee (south & west present).
|
|
|
|
//!
|
|
|
|
//! \image html _do_xG_xM3.png "_do_xG_xM3()"
|
|
|
|
|
|
|
|
//! \function void singleGCell ( KatabaticEngine* ktbt, Net* net );
|
|
|
|
//!
|
|
|
|
//! All the RoutingPads of the net are concentrated under a single
|
|
|
|
//! GCell. This function assumes that all the terminals are in
|
|
|
|
//! \c METAL1 (vertical), and link them two by two by horizontal
|
|
|
|
//! wires.
|
|
|
|
|
|
|
|
//! \}
|
2010-03-09 09:24:29 -06:00
|
|
|
|
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
using namespace CRL;
|
|
|
|
using namespace Hurricane;
|
|
|
|
using namespace Katabatic;
|
|
|
|
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------
|
|
|
|
// Local Enum/Types.
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
enum LocalFunctionFlag { NoFlags = 0x00000000
|
|
|
|
, SortDecreasing = 0x00000001
|
|
|
|
, HAccess = 0x00000002
|
|
|
|
, VSmall = 0x00000004
|
|
|
|
, HSmall = 0x00000008
|
|
|
|
, Punctual = 0x00000010
|
|
|
|
, HCollapse = 0x00000020
|
|
|
|
, VCollapse = 0x00000040
|
|
|
|
, Terminal = 0x00000080
|
|
|
|
, DoSourceContact = 0x00000100
|
|
|
|
, DoTargetContact = 0x00000200
|
|
|
|
, SouthBound = 0x00010000
|
|
|
|
, NorthBound = 0x00020000
|
|
|
|
, WestBound = 0x00040000
|
|
|
|
, EastBound = 0x00080000
|
|
|
|
};
|
2010-03-09 09:24:29 -06:00
|
|
|
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------
|
|
|
|
// Local Variables.
|
|
|
|
|
|
|
|
const char* invalidGCell =
|
2013-12-03 18:58:58 -06:00
|
|
|
"Katabatic::GCellTopology () :\n\n"
|
2010-03-09 09:24:29 -06:00
|
|
|
" No GCell under point.\n";
|
|
|
|
|
|
|
|
const char* mismatchGCell =
|
2013-12-03 18:58:58 -06:00
|
|
|
"Katabatic::GCellTopology () :\n\n"
|
2010-03-09 09:24:29 -06:00
|
|
|
" Contacts under two different GCells.\n";
|
|
|
|
|
|
|
|
const char* missingGCell =
|
2013-12-03 18:58:58 -06:00
|
|
|
"Katabatic::GCellTopology () :\n\n"
|
2010-03-09 09:24:29 -06:00
|
|
|
" No Contact in GCell.\n";
|
|
|
|
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
map<Component*,AutoSegment*> __routingPadAutoSegments;
|
2010-03-09 09:24:29 -06:00
|
|
|
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------
|
|
|
|
// LoadGrByNet Local Classes.
|
|
|
|
|
|
|
|
|
|
|
|
struct NetCompareByName {
|
|
|
|
inline bool operator() ( const Net* lhs, const Net* rhs ) const;
|
|
|
|
};
|
|
|
|
|
|
|
|
inline bool NetCompareByName::operator() ( const Net* lhs, const Net* rhs ) const
|
2013-12-03 18:58:58 -06:00
|
|
|
{ return lhs->getName() < rhs->getName(); }
|
2010-03-09 09:24:29 -06:00
|
|
|
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------
|
|
|
|
// LoadGrByNet Local Functions.
|
|
|
|
|
|
|
|
|
|
|
|
void lookupClear ()
|
2013-12-03 18:58:58 -06:00
|
|
|
{ __routingPadAutoSegments.clear (); }
|
|
|
|
|
|
|
|
|
|
|
|
void getPositions ( Component* anchor, Point& source, Point& target )
|
2010-03-09 09:24:29 -06:00
|
|
|
{
|
2013-12-03 18:58:58 -06:00
|
|
|
Segment* segment = dynamic_cast<Segment*>( anchor );
|
|
|
|
if (segment) {
|
|
|
|
source = segment->getSourcePosition();
|
|
|
|
target = segment->getTargetPosition();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
RoutingPad* rp = dynamic_cast<RoutingPad*>( anchor );
|
|
|
|
if (rp) {
|
|
|
|
source = rp->getSourcePosition();
|
|
|
|
target = rp->getTargetPosition();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
source = anchor->getPosition();
|
|
|
|
target = anchor->getPosition();
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
unsigned int checkRoutingPadSize ( Component* anchor )
|
2010-03-09 09:24:29 -06:00
|
|
|
{
|
2013-12-03 18:58:58 -06:00
|
|
|
Point source;
|
|
|
|
Point target;
|
|
|
|
|
Improved UpdateSession & exception catching. Start of RoutingGauge implem.
Miscellaneous:
* Change: In <crlcore>, in display.conf use the same display threshold
for both METAL2 & METAL3.
In alliance.conf, the side of VIAs in the gauge is 2l (not 3l).
In kite.conf, separate edge densities for H/V.
* Change: In <Cell>, in flattenNets() use flag as argument, not a
boolean. Do not create rings for clock or supply nets.
* Change: In <DeepNet>, in _createRoutingPads() do not create rings
for clock or supply net (duplicated policy as in Cell::flattenNets()).
* Bug: In <ControllerWidget>, at last find the bad signal disconnect
that was causing ungraceful messages.
* Change: In <knik>, in Edge display occupancy/capacity in the string
name. Improved display progress and debugging capabilities.
Improved exception catch & breakpoint managment:
* Bug: In <PaletteWidget>, in updateExtensions() replace the calls to
deleteLayer() by delete. This cause the widget to be immediatly
erased instead of waiting for the event queue to be completly
processed. This was causing the widget to be left in a incoherent
state when stoping at a breakpoint.
* Bug: In <BreakpointWidget>, in execNoModal(), flush the main event
loop (QApplication::flush()) *before* lauching the *local* event
loop. This is to ensure all widgets are in their final state when
waiting (especially <PaletteWidget>).
* Change: In <ExceptionWidget>, new method catchAllWrapper() to
execute any std::function< void() > function/method with a "try"/
"catch" wraparound and lauch the widget in case something is catch.
* New: In <hurricane>, support for a oberver pattern, backported from
<katabatic> with an Obervable capable of being linked to any
number of Obervers.
* New: In <Cell>, made it observable to detect Cell change, currently
emit two kind of signals:
- Cell::CellAboutToChange : *before* any change.
- Cell::CellChanged : *after* the change has been completed.
* New: In <UpdateSession>, in Go::invalidate() add the Cell owning the
Go to the UPDATOR_STACK (of course the cell is added only once).
In addition, when the Cell is added, send a notification of
Cell::CellAboutToChange to all it's observers. The slave instances
are also invalidated.
Conversely in UpdateSession::_preDestroy() for each invalidated
Cell send a Cell::CellChanged notification to all observer.
The UPDATOR_STACK has been slightly amended to accept Cell which
are not Gos. Prior to this, the Cell where completly excluded from
the UpdateSession mechanism, so it's instances where never actualised
of anything referring to the Cell for that matter.
Note: we use two different mechanisms to transmit a Cell change,
observers and the slave instance map. I think at some point it
should be unificated.
* Change: In <CellViewer>, make it a Cell observer to redraw when the
cell is modificated (also update the palette).
Uses the catchAllWrapper() to protect all critical actions.
* Change: In <GraphicTool>, no longer need of cellPreModificated and
cellPostModificated signals. Now done through the Cell obersvers.
* Change: In <mauka>, <etesian> & <kite> now uses the catchAllWrapper
method for protection (need to split methods in two, to be able
to pass it as argument). No longer emit cellPreModificated and
cellPostModificated.
Support for RoutingGauge in P&R:
* Bug: In <placeandroute.py>, the connection from the internal power
ring to the connectors was not done correctly. Wrong contact layers
leading to a gap.
* Change: In <BuildPowerRails>, detection of the corona signals based
on how the "pck_px" pad is connected. No longer based on name
matching.
* Change: In <placeandroute.py>, support for 2 routing metal only
(3 metal in the technology).
* Change: In <katabatic> & <kite> support for a "top layer" limitation
on the routing gauge, this allows to use only two routing metals
(METAL2 & METAL3). Work in progress.
2014-04-20 12:25:08 -05:00
|
|
|
size_t anchorDepth = Session::getLayerDepth( anchor->getLayer() );
|
|
|
|
if (anchorDepth == 0) ++anchorDepth;
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
getPositions( anchor, source, target );
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
DbU::Unit width = abs( target.getX() - source.getX() );
|
|
|
|
DbU::Unit height = abs( target.getY() - source.getY() );
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
unsigned int flags = 0;
|
Improved UpdateSession & exception catching. Start of RoutingGauge implem.
Miscellaneous:
* Change: In <crlcore>, in display.conf use the same display threshold
for both METAL2 & METAL3.
In alliance.conf, the side of VIAs in the gauge is 2l (not 3l).
In kite.conf, separate edge densities for H/V.
* Change: In <Cell>, in flattenNets() use flag as argument, not a
boolean. Do not create rings for clock or supply nets.
* Change: In <DeepNet>, in _createRoutingPads() do not create rings
for clock or supply net (duplicated policy as in Cell::flattenNets()).
* Bug: In <ControllerWidget>, at last find the bad signal disconnect
that was causing ungraceful messages.
* Change: In <knik>, in Edge display occupancy/capacity in the string
name. Improved display progress and debugging capabilities.
Improved exception catch & breakpoint managment:
* Bug: In <PaletteWidget>, in updateExtensions() replace the calls to
deleteLayer() by delete. This cause the widget to be immediatly
erased instead of waiting for the event queue to be completly
processed. This was causing the widget to be left in a incoherent
state when stoping at a breakpoint.
* Bug: In <BreakpointWidget>, in execNoModal(), flush the main event
loop (QApplication::flush()) *before* lauching the *local* event
loop. This is to ensure all widgets are in their final state when
waiting (especially <PaletteWidget>).
* Change: In <ExceptionWidget>, new method catchAllWrapper() to
execute any std::function< void() > function/method with a "try"/
"catch" wraparound and lauch the widget in case something is catch.
* New: In <hurricane>, support for a oberver pattern, backported from
<katabatic> with an Obervable capable of being linked to any
number of Obervers.
* New: In <Cell>, made it observable to detect Cell change, currently
emit two kind of signals:
- Cell::CellAboutToChange : *before* any change.
- Cell::CellChanged : *after* the change has been completed.
* New: In <UpdateSession>, in Go::invalidate() add the Cell owning the
Go to the UPDATOR_STACK (of course the cell is added only once).
In addition, when the Cell is added, send a notification of
Cell::CellAboutToChange to all it's observers. The slave instances
are also invalidated.
Conversely in UpdateSession::_preDestroy() for each invalidated
Cell send a Cell::CellChanged notification to all observer.
The UPDATOR_STACK has been slightly amended to accept Cell which
are not Gos. Prior to this, the Cell where completly excluded from
the UpdateSession mechanism, so it's instances where never actualised
of anything referring to the Cell for that matter.
Note: we use two different mechanisms to transmit a Cell change,
observers and the slave instance map. I think at some point it
should be unificated.
* Change: In <CellViewer>, make it a Cell observer to redraw when the
cell is modificated (also update the palette).
Uses the catchAllWrapper() to protect all critical actions.
* Change: In <GraphicTool>, no longer need of cellPreModificated and
cellPostModificated signals. Now done through the Cell obersvers.
* Change: In <mauka>, <etesian> & <kite> now uses the catchAllWrapper
method for protection (need to split methods in two, to be able
to pass it as argument). No longer emit cellPreModificated and
cellPostModificated.
Support for RoutingGauge in P&R:
* Bug: In <placeandroute.py>, the connection from the internal power
ring to the connectors was not done correctly. Wrong contact layers
leading to a gap.
* Change: In <BuildPowerRails>, detection of the corona signals based
on how the "pck_px" pad is connected. No longer based on name
matching.
* Change: In <placeandroute.py>, support for 2 routing metal only
(3 metal in the technology).
* Change: In <katabatic> & <kite> support for a "top layer" limitation
on the routing gauge, this allows to use only two routing metals
(METAL2 & METAL3). Work in progress.
2014-04-20 12:25:08 -05:00
|
|
|
flags |= (width < 3*Session::getPitch(anchorDepth)) ? HSmall : 0;
|
|
|
|
flags |= (height < 3*Session::getPitch(anchorDepth)) ? VSmall : 0;
|
|
|
|
flags |= ((width == 0) && (height == 0)) ? Punctual : 0;
|
2013-12-03 18:58:58 -06:00
|
|
|
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "::checkRoutingPadSize(): pitch[" << anchorDepth << "]:"
|
ExtensionCap support and source/target terminal flags in Katabatic & Kite.
Placement management:
* Change: In <metis>, always disable the hMetis support regardless of
it being detected or not as the placer is still unable manage the
final bin contents.
Routing gauge management:
* Bug: In CRL Core, <vsclib/alliance.conf>, set the correct pitches and
size for the routing layers and the cell gauge.
* Change: In Katabatic & Kite, extract the correct extension cap for each
routing layer from the layers characteristics (cache then in
Katabatic::Configuration).
* Change: In Katabatic, <AutoSegment>, create segment with the wire width
defined in the gauge. For AutoSegment created on already existing
Segment from the global routing, adjust the width.
* Change: In Katabatic, <AutoSegment>, more accurate information about how
a segment is connected to terminal via source and/or target.
The flag SegStrongTerminal is splitted into SegSourceTerminal and
SegSourceTarget (but still used as a mask). So now we can know by
which end an AutoSegment is connected to a terminal.
* Change: In Katabatic, ::doRp_Access(), create constraint freeing segments
not only when HSmall but also when VSmall (more critical for <vsclib>).
Otherwise we may see AutoSegments with incompatible source/target
constraints.
* Change: In Kite, BuildPowerRails, do not create blockage on PinOnly
layers *but* still create power rails planes. This is a temporary
workaround for <vsclib> where the METAL1 blockages overlaps the
terminals (it was fine for Nero, but they shouldn't for Kite).
* Change: In Kite, <RoutingEvent>, if a TrackSegment is overconstrained,
directly bybass it's slackening state to DataNegociate::Slacken.
Also rename the flag "_canHandleConstraints" to "_overConstrained",
seems clearer to me.
Miscellaneous:
* Change: In CRL Core, <Utilities>, add a "pass-though" capability on the
mstream to temporarily make them print everything.
2014-05-25 08:00:35 -05:00
|
|
|
<< DbU::toLambda(Session::getPitch(anchorDepth)) << " "
|
|
|
|
<< ((flags & HSmall) ? "HSmall " : " ")
|
|
|
|
<< ((flags & VSmall) ? "VSmall " : " ")
|
|
|
|
<< endl;
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
return flags;
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Hook* getSegmentOppositeHook ( Hook* hook )
|
|
|
|
{
|
|
|
|
Segment* segment = static_cast<Segment*>( hook->getComponent() );
|
2013-12-03 18:58:58 -06:00
|
|
|
return segment->getOppositeHook( hook );
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
unsigned int getSegmentHookType ( Hook* hook )
|
2010-03-09 09:24:29 -06:00
|
|
|
{
|
|
|
|
Horizontal* horizontal = dynamic_cast<Horizontal*>( hook->getComponent() );
|
2013-12-03 18:58:58 -06:00
|
|
|
if (horizontal) {
|
|
|
|
if (horizontal->getSourceX() > horizontal->getTargetX())
|
2010-03-09 09:24:29 -06:00
|
|
|
cerr << Warning("Bad orientation of %s",getString(horizontal).c_str()) << endl;
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
if (dynamic_cast<Segment::SourceHook*>(hook))
|
|
|
|
return EastBound;
|
|
|
|
return WestBound;
|
|
|
|
} else {
|
|
|
|
Vertical* vertical = dynamic_cast<Vertical*>( hook->getComponent() );
|
|
|
|
if (vertical) {
|
|
|
|
if (vertical->getSourceY() > vertical->getTargetY())
|
|
|
|
cerr << Warning("Bad orientation of %s",getString(vertical).c_str()) << endl;
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
if (dynamic_cast<Segment::SourceHook*>(hook))
|
|
|
|
return NorthBound;
|
|
|
|
} else {
|
|
|
|
cerr << Warning("Unmanaged Hook %s",getString(hook).c_str()) << endl;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return SouthBound;
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------
|
|
|
|
// Class : "SortRpByX".
|
|
|
|
|
|
|
|
class SortRpByX {
|
|
|
|
public:
|
2013-12-03 18:58:58 -06:00
|
|
|
inline SortRpByX ( unsigned int flags );
|
|
|
|
inline bool operator() ( Component* rp1, Component* rp2 );
|
2010-03-09 09:24:29 -06:00
|
|
|
protected:
|
2013-12-03 18:58:58 -06:00
|
|
|
unsigned int _flags;
|
2010-03-09 09:24:29 -06:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
inline SortRpByX::SortRpByX ( unsigned int flags )
|
|
|
|
: _flags(flags)
|
2010-03-09 09:24:29 -06:00
|
|
|
{ }
|
|
|
|
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
inline bool SortRpByX::operator() ( Component* rp1, Component* rp2 )
|
2010-03-09 09:24:29 -06:00
|
|
|
{
|
|
|
|
DbU::Unit x1 = rp1->getCenter().getX();
|
|
|
|
DbU::Unit x2 = rp2->getCenter().getX();
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
if (x1 == x2) return false;
|
|
|
|
return (_flags & SortDecreasing) xor (x1 < x2);
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------
|
|
|
|
// Class : "SortRpByY".
|
|
|
|
|
|
|
|
class SortRpByY {
|
|
|
|
public:
|
2013-12-03 18:58:58 -06:00
|
|
|
inline SortRpByY ( unsigned int flags );
|
|
|
|
inline bool operator() ( Component* rp1, Component* rp2 );
|
2010-03-09 09:24:29 -06:00
|
|
|
protected:
|
2013-12-03 18:58:58 -06:00
|
|
|
unsigned int _flags;
|
2010-03-09 09:24:29 -06:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
inline SortRpByY::SortRpByY ( unsigned int flags )
|
|
|
|
: _flags(flags)
|
2010-03-09 09:24:29 -06:00
|
|
|
{ }
|
|
|
|
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
inline bool SortRpByY::operator() ( Component* rp1, Component* rp2 )
|
2010-03-09 09:24:29 -06:00
|
|
|
{
|
|
|
|
DbU::Unit y1 = rp1->getCenter().getY();
|
|
|
|
DbU::Unit y2 = rp2->getCenter().getY();
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
if (y1 == y2) return false;
|
|
|
|
return (_flags & SortDecreasing) xor (y1 < y2);
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------
|
|
|
|
// Class : "ForkStack".
|
|
|
|
|
|
|
|
class ForkStack {
|
|
|
|
public:
|
|
|
|
inline void push ( Hook* from, AutoContact* contact );
|
|
|
|
inline void pop ();
|
|
|
|
inline Hook* getFrom () const;
|
|
|
|
inline AutoContact* getContact () const;
|
|
|
|
private:
|
|
|
|
struct Element {
|
|
|
|
Hook* _from;
|
|
|
|
AutoContact* _contact;
|
|
|
|
inline Element ( Hook* from, AutoContact* contact );
|
|
|
|
};
|
|
|
|
private:
|
|
|
|
list<Element> _stack;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
inline ForkStack::Element::Element ( Hook* from, AutoContact* contact ) : _from(from), _contact(contact) {}
|
2013-12-03 18:58:58 -06:00
|
|
|
inline void ForkStack::pop () { if (not _stack.empty()) _stack.pop_back(); }
|
2010-03-09 09:24:29 -06:00
|
|
|
inline Hook* ForkStack::getFrom () const { return _stack.empty() ? NULL : _stack.back()._from; }
|
|
|
|
inline AutoContact* ForkStack::getContact () const { return _stack.empty() ? NULL : _stack.back()._contact; }
|
|
|
|
|
|
|
|
|
|
|
|
inline void ForkStack::push ( Hook* from, AutoContact* contact )
|
|
|
|
{
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << " Stacking " << from << " + " << contact << endl;
|
2013-12-03 18:58:58 -06:00
|
|
|
_stack.push_back( Element(from,contact) );
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------
|
2013-12-03 18:58:58 -06:00
|
|
|
// Class : "GGellTopology".
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
class GCellTopology {
|
2010-03-09 09:24:29 -06:00
|
|
|
|
|
|
|
public:
|
2013-12-03 18:58:58 -06:00
|
|
|
static void init ( unsigned int degree );
|
|
|
|
static void fixSegments ();
|
|
|
|
GCellTopology ( GCellGrid*, Hook* fromHook, AutoContact* sourceContact=NULL );
|
|
|
|
void construct ( ForkStack& forks );
|
|
|
|
inline unsigned int getStateG () const;
|
|
|
|
inline GCell* getGCell () const;
|
|
|
|
static void doRp_AutoContacts ( GCell*, Component*, AutoContact*& source, AutoContact*& target, unsigned int flags );
|
|
|
|
static AutoContact* doRp_Access ( GCell*, Component*, unsigned int flags );
|
Corrections in how to access pads terminals in Katabatic/Kite.
* Change: In Katabatic, in GCellTopology::doRp_AccessPad(), if the supporting
RoutingPad is big (more than two pitch), do not put the access
contact in the center but on the edge. This is to avoid cut violations
between the VIAs and the matrix of VIAs that may be generated under the
RoutingPad itself.
* Change: In Kite, in TrackSegment destructor, if the legnth of the wire,
without extensions is less than one picth, enlarge it so it encompass
it's source & target VIAs (to avoid notches).
* Bug: In Kite, in Track destructor, the TrackElements where detacheds,
but not deleted, causing a memory link.
2016-03-02 10:15:58 -06:00
|
|
|
static AutoContact* doRp_AccessPad ( RoutingPad*, unsigned int flags );
|
2013-12-03 18:58:58 -06:00
|
|
|
static void doRp_StairCaseH ( GCell*, Component* rp1, Component* rp2 );
|
|
|
|
static void doRp_StairCaseV ( GCell*, Component* rp1, Component* rp2 );
|
|
|
|
private:
|
|
|
|
void _do_xG ();
|
|
|
|
void _do_xG_1Pad ();
|
|
|
|
void _do_1G_1PinM2 ();
|
|
|
|
void _do_1G_1M1 ();
|
|
|
|
void _do_1G_xM1 ();
|
|
|
|
void _do_xG_xM1_xM3 ();
|
|
|
|
void _do_xG_1M1_1M2 ();
|
2016-04-22 15:43:53 -05:00
|
|
|
void _do_4G_1M2 ();
|
2013-12-03 18:58:58 -06:00
|
|
|
void _do_xG_xM2 ();
|
|
|
|
void _do_1G_1M3 ();
|
|
|
|
void _do_xG_xM3 ();
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
private:
|
2016-03-17 10:54:53 -05:00
|
|
|
enum ConnexityBits { GlobalBSize = 4
|
|
|
|
, Metal1BSize = 4
|
|
|
|
, Metal2BSize = 4
|
|
|
|
, Metal3BSize = 4
|
|
|
|
, PadsBSize = 4
|
|
|
|
, PinsBSize = 4
|
2013-12-03 18:58:58 -06:00
|
|
|
};
|
|
|
|
|
|
|
|
#define CONNEXITY_VALUE( Gs, M1s, M2s, M3s, pads, pins ) \
|
|
|
|
Gs + ((M1s ) << GlobalBSize) \
|
|
|
|
+ ((M2s ) << (GlobalBSize+Metal1BSize)) \
|
|
|
|
+ ((M3s ) << (GlobalBSize+Metal1BSize+Metal2BSize)) \
|
|
|
|
+ ((pads) << (GlobalBSize+Metal1BSize+Metal2BSize+Metal3BSize)) \
|
|
|
|
+ ((pins) << (GlobalBSize+Metal1BSize+Metal2BSize+Metal3BSize+PadsBSize))
|
|
|
|
|
|
|
|
// Connexity Name | G|M1|M2|M2|Pad|Pin|
|
|
|
|
enum ConnexityFlag { Conn_0G = CONNEXITY_VALUE( 0, 0, 0, 0, 0 , 0 )
|
|
|
|
, Conn_2G = CONNEXITY_VALUE( 2, 0, 0, 0, 0 , 0 )
|
|
|
|
, Conn_3G = CONNEXITY_VALUE( 3, 0, 0, 0, 0 , 0 )
|
|
|
|
, Conn_4G = CONNEXITY_VALUE( 4, 0, 0, 0, 0 , 0 )
|
|
|
|
, Conn_0G_2M1 = CONNEXITY_VALUE( 0, 2, 0, 0, 0 , 0 )
|
|
|
|
, Conn_1G_1M1 = CONNEXITY_VALUE( 1, 1, 0, 0, 0 , 0 )
|
|
|
|
, Conn_1G_2M1 = CONNEXITY_VALUE( 1, 2, 0, 0, 0 , 0 )
|
|
|
|
, Conn_1G_3M1 = CONNEXITY_VALUE( 1, 3, 0, 0, 0 , 0 )
|
|
|
|
, Conn_1G_4M1 = CONNEXITY_VALUE( 1, 4, 0, 0, 0 , 0 )
|
2016-03-17 10:54:53 -05:00
|
|
|
, Conn_1G_5M1 = CONNEXITY_VALUE( 1, 5, 0, 0, 0 , 0 )
|
2013-12-03 18:58:58 -06:00
|
|
|
, Conn_1G_1M2 = CONNEXITY_VALUE( 1, 0, 1, 0, 0 , 0 )
|
|
|
|
, Conn_1G_2M2 = CONNEXITY_VALUE( 1, 0, 2, 0, 0 , 0 )
|
|
|
|
, Conn_1G_3M2 = CONNEXITY_VALUE( 1, 0, 3, 0, 0 , 0 )
|
|
|
|
, Conn_1G_4M2 = CONNEXITY_VALUE( 1, 0, 4, 0, 0 , 0 )
|
|
|
|
, Conn_1G_1M3 = CONNEXITY_VALUE( 1, 0, 0, 1, 0 , 0 )
|
|
|
|
, Conn_1G_2M3 = CONNEXITY_VALUE( 1, 0, 0, 2, 0 , 0 )
|
|
|
|
, Conn_1G_3M3 = CONNEXITY_VALUE( 1, 0, 0, 3, 0 , 0 )
|
|
|
|
, Conn_1G_4M3 = CONNEXITY_VALUE( 1, 0, 0, 4, 0 , 0 )
|
|
|
|
, Conn_1G_1M1_1M2 = CONNEXITY_VALUE( 1, 1, 1, 0, 0 , 0 )
|
|
|
|
, Conn_1G_1M1_1M3 = CONNEXITY_VALUE( 1, 1, 0, 1, 0 , 0 )
|
|
|
|
// Connexity Name | G|M1|M2|M2|Pad|Pin|
|
|
|
|
, Conn_2G_1M1 = CONNEXITY_VALUE( 2, 1, 0, 0, 0 , 0 )
|
|
|
|
, Conn_2G_2M1 = CONNEXITY_VALUE( 2, 2, 0, 0, 0 , 0 )
|
|
|
|
, Conn_2G_3M1 = CONNEXITY_VALUE( 2, 3, 0, 0, 0 , 0 )
|
|
|
|
, Conn_2G_4M1 = CONNEXITY_VALUE( 2, 4, 0, 0, 0 , 0 )
|
Clocks in BLIF parser. In RSavePlugin, flags to select views to save.
* Bug: In CRL Core, in BlifParser, recognize clocks (Alliance patterns).
* Change: In Cumulus, in RSavePlugin, "kw" manage a new "views" to
specify which views must be saved. Physical by default, but sometimes
we need logical as well. If the design contains uniquified cells,
save the logical view.
In ClockTree, abort the clock tree building if the design has no
top level clock.
* Change: In Katabatic, in GCellTopology, adds 2G_5M1 configuration.
* Bug: Kite, in BuildPowerRails, if we are not in a chip the nets
composing the H-Tree must be protecteds be blockages.
2016-03-26 05:59:32 -05:00
|
|
|
, Conn_2G_5M1 = CONNEXITY_VALUE( 2, 5, 0, 0, 0 , 0 )
|
2013-12-03 18:58:58 -06:00
|
|
|
, Conn_2G_1M2 = CONNEXITY_VALUE( 2, 0, 1, 0, 0 , 0 )
|
|
|
|
, Conn_2G_2M2 = CONNEXITY_VALUE( 2, 0, 2, 0, 0 , 0 )
|
|
|
|
, Conn_2G_3M2 = CONNEXITY_VALUE( 2, 0, 3, 0, 0 , 0 )
|
|
|
|
, Conn_2G_4M2 = CONNEXITY_VALUE( 2, 0, 4, 0, 0 , 0 )
|
|
|
|
, Conn_2G_1M3 = CONNEXITY_VALUE( 2, 0, 0, 1, 0 , 0 )
|
|
|
|
, Conn_2G_2M3 = CONNEXITY_VALUE( 2, 0, 0, 2, 0 , 0 )
|
|
|
|
, Conn_2G_3M3 = CONNEXITY_VALUE( 2, 0, 0, 3, 0 , 0 )
|
|
|
|
, Conn_2G_4M3 = CONNEXITY_VALUE( 2, 0, 0, 4, 0 , 0 )
|
|
|
|
, Conn_2G_1M1_1M2 = CONNEXITY_VALUE( 2, 1, 1, 0, 0 , 0 )
|
|
|
|
// Connexity Name | G|M1|M2|M2|Pad|Pin|
|
|
|
|
, Conn_3G_1M1 = CONNEXITY_VALUE( 3, 1, 0, 0, 0 , 0 )
|
|
|
|
, Conn_3G_2M1 = CONNEXITY_VALUE( 3, 2, 0, 0, 0 , 0 )
|
|
|
|
, Conn_3G_3M1 = CONNEXITY_VALUE( 3, 3, 0, 0, 0 , 0 )
|
|
|
|
, Conn_3G_4M1 = CONNEXITY_VALUE( 3, 4, 0, 0, 0 , 0 )
|
|
|
|
, Conn_3G_1M2 = CONNEXITY_VALUE( 3, 0, 1, 0, 0 , 0 )
|
2016-04-13 11:42:55 -05:00
|
|
|
, Conn_3G_2M2 = CONNEXITY_VALUE( 3, 0, 2, 0, 0 , 0 )
|
2013-12-03 18:58:58 -06:00
|
|
|
, Conn_3G_1M3 = CONNEXITY_VALUE( 3, 0, 0, 1, 0 , 0 )
|
|
|
|
, Conn_3G_2M3 = CONNEXITY_VALUE( 3, 0, 0, 2, 0 , 0 )
|
|
|
|
, Conn_3G_3M3 = CONNEXITY_VALUE( 3, 0, 0, 3, 0 , 0 )
|
|
|
|
, Conn_3G_4M3 = CONNEXITY_VALUE( 3, 0, 0, 4, 0 , 0 )
|
|
|
|
// Connexity Name | G|M1|M2|M2|Pad|Pin|
|
|
|
|
, Conn_4G_1M1 = CONNEXITY_VALUE( 4, 1, 0, 0, 0 , 0 )
|
|
|
|
, Conn_4G_2M1 = CONNEXITY_VALUE( 4, 2, 0, 0, 0 , 0 )
|
|
|
|
, Conn_4G_3M1 = CONNEXITY_VALUE( 4, 3, 0, 0, 0 , 0 )
|
|
|
|
, Conn_4G_4M1 = CONNEXITY_VALUE( 4, 4, 0, 0, 0 , 0 )
|
2016-04-22 15:43:53 -05:00
|
|
|
, Conn_4G_1M2 = CONNEXITY_VALUE( 4, 0, 1, 0, 0 , 0 )
|
2013-12-03 18:58:58 -06:00
|
|
|
, Conn_4G_1M3 = CONNEXITY_VALUE( 4, 0, 0, 1, 0 , 0 )
|
|
|
|
, Conn_1G_1Pad = CONNEXITY_VALUE( 1, 0, 0, 0, 1 , 0 )
|
|
|
|
, Conn_2G_1Pad = CONNEXITY_VALUE( 2, 0, 0, 0, 1 , 0 )
|
|
|
|
, Conn_3G_1Pad = CONNEXITY_VALUE( 3, 0, 0, 0, 1 , 0 )
|
|
|
|
, Conn_1G_1PinM2 = CONNEXITY_VALUE( 1, 0, 1, 0, 0 , 1 )
|
|
|
|
};
|
|
|
|
|
|
|
|
// Connexity Union Type.
|
|
|
|
union UConnexity {
|
|
|
|
unsigned int connexity;
|
2010-03-09 09:24:29 -06:00
|
|
|
struct {
|
2013-12-03 18:58:58 -06:00
|
|
|
unsigned int globals : GlobalBSize;
|
|
|
|
unsigned int M1 : Metal1BSize;
|
|
|
|
unsigned int M2 : Metal2BSize;
|
|
|
|
unsigned int M3 : Metal3BSize;
|
|
|
|
unsigned int Pad : PadsBSize;
|
|
|
|
unsigned int Pin : PinsBSize;
|
2010-03-09 09:24:29 -06:00
|
|
|
} fields;
|
|
|
|
};
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
enum TopologyFlag { Global_Vertical_End = 0x00000001
|
|
|
|
, Global_Horizontal_End = 0x00000002
|
|
|
|
, Global_Horizontal = 0x00000004
|
|
|
|
, Global_Vertical = 0x00000008
|
|
|
|
, Global_Turn = 0x00000010
|
|
|
|
, Global_Fork = 0x00000020
|
|
|
|
, Global_Fixed = 0x00000040
|
|
|
|
, Global_End = Global_Vertical_End | Global_Horizontal_End
|
|
|
|
, Global_Split = Global_Horizontal | Global_Vertical | Global_Fork
|
|
|
|
};
|
|
|
|
|
2010-03-09 09:24:29 -06:00
|
|
|
// Attributes.
|
* ./katabatic:
- New: ChipTools, regroup all datas and utilities to manage a full-chip
design.
- Change: In LoadGrByNet/GCellConfiguration::_GCell_xG_1Pad(), uses straight
perpandicular wires on top & right pads. The GCells under those connectors
are fully saturated, wires must go out as straight as possible.
- Change: In AutoHorizontal/AutoVertical, update the "terminal" flag after
slackening. This is to avoid global that are no longer connected to
terminals behave as such.
- Change: In AutoSegment::canMoveUp() & canPivotUp(), prevent all local segments
to go up. This is to avoid cluttering upper levels with small segments.
- Change: In GCellConfiguration, for xG_xL3, detect straight vertical topologies
and for them to be fixed (new flag FIXED_GLOBAL set by the topological
builder instead of the constructor). This prevent the router to do
stupid things...
To sets the "Fixed" flag *after* the axis of global segments have been
positionned correctly adds a "_toFixGlobals" static table lookup.
- Change: In GCell::stepDesaturate(), if less than one free track remains in
the upper layer, do not move up the segment. This allows from a minimum
free room for expansion.
- Change: In GCell::_getString(), display indexes and layer names in various
saturation tables.
- Change: In AutoSegment, allow move up of local Segment. ::moveUp() and
::canMoveUp() arguments go from booleans to flags, which are more
explicits.
2010-12-04 09:25:18 -06:00
|
|
|
private:
|
|
|
|
static vector<AutoSegment*> _toFixSegments;
|
2011-01-25 11:16:27 -06:00
|
|
|
static unsigned int _degree;
|
2013-12-03 18:58:58 -06:00
|
|
|
UConnexity _connexity;
|
* ./katabatic:
- New: ChipTools, regroup all datas and utilities to manage a full-chip
design.
- Change: In LoadGrByNet/GCellConfiguration::_GCell_xG_1Pad(), uses straight
perpandicular wires on top & right pads. The GCells under those connectors
are fully saturated, wires must go out as straight as possible.
- Change: In AutoHorizontal/AutoVertical, update the "terminal" flag after
slackening. This is to avoid global that are no longer connected to
terminals behave as such.
- Change: In AutoSegment::canMoveUp() & canPivotUp(), prevent all local segments
to go up. This is to avoid cluttering upper levels with small segments.
- Change: In GCellConfiguration, for xG_xL3, detect straight vertical topologies
and for them to be fixed (new flag FIXED_GLOBAL set by the topological
builder instead of the constructor). This prevent the router to do
stupid things...
To sets the "Fixed" flag *after* the axis of global segments have been
positionned correctly adds a "_toFixGlobals" static table lookup.
- Change: In GCell::stepDesaturate(), if less than one free track remains in
the upper layer, do not move up the segment. This allows from a minimum
free room for expansion.
- Change: In GCell::_getString(), display indexes and layer names in various
saturation tables.
- Change: In AutoSegment, allow move up of local Segment. ::moveUp() and
::canMoveUp() arguments go from booleans to flags, which are more
explicits.
2010-12-04 09:25:18 -06:00
|
|
|
unsigned int _topology;
|
|
|
|
Net* _net;
|
|
|
|
GCell* _gcell;
|
|
|
|
AutoContact* _sourceContact;
|
|
|
|
AutoContact* _southWestContact;
|
|
|
|
AutoContact* _northEastContact;
|
|
|
|
Hook* _fromHook;
|
|
|
|
Hook* _east;
|
|
|
|
Hook* _west;
|
|
|
|
Hook* _north;
|
|
|
|
Hook* _south;
|
Corrections in how to access pads terminals in Katabatic/Kite.
* Change: In Katabatic, in GCellTopology::doRp_AccessPad(), if the supporting
RoutingPad is big (more than two pitch), do not put the access
contact in the center but on the edge. This is to avoid cut violations
between the VIAs and the matrix of VIAs that may be generated under the
RoutingPad itself.
* Change: In Kite, in TrackSegment destructor, if the legnth of the wire,
without extensions is less than one picth, enlarge it so it encompass
it's source & target VIAs (to avoid notches).
* Bug: In Kite, in Track destructor, the TrackElements where detacheds,
but not deleted, causing a memory link.
2016-03-02 10:15:58 -06:00
|
|
|
vector<RoutingPad*> _routingPads;
|
2010-03-09 09:24:29 -06:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
inline unsigned int GCellTopology::getStateG () const { return _connexity.fields.globals; }
|
|
|
|
inline GCell* GCellTopology::getGCell () const { return _gcell; }
|
2010-03-09 09:24:29 -06:00
|
|
|
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
vector<AutoSegment*> GCellTopology::_toFixSegments;
|
|
|
|
unsigned int GCellTopology::_degree = 0;
|
* ./katabatic:
- New: ChipTools, regroup all datas and utilities to manage a full-chip
design.
- Change: In LoadGrByNet/GCellConfiguration::_GCell_xG_1Pad(), uses straight
perpandicular wires on top & right pads. The GCells under those connectors
are fully saturated, wires must go out as straight as possible.
- Change: In AutoHorizontal/AutoVertical, update the "terminal" flag after
slackening. This is to avoid global that are no longer connected to
terminals behave as such.
- Change: In AutoSegment::canMoveUp() & canPivotUp(), prevent all local segments
to go up. This is to avoid cluttering upper levels with small segments.
- Change: In GCellConfiguration, for xG_xL3, detect straight vertical topologies
and for them to be fixed (new flag FIXED_GLOBAL set by the topological
builder instead of the constructor). This prevent the router to do
stupid things...
To sets the "Fixed" flag *after* the axis of global segments have been
positionned correctly adds a "_toFixGlobals" static table lookup.
- Change: In GCell::stepDesaturate(), if less than one free track remains in
the upper layer, do not move up the segment. This allows from a minimum
free room for expansion.
- Change: In GCell::_getString(), display indexes and layer names in various
saturation tables.
- Change: In AutoSegment, allow move up of local Segment. ::moveUp() and
::canMoveUp() arguments go from booleans to flags, which are more
explicits.
2010-12-04 09:25:18 -06:00
|
|
|
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
void GCellTopology::fixSegments ()
|
* ./katabatic:
- New: ChipTools, regroup all datas and utilities to manage a full-chip
design.
- Change: In LoadGrByNet/GCellConfiguration::_GCell_xG_1Pad(), uses straight
perpandicular wires on top & right pads. The GCells under those connectors
are fully saturated, wires must go out as straight as possible.
- Change: In AutoHorizontal/AutoVertical, update the "terminal" flag after
slackening. This is to avoid global that are no longer connected to
terminals behave as such.
- Change: In AutoSegment::canMoveUp() & canPivotUp(), prevent all local segments
to go up. This is to avoid cluttering upper levels with small segments.
- Change: In GCellConfiguration, for xG_xL3, detect straight vertical topologies
and for them to be fixed (new flag FIXED_GLOBAL set by the topological
builder instead of the constructor). This prevent the router to do
stupid things...
To sets the "Fixed" flag *after* the axis of global segments have been
positionned correctly adds a "_toFixGlobals" static table lookup.
- Change: In GCell::stepDesaturate(), if less than one free track remains in
the upper layer, do not move up the segment. This allows from a minimum
free room for expansion.
- Change: In GCell::_getString(), display indexes and layer names in various
saturation tables.
- Change: In AutoSegment, allow move up of local Segment. ::moveUp() and
::canMoveUp() arguments go from booleans to flags, which are more
explicits.
2010-12-04 09:25:18 -06:00
|
|
|
{
|
|
|
|
for ( size_t i=0 ; i<_toFixSegments.size() ; ++i )
|
2013-12-03 18:58:58 -06:00
|
|
|
_toFixSegments[i]->setFlags( SegFixed );
|
|
|
|
_toFixSegments.clear();
|
* ./katabatic:
- New: ChipTools, regroup all datas and utilities to manage a full-chip
design.
- Change: In LoadGrByNet/GCellConfiguration::_GCell_xG_1Pad(), uses straight
perpandicular wires on top & right pads. The GCells under those connectors
are fully saturated, wires must go out as straight as possible.
- Change: In AutoHorizontal/AutoVertical, update the "terminal" flag after
slackening. This is to avoid global that are no longer connected to
terminals behave as such.
- Change: In AutoSegment::canMoveUp() & canPivotUp(), prevent all local segments
to go up. This is to avoid cluttering upper levels with small segments.
- Change: In GCellConfiguration, for xG_xL3, detect straight vertical topologies
and for them to be fixed (new flag FIXED_GLOBAL set by the topological
builder instead of the constructor). This prevent the router to do
stupid things...
To sets the "Fixed" flag *after* the axis of global segments have been
positionned correctly adds a "_toFixGlobals" static table lookup.
- Change: In GCell::stepDesaturate(), if less than one free track remains in
the upper layer, do not move up the segment. This allows from a minimum
free room for expansion.
- Change: In GCell::_getString(), display indexes and layer names in various
saturation tables.
- Change: In AutoSegment, allow move up of local Segment. ::moveUp() and
::canMoveUp() arguments go from booleans to flags, which are more
explicits.
2010-12-04 09:25:18 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
void GCellTopology::init ( unsigned int degree )
|
2011-01-25 11:16:27 -06:00
|
|
|
{
|
2013-12-03 18:58:58 -06:00
|
|
|
_degree = degree;
|
|
|
|
_toFixSegments.clear();
|
2011-01-25 11:16:27 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
GCellTopology::GCellTopology ( GCellGrid* gcellGrid
|
|
|
|
, Hook* fromHook
|
|
|
|
, AutoContact* sourceContact )
|
|
|
|
: _connexity ()
|
* ./katabatic:
- Change: In loadGlobalRouting(), more exlicit message as to why a net is
filtered out the routing set (POWER, GROUND, CLOCK or BLOCKAGE).
- New: ChipTool, module with utilities specific to chip routing. Containing
a function to pre-break wires around a block.
- New: In loadGlobalRouting/GCellConfiguration, adds supports for RoutingPad
connected on chip's pad.
- New: In GCellGrid/GCell, can compute the density in three modes: average,
max of H/V and max of layer. Currently implemented in getMaxHVDensity(),
should be unificated with getDensity(). Used for display purposes.
- Bug: In AutoContact, when splitting a contact, add a specific check for
"one layer span".
- Bug: In AutoSegment::canMoveUp(), do not move up fixed segments.
2010-11-16 07:59:38 -06:00
|
|
|
, _topology (0)
|
|
|
|
, _gcell (NULL)
|
|
|
|
, _sourceContact (sourceContact)
|
2010-03-09 09:24:29 -06:00
|
|
|
, _southWestContact(NULL)
|
|
|
|
, _northEastContact(NULL)
|
* ./katabatic:
- Change: In loadGlobalRouting(), more exlicit message as to why a net is
filtered out the routing set (POWER, GROUND, CLOCK or BLOCKAGE).
- New: ChipTool, module with utilities specific to chip routing. Containing
a function to pre-break wires around a block.
- New: In loadGlobalRouting/GCellConfiguration, adds supports for RoutingPad
connected on chip's pad.
- New: In GCellGrid/GCell, can compute the density in three modes: average,
max of H/V and max of layer. Currently implemented in getMaxHVDensity(),
should be unificated with getDensity(). Used for display purposes.
- Bug: In AutoContact, when splitting a contact, add a specific check for
"one layer span".
- Bug: In AutoSegment::canMoveUp(), do not move up fixed segments.
2010-11-16 07:59:38 -06:00
|
|
|
, _fromHook (fromHook)
|
|
|
|
, _east (NULL)
|
|
|
|
, _west (NULL)
|
|
|
|
, _north (NULL)
|
|
|
|
, _south (NULL)
|
|
|
|
, _routingPads ()
|
2010-03-09 09:24:29 -06:00
|
|
|
{
|
2013-12-03 18:58:58 -06:00
|
|
|
_connexity.connexity = 0;
|
|
|
|
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,1) << "GCellTopology::GCellTopology()" << endl;
|
|
|
|
cdebug_log(145,0) << getString(fromHook) << endl;
|
|
|
|
cdebug_log(145,0) << sourceContact << endl;
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
Segment* fromSegment = static_cast<Segment*>( _fromHook->getComponent() );
|
|
|
|
_net = fromSegment->getNet();
|
2010-03-09 09:24:29 -06:00
|
|
|
|
|
|
|
forEach ( Hook*, hook, fromHook->getHooks() ) {
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "Topology [" << _connexity.connexity << "] = "
|
Solve the template lookup problem in tstream.
* Bug: In Hurricane, in tstream (Commons.h), in the *template* overload of
operator<<(), do not use the operator<<() of ostream as it will be
looked up in "stage 1" (template definition) and so will miss all the
overloads added later and built over getString<>(). Instead, make use
of getString<>(), which, as another template will be looked up in
"stage 2" (template instanciation) and at that point will have all the
needed template specialisation of getString<>().
We also need to define new stream manipulators to be able to create
a matching template overload not dependant from the implementation.
To avoid name clashes, we prefix a 't'. For now, only 'tsetw()' is
refined.
As a side effect, we cannot directly print bit-fields into the stream,
we must go through an intermediate variable (happens once in AutoContact).
2016-07-19 09:02:55 -05:00
|
|
|
<< "[" << (int)_connexity.fields.globals
|
|
|
|
<< "+" << (int)_connexity.fields.M1
|
|
|
|
<< "+" << (int)_connexity.fields.M2
|
|
|
|
<< "+" << (int)_connexity.fields.M3
|
|
|
|
<< "+" << (int)_connexity.fields.Pin
|
|
|
|
<< "+" << (int)_connexity.fields.Pad
|
2016-03-17 10:54:53 -05:00
|
|
|
<< "] " << _gcell
|
|
|
|
<< endl;
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
Segment* toSegment = dynamic_cast<Segment*>( hook->getComponent() );
|
|
|
|
|
|
|
|
if (toSegment) {
|
2010-03-09 09:24:29 -06:00
|
|
|
switch ( getSegmentHookType(*hook) ) {
|
2013-12-03 18:58:58 -06:00
|
|
|
case WestBound: _west = *hook; break;
|
|
|
|
case EastBound: _east = *hook; break;
|
|
|
|
case SouthBound: _south = *hook; break;
|
|
|
|
case NorthBound: _north = *hook; break;
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
_connexity.fields.globals++;
|
|
|
|
} else {
|
|
|
|
Component* anchor = hook->getComponent();
|
|
|
|
RoutingPad* rp = dynamic_cast<RoutingPad*>( anchor );
|
|
|
|
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "| Looking for Anchor:" << anchor << " rp:" << rp << endl;
|
2013-12-03 18:58:58 -06:00
|
|
|
|
|
|
|
if (anchor) {
|
|
|
|
Contact* contact = dynamic_cast<Contact*>( anchor );
|
|
|
|
if (contact
|
|
|
|
and ( Session::getKatabatic()->isGContact( anchor->getLayer() )
|
|
|
|
or Session::getKatabatic()->isGMetal ( anchor->getLayer() )) ) {
|
|
|
|
// Global routing articulation contact are in <gmetalh> not <gcut> ?
|
|
|
|
GCell* gcell = gcellGrid->getGCell( contact->getCenter() );
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "* Global Routing Articulation: " << contact << endl;
|
|
|
|
cdebug_log(145,0) << "| " << gcell << endl;
|
2013-12-03 18:58:58 -06:00
|
|
|
if (gcell == NULL)
|
|
|
|
throw Error( invalidGCell );
|
|
|
|
if (_gcell == NULL) _gcell = gcell;
|
|
|
|
else if (_gcell != gcell) {
|
|
|
|
throw Error( mismatchGCell );
|
|
|
|
}
|
* ./katabatic:
- Change: In loadGlobalRouting(), more exlicit message as to why a net is
filtered out the routing set (POWER, GROUND, CLOCK or BLOCKAGE).
- New: ChipTool, module with utilities specific to chip routing. Containing
a function to pre-break wires around a block.
- New: In loadGlobalRouting/GCellConfiguration, adds supports for RoutingPad
connected on chip's pad.
- New: In GCellGrid/GCell, can compute the density in three modes: average,
max of H/V and max of layer. Currently implemented in getMaxHVDensity(),
should be unificated with getDensity(). Used for display purposes.
- Bug: In AutoContact, when splitting a contact, add a specific check for
"one layer span".
- Bug: In AutoSegment::canMoveUp(), do not move up fixed segments.
2010-11-16 07:59:38 -06:00
|
|
|
} else {
|
2013-12-03 18:58:58 -06:00
|
|
|
if (rp and AllianceFramework::get()->isPad(rp->_getEntityAsComponent()->getCell())) {
|
|
|
|
_connexity.fields.Pad++;
|
|
|
|
} else {
|
|
|
|
const Layer* layer = anchor->getLayer();
|
|
|
|
|
|
|
|
if (layer == Session::getRoutingLayer(0)) _connexity.fields.M1++; // M1 V
|
|
|
|
else if (layer == Session::getRoutingLayer(1)) _connexity.fields.M2++; // M2 H
|
|
|
|
else if (layer == Session::getRoutingLayer(2)) _connexity.fields.M3++; // M3 V
|
|
|
|
else if (layer == Session::getRoutingLayer(3)) _connexity.fields.M2++; // M4 H
|
|
|
|
else if (layer == Session::getRoutingLayer(4)) _connexity.fields.M3++; // M5 V
|
|
|
|
else {
|
|
|
|
cerr << Warning( "Terminal layer \"%s\" of %s is not managed yet (ignored)."
|
|
|
|
, getString(layer->getName()).c_str()
|
|
|
|
, getString(anchor).c_str() )
|
|
|
|
<< endl;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dynamic_cast<Pin*>(rp->getOccurrence().getEntity())) _connexity.fields.Pin++;
|
* ./katabatic:
- Change: In loadGlobalRouting(), more exlicit message as to why a net is
filtered out the routing set (POWER, GROUND, CLOCK or BLOCKAGE).
- New: ChipTool, module with utilities specific to chip routing. Containing
a function to pre-break wires around a block.
- New: In loadGlobalRouting/GCellConfiguration, adds supports for RoutingPad
connected on chip's pad.
- New: In GCellGrid/GCell, can compute the density in three modes: average,
max of H/V and max of layer. Currently implemented in getMaxHVDensity(),
should be unificated with getDensity(). Used for display purposes.
- Bug: In AutoContact, when splitting a contact, add a specific check for
"one layer span".
- Bug: In AutoSegment::canMoveUp(), do not move up fixed segments.
2010-11-16 07:59:38 -06:00
|
|
|
}
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "| Component to connect: " << anchor << endl;
|
Corrections in how to access pads terminals in Katabatic/Kite.
* Change: In Katabatic, in GCellTopology::doRp_AccessPad(), if the supporting
RoutingPad is big (more than two pitch), do not put the access
contact in the center but on the edge. This is to avoid cut violations
between the VIAs and the matrix of VIAs that may be generated under the
RoutingPad itself.
* Change: In Kite, in TrackSegment destructor, if the legnth of the wire,
without extensions is less than one picth, enlarge it so it encompass
it's source & target VIAs (to avoid notches).
* Bug: In Kite, in Track destructor, the TrackElements where detacheds,
but not deleted, causing a memory link.
2016-03-02 10:15:58 -06:00
|
|
|
_routingPads.push_back( rp );
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "east: " << _east << endl;
|
|
|
|
cdebug_log(145,0) << "west: " << _west << endl;
|
|
|
|
cdebug_log(145,0) << "north:" << _north << endl;
|
|
|
|
cdebug_log(145,0) << "south:" << _south << endl;
|
2013-12-03 18:58:58 -06:00
|
|
|
|
|
|
|
if (_connexity.fields.globals == 1) {
|
|
|
|
if ( _north or _south ) _topology |= Global_Vertical_End;
|
|
|
|
else _topology |= Global_Horizontal_End;
|
|
|
|
} else if (_connexity.fields.globals == 2) {
|
|
|
|
if ( _east && _west ) _topology |= Global_Horizontal;
|
|
|
|
else if ( _north && _south ) _topology |= Global_Vertical;
|
|
|
|
else _topology |= Global_Turn;
|
2010-03-09 09:24:29 -06:00
|
|
|
} else {
|
2013-12-03 18:58:58 -06:00
|
|
|
_topology |= Global_Fork;
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_tabw(145,-1);
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
if (_gcell == NULL) throw Error( missingGCell );
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
void GCellTopology::construct ( ForkStack& forks )
|
2010-03-09 09:24:29 -06:00
|
|
|
{
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,1) << "GCellTopology::construct() [" << _connexity.connexity << "] in " << _gcell << endl;
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
_southWestContact = NULL;
|
|
|
|
_northEastContact = NULL;
|
|
|
|
|
2010-03-09 09:24:29 -06:00
|
|
|
bool straightLine = false;
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
switch ( _connexity.connexity ) {
|
|
|
|
case Conn_1G_1Pad:
|
|
|
|
case Conn_2G_1Pad:
|
|
|
|
case Conn_3G_1Pad: _do_xG_1Pad(); break;
|
|
|
|
case Conn_1G_1PinM2: _do_1G_1PinM2(); break;
|
|
|
|
case Conn_1G_1M1: _do_1G_1M1(); break;
|
|
|
|
case Conn_1G_2M1:
|
|
|
|
case Conn_1G_3M1:
|
2016-03-17 10:54:53 -05:00
|
|
|
case Conn_1G_4M1:
|
|
|
|
case Conn_1G_5M1: _do_1G_xM1(); break;
|
2013-12-03 18:58:58 -06:00
|
|
|
case Conn_1G_1M2:
|
|
|
|
case Conn_1G_2M2:
|
|
|
|
case Conn_1G_3M2:
|
|
|
|
case Conn_1G_4M2: _do_xG_xM2(); break;
|
|
|
|
case Conn_1G_1M3: _do_1G_1M3(); break;
|
|
|
|
case Conn_1G_2M3:
|
|
|
|
case Conn_1G_3M3:
|
|
|
|
case Conn_1G_4M3: _do_xG_xM3 (); break;
|
|
|
|
case Conn_1G_1M1_1M2: _do_xG_1M1_1M2(); break;
|
|
|
|
case Conn_1G_1M1_1M3: _do_1G_xM1 (); break;
|
|
|
|
case Conn_2G_1M1:
|
|
|
|
case Conn_2G_2M1:
|
|
|
|
case Conn_2G_3M1:
|
|
|
|
case Conn_2G_4M1:
|
Clocks in BLIF parser. In RSavePlugin, flags to select views to save.
* Bug: In CRL Core, in BlifParser, recognize clocks (Alliance patterns).
* Change: In Cumulus, in RSavePlugin, "kw" manage a new "views" to
specify which views must be saved. Physical by default, but sometimes
we need logical as well. If the design contains uniquified cells,
save the logical view.
In ClockTree, abort the clock tree building if the design has no
top level clock.
* Change: In Katabatic, in GCellTopology, adds 2G_5M1 configuration.
* Bug: Kite, in BuildPowerRails, if we are not in a chip the nets
composing the H-Tree must be protecteds be blockages.
2016-03-26 05:59:32 -05:00
|
|
|
case Conn_2G_5M1:
|
2013-12-03 18:58:58 -06:00
|
|
|
case Conn_3G_1M1:
|
|
|
|
case Conn_3G_2M1:
|
|
|
|
case Conn_3G_3M1:
|
|
|
|
case Conn_3G_4M1:
|
|
|
|
case Conn_3G_2M3:
|
|
|
|
case Conn_3G_3M3:
|
|
|
|
case Conn_3G_4M3:
|
|
|
|
case Conn_4G_1M1:
|
|
|
|
case Conn_4G_2M1:
|
|
|
|
case Conn_4G_3M1:
|
|
|
|
case Conn_4G_4M1: _do_xG_xM1_xM3(); break;
|
2016-04-22 15:43:53 -05:00
|
|
|
case Conn_4G_1M2: _do_4G_1M2(); break;
|
2013-12-03 18:58:58 -06:00
|
|
|
case Conn_2G_1M2:
|
|
|
|
case Conn_2G_2M2:
|
|
|
|
case Conn_2G_3M2:
|
|
|
|
case Conn_2G_4M2:
|
2016-04-13 11:42:55 -05:00
|
|
|
case Conn_3G_1M2:
|
|
|
|
case Conn_3G_2M2: _do_xG_xM2(); break;
|
2013-12-03 18:58:58 -06:00
|
|
|
case Conn_2G_1M3:
|
|
|
|
case Conn_2G_2M3:
|
|
|
|
case Conn_2G_3M3:
|
|
|
|
case Conn_2G_4M3:
|
|
|
|
case Conn_3G_1M3: _do_xG_xM3 (); break;
|
|
|
|
case Conn_2G_1M1_1M2: _do_xG_1M1_1M2(); break;
|
|
|
|
case Conn_2G:
|
|
|
|
if ( (_east and _west) or (_north and _south) ) {
|
2010-03-09 09:24:29 -06:00
|
|
|
straightLine = true;
|
|
|
|
break;
|
|
|
|
}
|
2013-12-03 18:58:58 -06:00
|
|
|
case Conn_3G:
|
|
|
|
case Conn_4G:
|
|
|
|
_do_xG();
|
2010-03-09 09:24:29 -06:00
|
|
|
break;
|
|
|
|
default:
|
Bug fixes for MOSIS SCMOS_DEEP support.
* Change: In Hurricane, the NetRoutingProperty is moved into Hurricane
from Katabatic. Needed for Knik to be able to access thoses
informations.
* Change: In Hurricane, in RoutingPad::setOnBestComponent(), now in
case of identical area, select the component of lowest id.
This should not be needed if the component ordering was fully
deterministic as it should be (will investigate later).
This is to ensure that the choosen component is always the
same, especially between save/load of a global routing.
* Bug: In Katabatic, in AutoContactHTee::updateTopology(), invalidate
the segments only if the topology is valid (no NULL in the
cached segments).
* Bug: In Katabatic, in GCellTopology::construct(), throw an error
if the topology is bad instead of trying to continue (and core
dump later... ).
* Bug: In Kite, in BuildPowerRails, distinguish the name of the master
net in the pad (for vddi, vssi, vdde, vsse, ck, cki & cko) and the
name of the net in the *chip* netlist. Must use the later to make
comparison as they may differs.
* Change: In Knik, in save/load solution, exclude nets that are not
globally routed by Knik. That is which NetRoutingProperty is not
*Automatic*.
* Bug: In Cumulus, in chip.BlockPower take account of the layer
width extention to sligthy shrink the connector thus avoiding a
notch with standart cell in some cases.
* Change: In Cumulus, in chip.ClockTree disable the use of fixed Steiner
trees for the leaf clocks, as it seems overconstrained for the
router. First move was to lower them in M2/M3 (instead of M3/M4)
but that was not sufficent.
* New: In Cumulus, RSavePlugin for recursively saving a physical
hierarchy.
* New: In documentation, first embryo for RDS file. Should have been
in Alliance git, but I prefer to keep newest doc in Coriolis.
2014-09-21 09:44:37 -05:00
|
|
|
throw Bug( "Unmanaged Configuration [%d] = [%d+%d+%d+%d,%d+%d] %s in %s\n"
|
|
|
|
" The global routing seems to be defective."
|
|
|
|
, _connexity.connexity
|
|
|
|
, _connexity.fields.globals
|
|
|
|
, _connexity.fields.M1
|
|
|
|
, _connexity.fields.M2
|
|
|
|
, _connexity.fields.M3
|
|
|
|
, _connexity.fields.Pin
|
|
|
|
, _connexity.fields.Pad
|
|
|
|
, _net->_getString().c_str()
|
|
|
|
, getString(_gcell).c_str()
|
|
|
|
);
|
2013-12-03 18:58:58 -06:00
|
|
|
_do_xG();
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
if (straightLine) {
|
|
|
|
// This a global router problem.
|
Bug fixes for MOSIS SCMOS_DEEP support.
* Change: In Hurricane, the NetRoutingProperty is moved into Hurricane
from Katabatic. Needed for Knik to be able to access thoses
informations.
* Change: In Hurricane, in RoutingPad::setOnBestComponent(), now in
case of identical area, select the component of lowest id.
This should not be needed if the component ordering was fully
deterministic as it should be (will investigate later).
This is to ensure that the choosen component is always the
same, especially between save/load of a global routing.
* Bug: In Katabatic, in AutoContactHTee::updateTopology(), invalidate
the segments only if the topology is valid (no NULL in the
cached segments).
* Bug: In Katabatic, in GCellTopology::construct(), throw an error
if the topology is bad instead of trying to continue (and core
dump later... ).
* Bug: In Kite, in BuildPowerRails, distinguish the name of the master
net in the pad (for vddi, vssi, vdde, vsse, ck, cki & cko) and the
name of the net in the *chip* netlist. Must use the later to make
comparison as they may differs.
* Change: In Knik, in save/load solution, exclude nets that are not
globally routed by Knik. That is which NetRoutingProperty is not
*Automatic*.
* Bug: In Cumulus, in chip.BlockPower take account of the layer
width extention to sligthy shrink the connector thus avoiding a
notch with standart cell in some cases.
* Change: In Cumulus, in chip.ClockTree disable the use of fixed Steiner
trees for the leaf clocks, as it seems overconstrained for the
router. First move was to lower them in M2/M3 (instead of M3/M4)
but that was not sufficent.
* New: In Cumulus, RSavePlugin for recursively saving a physical
hierarchy.
* New: In documentation, first embryo for RDS file. Should have been
in Alliance git, but I prefer to keep newest doc in Coriolis.
2014-09-21 09:44:37 -05:00
|
|
|
cerr << Bug( "Unmanaged configuration: straight line in %s,\n"
|
2013-12-03 18:58:58 -06:00
|
|
|
" The global routing seems to be defective."
|
Bug fixes for MOSIS SCMOS_DEEP support.
* Change: In Hurricane, the NetRoutingProperty is moved into Hurricane
from Katabatic. Needed for Knik to be able to access thoses
informations.
* Change: In Hurricane, in RoutingPad::setOnBestComponent(), now in
case of identical area, select the component of lowest id.
This should not be needed if the component ordering was fully
deterministic as it should be (will investigate later).
This is to ensure that the choosen component is always the
same, especially between save/load of a global routing.
* Bug: In Katabatic, in AutoContactHTee::updateTopology(), invalidate
the segments only if the topology is valid (no NULL in the
cached segments).
* Bug: In Katabatic, in GCellTopology::construct(), throw an error
if the topology is bad instead of trying to continue (and core
dump later... ).
* Bug: In Kite, in BuildPowerRails, distinguish the name of the master
net in the pad (for vddi, vssi, vdde, vsse, ck, cki & cko) and the
name of the net in the *chip* netlist. Must use the later to make
comparison as they may differs.
* Change: In Knik, in save/load solution, exclude nets that are not
globally routed by Knik. That is which NetRoutingProperty is not
*Automatic*.
* Bug: In Cumulus, in chip.BlockPower take account of the layer
width extention to sligthy shrink the connector thus avoiding a
notch with standart cell in some cases.
* Change: In Cumulus, in chip.ClockTree disable the use of fixed Steiner
trees for the leaf clocks, as it seems overconstrained for the
router. First move was to lower them in M2/M3 (instead of M3/M4)
but that was not sufficent.
* New: In Cumulus, RSavePlugin for recursively saving a physical
hierarchy.
* New: In documentation, first embryo for RDS file. Should have been
in Alliance git, but I prefer to keep newest doc in Coriolis.
2014-09-21 09:44:37 -05:00
|
|
|
, _net->_getString().c_str()
|
2013-12-03 18:58:58 -06:00
|
|
|
) << endl;
|
2010-03-09 09:24:29 -06:00
|
|
|
return;
|
2013-12-03 18:58:58 -06:00
|
|
|
} else {
|
|
|
|
if (_sourceContact) {
|
|
|
|
AutoContact* targetContact
|
|
|
|
= ( getSegmentHookType(_fromHook) & (NorthBound|EastBound) )
|
|
|
|
? _northEastContact : _southWestContact ;
|
|
|
|
AutoSegment* globalSegment = AutoSegment::create( _sourceContact
|
|
|
|
, targetContact
|
|
|
|
, static_cast<Segment*>( _fromHook->getComponent() )
|
|
|
|
);
|
|
|
|
globalSegment->setFlags( (_degree == 2) ? SegBipoint : 0 );
|
|
|
|
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "Create global segment: " << globalSegment << endl;
|
2013-12-03 18:58:58 -06:00
|
|
|
|
|
|
|
#if THIS_IS_DEPRECATED
|
|
|
|
if ( globalSegment->isHorizontal()
|
|
|
|
and ( (Session::getRoutingGauge()->getLayerDepth(_sourceContact->getLayer()->getBottom()) > 1)
|
|
|
|
or (Session::getRoutingGauge()->getLayerDepth(targetContact ->getLayer()->getBottom()) > 1)) ) {
|
|
|
|
globalSegment->setLayer ( Session::getRoutingLayer(3) );
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "Source:" << _sourceContact << endl;
|
|
|
|
cdebug_log(145,0) << "Target:" << targetContact << endl;
|
|
|
|
cdebug_log(145,0) << "Moving up global:" << globalSegment << endl;
|
2013-12-03 18:58:58 -06:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
// HARDCODED VALUE.
|
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
|
|
|
if ( (_topology & Global_Fixed) and (globalSegment->getLength() < 2*Session::getSliceHeight()) )
|
2013-12-03 18:58:58 -06:00
|
|
|
_toFixSegments.push_back( globalSegment );
|
|
|
|
|
|
|
|
if (_connexity.fields.globals < 2) {
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_tabw(145,-1);
|
2013-12-03 18:58:58 -06:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else
|
|
|
|
_fromHook = NULL;
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
if ( _east and (_fromHook != _east) ) {
|
|
|
|
Hook* toHook = getSegmentOppositeHook( _east );
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "Pushing East (to) " << getString(toHook) << endl;
|
|
|
|
cdebug_log(145,0) << "Pushing East (from) " << _northEastContact << endl;
|
2013-12-03 18:58:58 -06:00
|
|
|
forks.push( toHook, _northEastContact );
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
2013-12-03 18:58:58 -06:00
|
|
|
if ( _west and (_fromHook != _west) ) {
|
|
|
|
Hook* toHook = getSegmentOppositeHook( _west );
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "Pushing West (to) " << getString(toHook) << endl;
|
|
|
|
cdebug_log(145,0) << "Pushing West (from) " << _southWestContact << endl;
|
2013-12-03 18:58:58 -06:00
|
|
|
forks.push( toHook, _southWestContact );
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
2013-12-03 18:58:58 -06:00
|
|
|
if ( _north and (_fromHook != _north) ) {
|
|
|
|
Hook* toHook = getSegmentOppositeHook( _north );
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "Pushing North (to) " << getString(toHook) << endl;
|
|
|
|
cdebug_log(145,0) << "Pushing North (from) " << _northEastContact << endl;
|
2013-12-03 18:58:58 -06:00
|
|
|
forks.push( toHook, _northEastContact );
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
2013-12-03 18:58:58 -06:00
|
|
|
if ( _south and (_fromHook != _south) ) {
|
|
|
|
Hook* toHook = getSegmentOppositeHook( _south );
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "Pushing South (to) " << getString(toHook) << endl;
|
|
|
|
cdebug_log(145,0) << "Pushing South (from) " << _southWestContact << endl;
|
2013-12-03 18:58:58 -06:00
|
|
|
forks.push( toHook, _southWestContact );
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_tabw(145,-1);
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
void GCellTopology::doRp_AutoContacts ( GCell* gcell
|
|
|
|
, Component* rp
|
|
|
|
, AutoContact*& source
|
|
|
|
, AutoContact*& target
|
|
|
|
, unsigned int flags
|
|
|
|
)
|
2010-03-09 09:24:29 -06:00
|
|
|
{
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,1) << "doRp_AutoContacts()" << endl;
|
|
|
|
cdebug_log(145,0) << rp << endl;
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
source = target = NULL;
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
Point sourcePosition;
|
|
|
|
Point targetPosition;
|
|
|
|
const Layer* rpLayer = rp->getLayer();
|
Improved UpdateSession & exception catching. Start of RoutingGauge implem.
Miscellaneous:
* Change: In <crlcore>, in display.conf use the same display threshold
for both METAL2 & METAL3.
In alliance.conf, the side of VIAs in the gauge is 2l (not 3l).
In kite.conf, separate edge densities for H/V.
* Change: In <Cell>, in flattenNets() use flag as argument, not a
boolean. Do not create rings for clock or supply nets.
* Change: In <DeepNet>, in _createRoutingPads() do not create rings
for clock or supply net (duplicated policy as in Cell::flattenNets()).
* Bug: In <ControllerWidget>, at last find the bad signal disconnect
that was causing ungraceful messages.
* Change: In <knik>, in Edge display occupancy/capacity in the string
name. Improved display progress and debugging capabilities.
Improved exception catch & breakpoint managment:
* Bug: In <PaletteWidget>, in updateExtensions() replace the calls to
deleteLayer() by delete. This cause the widget to be immediatly
erased instead of waiting for the event queue to be completly
processed. This was causing the widget to be left in a incoherent
state when stoping at a breakpoint.
* Bug: In <BreakpointWidget>, in execNoModal(), flush the main event
loop (QApplication::flush()) *before* lauching the *local* event
loop. This is to ensure all widgets are in their final state when
waiting (especially <PaletteWidget>).
* Change: In <ExceptionWidget>, new method catchAllWrapper() to
execute any std::function< void() > function/method with a "try"/
"catch" wraparound and lauch the widget in case something is catch.
* New: In <hurricane>, support for a oberver pattern, backported from
<katabatic> with an Obervable capable of being linked to any
number of Obervers.
* New: In <Cell>, made it observable to detect Cell change, currently
emit two kind of signals:
- Cell::CellAboutToChange : *before* any change.
- Cell::CellChanged : *after* the change has been completed.
* New: In <UpdateSession>, in Go::invalidate() add the Cell owning the
Go to the UPDATOR_STACK (of course the cell is added only once).
In addition, when the Cell is added, send a notification of
Cell::CellAboutToChange to all it's observers. The slave instances
are also invalidated.
Conversely in UpdateSession::_preDestroy() for each invalidated
Cell send a Cell::CellChanged notification to all observer.
The UPDATOR_STACK has been slightly amended to accept Cell which
are not Gos. Prior to this, the Cell where completly excluded from
the UpdateSession mechanism, so it's instances where never actualised
of anything referring to the Cell for that matter.
Note: we use two different mechanisms to transmit a Cell change,
observers and the slave instance map. I think at some point it
should be unificated.
* Change: In <CellViewer>, make it a Cell observer to redraw when the
cell is modificated (also update the palette).
Uses the catchAllWrapper() to protect all critical actions.
* Change: In <GraphicTool>, no longer need of cellPreModificated and
cellPostModificated signals. Now done through the Cell obersvers.
* Change: In <mauka>, <etesian> & <kite> now uses the catchAllWrapper
method for protection (need to split methods in two, to be able
to pass it as argument). No longer emit cellPreModificated and
cellPostModificated.
Support for RoutingGauge in P&R:
* Bug: In <placeandroute.py>, the connection from the internal power
ring to the connectors was not done correctly. Wrong contact layers
leading to a gap.
* Change: In <BuildPowerRails>, detection of the corona signals based
on how the "pck_px" pad is connected. No longer based on name
matching.
* Change: In <placeandroute.py>, support for 2 routing metal only
(3 metal in the technology).
* Change: In <katabatic> & <kite> support for a "top layer" limitation
on the routing gauge, this allows to use only two routing metals
(METAL2 & METAL3). Work in progress.
2014-04-20 12:25:08 -05:00
|
|
|
size_t rpDepth = Session::getLayerDepth( rp->getLayer() );
|
|
|
|
unsigned int direction = Session::getDirection ( rpDepth );
|
|
|
|
DbU::Unit viaSide = Session::getWireWidth ( rpDepth );
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
getPositions( rp, sourcePosition, targetPosition );
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
if (sourcePosition.getX() > targetPosition.getX()) swap( sourcePosition, targetPosition );
|
|
|
|
if (sourcePosition.getY() > targetPosition.getY()) swap( sourcePosition, targetPosition );
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
GCell* sourceGCell = Session::getKatabatic()->getGCellGrid()->getGCell( sourcePosition );
|
|
|
|
GCell* targetGCell = Session::getKatabatic()->getGCellGrid()->getGCell( targetPosition );
|
2010-03-09 09:24:29 -06:00
|
|
|
|
Improved UpdateSession & exception catching. Start of RoutingGauge implem.
Miscellaneous:
* Change: In <crlcore>, in display.conf use the same display threshold
for both METAL2 & METAL3.
In alliance.conf, the side of VIAs in the gauge is 2l (not 3l).
In kite.conf, separate edge densities for H/V.
* Change: In <Cell>, in flattenNets() use flag as argument, not a
boolean. Do not create rings for clock or supply nets.
* Change: In <DeepNet>, in _createRoutingPads() do not create rings
for clock or supply net (duplicated policy as in Cell::flattenNets()).
* Bug: In <ControllerWidget>, at last find the bad signal disconnect
that was causing ungraceful messages.
* Change: In <knik>, in Edge display occupancy/capacity in the string
name. Improved display progress and debugging capabilities.
Improved exception catch & breakpoint managment:
* Bug: In <PaletteWidget>, in updateExtensions() replace the calls to
deleteLayer() by delete. This cause the widget to be immediatly
erased instead of waiting for the event queue to be completly
processed. This was causing the widget to be left in a incoherent
state when stoping at a breakpoint.
* Bug: In <BreakpointWidget>, in execNoModal(), flush the main event
loop (QApplication::flush()) *before* lauching the *local* event
loop. This is to ensure all widgets are in their final state when
waiting (especially <PaletteWidget>).
* Change: In <ExceptionWidget>, new method catchAllWrapper() to
execute any std::function< void() > function/method with a "try"/
"catch" wraparound and lauch the widget in case something is catch.
* New: In <hurricane>, support for a oberver pattern, backported from
<katabatic> with an Obervable capable of being linked to any
number of Obervers.
* New: In <Cell>, made it observable to detect Cell change, currently
emit two kind of signals:
- Cell::CellAboutToChange : *before* any change.
- Cell::CellChanged : *after* the change has been completed.
* New: In <UpdateSession>, in Go::invalidate() add the Cell owning the
Go to the UPDATOR_STACK (of course the cell is added only once).
In addition, when the Cell is added, send a notification of
Cell::CellAboutToChange to all it's observers. The slave instances
are also invalidated.
Conversely in UpdateSession::_preDestroy() for each invalidated
Cell send a Cell::CellChanged notification to all observer.
The UPDATOR_STACK has been slightly amended to accept Cell which
are not Gos. Prior to this, the Cell where completly excluded from
the UpdateSession mechanism, so it's instances where never actualised
of anything referring to the Cell for that matter.
Note: we use two different mechanisms to transmit a Cell change,
observers and the slave instance map. I think at some point it
should be unificated.
* Change: In <CellViewer>, make it a Cell observer to redraw when the
cell is modificated (also update the palette).
Uses the catchAllWrapper() to protect all critical actions.
* Change: In <GraphicTool>, no longer need of cellPreModificated and
cellPostModificated signals. Now done through the Cell obersvers.
* Change: In <mauka>, <etesian> & <kite> now uses the catchAllWrapper
method for protection (need to split methods in two, to be able
to pass it as argument). No longer emit cellPreModificated and
cellPostModificated.
Support for RoutingGauge in P&R:
* Bug: In <placeandroute.py>, the connection from the internal power
ring to the connectors was not done correctly. Wrong contact layers
leading to a gap.
* Change: In <BuildPowerRails>, detection of the corona signals based
on how the "pck_px" pad is connected. No longer based on name
matching.
* Change: In <placeandroute.py>, support for 2 routing metal only
(3 metal in the technology).
* Change: In <katabatic> & <kite> support for a "top layer" limitation
on the routing gauge, this allows to use only two routing metals
(METAL2 & METAL3). Work in progress.
2014-04-20 12:25:08 -05:00
|
|
|
if (rpDepth == 0) {
|
2013-12-03 18:58:58 -06:00
|
|
|
rpLayer = Session::getContactLayer(0);
|
|
|
|
direction = KbHorizontal;
|
Improved UpdateSession & exception catching. Start of RoutingGauge implem.
Miscellaneous:
* Change: In <crlcore>, in display.conf use the same display threshold
for both METAL2 & METAL3.
In alliance.conf, the side of VIAs in the gauge is 2l (not 3l).
In kite.conf, separate edge densities for H/V.
* Change: In <Cell>, in flattenNets() use flag as argument, not a
boolean. Do not create rings for clock or supply nets.
* Change: In <DeepNet>, in _createRoutingPads() do not create rings
for clock or supply net (duplicated policy as in Cell::flattenNets()).
* Bug: In <ControllerWidget>, at last find the bad signal disconnect
that was causing ungraceful messages.
* Change: In <knik>, in Edge display occupancy/capacity in the string
name. Improved display progress and debugging capabilities.
Improved exception catch & breakpoint managment:
* Bug: In <PaletteWidget>, in updateExtensions() replace the calls to
deleteLayer() by delete. This cause the widget to be immediatly
erased instead of waiting for the event queue to be completly
processed. This was causing the widget to be left in a incoherent
state when stoping at a breakpoint.
* Bug: In <BreakpointWidget>, in execNoModal(), flush the main event
loop (QApplication::flush()) *before* lauching the *local* event
loop. This is to ensure all widgets are in their final state when
waiting (especially <PaletteWidget>).
* Change: In <ExceptionWidget>, new method catchAllWrapper() to
execute any std::function< void() > function/method with a "try"/
"catch" wraparound and lauch the widget in case something is catch.
* New: In <hurricane>, support for a oberver pattern, backported from
<katabatic> with an Obervable capable of being linked to any
number of Obervers.
* New: In <Cell>, made it observable to detect Cell change, currently
emit two kind of signals:
- Cell::CellAboutToChange : *before* any change.
- Cell::CellChanged : *after* the change has been completed.
* New: In <UpdateSession>, in Go::invalidate() add the Cell owning the
Go to the UPDATOR_STACK (of course the cell is added only once).
In addition, when the Cell is added, send a notification of
Cell::CellAboutToChange to all it's observers. The slave instances
are also invalidated.
Conversely in UpdateSession::_preDestroy() for each invalidated
Cell send a Cell::CellChanged notification to all observer.
The UPDATOR_STACK has been slightly amended to accept Cell which
are not Gos. Prior to this, the Cell where completly excluded from
the UpdateSession mechanism, so it's instances where never actualised
of anything referring to the Cell for that matter.
Note: we use two different mechanisms to transmit a Cell change,
observers and the slave instance map. I think at some point it
should be unificated.
* Change: In <CellViewer>, make it a Cell observer to redraw when the
cell is modificated (also update the palette).
Uses the catchAllWrapper() to protect all critical actions.
* Change: In <GraphicTool>, no longer need of cellPreModificated and
cellPostModificated signals. Now done through the Cell obersvers.
* Change: In <mauka>, <etesian> & <kite> now uses the catchAllWrapper
method for protection (need to split methods in two, to be able
to pass it as argument). No longer emit cellPreModificated and
cellPostModificated.
Support for RoutingGauge in P&R:
* Bug: In <placeandroute.py>, the connection from the internal power
ring to the connectors was not done correctly. Wrong contact layers
leading to a gap.
* Change: In <BuildPowerRails>, detection of the corona signals based
on how the "pck_px" pad is connected. No longer based on name
matching.
* Change: In <placeandroute.py>, support for 2 routing metal only
(3 metal in the technology).
* Change: In <katabatic> & <kite> support for a "top layer" limitation
on the routing gauge, this allows to use only two routing metals
(METAL2 & METAL3). Work in progress.
2014-04-20 12:25:08 -05:00
|
|
|
viaSide = Session::getViaWidth( rpDepth );
|
2013-12-03 18:58:58 -06:00
|
|
|
}
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
// Non-M1 terminal or punctual M1 protections.
|
Improved UpdateSession & exception catching. Start of RoutingGauge implem.
Miscellaneous:
* Change: In <crlcore>, in display.conf use the same display threshold
for both METAL2 & METAL3.
In alliance.conf, the side of VIAs in the gauge is 2l (not 3l).
In kite.conf, separate edge densities for H/V.
* Change: In <Cell>, in flattenNets() use flag as argument, not a
boolean. Do not create rings for clock or supply nets.
* Change: In <DeepNet>, in _createRoutingPads() do not create rings
for clock or supply net (duplicated policy as in Cell::flattenNets()).
* Bug: In <ControllerWidget>, at last find the bad signal disconnect
that was causing ungraceful messages.
* Change: In <knik>, in Edge display occupancy/capacity in the string
name. Improved display progress and debugging capabilities.
Improved exception catch & breakpoint managment:
* Bug: In <PaletteWidget>, in updateExtensions() replace the calls to
deleteLayer() by delete. This cause the widget to be immediatly
erased instead of waiting for the event queue to be completly
processed. This was causing the widget to be left in a incoherent
state when stoping at a breakpoint.
* Bug: In <BreakpointWidget>, in execNoModal(), flush the main event
loop (QApplication::flush()) *before* lauching the *local* event
loop. This is to ensure all widgets are in their final state when
waiting (especially <PaletteWidget>).
* Change: In <ExceptionWidget>, new method catchAllWrapper() to
execute any std::function< void() > function/method with a "try"/
"catch" wraparound and lauch the widget in case something is catch.
* New: In <hurricane>, support for a oberver pattern, backported from
<katabatic> with an Obervable capable of being linked to any
number of Obervers.
* New: In <Cell>, made it observable to detect Cell change, currently
emit two kind of signals:
- Cell::CellAboutToChange : *before* any change.
- Cell::CellChanged : *after* the change has been completed.
* New: In <UpdateSession>, in Go::invalidate() add the Cell owning the
Go to the UPDATOR_STACK (of course the cell is added only once).
In addition, when the Cell is added, send a notification of
Cell::CellAboutToChange to all it's observers. The slave instances
are also invalidated.
Conversely in UpdateSession::_preDestroy() for each invalidated
Cell send a Cell::CellChanged notification to all observer.
The UPDATOR_STACK has been slightly amended to accept Cell which
are not Gos. Prior to this, the Cell where completly excluded from
the UpdateSession mechanism, so it's instances where never actualised
of anything referring to the Cell for that matter.
Note: we use two different mechanisms to transmit a Cell change,
observers and the slave instance map. I think at some point it
should be unificated.
* Change: In <CellViewer>, make it a Cell observer to redraw when the
cell is modificated (also update the palette).
Uses the catchAllWrapper() to protect all critical actions.
* Change: In <GraphicTool>, no longer need of cellPreModificated and
cellPostModificated signals. Now done through the Cell obersvers.
* Change: In <mauka>, <etesian> & <kite> now uses the catchAllWrapper
method for protection (need to split methods in two, to be able
to pass it as argument). No longer emit cellPreModificated and
cellPostModificated.
Support for RoutingGauge in P&R:
* Bug: In <placeandroute.py>, the connection from the internal power
ring to the connectors was not done correctly. Wrong contact layers
leading to a gap.
* Change: In <BuildPowerRails>, detection of the corona signals based
on how the "pck_px" pad is connected. No longer based on name
matching.
* Change: In <placeandroute.py>, support for 2 routing metal only
(3 metal in the technology).
* Change: In <katabatic> & <kite> support for a "top layer" limitation
on the routing gauge, this allows to use only two routing metals
(METAL2 & METAL3). Work in progress.
2014-04-20 12:25:08 -05:00
|
|
|
if ((rpDepth != 0) or (sourcePosition == targetPosition)) {
|
2013-12-03 18:58:58 -06:00
|
|
|
map<Component*,AutoSegment*>::iterator irp = __routingPadAutoSegments.find( rp );
|
|
|
|
if (irp == __routingPadAutoSegments.end()) {
|
|
|
|
AutoContact* sourceProtect = AutoContactTerminal::create( sourceGCell
|
|
|
|
, rp
|
|
|
|
, rpLayer
|
|
|
|
, sourcePosition
|
Improved UpdateSession & exception catching. Start of RoutingGauge implem.
Miscellaneous:
* Change: In <crlcore>, in display.conf use the same display threshold
for both METAL2 & METAL3.
In alliance.conf, the side of VIAs in the gauge is 2l (not 3l).
In kite.conf, separate edge densities for H/V.
* Change: In <Cell>, in flattenNets() use flag as argument, not a
boolean. Do not create rings for clock or supply nets.
* Change: In <DeepNet>, in _createRoutingPads() do not create rings
for clock or supply net (duplicated policy as in Cell::flattenNets()).
* Bug: In <ControllerWidget>, at last find the bad signal disconnect
that was causing ungraceful messages.
* Change: In <knik>, in Edge display occupancy/capacity in the string
name. Improved display progress and debugging capabilities.
Improved exception catch & breakpoint managment:
* Bug: In <PaletteWidget>, in updateExtensions() replace the calls to
deleteLayer() by delete. This cause the widget to be immediatly
erased instead of waiting for the event queue to be completly
processed. This was causing the widget to be left in a incoherent
state when stoping at a breakpoint.
* Bug: In <BreakpointWidget>, in execNoModal(), flush the main event
loop (QApplication::flush()) *before* lauching the *local* event
loop. This is to ensure all widgets are in their final state when
waiting (especially <PaletteWidget>).
* Change: In <ExceptionWidget>, new method catchAllWrapper() to
execute any std::function< void() > function/method with a "try"/
"catch" wraparound and lauch the widget in case something is catch.
* New: In <hurricane>, support for a oberver pattern, backported from
<katabatic> with an Obervable capable of being linked to any
number of Obervers.
* New: In <Cell>, made it observable to detect Cell change, currently
emit two kind of signals:
- Cell::CellAboutToChange : *before* any change.
- Cell::CellChanged : *after* the change has been completed.
* New: In <UpdateSession>, in Go::invalidate() add the Cell owning the
Go to the UPDATOR_STACK (of course the cell is added only once).
In addition, when the Cell is added, send a notification of
Cell::CellAboutToChange to all it's observers. The slave instances
are also invalidated.
Conversely in UpdateSession::_preDestroy() for each invalidated
Cell send a Cell::CellChanged notification to all observer.
The UPDATOR_STACK has been slightly amended to accept Cell which
are not Gos. Prior to this, the Cell where completly excluded from
the UpdateSession mechanism, so it's instances where never actualised
of anything referring to the Cell for that matter.
Note: we use two different mechanisms to transmit a Cell change,
observers and the slave instance map. I think at some point it
should be unificated.
* Change: In <CellViewer>, make it a Cell observer to redraw when the
cell is modificated (also update the palette).
Uses the catchAllWrapper() to protect all critical actions.
* Change: In <GraphicTool>, no longer need of cellPreModificated and
cellPostModificated signals. Now done through the Cell obersvers.
* Change: In <mauka>, <etesian> & <kite> now uses the catchAllWrapper
method for protection (need to split methods in two, to be able
to pass it as argument). No longer emit cellPreModificated and
cellPostModificated.
Support for RoutingGauge in P&R:
* Bug: In <placeandroute.py>, the connection from the internal power
ring to the connectors was not done correctly. Wrong contact layers
leading to a gap.
* Change: In <BuildPowerRails>, detection of the corona signals based
on how the "pck_px" pad is connected. No longer based on name
matching.
* Change: In <placeandroute.py>, support for 2 routing metal only
(3 metal in the technology).
* Change: In <katabatic> & <kite> support for a "top layer" limitation
on the routing gauge, this allows to use only two routing metals
(METAL2 & METAL3). Work in progress.
2014-04-20 12:25:08 -05:00
|
|
|
, viaSide, viaSide
|
2013-12-03 18:58:58 -06:00
|
|
|
);
|
|
|
|
AutoContact* targetProtect = AutoContactTerminal::create( targetGCell
|
|
|
|
, rp
|
|
|
|
, rpLayer
|
|
|
|
, targetPosition
|
Improved UpdateSession & exception catching. Start of RoutingGauge implem.
Miscellaneous:
* Change: In <crlcore>, in display.conf use the same display threshold
for both METAL2 & METAL3.
In alliance.conf, the side of VIAs in the gauge is 2l (not 3l).
In kite.conf, separate edge densities for H/V.
* Change: In <Cell>, in flattenNets() use flag as argument, not a
boolean. Do not create rings for clock or supply nets.
* Change: In <DeepNet>, in _createRoutingPads() do not create rings
for clock or supply net (duplicated policy as in Cell::flattenNets()).
* Bug: In <ControllerWidget>, at last find the bad signal disconnect
that was causing ungraceful messages.
* Change: In <knik>, in Edge display occupancy/capacity in the string
name. Improved display progress and debugging capabilities.
Improved exception catch & breakpoint managment:
* Bug: In <PaletteWidget>, in updateExtensions() replace the calls to
deleteLayer() by delete. This cause the widget to be immediatly
erased instead of waiting for the event queue to be completly
processed. This was causing the widget to be left in a incoherent
state when stoping at a breakpoint.
* Bug: In <BreakpointWidget>, in execNoModal(), flush the main event
loop (QApplication::flush()) *before* lauching the *local* event
loop. This is to ensure all widgets are in their final state when
waiting (especially <PaletteWidget>).
* Change: In <ExceptionWidget>, new method catchAllWrapper() to
execute any std::function< void() > function/method with a "try"/
"catch" wraparound and lauch the widget in case something is catch.
* New: In <hurricane>, support for a oberver pattern, backported from
<katabatic> with an Obervable capable of being linked to any
number of Obervers.
* New: In <Cell>, made it observable to detect Cell change, currently
emit two kind of signals:
- Cell::CellAboutToChange : *before* any change.
- Cell::CellChanged : *after* the change has been completed.
* New: In <UpdateSession>, in Go::invalidate() add the Cell owning the
Go to the UPDATOR_STACK (of course the cell is added only once).
In addition, when the Cell is added, send a notification of
Cell::CellAboutToChange to all it's observers. The slave instances
are also invalidated.
Conversely in UpdateSession::_preDestroy() for each invalidated
Cell send a Cell::CellChanged notification to all observer.
The UPDATOR_STACK has been slightly amended to accept Cell which
are not Gos. Prior to this, the Cell where completly excluded from
the UpdateSession mechanism, so it's instances where never actualised
of anything referring to the Cell for that matter.
Note: we use two different mechanisms to transmit a Cell change,
observers and the slave instance map. I think at some point it
should be unificated.
* Change: In <CellViewer>, make it a Cell observer to redraw when the
cell is modificated (also update the palette).
Uses the catchAllWrapper() to protect all critical actions.
* Change: In <GraphicTool>, no longer need of cellPreModificated and
cellPostModificated signals. Now done through the Cell obersvers.
* Change: In <mauka>, <etesian> & <kite> now uses the catchAllWrapper
method for protection (need to split methods in two, to be able
to pass it as argument). No longer emit cellPreModificated and
cellPostModificated.
Support for RoutingGauge in P&R:
* Bug: In <placeandroute.py>, the connection from the internal power
ring to the connectors was not done correctly. Wrong contact layers
leading to a gap.
* Change: In <BuildPowerRails>, detection of the corona signals based
on how the "pck_px" pad is connected. No longer based on name
matching.
* Change: In <placeandroute.py>, support for 2 routing metal only
(3 metal in the technology).
* Change: In <katabatic> & <kite> support for a "top layer" limitation
on the routing gauge, this allows to use only two routing metals
(METAL2 & METAL3). Work in progress.
2014-04-20 12:25:08 -05:00
|
|
|
, viaSide, viaSide
|
2013-12-03 18:58:58 -06:00
|
|
|
);
|
|
|
|
sourceProtect->setFlags( CntFixed );
|
|
|
|
targetProtect->setFlags( CntFixed );
|
|
|
|
|
|
|
|
AutoSegment* segment = AutoSegment::create( sourceProtect, targetProtect, direction );
|
|
|
|
segment->setFlags( SegFixed );
|
|
|
|
|
|
|
|
__routingPadAutoSegments.insert( make_pair(rp,segment) );
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
2013-12-03 18:58:58 -06:00
|
|
|
}
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
if (sourcePosition != targetPosition) {
|
|
|
|
if (flags & DoSourceContact)
|
|
|
|
source = AutoContactTerminal::create( sourceGCell
|
|
|
|
, rp
|
|
|
|
, rpLayer
|
|
|
|
, sourcePosition
|
Improved UpdateSession & exception catching. Start of RoutingGauge implem.
Miscellaneous:
* Change: In <crlcore>, in display.conf use the same display threshold
for both METAL2 & METAL3.
In alliance.conf, the side of VIAs in the gauge is 2l (not 3l).
In kite.conf, separate edge densities for H/V.
* Change: In <Cell>, in flattenNets() use flag as argument, not a
boolean. Do not create rings for clock or supply nets.
* Change: In <DeepNet>, in _createRoutingPads() do not create rings
for clock or supply net (duplicated policy as in Cell::flattenNets()).
* Bug: In <ControllerWidget>, at last find the bad signal disconnect
that was causing ungraceful messages.
* Change: In <knik>, in Edge display occupancy/capacity in the string
name. Improved display progress and debugging capabilities.
Improved exception catch & breakpoint managment:
* Bug: In <PaletteWidget>, in updateExtensions() replace the calls to
deleteLayer() by delete. This cause the widget to be immediatly
erased instead of waiting for the event queue to be completly
processed. This was causing the widget to be left in a incoherent
state when stoping at a breakpoint.
* Bug: In <BreakpointWidget>, in execNoModal(), flush the main event
loop (QApplication::flush()) *before* lauching the *local* event
loop. This is to ensure all widgets are in their final state when
waiting (especially <PaletteWidget>).
* Change: In <ExceptionWidget>, new method catchAllWrapper() to
execute any std::function< void() > function/method with a "try"/
"catch" wraparound and lauch the widget in case something is catch.
* New: In <hurricane>, support for a oberver pattern, backported from
<katabatic> with an Obervable capable of being linked to any
number of Obervers.
* New: In <Cell>, made it observable to detect Cell change, currently
emit two kind of signals:
- Cell::CellAboutToChange : *before* any change.
- Cell::CellChanged : *after* the change has been completed.
* New: In <UpdateSession>, in Go::invalidate() add the Cell owning the
Go to the UPDATOR_STACK (of course the cell is added only once).
In addition, when the Cell is added, send a notification of
Cell::CellAboutToChange to all it's observers. The slave instances
are also invalidated.
Conversely in UpdateSession::_preDestroy() for each invalidated
Cell send a Cell::CellChanged notification to all observer.
The UPDATOR_STACK has been slightly amended to accept Cell which
are not Gos. Prior to this, the Cell where completly excluded from
the UpdateSession mechanism, so it's instances where never actualised
of anything referring to the Cell for that matter.
Note: we use two different mechanisms to transmit a Cell change,
observers and the slave instance map. I think at some point it
should be unificated.
* Change: In <CellViewer>, make it a Cell observer to redraw when the
cell is modificated (also update the palette).
Uses the catchAllWrapper() to protect all critical actions.
* Change: In <GraphicTool>, no longer need of cellPreModificated and
cellPostModificated signals. Now done through the Cell obersvers.
* Change: In <mauka>, <etesian> & <kite> now uses the catchAllWrapper
method for protection (need to split methods in two, to be able
to pass it as argument). No longer emit cellPreModificated and
cellPostModificated.
Support for RoutingGauge in P&R:
* Bug: In <placeandroute.py>, the connection from the internal power
ring to the connectors was not done correctly. Wrong contact layers
leading to a gap.
* Change: In <BuildPowerRails>, detection of the corona signals based
on how the "pck_px" pad is connected. No longer based on name
matching.
* Change: In <placeandroute.py>, support for 2 routing metal only
(3 metal in the technology).
* Change: In <katabatic> & <kite> support for a "top layer" limitation
on the routing gauge, this allows to use only two routing metals
(METAL2 & METAL3). Work in progress.
2014-04-20 12:25:08 -05:00
|
|
|
, viaSide, viaSide
|
2013-12-03 18:58:58 -06:00
|
|
|
);
|
|
|
|
if (flags & DoTargetContact)
|
|
|
|
target = AutoContactTerminal::create( targetGCell
|
|
|
|
, rp
|
|
|
|
, rpLayer
|
|
|
|
, targetPosition
|
Improved UpdateSession & exception catching. Start of RoutingGauge implem.
Miscellaneous:
* Change: In <crlcore>, in display.conf use the same display threshold
for both METAL2 & METAL3.
In alliance.conf, the side of VIAs in the gauge is 2l (not 3l).
In kite.conf, separate edge densities for H/V.
* Change: In <Cell>, in flattenNets() use flag as argument, not a
boolean. Do not create rings for clock or supply nets.
* Change: In <DeepNet>, in _createRoutingPads() do not create rings
for clock or supply net (duplicated policy as in Cell::flattenNets()).
* Bug: In <ControllerWidget>, at last find the bad signal disconnect
that was causing ungraceful messages.
* Change: In <knik>, in Edge display occupancy/capacity in the string
name. Improved display progress and debugging capabilities.
Improved exception catch & breakpoint managment:
* Bug: In <PaletteWidget>, in updateExtensions() replace the calls to
deleteLayer() by delete. This cause the widget to be immediatly
erased instead of waiting for the event queue to be completly
processed. This was causing the widget to be left in a incoherent
state when stoping at a breakpoint.
* Bug: In <BreakpointWidget>, in execNoModal(), flush the main event
loop (QApplication::flush()) *before* lauching the *local* event
loop. This is to ensure all widgets are in their final state when
waiting (especially <PaletteWidget>).
* Change: In <ExceptionWidget>, new method catchAllWrapper() to
execute any std::function< void() > function/method with a "try"/
"catch" wraparound and lauch the widget in case something is catch.
* New: In <hurricane>, support for a oberver pattern, backported from
<katabatic> with an Obervable capable of being linked to any
number of Obervers.
* New: In <Cell>, made it observable to detect Cell change, currently
emit two kind of signals:
- Cell::CellAboutToChange : *before* any change.
- Cell::CellChanged : *after* the change has been completed.
* New: In <UpdateSession>, in Go::invalidate() add the Cell owning the
Go to the UPDATOR_STACK (of course the cell is added only once).
In addition, when the Cell is added, send a notification of
Cell::CellAboutToChange to all it's observers. The slave instances
are also invalidated.
Conversely in UpdateSession::_preDestroy() for each invalidated
Cell send a Cell::CellChanged notification to all observer.
The UPDATOR_STACK has been slightly amended to accept Cell which
are not Gos. Prior to this, the Cell where completly excluded from
the UpdateSession mechanism, so it's instances where never actualised
of anything referring to the Cell for that matter.
Note: we use two different mechanisms to transmit a Cell change,
observers and the slave instance map. I think at some point it
should be unificated.
* Change: In <CellViewer>, make it a Cell observer to redraw when the
cell is modificated (also update the palette).
Uses the catchAllWrapper() to protect all critical actions.
* Change: In <GraphicTool>, no longer need of cellPreModificated and
cellPostModificated signals. Now done through the Cell obersvers.
* Change: In <mauka>, <etesian> & <kite> now uses the catchAllWrapper
method for protection (need to split methods in two, to be able
to pass it as argument). No longer emit cellPreModificated and
cellPostModificated.
Support for RoutingGauge in P&R:
* Bug: In <placeandroute.py>, the connection from the internal power
ring to the connectors was not done correctly. Wrong contact layers
leading to a gap.
* Change: In <BuildPowerRails>, detection of the corona signals based
on how the "pck_px" pad is connected. No longer based on name
matching.
* Change: In <placeandroute.py>, support for 2 routing metal only
(3 metal in the technology).
* Change: In <katabatic> & <kite> support for a "top layer" limitation
on the routing gauge, this allows to use only two routing metals
(METAL2 & METAL3). Work in progress.
2014-04-20 12:25:08 -05:00
|
|
|
, viaSide, viaSide
|
2013-12-03 18:58:58 -06:00
|
|
|
);
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
if (not source and not target) {
|
|
|
|
source = target = AutoContactTerminal::create( gcell
|
|
|
|
, rp
|
|
|
|
, rpLayer
|
|
|
|
, rp->getCenter()
|
Improved UpdateSession & exception catching. Start of RoutingGauge implem.
Miscellaneous:
* Change: In <crlcore>, in display.conf use the same display threshold
for both METAL2 & METAL3.
In alliance.conf, the side of VIAs in the gauge is 2l (not 3l).
In kite.conf, separate edge densities for H/V.
* Change: In <Cell>, in flattenNets() use flag as argument, not a
boolean. Do not create rings for clock or supply nets.
* Change: In <DeepNet>, in _createRoutingPads() do not create rings
for clock or supply net (duplicated policy as in Cell::flattenNets()).
* Bug: In <ControllerWidget>, at last find the bad signal disconnect
that was causing ungraceful messages.
* Change: In <knik>, in Edge display occupancy/capacity in the string
name. Improved display progress and debugging capabilities.
Improved exception catch & breakpoint managment:
* Bug: In <PaletteWidget>, in updateExtensions() replace the calls to
deleteLayer() by delete. This cause the widget to be immediatly
erased instead of waiting for the event queue to be completly
processed. This was causing the widget to be left in a incoherent
state when stoping at a breakpoint.
* Bug: In <BreakpointWidget>, in execNoModal(), flush the main event
loop (QApplication::flush()) *before* lauching the *local* event
loop. This is to ensure all widgets are in their final state when
waiting (especially <PaletteWidget>).
* Change: In <ExceptionWidget>, new method catchAllWrapper() to
execute any std::function< void() > function/method with a "try"/
"catch" wraparound and lauch the widget in case something is catch.
* New: In <hurricane>, support for a oberver pattern, backported from
<katabatic> with an Obervable capable of being linked to any
number of Obervers.
* New: In <Cell>, made it observable to detect Cell change, currently
emit two kind of signals:
- Cell::CellAboutToChange : *before* any change.
- Cell::CellChanged : *after* the change has been completed.
* New: In <UpdateSession>, in Go::invalidate() add the Cell owning the
Go to the UPDATOR_STACK (of course the cell is added only once).
In addition, when the Cell is added, send a notification of
Cell::CellAboutToChange to all it's observers. The slave instances
are also invalidated.
Conversely in UpdateSession::_preDestroy() for each invalidated
Cell send a Cell::CellChanged notification to all observer.
The UPDATOR_STACK has been slightly amended to accept Cell which
are not Gos. Prior to this, the Cell where completly excluded from
the UpdateSession mechanism, so it's instances where never actualised
of anything referring to the Cell for that matter.
Note: we use two different mechanisms to transmit a Cell change,
observers and the slave instance map. I think at some point it
should be unificated.
* Change: In <CellViewer>, make it a Cell observer to redraw when the
cell is modificated (also update the palette).
Uses the catchAllWrapper() to protect all critical actions.
* Change: In <GraphicTool>, no longer need of cellPreModificated and
cellPostModificated signals. Now done through the Cell obersvers.
* Change: In <mauka>, <etesian> & <kite> now uses the catchAllWrapper
method for protection (need to split methods in two, to be able
to pass it as argument). No longer emit cellPreModificated and
cellPostModificated.
Support for RoutingGauge in P&R:
* Bug: In <placeandroute.py>, the connection from the internal power
ring to the connectors was not done correctly. Wrong contact layers
leading to a gap.
* Change: In <BuildPowerRails>, detection of the corona signals based
on how the "pck_px" pad is connected. No longer based on name
matching.
* Change: In <placeandroute.py>, support for 2 routing metal only
(3 metal in the technology).
* Change: In <katabatic> & <kite> support for a "top layer" limitation
on the routing gauge, this allows to use only two routing metals
(METAL2 & METAL3). Work in progress.
2014-04-20 12:25:08 -05:00
|
|
|
, viaSide, viaSide
|
2010-03-09 09:24:29 -06:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_tabw(145,-1);
|
2013-12-03 18:58:58 -06:00
|
|
|
return;
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
AutoContact* GCellTopology::doRp_Access ( GCell* gcell, Component* rp, unsigned int flags )
|
2010-03-09 09:24:29 -06:00
|
|
|
{
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,1) << "doRp_Access() - flags:" << flags << endl;
|
2010-03-09 09:24:29 -06:00
|
|
|
|
|
|
|
AutoContact* rpContactSource;
|
|
|
|
AutoContact* rpContactTarget;
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
flags |= checkRoutingPadSize( rp );
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
doRp_AutoContacts( gcell, rp, rpContactSource, rpContactTarget, flags );
|
2010-03-09 09:24:29 -06:00
|
|
|
|
ExtensionCap support and source/target terminal flags in Katabatic & Kite.
Placement management:
* Change: In <metis>, always disable the hMetis support regardless of
it being detected or not as the placer is still unable manage the
final bin contents.
Routing gauge management:
* Bug: In CRL Core, <vsclib/alliance.conf>, set the correct pitches and
size for the routing layers and the cell gauge.
* Change: In Katabatic & Kite, extract the correct extension cap for each
routing layer from the layers characteristics (cache then in
Katabatic::Configuration).
* Change: In Katabatic, <AutoSegment>, create segment with the wire width
defined in the gauge. For AutoSegment created on already existing
Segment from the global routing, adjust the width.
* Change: In Katabatic, <AutoSegment>, more accurate information about how
a segment is connected to terminal via source and/or target.
The flag SegStrongTerminal is splitted into SegSourceTerminal and
SegSourceTarget (but still used as a mask). So now we can know by
which end an AutoSegment is connected to a terminal.
* Change: In Katabatic, ::doRp_Access(), create constraint freeing segments
not only when HSmall but also when VSmall (more critical for <vsclib>).
Otherwise we may see AutoSegments with incompatible source/target
constraints.
* Change: In Kite, BuildPowerRails, do not create blockage on PinOnly
layers *but* still create power rails planes. This is a temporary
workaround for <vsclib> where the METAL1 blockages overlaps the
terminals (it was fine for Nero, but they shouldn't for Kite).
* Change: In Kite, <RoutingEvent>, if a TrackSegment is overconstrained,
directly bybass it's slackening state to DataNegociate::Slacken.
Also rename the flag "_canHandleConstraints" to "_overConstrained",
seems clearer to me.
Miscellaneous:
* Change: In CRL Core, <Utilities>, add a "pass-though" capability on the
mstream to temporarily make them print everything.
2014-05-25 08:00:35 -05:00
|
|
|
if (flags & HAccess) {
|
|
|
|
if (flags & VSmall) {
|
|
|
|
AutoContact* subContact1 = AutoContactTurn::create( gcell, rp->getNet(), Session::getContactLayer(1) );
|
|
|
|
AutoContact* subContact2 = AutoContactTurn::create( gcell, rp->getNet(), Session::getContactLayer(1) );
|
|
|
|
AutoSegment::create( rpContactSource, subContact1, KbHorizontal );
|
|
|
|
AutoSegment::create( subContact1, subContact2, KbVertical );
|
|
|
|
rpContactSource = subContact2;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (flags & HSmall) {
|
|
|
|
AutoContact* subContact1 = AutoContactTurn::create( gcell, rp->getNet(), Session::getContactLayer(1) );
|
|
|
|
AutoSegment::create( rpContactSource, subContact1, KbHorizontal );
|
|
|
|
rpContactSource = subContact1;
|
|
|
|
}
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_tabw(145,-1);
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
return rpContactSource;
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
Corrections in how to access pads terminals in Katabatic/Kite.
* Change: In Katabatic, in GCellTopology::doRp_AccessPad(), if the supporting
RoutingPad is big (more than two pitch), do not put the access
contact in the center but on the edge. This is to avoid cut violations
between the VIAs and the matrix of VIAs that may be generated under the
RoutingPad itself.
* Change: In Kite, in TrackSegment destructor, if the legnth of the wire,
without extensions is less than one picth, enlarge it so it encompass
it's source & target VIAs (to avoid notches).
* Bug: In Kite, in Track destructor, the TrackElements where detacheds,
but not deleted, causing a memory link.
2016-03-02 10:15:58 -06:00
|
|
|
AutoContact* GCellTopology::doRp_AccessPad ( RoutingPad* rp, unsigned int flags )
|
Added support for "same layer" dogleg. Big fix for pad routing.
* Change: In Knik, in Vertex, add a "blocked" flag to signal disabled
vertexes in the grid (must not be used by the global router).
Modificate the Graph::getVertex() method so that when a vertex
is geometrically queried, if is a blocked one, return a non-blocked
neighbor. This mechanism is introduced to, at last, prevent the
global router to go *under* the pad in case of a commplete chip.
* New: In Katabatic, in AutoSegment, a new state has been added: "reduced".
A reduced segment is in the same layer as it's perpandiculars.
To be reduced, a segments has to be connected on source & target to
AutoContactTurn, both of the perpandiculars must be of the same layer
(below or above) and it's length must not exceed one pitch in the
perpandicular direction.
To reduce an AutoSegment, call ::reduce() and to revert the state,
call ::raise(). Two associated predicates are associated:
::canReduce() and ::mustRaise().
Note: No two adjacent segments can be reduced at the same time.
* Bug: In Katabatic, in GCellTopology, add a new method ::doRp_AccessPad()
to connect to the pads. Create wiring, fixed and non managed by
Katabatic, to connect the pad connector layer to the lowest routing
layers (depth 1 & 2). The former implementation was sometimes leading
to gaps (sheared contact) that *must not* occurs during the building
stage.
Remark: This bug did put under the light the fact that the initial
wiring must be created without gaps. Gaps are closed by making doglegs
on contacts. But this mechanism could only work when the database if
fully initialised (the cache is up to date). Otherwise various problems
arise, in the canonization process for example.
* New: In Katabatic, in AutoContactTerminal::getNativeConstraintBox(),
when anchored on a RoutingPad, now take account the potential rotation
of the Path's transformation. Here again, for the chip's pads.
* New: In Kite, support for reduced AutoSegment. TrackSegment associateds
to reduced AutoSegment are *not* inserted into track to become
effectively invisibles. When a segment becomes reduced, a TrackEvent
is generated to remove it. Conversely when it is raised a RoutingEvent
is created/rescheduled to insert it. All this is mostly managed inside
the Session::revalidate() method.
* New: In Kite, in KiteEngine::createGlobalGraph(), in case of a chip,
mark all global routing vertexes (Knik) that are under a pad, as blockeds.
* Bug: In Cumulus, in PadsCorona.Side.getAxis(), inversion between X and
Y coordinate of the chip size. Did not show until a non-square chip
was routed (i.e. our MIPS R3000).
* Change: In Stratus1, in st_placement.py add the ClockBuffer class for
backward compatibility with the MIPS32 bench. Have to review this
functionnality coming from the deprecated placeAndroute.py.
In st_instance.py, no longer creates the Plug ring of a Net.
In my opinion it just clutter the display until the P&R is called.
Can re-enable later as an option (in Unicorn).
* Change: In Unicorn, in cgt.py, more reliable way of loading then running
user supplied scripts. Borrowed from alliance-checker-toolkit doChip.py .
2015-08-16 16:29:28 -05:00
|
|
|
{
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,1) << "doRp_AccessPad()" << endl;
|
|
|
|
cdebug_log(145,0) << rp << endl;
|
Added support for "same layer" dogleg. Big fix for pad routing.
* Change: In Knik, in Vertex, add a "blocked" flag to signal disabled
vertexes in the grid (must not be used by the global router).
Modificate the Graph::getVertex() method so that when a vertex
is geometrically queried, if is a blocked one, return a non-blocked
neighbor. This mechanism is introduced to, at last, prevent the
global router to go *under* the pad in case of a commplete chip.
* New: In Katabatic, in AutoSegment, a new state has been added: "reduced".
A reduced segment is in the same layer as it's perpandiculars.
To be reduced, a segments has to be connected on source & target to
AutoContactTurn, both of the perpandiculars must be of the same layer
(below or above) and it's length must not exceed one pitch in the
perpandicular direction.
To reduce an AutoSegment, call ::reduce() and to revert the state,
call ::raise(). Two associated predicates are associated:
::canReduce() and ::mustRaise().
Note: No two adjacent segments can be reduced at the same time.
* Bug: In Katabatic, in GCellTopology, add a new method ::doRp_AccessPad()
to connect to the pads. Create wiring, fixed and non managed by
Katabatic, to connect the pad connector layer to the lowest routing
layers (depth 1 & 2). The former implementation was sometimes leading
to gaps (sheared contact) that *must not* occurs during the building
stage.
Remark: This bug did put under the light the fact that the initial
wiring must be created without gaps. Gaps are closed by making doglegs
on contacts. But this mechanism could only work when the database if
fully initialised (the cache is up to date). Otherwise various problems
arise, in the canonization process for example.
* New: In Katabatic, in AutoContactTerminal::getNativeConstraintBox(),
when anchored on a RoutingPad, now take account the potential rotation
of the Path's transformation. Here again, for the chip's pads.
* New: In Kite, support for reduced AutoSegment. TrackSegment associateds
to reduced AutoSegment are *not* inserted into track to become
effectively invisibles. When a segment becomes reduced, a TrackEvent
is generated to remove it. Conversely when it is raised a RoutingEvent
is created/rescheduled to insert it. All this is mostly managed inside
the Session::revalidate() method.
* New: In Kite, in KiteEngine::createGlobalGraph(), in case of a chip,
mark all global routing vertexes (Knik) that are under a pad, as blockeds.
* Bug: In Cumulus, in PadsCorona.Side.getAxis(), inversion between X and
Y coordinate of the chip size. Did not show until a non-square chip
was routed (i.e. our MIPS R3000).
* Change: In Stratus1, in st_placement.py add the ClockBuffer class for
backward compatibility with the MIPS32 bench. Have to review this
functionnality coming from the deprecated placeAndroute.py.
In st_instance.py, no longer creates the Plug ring of a Net.
In my opinion it just clutter the display until the P&R is called.
Can re-enable later as an option (in Unicorn).
* Change: In Unicorn, in cgt.py, more reliable way of loading then running
user supplied scripts. Borrowed from alliance-checker-toolkit doChip.py .
2015-08-16 16:29:28 -05:00
|
|
|
|
|
|
|
// Hardcoded: H access is METAL2 (depth=1), V access is METAL3 (depth=2).
|
|
|
|
size_t accessDepth = (flags & HAccess) ? 1 : 2 ;
|
|
|
|
size_t padDepth = Session::getLayerDepth(rp->getLayer());
|
|
|
|
if (padDepth > Session::getAllowedDepth()) {
|
|
|
|
cerr << Error( "GCellTopology::doRp_AccessPad(): Pad RoutingPad %s\n"
|
|
|
|
" has a layer unreachable by the router (top layer is: %s)"
|
|
|
|
, getString(rp).c_str()
|
|
|
|
, getString(Session::getRoutingLayer(Session::getAllowedDepth())).c_str()
|
|
|
|
) << endl;
|
|
|
|
padDepth = Session::getAllowedDepth();
|
|
|
|
}
|
|
|
|
|
|
|
|
rp->getBodyHook()->detach();
|
|
|
|
|
Corrections in how to access pads terminals in Katabatic/Kite.
* Change: In Katabatic, in GCellTopology::doRp_AccessPad(), if the supporting
RoutingPad is big (more than two pitch), do not put the access
contact in the center but on the edge. This is to avoid cut violations
between the VIAs and the matrix of VIAs that may be generated under the
RoutingPad itself.
* Change: In Kite, in TrackSegment destructor, if the legnth of the wire,
without extensions is less than one picth, enlarge it so it encompass
it's source & target VIAs (to avoid notches).
* Bug: In Kite, in Track destructor, the TrackElements where detacheds,
but not deleted, causing a memory link.
2016-03-02 10:15:58 -06:00
|
|
|
Point rpPosition = rp->getCenter();
|
|
|
|
Point position = rp->getCenter();
|
|
|
|
Box rpbb = rp->getBoundingBox();
|
|
|
|
if ( (rpbb.getWidth () > 2*Session::getWireWidth(padDepth))
|
|
|
|
or (rpbb.getHeight() > 2*Session::getWireWidth(padDepth)) ) {
|
|
|
|
//cerr << "doRp_AccessPad(): connecting to non-punctual connector (RoutingPad).\n"
|
|
|
|
// << " " << rp->getNet() << "pad:" << rp->getOccurrence().getMasterCell() << endl;
|
|
|
|
|
|
|
|
Transformation transf = rp->getOccurrence().getPath().getTransformation();
|
|
|
|
switch ( transf.getOrientation() ) {
|
|
|
|
case Transformation::Orientation::ID: position.setY( rpbb.getYMin() ); break;
|
|
|
|
case Transformation::Orientation::MY: position.setY( rpbb.getYMax() ); break;
|
|
|
|
case Transformation::Orientation::YR:
|
|
|
|
case Transformation::Orientation::R3: position.setX( rpbb.getXMin() ); break;
|
|
|
|
case Transformation::Orientation::R1: position.setX( rpbb.getXMax() ); break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
GCell* gcell = Session::getKatabatic()->getGCellGrid()->getGCell(position);
|
|
|
|
Component* anchor = rp;
|
Added support for "same layer" dogleg. Big fix for pad routing.
* Change: In Knik, in Vertex, add a "blocked" flag to signal disabled
vertexes in the grid (must not be used by the global router).
Modificate the Graph::getVertex() method so that when a vertex
is geometrically queried, if is a blocked one, return a non-blocked
neighbor. This mechanism is introduced to, at last, prevent the
global router to go *under* the pad in case of a commplete chip.
* New: In Katabatic, in AutoSegment, a new state has been added: "reduced".
A reduced segment is in the same layer as it's perpandiculars.
To be reduced, a segments has to be connected on source & target to
AutoContactTurn, both of the perpandiculars must be of the same layer
(below or above) and it's length must not exceed one pitch in the
perpandicular direction.
To reduce an AutoSegment, call ::reduce() and to revert the state,
call ::raise(). Two associated predicates are associated:
::canReduce() and ::mustRaise().
Note: No two adjacent segments can be reduced at the same time.
* Bug: In Katabatic, in GCellTopology, add a new method ::doRp_AccessPad()
to connect to the pads. Create wiring, fixed and non managed by
Katabatic, to connect the pad connector layer to the lowest routing
layers (depth 1 & 2). The former implementation was sometimes leading
to gaps (sheared contact) that *must not* occurs during the building
stage.
Remark: This bug did put under the light the fact that the initial
wiring must be created without gaps. Gaps are closed by making doglegs
on contacts. But this mechanism could only work when the database if
fully initialised (the cache is up to date). Otherwise various problems
arise, in the canonization process for example.
* New: In Katabatic, in AutoContactTerminal::getNativeConstraintBox(),
when anchored on a RoutingPad, now take account the potential rotation
of the Path's transformation. Here again, for the chip's pads.
* New: In Kite, support for reduced AutoSegment. TrackSegment associateds
to reduced AutoSegment are *not* inserted into track to become
effectively invisibles. When a segment becomes reduced, a TrackEvent
is generated to remove it. Conversely when it is raised a RoutingEvent
is created/rescheduled to insert it. All this is mostly managed inside
the Session::revalidate() method.
* New: In Kite, in KiteEngine::createGlobalGraph(), in case of a chip,
mark all global routing vertexes (Knik) that are under a pad, as blockeds.
* Bug: In Cumulus, in PadsCorona.Side.getAxis(), inversion between X and
Y coordinate of the chip size. Did not show until a non-square chip
was routed (i.e. our MIPS R3000).
* Change: In Stratus1, in st_placement.py add the ClockBuffer class for
backward compatibility with the MIPS32 bench. Have to review this
functionnality coming from the deprecated placeAndroute.py.
In st_instance.py, no longer creates the Plug ring of a Net.
In my opinion it just clutter the display until the P&R is called.
Can re-enable later as an option (in Unicorn).
* Change: In Unicorn, in cgt.py, more reliable way of loading then running
user supplied scripts. Borrowed from alliance-checker-toolkit doChip.py .
2015-08-16 16:29:28 -05:00
|
|
|
|
|
|
|
if (padDepth != accessDepth) {
|
|
|
|
if (padDepth > accessDepth) {
|
|
|
|
// Go *down* from the pad's RoutingPad.
|
|
|
|
--padDepth;
|
|
|
|
|
|
|
|
Contact* target = NULL;
|
|
|
|
Contact* source = Contact::create ( rp
|
|
|
|
, Session::getContactLayer(padDepth)
|
Corrections in how to access pads terminals in Katabatic/Kite.
* Change: In Katabatic, in GCellTopology::doRp_AccessPad(), if the supporting
RoutingPad is big (more than two pitch), do not put the access
contact in the center but on the edge. This is to avoid cut violations
between the VIAs and the matrix of VIAs that may be generated under the
RoutingPad itself.
* Change: In Kite, in TrackSegment destructor, if the legnth of the wire,
without extensions is less than one picth, enlarge it so it encompass
it's source & target VIAs (to avoid notches).
* Bug: In Kite, in Track destructor, the TrackElements where detacheds,
but not deleted, causing a memory link.
2016-03-02 10:15:58 -06:00
|
|
|
, position.getX() - rpPosition.getX()
|
|
|
|
, position.getY() - rpPosition.getY()
|
Added support for "same layer" dogleg. Big fix for pad routing.
* Change: In Knik, in Vertex, add a "blocked" flag to signal disabled
vertexes in the grid (must not be used by the global router).
Modificate the Graph::getVertex() method so that when a vertex
is geometrically queried, if is a blocked one, return a non-blocked
neighbor. This mechanism is introduced to, at last, prevent the
global router to go *under* the pad in case of a commplete chip.
* New: In Katabatic, in AutoSegment, a new state has been added: "reduced".
A reduced segment is in the same layer as it's perpandiculars.
To be reduced, a segments has to be connected on source & target to
AutoContactTurn, both of the perpandiculars must be of the same layer
(below or above) and it's length must not exceed one pitch in the
perpandicular direction.
To reduce an AutoSegment, call ::reduce() and to revert the state,
call ::raise(). Two associated predicates are associated:
::canReduce() and ::mustRaise().
Note: No two adjacent segments can be reduced at the same time.
* Bug: In Katabatic, in GCellTopology, add a new method ::doRp_AccessPad()
to connect to the pads. Create wiring, fixed and non managed by
Katabatic, to connect the pad connector layer to the lowest routing
layers (depth 1 & 2). The former implementation was sometimes leading
to gaps (sheared contact) that *must not* occurs during the building
stage.
Remark: This bug did put under the light the fact that the initial
wiring must be created without gaps. Gaps are closed by making doglegs
on contacts. But this mechanism could only work when the database if
fully initialised (the cache is up to date). Otherwise various problems
arise, in the canonization process for example.
* New: In Katabatic, in AutoContactTerminal::getNativeConstraintBox(),
when anchored on a RoutingPad, now take account the potential rotation
of the Path's transformation. Here again, for the chip's pads.
* New: In Kite, support for reduced AutoSegment. TrackSegment associateds
to reduced AutoSegment are *not* inserted into track to become
effectively invisibles. When a segment becomes reduced, a TrackEvent
is generated to remove it. Conversely when it is raised a RoutingEvent
is created/rescheduled to insert it. All this is mostly managed inside
the Session::revalidate() method.
* New: In Kite, in KiteEngine::createGlobalGraph(), in case of a chip,
mark all global routing vertexes (Knik) that are under a pad, as blockeds.
* Bug: In Cumulus, in PadsCorona.Side.getAxis(), inversion between X and
Y coordinate of the chip size. Did not show until a non-square chip
was routed (i.e. our MIPS R3000).
* Change: In Stratus1, in st_placement.py add the ClockBuffer class for
backward compatibility with the MIPS32 bench. Have to review this
functionnality coming from the deprecated placeAndroute.py.
In st_instance.py, no longer creates the Plug ring of a Net.
In my opinion it just clutter the display until the P&R is called.
Can re-enable later as an option (in Unicorn).
* Change: In Unicorn, in cgt.py, more reliable way of loading then running
user supplied scripts. Borrowed from alliance-checker-toolkit doChip.py .
2015-08-16 16:29:28 -05:00
|
|
|
, Session::getViaWidth(padDepth)
|
|
|
|
, Session::getViaWidth(padDepth)
|
|
|
|
);
|
|
|
|
|
|
|
|
for ( size_t depth = padDepth ; depth >= accessDepth ; --depth ) {
|
|
|
|
const Layer* segmentLayer = Session::getRoutingLayer(depth);
|
|
|
|
const Layer* targetLayer = (depth == accessDepth) ? segmentLayer
|
|
|
|
: Session::getContactLayer(depth-1);
|
|
|
|
DbU::Unit targetSide = (depth == accessDepth) ? Session::getWireWidth(depth)
|
|
|
|
: Session::getViaWidth (depth-1);
|
|
|
|
|
|
|
|
target = Contact::create( rp->getNet()
|
|
|
|
, targetLayer
|
|
|
|
, position.getX()
|
|
|
|
, position.getY()
|
|
|
|
, targetSide
|
|
|
|
, targetSide
|
|
|
|
);
|
|
|
|
if (Session::getDirection(depth) == KbHorizontal) {
|
|
|
|
anchor = Horizontal::create( source
|
|
|
|
, target
|
|
|
|
, segmentLayer
|
|
|
|
, position.getY()
|
|
|
|
, Session::getWireWidth(depth)
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
anchor = Vertical::create( source
|
|
|
|
, target
|
|
|
|
, segmentLayer
|
|
|
|
, position.getX()
|
|
|
|
, Session::getWireWidth(depth)
|
|
|
|
);
|
|
|
|
}
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "Pad strap: " << anchor << endl;
|
Added support for "same layer" dogleg. Big fix for pad routing.
* Change: In Knik, in Vertex, add a "blocked" flag to signal disabled
vertexes in the grid (must not be used by the global router).
Modificate the Graph::getVertex() method so that when a vertex
is geometrically queried, if is a blocked one, return a non-blocked
neighbor. This mechanism is introduced to, at last, prevent the
global router to go *under* the pad in case of a commplete chip.
* New: In Katabatic, in AutoSegment, a new state has been added: "reduced".
A reduced segment is in the same layer as it's perpandiculars.
To be reduced, a segments has to be connected on source & target to
AutoContactTurn, both of the perpandiculars must be of the same layer
(below or above) and it's length must not exceed one pitch in the
perpandicular direction.
To reduce an AutoSegment, call ::reduce() and to revert the state,
call ::raise(). Two associated predicates are associated:
::canReduce() and ::mustRaise().
Note: No two adjacent segments can be reduced at the same time.
* Bug: In Katabatic, in GCellTopology, add a new method ::doRp_AccessPad()
to connect to the pads. Create wiring, fixed and non managed by
Katabatic, to connect the pad connector layer to the lowest routing
layers (depth 1 & 2). The former implementation was sometimes leading
to gaps (sheared contact) that *must not* occurs during the building
stage.
Remark: This bug did put under the light the fact that the initial
wiring must be created without gaps. Gaps are closed by making doglegs
on contacts. But this mechanism could only work when the database if
fully initialised (the cache is up to date). Otherwise various problems
arise, in the canonization process for example.
* New: In Katabatic, in AutoContactTerminal::getNativeConstraintBox(),
when anchored on a RoutingPad, now take account the potential rotation
of the Path's transformation. Here again, for the chip's pads.
* New: In Kite, support for reduced AutoSegment. TrackSegment associateds
to reduced AutoSegment are *not* inserted into track to become
effectively invisibles. When a segment becomes reduced, a TrackEvent
is generated to remove it. Conversely when it is raised a RoutingEvent
is created/rescheduled to insert it. All this is mostly managed inside
the Session::revalidate() method.
* New: In Kite, in KiteEngine::createGlobalGraph(), in case of a chip,
mark all global routing vertexes (Knik) that are under a pad, as blockeds.
* Bug: In Cumulus, in PadsCorona.Side.getAxis(), inversion between X and
Y coordinate of the chip size. Did not show until a non-square chip
was routed (i.e. our MIPS R3000).
* Change: In Stratus1, in st_placement.py add the ClockBuffer class for
backward compatibility with the MIPS32 bench. Have to review this
functionnality coming from the deprecated placeAndroute.py.
In st_instance.py, no longer creates the Plug ring of a Net.
In my opinion it just clutter the display until the P&R is called.
Can re-enable later as an option (in Unicorn).
* Change: In Unicorn, in cgt.py, more reliable way of loading then running
user supplied scripts. Borrowed from alliance-checker-toolkit doChip.py .
2015-08-16 16:29:28 -05:00
|
|
|
source = target;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Go *up* from the pad's RoutingPad.
|
|
|
|
Contact* target = NULL;
|
|
|
|
Contact* source = Contact::create ( rp
|
|
|
|
, Session::getContactLayer(padDepth)
|
|
|
|
, 0
|
|
|
|
, 0
|
|
|
|
, Session::getViaWidth(padDepth)
|
|
|
|
, Session::getViaWidth(padDepth)
|
|
|
|
);
|
|
|
|
|
|
|
|
for ( size_t depth = padDepth ; depth <= accessDepth ; ++depth ) {
|
|
|
|
const Layer* segmentLayer = Session::getRoutingLayer(depth);
|
|
|
|
const Layer* targetLayer = (depth == accessDepth) ? segmentLayer
|
|
|
|
: Session::getContactLayer(depth);
|
|
|
|
DbU::Unit targetSide = (depth == accessDepth) ? Session::getWireWidth(depth)
|
|
|
|
: Session::getViaWidth (depth);
|
|
|
|
|
|
|
|
target = Contact::create( rp->getNet()
|
|
|
|
, targetLayer
|
|
|
|
, position.getX()
|
|
|
|
, position.getY()
|
|
|
|
, targetSide
|
|
|
|
, targetSide
|
|
|
|
);
|
|
|
|
if (Session::getDirection(depth) == KbHorizontal) {
|
|
|
|
anchor = Horizontal::create( source
|
|
|
|
, target
|
|
|
|
, segmentLayer
|
|
|
|
, position.getY()
|
|
|
|
, Session::getWireWidth(depth)
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
anchor = Vertical::create( source
|
|
|
|
, target
|
|
|
|
, segmentLayer
|
|
|
|
, position.getX()
|
|
|
|
, Session::getWireWidth(depth)
|
|
|
|
);
|
|
|
|
}
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "Pad strap: " << anchor << endl;
|
Added support for "same layer" dogleg. Big fix for pad routing.
* Change: In Knik, in Vertex, add a "blocked" flag to signal disabled
vertexes in the grid (must not be used by the global router).
Modificate the Graph::getVertex() method so that when a vertex
is geometrically queried, if is a blocked one, return a non-blocked
neighbor. This mechanism is introduced to, at last, prevent the
global router to go *under* the pad in case of a commplete chip.
* New: In Katabatic, in AutoSegment, a new state has been added: "reduced".
A reduced segment is in the same layer as it's perpandiculars.
To be reduced, a segments has to be connected on source & target to
AutoContactTurn, both of the perpandiculars must be of the same layer
(below or above) and it's length must not exceed one pitch in the
perpandicular direction.
To reduce an AutoSegment, call ::reduce() and to revert the state,
call ::raise(). Two associated predicates are associated:
::canReduce() and ::mustRaise().
Note: No two adjacent segments can be reduced at the same time.
* Bug: In Katabatic, in GCellTopology, add a new method ::doRp_AccessPad()
to connect to the pads. Create wiring, fixed and non managed by
Katabatic, to connect the pad connector layer to the lowest routing
layers (depth 1 & 2). The former implementation was sometimes leading
to gaps (sheared contact) that *must not* occurs during the building
stage.
Remark: This bug did put under the light the fact that the initial
wiring must be created without gaps. Gaps are closed by making doglegs
on contacts. But this mechanism could only work when the database if
fully initialised (the cache is up to date). Otherwise various problems
arise, in the canonization process for example.
* New: In Katabatic, in AutoContactTerminal::getNativeConstraintBox(),
when anchored on a RoutingPad, now take account the potential rotation
of the Path's transformation. Here again, for the chip's pads.
* New: In Kite, support for reduced AutoSegment. TrackSegment associateds
to reduced AutoSegment are *not* inserted into track to become
effectively invisibles. When a segment becomes reduced, a TrackEvent
is generated to remove it. Conversely when it is raised a RoutingEvent
is created/rescheduled to insert it. All this is mostly managed inside
the Session::revalidate() method.
* New: In Kite, in KiteEngine::createGlobalGraph(), in case of a chip,
mark all global routing vertexes (Knik) that are under a pad, as blockeds.
* Bug: In Cumulus, in PadsCorona.Side.getAxis(), inversion between X and
Y coordinate of the chip size. Did not show until a non-square chip
was routed (i.e. our MIPS R3000).
* Change: In Stratus1, in st_placement.py add the ClockBuffer class for
backward compatibility with the MIPS32 bench. Have to review this
functionnality coming from the deprecated placeAndroute.py.
In st_instance.py, no longer creates the Plug ring of a Net.
In my opinion it just clutter the display until the P&R is called.
Can re-enable later as an option (in Unicorn).
* Change: In Unicorn, in cgt.py, more reliable way of loading then running
user supplied scripts. Borrowed from alliance-checker-toolkit doChip.py .
2015-08-16 16:29:28 -05:00
|
|
|
source = target;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
AutoContact* autoSource
|
|
|
|
= AutoContactTerminal::create ( gcell
|
|
|
|
, anchor
|
|
|
|
, Session::getRoutingLayer(accessDepth)
|
|
|
|
, position
|
|
|
|
, Session::getWireWidth(accessDepth)
|
|
|
|
, Session::getWireWidth(accessDepth)
|
|
|
|
);
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_tabw(145,-1);
|
Added support for "same layer" dogleg. Big fix for pad routing.
* Change: In Knik, in Vertex, add a "blocked" flag to signal disabled
vertexes in the grid (must not be used by the global router).
Modificate the Graph::getVertex() method so that when a vertex
is geometrically queried, if is a blocked one, return a non-blocked
neighbor. This mechanism is introduced to, at last, prevent the
global router to go *under* the pad in case of a commplete chip.
* New: In Katabatic, in AutoSegment, a new state has been added: "reduced".
A reduced segment is in the same layer as it's perpandiculars.
To be reduced, a segments has to be connected on source & target to
AutoContactTurn, both of the perpandiculars must be of the same layer
(below or above) and it's length must not exceed one pitch in the
perpandicular direction.
To reduce an AutoSegment, call ::reduce() and to revert the state,
call ::raise(). Two associated predicates are associated:
::canReduce() and ::mustRaise().
Note: No two adjacent segments can be reduced at the same time.
* Bug: In Katabatic, in GCellTopology, add a new method ::doRp_AccessPad()
to connect to the pads. Create wiring, fixed and non managed by
Katabatic, to connect the pad connector layer to the lowest routing
layers (depth 1 & 2). The former implementation was sometimes leading
to gaps (sheared contact) that *must not* occurs during the building
stage.
Remark: This bug did put under the light the fact that the initial
wiring must be created without gaps. Gaps are closed by making doglegs
on contacts. But this mechanism could only work when the database if
fully initialised (the cache is up to date). Otherwise various problems
arise, in the canonization process for example.
* New: In Katabatic, in AutoContactTerminal::getNativeConstraintBox(),
when anchored on a RoutingPad, now take account the potential rotation
of the Path's transformation. Here again, for the chip's pads.
* New: In Kite, support for reduced AutoSegment. TrackSegment associateds
to reduced AutoSegment are *not* inserted into track to become
effectively invisibles. When a segment becomes reduced, a TrackEvent
is generated to remove it. Conversely when it is raised a RoutingEvent
is created/rescheduled to insert it. All this is mostly managed inside
the Session::revalidate() method.
* New: In Kite, in KiteEngine::createGlobalGraph(), in case of a chip,
mark all global routing vertexes (Knik) that are under a pad, as blockeds.
* Bug: In Cumulus, in PadsCorona.Side.getAxis(), inversion between X and
Y coordinate of the chip size. Did not show until a non-square chip
was routed (i.e. our MIPS R3000).
* Change: In Stratus1, in st_placement.py add the ClockBuffer class for
backward compatibility with the MIPS32 bench. Have to review this
functionnality coming from the deprecated placeAndroute.py.
In st_instance.py, no longer creates the Plug ring of a Net.
In my opinion it just clutter the display until the P&R is called.
Can re-enable later as an option (in Unicorn).
* Change: In Unicorn, in cgt.py, more reliable way of loading then running
user supplied scripts. Borrowed from alliance-checker-toolkit doChip.py .
2015-08-16 16:29:28 -05:00
|
|
|
return autoSource;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
void GCellTopology::doRp_StairCaseH ( GCell* gcell, Component* rp1, Component* rp2 )
|
2010-03-09 09:24:29 -06:00
|
|
|
{
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "doRp_StairCaseH()" << endl;
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
if (rp1->getCenter().getX() > rp2->getCenter().getX()) swap( rp1, rp2 );
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
AutoContact* rp1ContactSource = NULL;
|
|
|
|
AutoContact* rp1ContactTarget = NULL;
|
|
|
|
AutoContact* rp2ContactSource = NULL;
|
|
|
|
AutoContact* rp2ContactTarget = NULL;
|
|
|
|
const Layer* viaLayer = NULL;
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
doRp_AutoContacts( gcell, rp1, rp1ContactSource, rp1ContactTarget, DoTargetContact );
|
|
|
|
doRp_AutoContacts( gcell, rp2, rp2ContactSource, rp2ContactTarget, DoSourceContact );
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
if (rp1ContactTarget->getY() == rp2ContactSource->getY()) {
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "Aligned horizontal routing pads : straight wire" << endl;
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
viaLayer = rp1->getLayer();
|
|
|
|
AutoSegment::create( rp1ContactTarget, rp2ContactSource, KbHorizontal );
|
2010-03-09 09:24:29 -06:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
viaLayer = Session::getContactLayer(1);
|
|
|
|
|
2014-03-15 04:47:37 -05:00
|
|
|
AutoContact* subContact1 = AutoContactTurn::create( gcell, rp1->getNet(), viaLayer );
|
|
|
|
AutoContact* subContact2 = AutoContactTurn::create( gcell, rp1->getNet(), viaLayer );
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
AutoSegment::create( rp1ContactTarget, subContact1 , KbHorizontal );
|
|
|
|
AutoSegment::create( subContact1 , subContact2 , KbVertical );
|
|
|
|
AutoSegment::create( subContact1 , rp2ContactSource, KbHorizontal );
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
void GCellTopology::doRp_StairCaseV ( GCell* gcell, Component* rp1, Component* rp2 )
|
2010-03-09 09:24:29 -06:00
|
|
|
{
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "doRp_StairCaseV()" << endl;
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
if (rp1->getCenter().getY() > rp2->getCenter().getY()) swap( rp1, rp2 );
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
AutoContact* rp1ContactSource = NULL;
|
|
|
|
AutoContact* rp1ContactTarget = NULL;
|
|
|
|
AutoContact* rp2ContactSource = NULL;
|
|
|
|
AutoContact* rp2ContactTarget = NULL;
|
|
|
|
const Layer* viaLayer = NULL;
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
doRp_AutoContacts( gcell, rp1, rp1ContactSource, rp1ContactTarget, DoTargetContact );
|
|
|
|
doRp_AutoContacts( gcell, rp2, rp2ContactSource, rp2ContactTarget, DoSourceContact );
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
if (rp1ContactTarget->getX() == rp2ContactSource->getX()) {
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "Aligned vertical routing pads : straight wire" << endl;
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
viaLayer = rp1->getLayer();
|
|
|
|
AutoSegment::create( rp1ContactTarget, rp2ContactSource, KbVertical );
|
2010-03-09 09:24:29 -06:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
viaLayer = Session::getContactLayer(1);
|
|
|
|
|
2014-03-15 04:47:37 -05:00
|
|
|
AutoContact* subContact1 = AutoContactTurn::create( gcell, rp1->getNet(), viaLayer );
|
|
|
|
AutoContact* subContact2 = AutoContactTurn::create( gcell, rp1->getNet(), viaLayer );
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
AutoSegment::create( rp1ContactTarget, subContact1 , KbVertical );
|
|
|
|
AutoSegment::create( subContact1 , subContact2 , KbHorizontal );
|
|
|
|
AutoSegment::create( subContact1 , rp2ContactSource, KbVertical );
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
void GCellTopology::_do_xG ()
|
2010-03-09 09:24:29 -06:00
|
|
|
{
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,1) << "_do_xG()" << endl;
|
2013-12-03 18:58:58 -06:00
|
|
|
|
|
|
|
if (_connexity.fields.globals == 2) {
|
|
|
|
_southWestContact
|
|
|
|
= _northEastContact
|
|
|
|
= AutoContactTurn::create( _gcell, _net, Session::getContactLayer(1) );
|
|
|
|
} else if (_connexity.fields.globals == 3) {
|
|
|
|
if (_east and _west) {
|
|
|
|
_southWestContact = AutoContactTurn::create( _gcell, _net, Session::getContactLayer(1) );
|
|
|
|
_northEastContact = AutoContactVTee::create( _gcell, _net, Session::getContactLayer(1) );
|
|
|
|
if (_south) swap( _southWestContact, _northEastContact );
|
|
|
|
|
|
|
|
AutoSegment::create( _southWestContact, _northEastContact, KbVertical );
|
|
|
|
} else {
|
|
|
|
_southWestContact = AutoContactTurn::create( _gcell, _net, Session::getContactLayer(1) );
|
|
|
|
_northEastContact = AutoContactHTee::create( _gcell, _net, Session::getContactLayer(1) );
|
|
|
|
if (_west) swap( _southWestContact, _northEastContact );
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
AutoSegment::create( _southWestContact, _northEastContact, KbHorizontal );
|
|
|
|
}
|
|
|
|
} else { // fields.globals == 4.
|
|
|
|
AutoContact* turn = AutoContactTurn::create( _gcell, _net, Session::getContactLayer(1) );
|
|
|
|
_southWestContact = AutoContactHTee::create( _gcell, _net, Session::getContactLayer(1) );
|
|
|
|
_northEastContact = AutoContactVTee::create( _gcell, _net, Session::getContactLayer(1) );
|
|
|
|
AutoSegment::create( _southWestContact, turn, KbHorizontal );
|
|
|
|
AutoSegment::create( turn, _northEastContact, KbVertical );
|
|
|
|
}
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_tabw(145,-1);
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
void GCellTopology::_do_xG_1Pad ()
|
* ./katabatic:
- Change: In loadGlobalRouting(), more exlicit message as to why a net is
filtered out the routing set (POWER, GROUND, CLOCK or BLOCKAGE).
- New: ChipTool, module with utilities specific to chip routing. Containing
a function to pre-break wires around a block.
- New: In loadGlobalRouting/GCellConfiguration, adds supports for RoutingPad
connected on chip's pad.
- New: In GCellGrid/GCell, can compute the density in three modes: average,
max of H/V and max of layer. Currently implemented in getMaxHVDensity(),
should be unificated with getDensity(). Used for display purposes.
- Bug: In AutoContact, when splitting a contact, add a specific check for
"one layer span".
- Bug: In AutoSegment::canMoveUp(), do not move up fixed segments.
2010-11-16 07:59:38 -06:00
|
|
|
{
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,1) << "_do_xG_1Pad() [Managed Configuration - Optimized] " << _topology << endl;
|
Solve the template lookup problem in tstream.
* Bug: In Hurricane, in tstream (Commons.h), in the *template* overload of
operator<<(), do not use the operator<<() of ostream as it will be
looked up in "stage 1" (template definition) and so will miss all the
overloads added later and built over getString<>(). Instead, make use
of getString<>(), which, as another template will be looked up in
"stage 2" (template instanciation) and at that point will have all the
needed template specialisation of getString<>().
We also need to define new stream manipulators to be able to create
a matching template overload not dependant from the implementation.
To avoid name clashes, we prefix a 't'. For now, only 'tsetw()' is
refined.
As a side effect, we cannot directly print bit-fields into the stream,
we must go through an intermediate variable (happens once in AutoContact).
2016-07-19 09:02:55 -05:00
|
|
|
cdebug_log(145,0) << "_connexity.globals:" << (int)_connexity.fields.globals << endl;
|
* ./katabatic:
- Change: In loadGlobalRouting(), more exlicit message as to why a net is
filtered out the routing set (POWER, GROUND, CLOCK or BLOCKAGE).
- New: ChipTool, module with utilities specific to chip routing. Containing
a function to pre-break wires around a block.
- New: In loadGlobalRouting/GCellConfiguration, adds supports for RoutingPad
connected on chip's pad.
- New: In GCellGrid/GCell, can compute the density in three modes: average,
max of H/V and max of layer. Currently implemented in getMaxHVDensity(),
should be unificated with getDensity(). Used for display purposes.
- Bug: In AutoContact, when splitting a contact, add a specific check for
"one layer span".
- Bug: In AutoSegment::canMoveUp(), do not move up fixed segments.
2010-11-16 07:59:38 -06:00
|
|
|
|
Added support for "same layer" dogleg. Big fix for pad routing.
* Change: In Knik, in Vertex, add a "blocked" flag to signal disabled
vertexes in the grid (must not be used by the global router).
Modificate the Graph::getVertex() method so that when a vertex
is geometrically queried, if is a blocked one, return a non-blocked
neighbor. This mechanism is introduced to, at last, prevent the
global router to go *under* the pad in case of a commplete chip.
* New: In Katabatic, in AutoSegment, a new state has been added: "reduced".
A reduced segment is in the same layer as it's perpandiculars.
To be reduced, a segments has to be connected on source & target to
AutoContactTurn, both of the perpandiculars must be of the same layer
(below or above) and it's length must not exceed one pitch in the
perpandicular direction.
To reduce an AutoSegment, call ::reduce() and to revert the state,
call ::raise(). Two associated predicates are associated:
::canReduce() and ::mustRaise().
Note: No two adjacent segments can be reduced at the same time.
* Bug: In Katabatic, in GCellTopology, add a new method ::doRp_AccessPad()
to connect to the pads. Create wiring, fixed and non managed by
Katabatic, to connect the pad connector layer to the lowest routing
layers (depth 1 & 2). The former implementation was sometimes leading
to gaps (sheared contact) that *must not* occurs during the building
stage.
Remark: This bug did put under the light the fact that the initial
wiring must be created without gaps. Gaps are closed by making doglegs
on contacts. But this mechanism could only work when the database if
fully initialised (the cache is up to date). Otherwise various problems
arise, in the canonization process for example.
* New: In Katabatic, in AutoContactTerminal::getNativeConstraintBox(),
when anchored on a RoutingPad, now take account the potential rotation
of the Path's transformation. Here again, for the chip's pads.
* New: In Kite, support for reduced AutoSegment. TrackSegment associateds
to reduced AutoSegment are *not* inserted into track to become
effectively invisibles. When a segment becomes reduced, a TrackEvent
is generated to remove it. Conversely when it is raised a RoutingEvent
is created/rescheduled to insert it. All this is mostly managed inside
the Session::revalidate() method.
* New: In Kite, in KiteEngine::createGlobalGraph(), in case of a chip,
mark all global routing vertexes (Knik) that are under a pad, as blockeds.
* Bug: In Cumulus, in PadsCorona.Side.getAxis(), inversion between X and
Y coordinate of the chip size. Did not show until a non-square chip
was routed (i.e. our MIPS R3000).
* Change: In Stratus1, in st_placement.py add the ClockBuffer class for
backward compatibility with the MIPS32 bench. Have to review this
functionnality coming from the deprecated placeAndroute.py.
In st_instance.py, no longer creates the Plug ring of a Net.
In my opinion it just clutter the display until the P&R is called.
Can re-enable later as an option (in Unicorn).
* Change: In Unicorn, in cgt.py, more reliable way of loading then running
user supplied scripts. Borrowed from alliance-checker-toolkit doChip.py .
2015-08-16 16:29:28 -05:00
|
|
|
unsigned int flags = NoFlags;
|
|
|
|
bool eastPad = false;
|
|
|
|
bool westPad = false;
|
|
|
|
bool northPad = false;
|
|
|
|
bool southPad = false;
|
Corrections in how to access pads terminals in Katabatic/Kite.
* Change: In Katabatic, in GCellTopology::doRp_AccessPad(), if the supporting
RoutingPad is big (more than two pitch), do not put the access
contact in the center but on the edge. This is to avoid cut violations
between the VIAs and the matrix of VIAs that may be generated under the
RoutingPad itself.
* Change: In Kite, in TrackSegment destructor, if the legnth of the wire,
without extensions is less than one picth, enlarge it so it encompass
it's source & target VIAs (to avoid notches).
* Bug: In Kite, in Track destructor, the TrackElements where detacheds,
but not deleted, causing a memory link.
2016-03-02 10:15:58 -06:00
|
|
|
Instance* padInstance = _routingPads[0]->getOccurrence().getPath().getHeadInstance();
|
* ./katabatic:
- New: ChipTools, regroup all datas and utilities to manage a full-chip
design.
- Change: In LoadGrByNet/GCellConfiguration::_GCell_xG_1Pad(), uses straight
perpandicular wires on top & right pads. The GCells under those connectors
are fully saturated, wires must go out as straight as possible.
- Change: In AutoHorizontal/AutoVertical, update the "terminal" flag after
slackening. This is to avoid global that are no longer connected to
terminals behave as such.
- Change: In AutoSegment::canMoveUp() & canPivotUp(), prevent all local segments
to go up. This is to avoid cluttering upper levels with small segments.
- Change: In GCellConfiguration, for xG_xL3, detect straight vertical topologies
and for them to be fixed (new flag FIXED_GLOBAL set by the topological
builder instead of the constructor). This prevent the router to do
stupid things...
To sets the "Fixed" flag *after* the axis of global segments have been
positionned correctly adds a "_toFixGlobals" static table lookup.
- Change: In GCell::stepDesaturate(), if less than one free track remains in
the upper layer, do not move up the segment. This allows from a minimum
free room for expansion.
- Change: In GCell::_getString(), display indexes and layer names in various
saturation tables.
- Change: In AutoSegment, allow move up of local Segment. ::moveUp() and
::canMoveUp() arguments go from booleans to flags, which are more
explicits.
2010-12-04 09:25:18 -06:00
|
|
|
|
|
|
|
switch ( padInstance->getTransformation().getOrientation() ) {
|
2013-12-03 18:58:58 -06:00
|
|
|
case Transformation::Orientation::ID: northPad = true; break;
|
|
|
|
case Transformation::Orientation::MY: southPad = true; break;
|
Added support for "same layer" dogleg. Big fix for pad routing.
* Change: In Knik, in Vertex, add a "blocked" flag to signal disabled
vertexes in the grid (must not be used by the global router).
Modificate the Graph::getVertex() method so that when a vertex
is geometrically queried, if is a blocked one, return a non-blocked
neighbor. This mechanism is introduced to, at last, prevent the
global router to go *under* the pad in case of a commplete chip.
* New: In Katabatic, in AutoSegment, a new state has been added: "reduced".
A reduced segment is in the same layer as it's perpandiculars.
To be reduced, a segments has to be connected on source & target to
AutoContactTurn, both of the perpandiculars must be of the same layer
(below or above) and it's length must not exceed one pitch in the
perpandicular direction.
To reduce an AutoSegment, call ::reduce() and to revert the state,
call ::raise(). Two associated predicates are associated:
::canReduce() and ::mustRaise().
Note: No two adjacent segments can be reduced at the same time.
* Bug: In Katabatic, in GCellTopology, add a new method ::doRp_AccessPad()
to connect to the pads. Create wiring, fixed and non managed by
Katabatic, to connect the pad connector layer to the lowest routing
layers (depth 1 & 2). The former implementation was sometimes leading
to gaps (sheared contact) that *must not* occurs during the building
stage.
Remark: This bug did put under the light the fact that the initial
wiring must be created without gaps. Gaps are closed by making doglegs
on contacts. But this mechanism could only work when the database if
fully initialised (the cache is up to date). Otherwise various problems
arise, in the canonization process for example.
* New: In Katabatic, in AutoContactTerminal::getNativeConstraintBox(),
when anchored on a RoutingPad, now take account the potential rotation
of the Path's transformation. Here again, for the chip's pads.
* New: In Kite, support for reduced AutoSegment. TrackSegment associateds
to reduced AutoSegment are *not* inserted into track to become
effectively invisibles. When a segment becomes reduced, a TrackEvent
is generated to remove it. Conversely when it is raised a RoutingEvent
is created/rescheduled to insert it. All this is mostly managed inside
the Session::revalidate() method.
* New: In Kite, in KiteEngine::createGlobalGraph(), in case of a chip,
mark all global routing vertexes (Knik) that are under a pad, as blockeds.
* Bug: In Cumulus, in PadsCorona.Side.getAxis(), inversion between X and
Y coordinate of the chip size. Did not show until a non-square chip
was routed (i.e. our MIPS R3000).
* Change: In Stratus1, in st_placement.py add the ClockBuffer class for
backward compatibility with the MIPS32 bench. Have to review this
functionnality coming from the deprecated placeAndroute.py.
In st_instance.py, no longer creates the Plug ring of a Net.
In my opinion it just clutter the display until the P&R is called.
Can re-enable later as an option (in Unicorn).
* Change: In Unicorn, in cgt.py, more reliable way of loading then running
user supplied scripts. Borrowed from alliance-checker-toolkit doChip.py .
2015-08-16 16:29:28 -05:00
|
|
|
case Transformation::Orientation::YR:
|
|
|
|
case Transformation::Orientation::R3: eastPad = true; flags |= HAccess; break;
|
|
|
|
case Transformation::Orientation::R1: westPad = true; flags |= HAccess; break;
|
* ./katabatic:
- New: ChipTools, regroup all datas and utilities to manage a full-chip
design.
- Change: In LoadGrByNet/GCellConfiguration::_GCell_xG_1Pad(), uses straight
perpandicular wires on top & right pads. The GCells under those connectors
are fully saturated, wires must go out as straight as possible.
- Change: In AutoHorizontal/AutoVertical, update the "terminal" flag after
slackening. This is to avoid global that are no longer connected to
terminals behave as such.
- Change: In AutoSegment::canMoveUp() & canPivotUp(), prevent all local segments
to go up. This is to avoid cluttering upper levels with small segments.
- Change: In GCellConfiguration, for xG_xL3, detect straight vertical topologies
and for them to be fixed (new flag FIXED_GLOBAL set by the topological
builder instead of the constructor). This prevent the router to do
stupid things...
To sets the "Fixed" flag *after* the axis of global segments have been
positionned correctly adds a "_toFixGlobals" static table lookup.
- Change: In GCell::stepDesaturate(), if less than one free track remains in
the upper layer, do not move up the segment. This allows from a minimum
free room for expansion.
- Change: In GCell::_getString(), display indexes and layer names in various
saturation tables.
- Change: In AutoSegment, allow move up of local Segment. ::moveUp() and
::canMoveUp() arguments go from booleans to flags, which are more
explicits.
2010-12-04 09:25:18 -06:00
|
|
|
default:
|
2013-12-03 18:58:58 -06:00
|
|
|
cerr << Warning( "Unmanaged orientation %s for pad <%s>."
|
|
|
|
, getString(padInstance->getTransformation().getOrientation()).c_str()
|
|
|
|
, getString(padInstance).c_str() ) << endl;
|
* ./katabatic:
- New: ChipTools, regroup all datas and utilities to manage a full-chip
design.
- Change: In LoadGrByNet/GCellConfiguration::_GCell_xG_1Pad(), uses straight
perpandicular wires on top & right pads. The GCells under those connectors
are fully saturated, wires must go out as straight as possible.
- Change: In AutoHorizontal/AutoVertical, update the "terminal" flag after
slackening. This is to avoid global that are no longer connected to
terminals behave as such.
- Change: In AutoSegment::canMoveUp() & canPivotUp(), prevent all local segments
to go up. This is to avoid cluttering upper levels with small segments.
- Change: In GCellConfiguration, for xG_xL3, detect straight vertical topologies
and for them to be fixed (new flag FIXED_GLOBAL set by the topological
builder instead of the constructor). This prevent the router to do
stupid things...
To sets the "Fixed" flag *after* the axis of global segments have been
positionned correctly adds a "_toFixGlobals" static table lookup.
- Change: In GCell::stepDesaturate(), if less than one free track remains in
the upper layer, do not move up the segment. This allows from a minimum
free room for expansion.
- Change: In GCell::_getString(), display indexes and layer names in various
saturation tables.
- Change: In AutoSegment, allow move up of local Segment. ::moveUp() and
::canMoveUp() arguments go from booleans to flags, which are more
explicits.
2010-12-04 09:25:18 -06:00
|
|
|
break;
|
|
|
|
}
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "eastPad:" << eastPad << ", "
|
Added support for "same layer" dogleg. Big fix for pad routing.
* Change: In Knik, in Vertex, add a "blocked" flag to signal disabled
vertexes in the grid (must not be used by the global router).
Modificate the Graph::getVertex() method so that when a vertex
is geometrically queried, if is a blocked one, return a non-blocked
neighbor. This mechanism is introduced to, at last, prevent the
global router to go *under* the pad in case of a commplete chip.
* New: In Katabatic, in AutoSegment, a new state has been added: "reduced".
A reduced segment is in the same layer as it's perpandiculars.
To be reduced, a segments has to be connected on source & target to
AutoContactTurn, both of the perpandiculars must be of the same layer
(below or above) and it's length must not exceed one pitch in the
perpandicular direction.
To reduce an AutoSegment, call ::reduce() and to revert the state,
call ::raise(). Two associated predicates are associated:
::canReduce() and ::mustRaise().
Note: No two adjacent segments can be reduced at the same time.
* Bug: In Katabatic, in GCellTopology, add a new method ::doRp_AccessPad()
to connect to the pads. Create wiring, fixed and non managed by
Katabatic, to connect the pad connector layer to the lowest routing
layers (depth 1 & 2). The former implementation was sometimes leading
to gaps (sheared contact) that *must not* occurs during the building
stage.
Remark: This bug did put under the light the fact that the initial
wiring must be created without gaps. Gaps are closed by making doglegs
on contacts. But this mechanism could only work when the database if
fully initialised (the cache is up to date). Otherwise various problems
arise, in the canonization process for example.
* New: In Katabatic, in AutoContactTerminal::getNativeConstraintBox(),
when anchored on a RoutingPad, now take account the potential rotation
of the Path's transformation. Here again, for the chip's pads.
* New: In Kite, support for reduced AutoSegment. TrackSegment associateds
to reduced AutoSegment are *not* inserted into track to become
effectively invisibles. When a segment becomes reduced, a TrackEvent
is generated to remove it. Conversely when it is raised a RoutingEvent
is created/rescheduled to insert it. All this is mostly managed inside
the Session::revalidate() method.
* New: In Kite, in KiteEngine::createGlobalGraph(), in case of a chip,
mark all global routing vertexes (Knik) that are under a pad, as blockeds.
* Bug: In Cumulus, in PadsCorona.Side.getAxis(), inversion between X and
Y coordinate of the chip size. Did not show until a non-square chip
was routed (i.e. our MIPS R3000).
* Change: In Stratus1, in st_placement.py add the ClockBuffer class for
backward compatibility with the MIPS32 bench. Have to review this
functionnality coming from the deprecated placeAndroute.py.
In st_instance.py, no longer creates the Plug ring of a Net.
In my opinion it just clutter the display until the P&R is called.
Can re-enable later as an option (in Unicorn).
* Change: In Unicorn, in cgt.py, more reliable way of loading then running
user supplied scripts. Borrowed from alliance-checker-toolkit doChip.py .
2015-08-16 16:29:28 -05:00
|
|
|
<< "westPad:" << westPad << ", "
|
|
|
|
<< "northPad:" << northPad << ", "
|
|
|
|
<< "southPad:" << southPad
|
|
|
|
<< endl;
|
* ./katabatic:
- New: ChipTools, regroup all datas and utilities to manage a full-chip
design.
- Change: In LoadGrByNet/GCellConfiguration::_GCell_xG_1Pad(), uses straight
perpandicular wires on top & right pads. The GCells under those connectors
are fully saturated, wires must go out as straight as possible.
- Change: In AutoHorizontal/AutoVertical, update the "terminal" flag after
slackening. This is to avoid global that are no longer connected to
terminals behave as such.
- Change: In AutoSegment::canMoveUp() & canPivotUp(), prevent all local segments
to go up. This is to avoid cluttering upper levels with small segments.
- Change: In GCellConfiguration, for xG_xL3, detect straight vertical topologies
and for them to be fixed (new flag FIXED_GLOBAL set by the topological
builder instead of the constructor). This prevent the router to do
stupid things...
To sets the "Fixed" flag *after* the axis of global segments have been
positionned correctly adds a "_toFixGlobals" static table lookup.
- Change: In GCell::stepDesaturate(), if less than one free track remains in
the upper layer, do not move up the segment. This allows from a minimum
free room for expansion.
- Change: In GCell::_getString(), display indexes and layer names in various
saturation tables.
- Change: In AutoSegment, allow move up of local Segment. ::moveUp() and
::canMoveUp() arguments go from booleans to flags, which are more
explicits.
2010-12-04 09:25:18 -06:00
|
|
|
|
Added support for "same layer" dogleg. Big fix for pad routing.
* Change: In Knik, in Vertex, add a "blocked" flag to signal disabled
vertexes in the grid (must not be used by the global router).
Modificate the Graph::getVertex() method so that when a vertex
is geometrically queried, if is a blocked one, return a non-blocked
neighbor. This mechanism is introduced to, at last, prevent the
global router to go *under* the pad in case of a commplete chip.
* New: In Katabatic, in AutoSegment, a new state has been added: "reduced".
A reduced segment is in the same layer as it's perpandiculars.
To be reduced, a segments has to be connected on source & target to
AutoContactTurn, both of the perpandiculars must be of the same layer
(below or above) and it's length must not exceed one pitch in the
perpandicular direction.
To reduce an AutoSegment, call ::reduce() and to revert the state,
call ::raise(). Two associated predicates are associated:
::canReduce() and ::mustRaise().
Note: No two adjacent segments can be reduced at the same time.
* Bug: In Katabatic, in GCellTopology, add a new method ::doRp_AccessPad()
to connect to the pads. Create wiring, fixed and non managed by
Katabatic, to connect the pad connector layer to the lowest routing
layers (depth 1 & 2). The former implementation was sometimes leading
to gaps (sheared contact) that *must not* occurs during the building
stage.
Remark: This bug did put under the light the fact that the initial
wiring must be created without gaps. Gaps are closed by making doglegs
on contacts. But this mechanism could only work when the database if
fully initialised (the cache is up to date). Otherwise various problems
arise, in the canonization process for example.
* New: In Katabatic, in AutoContactTerminal::getNativeConstraintBox(),
when anchored on a RoutingPad, now take account the potential rotation
of the Path's transformation. Here again, for the chip's pads.
* New: In Kite, support for reduced AutoSegment. TrackSegment associateds
to reduced AutoSegment are *not* inserted into track to become
effectively invisibles. When a segment becomes reduced, a TrackEvent
is generated to remove it. Conversely when it is raised a RoutingEvent
is created/rescheduled to insert it. All this is mostly managed inside
the Session::revalidate() method.
* New: In Kite, in KiteEngine::createGlobalGraph(), in case of a chip,
mark all global routing vertexes (Knik) that are under a pad, as blockeds.
* Bug: In Cumulus, in PadsCorona.Side.getAxis(), inversion between X and
Y coordinate of the chip size. Did not show until a non-square chip
was routed (i.e. our MIPS R3000).
* Change: In Stratus1, in st_placement.py add the ClockBuffer class for
backward compatibility with the MIPS32 bench. Have to review this
functionnality coming from the deprecated placeAndroute.py.
In st_instance.py, no longer creates the Plug ring of a Net.
In my opinion it just clutter the display until the P&R is called.
Can re-enable later as an option (in Unicorn).
* Change: In Unicorn, in cgt.py, more reliable way of loading then running
user supplied scripts. Borrowed from alliance-checker-toolkit doChip.py .
2015-08-16 16:29:28 -05:00
|
|
|
AutoContact* source = doRp_AccessPad( _routingPads[0], flags );
|
|
|
|
// Point position = _routingPads[0]->getCenter();
|
|
|
|
// AutoContact* source = NULL;
|
|
|
|
// GCell* gcell = Session::getKatabatic()->getGCellGrid()->getGCell(position);
|
|
|
|
|
|
|
|
// source = AutoContactTerminal::create ( gcell
|
|
|
|
// , _routingPads[0]
|
|
|
|
// , Session::getContactLayer(3)
|
|
|
|
// , position
|
|
|
|
// , Session::getViaWidth(3), Session::getViaWidth(3)
|
|
|
|
// );
|
|
|
|
// source->setFlags( CntFixed );
|
|
|
|
|
|
|
|
// if (northPad or eastPad) {
|
|
|
|
// _southWestContact = _northEastContact = source;
|
2016-06-11 14:56:12 -05:00
|
|
|
// cdebug_tabw(145,-1);
|
Added support for "same layer" dogleg. Big fix for pad routing.
* Change: In Knik, in Vertex, add a "blocked" flag to signal disabled
vertexes in the grid (must not be used by the global router).
Modificate the Graph::getVertex() method so that when a vertex
is geometrically queried, if is a blocked one, return a non-blocked
neighbor. This mechanism is introduced to, at last, prevent the
global router to go *under* the pad in case of a commplete chip.
* New: In Katabatic, in AutoSegment, a new state has been added: "reduced".
A reduced segment is in the same layer as it's perpandiculars.
To be reduced, a segments has to be connected on source & target to
AutoContactTurn, both of the perpandiculars must be of the same layer
(below or above) and it's length must not exceed one pitch in the
perpandicular direction.
To reduce an AutoSegment, call ::reduce() and to revert the state,
call ::raise(). Two associated predicates are associated:
::canReduce() and ::mustRaise().
Note: No two adjacent segments can be reduced at the same time.
* Bug: In Katabatic, in GCellTopology, add a new method ::doRp_AccessPad()
to connect to the pads. Create wiring, fixed and non managed by
Katabatic, to connect the pad connector layer to the lowest routing
layers (depth 1 & 2). The former implementation was sometimes leading
to gaps (sheared contact) that *must not* occurs during the building
stage.
Remark: This bug did put under the light the fact that the initial
wiring must be created without gaps. Gaps are closed by making doglegs
on contacts. But this mechanism could only work when the database if
fully initialised (the cache is up to date). Otherwise various problems
arise, in the canonization process for example.
* New: In Katabatic, in AutoContactTerminal::getNativeConstraintBox(),
when anchored on a RoutingPad, now take account the potential rotation
of the Path's transformation. Here again, for the chip's pads.
* New: In Kite, support for reduced AutoSegment. TrackSegment associateds
to reduced AutoSegment are *not* inserted into track to become
effectively invisibles. When a segment becomes reduced, a TrackEvent
is generated to remove it. Conversely when it is raised a RoutingEvent
is created/rescheduled to insert it. All this is mostly managed inside
the Session::revalidate() method.
* New: In Kite, in KiteEngine::createGlobalGraph(), in case of a chip,
mark all global routing vertexes (Knik) that are under a pad, as blockeds.
* Bug: In Cumulus, in PadsCorona.Side.getAxis(), inversion between X and
Y coordinate of the chip size. Did not show until a non-square chip
was routed (i.e. our MIPS R3000).
* Change: In Stratus1, in st_placement.py add the ClockBuffer class for
backward compatibility with the MIPS32 bench. Have to review this
functionnality coming from the deprecated placeAndroute.py.
In st_instance.py, no longer creates the Plug ring of a Net.
In my opinion it just clutter the display until the P&R is called.
Can re-enable later as an option (in Unicorn).
* Change: In Unicorn, in cgt.py, more reliable way of loading then running
user supplied scripts. Borrowed from alliance-checker-toolkit doChip.py .
2015-08-16 16:29:28 -05:00
|
|
|
// return;
|
|
|
|
// }
|
* ./katabatic:
- New: ChipTools, regroup all datas and utilities to manage a full-chip
design.
- Change: In LoadGrByNet/GCellConfiguration::_GCell_xG_1Pad(), uses straight
perpandicular wires on top & right pads. The GCells under those connectors
are fully saturated, wires must go out as straight as possible.
- Change: In AutoHorizontal/AutoVertical, update the "terminal" flag after
slackening. This is to avoid global that are no longer connected to
terminals behave as such.
- Change: In AutoSegment::canMoveUp() & canPivotUp(), prevent all local segments
to go up. This is to avoid cluttering upper levels with small segments.
- Change: In GCellConfiguration, for xG_xL3, detect straight vertical topologies
and for them to be fixed (new flag FIXED_GLOBAL set by the topological
builder instead of the constructor). This prevent the router to do
stupid things...
To sets the "Fixed" flag *after* the axis of global segments have been
positionned correctly adds a "_toFixGlobals" static table lookup.
- Change: In GCell::stepDesaturate(), if less than one free track remains in
the upper layer, do not move up the segment. This allows from a minimum
free room for expansion.
- Change: In GCell::_getString(), display indexes and layer names in various
saturation tables.
- Change: In AutoSegment, allow move up of local Segment. ::moveUp() and
::canMoveUp() arguments go from booleans to flags, which are more
explicits.
2010-12-04 09:25:18 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
// Check for straight lines, which are not managed by _do_xG().
|
|
|
|
if (_connexity.fields.globals == 1) {
|
Added support for "same layer" dogleg. Big fix for pad routing.
* Change: In Knik, in Vertex, add a "blocked" flag to signal disabled
vertexes in the grid (must not be used by the global router).
Modificate the Graph::getVertex() method so that when a vertex
is geometrically queried, if is a blocked one, return a non-blocked
neighbor. This mechanism is introduced to, at last, prevent the
global router to go *under* the pad in case of a commplete chip.
* New: In Katabatic, in AutoSegment, a new state has been added: "reduced".
A reduced segment is in the same layer as it's perpandiculars.
To be reduced, a segments has to be connected on source & target to
AutoContactTurn, both of the perpandiculars must be of the same layer
(below or above) and it's length must not exceed one pitch in the
perpandicular direction.
To reduce an AutoSegment, call ::reduce() and to revert the state,
call ::raise(). Two associated predicates are associated:
::canReduce() and ::mustRaise().
Note: No two adjacent segments can be reduced at the same time.
* Bug: In Katabatic, in GCellTopology, add a new method ::doRp_AccessPad()
to connect to the pads. Create wiring, fixed and non managed by
Katabatic, to connect the pad connector layer to the lowest routing
layers (depth 1 & 2). The former implementation was sometimes leading
to gaps (sheared contact) that *must not* occurs during the building
stage.
Remark: This bug did put under the light the fact that the initial
wiring must be created without gaps. Gaps are closed by making doglegs
on contacts. But this mechanism could only work when the database if
fully initialised (the cache is up to date). Otherwise various problems
arise, in the canonization process for example.
* New: In Katabatic, in AutoContactTerminal::getNativeConstraintBox(),
when anchored on a RoutingPad, now take account the potential rotation
of the Path's transformation. Here again, for the chip's pads.
* New: In Kite, support for reduced AutoSegment. TrackSegment associateds
to reduced AutoSegment are *not* inserted into track to become
effectively invisibles. When a segment becomes reduced, a TrackEvent
is generated to remove it. Conversely when it is raised a RoutingEvent
is created/rescheduled to insert it. All this is mostly managed inside
the Session::revalidate() method.
* New: In Kite, in KiteEngine::createGlobalGraph(), in case of a chip,
mark all global routing vertexes (Knik) that are under a pad, as blockeds.
* Bug: In Cumulus, in PadsCorona.Side.getAxis(), inversion between X and
Y coordinate of the chip size. Did not show until a non-square chip
was routed (i.e. our MIPS R3000).
* Change: In Stratus1, in st_placement.py add the ClockBuffer class for
backward compatibility with the MIPS32 bench. Have to review this
functionnality coming from the deprecated placeAndroute.py.
In st_instance.py, no longer creates the Plug ring of a Net.
In my opinion it just clutter the display until the P&R is called.
Can re-enable later as an option (in Unicorn).
* Change: In Unicorn, in cgt.py, more reliable way of loading then running
user supplied scripts. Borrowed from alliance-checker-toolkit doChip.py .
2015-08-16 16:29:28 -05:00
|
|
|
if ( (westPad and (_east != NULL))
|
|
|
|
or (eastPad and (_west != NULL)) ) {
|
2013-12-03 18:58:58 -06:00
|
|
|
AutoContact* turn = AutoContactTurn::create( _gcell, _net, Session::getContactLayer(1) );
|
Added support for "same layer" dogleg. Big fix for pad routing.
* Change: In Knik, in Vertex, add a "blocked" flag to signal disabled
vertexes in the grid (must not be used by the global router).
Modificate the Graph::getVertex() method so that when a vertex
is geometrically queried, if is a blocked one, return a non-blocked
neighbor. This mechanism is introduced to, at last, prevent the
global router to go *under* the pad in case of a commplete chip.
* New: In Katabatic, in AutoSegment, a new state has been added: "reduced".
A reduced segment is in the same layer as it's perpandiculars.
To be reduced, a segments has to be connected on source & target to
AutoContactTurn, both of the perpandiculars must be of the same layer
(below or above) and it's length must not exceed one pitch in the
perpandicular direction.
To reduce an AutoSegment, call ::reduce() and to revert the state,
call ::raise(). Two associated predicates are associated:
::canReduce() and ::mustRaise().
Note: No two adjacent segments can be reduced at the same time.
* Bug: In Katabatic, in GCellTopology, add a new method ::doRp_AccessPad()
to connect to the pads. Create wiring, fixed and non managed by
Katabatic, to connect the pad connector layer to the lowest routing
layers (depth 1 & 2). The former implementation was sometimes leading
to gaps (sheared contact) that *must not* occurs during the building
stage.
Remark: This bug did put under the light the fact that the initial
wiring must be created without gaps. Gaps are closed by making doglegs
on contacts. But this mechanism could only work when the database if
fully initialised (the cache is up to date). Otherwise various problems
arise, in the canonization process for example.
* New: In Katabatic, in AutoContactTerminal::getNativeConstraintBox(),
when anchored on a RoutingPad, now take account the potential rotation
of the Path's transformation. Here again, for the chip's pads.
* New: In Kite, support for reduced AutoSegment. TrackSegment associateds
to reduced AutoSegment are *not* inserted into track to become
effectively invisibles. When a segment becomes reduced, a TrackEvent
is generated to remove it. Conversely when it is raised a RoutingEvent
is created/rescheduled to insert it. All this is mostly managed inside
the Session::revalidate() method.
* New: In Kite, in KiteEngine::createGlobalGraph(), in case of a chip,
mark all global routing vertexes (Knik) that are under a pad, as blockeds.
* Bug: In Cumulus, in PadsCorona.Side.getAxis(), inversion between X and
Y coordinate of the chip size. Did not show until a non-square chip
was routed (i.e. our MIPS R3000).
* Change: In Stratus1, in st_placement.py add the ClockBuffer class for
backward compatibility with the MIPS32 bench. Have to review this
functionnality coming from the deprecated placeAndroute.py.
In st_instance.py, no longer creates the Plug ring of a Net.
In my opinion it just clutter the display until the P&R is called.
Can re-enable later as an option (in Unicorn).
* Change: In Unicorn, in cgt.py, more reliable way of loading then running
user supplied scripts. Borrowed from alliance-checker-toolkit doChip.py .
2015-08-16 16:29:28 -05:00
|
|
|
_northEastContact = _southWestContact
|
|
|
|
= AutoContactTurn::create( _gcell, _net, Session::getContactLayer(1) );
|
2013-12-03 18:58:58 -06:00
|
|
|
AutoSegment::create( source, turn, KbHorizontal );
|
|
|
|
AutoSegment::create( turn, _northEastContact, KbVertical );
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_tabw(145,-1);
|
2013-12-03 18:58:58 -06:00
|
|
|
return;
|
Added support for "same layer" dogleg. Big fix for pad routing.
* Change: In Knik, in Vertex, add a "blocked" flag to signal disabled
vertexes in the grid (must not be used by the global router).
Modificate the Graph::getVertex() method so that when a vertex
is geometrically queried, if is a blocked one, return a non-blocked
neighbor. This mechanism is introduced to, at last, prevent the
global router to go *under* the pad in case of a commplete chip.
* New: In Katabatic, in AutoSegment, a new state has been added: "reduced".
A reduced segment is in the same layer as it's perpandiculars.
To be reduced, a segments has to be connected on source & target to
AutoContactTurn, both of the perpandiculars must be of the same layer
(below or above) and it's length must not exceed one pitch in the
perpandicular direction.
To reduce an AutoSegment, call ::reduce() and to revert the state,
call ::raise(). Two associated predicates are associated:
::canReduce() and ::mustRaise().
Note: No two adjacent segments can be reduced at the same time.
* Bug: In Katabatic, in GCellTopology, add a new method ::doRp_AccessPad()
to connect to the pads. Create wiring, fixed and non managed by
Katabatic, to connect the pad connector layer to the lowest routing
layers (depth 1 & 2). The former implementation was sometimes leading
to gaps (sheared contact) that *must not* occurs during the building
stage.
Remark: This bug did put under the light the fact that the initial
wiring must be created without gaps. Gaps are closed by making doglegs
on contacts. But this mechanism could only work when the database if
fully initialised (the cache is up to date). Otherwise various problems
arise, in the canonization process for example.
* New: In Katabatic, in AutoContactTerminal::getNativeConstraintBox(),
when anchored on a RoutingPad, now take account the potential rotation
of the Path's transformation. Here again, for the chip's pads.
* New: In Kite, support for reduced AutoSegment. TrackSegment associateds
to reduced AutoSegment are *not* inserted into track to become
effectively invisibles. When a segment becomes reduced, a TrackEvent
is generated to remove it. Conversely when it is raised a RoutingEvent
is created/rescheduled to insert it. All this is mostly managed inside
the Session::revalidate() method.
* New: In Kite, in KiteEngine::createGlobalGraph(), in case of a chip,
mark all global routing vertexes (Knik) that are under a pad, as blockeds.
* Bug: In Cumulus, in PadsCorona.Side.getAxis(), inversion between X and
Y coordinate of the chip size. Did not show until a non-square chip
was routed (i.e. our MIPS R3000).
* Change: In Stratus1, in st_placement.py add the ClockBuffer class for
backward compatibility with the MIPS32 bench. Have to review this
functionnality coming from the deprecated placeAndroute.py.
In st_instance.py, no longer creates the Plug ring of a Net.
In my opinion it just clutter the display until the P&R is called.
Can re-enable later as an option (in Unicorn).
* Change: In Unicorn, in cgt.py, more reliable way of loading then running
user supplied scripts. Borrowed from alliance-checker-toolkit doChip.py .
2015-08-16 16:29:28 -05:00
|
|
|
} else if ( (southPad and (_north != NULL))
|
|
|
|
or (northPad and (_south != NULL)) ) {
|
2013-12-03 18:58:58 -06:00
|
|
|
AutoContact* turn = AutoContactTurn::create( _gcell, _net, Session::getContactLayer(1) );
|
Added support for "same layer" dogleg. Big fix for pad routing.
* Change: In Knik, in Vertex, add a "blocked" flag to signal disabled
vertexes in the grid (must not be used by the global router).
Modificate the Graph::getVertex() method so that when a vertex
is geometrically queried, if is a blocked one, return a non-blocked
neighbor. This mechanism is introduced to, at last, prevent the
global router to go *under* the pad in case of a commplete chip.
* New: In Katabatic, in AutoSegment, a new state has been added: "reduced".
A reduced segment is in the same layer as it's perpandiculars.
To be reduced, a segments has to be connected on source & target to
AutoContactTurn, both of the perpandiculars must be of the same layer
(below or above) and it's length must not exceed one pitch in the
perpandicular direction.
To reduce an AutoSegment, call ::reduce() and to revert the state,
call ::raise(). Two associated predicates are associated:
::canReduce() and ::mustRaise().
Note: No two adjacent segments can be reduced at the same time.
* Bug: In Katabatic, in GCellTopology, add a new method ::doRp_AccessPad()
to connect to the pads. Create wiring, fixed and non managed by
Katabatic, to connect the pad connector layer to the lowest routing
layers (depth 1 & 2). The former implementation was sometimes leading
to gaps (sheared contact) that *must not* occurs during the building
stage.
Remark: This bug did put under the light the fact that the initial
wiring must be created without gaps. Gaps are closed by making doglegs
on contacts. But this mechanism could only work when the database if
fully initialised (the cache is up to date). Otherwise various problems
arise, in the canonization process for example.
* New: In Katabatic, in AutoContactTerminal::getNativeConstraintBox(),
when anchored on a RoutingPad, now take account the potential rotation
of the Path's transformation. Here again, for the chip's pads.
* New: In Kite, support for reduced AutoSegment. TrackSegment associateds
to reduced AutoSegment are *not* inserted into track to become
effectively invisibles. When a segment becomes reduced, a TrackEvent
is generated to remove it. Conversely when it is raised a RoutingEvent
is created/rescheduled to insert it. All this is mostly managed inside
the Session::revalidate() method.
* New: In Kite, in KiteEngine::createGlobalGraph(), in case of a chip,
mark all global routing vertexes (Knik) that are under a pad, as blockeds.
* Bug: In Cumulus, in PadsCorona.Side.getAxis(), inversion between X and
Y coordinate of the chip size. Did not show until a non-square chip
was routed (i.e. our MIPS R3000).
* Change: In Stratus1, in st_placement.py add the ClockBuffer class for
backward compatibility with the MIPS32 bench. Have to review this
functionnality coming from the deprecated placeAndroute.py.
In st_instance.py, no longer creates the Plug ring of a Net.
In my opinion it just clutter the display until the P&R is called.
Can re-enable later as an option (in Unicorn).
* Change: In Unicorn, in cgt.py, more reliable way of loading then running
user supplied scripts. Borrowed from alliance-checker-toolkit doChip.py .
2015-08-16 16:29:28 -05:00
|
|
|
_northEastContact = _southWestContact
|
|
|
|
= AutoContactTurn::create( _gcell, _net, Session::getContactLayer(1) );
|
2013-12-03 18:58:58 -06:00
|
|
|
AutoSegment::create( source, turn, KbVertical );
|
|
|
|
AutoSegment::create( turn, _northEastContact, KbHorizontal );
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_tabw(145,-1);
|
2013-12-03 18:58:58 -06:00
|
|
|
return;
|
|
|
|
}
|
* ./katabatic:
- Change: In loadGlobalRouting(), more exlicit message as to why a net is
filtered out the routing set (POWER, GROUND, CLOCK or BLOCKAGE).
- New: ChipTool, module with utilities specific to chip routing. Containing
a function to pre-break wires around a block.
- New: In loadGlobalRouting/GCellConfiguration, adds supports for RoutingPad
connected on chip's pad.
- New: In GCellGrid/GCell, can compute the density in three modes: average,
max of H/V and max of layer. Currently implemented in getMaxHVDensity(),
should be unificated with getDensity(). Used for display purposes.
- Bug: In AutoContact, when splitting a contact, add a specific check for
"one layer span".
- Bug: In AutoSegment::canMoveUp(), do not move up fixed segments.
2010-11-16 07:59:38 -06:00
|
|
|
}
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
++_connexity.fields.globals;
|
|
|
|
--_connexity.fields.Pad;
|
* ./katabatic:
- Change: In loadGlobalRouting(), more exlicit message as to why a net is
filtered out the routing set (POWER, GROUND, CLOCK or BLOCKAGE).
- New: ChipTool, module with utilities specific to chip routing. Containing
a function to pre-break wires around a block.
- New: In loadGlobalRouting/GCellConfiguration, adds supports for RoutingPad
connected on chip's pad.
- New: In GCellGrid/GCell, can compute the density in three modes: average,
max of H/V and max of layer. Currently implemented in getMaxHVDensity(),
should be unificated with getDensity(). Used for display purposes.
- Bug: In AutoContact, when splitting a contact, add a specific check for
"one layer span".
- Bug: In AutoSegment::canMoveUp(), do not move up fixed segments.
2010-11-16 07:59:38 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
if (westPad ) _west = source->getBodyHook();
|
Added support for "same layer" dogleg. Big fix for pad routing.
* Change: In Knik, in Vertex, add a "blocked" flag to signal disabled
vertexes in the grid (must not be used by the global router).
Modificate the Graph::getVertex() method so that when a vertex
is geometrically queried, if is a blocked one, return a non-blocked
neighbor. This mechanism is introduced to, at last, prevent the
global router to go *under* the pad in case of a commplete chip.
* New: In Katabatic, in AutoSegment, a new state has been added: "reduced".
A reduced segment is in the same layer as it's perpandiculars.
To be reduced, a segments has to be connected on source & target to
AutoContactTurn, both of the perpandiculars must be of the same layer
(below or above) and it's length must not exceed one pitch in the
perpandicular direction.
To reduce an AutoSegment, call ::reduce() and to revert the state,
call ::raise(). Two associated predicates are associated:
::canReduce() and ::mustRaise().
Note: No two adjacent segments can be reduced at the same time.
* Bug: In Katabatic, in GCellTopology, add a new method ::doRp_AccessPad()
to connect to the pads. Create wiring, fixed and non managed by
Katabatic, to connect the pad connector layer to the lowest routing
layers (depth 1 & 2). The former implementation was sometimes leading
to gaps (sheared contact) that *must not* occurs during the building
stage.
Remark: This bug did put under the light the fact that the initial
wiring must be created without gaps. Gaps are closed by making doglegs
on contacts. But this mechanism could only work when the database if
fully initialised (the cache is up to date). Otherwise various problems
arise, in the canonization process for example.
* New: In Katabatic, in AutoContactTerminal::getNativeConstraintBox(),
when anchored on a RoutingPad, now take account the potential rotation
of the Path's transformation. Here again, for the chip's pads.
* New: In Kite, support for reduced AutoSegment. TrackSegment associateds
to reduced AutoSegment are *not* inserted into track to become
effectively invisibles. When a segment becomes reduced, a TrackEvent
is generated to remove it. Conversely when it is raised a RoutingEvent
is created/rescheduled to insert it. All this is mostly managed inside
the Session::revalidate() method.
* New: In Kite, in KiteEngine::createGlobalGraph(), in case of a chip,
mark all global routing vertexes (Knik) that are under a pad, as blockeds.
* Bug: In Cumulus, in PadsCorona.Side.getAxis(), inversion between X and
Y coordinate of the chip size. Did not show until a non-square chip
was routed (i.e. our MIPS R3000).
* Change: In Stratus1, in st_placement.py add the ClockBuffer class for
backward compatibility with the MIPS32 bench. Have to review this
functionnality coming from the deprecated placeAndroute.py.
In st_instance.py, no longer creates the Plug ring of a Net.
In my opinion it just clutter the display until the P&R is called.
Can re-enable later as an option (in Unicorn).
* Change: In Unicorn, in cgt.py, more reliable way of loading then running
user supplied scripts. Borrowed from alliance-checker-toolkit doChip.py .
2015-08-16 16:29:28 -05:00
|
|
|
if (eastPad ) _east = source->getBodyHook();
|
2013-12-03 18:58:58 -06:00
|
|
|
if (southPad) _south = source->getBodyHook();
|
Added support for "same layer" dogleg. Big fix for pad routing.
* Change: In Knik, in Vertex, add a "blocked" flag to signal disabled
vertexes in the grid (must not be used by the global router).
Modificate the Graph::getVertex() method so that when a vertex
is geometrically queried, if is a blocked one, return a non-blocked
neighbor. This mechanism is introduced to, at last, prevent the
global router to go *under* the pad in case of a commplete chip.
* New: In Katabatic, in AutoSegment, a new state has been added: "reduced".
A reduced segment is in the same layer as it's perpandiculars.
To be reduced, a segments has to be connected on source & target to
AutoContactTurn, both of the perpandiculars must be of the same layer
(below or above) and it's length must not exceed one pitch in the
perpandicular direction.
To reduce an AutoSegment, call ::reduce() and to revert the state,
call ::raise(). Two associated predicates are associated:
::canReduce() and ::mustRaise().
Note: No two adjacent segments can be reduced at the same time.
* Bug: In Katabatic, in GCellTopology, add a new method ::doRp_AccessPad()
to connect to the pads. Create wiring, fixed and non managed by
Katabatic, to connect the pad connector layer to the lowest routing
layers (depth 1 & 2). The former implementation was sometimes leading
to gaps (sheared contact) that *must not* occurs during the building
stage.
Remark: This bug did put under the light the fact that the initial
wiring must be created without gaps. Gaps are closed by making doglegs
on contacts. But this mechanism could only work when the database if
fully initialised (the cache is up to date). Otherwise various problems
arise, in the canonization process for example.
* New: In Katabatic, in AutoContactTerminal::getNativeConstraintBox(),
when anchored on a RoutingPad, now take account the potential rotation
of the Path's transformation. Here again, for the chip's pads.
* New: In Kite, support for reduced AutoSegment. TrackSegment associateds
to reduced AutoSegment are *not* inserted into track to become
effectively invisibles. When a segment becomes reduced, a TrackEvent
is generated to remove it. Conversely when it is raised a RoutingEvent
is created/rescheduled to insert it. All this is mostly managed inside
the Session::revalidate() method.
* New: In Kite, in KiteEngine::createGlobalGraph(), in case of a chip,
mark all global routing vertexes (Knik) that are under a pad, as blockeds.
* Bug: In Cumulus, in PadsCorona.Side.getAxis(), inversion between X and
Y coordinate of the chip size. Did not show until a non-square chip
was routed (i.e. our MIPS R3000).
* Change: In Stratus1, in st_placement.py add the ClockBuffer class for
backward compatibility with the MIPS32 bench. Have to review this
functionnality coming from the deprecated placeAndroute.py.
In st_instance.py, no longer creates the Plug ring of a Net.
In my opinion it just clutter the display until the P&R is called.
Can re-enable later as an option (in Unicorn).
* Change: In Unicorn, in cgt.py, more reliable way of loading then running
user supplied scripts. Borrowed from alliance-checker-toolkit doChip.py .
2015-08-16 16:29:28 -05:00
|
|
|
if (northPad) _north = source->getBodyHook();
|
* ./katabatic:
- Change: In loadGlobalRouting(), more exlicit message as to why a net is
filtered out the routing set (POWER, GROUND, CLOCK or BLOCKAGE).
- New: ChipTool, module with utilities specific to chip routing. Containing
a function to pre-break wires around a block.
- New: In loadGlobalRouting/GCellConfiguration, adds supports for RoutingPad
connected on chip's pad.
- New: In GCellGrid/GCell, can compute the density in three modes: average,
max of H/V and max of layer. Currently implemented in getMaxHVDensity(),
should be unificated with getDensity(). Used for display purposes.
- Bug: In AutoContact, when splitting a contact, add a specific check for
"one layer span".
- Bug: In AutoSegment::canMoveUp(), do not move up fixed segments.
2010-11-16 07:59:38 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
_do_xG();
|
* ./katabatic:
- Change: In loadGlobalRouting(), more exlicit message as to why a net is
filtered out the routing set (POWER, GROUND, CLOCK or BLOCKAGE).
- New: ChipTool, module with utilities specific to chip routing. Containing
a function to pre-break wires around a block.
- New: In loadGlobalRouting/GCellConfiguration, adds supports for RoutingPad
connected on chip's pad.
- New: In GCellGrid/GCell, can compute the density in three modes: average,
max of H/V and max of layer. Currently implemented in getMaxHVDensity(),
should be unificated with getDensity(). Used for display purposes.
- Bug: In AutoContact, when splitting a contact, add a specific check for
"one layer span".
- Bug: In AutoSegment::canMoveUp(), do not move up fixed segments.
2010-11-16 07:59:38 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
if (westPad) {
|
|
|
|
AutoSegment::create( source, _southWestContact, KbHorizontal );
|
|
|
|
_west = NULL;
|
* ./katabatic:
- Change: In loadGlobalRouting(), more exlicit message as to why a net is
filtered out the routing set (POWER, GROUND, CLOCK or BLOCKAGE).
- New: ChipTool, module with utilities specific to chip routing. Containing
a function to pre-break wires around a block.
- New: In loadGlobalRouting/GCellConfiguration, adds supports for RoutingPad
connected on chip's pad.
- New: In GCellGrid/GCell, can compute the density in three modes: average,
max of H/V and max of layer. Currently implemented in getMaxHVDensity(),
should be unificated with getDensity(). Used for display purposes.
- Bug: In AutoContact, when splitting a contact, add a specific check for
"one layer span".
- Bug: In AutoSegment::canMoveUp(), do not move up fixed segments.
2010-11-16 07:59:38 -06:00
|
|
|
}
|
Added support for "same layer" dogleg. Big fix for pad routing.
* Change: In Knik, in Vertex, add a "blocked" flag to signal disabled
vertexes in the grid (must not be used by the global router).
Modificate the Graph::getVertex() method so that when a vertex
is geometrically queried, if is a blocked one, return a non-blocked
neighbor. This mechanism is introduced to, at last, prevent the
global router to go *under* the pad in case of a commplete chip.
* New: In Katabatic, in AutoSegment, a new state has been added: "reduced".
A reduced segment is in the same layer as it's perpandiculars.
To be reduced, a segments has to be connected on source & target to
AutoContactTurn, both of the perpandiculars must be of the same layer
(below or above) and it's length must not exceed one pitch in the
perpandicular direction.
To reduce an AutoSegment, call ::reduce() and to revert the state,
call ::raise(). Two associated predicates are associated:
::canReduce() and ::mustRaise().
Note: No two adjacent segments can be reduced at the same time.
* Bug: In Katabatic, in GCellTopology, add a new method ::doRp_AccessPad()
to connect to the pads. Create wiring, fixed and non managed by
Katabatic, to connect the pad connector layer to the lowest routing
layers (depth 1 & 2). The former implementation was sometimes leading
to gaps (sheared contact) that *must not* occurs during the building
stage.
Remark: This bug did put under the light the fact that the initial
wiring must be created without gaps. Gaps are closed by making doglegs
on contacts. But this mechanism could only work when the database if
fully initialised (the cache is up to date). Otherwise various problems
arise, in the canonization process for example.
* New: In Katabatic, in AutoContactTerminal::getNativeConstraintBox(),
when anchored on a RoutingPad, now take account the potential rotation
of the Path's transformation. Here again, for the chip's pads.
* New: In Kite, support for reduced AutoSegment. TrackSegment associateds
to reduced AutoSegment are *not* inserted into track to become
effectively invisibles. When a segment becomes reduced, a TrackEvent
is generated to remove it. Conversely when it is raised a RoutingEvent
is created/rescheduled to insert it. All this is mostly managed inside
the Session::revalidate() method.
* New: In Kite, in KiteEngine::createGlobalGraph(), in case of a chip,
mark all global routing vertexes (Knik) that are under a pad, as blockeds.
* Bug: In Cumulus, in PadsCorona.Side.getAxis(), inversion between X and
Y coordinate of the chip size. Did not show until a non-square chip
was routed (i.e. our MIPS R3000).
* Change: In Stratus1, in st_placement.py add the ClockBuffer class for
backward compatibility with the MIPS32 bench. Have to review this
functionnality coming from the deprecated placeAndroute.py.
In st_instance.py, no longer creates the Plug ring of a Net.
In my opinion it just clutter the display until the P&R is called.
Can re-enable later as an option (in Unicorn).
* Change: In Unicorn, in cgt.py, more reliable way of loading then running
user supplied scripts. Borrowed from alliance-checker-toolkit doChip.py .
2015-08-16 16:29:28 -05:00
|
|
|
if (eastPad) {
|
|
|
|
AutoSegment::create( source, _northEastContact, KbHorizontal );
|
|
|
|
_east = NULL;
|
|
|
|
}
|
2013-12-03 18:58:58 -06:00
|
|
|
if (southPad) {
|
|
|
|
AutoSegment::create( source, _southWestContact, KbVertical );
|
|
|
|
_south = NULL;
|
|
|
|
}
|
Added support for "same layer" dogleg. Big fix for pad routing.
* Change: In Knik, in Vertex, add a "blocked" flag to signal disabled
vertexes in the grid (must not be used by the global router).
Modificate the Graph::getVertex() method so that when a vertex
is geometrically queried, if is a blocked one, return a non-blocked
neighbor. This mechanism is introduced to, at last, prevent the
global router to go *under* the pad in case of a commplete chip.
* New: In Katabatic, in AutoSegment, a new state has been added: "reduced".
A reduced segment is in the same layer as it's perpandiculars.
To be reduced, a segments has to be connected on source & target to
AutoContactTurn, both of the perpandiculars must be of the same layer
(below or above) and it's length must not exceed one pitch in the
perpandicular direction.
To reduce an AutoSegment, call ::reduce() and to revert the state,
call ::raise(). Two associated predicates are associated:
::canReduce() and ::mustRaise().
Note: No two adjacent segments can be reduced at the same time.
* Bug: In Katabatic, in GCellTopology, add a new method ::doRp_AccessPad()
to connect to the pads. Create wiring, fixed and non managed by
Katabatic, to connect the pad connector layer to the lowest routing
layers (depth 1 & 2). The former implementation was sometimes leading
to gaps (sheared contact) that *must not* occurs during the building
stage.
Remark: This bug did put under the light the fact that the initial
wiring must be created without gaps. Gaps are closed by making doglegs
on contacts. But this mechanism could only work when the database if
fully initialised (the cache is up to date). Otherwise various problems
arise, in the canonization process for example.
* New: In Katabatic, in AutoContactTerminal::getNativeConstraintBox(),
when anchored on a RoutingPad, now take account the potential rotation
of the Path's transformation. Here again, for the chip's pads.
* New: In Kite, support for reduced AutoSegment. TrackSegment associateds
to reduced AutoSegment are *not* inserted into track to become
effectively invisibles. When a segment becomes reduced, a TrackEvent
is generated to remove it. Conversely when it is raised a RoutingEvent
is created/rescheduled to insert it. All this is mostly managed inside
the Session::revalidate() method.
* New: In Kite, in KiteEngine::createGlobalGraph(), in case of a chip,
mark all global routing vertexes (Knik) that are under a pad, as blockeds.
* Bug: In Cumulus, in PadsCorona.Side.getAxis(), inversion between X and
Y coordinate of the chip size. Did not show until a non-square chip
was routed (i.e. our MIPS R3000).
* Change: In Stratus1, in st_placement.py add the ClockBuffer class for
backward compatibility with the MIPS32 bench. Have to review this
functionnality coming from the deprecated placeAndroute.py.
In st_instance.py, no longer creates the Plug ring of a Net.
In my opinion it just clutter the display until the P&R is called.
Can re-enable later as an option (in Unicorn).
* Change: In Unicorn, in cgt.py, more reliable way of loading then running
user supplied scripts. Borrowed from alliance-checker-toolkit doChip.py .
2015-08-16 16:29:28 -05:00
|
|
|
if (northPad) {
|
|
|
|
AutoSegment::create( source, _northEastContact, KbVertical );
|
|
|
|
_north = NULL;
|
|
|
|
}
|
2013-12-03 18:58:58 -06:00
|
|
|
--_connexity.fields.globals;
|
* ./katabatic:
- Change: In loadGlobalRouting(), more exlicit message as to why a net is
filtered out the routing set (POWER, GROUND, CLOCK or BLOCKAGE).
- New: ChipTool, module with utilities specific to chip routing. Containing
a function to pre-break wires around a block.
- New: In loadGlobalRouting/GCellConfiguration, adds supports for RoutingPad
connected on chip's pad.
- New: In GCellGrid/GCell, can compute the density in three modes: average,
max of H/V and max of layer. Currently implemented in getMaxHVDensity(),
should be unificated with getDensity(). Used for display purposes.
- Bug: In AutoContact, when splitting a contact, add a specific check for
"one layer span".
- Bug: In AutoSegment::canMoveUp(), do not move up fixed segments.
2010-11-16 07:59:38 -06:00
|
|
|
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_tabw(145,-1);
|
* ./katabatic:
- Change: In loadGlobalRouting(), more exlicit message as to why a net is
filtered out the routing set (POWER, GROUND, CLOCK or BLOCKAGE).
- New: ChipTool, module with utilities specific to chip routing. Containing
a function to pre-break wires around a block.
- New: In loadGlobalRouting/GCellConfiguration, adds supports for RoutingPad
connected on chip's pad.
- New: In GCellGrid/GCell, can compute the density in three modes: average,
max of H/V and max of layer. Currently implemented in getMaxHVDensity(),
should be unificated with getDensity(). Used for display purposes.
- Bug: In AutoContact, when splitting a contact, add a specific check for
"one layer span".
- Bug: In AutoSegment::canMoveUp(), do not move up fixed segments.
2010-11-16 07:59:38 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
void GCellTopology::_do_1G_1PinM2 ()
|
2010-03-09 09:24:29 -06:00
|
|
|
{
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,1) << "_do_1G_1PinM2() [Managed Configuration - Optimized] " << _topology << endl;
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
AutoContact* rpContact = doRp_Access( _gcell, _routingPads[0], NoFlags );
|
|
|
|
AutoContact* turn1 = AutoContactTurn::create( _gcell, _net, Session::getContactLayer(1) );
|
|
|
|
AutoSegment::create( rpContact, turn1, KbVertical );
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
if (_north or _south) {
|
|
|
|
AutoContact* turn2 = AutoContactTurn::create( _gcell, _net, Session::getContactLayer(1) );
|
|
|
|
AutoSegment::create( turn1, turn2, KbHorizontal );
|
|
|
|
turn1 = turn2;
|
|
|
|
}
|
|
|
|
_southWestContact = _northEastContact = turn1;
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_tabw(145,-1);
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
void GCellTopology::_do_1G_1M1 ()
|
2010-03-09 09:24:29 -06:00
|
|
|
{
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,1) << "_do_1G_1M1() [Managed Configuration - Optimized] " << _topology << endl;
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
unsigned int flags = NoFlags;
|
2014-03-15 04:47:37 -05:00
|
|
|
if (_east ) { flags |= HAccess; }
|
|
|
|
else if (_west ) { flags |= HAccess; }
|
|
|
|
else if (_north) { flags |= VSmall; }
|
|
|
|
else if (_south) { flags |= VSmall; }
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
_southWestContact = _northEastContact = doRp_Access( _gcell, _routingPads[0], flags );
|
|
|
|
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_tabw(145,-1);
|
2013-12-03 18:58:58 -06:00
|
|
|
}
|
2010-03-09 09:24:29 -06:00
|
|
|
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
void GCellTopology::_do_1G_xM1 ()
|
|
|
|
{
|
Solve the template lookup problem in tstream.
* Bug: In Hurricane, in tstream (Commons.h), in the *template* overload of
operator<<(), do not use the operator<<() of ostream as it will be
looked up in "stage 1" (template definition) and so will miss all the
overloads added later and built over getString<>(). Instead, make use
of getString<>(), which, as another template will be looked up in
"stage 2" (template instanciation) and at that point will have all the
needed template specialisation of getString<>().
We also need to define new stream manipulators to be able to create
a matching template overload not dependant from the implementation.
To avoid name clashes, we prefix a 't'. For now, only 'tsetw()' is
refined.
As a side effect, we cannot directly print bit-fields into the stream,
we must go through an intermediate variable (happens once in AutoContact).
2016-07-19 09:02:55 -05:00
|
|
|
cdebug_log(145,1) << "_do_1G_" << (int)_connexity.fields.M1 << "M1() [Managed Configuration]" << endl;
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
sort( _routingPads.begin(), _routingPads.end(), SortRpByX(NoFlags) ); // increasing X.
|
|
|
|
for ( unsigned int i=1 ; i<_routingPads.size() ; ++i ) {
|
|
|
|
AutoContact* leftContact = doRp_Access( _gcell, _routingPads[i-1], HAccess );
|
|
|
|
AutoContact* rightContact = doRp_Access( _gcell, _routingPads[i ], HAccess );
|
|
|
|
AutoSegment::create( leftContact, rightContact, KbHorizontal );
|
|
|
|
}
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
Component* globalRp = NULL;
|
|
|
|
if (_east) globalRp = _routingPads[_routingPads.size()-1];
|
|
|
|
else if (_west) globalRp = _routingPads[0];
|
|
|
|
else {
|
|
|
|
globalRp = _routingPads[0];
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "| Initial N/S Global RP: " << globalRp << endl;
|
2013-12-03 18:58:58 -06:00
|
|
|
for ( unsigned int i=1 ; i<_routingPads.size() ; ++i ) {
|
|
|
|
if (_routingPads[i]->getBoundingBox().getHeight() > globalRp->getBoundingBox().getHeight()) {
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "| Better RP: " << globalRp << endl;
|
2013-12-03 18:58:58 -06:00
|
|
|
globalRp = _routingPads[i];
|
|
|
|
}
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
}
|
2013-12-03 18:58:58 -06:00
|
|
|
|
|
|
|
AutoContact* globalContact = doRp_Access( _gcell, globalRp, HAccess );
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
if (_north or _south) {
|
|
|
|
AutoContact* turn = globalContact;
|
|
|
|
globalContact = AutoContactTurn::create( _gcell, _net, Session::getContactLayer(1) );
|
|
|
|
AutoSegment::create( globalContact, turn, KbHorizontal );
|
|
|
|
}
|
|
|
|
|
|
|
|
_northEastContact = _southWestContact = globalContact;
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_tabw(145,-1);
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
void GCellTopology::_do_xG_1M1_1M2 ()
|
2010-03-09 09:24:29 -06:00
|
|
|
{
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,1) << "_do_xG_1M1_1M2() [Managed Configuration]" << endl;
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
Component* rpL1;
|
|
|
|
Component* rpL2;
|
|
|
|
if (_routingPads[0]->getLayer() == Session::getRoutingLayer(0)) {
|
2010-03-09 09:24:29 -06:00
|
|
|
rpL1 = _routingPads[0];
|
|
|
|
rpL2 = _routingPads[1];
|
|
|
|
} else {
|
|
|
|
rpL1 = _routingPads[1];
|
|
|
|
rpL2 = _routingPads[0];
|
|
|
|
}
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "rpL1 := " << rpL1 << endl;
|
|
|
|
cdebug_log(145,0) << "rpL2 := " << rpL2 << endl;
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
AutoContact* rpL1ContactSource = NULL;
|
|
|
|
AutoContact* rpL1ContactTarget = NULL;
|
|
|
|
AutoContact* rpL2ContactSource = NULL;
|
|
|
|
AutoContact* rpL2ContactTarget = NULL;
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
doRp_AutoContacts( _gcell, rpL1, rpL1ContactSource, rpL1ContactTarget, NoFlags );
|
|
|
|
doRp_AutoContacts( _gcell, rpL2, rpL2ContactSource, rpL2ContactTarget, NoFlags );
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
AutoContact* subContact = AutoContactTurn::create( _gcell, _net, Session::getContactLayer(1) );
|
|
|
|
AutoSegment::create( rpL1ContactSource, subContact, KbHorizontal );
|
|
|
|
AutoSegment::create( rpL2ContactSource, subContact, KbVertical );
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
if (_south or _west) {
|
|
|
|
doRp_AutoContacts( _gcell, rpL2, rpL2ContactSource, rpL2ContactTarget, DoSourceContact );
|
|
|
|
if (_south and _west) {
|
|
|
|
_southWestContact = AutoContactHTee::create( _gcell, _net, Session::getContactLayer(2) );
|
|
|
|
AutoSegment::create( rpL2ContactSource, _southWestContact, KbHorizontal );
|
|
|
|
} else {
|
|
|
|
if (_south) {
|
|
|
|
_southWestContact = AutoContactTurn::create( _gcell, _net, Session::getContactLayer(2) );
|
|
|
|
AutoSegment::create( rpL2ContactSource, _southWestContact, KbHorizontal );
|
|
|
|
} else {
|
|
|
|
_southWestContact = rpL2ContactSource;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_north or _east) {
|
|
|
|
doRp_AutoContacts( _gcell, rpL2, rpL2ContactSource, rpL2ContactTarget, DoTargetContact );
|
|
|
|
if (_north and _east) {
|
|
|
|
_northEastContact = AutoContactHTee::create( _gcell, _net, Session::getContactLayer(2) );
|
|
|
|
AutoSegment::create( rpL2ContactTarget, _northEastContact, KbHorizontal );
|
|
|
|
} else {
|
|
|
|
if (_north) {
|
|
|
|
_northEastContact = AutoContactTurn::create( _gcell, _net, Session::getContactLayer(2) );
|
|
|
|
AutoSegment::create( rpL2ContactTarget, _northEastContact, KbHorizontal );
|
|
|
|
} else {
|
|
|
|
_northEastContact = rpL2ContactTarget;
|
|
|
|
}
|
|
|
|
}
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_tabw(145,-1);
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
void GCellTopology::_do_xG_xM1_xM3 ()
|
2010-03-09 09:24:29 -06:00
|
|
|
{
|
Solve the template lookup problem in tstream.
* Bug: In Hurricane, in tstream (Commons.h), in the *template* overload of
operator<<(), do not use the operator<<() of ostream as it will be
looked up in "stage 1" (template definition) and so will miss all the
overloads added later and built over getString<>(). Instead, make use
of getString<>(), which, as another template will be looked up in
"stage 2" (template instanciation) and at that point will have all the
needed template specialisation of getString<>().
We also need to define new stream manipulators to be able to create
a matching template overload not dependant from the implementation.
To avoid name clashes, we prefix a 't'. For now, only 'tsetw()' is
refined.
As a side effect, we cannot directly print bit-fields into the stream,
we must go through an intermediate variable (happens once in AutoContact).
2016-07-19 09:02:55 -05:00
|
|
|
cdebug_log(145,1) << "_do_xG_" << (int)_connexity.fields.M1
|
|
|
|
<< "M1_" << (int)_connexity.fields.M3
|
|
|
|
<< "M3() [G:" << (int)_connexity.fields.globals << " Managed Configuration]" << endl;
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "_connexity: " << _connexity.connexity << endl;
|
|
|
|
cdebug_log(145,0) << "_north: " << _north << endl;
|
|
|
|
cdebug_log(145,0) << "_south: " << _south << endl;
|
|
|
|
cdebug_log(145,0) << "_east: " << _east << endl;
|
|
|
|
cdebug_log(145,0) << "_west: " << _west << endl;
|
2013-12-03 18:58:58 -06:00
|
|
|
|
|
|
|
Component* rpM3 = NULL;
|
|
|
|
if (_routingPads[0]->getLayer() == Session::getRoutingLayer(2))
|
|
|
|
rpM3 = _routingPads[0];
|
|
|
|
|
|
|
|
sort( _routingPads.begin(), _routingPads.end(), SortRpByX(NoFlags) ); // increasing X.
|
|
|
|
for ( unsigned int i=1 ; i<_routingPads.size() ; ++i ) {
|
|
|
|
AutoContact* leftContact = doRp_Access( _gcell, _routingPads[i-1], HAccess );
|
|
|
|
AutoContact* rightContact = doRp_Access( _gcell, _routingPads[i ], HAccess );
|
|
|
|
AutoSegment::create( leftContact, rightContact, KbHorizontal );
|
|
|
|
|
|
|
|
if (not rpM3 and (_routingPads[i]->getLayer() == Session::getRoutingLayer(2)))
|
|
|
|
rpM3 = _routingPads[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
AutoContact* unusedContact = NULL;
|
|
|
|
|
|
|
|
if (rpM3) {
|
|
|
|
// At least one M3 RoutingPad is present: use it.
|
|
|
|
if (_west and not _south) {
|
|
|
|
_southWestContact = doRp_Access( _gcell, _routingPads[0], HAccess );
|
|
|
|
} else if (not _west and _south) {
|
|
|
|
doRp_AutoContacts( _gcell, rpM3, _southWestContact, unusedContact, DoSourceContact );
|
|
|
|
} else if (_west and _south) {
|
|
|
|
AutoContact* rpContact = NULL;
|
|
|
|
doRp_AutoContacts( _gcell, rpM3, rpContact, unusedContact, DoSourceContact );
|
|
|
|
_southWestContact = AutoContactVTee::create( _gcell, _net, Session::getContactLayer(1) );
|
|
|
|
AutoSegment::create( rpContact, _southWestContact, KbVertical );
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_east and not _north) {
|
|
|
|
_northEastContact = doRp_Access( _gcell, _routingPads[_routingPads.size()-1], HAccess );
|
|
|
|
} else if (not _east and _north) {
|
|
|
|
doRp_AutoContacts( _gcell, rpM3, unusedContact, _northEastContact, DoTargetContact );
|
|
|
|
} else if (_east and _north) {
|
|
|
|
AutoContact* rpContact = NULL;
|
|
|
|
doRp_AutoContacts( _gcell, rpM3, unusedContact, rpContact, DoTargetContact );
|
|
|
|
_northEastContact = AutoContactVTee::create( _gcell, _net, Session::getContactLayer(1) );
|
|
|
|
AutoSegment::create( rpContact, _northEastContact, KbVertical );
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
} else {
|
Improved UpdateSession & exception catching. Start of RoutingGauge implem.
Miscellaneous:
* Change: In <crlcore>, in display.conf use the same display threshold
for both METAL2 & METAL3.
In alliance.conf, the side of VIAs in the gauge is 2l (not 3l).
In kite.conf, separate edge densities for H/V.
* Change: In <Cell>, in flattenNets() use flag as argument, not a
boolean. Do not create rings for clock or supply nets.
* Change: In <DeepNet>, in _createRoutingPads() do not create rings
for clock or supply net (duplicated policy as in Cell::flattenNets()).
* Bug: In <ControllerWidget>, at last find the bad signal disconnect
that was causing ungraceful messages.
* Change: In <knik>, in Edge display occupancy/capacity in the string
name. Improved display progress and debugging capabilities.
Improved exception catch & breakpoint managment:
* Bug: In <PaletteWidget>, in updateExtensions() replace the calls to
deleteLayer() by delete. This cause the widget to be immediatly
erased instead of waiting for the event queue to be completly
processed. This was causing the widget to be left in a incoherent
state when stoping at a breakpoint.
* Bug: In <BreakpointWidget>, in execNoModal(), flush the main event
loop (QApplication::flush()) *before* lauching the *local* event
loop. This is to ensure all widgets are in their final state when
waiting (especially <PaletteWidget>).
* Change: In <ExceptionWidget>, new method catchAllWrapper() to
execute any std::function< void() > function/method with a "try"/
"catch" wraparound and lauch the widget in case something is catch.
* New: In <hurricane>, support for a oberver pattern, backported from
<katabatic> with an Obervable capable of being linked to any
number of Obervers.
* New: In <Cell>, made it observable to detect Cell change, currently
emit two kind of signals:
- Cell::CellAboutToChange : *before* any change.
- Cell::CellChanged : *after* the change has been completed.
* New: In <UpdateSession>, in Go::invalidate() add the Cell owning the
Go to the UPDATOR_STACK (of course the cell is added only once).
In addition, when the Cell is added, send a notification of
Cell::CellAboutToChange to all it's observers. The slave instances
are also invalidated.
Conversely in UpdateSession::_preDestroy() for each invalidated
Cell send a Cell::CellChanged notification to all observer.
The UPDATOR_STACK has been slightly amended to accept Cell which
are not Gos. Prior to this, the Cell where completly excluded from
the UpdateSession mechanism, so it's instances where never actualised
of anything referring to the Cell for that matter.
Note: we use two different mechanisms to transmit a Cell change,
observers and the slave instance map. I think at some point it
should be unificated.
* Change: In <CellViewer>, make it a Cell observer to redraw when the
cell is modificated (also update the palette).
Uses the catchAllWrapper() to protect all critical actions.
* Change: In <GraphicTool>, no longer need of cellPreModificated and
cellPostModificated signals. Now done through the Cell obersvers.
* Change: In <mauka>, <etesian> & <kite> now uses the catchAllWrapper
method for protection (need to split methods in two, to be able
to pass it as argument). No longer emit cellPreModificated and
cellPostModificated.
Support for RoutingGauge in P&R:
* Bug: In <placeandroute.py>, the connection from the internal power
ring to the connectors was not done correctly. Wrong contact layers
leading to a gap.
* Change: In <BuildPowerRails>, detection of the corona signals based
on how the "pck_px" pad is connected. No longer based on name
matching.
* Change: In <placeandroute.py>, support for 2 routing metal only
(3 metal in the technology).
* Change: In <katabatic> & <kite> support for a "top layer" limitation
on the routing gauge, this allows to use only two routing metals
(METAL2 & METAL3). Work in progress.
2014-04-20 12:25:08 -05:00
|
|
|
// All RoutingPad are M1.
|
2013-12-03 18:58:58 -06:00
|
|
|
Component* southWestRp = _routingPads[0];
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "| Initial S-W Global RP: " << southWestRp << endl;
|
2013-12-03 18:58:58 -06:00
|
|
|
for ( unsigned int i=1 ; i<_routingPads.size() ; ++i ) {
|
Improved UpdateSession & exception catching. Start of RoutingGauge implem.
Miscellaneous:
* Change: In <crlcore>, in display.conf use the same display threshold
for both METAL2 & METAL3.
In alliance.conf, the side of VIAs in the gauge is 2l (not 3l).
In kite.conf, separate edge densities for H/V.
* Change: In <Cell>, in flattenNets() use flag as argument, not a
boolean. Do not create rings for clock or supply nets.
* Change: In <DeepNet>, in _createRoutingPads() do not create rings
for clock or supply net (duplicated policy as in Cell::flattenNets()).
* Bug: In <ControllerWidget>, at last find the bad signal disconnect
that was causing ungraceful messages.
* Change: In <knik>, in Edge display occupancy/capacity in the string
name. Improved display progress and debugging capabilities.
Improved exception catch & breakpoint managment:
* Bug: In <PaletteWidget>, in updateExtensions() replace the calls to
deleteLayer() by delete. This cause the widget to be immediatly
erased instead of waiting for the event queue to be completly
processed. This was causing the widget to be left in a incoherent
state when stoping at a breakpoint.
* Bug: In <BreakpointWidget>, in execNoModal(), flush the main event
loop (QApplication::flush()) *before* lauching the *local* event
loop. This is to ensure all widgets are in their final state when
waiting (especially <PaletteWidget>).
* Change: In <ExceptionWidget>, new method catchAllWrapper() to
execute any std::function< void() > function/method with a "try"/
"catch" wraparound and lauch the widget in case something is catch.
* New: In <hurricane>, support for a oberver pattern, backported from
<katabatic> with an Obervable capable of being linked to any
number of Obervers.
* New: In <Cell>, made it observable to detect Cell change, currently
emit two kind of signals:
- Cell::CellAboutToChange : *before* any change.
- Cell::CellChanged : *after* the change has been completed.
* New: In <UpdateSession>, in Go::invalidate() add the Cell owning the
Go to the UPDATOR_STACK (of course the cell is added only once).
In addition, when the Cell is added, send a notification of
Cell::CellAboutToChange to all it's observers. The slave instances
are also invalidated.
Conversely in UpdateSession::_preDestroy() for each invalidated
Cell send a Cell::CellChanged notification to all observer.
The UPDATOR_STACK has been slightly amended to accept Cell which
are not Gos. Prior to this, the Cell where completly excluded from
the UpdateSession mechanism, so it's instances where never actualised
of anything referring to the Cell for that matter.
Note: we use two different mechanisms to transmit a Cell change,
observers and the slave instance map. I think at some point it
should be unificated.
* Change: In <CellViewer>, make it a Cell observer to redraw when the
cell is modificated (also update the palette).
Uses the catchAllWrapper() to protect all critical actions.
* Change: In <GraphicTool>, no longer need of cellPreModificated and
cellPostModificated signals. Now done through the Cell obersvers.
* Change: In <mauka>, <etesian> & <kite> now uses the catchAllWrapper
method for protection (need to split methods in two, to be able
to pass it as argument). No longer emit cellPreModificated and
cellPostModificated.
Support for RoutingGauge in P&R:
* Bug: In <placeandroute.py>, the connection from the internal power
ring to the connectors was not done correctly. Wrong contact layers
leading to a gap.
* Change: In <BuildPowerRails>, detection of the corona signals based
on how the "pck_px" pad is connected. No longer based on name
matching.
* Change: In <placeandroute.py>, support for 2 routing metal only
(3 metal in the technology).
* Change: In <katabatic> & <kite> support for a "top layer" limitation
on the routing gauge, this allows to use only two routing metals
(METAL2 & METAL3). Work in progress.
2014-04-20 12:25:08 -05:00
|
|
|
if (southWestRp->getBoundingBox().getHeight() >= 4*Session::getPitch(1)) break;
|
2013-12-03 18:58:58 -06:00
|
|
|
if (_routingPads[i]->getBoundingBox().getHeight() > southWestRp->getBoundingBox().getHeight()) {
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "| Better RP: " << southWestRp << endl;
|
2013-12-03 18:58:58 -06:00
|
|
|
southWestRp = _routingPads[i];
|
2010-04-12 06:22:11 -05:00
|
|
|
}
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
if (_west and not _south) {
|
|
|
|
_southWestContact = doRp_Access( _gcell, southWestRp, HAccess );
|
|
|
|
} else if (not _west and _south) {
|
|
|
|
AutoContact* rpContact = doRp_Access( _gcell, southWestRp, HAccess );
|
|
|
|
_southWestContact = AutoContactTurn::create( _gcell, _net, Session::getContactLayer(1) );
|
|
|
|
AutoSegment::create( rpContact, _southWestContact, KbHorizontal );
|
|
|
|
} else if (_west and _south) {
|
|
|
|
AutoContact* rpContact = doRp_Access( _gcell, southWestRp, HAccess );
|
|
|
|
_southWestContact = AutoContactHTee::create( _gcell, _net, Session::getContactLayer(1) );
|
|
|
|
AutoSegment::create( rpContact, _southWestContact, KbHorizontal );
|
* ./katabatic:
- New: In AutoSegment, adds a "_parent" attribute to keep track of the fragmentation
processus. Currently used only for strap segments, points to the original
segment in the appropriate direction (before the split).
- New: In GCell & LayerAssign, new method of layer assignment. Move up
the whole net trunk if only one of it's segment is inside an over-saturated
GCell. AutoSegment are moved up only if there is at least 2 free tracks
remaining on the upper level.
- Change: In Session::_canonize(), uses the lowest segment Id as canonical.
More reliable than geometricals criterions in the end. Assuming that the
segments are being created in deterministic order, which *should* be the
case consediring the way we are walking through the global routing.
- Change: In AutoSegment, completly suppress the CompareCanonical(), replace
it by the much simpler CompareId().
- Change: In GCell::rpDesaturate(), stops desaturation when bottom density
is under 0.5, otherwise we are causing a severe imbalance in M2/M4
densities. All wires pushed up to M4...
- Change: In ChipTools, for the Pad's RoutingPad, reslect the best component
using the one in the lowest layer. To avoid problem when splitting
AutoContact as we expect the base Contact to be on the lower layer.
- Bug: In GCellConfiguration::_GCell_xG_xL1_xL3(), add H/V alignement constraints
in fork case. This allow NE/SW contact to be splitted correctly later.
- Bug: In AutoContact::split(), the connexity on the splitted contacts was
not correctly restored, leading to canonization and parentage looping
errors. This was concealed by the Kite Track::_check() bug (incomplete
individual TrackSegment checking).
2010-12-30 12:41:19 -06:00
|
|
|
}
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
Component* northEastRp = _routingPads[_routingPads.size()-1];
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "| Initial N-E Global RP: " << northEastRp << endl;
|
2013-12-03 18:58:58 -06:00
|
|
|
|
|
|
|
if (_routingPads.size() > 1) {
|
Correction in plugins to support msxlib compatible pads.
* New: In CRL Core, in helpers & alliance.conf, set and read a "PAD"
variable to define the pad model name extension ("px" for "sxlib
and "pxr" for vsxlib, this is provisional).
* New: In CRL Core, in plugin.conf, add parameters to define the name
of used for power & clock supply. We may remove the extention in
the future (to be more coherent with the previous modification).
* New: In Cumulus, in chip.Configuration.GaugeConf._rpAccess(), no
longer place the accessing contact *at the center* of the
RoutingPad. It works under sxlib because buffers & registers all
have same size terminals. But this is not true under vsxlib,
leading to misaligned contacts & wires. Now systematically place
on the slice midlle track (maybe with one pitch above or below).
This is still very weak as we do not check if the terminal
reach were the contact is being put. Has to be strenthened in
the future.
* New: In Cumulus, in chip.Configuration.ChipConf, read the new
clock & power pad parameters.
* Change: In Isobar (and all other Python wrappers), uses PyLong instead
of PyInt for DbU conversions. In PyHurricane argument converter,
automatically check for both PyLong and then PyInt.
* Change: In Cumulus, in chip.PadsCorona, more accurate error message
in case of discrepency in global net connections (i.e. no net
of the same name in instance model and instance model owner.
* Change: In Kite, in BuildPowerRails, when looking up at the pads
model name to find "pck_" or "pvddeck_", do not compare the
extension part. But we still use hard-coded stem pad names,
maybe we shouldn't.
* Bug: In Katabatic, in GCellConfiguration::_do_xG_xM1_xM3(), there
was a loop in the search of the best N/E initial RoutingPad.
* Bug: In Kite, in KiteEngine::protectRoutingPads(), *do not* protect
RoutingPads of fixed nets, they are already through the
BuildPowerRails stage (and it's causing scary overlap warning
messages).
* Bug: In Cumulus, in ClockTree.HTreeNode.addLeaf(), do not create
deep-plug when the core is flat (not sub-modules). All the new
nets are at core level.
* Bug: In Cumulus, in ChipPlugin.PlaceCore.doFloorplan(), ensure
that the core is aligned on the GCell grid (i.e. the slice
grid of the overall chip).
* Bug: In Kite, in GCellTopology::_do_xG_xM1_xM3(), infinite loop
while looking for the bigger N-E RoutingPad. Forgot to decrement
the index...
2014-09-13 10:45:30 -05:00
|
|
|
for ( unsigned int i=_routingPads.size()-1; i != 0 ; ) {
|
|
|
|
i -= 1;
|
Improved UpdateSession & exception catching. Start of RoutingGauge implem.
Miscellaneous:
* Change: In <crlcore>, in display.conf use the same display threshold
for both METAL2 & METAL3.
In alliance.conf, the side of VIAs in the gauge is 2l (not 3l).
In kite.conf, separate edge densities for H/V.
* Change: In <Cell>, in flattenNets() use flag as argument, not a
boolean. Do not create rings for clock or supply nets.
* Change: In <DeepNet>, in _createRoutingPads() do not create rings
for clock or supply net (duplicated policy as in Cell::flattenNets()).
* Bug: In <ControllerWidget>, at last find the bad signal disconnect
that was causing ungraceful messages.
* Change: In <knik>, in Edge display occupancy/capacity in the string
name. Improved display progress and debugging capabilities.
Improved exception catch & breakpoint managment:
* Bug: In <PaletteWidget>, in updateExtensions() replace the calls to
deleteLayer() by delete. This cause the widget to be immediatly
erased instead of waiting for the event queue to be completly
processed. This was causing the widget to be left in a incoherent
state when stoping at a breakpoint.
* Bug: In <BreakpointWidget>, in execNoModal(), flush the main event
loop (QApplication::flush()) *before* lauching the *local* event
loop. This is to ensure all widgets are in their final state when
waiting (especially <PaletteWidget>).
* Change: In <ExceptionWidget>, new method catchAllWrapper() to
execute any std::function< void() > function/method with a "try"/
"catch" wraparound and lauch the widget in case something is catch.
* New: In <hurricane>, support for a oberver pattern, backported from
<katabatic> with an Obervable capable of being linked to any
number of Obervers.
* New: In <Cell>, made it observable to detect Cell change, currently
emit two kind of signals:
- Cell::CellAboutToChange : *before* any change.
- Cell::CellChanged : *after* the change has been completed.
* New: In <UpdateSession>, in Go::invalidate() add the Cell owning the
Go to the UPDATOR_STACK (of course the cell is added only once).
In addition, when the Cell is added, send a notification of
Cell::CellAboutToChange to all it's observers. The slave instances
are also invalidated.
Conversely in UpdateSession::_preDestroy() for each invalidated
Cell send a Cell::CellChanged notification to all observer.
The UPDATOR_STACK has been slightly amended to accept Cell which
are not Gos. Prior to this, the Cell where completly excluded from
the UpdateSession mechanism, so it's instances where never actualised
of anything referring to the Cell for that matter.
Note: we use two different mechanisms to transmit a Cell change,
observers and the slave instance map. I think at some point it
should be unificated.
* Change: In <CellViewer>, make it a Cell observer to redraw when the
cell is modificated (also update the palette).
Uses the catchAllWrapper() to protect all critical actions.
* Change: In <GraphicTool>, no longer need of cellPreModificated and
cellPostModificated signals. Now done through the Cell obersvers.
* Change: In <mauka>, <etesian> & <kite> now uses the catchAllWrapper
method for protection (need to split methods in two, to be able
to pass it as argument). No longer emit cellPreModificated and
cellPostModificated.
Support for RoutingGauge in P&R:
* Bug: In <placeandroute.py>, the connection from the internal power
ring to the connectors was not done correctly. Wrong contact layers
leading to a gap.
* Change: In <BuildPowerRails>, detection of the corona signals based
on how the "pck_px" pad is connected. No longer based on name
matching.
* Change: In <placeandroute.py>, support for 2 routing metal only
(3 metal in the technology).
* Change: In <katabatic> & <kite> support for a "top layer" limitation
on the routing gauge, this allows to use only two routing metals
(METAL2 & METAL3). Work in progress.
2014-04-20 12:25:08 -05:00
|
|
|
if (northEastRp->getBoundingBox().getHeight() >= 4*Session::getPitch(1)) break;
|
2013-12-03 18:58:58 -06:00
|
|
|
if (_routingPads[i]->getBoundingBox().getHeight() > northEastRp->getBoundingBox().getHeight()) {
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "| Better RP: " << northEastRp << endl;
|
2013-12-03 18:58:58 -06:00
|
|
|
northEastRp = _routingPads[i];
|
|
|
|
}
|
Correction in plugins to support msxlib compatible pads.
* New: In CRL Core, in helpers & alliance.conf, set and read a "PAD"
variable to define the pad model name extension ("px" for "sxlib
and "pxr" for vsxlib, this is provisional).
* New: In CRL Core, in plugin.conf, add parameters to define the name
of used for power & clock supply. We may remove the extention in
the future (to be more coherent with the previous modification).
* New: In Cumulus, in chip.Configuration.GaugeConf._rpAccess(), no
longer place the accessing contact *at the center* of the
RoutingPad. It works under sxlib because buffers & registers all
have same size terminals. But this is not true under vsxlib,
leading to misaligned contacts & wires. Now systematically place
on the slice midlle track (maybe with one pitch above or below).
This is still very weak as we do not check if the terminal
reach were the contact is being put. Has to be strenthened in
the future.
* New: In Cumulus, in chip.Configuration.ChipConf, read the new
clock & power pad parameters.
* Change: In Isobar (and all other Python wrappers), uses PyLong instead
of PyInt for DbU conversions. In PyHurricane argument converter,
automatically check for both PyLong and then PyInt.
* Change: In Cumulus, in chip.PadsCorona, more accurate error message
in case of discrepency in global net connections (i.e. no net
of the same name in instance model and instance model owner.
* Change: In Kite, in BuildPowerRails, when looking up at the pads
model name to find "pck_" or "pvddeck_", do not compare the
extension part. But we still use hard-coded stem pad names,
maybe we shouldn't.
* Bug: In Katabatic, in GCellConfiguration::_do_xG_xM1_xM3(), there
was a loop in the search of the best N/E initial RoutingPad.
* Bug: In Kite, in KiteEngine::protectRoutingPads(), *do not* protect
RoutingPads of fixed nets, they are already through the
BuildPowerRails stage (and it's causing scary overlap warning
messages).
* Bug: In Cumulus, in ClockTree.HTreeNode.addLeaf(), do not create
deep-plug when the core is flat (not sub-modules). All the new
nets are at core level.
* Bug: In Cumulus, in ChipPlugin.PlaceCore.doFloorplan(), ensure
that the core is aligned on the GCell grid (i.e. the slice
grid of the overall chip).
* Bug: In Kite, in GCellTopology::_do_xG_xM1_xM3(), infinite loop
while looking for the bigger N-E RoutingPad. Forgot to decrement
the index...
2014-09-13 10:45:30 -05:00
|
|
|
}
|
2013-12-03 18:58:58 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
if (_east and not _north) {
|
|
|
|
_northEastContact = doRp_Access( _gcell, northEastRp, HAccess );
|
|
|
|
} else if (not _east and _north) {
|
|
|
|
AutoContact* rpContact = doRp_Access( _gcell, northEastRp, HAccess );
|
|
|
|
_northEastContact = AutoContactTurn::create( _gcell, _net, Session::getContactLayer(1) );
|
|
|
|
AutoSegment::create( rpContact, _northEastContact, KbHorizontal );
|
|
|
|
} else if (_east and _north) {
|
|
|
|
AutoContact* rpContact = doRp_Access( _gcell, northEastRp, HAccess );
|
|
|
|
_northEastContact = AutoContactHTee::create( _gcell, _net, Session::getContactLayer(1) );
|
|
|
|
AutoSegment::create( rpContact, _northEastContact, KbHorizontal );
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_tabw(145,-1);
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-04-22 15:43:53 -05:00
|
|
|
void GCellTopology::_do_4G_1M2 ()
|
|
|
|
{
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,1) << "_do_4G_1M2() [Managed Configuration]" << endl;
|
2016-04-22 15:43:53 -05:00
|
|
|
|
|
|
|
Component* rpL2 = _routingPads[0];
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "rpL2 := " << rpL2 << endl;
|
2016-04-22 15:43:53 -05:00
|
|
|
|
|
|
|
AutoContact* rpL2ContactSource = NULL;
|
|
|
|
AutoContact* rpL2ContactTarget = NULL;
|
|
|
|
|
|
|
|
doRp_AutoContacts( _gcell, rpL2, rpL2ContactSource, rpL2ContactTarget, DoSourceContact|DoTargetContact );
|
|
|
|
|
|
|
|
_southWestContact = AutoContactHTee::create( _gcell, _net, Session::getContactLayer(2) );
|
|
|
|
_northEastContact = AutoContactHTee::create( _gcell, _net, Session::getContactLayer(2) );
|
|
|
|
|
|
|
|
AutoSegment::create( _southWestContact, rpL2ContactSource, KbHorizontal );
|
|
|
|
AutoSegment::create( rpL2ContactTarget, _northEastContact, KbHorizontal );
|
|
|
|
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_tabw(145,-1);
|
2016-04-22 15:43:53 -05:00
|
|
|
}
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
void GCellTopology::_do_xG_xM2 ()
|
2010-03-09 09:24:29 -06:00
|
|
|
{
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,1) << "_do_"
|
Solve the template lookup problem in tstream.
* Bug: In Hurricane, in tstream (Commons.h), in the *template* overload of
operator<<(), do not use the operator<<() of ostream as it will be
looked up in "stage 1" (template definition) and so will miss all the
overloads added later and built over getString<>(). Instead, make use
of getString<>(), which, as another template will be looked up in
"stage 2" (template instanciation) and at that point will have all the
needed template specialisation of getString<>().
We also need to define new stream manipulators to be able to create
a matching template overload not dependant from the implementation.
To avoid name clashes, we prefix a 't'. For now, only 'tsetw()' is
refined.
As a side effect, we cannot directly print bit-fields into the stream,
we must go through an intermediate variable (happens once in AutoContact).
2016-07-19 09:02:55 -05:00
|
|
|
<< (int)_connexity.fields.globals << "G_"
|
|
|
|
<< (int)_connexity.fields.M2 << "M2() [Managed Configuration - x]" << endl;
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
Component* biggestRp = _routingPads[0];
|
|
|
|
for ( unsigned int i=1 ; i<_routingPads.size() ; ++i ) {
|
|
|
|
doRp_StairCaseH( _gcell, _routingPads[i-1], _routingPads[i] );
|
|
|
|
if (_routingPads[i]->getBoundingBox().getWidth() > biggestRp->getBoundingBox().getWidth())
|
|
|
|
biggestRp = _routingPads[i];
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
AutoContact* unusedContact = NULL;
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
if (_west and not _south) {
|
|
|
|
doRp_AutoContacts( _gcell, _routingPads[0], _southWestContact, unusedContact, DoSourceContact );
|
|
|
|
} else if (not _west and _south) {
|
|
|
|
_southWestContact = doRp_Access( _gcell, biggestRp, NoFlags );
|
|
|
|
} else if (_west and _south) {
|
|
|
|
AutoContact* rpContact = doRp_Access( _gcell, biggestRp, NoFlags );
|
|
|
|
_southWestContact = AutoContactVTee::create( _gcell, _net, Session::getContactLayer(1) );
|
|
|
|
AutoSegment::create( rpContact, _southWestContact, KbVertical );
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
if (_east and not _north) {
|
|
|
|
doRp_AutoContacts( _gcell, _routingPads[_routingPads.size()-1], _northEastContact, unusedContact, DoSourceContact );
|
|
|
|
} else if (not _east and _north) {
|
|
|
|
_northEastContact = doRp_Access( _gcell, biggestRp, NoFlags );
|
|
|
|
} else if (_east and _north) {
|
|
|
|
AutoContact* rpContact = doRp_Access( _gcell, biggestRp, NoFlags );
|
|
|
|
_northEastContact = AutoContactVTee::create( _gcell, _net, Session::getContactLayer(1) );
|
|
|
|
AutoSegment::create( rpContact, _northEastContact, KbVertical );
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_tabw(145,-1);
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
void GCellTopology::_do_1G_1M3 ()
|
2010-03-09 09:24:29 -06:00
|
|
|
{
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,1) << "_do_1G_1M3() [Optimised Configuration]" << endl;
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
unsigned int flags = (_east or _west) ? HAccess : NoFlags;
|
|
|
|
flags |= (_north) ? DoTargetContact : NoFlags;
|
|
|
|
flags |= (_south) ? DoSourceContact : NoFlags;
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
doRp_AutoContacts( _gcell
|
|
|
|
, _routingPads[0]
|
|
|
|
, _southWestContact
|
|
|
|
, _northEastContact
|
|
|
|
, flags
|
|
|
|
);
|
|
|
|
if (not _southWestContact) _southWestContact = _northEastContact;
|
|
|
|
if (not _northEastContact) _northEastContact = _southWestContact;
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "_southWest: " << _southWestContact << endl;
|
|
|
|
cdebug_log(145,0) << "_northEast: " << _northEastContact << endl;
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
if (flags & HAccess) {
|
|
|
|
// HARDCODED VALUE.
|
Improved UpdateSession & exception catching. Start of RoutingGauge implem.
Miscellaneous:
* Change: In <crlcore>, in display.conf use the same display threshold
for both METAL2 & METAL3.
In alliance.conf, the side of VIAs in the gauge is 2l (not 3l).
In kite.conf, separate edge densities for H/V.
* Change: In <Cell>, in flattenNets() use flag as argument, not a
boolean. Do not create rings for clock or supply nets.
* Change: In <DeepNet>, in _createRoutingPads() do not create rings
for clock or supply net (duplicated policy as in Cell::flattenNets()).
* Bug: In <ControllerWidget>, at last find the bad signal disconnect
that was causing ungraceful messages.
* Change: In <knik>, in Edge display occupancy/capacity in the string
name. Improved display progress and debugging capabilities.
Improved exception catch & breakpoint managment:
* Bug: In <PaletteWidget>, in updateExtensions() replace the calls to
deleteLayer() by delete. This cause the widget to be immediatly
erased instead of waiting for the event queue to be completly
processed. This was causing the widget to be left in a incoherent
state when stoping at a breakpoint.
* Bug: In <BreakpointWidget>, in execNoModal(), flush the main event
loop (QApplication::flush()) *before* lauching the *local* event
loop. This is to ensure all widgets are in their final state when
waiting (especially <PaletteWidget>).
* Change: In <ExceptionWidget>, new method catchAllWrapper() to
execute any std::function< void() > function/method with a "try"/
"catch" wraparound and lauch the widget in case something is catch.
* New: In <hurricane>, support for a oberver pattern, backported from
<katabatic> with an Obervable capable of being linked to any
number of Obervers.
* New: In <Cell>, made it observable to detect Cell change, currently
emit two kind of signals:
- Cell::CellAboutToChange : *before* any change.
- Cell::CellChanged : *after* the change has been completed.
* New: In <UpdateSession>, in Go::invalidate() add the Cell owning the
Go to the UPDATOR_STACK (of course the cell is added only once).
In addition, when the Cell is added, send a notification of
Cell::CellAboutToChange to all it's observers. The slave instances
are also invalidated.
Conversely in UpdateSession::_preDestroy() for each invalidated
Cell send a Cell::CellChanged notification to all observer.
The UPDATOR_STACK has been slightly amended to accept Cell which
are not Gos. Prior to this, the Cell where completly excluded from
the UpdateSession mechanism, so it's instances where never actualised
of anything referring to the Cell for that matter.
Note: we use two different mechanisms to transmit a Cell change,
observers and the slave instance map. I think at some point it
should be unificated.
* Change: In <CellViewer>, make it a Cell observer to redraw when the
cell is modificated (also update the palette).
Uses the catchAllWrapper() to protect all critical actions.
* Change: In <GraphicTool>, no longer need of cellPreModificated and
cellPostModificated signals. Now done through the Cell obersvers.
* Change: In <mauka>, <etesian> & <kite> now uses the catchAllWrapper
method for protection (need to split methods in two, to be able
to pass it as argument). No longer emit cellPreModificated and
cellPostModificated.
Support for RoutingGauge in P&R:
* Bug: In <placeandroute.py>, the connection from the internal power
ring to the connectors was not done correctly. Wrong contact layers
leading to a gap.
* Change: In <BuildPowerRails>, detection of the corona signals based
on how the "pck_px" pad is connected. No longer based on name
matching.
* Change: In <placeandroute.py>, support for 2 routing metal only
(3 metal in the technology).
* Change: In <katabatic> & <kite> support for a "top layer" limitation
on the routing gauge, this allows to use only two routing metals
(METAL2 & METAL3). Work in progress.
2014-04-20 12:25:08 -05:00
|
|
|
if (_routingPads[0]->getBoundingBox().getHeight() < 3*Session::getPitch(1)) {
|
2013-12-03 18:58:58 -06:00
|
|
|
AutoContact* subContact = AutoContactTurn::create( _gcell, _net, Session::getContactLayer(1) );
|
|
|
|
AutoSegment::create( _southWestContact, subContact, KbVertical );
|
2010-03-09 09:24:29 -06:00
|
|
|
|
|
|
|
_southWestContact = _northEastContact = subContact;
|
|
|
|
}
|
2013-12-03 18:58:58 -06:00
|
|
|
} else {
|
|
|
|
if (_sourceContact) {
|
|
|
|
if (_sourceContact->getX() != _southWestContact->getX()) {
|
|
|
|
AutoContactTurn* turn1 = AutoContactTurn::create( _gcell, _net, Session::getContactLayer(1) );
|
|
|
|
AutoContactTurn* turn2 = AutoContactTurn::create( _gcell, _net, Session::getContactLayer(1) );
|
|
|
|
AutoSegment::create( _southWestContact, turn1, KbVertical );
|
|
|
|
AutoSegment::create( turn1 , turn2, KbHorizontal );
|
|
|
|
_southWestContact = _northEastContact = turn2;
|
|
|
|
}
|
|
|
|
}
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_tabw(145,-1);
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
void GCellTopology::_do_xG_xM3 ()
|
2010-03-09 09:24:29 -06:00
|
|
|
{
|
Solve the template lookup problem in tstream.
* Bug: In Hurricane, in tstream (Commons.h), in the *template* overload of
operator<<(), do not use the operator<<() of ostream as it will be
looked up in "stage 1" (template definition) and so will miss all the
overloads added later and built over getString<>(). Instead, make use
of getString<>(), which, as another template will be looked up in
"stage 2" (template instanciation) and at that point will have all the
needed template specialisation of getString<>().
We also need to define new stream manipulators to be able to create
a matching template overload not dependant from the implementation.
To avoid name clashes, we prefix a 't'. For now, only 'tsetw()' is
refined.
As a side effect, we cannot directly print bit-fields into the stream,
we must go through an intermediate variable (happens once in AutoContact).
2016-07-19 09:02:55 -05:00
|
|
|
cdebug_log(145,1) << "_do_xG_" << (int)_connexity.fields.M3
|
2016-05-17 16:00:06 -05:00
|
|
|
<< "M3() [Managed Configuration]" << endl;
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "_west:" << _west << endl;
|
|
|
|
cdebug_log(145,0) << "_east:" << _east << endl;
|
|
|
|
cdebug_log(145,0) << "_south:" << _south << endl;
|
|
|
|
cdebug_log(145,0) << "_north:" << _north << endl;
|
2013-12-03 18:58:58 -06:00
|
|
|
|
|
|
|
sort( _routingPads.begin(), _routingPads.end(), SortRpByY(NoFlags) ); // increasing Y.
|
|
|
|
for ( unsigned int i=1 ; i<_routingPads.size() ; i++ ) {
|
|
|
|
doRp_StairCaseV( _gcell, _routingPads[i-1], _routingPads[i] );
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
AutoContact* unusedContact = NULL;
|
|
|
|
Component* rp = _routingPads[0];
|
|
|
|
|
|
|
|
if (_west and not _south) {
|
|
|
|
_southWestContact = doRp_Access( _gcell, rp, HAccess );
|
|
|
|
} else if (not _west and _south) {
|
|
|
|
doRp_AutoContacts( _gcell, rp, _southWestContact, unusedContact, DoSourceContact );
|
|
|
|
if (_sourceContact) {
|
|
|
|
if (_sourceContact->getX() != _southWestContact->getX()) {
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(149,0) << "Misaligned South: _source:" << DbU::getValueString(_sourceContact->getX())
|
2013-12-03 18:58:58 -06:00
|
|
|
<< "_southWest:" << DbU::getValueString(_southWestContact->getX()) << endl;
|
|
|
|
|
|
|
|
AutoContactTurn* turn1 = AutoContactTurn::create( _gcell, _net, Session::getContactLayer(1) );
|
|
|
|
AutoContactTurn* turn2 = AutoContactTurn::create( _gcell, _net, Session::getContactLayer(1) );
|
|
|
|
AutoSegment::create( _southWestContact, turn1, KbVertical );
|
|
|
|
AutoSegment::create( turn1 , turn2, KbHorizontal );
|
|
|
|
_southWestContact = turn2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (_west and _south) {
|
|
|
|
AutoContact* rpContact = NULL;
|
|
|
|
doRp_AutoContacts( _gcell, rp, rpContact, unusedContact, DoSourceContact );
|
|
|
|
_southWestContact = AutoContactVTee::create( _gcell, _net, Session::getContactLayer(1) );
|
|
|
|
AutoSegment::create( rpContact, _southWestContact, KbVertical );
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
rp = _routingPads[_routingPads.size()-1];
|
|
|
|
if (_east and not _north) {
|
|
|
|
_northEastContact = doRp_Access( _gcell, rp, HAccess );
|
|
|
|
} else if (not _east and _north) {
|
|
|
|
doRp_AutoContacts( _gcell, rp, unusedContact, _northEastContact, DoTargetContact );
|
|
|
|
if (_sourceContact) {
|
|
|
|
if (_sourceContact->getX() != _northEastContact->getX()) {
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(149,0) << "Misaligned North: _source:" << DbU::getValueString(_sourceContact->getX())
|
2013-12-03 18:58:58 -06:00
|
|
|
<< "_southWest:" << DbU::getValueString(_northEastContact->getX()) << endl;
|
|
|
|
|
|
|
|
AutoContactTurn* turn1 = AutoContactTurn::create( _gcell, _net, Session::getContactLayer(1) );
|
|
|
|
AutoContactTurn* turn2 = AutoContactTurn::create( _gcell, _net, Session::getContactLayer(1) );
|
|
|
|
AutoSegment::create( _northEastContact, turn1, KbVertical );
|
|
|
|
AutoSegment::create( turn1 , turn2, KbHorizontal );
|
|
|
|
_northEastContact = turn2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (_east and _north) {
|
|
|
|
AutoContact* rpContact = NULL;
|
|
|
|
doRp_AutoContacts( _gcell, rp, unusedContact, rpContact, DoTargetContact );
|
|
|
|
_northEastContact = AutoContactVTee::create( _gcell, _net, Session::getContactLayer(1) );
|
|
|
|
AutoSegment::create( rpContact, _northEastContact, KbVertical );
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_tabw(145,-1);
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void singleGCell ( KatabaticEngine* ktbt, Net* net )
|
|
|
|
{
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,1) << "singleGCell() " << net << endl;
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
vector<Component*> rpM1s;
|
|
|
|
Component* rpM2 = NULL;
|
2010-03-09 09:24:29 -06:00
|
|
|
|
|
|
|
forEach ( RoutingPad*, irp, net->getRoutingPads() ) {
|
2013-12-03 18:58:58 -06:00
|
|
|
if (Session::getRoutingGauge()->getLayerDepth(irp->getLayer()) == 1)
|
|
|
|
rpM2 = *irp;
|
|
|
|
else
|
|
|
|
rpM1s.push_back( *irp );
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
if ((rpM1s.size() < 2) and not rpM2) {
|
|
|
|
cerr << Error( "For %s, less than two Plugs/Pins (%d)."
|
|
|
|
, getString(net).c_str()
|
|
|
|
, rpM1s.size() ) << endl;
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_tabw(145,-1);
|
2010-03-09 09:24:29 -06:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
sort( rpM1s.begin(), rpM1s.end(), SortRpByX(NoFlags) ); // increasing X.
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
GCell* gcell = ktbt->getGCellGrid()->getGCell( (*rpM1s.begin ())->getCenter()
|
|
|
|
, (*rpM1s.rbegin())->getCenter() );
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
if (not gcell) {
|
|
|
|
cerr << Error( "No GCell under %s.", getString(rpM1s[0]).c_str() ) << endl;
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_tabw(145,-1);
|
2010-03-09 09:24:29 -06:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "singleGCell " << gcell << endl;
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
AutoContact* turn = NULL;
|
2010-03-09 09:24:29 -06:00
|
|
|
AutoContact* source = NULL;
|
|
|
|
AutoContact* target = NULL;
|
2010-04-23 08:13:54 -05:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
for ( size_t irp=1 ; irp<rpM1s.size() ; ++irp ) {
|
|
|
|
GCellTopology::doRp_AutoContacts( gcell, rpM1s[irp-1], source, turn, DoSourceContact );
|
|
|
|
GCellTopology::doRp_AutoContacts( gcell, rpM1s[irp ], target, turn, DoSourceContact );
|
|
|
|
AutoSegment::create( source, target, KbHorizontal );
|
2010-04-23 08:13:54 -05:00
|
|
|
}
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
if (rpM2) {
|
|
|
|
GCellTopology::doRp_AutoContacts( gcell, rpM1s[0], source, turn, DoSourceContact );
|
|
|
|
GCellTopology::doRp_AutoContacts( gcell, rpM2 , target, turn, DoSourceContact );
|
|
|
|
turn = AutoContactTurn::create( gcell, rpM2->getNet(), Session::getContactLayer(1) );
|
|
|
|
AutoSegment::create( source, turn , KbHorizontal );
|
|
|
|
AutoSegment::create( turn , target, KbVertical );
|
|
|
|
}
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_tabw(145,-1);
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
} // Anonymous namespace.
|
2010-03-09 09:24:29 -06:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
namespace Katabatic {
|
|
|
|
|
2010-04-23 08:13:54 -05:00
|
|
|
using Hurricane::Name;
|
2010-03-09 09:24:29 -06:00
|
|
|
using Hurricane::DebugSession;
|
|
|
|
using Hurricane::Error;
|
|
|
|
using Hurricane::Warning;
|
|
|
|
using Hurricane::Bug;
|
2010-04-23 08:13:54 -05:00
|
|
|
using CRL::addMeasure;
|
2013-12-03 18:58:58 -06:00
|
|
|
using CRL::getMeasure;
|
2010-03-09 09:24:29 -06:00
|
|
|
|
|
|
|
|
|
|
|
void KatabaticEngine::_loadGrByNet ()
|
|
|
|
{
|
|
|
|
cmess1 << " o Loading Nets global routing from Knik." << endl;
|
* ./katabatic:
- Change: In loadGlobalRouting(), more exlicit message as to why a net is
filtered out the routing set (POWER, GROUND, CLOCK or BLOCKAGE).
- New: ChipTool, module with utilities specific to chip routing. Containing
a function to pre-break wires around a block.
- New: In loadGlobalRouting/GCellConfiguration, adds supports for RoutingPad
connected on chip's pad.
- New: In GCellGrid/GCell, can compute the density in three modes: average,
max of H/V and max of layer. Currently implemented in getMaxHVDensity(),
should be unificated with getDensity(). Used for display purposes.
- Bug: In AutoContact, when splitting a contact, add a specific check for
"one layer span".
- Bug: In AutoSegment::canMoveUp(), do not move up fixed segments.
2010-11-16 07:59:38 -06:00
|
|
|
cmess1 << Dots::asDouble(" - Saturation",getMeasure<double>(getCell(),"Sat.")->getData()) << endl;
|
2010-05-13 04:46:14 -05:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
startMeasures();
|
|
|
|
Session::open( this );
|
2010-03-09 09:24:29 -06:00
|
|
|
|
Add NetRoutingState (Property) on Nets to tell Kite what to do.
* New: In Katabatic, add the ability to decorate some (i.e. few) nets
with a state that indicate to Katabatic & Kite what to do with it.
The possible states are:
1. Fixed : all the wire are in fixed positions. The router cannot
move them and account them as obstacles.
2. ManualGlobalRoute : a user-defined topology is supplied. The
wires still have to be detailed route in "Detailed Pre-Route"
but will be skipped for the global routing and fixed for the
general Detailed route.
3. AutomaticGlobalRoute : ordinary nets, to be global routed then
detail routed.
4. Excluded : do not try to global or detail route thoses nets.
5. MixedPreRoute : mask combining Fixed and ManualGlobalRoute.
Not all nets have this property, only those that needs a special
processing.
To ease the access to the state, it is nested inside a
PrivateProperty in the net (NetRoutingProperty), with an extension
access class (NetRoutingExtension).
* New: In Kite, take account of NetRoutingState. Pointers to the
net's states are strored inside an internal map for faster access.
The property is *not* deleted when Kite is destroyed. The property
will remains until the Net itself is destroyed.
As a consequence, the lists that where passed to high level
function are removed as the information can now be accessed directly
through the net NetRoutingProperty.
* New: In Unicorn, in CgtMain, comply with the update interface.
* New: In documentation, update the User's Guide to explain the Pre-routed
step of Kite.
2014-07-02 07:36:58 -05:00
|
|
|
forEach ( Net*, inet, getCell()->getNets() ) {
|
|
|
|
if (NetRoutingExtension::isAutomaticGlobalRoute(*inet)) {
|
2016-05-17 16:00:06 -05:00
|
|
|
DebugSession::open( *inet, 140, 150 );
|
Add NetRoutingState (Property) on Nets to tell Kite what to do.
* New: In Katabatic, add the ability to decorate some (i.e. few) nets
with a state that indicate to Katabatic & Kite what to do with it.
The possible states are:
1. Fixed : all the wire are in fixed positions. The router cannot
move them and account them as obstacles.
2. ManualGlobalRoute : a user-defined topology is supplied. The
wires still have to be detailed route in "Detailed Pre-Route"
but will be skipped for the global routing and fixed for the
general Detailed route.
3. AutomaticGlobalRoute : ordinary nets, to be global routed then
detail routed.
4. Excluded : do not try to global or detail route thoses nets.
5. MixedPreRoute : mask combining Fixed and ManualGlobalRoute.
Not all nets have this property, only those that needs a special
processing.
To ease the access to the state, it is nested inside a
PrivateProperty in the net (NetRoutingProperty), with an extension
access class (NetRoutingExtension).
* New: In Kite, take account of NetRoutingState. Pointers to the
net's states are strored inside an internal map for faster access.
The property is *not* deleted when Kite is destroyed. The property
will remains until the Net itself is destroyed.
As a consequence, the lists that where passed to high level
function are removed as the information can now be accessed directly
through the net NetRoutingProperty.
* New: In Unicorn, in CgtMain, comply with the update interface.
* New: In documentation, update the User's Guide to explain the Pre-routed
step of Kite.
2014-07-02 07:36:58 -05:00
|
|
|
_loadNetGlobalRouting( *inet );
|
|
|
|
Session::revalidate();
|
|
|
|
DebugSession::close();
|
|
|
|
}
|
|
|
|
} // forEach(Net*)
|
2010-03-09 09:24:29 -06:00
|
|
|
|
|
|
|
#if defined(CHECK_DATABASE)
|
|
|
|
_check ( "after Katabatic loading" );
|
|
|
|
#endif
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
_print();
|
|
|
|
Session::close();
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
stopMeasures();
|
|
|
|
printMeasures( "load" );
|
2010-04-23 08:13:54 -05:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
addMeasure<size_t>( getCell(), "Globals", AutoSegment::getGlobalsCount() );
|
|
|
|
addMeasure<size_t>( getCell(), "Edges" , AutoSegment::getAllocateds() );
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void KatabaticEngine::_loadNetGlobalRouting ( Net* net )
|
|
|
|
{
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(149,0) << "Katabatic::_loadNetGlobalRouting( " << net << " )" << endl;
|
|
|
|
cdebug_tabw(145,1);
|
2010-03-09 09:24:29 -06:00
|
|
|
|
|
|
|
ForkStack forks;
|
|
|
|
Hook* sourceHook = NULL;
|
|
|
|
AutoContact* sourceContact = NULL;
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
lookupClear();
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
RoutingPads routingPads = net->getRoutingPads();
|
2011-01-25 11:16:27 -06:00
|
|
|
size_t degree = routingPads.getSize();
|
2013-12-03 18:58:58 -06:00
|
|
|
|
|
|
|
if (degree == 0) {
|
|
|
|
cmess2 << Warning("Net \"%s\" do not have any RoutingPad (ignored)."
|
|
|
|
,getString(net->getName()).c_str()) << endl;
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_tabw(145,-1);
|
2013-12-03 18:58:58 -06:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (degree < 2) {
|
2010-03-09 09:24:29 -06:00
|
|
|
#if 0
|
|
|
|
if ( !getDemoMode() )
|
|
|
|
cmess2 << Warning("Net \"%s\" have less than 2 plugs/pins (ignored)."
|
|
|
|
,getString(net->getName()).c_str()) << endl;
|
|
|
|
#endif
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_tabw(145,-1);
|
2010-03-09 09:24:29 -06:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_tabw(145,1);
|
2010-05-11 06:04:23 -05:00
|
|
|
Hook* startHook = NULL;
|
|
|
|
GCell* lowestGCell = NULL;
|
|
|
|
size_t unconnecteds = 0;
|
|
|
|
size_t connecteds = 0;
|
2011-01-25 11:16:27 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
GCellTopology::init( degree );
|
2011-01-25 11:16:27 -06:00
|
|
|
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "Start RoutingPad Ring" << endl;
|
2010-03-09 09:24:29 -06:00
|
|
|
forEach ( RoutingPad*, startRp, routingPads ) {
|
2010-08-18 15:24:30 -05:00
|
|
|
bool segmentFound = false;
|
2013-12-03 18:58:58 -06:00
|
|
|
|
2010-03-09 09:24:29 -06:00
|
|
|
forEach ( Hook*, ihook, startRp->getBodyHook()->getHooks() ) {
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "Component " << ihook->getComponent() << endl;
|
2013-12-03 18:58:58 -06:00
|
|
|
Segment* segment = dynamic_cast<Segment*>( ihook->getComponent() );
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
if (segment) {
|
2010-05-11 06:04:23 -05:00
|
|
|
++connecteds;
|
2010-08-18 15:24:30 -05:00
|
|
|
segmentFound = true;
|
2010-05-11 06:04:23 -05:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
GCellTopology gcellConf ( getGCellGrid(), *ihook, NULL );
|
|
|
|
if (gcellConf.getStateG() == 1) {
|
2010-05-11 06:04:23 -05:00
|
|
|
if ( (lowestGCell == NULL) or (lowestGCell->getIndex() > gcellConf.getGCell()->getIndex()) ) {
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "Starting from GCell " << gcellConf.getGCell() << endl;
|
2010-03-09 09:24:29 -06:00
|
|
|
lowestGCell = gcellConf.getGCell();
|
|
|
|
startHook = *ihook;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-12-03 18:58:58 -06:00
|
|
|
|
2010-08-18 15:24:30 -05:00
|
|
|
unconnecteds += (segmentFound) ? 0 : 1;
|
2010-05-11 06:04:23 -05:00
|
|
|
if ( (unconnecteds > 10) and (connecteds == 0) ) {
|
|
|
|
cerr << Warning("More than 10 unconnected RoutingPads (%u) on %s, missing global routing?"
|
|
|
|
,unconnecteds, getString(net->getName()).c_str() ) << endl;
|
|
|
|
|
Add NetRoutingState (Property) on Nets to tell Kite what to do.
* New: In Katabatic, add the ability to decorate some (i.e. few) nets
with a state that indicate to Katabatic & Kite what to do with it.
The possible states are:
1. Fixed : all the wire are in fixed positions. The router cannot
move them and account them as obstacles.
2. ManualGlobalRoute : a user-defined topology is supplied. The
wires still have to be detailed route in "Detailed Pre-Route"
but will be skipped for the global routing and fixed for the
general Detailed route.
3. AutomaticGlobalRoute : ordinary nets, to be global routed then
detail routed.
4. Excluded : do not try to global or detail route thoses nets.
5. MixedPreRoute : mask combining Fixed and ManualGlobalRoute.
Not all nets have this property, only those that needs a special
processing.
To ease the access to the state, it is nested inside a
PrivateProperty in the net (NetRoutingProperty), with an extension
access class (NetRoutingExtension).
* New: In Kite, take account of NetRoutingState. Pointers to the
net's states are strored inside an internal map for faster access.
The property is *not* deleted when Kite is destroyed. The property
will remains until the Net itself is destroyed.
As a consequence, the lists that where passed to high level
function are removed as the information can now be accessed directly
through the net NetRoutingProperty.
* New: In Unicorn, in CgtMain, comply with the update interface.
* New: In documentation, update the User's Guide to explain the Pre-routed
step of Kite.
2014-07-02 07:36:58 -05:00
|
|
|
NetRoutingExtension::create( net )->setFlags ( NetRoutingState::Excluded );
|
|
|
|
NetRoutingExtension::create( net )->unsetFlags( NetRoutingState::AutomaticGlobalRoute );
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_tabw(145,-1);
|
2010-05-11 06:04:23 -05:00
|
|
|
return;
|
|
|
|
}
|
2013-12-03 18:58:58 -06:00
|
|
|
// Uncomment the next line to disable the lowest GCell search.
|
|
|
|
// (takes first GCell with exactly one global).
|
|
|
|
//if (startHook) break;
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_tabw(145,-1);
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2016-06-11 14:56:12 -05:00
|
|
|
if (startHook == NULL) { singleGCell( this, net ); cdebug_tabw(145,-1); return; }
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
GCellTopology startGCellConf ( getGCellGrid(), startHook, NULL );
|
|
|
|
startGCellConf.construct( forks );
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
sourceHook = forks.getFrom ();
|
|
|
|
sourceContact = forks.getContact();
|
|
|
|
forks.pop();
|
2010-03-09 09:24:29 -06:00
|
|
|
|
|
|
|
while ( sourceHook ) {
|
2013-12-03 18:58:58 -06:00
|
|
|
GCellTopology gcellConf ( getGCellGrid(), sourceHook, sourceContact );
|
|
|
|
gcellConf.construct( forks );
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
sourceHook = forks.getFrom();
|
|
|
|
sourceContact = forks.getContact();
|
|
|
|
forks.pop();
|
2010-03-09 09:24:29 -06:00
|
|
|
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_log(145,0) << "Popping (from) " << sourceHook << endl;
|
|
|
|
cdebug_log(145,0) << "Popping (to) " << sourceContact << endl;
|
2013-12-03 18:58:58 -06:00
|
|
|
}
|
2010-05-11 06:04:23 -05:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
lookupClear();
|
|
|
|
Session::revalidate();
|
2010-05-11 06:04:23 -05:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
#if THIS_IS_DISABLED
|
2010-05-11 06:04:23 -05:00
|
|
|
set<AutoSegment*>::iterator iover = overconstraineds.begin();
|
|
|
|
for ( ; iover != overconstraineds.end() ; ++iover ) {
|
2013-12-03 18:58:58 -06:00
|
|
|
(*iover)->makeDogLeg( (*iover)->getAutoSource()->getGCell(), true );
|
2010-05-11 06:04:23 -05:00
|
|
|
}
|
2013-12-03 18:58:58 -06:00
|
|
|
#endif
|
2010-05-11 06:04:23 -05:00
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
Session::revalidate();
|
|
|
|
GCellTopology::fixSegments();
|
2016-06-11 14:56:12 -05:00
|
|
|
cdebug_tabw(145,-1);
|
2010-03-09 09:24:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-12-03 18:58:58 -06:00
|
|
|
} // Katabatic namespace.
|