/* * This program source code file is part of KICAD, a free EDA CAD application. * * Copyright (C) 2013-2017 CERN * Copyright (C) 2019-2022 KiCad Developers, see AUTHORS.txt for contributors. * * @author Maciej Suminski * @author Tomasz Wlostowski * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, you may find one here: * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html * or you may search the http://www.gnu.org website for the version 2 license, * or you may write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA */ // #define CONNECTIVITY_DEBUG #ifndef __CONNECTIVITY_ALGO_H #define __CONNECTIVITY_ALGO_H #include #include #include #include #include #include #include #include #include #include #include #include #include #include class CN_RATSNEST_NODES; class BOARD; class BOARD_CONNECTED_ITEM; class BOARD_ITEM; class ZONE; class PROGRESS_REPORTER; /** * CN_EDGE represents a point-to-point connection, whether realized or unrealized (ie: tracks etc. * or a ratsnest line). */ class CN_EDGE { public: CN_EDGE() : m_weight( 0 ), m_visible( true ) {} CN_EDGE( const std::shared_ptr& aSource, const std::shared_ptr& aTarget, unsigned aWeight = 0 ) : m_source( aSource ), m_target( aTarget ), m_weight( aWeight ), m_visible( true ) {} /** * This sort operator provides a sort-by-weight for the ratsnest operation. * * @param aOther the other edge to compare. * @return true if our weight is smaller than the other weight. */ bool operator<( CN_EDGE aOther ) const { return m_weight < aOther.m_weight; } std::shared_ptr GetSourceNode() const { return m_source; } std::shared_ptr GetTargetNode() const { return m_target; } void SetSourceNode( const std::shared_ptr& aNode ) { m_source = aNode; } void SetTargetNode( const std::shared_ptr& aNode ) { m_target = aNode; } void SetWeight( unsigned weight ) { m_weight = weight; } unsigned GetWeight() const { return m_weight; } void SetVisible( bool aVisible ) { m_visible = aVisible; } bool IsVisible() const { return m_visible; } const VECTOR2I GetSourcePos() const { return m_source->Pos(); } const VECTOR2I GetTargetPos() const { return m_target->Pos(); } private: std::shared_ptr m_source; std::shared_ptr m_target; unsigned m_weight; bool m_visible; }; class CN_CONNECTIVITY_ALGO { public: enum CLUSTER_SEARCH_MODE { CSM_PROPAGATE, CSM_CONNECTIVITY_CHECK, CSM_RATSNEST }; using CLUSTERS = std::vector>; class ITEM_MAP_ENTRY { public: ITEM_MAP_ENTRY( CN_ITEM* aItem = nullptr ) { if( aItem ) m_items.push_back( aItem ); } void MarkItemsAsInvalid() { for( CN_ITEM* item : m_items ) item->SetValid( false ); } void Link( CN_ITEM* aItem ) { m_items.push_back( aItem ); } const std::list GetItems() const { return m_items; } std::list m_items; }; CN_CONNECTIVITY_ALGO() {} ~CN_CONNECTIVITY_ALGO() { Clear(); } bool ItemExists( const BOARD_CONNECTED_ITEM* aItem ) const { return m_itemMap.find( aItem ) != m_itemMap.end(); } ITEM_MAP_ENTRY& ItemEntry( const BOARD_CONNECTED_ITEM* aItem ) { return m_itemMap[ aItem ]; } bool IsNetDirty( int aNet ) const { if( aNet < 0 ) return false; return m_dirtyNets[ aNet ]; } void ClearDirtyFlags() { for( size_t ii = 0; ii < m_dirtyNets.size(); ii++ ) m_dirtyNets[ii] = false; } void GetDirtyClusters( CLUSTERS& aClusters ) const { for( const std::shared_ptr& cl : m_ratsnestClusters ) { int net = cl->OriginNet(); if( net >= 0 && m_dirtyNets[net] ) aClusters.push_back( cl ); } } int NetCount() const { return m_dirtyNets.size(); } void Build( BOARD* aBoard, PROGRESS_REPORTER* aReporter = nullptr ); void LocalBuild( const std::vector& aItems ); void Clear(); bool Remove( BOARD_ITEM* aItem ); bool Add( BOARD_ITEM* aItem ); const CLUSTERS SearchClusters( CLUSTER_SEARCH_MODE aMode, const KICAD_T aTypes[], int aSingleNet, CN_ITEM* rootItem = nullptr ); const CLUSTERS SearchClusters( CLUSTER_SEARCH_MODE aMode ); /** * Propagate nets from pads to other items in clusters. * @param aCommit is used to store undo information for items modified by the call. * @param aMode controls how clusters with conflicting nets are resolved. */ void PropagateNets( BOARD_COMMIT* aCommit = nullptr, PROPAGATE_MODE aMode = PROPAGATE_MODE::SKIP_CONFLICTS ); void FindIsolatedCopperIslands( ZONE* aZone, PCB_LAYER_ID aLayer, std::vector& aIslands ); /** * Find the copper islands that are not connected to a net. * * These are added to the m_islands vector. * N.B. This must be called after aZones has been refreshed. * * @param: aZones is the set of zones to search for islands. */ void FindIsolatedCopperIslands( std::vector& aZones ); const CLUSTERS& GetClusters(); const CN_LIST& ItemList() const { return m_itemList; } template void ForEachAnchor( Func&& aFunc ) const { for( CN_ITEM* item : m_itemList ) { for( std::shared_ptr& anchor : item->Anchors() ) aFunc( *anchor ); } } template void ForEachItem( Func&& aFunc ) const { for( CN_ITEM* item : m_itemList ) aFunc( *item ); } void MarkNetAsDirty( int aNet ); void SetProgressReporter( PROGRESS_REPORTER* aReporter ); private: void searchConnections(); void propagateConnections( BOARD_COMMIT* aCommit = nullptr, PROPAGATE_MODE aMode = PROPAGATE_MODE::SKIP_CONFLICTS ); template void add( Container& c, BItem brditem ) { CN_ITEM* item = c.Add( brditem ); m_itemMap[ brditem ] = ITEM_MAP_ENTRY( item ); } void markItemNetAsDirty( const BOARD_ITEM* aItem ); private: CN_LIST m_itemList; std::unordered_map m_itemMap; std::vector> m_connClusters; std::vector> m_ratsnestClusters; std::vector m_dirtyNets; PROGRESS_REPORTER* m_progressReporter = nullptr; }; class CN_VISITOR { public: CN_VISITOR( CN_ITEM* aItem ) : m_item( aItem ) {} bool operator()( CN_ITEM* aCandidate ); protected: void checkZoneItemConnection( CN_ZONE_LAYER* aZoneLayer, CN_ITEM* aItem ); void checkZoneZoneConnection( CN_ZONE_LAYER* aZoneLayerA, CN_ZONE_LAYER* aZoneLayerB ); protected: CN_ITEM* m_item; ///< The item we are looking for connections to. }; #endif