|
|
/*
* This program source code file is part of KiCad, a free EDA CAD application. * * Copyright (C) 2018 CERN * Copyright The KiCad Developers, see AUTHORS.txt for contributors. * * @author Jon Evans <jon@craftyjon.com> * * 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, see <http://www.gnu.org/licenses/>.
*/
#include <list>
#include <future>
#include <vector>
#include <unordered_map>
#include <core/profile.h>
#include <core/kicad_algo.h>
#include <common.h>
#include <core/kicad_algo.h>
#include <erc/erc.h>
#include <pin_type.h>
#include <sch_bus_entry.h>
#include <sch_symbol.h>
#include <sch_edit_frame.h>
#include <sch_line.h>
#include <sch_marker.h>
#include <sch_pin.h>
#include <sch_rule_area.h>
#include <sch_sheet.h>
#include <sch_sheet_path.h>
#include <sch_sheet_pin.h>
#include <sch_text.h>
#include <schematic.h>
#include <connection_graph.h>
#include <project/project_file.h>
#include <project/net_settings.h>
#include <widgets/ui_common.h>
#include <string_utils.h>
#include <core/thread_pool.h>
#include <wx/log.h>
#include <advanced_config.h> // for realtime connectivity switch in release builds
/**
* Flag to enable connectivity profiling * @ingroup trace_env_vars */static const wxChar DanglingProfileMask[] = wxT( "CONN_PROFILE" );
/**
* Flag to enable connectivity tracing * @ingroup trace_env_vars */static const wxChar ConnTrace[] = wxT( "CONN" );
void CONNECTION_SUBGRAPH::RemoveItem( SCH_ITEM* aItem ){ m_items.erase( aItem ); m_drivers.erase( aItem );
if( aItem == m_driver ) { m_driver = nullptr; m_driver_connection = nullptr; }
if( aItem->Type() == SCH_SHEET_PIN_T ) m_hier_pins.erase( static_cast<SCH_SHEET_PIN*>( aItem ) );
if( aItem->Type() == SCH_HIER_LABEL_T ) m_hier_ports.erase( static_cast<SCH_HIERLABEL*>( aItem ) );}
void CONNECTION_SUBGRAPH::ExchangeItem( SCH_ITEM* aOldItem, SCH_ITEM* aNewItem ){ m_items.erase( aOldItem ); m_items.insert( aNewItem );
m_drivers.erase( aOldItem ); m_drivers.insert( aNewItem );
if( aOldItem == m_driver ) { m_driver = aNewItem; m_driver_connection = aNewItem->GetOrInitConnection( m_sheet, m_graph ); }
SCH_CONNECTION* old_conn = aOldItem->Connection( &m_sheet ); SCH_CONNECTION* new_conn = aNewItem->GetOrInitConnection( m_sheet, m_graph );
if( old_conn && new_conn ) { new_conn->Clone( *old_conn );
if( old_conn->IsDriver() ) new_conn->SetDriver( aNewItem );
new_conn->ClearDirty(); }
if( aOldItem->Type() == SCH_SHEET_PIN_T ) { m_hier_pins.erase( static_cast<SCH_SHEET_PIN*>( aOldItem ) ); m_hier_pins.insert( static_cast<SCH_SHEET_PIN*>( aNewItem ) ); }
if( aOldItem->Type() == SCH_HIER_LABEL_T ) { m_hier_ports.erase( static_cast<SCH_HIERLABEL*>( aOldItem ) ); m_hier_ports.insert( static_cast<SCH_HIERLABEL*>( aNewItem ) ); }}
bool CONNECTION_SUBGRAPH::ResolveDrivers( bool aCheckMultipleDrivers ){ std::lock_guard lock( m_driver_mutex );
auto candidate_cmp = [&]( SCH_ITEM* a, SCH_ITEM* b ) -> bool { // meet irreflexive requirements of std::sort
if( a == b ) return false;
SCH_CONNECTION* ac = a->Connection( &m_sheet ); SCH_CONNECTION* bc = b->Connection( &m_sheet );
// Ensure we don't pick the subset over the superset
if( ac->IsBus() && bc->IsBus() ) return bc->IsSubsetOf( ac );
// Ensure we don't pick a hidden power pin on a regular symbol over
// one on a power symbol
if( a->Type() == SCH_PIN_T && b->Type() == SCH_PIN_T ) { SCH_PIN* pa = static_cast<SCH_PIN*>( a ); SCH_PIN* pb = static_cast<SCH_PIN*>( b );
bool aPower = pa->GetLibPin()->GetParentSymbol()->IsPower(); bool bPower = pb->GetLibPin()->GetParentSymbol()->IsPower();
if( aPower && !bPower ) return true; else if( bPower && !aPower ) return false; }
const wxString& a_name = GetNameForDriver( a ); const wxString& b_name = GetNameForDriver( b ); bool a_lowQualityName = a_name.Contains( "-Pad" ); bool b_lowQualityName = b_name.Contains( "-Pad" );
if( a_lowQualityName && !b_lowQualityName ) return false; else if( b_lowQualityName && !a_lowQualityName ) return true; else return a_name < b_name; };
PRIORITY highest_priority = PRIORITY::INVALID; std::set<SCH_ITEM*, decltype( candidate_cmp )> candidates( candidate_cmp ); std::set<SCH_ITEM*> strong_drivers;
m_driver = nullptr;
// Hierarchical labels are lower priority than local labels here,
// because on the first pass we want local labels to drive subgraphs
// so that we can identify same-sheet neighbors and link them together.
// Hierarchical labels will end up overriding the final net name if
// a higher-level sheet has a different name during the hierarchical
// pass.
for( SCH_ITEM* item : m_drivers ) { PRIORITY item_priority = GetDriverPriority( item );
if( item_priority == PRIORITY::PIN ) { SCH_PIN* pin = static_cast<SCH_PIN*>( item );
if( !static_cast<SCH_SYMBOL*>( pin->GetParentSymbol() )->IsInNetlist() ) continue; }
if( item_priority >= PRIORITY::HIER_LABEL ) strong_drivers.insert( item );
if( item_priority > highest_priority ) { candidates.clear(); candidates.insert( item ); highest_priority = item_priority; } else if( !candidates.empty() && ( item_priority == highest_priority ) ) { candidates.insert( item ); } }
if( highest_priority >= PRIORITY::HIER_LABEL ) m_strong_driver = true;
// Power pins are 5, global labels are 6
m_local_driver = ( highest_priority < PRIORITY::POWER_PIN );
if( !candidates.empty() ) { if( candidates.size() > 1 ) { if( highest_priority == PRIORITY::SHEET_PIN ) { // We have multiple options, and they are all hierarchical
// sheet pins. Let's prefer outputs over inputs.
for( SCH_ITEM* c : candidates ) { SCH_SHEET_PIN* p = static_cast<SCH_SHEET_PIN*>( c );
if( p->GetShape() == LABEL_FLAG_SHAPE::L_OUTPUT ) { m_driver = c; break; } } } }
if( !m_driver ) m_driver = *candidates.begin(); }
if( strong_drivers.size() > 1 ) m_multiple_drivers = true;
// Drop weak drivers
if( m_strong_driver ) { m_drivers.clear(); m_drivers.insert( strong_drivers.begin(), strong_drivers.end() ); }
// Cache driver connection
if( m_driver ) { m_driver_connection = m_driver->Connection( &m_sheet ); m_driver_connection->ConfigureFromLabel( GetNameForDriver( m_driver ) ); m_driver_connection->SetDriver( m_driver ); m_driver_connection->ClearDirty(); } else if( !m_is_bus_member ) { m_driver_connection = nullptr; }
return ( m_driver != nullptr );}
void CONNECTION_SUBGRAPH::getAllConnectedItems( std::set<std::pair<SCH_SHEET_PATH, SCH_ITEM*>>& aItems, std::set<CONNECTION_SUBGRAPH*>& aSubgraphs ){ CONNECTION_SUBGRAPH* sg = this;
while( sg->m_absorbed_by ) { wxASSERT( sg->m_graph == sg->m_absorbed_by->m_graph ); sg = sg->m_absorbed_by; }
// If we are unable to insert the subgraph into the set, then we have already
// visited it and don't need to add it again.
if( aSubgraphs.insert( sg ).second == false ) return;
aSubgraphs.insert( sg->m_absorbed_subgraphs.begin(), sg->m_absorbed_subgraphs.end() );
for( SCH_ITEM* item : sg->m_items ) aItems.emplace( m_sheet, item );
for( CONNECTION_SUBGRAPH* child_sg : sg->m_hier_children ) child_sg->getAllConnectedItems( aItems, aSubgraphs );}
wxString CONNECTION_SUBGRAPH::GetNetName() const{ if( !m_driver || m_dirty ) return "";
if( !m_driver->Connection( &m_sheet ) ) {#ifdef CONNECTIVITY_DEBUG
wxASSERT_MSG( false, wxS( "Tried to get the net name of an item with no connection" ) );#endif
return ""; }
return m_driver->Connection( &m_sheet )->Name();}
std::vector<SCH_ITEM*> CONNECTION_SUBGRAPH::GetAllBusLabels() const{ std::vector<SCH_ITEM*> labels;
for( SCH_ITEM* item : m_drivers ) { switch( item->Type() ) { case SCH_LABEL_T: case SCH_GLOBAL_LABEL_T: { CONNECTION_TYPE type = item->Connection( &m_sheet )->Type();
// Only consider bus vectors
if( type == CONNECTION_TYPE::BUS || type == CONNECTION_TYPE::BUS_GROUP ) labels.push_back( item );
break; }
default: break; } }
return labels;}
std::vector<SCH_ITEM*> CONNECTION_SUBGRAPH::GetVectorBusLabels() const{ std::vector<SCH_ITEM*> labels;
for( SCH_ITEM* item : m_drivers ) { switch( item->Type() ) { case SCH_LABEL_T: case SCH_GLOBAL_LABEL_T: { SCH_CONNECTION* label_conn = item->Connection( &m_sheet );
// Only consider bus vectors
if( label_conn->Type() == CONNECTION_TYPE::BUS ) labels.push_back( item );
break; }
default: break; } }
return labels;}
wxString CONNECTION_SUBGRAPH::driverName( SCH_ITEM* aItem ) const{ switch( aItem->Type() ) { case SCH_PIN_T: { SCH_PIN* pin = static_cast<SCH_PIN*>( aItem ); bool forceNoConnect = m_no_connect != nullptr;
return pin->GetDefaultNetName( m_sheet, forceNoConnect ); }
case SCH_LABEL_T: case SCH_GLOBAL_LABEL_T: case SCH_HIER_LABEL_T: { SCH_LABEL_BASE* label = static_cast<SCH_LABEL_BASE*>( aItem );
// NB: any changes here will need corresponding changes in SCH_LABEL_BASE::cacheShownText()
return EscapeString( label->GetShownText( &m_sheet, false ), CTX_NETNAME ); }
case SCH_SHEET_PIN_T: { // Sheet pins need to use their parent sheet as their starting sheet or they will resolve
// variables on the current sheet first
SCH_SHEET_PIN* sheetPin = static_cast<SCH_SHEET_PIN*>( aItem ); SCH_SHEET_PATH path = m_sheet;
if( path.Last() != sheetPin->GetParent() ) path.push_back( sheetPin->GetParent() );
return EscapeString( sheetPin->GetShownText( &path, false ), CTX_NETNAME ); }
default: wxFAIL_MSG( wxS( "Unhandled item type in GetNameForDriver" ) ); return wxEmptyString; }}
const wxString& CONNECTION_SUBGRAPH::GetNameForDriver( SCH_ITEM* aItem ) const{ if( aItem->HasCachedDriverName() ) return aItem->GetCachedDriverName();
auto it = m_driver_name_cache.find( aItem );
if( it != m_driver_name_cache.end() ) return it->second;
return m_driver_name_cache.emplace( aItem, driverName( aItem ) ).first->second;}
const std::vector<std::pair<wxString, SCH_ITEM*>>CONNECTION_SUBGRAPH::GetNetclassesForDriver( SCH_ITEM* aItem ) const{ std::vector<std::pair<wxString, SCH_ITEM*>> foundNetclasses;
const std::unordered_set<SCH_RULE_AREA*>& ruleAreaCache = aItem->GetRuleAreaCache();
// Get netclasses on attached rule areas
for( SCH_RULE_AREA* ruleArea : ruleAreaCache ) { const std::vector<std::pair<wxString, SCH_ITEM*>> ruleNetclasses = ruleArea->GetResolvedNetclasses();
if( ruleNetclasses.size() > 0 ) { foundNetclasses.insert( foundNetclasses.end(), ruleNetclasses.begin(), ruleNetclasses.end() ); } }
// Get netclasses on child fields
aItem->RunOnChildren( [&]( SCH_ITEM* aChild ) { if( aChild->Type() == SCH_FIELD_T ) { SCH_FIELD* field = static_cast<SCH_FIELD*>( aChild );
if( field->GetCanonicalName() == wxT( "Netclass" ) ) { wxString netclass = field->GetShownText( &m_sheet, false );
if( netclass != wxEmptyString ) foundNetclasses.push_back( { netclass, aItem } ); } } } );
std::sort( foundNetclasses.begin(), foundNetclasses.end(), []( const std::pair<wxString, SCH_ITEM*>& i1, const std::pair<wxString, SCH_ITEM*>& i2 ) { return i1.first < i2.first; } );
return foundNetclasses;}
void CONNECTION_SUBGRAPH::Absorb( CONNECTION_SUBGRAPH* aOther ){ wxASSERT( m_sheet == aOther->m_sheet );
for( SCH_ITEM* item : aOther->m_items ) { item->Connection( &m_sheet )->SetSubgraphCode( m_code ); AddItem( item ); }
m_absorbed_subgraphs.insert( aOther ); m_absorbed_subgraphs.insert( aOther->m_absorbed_subgraphs.begin(), aOther->m_absorbed_subgraphs.end() );
m_bus_neighbors.insert( aOther->m_bus_neighbors.begin(), aOther->m_bus_neighbors.end() ); m_bus_parents.insert( aOther->m_bus_parents.begin(), aOther->m_bus_parents.end() );
m_multiple_drivers |= aOther->m_multiple_drivers;
std::function<void( CONNECTION_SUBGRAPH* )> set_absorbed_by = [ & ]( CONNECTION_SUBGRAPH *child ) { child->m_absorbed_by = this;
for( CONNECTION_SUBGRAPH* subchild : child->m_absorbed_subgraphs ) set_absorbed_by( subchild ); };
aOther->m_absorbed = true; aOther->m_dirty = false; aOther->m_driver = nullptr; aOther->m_driver_connection = nullptr;
set_absorbed_by( aOther );}
void CONNECTION_SUBGRAPH::AddItem( SCH_ITEM* aItem ){ m_items.insert( aItem );
if( aItem->Connection( &m_sheet )->IsDriver() ) m_drivers.insert( aItem );
if( aItem->Type() == SCH_SHEET_PIN_T ) m_hier_pins.insert( static_cast<SCH_SHEET_PIN*>( aItem ) ); else if( aItem->Type() == SCH_HIER_LABEL_T ) m_hier_ports.insert( static_cast<SCH_HIERLABEL*>( aItem ) );}
void CONNECTION_SUBGRAPH::UpdateItemConnections(){ if( !m_driver_connection ) return;
for( SCH_ITEM* item : m_items ) { SCH_CONNECTION* item_conn = item->GetOrInitConnection( m_sheet, m_graph );
if( !item_conn ) continue;
if( ( m_driver_connection->IsBus() && item_conn->IsNet() ) || ( m_driver_connection->IsNet() && item_conn->IsBus() ) ) { continue; }
if( item != m_driver ) { item_conn->Clone( *m_driver_connection ); item_conn->ClearDirty(); } }}
CONNECTION_SUBGRAPH::PRIORITY CONNECTION_SUBGRAPH::GetDriverPriority( SCH_ITEM* aDriver ){ if( !aDriver ) return PRIORITY::NONE;
switch( aDriver->Type() ) { case SCH_SHEET_PIN_T: return PRIORITY::SHEET_PIN; case SCH_HIER_LABEL_T: return PRIORITY::HIER_LABEL; case SCH_LABEL_T: return PRIORITY::LOCAL_LABEL; case SCH_GLOBAL_LABEL_T: return PRIORITY::GLOBAL; case SCH_PIN_T: { SCH_PIN* sch_pin = static_cast<SCH_PIN*>( aDriver ); const SCH_SYMBOL* sym = static_cast<SCH_SYMBOL*>( sch_pin->GetParentSymbol() );
if( sch_pin->IsGlobalPower() ) return PRIORITY::POWER_PIN; else if( !sym || sym->GetExcludedFromBoard() || sym->GetLibSymbolRef()->GetReferenceField().GetText().StartsWith( '#' ) ) return PRIORITY::NONE; else return PRIORITY::PIN; }
default: return PRIORITY::NONE; }}
void CONNECTION_GRAPH::Merge( CONNECTION_GRAPH& aGraph ){ std::copy( aGraph.m_items.begin(), aGraph.m_items.end(), std::back_inserter( m_items ) );
for( SCH_ITEM* item : aGraph.m_items ) item->SetConnectionGraph( this );
std::copy( aGraph.m_subgraphs.begin(), aGraph.m_subgraphs.end(), std::back_inserter( m_subgraphs ) );
for( CONNECTION_SUBGRAPH* sg : aGraph.m_subgraphs ) { if( sg->m_driver_connection ) sg->m_driver_connection->SetGraph( this );
sg->m_graph = this; }
std::copy( aGraph.m_driver_subgraphs.begin(), aGraph.m_driver_subgraphs.end(), std::back_inserter( m_driver_subgraphs ) );
std::copy( aGraph.m_global_power_pins.begin(), aGraph.m_global_power_pins.end(), std::back_inserter( m_global_power_pins ) );
for( auto& [key, value] : aGraph.m_net_name_to_subgraphs_map ) m_net_name_to_subgraphs_map.insert_or_assign( key, value );
for( auto& [key, value] : aGraph.m_sheet_to_subgraphs_map ) m_sheet_to_subgraphs_map.insert_or_assign( key, value );
for( auto& [key, value] : aGraph.m_net_name_to_code_map ) m_net_name_to_code_map.insert_or_assign( key, value );
for( auto& [key, value] : aGraph.m_bus_name_to_code_map ) m_bus_name_to_code_map.insert_or_assign( key, value );
for( auto& [key, value] : aGraph.m_net_code_to_subgraphs_map ) m_net_code_to_subgraphs_map.insert_or_assign( key, value );
for( auto& [key, value] : aGraph.m_item_to_subgraph_map ) m_item_to_subgraph_map.insert_or_assign( key, value );
for( auto& [key, value] : aGraph.m_local_label_cache ) m_local_label_cache.insert_or_assign( key, value );
for( auto& [key, value] : aGraph.m_global_label_cache ) m_global_label_cache.insert_or_assign( key, value );
m_last_bus_code = std::max( m_last_bus_code, aGraph.m_last_bus_code ); m_last_net_code = std::max( m_last_net_code, aGraph.m_last_net_code ); m_last_subgraph_code = std::max( m_last_subgraph_code, aGraph.m_last_subgraph_code );
}
void CONNECTION_GRAPH::ExchangeItem( SCH_ITEM* aOldItem, SCH_ITEM* aNewItem ){ wxCHECK2( aOldItem->Type() == aNewItem->Type(), return );
auto exchange = [&]( SCH_ITEM* aOld, SCH_ITEM* aNew ) { auto it = m_item_to_subgraph_map.find( aOld );
if( it == m_item_to_subgraph_map.end() ) return;
CONNECTION_SUBGRAPH* sg = it->second;
sg->ExchangeItem( aOld, aNew );
m_item_to_subgraph_map.erase( it ); m_item_to_subgraph_map.emplace( aNew, sg );
for( auto it2 = m_items.begin(); it2 != m_items.end(); ++it2 ) { if( *it2 == aOld ) { *it2 = aNew; break; } } };
exchange( aOldItem, aNewItem );
if( aOldItem->Type() == SCH_SYMBOL_T ) { SCH_SYMBOL* oldSymbol = static_cast<SCH_SYMBOL*>( aOldItem ); SCH_SYMBOL* newSymbol = static_cast<SCH_SYMBOL*>( aNewItem ); std::vector<SCH_PIN*> oldPins = oldSymbol->GetPins( &m_schematic->CurrentSheet() ); std::vector<SCH_PIN*> newPins = newSymbol->GetPins( &m_schematic->CurrentSheet() );
wxCHECK2( oldPins.size() == newPins.size(), return );
for( size_t ii = 0; ii < oldPins.size(); ii++ ) { exchange( oldPins[ii], newPins[ii] ); } }}
void CONNECTION_GRAPH::Reset(){ for( auto& subgraph : m_subgraphs ) { /// Only delete subgraphs of which we are the owner
if( subgraph->m_graph == this ) delete subgraph; }
m_items.clear(); m_subgraphs.clear(); m_driver_subgraphs.clear(); m_sheet_to_subgraphs_map.clear(); m_global_power_pins.clear(); m_bus_alias_cache.clear(); m_net_name_to_code_map.clear(); m_bus_name_to_code_map.clear(); m_net_code_to_subgraphs_map.clear(); m_net_name_to_subgraphs_map.clear(); m_item_to_subgraph_map.clear(); m_local_label_cache.clear(); m_global_label_cache.clear(); m_last_net_code = 1; m_last_bus_code = 1; m_last_subgraph_code = 1;}
void CONNECTION_GRAPH::Recalculate( const SCH_SHEET_LIST& aSheetList, bool aUnconditional, std::function<void( SCH_ITEM* )>* aChangedItemHandler ){ PROF_TIMER recalc_time( "CONNECTION_GRAPH::Recalculate" );
if( aUnconditional ) Reset();
PROF_TIMER update_items( "updateItemConnectivity" );
m_sheetList = aSheetList; std::set<SCH_ITEM*> dirty_items;
for( const SCH_SHEET_PATH& sheet : aSheetList ) { std::vector<SCH_ITEM*> items;
// Store current unit value, to replace it after calculations
std::vector<std::pair<SCH_SYMBOL*, int>> symbolsChanged;
for( SCH_ITEM* item : sheet.LastScreen()->Items() ) { if( item->IsConnectable() && ( aUnconditional || item->IsConnectivityDirty() ) ) { wxLogTrace( ConnTrace, wxT( "Adding item %s to connectivity graph update" ), item->GetTypeDesc() ); items.push_back( item ); dirty_items.insert( item );
// Add any symbol dirty pins to the dirty_items list
if( item->Type() == SCH_SYMBOL_T ) { SCH_SYMBOL* symbol = static_cast<SCH_SYMBOL*>( item );
for( SCH_PIN* pin : symbol->GetPins( &sheet ) ) { if( pin->IsConnectivityDirty() ) { dirty_items.insert( pin ); } } } } // If the symbol isn't dirty, look at the pins
// TODO: remove symbols from connectivity graph and only use pins
else if( item->Type() == SCH_SYMBOL_T ) { SCH_SYMBOL* symbol = static_cast<SCH_SYMBOL*>( item );
for( SCH_PIN* pin : symbol->GetPins( &sheet ) ) { if( pin->IsConnectivityDirty() ) { items.push_back( pin ); dirty_items.insert( pin ); } } } else if( item->Type() == SCH_SHEET_T ) { SCH_SHEET* sheetItem = static_cast<SCH_SHEET*>( item );
for( SCH_SHEET_PIN* pin : sheetItem->GetPins() ) { if( pin->IsConnectivityDirty() ) { items.push_back( pin ); dirty_items.insert( pin ); } } }
// Ensure the hierarchy info stored in the SCH_SCREEN (such as symbol units) reflects
// the current SCH_SHEET_PATH
if( item->Type() == SCH_SYMBOL_T ) { SCH_SYMBOL* symbol = static_cast<SCH_SYMBOL*>( item ); int new_unit = symbol->GetUnitSelection( &sheet );
// Store the initial unit value so we can restore it after calculations
if( symbol->GetUnit() != new_unit ) symbolsChanged.push_back( { symbol, symbol->GetUnit() } );
symbol->SetUnit( new_unit ); } }
m_items.reserve( m_items.size() + items.size() );
updateItemConnectivity( sheet, items );
// UpdateDanglingState() also adds connected items for SCH_TEXT
sheet.LastScreen()->TestDanglingEnds( &sheet, aChangedItemHandler );
// Restore the m_unit member variables where we had to change them
for( const auto& [ symbol, originalUnit ] : symbolsChanged ) symbol->SetUnit( originalUnit ); }
// Restore the danlging states of items in the current SCH_SCREEN to match the current
// SCH_SHEET_PATH.
m_schematic->CurrentSheet().LastScreen()->TestDanglingEnds( &m_schematic->CurrentSheet(), aChangedItemHandler );
for( SCH_ITEM* item : dirty_items ) item->SetConnectivityDirty( false );
if( wxLog::IsAllowedTraceMask( DanglingProfileMask ) ) update_items.Show();
PROF_TIMER build_graph( "buildConnectionGraph" );
buildConnectionGraph( aChangedItemHandler, aUnconditional );
if( wxLog::IsAllowedTraceMask( DanglingProfileMask ) ) build_graph.Show();
recalc_time.Stop();
if( wxLog::IsAllowedTraceMask( DanglingProfileMask ) ) recalc_time.Show();}
std::set<std::pair<SCH_SHEET_PATH, SCH_ITEM*>> CONNECTION_GRAPH::ExtractAffectedItems( const std::set<SCH_ITEM*> &aItems ){ std::set<std::pair<SCH_SHEET_PATH, SCH_ITEM*>> retvals; std::set<CONNECTION_SUBGRAPH*> subgraphs;
auto traverse_subgraph = [&retvals, &subgraphs]( CONNECTION_SUBGRAPH* aSubgraph ) { // Find the primary subgraph on this sheet
while( aSubgraph->m_absorbed_by ) { wxASSERT( aSubgraph->m_graph == aSubgraph->m_absorbed_by->m_graph ); aSubgraph = aSubgraph->m_absorbed_by; }
// Find the top most connected subgraph on all sheets
while( aSubgraph->m_hier_parent ) { wxASSERT( aSubgraph->m_graph == aSubgraph->m_hier_parent->m_graph ); aSubgraph = aSubgraph->m_hier_parent; }
// Recurse through all subsheets to collect connected items
aSubgraph->getAllConnectedItems( retvals, subgraphs ); };
auto extract_element = [&]( SCH_ITEM* aItem ) { CONNECTION_SUBGRAPH* item_sg = GetSubgraphForItem( aItem );
if( !item_sg ) { wxLogTrace( ConnTrace, wxT( "Item %s not found in connection graph" ), aItem->GetTypeDesc() ); return; } if( !item_sg->ResolveDrivers( true ) ) { wxLogTrace( ConnTrace, wxT( "Item %s in subgraph %ld (%p) has no driver" ), aItem->GetTypeDesc(), item_sg->m_code, item_sg ); }
std::vector<CONNECTION_SUBGRAPH*> sg_to_scan = GetAllSubgraphs( item_sg->GetNetName() );
if( sg_to_scan.empty() ) { wxLogTrace( ConnTrace, wxT( "Item %s in subgraph %ld with net %s has no neighbors" ), aItem->GetTypeDesc(), item_sg->m_code, item_sg->GetNetName() ); sg_to_scan.push_back( item_sg ); }
wxLogTrace( ConnTrace, wxT( "Removing all item %s connections from subgraph %ld with net %s: Found " "%zu subgraphs" ), aItem->GetTypeDesc(), item_sg->m_code, item_sg->GetNetName(), sg_to_scan.size() );
for( CONNECTION_SUBGRAPH* sg : sg_to_scan )
{ traverse_subgraph( sg );
for( auto& bus_it : sg->m_bus_neighbors ) { for( CONNECTION_SUBGRAPH* bus_sg : bus_it.second ) traverse_subgraph( bus_sg ); }
for( auto& bus_it : sg->m_bus_parents ) { for( CONNECTION_SUBGRAPH* bus_sg : bus_it.second ) traverse_subgraph( bus_sg ); } }
alg::delete_matching( m_items, aItem ); };
for( SCH_ITEM* item : aItems ) { if( item->Type() == SCH_SHEET_T ) { SCH_SHEET* sheet = static_cast<SCH_SHEET*>( item );
for( SCH_SHEET_PIN* pin : sheet->GetPins() ) extract_element( pin ); } else if ( item->Type() == SCH_SYMBOL_T ) { SCH_SYMBOL* symbol = static_cast<SCH_SYMBOL*>( item );
for( SCH_PIN* pin : symbol->GetPins( &m_schematic->CurrentSheet() ) ) extract_element( pin ); } else { extract_element( item ); } }
removeSubgraphs( subgraphs );
for( const auto& [path, item] : retvals ) alg::delete_matching( m_items, item );
return retvals;}
void CONNECTION_GRAPH::RemoveItem( SCH_ITEM* aItem ){ auto it = m_item_to_subgraph_map.find( aItem );
if( it == m_item_to_subgraph_map.end() ) return;
CONNECTION_SUBGRAPH* subgraph = it->second;
while(subgraph->m_absorbed_by ) subgraph = subgraph->m_absorbed_by;
subgraph->RemoveItem( aItem ); alg::delete_matching( m_items, aItem ); m_item_to_subgraph_map.erase( it );}
void CONNECTION_GRAPH::removeSubgraphs( std::set<CONNECTION_SUBGRAPH*>& aSubgraphs ){ wxLogTrace( ConnTrace, wxT( "Removing %zu subgraphs" ), aSubgraphs.size() ); std::sort( m_driver_subgraphs.begin(), m_driver_subgraphs.end() ); std::sort( m_subgraphs.begin(), m_subgraphs.end() ); std::set<int> codes_to_remove;
for( auto& el : m_sheet_to_subgraphs_map ) { std::sort( el.second.begin(), el.second.end() ); }
for( CONNECTION_SUBGRAPH* sg : aSubgraphs ) { for( auto& it : sg->m_bus_neighbors ) { for( CONNECTION_SUBGRAPH* neighbor : it.second ) { auto& parents = neighbor->m_bus_parents[it.first];
for( auto test = parents.begin(); test != parents.end(); ) { if( *test == sg ) test = parents.erase( test ); else ++test; }
if( parents.empty() ) neighbor->m_bus_parents.erase( it.first ); } }
for( auto& it : sg->m_bus_parents ) { for( CONNECTION_SUBGRAPH* parent : it.second ) { auto& neighbors = parent->m_bus_neighbors[it.first];
for( auto test = neighbors.begin(); test != neighbors.end(); ) { if( *test == sg ) test = neighbors.erase( test ); else ++test; }
if( neighbors.empty() ) parent->m_bus_neighbors.erase( it.first ); } }
{ auto it = std::lower_bound( m_driver_subgraphs.begin(), m_driver_subgraphs.end(), sg );
while( it != m_driver_subgraphs.end() && *it == sg ) it = m_driver_subgraphs.erase( it ); }
{ auto it = std::lower_bound( m_subgraphs.begin(), m_subgraphs.end(), sg );
while( it != m_subgraphs.end() && *it == sg ) it = m_subgraphs.erase( it ); }
for( auto& el : m_sheet_to_subgraphs_map ) { auto it = std::lower_bound( el.second.begin(), el.second.end(), sg );
while( it != el.second.end() && *it == sg ) it = el.second.erase( it ); }
auto remove_sg = [sg]( auto it ) -> bool { for( const CONNECTION_SUBGRAPH* test_sg : it->second ) { if( sg == test_sg ) return true; }
return false; };
for( auto it = m_global_label_cache.begin(); it != m_global_label_cache.end(); ) { if( remove_sg( it ) ) it = m_global_label_cache.erase( it ); else ++it; }
for( auto it = m_local_label_cache.begin(); it != m_local_label_cache.end(); ) { if( remove_sg( it ) ) it = m_local_label_cache.erase( it ); else ++it; }
for( auto it = m_net_code_to_subgraphs_map.begin(); it != m_net_code_to_subgraphs_map.end(); ) { if( remove_sg( it ) ) { codes_to_remove.insert( it->first.Netcode ); it = m_net_code_to_subgraphs_map.erase( it ); } else ++it; }
for( auto it = m_net_name_to_subgraphs_map.begin(); it != m_net_name_to_subgraphs_map.end(); ) { if( remove_sg( it ) ) it = m_net_name_to_subgraphs_map.erase( it ); else ++it; }
for( auto it = m_item_to_subgraph_map.begin(); it != m_item_to_subgraph_map.end(); ) { if( it->second == sg ) it = m_item_to_subgraph_map.erase( it ); else ++it; }
}
for( auto it = m_net_name_to_code_map.begin(); it != m_net_name_to_code_map.end(); ) { if( codes_to_remove.contains( it->second ) ) it = m_net_name_to_code_map.erase( it ); else ++it; }
for( auto it = m_bus_name_to_code_map.begin(); it != m_bus_name_to_code_map.end(); ) { if( codes_to_remove.contains( it->second ) ) it = m_bus_name_to_code_map.erase( it ); else ++it; }
for( CONNECTION_SUBGRAPH* sg : aSubgraphs ) { sg->m_code = -1; sg->m_graph = nullptr; delete sg; }}
void CONNECTION_GRAPH::updateItemConnectivity( const SCH_SHEET_PATH& aSheet, const std::vector<SCH_ITEM*>& aItemList ){ wxLogTrace( wxT( "Updating connectivity for sheet %s with %zu items" ), aSheet.Last()->GetFileName(), aItemList.size() ); std::map<VECTOR2I, std::vector<SCH_ITEM*>> connection_map;
auto updatePin = [&]( SCH_PIN* aPin, SCH_CONNECTION* aConn ) { aConn->SetType( CONNECTION_TYPE::NET );
// because calling the first time is not thread-safe
wxString name = aPin->GetDefaultNetName( aSheet ); aPin->ClearConnectedItems( aSheet );
// power symbol pins need to be post-processed later
if( aPin->IsGlobalPower() ) { aConn->SetName( name ); m_global_power_pins.emplace_back( std::make_pair( aSheet, aPin ) ); } };
for( SCH_ITEM* item : aItemList ) { std::vector<VECTOR2I> points = item->GetConnectionPoints(); item->ClearConnectedItems( aSheet );
if( item->Type() == SCH_SHEET_T ) { for( SCH_SHEET_PIN* pin : static_cast<SCH_SHEET*>( item )->GetPins() ) { pin->InitializeConnection( aSheet, this );
pin->ClearConnectedItems( aSheet );
connection_map[ pin->GetTextPos() ].push_back( pin ); m_items.emplace_back( pin ); } } else if( item->Type() == SCH_SYMBOL_T ) { SCH_SYMBOL* symbol = static_cast<SCH_SYMBOL*>( item );
for( SCH_PIN* pin : symbol->GetPins( &aSheet ) ) { m_items.emplace_back( pin ); SCH_CONNECTION* conn = pin->InitializeConnection( aSheet, this ); updatePin( pin, conn ); connection_map[ pin->GetPosition() ].push_back( pin ); } } else { m_items.emplace_back( item ); SCH_CONNECTION* conn = item->InitializeConnection( aSheet, this );
// Set bus/net property here so that the propagation code uses it
switch( item->Type() ) { case SCH_LINE_T: conn->SetType( item->GetLayer() == LAYER_BUS ? CONNECTION_TYPE::BUS : CONNECTION_TYPE::NET ); break;
case SCH_BUS_BUS_ENTRY_T: conn->SetType( CONNECTION_TYPE::BUS );
// clean previous (old) links:
static_cast<SCH_BUS_BUS_ENTRY*>( item )->m_connected_bus_items[0] = nullptr; static_cast<SCH_BUS_BUS_ENTRY*>( item )->m_connected_bus_items[1] = nullptr; break;
case SCH_PIN_T: if( points.empty() ) points = { static_cast<SCH_PIN*>( item )->GetPosition() };
updatePin( static_cast<SCH_PIN*>( item ), conn ); break;
case SCH_BUS_WIRE_ENTRY_T: conn->SetType( CONNECTION_TYPE::NET );
// clean previous (old) link:
static_cast<SCH_BUS_WIRE_ENTRY*>( item )->m_connected_bus_item = nullptr; break;
default: break; }
for( const VECTOR2I& point : points ) connection_map[ point ].push_back( item ); } }
for( const auto& it : connection_map ) { std::vector<SCH_ITEM*> connection_vec = it.second; std::sort( connection_vec.begin(), connection_vec.end() ); alg::remove_duplicates( connection_vec );
// Pre-scan to see if we have a bus at this location
SCH_LINE* busLine = aSheet.LastScreen()->GetBus( it.first );
std::mutex update_mutex;
auto update_lambda = [&]( SCH_ITEM* connected_item ) -> size_t { // Bus entries are special: they can have connection points in the
// middle of a wire segment, because the junction algo doesn't split
// the segment in two where you place a bus entry. This means that
// bus entries that don't land on the end of a line segment need to
// have "virtual" connection points to the segments they graphically
// touch.
if( connected_item->Type() == SCH_BUS_WIRE_ENTRY_T ) { // If this location only has the connection point of the bus
// entry itself, this means that either the bus entry is not
// connected to anything graphically, or that it is connected to
// a segment at some point other than at one of the endpoints.
if( connection_vec.size() == 1 ) { if( busLine ) { auto bus_entry = static_cast<SCH_BUS_WIRE_ENTRY*>( connected_item ); bus_entry->m_connected_bus_item = busLine; } } }
// Bus-to-bus entries are treated just like bus wires
else if( connected_item->Type() == SCH_BUS_BUS_ENTRY_T ) { if( connection_vec.size() < 2 ) { if( busLine ) { auto bus_entry = static_cast<SCH_BUS_BUS_ENTRY*>( connected_item );
if( it.first == bus_entry->GetPosition() ) bus_entry->m_connected_bus_items[0] = busLine; else bus_entry->m_connected_bus_items[1] = busLine;
std::lock_guard<std::mutex> lock( update_mutex ); bus_entry->AddConnectionTo( aSheet, busLine ); busLine->AddConnectionTo( aSheet, bus_entry ); } } }
// Change junctions to be on bus junction layer if they are touching a bus
else if( connected_item->Type() == SCH_JUNCTION_T ) { connected_item->SetLayer( busLine ? LAYER_BUS_JUNCTION : LAYER_JUNCTION ); }
for( SCH_ITEM* test_item : connection_vec ) { bool bus_connection_ok = true;
if( test_item == connected_item ) continue;
// Set up the link between the bus entry net and the bus
if( connected_item->Type() == SCH_BUS_WIRE_ENTRY_T ) { if( test_item->GetLayer() == LAYER_BUS ) { auto bus_entry = static_cast<SCH_BUS_WIRE_ENTRY*>( connected_item ); bus_entry->m_connected_bus_item = test_item; } }
// Bus entries only connect to bus lines on the end that is touching a bus line.
// If the user has overlapped another net line with the endpoint of the bus entry
// where the entry connects to a bus, we don't want to short-circuit it.
if( connected_item->Type() == SCH_BUS_WIRE_ENTRY_T ) { bus_connection_ok = !busLine || test_item->GetLayer() == LAYER_BUS; } else if( test_item->Type() == SCH_BUS_WIRE_ENTRY_T ) { bus_connection_ok = !busLine || connected_item->GetLayer() == LAYER_BUS; }
if( connected_item->ConnectionPropagatesTo( test_item ) && test_item->ConnectionPropagatesTo( connected_item ) && bus_connection_ok ) { connected_item->AddConnectionTo( aSheet, test_item ); } }
// If we got this far and did not find a connected bus item for a bus entry,
// we should do a manual scan in case there is a bus item on this connection
// point but we didn't pick it up earlier because there is *also* a net item here.
if( connected_item->Type() == SCH_BUS_WIRE_ENTRY_T ) { auto bus_entry = static_cast<SCH_BUS_WIRE_ENTRY*>( connected_item );
if( !bus_entry->m_connected_bus_item ) { SCH_SCREEN* screen = aSheet.LastScreen(); SCH_LINE* bus = screen->GetBus( it.first );
if( bus ) bus_entry->m_connected_bus_item = bus; } }
return 1; };
thread_pool& tp = GetKiCadThreadPool();
tp.push_loop( connection_vec.size(), [&]( const int a, const int b) { for( int ii = a; ii < b; ++ii ) update_lambda( connection_vec[ii] ); }); tp.wait_for_tasks(); }}
void CONNECTION_GRAPH::buildItemSubGraphs(){ // Recache all bus aliases for later use
wxCHECK_RET( m_schematic, wxS( "Connection graph cannot be built without schematic pointer" ) );
SCH_SCREENS screens( m_schematic->Root() );
for( SCH_SCREEN* screen = screens.GetFirst(); screen; screen = screens.GetNext() ) { for( const std::shared_ptr<BUS_ALIAS>& alias : screen->GetBusAliases() ) m_bus_alias_cache[alias->GetName()] = alias; }
// Build subgraphs from items (on a per-sheet basis)
for( SCH_ITEM* item : m_items ) { for( const auto& it : item->m_connection_map ) { const SCH_SHEET_PATH& sheet = it.first; SCH_CONNECTION* connection = it.second;
if( connection->SubgraphCode() == 0 ) { CONNECTION_SUBGRAPH* subgraph = new CONNECTION_SUBGRAPH( this );
subgraph->m_code = m_last_subgraph_code++; subgraph->m_sheet = sheet;
subgraph->AddItem( item );
connection->SetSubgraphCode( subgraph->m_code ); m_item_to_subgraph_map[item] = subgraph;
std::list<SCH_ITEM*> memberlist;
auto get_items = [&]( SCH_ITEM* aItem ) -> bool { SCH_CONNECTION* conn = aItem->GetOrInitConnection( sheet, this ); bool unique = !( aItem->GetFlags() & CANDIDATE );
if( conn && !conn->SubgraphCode() ) aItem->SetFlags( CANDIDATE );
return ( unique && conn && ( conn->SubgraphCode() == 0 ) ); };
std::copy_if( item->ConnectedItems( sheet ).begin(), item->ConnectedItems( sheet ).end(), std::back_inserter( memberlist ), get_items );
for( SCH_ITEM* connected_item : memberlist ) { if( connected_item->Type() == SCH_NO_CONNECT_T ) subgraph->m_no_connect = connected_item;
SCH_CONNECTION* connected_conn = connected_item->Connection( &sheet );
wxASSERT( connected_conn );
if( connected_conn->SubgraphCode() == 0 ) { connected_conn->SetSubgraphCode( subgraph->m_code ); m_item_to_subgraph_map[connected_item] = subgraph; subgraph->AddItem( connected_item ); const SCH_ITEM_VEC& citemset = connected_item->ConnectedItems( sheet );
for( SCH_ITEM* citem : citemset ) { if( citem->HasFlag( CANDIDATE ) ) continue;
if( get_items( citem ) ) memberlist.push_back( citem ); } } }
for( SCH_ITEM* connected_item : memberlist ) connected_item->ClearFlags( CANDIDATE );
subgraph->m_dirty = true; m_subgraphs.push_back( subgraph ); } } }
}
void CONNECTION_GRAPH::resolveAllDrivers(){ // Resolve drivers for subgraphs and propagate connectivity info
std::vector<CONNECTION_SUBGRAPH*> dirty_graphs;
std::copy_if( m_subgraphs.begin(), m_subgraphs.end(), std::back_inserter( dirty_graphs ), [&] ( const CONNECTION_SUBGRAPH* candidate ) { return candidate->m_dirty; } );
wxLogTrace( ConnTrace, wxT( "Resolving drivers for %zu subgraphs" ), dirty_graphs.size() );
std::vector<std::future<size_t>> returns( dirty_graphs.size() );
auto update_lambda = []( CONNECTION_SUBGRAPH* subgraph ) -> size_t { if( !subgraph->m_dirty ) return 0;
// Special processing for some items
for( SCH_ITEM* item : subgraph->m_items ) { switch( item->Type() ) { case SCH_NO_CONNECT_T: subgraph->m_no_connect = item; break;
case SCH_BUS_WIRE_ENTRY_T: subgraph->m_bus_entry = item; break;
case SCH_PIN_T: { auto pin = static_cast<SCH_PIN*>( item );
if( pin->GetType() == ELECTRICAL_PINTYPE::PT_NC ) subgraph->m_no_connect = item;
break; }
default: break; } }
subgraph->ResolveDrivers( true ); subgraph->m_dirty = false;
return 1; };
thread_pool& tp = GetKiCadThreadPool();
tp.push_loop( dirty_graphs.size(), [&]( const int a, const int b) { for( int ii = a; ii < b; ++ii ) update_lambda( dirty_graphs[ii] ); }); tp.wait_for_tasks();
// Now discard any non-driven subgraphs from further consideration
std::copy_if( m_subgraphs.begin(), m_subgraphs.end(), std::back_inserter( m_driver_subgraphs ), [&] ( const CONNECTION_SUBGRAPH* candidate ) -> bool { return candidate->m_driver; } );}
void CONNECTION_GRAPH::collectAllDriverValues(){ // Check for subgraphs with the same net name but only weak drivers.
// For example, two wires that are both connected to hierarchical
// sheet pins that happen to have the same name, but are not the same.
for( auto&& subgraph : m_driver_subgraphs ) { wxString full_name = subgraph->m_driver_connection->Name(); wxString name = subgraph->m_driver_connection->Name( true ); m_net_name_to_subgraphs_map[full_name].emplace_back( subgraph );
// For vector buses, we need to cache the prefix also, as two different instances of the
// weakly driven pin may have the same prefix but different vector start and end. We need
// to treat those as needing renaming also, because otherwise if they end up on a sheet with
// common usage, they will be incorrectly merged.
if( subgraph->m_driver_connection->Type() == CONNECTION_TYPE::BUS ) { wxString prefixOnly = full_name.BeforeFirst( '[' ) + wxT( "[]" ); m_net_name_to_subgraphs_map[prefixOnly].emplace_back( subgraph ); }
subgraph->m_dirty = true;
if( subgraph->m_strong_driver ) { SCH_ITEM* driver = subgraph->m_driver; SCH_SHEET_PATH sheet = subgraph->m_sheet;
switch( driver->Type() ) { case SCH_LABEL_T: case SCH_HIER_LABEL_T: { m_local_label_cache[std::make_pair( sheet, name )].push_back( subgraph ); break; } case SCH_GLOBAL_LABEL_T: { m_global_label_cache[name].push_back( subgraph ); break; } case SCH_PIN_T: { SCH_PIN* pin = static_cast<SCH_PIN*>( driver ); wxASSERT( pin->IsGlobalPower() ); m_global_label_cache[name].push_back( subgraph ); break; } default: { UNITS_PROVIDER unitsProvider( schIUScale, EDA_UNITS::MILLIMETRES );
wxLogTrace( ConnTrace, wxS( "Unexpected strong driver %s" ), driver->GetItemDescription( &unitsProvider, true ) ); break; } } } }}
void CONNECTION_GRAPH::generateBusAliasMembers(){ std::vector<CONNECTION_SUBGRAPH*> new_subgraphs;
for( CONNECTION_SUBGRAPH* subgraph : m_driver_subgraphs ) { SCH_ITEM_VEC vec = subgraph->GetAllBusLabels();
for( SCH_ITEM* item : vec ) { SCH_LABEL_BASE* label = static_cast<SCH_LABEL_BASE*>( item );
SCH_CONNECTION dummy( item, subgraph->m_sheet ); dummy.SetGraph( this ); dummy.ConfigureFromLabel( label->GetShownText( &subgraph->m_sheet, false ) );
wxLogTrace( ConnTrace, wxS( "new bus label (%s)" ), label->GetShownText( &subgraph->m_sheet, false ) );
for( const auto& conn : dummy.Members() ) { wxString name = conn->FullLocalName();
CONNECTION_SUBGRAPH* new_sg = new CONNECTION_SUBGRAPH( this );
// This connection cannot form a part of the item because the item is not, itself
// connected to this subgraph. It exists as part of a virtual item that may be
// connected to other items but is not in the schematic.
SCH_CONNECTION* new_conn = new SCH_CONNECTION( item, subgraph->m_sheet ); new_conn->SetGraph( this ); new_conn->SetName( name ); new_conn->SetType( CONNECTION_TYPE::NET ); subgraph->StoreImplicitConnection( new_conn ); int code = assignNewNetCode( *new_conn );
wxLogTrace( ConnTrace, wxS( "SG(%ld), Adding full local name (%s) with sg (%d) " "on subsheet %s" ), subgraph->m_code, name, code, subgraph->m_sheet.PathHumanReadable() );
new_sg->m_driver_connection = new_conn; new_sg->m_code = m_last_subgraph_code++; new_sg->m_sheet = subgraph->GetSheet(); new_sg->m_is_bus_member = true; new_sg->m_strong_driver = true;
/// Need to figure out why these sgs are not getting connected to their bus parents
NET_NAME_CODE_CACHE_KEY key = { new_sg->GetNetName(), code }; m_net_code_to_subgraphs_map[ key ].push_back( new_sg ); m_net_name_to_subgraphs_map[ name ].push_back( new_sg ); m_subgraphs.push_back( new_sg ); new_subgraphs.push_back( new_sg ); } } }
std::copy( new_subgraphs.begin(), new_subgraphs.end(), std::back_inserter( m_driver_subgraphs ) );}
void CONNECTION_GRAPH::generateGlobalPowerPinSubGraphs(){ // Generate subgraphs for global power pins. These will be merged with other subgraphs
// on the same sheet in the next loop.
// These are NOT limited to power symbols, we support legacy invisible + power-in pins
// on non-power symbols.
std::unordered_map<int, CONNECTION_SUBGRAPH*> global_power_pin_subgraphs;
for( const auto& it : m_global_power_pins ) { SCH_SHEET_PATH sheet = it.first; SCH_PIN* pin = it.second;
if( !pin->ConnectedItems( sheet ).empty() && !pin->GetLibPin()->GetParentSymbol()->IsPower() ) { // ERC will warn about this: user has wired up an invisible pin
continue; }
SCH_CONNECTION* connection = pin->GetOrInitConnection( sheet, this );
// If this pin already has a subgraph, don't need to process
if( !connection || connection->SubgraphCode() > 0 ) continue;
// Proper modern power symbols get their net name from the value field
// in the symbol, but we support legacy non-power symbols with global
// power connections based on invisible, power-in, pin's names.
if( pin->GetLibPin()->GetParentSymbol()->IsPower() ) connection->SetName( pin->GetParentSymbol()->GetValue( true, &sheet, false ) ); else connection->SetName( pin->GetShownName() );
int code = assignNewNetCode( *connection );
connection->SetNetCode( code );
CONNECTION_SUBGRAPH* subgraph; auto jj = global_power_pin_subgraphs.find( code );
if( jj != global_power_pin_subgraphs.end() ) { subgraph = jj->second; subgraph->AddItem( pin ); } else { subgraph = new CONNECTION_SUBGRAPH( this );
subgraph->m_code = m_last_subgraph_code++; subgraph->m_sheet = sheet;
subgraph->AddItem( pin ); subgraph->ResolveDrivers();
NET_NAME_CODE_CACHE_KEY key = { subgraph->GetNetName(), code }; m_net_code_to_subgraphs_map[ key ].push_back( subgraph ); m_subgraphs.push_back( subgraph ); m_driver_subgraphs.push_back( subgraph );
global_power_pin_subgraphs[code] = subgraph; }
connection->SetSubgraphCode( subgraph->m_code ); }}
void CONNECTION_GRAPH::processSubGraphs(){ // Here we do all the local (sheet) processing of each subgraph, including assigning net
// codes, merging subgraphs together that use label connections, etc.
// Cache remaining valid subgraphs by sheet path
for( CONNECTION_SUBGRAPH* subgraph : m_driver_subgraphs ) m_sheet_to_subgraphs_map[ subgraph->m_sheet ].emplace_back( subgraph );
std::unordered_set<CONNECTION_SUBGRAPH*> invalidated_subgraphs;
for( CONNECTION_SUBGRAPH* subgraph : m_driver_subgraphs ) { if( subgraph->m_absorbed ) continue;
SCH_CONNECTION* connection = subgraph->m_driver_connection; SCH_SHEET_PATH sheet = subgraph->m_sheet; wxString name = connection->Name();
// Test subgraphs with weak drivers for net name conflicts and fix them
unsigned suffix = 1;
auto create_new_name = [&suffix]( SCH_CONNECTION* aConn ) -> wxString { wxString newName; wxString suffixStr = std::to_wstring( suffix );
// For group buses with a prefix, we can add the suffix to the prefix.
// If they don't have a prefix, we force the creation of a prefix so that
// two buses don't get inadvertently shorted together.
if( aConn->Type() == CONNECTION_TYPE::BUS_GROUP ) { wxString prefix = aConn->BusPrefix();
if( prefix.empty() ) prefix = wxT( "BUS" ); // So result will be "BUS_1{...}"
wxString oldName = aConn->Name().AfterFirst( '{' );
newName << prefix << wxT( "_" ) << suffixStr << wxT( "{" ) << oldName;
aConn->ConfigureFromLabel( newName ); } else { newName << aConn->Name() << wxT( "_" ) << suffixStr; aConn->SetSuffix( wxString( wxT( "_" ) ) << suffixStr ); }
suffix++; return newName; };
if( !subgraph->m_strong_driver ) { std::vector<CONNECTION_SUBGRAPH*> vec_empty; std::vector<CONNECTION_SUBGRAPH*>* vec = &vec_empty;
if( m_net_name_to_subgraphs_map.count( name ) ) vec = &m_net_name_to_subgraphs_map.at( name );
// If we are a unique bus vector, check if we aren't actually unique because of another
// subgraph with a similar bus vector
if( vec->size() <= 1 && subgraph->m_driver_connection->Type() == CONNECTION_TYPE::BUS ) { wxString prefixOnly = name.BeforeFirst( '[' ) + wxT( "[]" );
if( m_net_name_to_subgraphs_map.count( prefixOnly ) ) vec = &m_net_name_to_subgraphs_map.at( prefixOnly ); }
if( vec->size() > 1 ) { wxString new_name = create_new_name( connection );
while( m_net_name_to_subgraphs_map.contains( new_name ) ) new_name = create_new_name( connection );
wxLogTrace( ConnTrace, wxS( "%ld (%s) is weakly driven and not unique. Changing to %s." ), subgraph->m_code, name, new_name );
alg::delete_matching( *vec, subgraph );
m_net_name_to_subgraphs_map[new_name].emplace_back( subgraph );
name = new_name; } else if( subgraph->m_driver ) { // If there is no conflict, promote sheet pins to be strong drivers so that they
// will be considered below for propagation/merging.
// It is possible for this to generate a conflict if the sheet pin has the same
// name as a global label on the same sheet, because global merging will then treat
// this subgraph as if it had a matching local label. So, for those cases, we
// don't apply this promotion
if( subgraph->m_driver->Type() == SCH_SHEET_PIN_T ) { bool conflict = false; wxString global_name = connection->Name( true ); auto kk = m_net_name_to_subgraphs_map.find( global_name );
if( kk != m_net_name_to_subgraphs_map.end() ) { // A global will conflict if it is on the same sheet as this subgraph, since
// it would be connected by implicit local label linking
std::vector<CONNECTION_SUBGRAPH*>& candidates = kk->second;
for( const CONNECTION_SUBGRAPH* candidate : candidates ) { if( candidate->m_sheet == sheet ) conflict = true; } }
if( conflict ) { wxLogTrace( ConnTrace, wxS( "%ld (%s) skipped for promotion due to potential " "conflict" ), subgraph->m_code, name ); } else { UNITS_PROVIDER unitsProvider( schIUScale, EDA_UNITS::MILLIMETRES );
wxLogTrace( ConnTrace, wxS( "%ld (%s) weakly driven by unique sheet pin %s, " "promoting" ), subgraph->m_code, name, subgraph->m_driver->GetItemDescription( &unitsProvider, true ) );
subgraph->m_strong_driver = true; } } } }
// Assign net codes
if( connection->IsBus() ) { int code = -1; auto it = m_bus_name_to_code_map.find( name );
if( it != m_bus_name_to_code_map.end() ) { code = it->second; } else { code = m_last_bus_code++; m_bus_name_to_code_map[ name ] = code; }
connection->SetBusCode( code ); assignNetCodesToBus( connection ); } else { assignNewNetCode( *connection ); }
// Reset the flag for the next loop below
subgraph->m_dirty = true;
// Next, we merge together subgraphs that have label connections, and create
// neighbor links for subgraphs that are part of a bus on the same sheet.
// For merging, we consider each possible strong driver.
// If this subgraph doesn't have a strong driver, let's skip it, since there is no
// way it will be merged with anything.
if( !subgraph->m_strong_driver ) continue;
// candidate_subgraphs will contain each valid, non-bus subgraph on the same sheet
// as the subgraph we are considering that has a strong driver.
// Weakly driven subgraphs are not considered since they will never be absorbed or
// form neighbor links.
std::vector<CONNECTION_SUBGRAPH*> candidate_subgraphs; std::copy_if( m_sheet_to_subgraphs_map[ subgraph->m_sheet ].begin(), m_sheet_to_subgraphs_map[ subgraph->m_sheet ].end(), std::back_inserter( candidate_subgraphs ), [&] ( const CONNECTION_SUBGRAPH* candidate ) { return ( !candidate->m_absorbed && candidate->m_strong_driver && candidate != subgraph ); } );
// This is a list of connections on the current subgraph to compare to the
// drivers of each candidate subgraph. If the current subgraph is a bus,
// we should consider each bus member.
std::vector< std::shared_ptr<SCH_CONNECTION> > connections_to_check;
// Also check the main driving connection
connections_to_check.push_back( std::make_shared<SCH_CONNECTION>( *connection ) );
auto add_connections_to_check = [&] ( CONNECTION_SUBGRAPH* aSubgraph ) { for( SCH_ITEM* possible_driver : aSubgraph->m_items ) { if( possible_driver == aSubgraph->m_driver ) continue;
auto c = getDefaultConnection( possible_driver, aSubgraph );
if( c ) { if( c->Type() != aSubgraph->m_driver_connection->Type() ) continue;
if( c->Name( true ) == aSubgraph->m_driver_connection->Name( true ) ) continue;
connections_to_check.push_back( c ); wxLogTrace( ConnTrace, wxS( "%lu (%s): Adding secondary driver %s" ), aSubgraph->m_code, aSubgraph->m_driver_connection->Name( true ), c->Name( true ) ); } } };
// Now add other strong drivers
// The actual connection attached to these items will have been overwritten
// by the chosen driver of the subgraph, so we need to create a dummy connection
add_connections_to_check( subgraph );
std::set<SCH_CONNECTION*> checked_connections;
for( unsigned i = 0; i < connections_to_check.size(); i++ ) { auto member = connections_to_check[i];
// Don't check the same connection twice
if( !checked_connections.insert( member.get() ).second ) continue;
if( member->IsBus() ) { connections_to_check.insert( connections_to_check.end(), member->Members().begin(), member->Members().end() ); }
wxString test_name = member->Name( true );
for( CONNECTION_SUBGRAPH* candidate : candidate_subgraphs ) { if( candidate->m_absorbed || candidate == subgraph ) continue;
bool match = false;
if( candidate->m_driver_connection->Name( true ) == test_name ) { match = true; } else { if( !candidate->m_multiple_drivers ) continue;
for( SCH_ITEM *driver : candidate->m_drivers ) { if( driver == candidate->m_driver ) continue;
// Sheet pins are not candidates for merging
if( driver->Type() == SCH_SHEET_PIN_T ) continue;
if( driver->Type() == SCH_PIN_T ) { auto pin = static_cast<SCH_PIN*>( driver );
if( pin->IsGlobalPower() && pin->GetDefaultNetName( sheet ) == test_name ) { match = true; break; } } else { wxASSERT( driver->Type() == SCH_LABEL_T || driver->Type() == SCH_GLOBAL_LABEL_T || driver->Type() == SCH_HIER_LABEL_T );
if( subgraph->GetNameForDriver( driver ) == test_name ) { match = true; break; } } } }
if( match ) { if( connection->IsBus() && candidate->m_driver_connection->IsNet() ) { wxLogTrace( ConnTrace, wxS( "%lu (%s) has bus child %lu (%s)" ), subgraph->m_code, connection->Name(), candidate->m_code, member->Name() );
subgraph->m_bus_neighbors[member].insert( candidate ); candidate->m_bus_parents[member].insert( subgraph ); } else if( !connection->IsBus() || connection->Type() == candidate->m_driver_connection->Type() ) { wxLogTrace( ConnTrace, wxS( "%lu (%s) absorbs neighbor %lu (%s)" ), subgraph->m_code, connection->Name(), candidate->m_code, candidate->m_driver_connection->Name() );
// Candidate may have other non-chosen drivers we need to follow
add_connections_to_check( candidate );
subgraph->Absorb( candidate ); invalidated_subgraphs.insert( subgraph ); } } } } }
// Update any subgraph that was invalidated above
for( CONNECTION_SUBGRAPH* subgraph : invalidated_subgraphs ) { if( subgraph->m_absorbed ) continue;
if( !subgraph->ResolveDrivers() ) continue;
if( subgraph->m_driver_connection->IsBus() ) assignNetCodesToBus( subgraph->m_driver_connection ); else assignNewNetCode( *subgraph->m_driver_connection );
wxLogTrace( ConnTrace, wxS( "Re-resolving drivers for %lu (%s)" ), subgraph->m_code, subgraph->m_driver_connection->Name() ); }
}
// TODO(JE) This won't give the same subgraph IDs (and eventually net/graph codes)
// to the same subgraph necessarily if it runs over and over again on the same
// sheet. We need:
//
// a) a cache of net/bus codes, like used before
// b) to persist the CONNECTION_GRAPH globally so the cache is persistent,
// c) some way of trying to avoid changing net names. so we should keep track
// of the previous driver of a net, and if it comes down to choosing between
// equally-prioritized drivers, choose the one that already exists as a driver
// on some portion of the items.
void CONNECTION_GRAPH::buildConnectionGraph( std::function<void( SCH_ITEM* )>* aChangedItemHandler, bool aUnconditional ){ // Recache all bus aliases for later use
wxCHECK_RET( m_schematic, wxT( "Connection graph cannot be built without schematic pointer" ) );
SCH_SCREENS screens( m_schematic->Root() );
for( SCH_SCREEN* screen = screens.GetFirst(); screen; screen = screens.GetNext() ) { for( const std::shared_ptr<BUS_ALIAS>& alias : screen->GetBusAliases() ) m_bus_alias_cache[alias->GetName()] = alias; }
PROF_TIMER sub_graph( "buildItemSubGraphs" ); buildItemSubGraphs();
if( wxLog::IsAllowedTraceMask( DanglingProfileMask ) ) sub_graph.Show();
/**
* TODO(JE): Net codes are non-deterministic. Fortunately, they are also not really used for * anything. We should consider removing them entirely and just using net names everywhere. */
resolveAllDrivers();
collectAllDriverValues();
generateGlobalPowerPinSubGraphs();
generateBusAliasMembers();
PROF_TIMER proc_sub_graph( "ProcessSubGraphs" ); processSubGraphs();
if( wxLog::IsAllowedTraceMask( DanglingProfileMask ) ) proc_sub_graph.Show();
// Absorbed subgraphs should no longer be considered
alg::delete_if( m_driver_subgraphs, [&]( const CONNECTION_SUBGRAPH* candidate ) -> bool { return candidate->m_absorbed; } );
// Store global subgraphs for later reference
std::vector<CONNECTION_SUBGRAPH*> global_subgraphs; std::copy_if( m_driver_subgraphs.begin(), m_driver_subgraphs.end(), std::back_inserter( global_subgraphs ), [&] ( const CONNECTION_SUBGRAPH* candidate ) -> bool { return !candidate->m_local_driver; } );
// Recache remaining valid subgraphs by sheet path
m_sheet_to_subgraphs_map.clear();
for( CONNECTION_SUBGRAPH* subgraph : m_driver_subgraphs ) m_sheet_to_subgraphs_map[ subgraph->m_sheet ].emplace_back( subgraph );
thread_pool& tp = GetKiCadThreadPool();
tp.push_loop( m_driver_subgraphs.size(), [&]( const int a, const int b) { for( int ii = a; ii < b; ++ii ) m_driver_subgraphs[ii]->UpdateItemConnections(); }); tp.wait_for_tasks();
// Next time through the subgraphs, we do some post-processing to handle things like
// connecting bus members to their neighboring subgraphs, and then propagate connections
// through the hierarchy
for( CONNECTION_SUBGRAPH* subgraph : m_driver_subgraphs ) { if( !subgraph->m_dirty ) continue;
wxLogTrace( ConnTrace, wxS( "Processing %lu (%s) for propagation" ), subgraph->m_code, subgraph->m_driver_connection->Name() );
// For subgraphs that are driven by a global (power port or label) and have more
// than one global driver, we need to seek out other subgraphs driven by the
// same name as the non-chosen driver and update them to match the chosen one.
if( !subgraph->m_local_driver && subgraph->m_multiple_drivers ) { for( SCH_ITEM* driver : subgraph->m_drivers ) { if( driver == subgraph->m_driver ) continue;
const wxString& secondary_name = subgraph->GetNameForDriver( driver );
if( secondary_name == subgraph->m_driver_connection->Name() ) continue;
bool secondary_is_global = CONNECTION_SUBGRAPH::GetDriverPriority( driver ) >= CONNECTION_SUBGRAPH::PRIORITY::POWER_PIN;
for( CONNECTION_SUBGRAPH* candidate : global_subgraphs ) { if( candidate == subgraph ) continue;
if( !secondary_is_global && candidate->m_sheet != subgraph->m_sheet ) continue;
for( SCH_ITEM* candidate_driver : candidate->m_drivers ) { if( candidate->GetNameForDriver( candidate_driver ) == secondary_name ) { wxLogTrace( ConnTrace, wxS( "Global %lu (%s) promoted to %s" ), candidate->m_code, candidate->m_driver_connection->Name(), subgraph->m_driver_connection->Name() );
candidate->m_driver_connection->Clone( *subgraph->m_driver_connection );
candidate->m_dirty = false; propagateToNeighbors( candidate, false ); } } } } }
// This call will handle descending the hierarchy and updating child subgraphs
propagateToNeighbors( subgraph, false ); }
// After processing and allowing some to be skipped if they have hierarchical
// pins connecting both up and down the hierarchy, we check to see if any of them
// have not been processed. This would indicate that they do not have off-sheet connections
// but we still need to handle the subgraph
for( CONNECTION_SUBGRAPH* subgraph : m_driver_subgraphs ) { if( subgraph->m_dirty ) propagateToNeighbors( subgraph, true ); }
// Handle buses that have been linked together somewhere by member (net) connections.
// This feels a bit hacky, perhaps this algorithm should be revisited in the future.
// For net subgraphs that have more than one bus parent, we need to ensure that those
// buses are linked together in the final netlist. The final name of each bus might not
// match the local name that was used to establish the parent-child relationship, because
// the bus may have been renamed by a hierarchical connection. So, for each of these cases,
// we need to identify the appropriate bus members to link together (and their final names),
// and then update all instances of the old name in the hierarchy.
for( CONNECTION_SUBGRAPH* subgraph : m_driver_subgraphs ) { // All SGs should have been processed by propagateToNeighbors above
wxASSERT_MSG( !subgraph->m_dirty, wxS( "Subgraph not processed by propagateToNeighbors!" ) );
if( subgraph->m_bus_parents.size() < 2 ) continue;
SCH_CONNECTION* conn = subgraph->m_driver_connection;
wxLogTrace( ConnTrace, wxS( "%lu (%s) has multiple bus parents" ), subgraph->m_code, conn->Name() );
wxASSERT( conn->IsNet() );
for( const auto& ii : subgraph->m_bus_parents ) { SCH_CONNECTION* link_member = ii.first.get();
for( CONNECTION_SUBGRAPH* parent : ii.second ) { while( parent->m_absorbed ) parent = parent->m_absorbed_by;
SCH_CONNECTION* match = matchBusMember( parent->m_driver_connection, link_member );
if( !match ) { wxLogTrace( ConnTrace, wxS( "Warning: could not match %s inside %lu (%s)" ), conn->Name(), parent->m_code, parent->m_driver_connection->Name() ); continue; }
if( conn->Name() != match->Name() ) { wxString old_name = match->Name();
wxLogTrace( ConnTrace, wxS( "Updating %lu (%s) member %s to %s" ), parent->m_code, parent->m_driver_connection->Name(), old_name, conn->Name() );
match->Clone( *conn );
auto jj = m_net_name_to_subgraphs_map.find( old_name );
if( jj == m_net_name_to_subgraphs_map.end() ) continue;
for( CONNECTION_SUBGRAPH* old_sg : jj->second ) { while( old_sg->m_absorbed ) old_sg = old_sg->m_absorbed_by;
old_sg->m_driver_connection->Clone( *conn ); } } } } }
auto updateItemConnectionsTask = [&]( CONNECTION_SUBGRAPH* subgraph ) -> size_t { // Make sure weakly-driven single-pin nets get the unconnected_ prefix
if( !subgraph->m_strong_driver && subgraph->m_drivers.size() == 1 && subgraph->m_driver->Type() == SCH_PIN_T ) { SCH_PIN* pin = static_cast<SCH_PIN*>( subgraph->m_driver ); wxString name = pin->GetDefaultNetName( subgraph->m_sheet, true );
subgraph->m_driver_connection->ConfigureFromLabel( name ); }
subgraph->m_dirty = false; subgraph->UpdateItemConnections();
// No other processing to do on buses
if( subgraph->m_driver_connection->IsBus() ) return 0;
// As a visual aid, we can check sheet pins that are driven by themselves to see
// if they should be promoted to buses
if( subgraph->m_driver && subgraph->m_driver->Type() == SCH_SHEET_PIN_T ) { SCH_SHEET_PIN* pin = static_cast<SCH_SHEET_PIN*>( subgraph->m_driver );
if( SCH_SHEET* sheet = pin->GetParent() ) { wxString pinText = pin->GetShownText( false ); SCH_SCREEN* screen = sheet->GetScreen();
for( SCH_ITEM* item : screen->Items().OfType( SCH_HIER_LABEL_T ) ) { SCH_HIERLABEL* label = static_cast<SCH_HIERLABEL*>( item );
if( label->GetShownText( &subgraph->m_sheet, false ) == pinText ) { SCH_SHEET_PATH path = subgraph->m_sheet; path.push_back( sheet );
SCH_CONNECTION* parent_conn = label->Connection( &path );
if( parent_conn && parent_conn->IsBus() ) subgraph->m_driver_connection->SetType( CONNECTION_TYPE::BUS );
break; } }
if( subgraph->m_driver_connection->IsBus() ) return 0; } }
return 1; };
tp.push_loop( m_driver_subgraphs.size(), [&]( const int a, const int b) { for( int ii = a; ii < b; ++ii ) updateItemConnectionsTask( m_driver_subgraphs[ii] ); }); tp.wait_for_tasks();
m_net_code_to_subgraphs_map.clear(); m_net_name_to_subgraphs_map.clear();
for( CONNECTION_SUBGRAPH* subgraph : m_driver_subgraphs ) { NET_NAME_CODE_CACHE_KEY key = { subgraph->GetNetName(), subgraph->m_driver_connection->NetCode() }; m_net_code_to_subgraphs_map[ key ].push_back( subgraph );
m_net_name_to_subgraphs_map[subgraph->m_driver_connection->Name()].push_back( subgraph ); }
std::shared_ptr<NET_SETTINGS>& netSettings = m_schematic->Prj().GetProjectFile().m_NetSettings; std::map<wxString, std::set<wxString>> oldAssignments = netSettings->GetNetclassLabelAssignments(); std::set<wxString> affectedNetclassNetAssignments;
netSettings->ClearNetclassLabelAssignments();
auto dirtySubgraphs = [&]( const std::vector<CONNECTION_SUBGRAPH*>& subgraphs ) { if( aChangedItemHandler ) { for( const CONNECTION_SUBGRAPH* subgraph : subgraphs ) { for( SCH_ITEM* item : subgraph->m_items ) (*aChangedItemHandler)( item ); } } };
auto checkNetclassDrivers = [&]( const wxString& netName, const std::vector<CONNECTION_SUBGRAPH*>& subgraphs ) { wxCHECK_RET( !subgraphs.empty(), wxS( "Invalid empty subgraph" ) );
std::set<wxString> netclasses;
// Collect all netclasses on all subgraphs for this net
for( const CONNECTION_SUBGRAPH* subgraph : subgraphs ) { for( SCH_ITEM* item : subgraph->m_items ) { std::vector<std::pair<wxString, SCH_ITEM*>> netclassesWithProviders = subgraph->GetNetclassesForDriver( item );
for( std::pair<wxString, SCH_ITEM*>& ncPair : netclassesWithProviders ) netclasses.insert( std::move( ncPair.first ) ); } }
// Append the netclasses to any included bus members
for( const CONNECTION_SUBGRAPH* subgraph : subgraphs ) { if( subgraph->m_driver_connection->IsBus() ) { auto processBusMember = [&, this]( const SCH_CONNECTION* member ) { if( !netclasses.empty() ) { netSettings->AppendNetclassLabelAssignment( member->Name(), netclasses ); }
auto ii = m_net_name_to_subgraphs_map.find( member->Name() );
if( oldAssignments.count( member->Name() ) ) { if( oldAssignments[member->Name()] != netclasses ) { affectedNetclassNetAssignments.insert( member->Name() );
if( ii != m_net_name_to_subgraphs_map.end() ) dirtySubgraphs( ii->second ); } } else if( !netclasses.empty() ) { affectedNetclassNetAssignments.insert( member->Name() );
if( ii != m_net_name_to_subgraphs_map.end() ) dirtySubgraphs( ii->second ); } };
for( const std::shared_ptr<SCH_CONNECTION>& member : subgraph->m_driver_connection->Members() ) { // Check if this member itself is a bus (which can be the case
// for vector buses as members of a bus, see
// https://gitlab.com/kicad/code/kicad/-/issues/16545
if( member->IsBus() ) { for( const std::shared_ptr<SCH_CONNECTION>& nestedMember : member->Members() ) { processBusMember( nestedMember.get() ); } } else { processBusMember( member.get() ); } } } }
// Assign the netclasses to the root netname
if( !netclasses.empty() ) { netSettings->AppendNetclassLabelAssignment( netName, netclasses ); }
if( oldAssignments.count( netName ) ) { if( oldAssignments[netName] != netclasses ) { affectedNetclassNetAssignments.insert( netName ); dirtySubgraphs( subgraphs ); } } else if( !netclasses.empty() ) { affectedNetclassNetAssignments.insert( netName ); dirtySubgraphs( subgraphs ); } };
// Check for netclass assignments
for( const auto& [ netname, subgraphs ] : m_net_name_to_subgraphs_map ) checkNetclassDrivers( netname, subgraphs );
if( !aUnconditional ) { for( auto& [netname, netclasses] : oldAssignments ) { if( netSettings->GetNetclassLabelAssignments().count( netname ) || affectedNetclassNetAssignments.count( netname ) ) { continue; }
netSettings->SetNetclassLabelAssignment( netname, netclasses ); } }}
int CONNECTION_GRAPH::getOrCreateNetCode( const wxString& aNetName ){ int code;
auto it = m_net_name_to_code_map.find( aNetName );
if( it == m_net_name_to_code_map.end() ) { code = m_last_net_code++; m_net_name_to_code_map[ aNetName ] = code; } else { code = it->second; }
return code;}
int CONNECTION_GRAPH::assignNewNetCode( SCH_CONNECTION& aConnection ){ int code = getOrCreateNetCode( aConnection.Name() );
aConnection.SetNetCode( code );
return code;}
void CONNECTION_GRAPH::assignNetCodesToBus( SCH_CONNECTION* aConnection ){ std::vector<std::shared_ptr<SCH_CONNECTION>> connections_to_check( aConnection->Members() );
for( unsigned i = 0; i < connections_to_check.size(); i++ ) { const std::shared_ptr<SCH_CONNECTION>& member = connections_to_check[i];
if( member->IsBus() ) { connections_to_check.insert( connections_to_check.end(), member->Members().begin(), member->Members().end() ); continue; }
assignNewNetCode( *member ); }}
void CONNECTION_GRAPH::propagateToNeighbors( CONNECTION_SUBGRAPH* aSubgraph, bool aForce ){ SCH_CONNECTION* conn = aSubgraph->m_driver_connection; std::vector<CONNECTION_SUBGRAPH*> search_list; std::unordered_set<CONNECTION_SUBGRAPH*> visited; std::unordered_set<SCH_CONNECTION*> stale_bus_members;
auto visit =[&]( CONNECTION_SUBGRAPH* aParent ) { for( SCH_SHEET_PIN* pin : aParent->m_hier_pins ) { SCH_SHEET_PATH path = aParent->m_sheet; path.push_back( pin->GetParent() );
auto it = m_sheet_to_subgraphs_map.find( path );
if( it == m_sheet_to_subgraphs_map.end() ) continue;
for( CONNECTION_SUBGRAPH* candidate : it->second ) { if( !candidate->m_strong_driver || candidate->m_hier_ports.empty() || visited.contains( candidate ) ) { continue; }
for( SCH_HIERLABEL* label : candidate->m_hier_ports ) { if( candidate->GetNameForDriver( label ) == aParent->GetNameForDriver( pin ) ) { wxLogTrace( ConnTrace, wxS( "%lu: found child %lu (%s)" ), aParent->m_code, candidate->m_code, candidate->m_driver_connection->Name() );
candidate->m_hier_parent = aParent; aParent->m_hier_children.insert( candidate );
wxASSERT( candidate->m_graph == aParent->m_graph );
search_list.push_back( candidate ); break; } } } }
for( SCH_HIERLABEL* label : aParent->m_hier_ports ) { SCH_SHEET_PATH path = aParent->m_sheet; path.pop_back();
auto it = m_sheet_to_subgraphs_map.find( path );
if( it == m_sheet_to_subgraphs_map.end() ) continue;
for( CONNECTION_SUBGRAPH* candidate : it->second ) { if( candidate->m_hier_pins.empty() || visited.contains( candidate ) || candidate->m_driver_connection->Type() != aParent->m_driver_connection->Type() ) { continue; }
const KIID& last_parent_uuid = aParent->m_sheet.Last()->m_Uuid;
for( SCH_SHEET_PIN* pin : candidate->m_hier_pins ) { // If the last sheet UUIDs won't match, no need to check the full path
if( pin->GetParent()->m_Uuid != last_parent_uuid ) continue;
SCH_SHEET_PATH pin_path = path; pin_path.push_back( pin->GetParent() );
if( pin_path != aParent->m_sheet ) continue;
if( aParent->GetNameForDriver( label ) == candidate->GetNameForDriver( pin ) ) { wxLogTrace( ConnTrace, wxS( "%lu: found additional parent %lu (%s)" ), aParent->m_code, candidate->m_code, candidate->m_driver_connection->Name() );
aParent->m_hier_children.insert( candidate ); search_list.push_back( candidate ); break; } } } } };
auto propagate_bus_neighbors = [&]( CONNECTION_SUBGRAPH* aParentGraph ) { for( const auto& kv : aParentGraph->m_bus_neighbors ) { for( CONNECTION_SUBGRAPH* neighbor : kv.second ) { // May have been absorbed but won't have been deleted
while( neighbor->m_absorbed ) neighbor = neighbor->m_absorbed_by;
SCH_CONNECTION* parent = aParentGraph->m_driver_connection;
// Now member may be out of date, since we just cloned the
// connection from higher up in the hierarchy. We need to
// figure out what the actual new connection is.
SCH_CONNECTION* member = matchBusMember( parent, kv.first.get() );
if( !member ) { // Try harder: we might match on a secondary driver
for( CONNECTION_SUBGRAPH* sg : kv.second ) { if( sg->m_multiple_drivers ) { SCH_SHEET_PATH sheet = sg->m_sheet;
for( SCH_ITEM* driver : sg->m_drivers ) { auto c = getDefaultConnection( driver, sg ); member = matchBusMember( parent, c.get() );
if( member ) break; } }
if( member ) break; } }
// This is bad, probably an ERC error
if( !member ) { wxLogTrace( ConnTrace, wxS( "Could not match bus member %s in %s" ), kv.first->Name(), parent->Name() ); continue; }
auto neighbor_conn = neighbor->m_driver_connection; auto neighbor_name = neighbor_conn->Name();
// Matching name: no update needed
if( neighbor_name == member->Name() ) continue;
// Was this neighbor already updated from a different sheet? Don't rename it again
if( neighbor_conn->Sheet() != neighbor->m_sheet ) continue;
// Safety check against infinite recursion
wxASSERT( neighbor_conn->IsNet() );
wxLogTrace( ConnTrace, wxS( "%lu (%s) connected to bus member %s (local %s)" ), neighbor->m_code, neighbor_name, member->Name(), member->LocalName() );
// Take whichever name is higher priority
if( CONNECTION_SUBGRAPH::GetDriverPriority( neighbor->m_driver ) >= CONNECTION_SUBGRAPH::PRIORITY::POWER_PIN ) { member->Clone( *neighbor_conn ); stale_bus_members.insert( member ); } else { neighbor_conn->Clone( *member );
recacheSubgraphName( neighbor, neighbor_name );
// Recurse onto this neighbor in case it needs to re-propagate
neighbor->m_dirty = true; propagateToNeighbors( neighbor, aForce ); } } } };
// If we are a bus, we must propagate to local neighbors and then the hierarchy
if( conn->IsBus() ) propagate_bus_neighbors( aSubgraph );
// If we have both ports and pins, skip processing as we'll be visited by a parent or child.
// If we only have one or the other, process (we can either go bottom-up or top-down depending
// on which subgraph comes up first)
if( !aForce && !aSubgraph->m_hier_ports.empty() && !aSubgraph->m_hier_pins.empty() ) { wxLogTrace( ConnTrace, wxS( "%lu (%s) has both hier ports and pins; deferring processing" ), aSubgraph->m_code, conn->Name() ); return; } else if( aSubgraph->m_hier_ports.empty() && aSubgraph->m_hier_pins.empty() ) { wxLogTrace( ConnTrace, wxS( "%lu (%s) has no hier pins or ports on sheet %s; marking clean" ), aSubgraph->m_code, conn->Name(), aSubgraph->m_sheet.PathHumanReadable() ); aSubgraph->m_dirty = false; return; }
visited.insert( aSubgraph );
wxLogTrace( ConnTrace, wxS( "Propagating %lu (%s) to subsheets" ), aSubgraph->m_code, aSubgraph->m_driver_connection->Name() );
visit( aSubgraph );
for( unsigned i = 0; i < search_list.size(); i++ ) { auto child = search_list[i];
if( visited.insert( child ).second ) visit( child );
child->m_dirty = false; }
// Now, find the best driver for this chain of subgraphs
CONNECTION_SUBGRAPH* bestDriver = aSubgraph; CONNECTION_SUBGRAPH::PRIORITY highest = CONNECTION_SUBGRAPH::GetDriverPriority( aSubgraph->m_driver ); bool bestIsStrong = ( highest >= CONNECTION_SUBGRAPH::PRIORITY::HIER_LABEL ); wxString bestName = aSubgraph->m_driver_connection->Name();
// Check if a subsheet has a higher-priority connection to the same net
if( highest < CONNECTION_SUBGRAPH::PRIORITY::POWER_PIN ) { for( CONNECTION_SUBGRAPH* subgraph : visited ) { if( subgraph == aSubgraph ) continue;
CONNECTION_SUBGRAPH::PRIORITY priority = CONNECTION_SUBGRAPH::GetDriverPriority( subgraph->m_driver );
bool candidateStrong = ( priority >= CONNECTION_SUBGRAPH::PRIORITY::HIER_LABEL ); wxString candidateName = subgraph->m_driver_connection->Name(); bool shorterPath = subgraph->m_sheet.size() < bestDriver->m_sheet.size(); bool asGoodPath = subgraph->m_sheet.size() <= bestDriver->m_sheet.size();
// Pick a better driving subgraph if it:
// a) has a power pin or global driver
// b) is a strong driver and we're a weak driver
// c) is a higher priority strong driver
// d) matches our priority, is a strong driver, and has a shorter path
// e) matches our strength and is at least as short, and is alphabetically lower
if( ( priority >= CONNECTION_SUBGRAPH::PRIORITY::POWER_PIN ) || ( !bestIsStrong && candidateStrong ) || ( priority > highest && candidateStrong ) || ( priority == highest && candidateStrong && shorterPath ) || ( ( bestIsStrong == candidateStrong ) && asGoodPath && ( priority == highest ) && ( candidateName < bestName ) ) ) { bestDriver = subgraph; highest = priority; bestIsStrong = candidateStrong; bestName = candidateName; } } }
if( bestDriver != aSubgraph ) { wxLogTrace( ConnTrace, wxS( "%lu (%s) overridden by new driver %lu (%s)" ), aSubgraph->m_code, aSubgraph->m_driver_connection->Name(), bestDriver->m_code, bestDriver->m_driver_connection->Name() ); }
conn = bestDriver->m_driver_connection;
for( CONNECTION_SUBGRAPH* subgraph : visited ) { wxString old_name = subgraph->m_driver_connection->Name();
subgraph->m_driver_connection->Clone( *conn );
if( old_name != conn->Name() ) recacheSubgraphName( subgraph, old_name );
if( conn->IsBus() ) propagate_bus_neighbors( subgraph ); }
// Somewhere along the way, a bus member may have been upgraded to a global or power label.
// Because this can happen anywhere, we need a second pass to update all instances of that bus
// member to have the correct connection info
if( conn->IsBus() && !stale_bus_members.empty() ) { std::unordered_set<SCH_CONNECTION*> cached_members = stale_bus_members;
for( SCH_CONNECTION* stale_member : cached_members ) { for( CONNECTION_SUBGRAPH* subgraph : visited ) { SCH_CONNECTION* member = matchBusMember( subgraph->m_driver_connection, stale_member );
if( !member ) { wxLogTrace( ConnTrace, wxS( "WARNING: failed to match stale member %s in %s." ), stale_member->Name(), subgraph->m_driver_connection->Name() ); continue; }
wxLogTrace( ConnTrace, wxS( "Updating %lu (%s) member %s to %s" ), subgraph->m_code, subgraph->m_driver_connection->Name(), member->LocalName(), stale_member->Name() );
member->Clone( *stale_member );
propagate_bus_neighbors( subgraph ); } } }
aSubgraph->m_dirty = false;}
std::shared_ptr<SCH_CONNECTION> CONNECTION_GRAPH::getDefaultConnection( SCH_ITEM* aItem, CONNECTION_SUBGRAPH* aSubgraph ){ std::shared_ptr<SCH_CONNECTION> c = std::shared_ptr<SCH_CONNECTION>( nullptr );
switch( aItem->Type() ) { case SCH_PIN_T: { SCH_PIN* pin = static_cast<SCH_PIN*>( aItem );
if( pin->IsGlobalPower() ) c = std::make_shared<SCH_CONNECTION>( aItem, aSubgraph->m_sheet );
break; }
case SCH_GLOBAL_LABEL_T: case SCH_HIER_LABEL_T: case SCH_LABEL_T: { c = std::make_shared<SCH_CONNECTION>( aItem, aSubgraph->m_sheet ); break; }
default: break; }
if( c ) { c->SetGraph( this ); c->ConfigureFromLabel( aSubgraph->GetNameForDriver( aItem ) ); }
return c;}
SCH_CONNECTION* CONNECTION_GRAPH::matchBusMember( SCH_CONNECTION* aBusConnection, SCH_CONNECTION* aSearch ){ wxASSERT( aBusConnection->IsBus() );
SCH_CONNECTION* match = nullptr;
if( aBusConnection->Type() == CONNECTION_TYPE::BUS ) { // Vector bus: compare against index, because we allow the name
// to be different
for( const std::shared_ptr<SCH_CONNECTION>& bus_member : aBusConnection->Members() ) { if( bus_member->VectorIndex() == aSearch->VectorIndex() ) { match = bus_member.get(); break; } } } else { // Group bus
for( const std::shared_ptr<SCH_CONNECTION>& c : aBusConnection->Members() ) { // Vector inside group: compare names, because for bus groups
// we expect the naming to be consistent across all usages
// TODO(JE) explain this in the docs
if( c->Type() == CONNECTION_TYPE::BUS ) { for( const std::shared_ptr<SCH_CONNECTION>& bus_member : c->Members() ) { if( bus_member->LocalName() == aSearch->LocalName() ) { match = bus_member.get(); break; } } } else if( c->LocalName() == aSearch->LocalName() ) { match = c.get(); break; } } }
return match;}
void CONNECTION_GRAPH::recacheSubgraphName( CONNECTION_SUBGRAPH* aSubgraph, const wxString& aOldName ){ auto it = m_net_name_to_subgraphs_map.find( aOldName );
if( it != m_net_name_to_subgraphs_map.end() ) { std::vector<CONNECTION_SUBGRAPH*>& vec = it->second; alg::delete_matching( vec, aSubgraph ); }
wxLogTrace( ConnTrace, wxS( "recacheSubgraphName: %s => %s" ), aOldName, aSubgraph->m_driver_connection->Name() );
m_net_name_to_subgraphs_map[aSubgraph->m_driver_connection->Name()].push_back( aSubgraph );}
std::shared_ptr<BUS_ALIAS> CONNECTION_GRAPH::GetBusAlias( const wxString& aName ){ auto it = m_bus_alias_cache.find( aName );
return it != m_bus_alias_cache.end() ? it->second : nullptr;}
std::vector<const CONNECTION_SUBGRAPH*> CONNECTION_GRAPH::GetBusesNeedingMigration(){ std::vector<const CONNECTION_SUBGRAPH*> ret;
for( CONNECTION_SUBGRAPH* subgraph : m_subgraphs ) { // Graph is supposed to be up-to-date before calling this
wxASSERT( !subgraph->m_dirty );
if( !subgraph->m_driver ) continue;
SCH_SHEET_PATH* sheet = &subgraph->m_sheet; SCH_CONNECTION* connection = subgraph->m_driver->Connection( sheet );
if( !connection->IsBus() ) continue;
auto labels = subgraph->GetVectorBusLabels();
if( labels.size() > 1 ) { bool different = false; wxString first = static_cast<SCH_TEXT*>( labels.at( 0 ) )->GetShownText( sheet, false );
for( unsigned i = 1; i < labels.size(); ++i ) { if( static_cast<SCH_TEXT*>( labels.at( i ) )->GetShownText( sheet, false ) != first ) { different = true; break; } }
if( !different ) continue;
wxLogTrace( ConnTrace, wxS( "SG %ld (%s) has multiple bus labels" ), subgraph->m_code, connection->Name() );
ret.push_back( subgraph ); } }
return ret;}
wxString CONNECTION_GRAPH::GetResolvedSubgraphName( const CONNECTION_SUBGRAPH* aSubGraph ) const{ wxString retval = aSubGraph->GetNetName(); bool found = false;
// This is a hacky way to find the true subgraph net name (why do we not store it?)
// TODO: Remove once the actual netname of the subgraph is stored with the subgraph
for( auto it = m_net_name_to_subgraphs_map.begin(); it != m_net_name_to_subgraphs_map.end() && !found; ++it ) { for( CONNECTION_SUBGRAPH* graph : it->second ) { if( graph == aSubGraph ) { retval = it->first; found = true; break; } } }
return retval;}
CONNECTION_SUBGRAPH* CONNECTION_GRAPH::FindSubgraphByName( const wxString& aNetName, const SCH_SHEET_PATH& aPath ){ auto it = m_net_name_to_subgraphs_map.find( aNetName );
if( it == m_net_name_to_subgraphs_map.end() ) return nullptr;
for( CONNECTION_SUBGRAPH* sg : it->second ) { // Cache is supposed to be valid by now
wxASSERT( sg && !sg->m_absorbed && sg->m_driver_connection );
if( sg->m_sheet == aPath && sg->m_driver_connection->Name() == aNetName ) return sg; }
return nullptr;}
CONNECTION_SUBGRAPH* CONNECTION_GRAPH::FindFirstSubgraphByName( const wxString& aNetName ){ auto it = m_net_name_to_subgraphs_map.find( aNetName );
if( it == m_net_name_to_subgraphs_map.end() ) return nullptr;
wxASSERT( !it->second.empty() );
return it->second[0];}
CONNECTION_SUBGRAPH* CONNECTION_GRAPH::GetSubgraphForItem( SCH_ITEM* aItem ) const{ auto it = m_item_to_subgraph_map.find( aItem ); CONNECTION_SUBGRAPH* ret = it != m_item_to_subgraph_map.end() ? it->second : nullptr;
while( ret && ret->m_absorbed ) ret = ret->m_absorbed_by;
return ret;}
const std::vector<CONNECTION_SUBGRAPH*>&CONNECTION_GRAPH::GetAllSubgraphs( const wxString& aNetName ) const{ static const std::vector<CONNECTION_SUBGRAPH*> subgraphs;
auto it = m_net_name_to_subgraphs_map.find( aNetName );
if( it == m_net_name_to_subgraphs_map.end() ) return subgraphs;
return it->second;}
int CONNECTION_GRAPH::RunERC(){ int error_count = 0;
wxCHECK_MSG( m_schematic, true, wxS( "Null m_schematic in CONNECTION_GRAPH::RunERC" ) );
ERC_SETTINGS& settings = m_schematic->ErcSettings();
// We don't want to run many ERC checks more than once on a given screen even though it may
// represent multiple sheets with multiple subgraphs. We can tell these apart by drivers.
std::set<SCH_ITEM*> seenDriverInstances;
for( CONNECTION_SUBGRAPH* subgraph : m_subgraphs ) { // There shouldn't be any null sub-graph pointers.
wxCHECK2( subgraph, continue );
// Graph is supposed to be up-to-date before calling RunERC()
wxASSERT( !subgraph->m_dirty );
if( subgraph->m_absorbed ) continue;
if( seenDriverInstances.count( subgraph->m_driver ) ) continue;
if( subgraph->m_driver ) seenDriverInstances.insert( subgraph->m_driver );
/**
* NOTE: * * We could check that labels attached to bus subgraphs follow the * proper format (i.e. actually define a bus). * * This check doesn't need to be here right now because labels * won't actually be connected to bus wires if they aren't in the right * format due to their TestDanglingEnds() implementation. */ if( settings.IsTestEnabled( ERCE_DRIVER_CONFLICT ) ) { if( !ercCheckMultipleDrivers( subgraph ) ) error_count++; }
subgraph->ResolveDrivers( false );
if( settings.IsTestEnabled( ERCE_BUS_TO_NET_CONFLICT ) ) { if( !ercCheckBusToNetConflicts( subgraph ) ) error_count++; }
if( settings.IsTestEnabled( ERCE_BUS_ENTRY_CONFLICT ) ) { if( !ercCheckBusToBusEntryConflicts( subgraph ) ) error_count++; }
if( settings.IsTestEnabled( ERCE_BUS_TO_BUS_CONFLICT ) ) { if( !ercCheckBusToBusConflicts( subgraph ) ) error_count++; }
if( settings.IsTestEnabled( ERCE_WIRE_DANGLING ) ) { if( !ercCheckFloatingWires( subgraph ) ) error_count++; }
if( settings.IsTestEnabled( ERCE_UNCONNECTED_WIRE_ENDPOINT ) ) { if( !ercCheckDanglingWireEndpoints( subgraph ) ) error_count++; }
if( settings.IsTestEnabled( ERCE_NOCONNECT_CONNECTED ) || settings.IsTestEnabled( ERCE_NOCONNECT_NOT_CONNECTED ) || settings.IsTestEnabled( ERCE_PIN_NOT_CONNECTED ) ) { if( !ercCheckNoConnects( subgraph ) ) error_count++; }
if( settings.IsTestEnabled( ERCE_LABEL_NOT_CONNECTED ) || settings.IsTestEnabled( ERCE_GLOBLABEL_DANGLING ) ) { if( !ercCheckLabels( subgraph ) ) error_count++; } }
if( settings.IsTestEnabled( ERCE_LABEL_NOT_CONNECTED ) ) { error_count += ercCheckDirectiveLabels(); }
// Hierarchical sheet checking is done at the schematic level
if( settings.IsTestEnabled( ERCE_HIERACHICAL_LABEL ) || settings.IsTestEnabled( ERCE_PIN_NOT_CONNECTED ) ) { error_count += ercCheckHierSheets(); }
if( settings.IsTestEnabled( ERCE_SINGLE_GLOBAL_LABEL ) ) { error_count += ercCheckSingleGlobalLabel(); }
return error_count;}
bool CONNECTION_GRAPH::ercCheckMultipleDrivers( const CONNECTION_SUBGRAPH* aSubgraph ){ wxCHECK( aSubgraph, false );
if( aSubgraph->m_multiple_drivers ) { for( SCH_ITEM* driver : aSubgraph->m_drivers ) { if( driver == aSubgraph->m_driver ) continue;
if( driver->Type() == SCH_GLOBAL_LABEL_T || driver->Type() == SCH_HIER_LABEL_T || driver->Type() == SCH_LABEL_T || ( driver->Type() == SCH_PIN_T && static_cast<SCH_PIN*>( driver )->IsGlobalPower() ) ) { const wxString& primaryName = aSubgraph->GetNameForDriver( aSubgraph->m_driver ); const wxString& secondaryName = aSubgraph->GetNameForDriver( driver );
if( primaryName == secondaryName ) continue;
wxString msg = wxString::Format( _( "Both %s and %s are attached to the same " "items; %s will be used in the netlist" ), primaryName, secondaryName, primaryName );
std::shared_ptr<ERC_ITEM> ercItem = ERC_ITEM::Create( ERCE_DRIVER_CONFLICT ); ercItem->SetItems( aSubgraph->m_driver, driver ); ercItem->SetSheetSpecificPath( aSubgraph->GetSheet() ); ercItem->SetItemsSheetPaths( aSubgraph->GetSheet(), aSubgraph->m_sheet ); ercItem->SetErrorMessage( msg );
SCH_MARKER* marker = new SCH_MARKER( ercItem, driver->GetPosition() ); aSubgraph->m_sheet.LastScreen()->Append( marker );
return false; } } }
return true;}
bool CONNECTION_GRAPH::ercCheckBusToNetConflicts( const CONNECTION_SUBGRAPH* aSubgraph ){ const SCH_SHEET_PATH& sheet = aSubgraph->m_sheet; SCH_SCREEN* screen = sheet.LastScreen();
SCH_ITEM* net_item = nullptr; SCH_ITEM* bus_item = nullptr; SCH_CONNECTION conn( this );
for( SCH_ITEM* item : aSubgraph->m_items ) { switch( item->Type() ) { case SCH_LINE_T: { if( item->GetLayer() == LAYER_BUS ) bus_item = ( !bus_item ) ? item : bus_item; else net_item = ( !net_item ) ? item : net_item;
break; }
case SCH_LABEL_T: case SCH_GLOBAL_LABEL_T: case SCH_SHEET_PIN_T: case SCH_HIER_LABEL_T: { SCH_TEXT* text = static_cast<SCH_TEXT*>( item ); conn.ConfigureFromLabel( EscapeString( text->GetShownText( &sheet, false ), CTX_NETNAME ) );
if( conn.IsBus() ) bus_item = ( !bus_item ) ? item : bus_item; else net_item = ( !net_item ) ? item : net_item;
break; }
default: break; } }
if( net_item && bus_item ) { std::shared_ptr<ERC_ITEM> ercItem = ERC_ITEM::Create( ERCE_BUS_TO_NET_CONFLICT ); ercItem->SetSheetSpecificPath( sheet ); ercItem->SetItems( net_item, bus_item );
SCH_MARKER* marker = new SCH_MARKER( ercItem, net_item->GetPosition() ); screen->Append( marker );
return false; }
return true;}
bool CONNECTION_GRAPH::ercCheckBusToBusConflicts( const CONNECTION_SUBGRAPH* aSubgraph ){ const SCH_SHEET_PATH& sheet = aSubgraph->m_sheet; SCH_SCREEN* screen = sheet.LastScreen();
SCH_ITEM* label = nullptr; SCH_ITEM* port = nullptr;
for( SCH_ITEM* item : aSubgraph->m_items ) { switch( item->Type() ) { case SCH_TEXT_T: case SCH_GLOBAL_LABEL_T: { if( !label && item->Connection( &sheet )->IsBus() ) label = item; break; }
case SCH_SHEET_PIN_T: case SCH_HIER_LABEL_T: { if( !port && item->Connection( &sheet )->IsBus() ) port = item; break; }
default: break; } }
if( label && port ) { bool match = false;
for( const auto& member : label->Connection( &sheet )->Members() ) { for( const auto& test : port->Connection( &sheet )->Members() ) { if( test != member && member->Name() == test->Name() ) { match = true; break; } }
if( match ) break; }
if( !match ) { std::shared_ptr<ERC_ITEM> ercItem = ERC_ITEM::Create( ERCE_BUS_TO_BUS_CONFLICT ); ercItem->SetSheetSpecificPath( sheet ); ercItem->SetItems( label, port );
SCH_MARKER* marker = new SCH_MARKER( ercItem, label->GetPosition() ); screen->Append( marker );
return false; } }
return true;}
bool CONNECTION_GRAPH::ercCheckBusToBusEntryConflicts( const CONNECTION_SUBGRAPH* aSubgraph ){ bool conflict = false; const SCH_SHEET_PATH& sheet = aSubgraph->m_sheet; SCH_SCREEN* screen = sheet.LastScreen();
SCH_BUS_WIRE_ENTRY* bus_entry = nullptr; SCH_ITEM* bus_wire = nullptr; wxString bus_name;
if( !aSubgraph->m_driver_connection ) { // Incomplete bus entry. Let the unconnected tests handle it.
return true; }
for( SCH_ITEM* item : aSubgraph->m_items ) { switch( item->Type() ) { case SCH_BUS_WIRE_ENTRY_T: { if( !bus_entry ) bus_entry = static_cast<SCH_BUS_WIRE_ENTRY*>( item ); break; }
default: break; } }
if( bus_entry && bus_entry->m_connected_bus_item ) { bus_wire = bus_entry->m_connected_bus_item;
wxASSERT( bus_wire->Type() == SCH_LINE_T );
// In some cases, the connection list (SCH_CONNECTION*) can be null.
// Skip null connections.
if( bus_entry->Connection( &sheet ) && bus_wire->Type() == SCH_LINE_T && bus_wire->Connection( &sheet ) ) { conflict = true; // Assume a conflict; we'll reset if we find it's OK
bus_name = bus_wire->Connection( &sheet )->Name();
std::set<wxString> test_names; test_names.insert( bus_entry->Connection( &sheet )->FullLocalName() );
wxString baseName = sheet.PathHumanReadable();
for( SCH_ITEM* driver : aSubgraph->m_drivers ) test_names.insert( baseName + aSubgraph->GetNameForDriver( driver ) );
for( const auto& member : bus_wire->Connection( &sheet )->Members() ) { if( member->Type() == CONNECTION_TYPE::BUS ) { for( const auto& sub_member : member->Members() ) { if( test_names.count( sub_member->FullLocalName() ) ) conflict = false; } } else if( test_names.count( member->FullLocalName() ) ) { conflict = false; } } } }
// Don't report warnings if this bus member has been overridden by a higher priority power pin
// or global label
if( conflict && CONNECTION_SUBGRAPH::GetDriverPriority( aSubgraph->m_driver ) >= CONNECTION_SUBGRAPH::PRIORITY::POWER_PIN ) { conflict = false; }
if( conflict ) { wxString netName = aSubgraph->m_driver_connection->Name(); wxString msg = wxString::Format( _( "Net %s is graphically connected to bus %s but is not a" " member of that bus" ), UnescapeString( netName ), UnescapeString( bus_name ) ); std::shared_ptr<ERC_ITEM> ercItem = ERC_ITEM::Create( ERCE_BUS_ENTRY_CONFLICT ); ercItem->SetSheetSpecificPath( sheet ); ercItem->SetItems( bus_entry, bus_wire ); ercItem->SetErrorMessage( msg );
SCH_MARKER* marker = new SCH_MARKER( ercItem, bus_entry->GetPosition() ); screen->Append( marker );
return false; }
return true;}
bool CONNECTION_GRAPH::ercCheckNoConnects( const CONNECTION_SUBGRAPH* aSubgraph ){ ERC_SETTINGS& settings = m_schematic->ErcSettings(); const SCH_SHEET_PATH& sheet = aSubgraph->m_sheet; SCH_SCREEN* screen = sheet.LastScreen(); bool ok = true; SCH_PIN* pin = nullptr;
std::set<SCH_PIN*> unique_pins; std::set<SCH_LABEL_BASE*> unique_labels;
wxString netName = GetResolvedSubgraphName( aSubgraph );
auto process_subgraph = [&]( const CONNECTION_SUBGRAPH* aProcessGraph ) { // Any subgraph that contains a no-connect should not
// more than one pin (which would indicate it is connected
for( SCH_ITEM* item : aProcessGraph->m_items ) { switch( item->Type() ) { case SCH_PIN_T: { SCH_PIN* test_pin = static_cast<SCH_PIN*>( item );
// Only link NC to pin on the current subgraph being checked
if( aProcessGraph == aSubgraph ) pin = test_pin;
if( std::none_of( unique_pins.begin(), unique_pins.end(), [test_pin]( SCH_PIN* aPin ) { return test_pin->IsStacked( aPin ); } )) { unique_pins.insert( test_pin ); }
break; }
case SCH_LABEL_T: case SCH_GLOBAL_LABEL_T: case SCH_HIER_LABEL_T: unique_labels.insert( static_cast<SCH_LABEL_BASE*>( item ) ); KI_FALLTHROUGH; default: break; } } };
auto it = m_net_name_to_subgraphs_map.find( netName );
if( it != m_net_name_to_subgraphs_map.end() ) { for( const CONNECTION_SUBGRAPH* subgraph : it->second ) { process_subgraph( subgraph ); } } else { process_subgraph( aSubgraph ); }
if( aSubgraph->m_no_connect != nullptr ) { // Special case: If the subgraph being checked consists of only a hier port/pin and
// a no-connect, we don't issue a "no-connect connected" warning just because
// connections exist on the sheet on the other side of the link.
VECTOR2I noConnectPos = aSubgraph->m_no_connect->GetPosition();
for( SCH_SHEET_PIN* hierPin : aSubgraph->m_hier_pins ) { if( hierPin->GetPosition() == noConnectPos ) return true; }
for( SCH_HIERLABEL* hierLabel : aSubgraph->m_hier_ports ) { if( hierLabel->GetPosition() == noConnectPos ) return true; }
if( unique_pins.size() > 1 && settings.IsTestEnabled( ERCE_NOCONNECT_CONNECTED ) ) { std::shared_ptr<ERC_ITEM> ercItem = ERC_ITEM::Create( ERCE_NOCONNECT_CONNECTED ); ercItem->SetSheetSpecificPath( sheet ); ercItem->SetItemsSheetPaths( sheet );
VECTOR2I pos;
if( pin ) { ercItem->SetItems( pin, aSubgraph->m_no_connect ); pos = pin->GetPosition(); } else { ercItem->SetItems( aSubgraph->m_no_connect ); pos = aSubgraph->m_no_connect->GetPosition(); }
SCH_MARKER* marker = new SCH_MARKER( ercItem, pos ); screen->Append( marker );
ok = false; }
if( unique_pins.empty() && unique_labels.empty() && settings.IsTestEnabled( ERCE_NOCONNECT_NOT_CONNECTED ) ) { std::shared_ptr<ERC_ITEM> ercItem = ERC_ITEM::Create( ERCE_NOCONNECT_NOT_CONNECTED ); ercItem->SetItems( aSubgraph->m_no_connect ); ercItem->SetSheetSpecificPath( sheet ); ercItem->SetItemsSheetPaths( sheet );
SCH_MARKER* marker = new SCH_MARKER( ercItem, aSubgraph->m_no_connect->GetPosition() ); screen->Append( marker );
ok = false; } } else { bool has_other_connections = false; std::vector<SCH_PIN*> pins;
// Any subgraph that lacks a no-connect and contains a pin should also
// contain at least one other potential driver
for( SCH_ITEM* item : aSubgraph->m_items ) { switch( item->Type() ) { case SCH_PIN_T: { SCH_PIN* test_pin = static_cast<SCH_PIN*>( item );
// Stacked pins do not count as other connections but non-stacked pins do
if( !has_other_connections && !pins.empty() && !test_pin->GetParentSymbol()->IsPower() ) { for( SCH_PIN* other_pin : pins ) { if( !test_pin->IsStacked( other_pin ) ) { has_other_connections = true; break; } } }
pins.emplace_back( static_cast<SCH_PIN*>( item ) );
break; }
default: if( aSubgraph->GetDriverPriority( item ) != CONNECTION_SUBGRAPH::PRIORITY::NONE ) has_other_connections = true;
break; } }
// For many checks, we can just use the first pin
pin = pins.empty() ? nullptr : pins[0];
// But if there is a power pin, it might be connected elsewhere
for( SCH_PIN* test_pin : pins ) { // Prefer the pin is part of a real component rather than some stray power symbol
// Or else we may fail walking connected components to a power symbol pin since we reject
// starting at a power symbol
if( test_pin->GetType() == ELECTRICAL_PINTYPE::PT_POWER_IN && !test_pin->IsGlobalPower() ) { pin = test_pin; break; } }
// Check if power input pins connect to anything else via net name,
// but not for power symbols (with visible or legacy invisible pins).
// We want to throw unconnected errors for power symbols even if they are connected to other
// net items by name, because usually failing to connect them graphically is a mistake
if( pin && !has_other_connections && !pin->IsGlobalPower() && !pin->GetLibPin()->GetParentSymbol()->IsPower() ) { wxString name = pin->Connection( &sheet )->Name(); wxString local_name = pin->Connection( &sheet )->Name( true );
if( m_global_label_cache.count( name ) || m_local_label_cache.count( std::make_pair( sheet, local_name ) ) ) { has_other_connections = true; } }
// Only one pin, and it's not a no-connect pin
if( pin && !has_other_connections && pin->GetType() != ELECTRICAL_PINTYPE::PT_NC && pin->GetType() != ELECTRICAL_PINTYPE::PT_NIC && settings.IsTestEnabled( ERCE_PIN_NOT_CONNECTED ) ) { std::shared_ptr<ERC_ITEM> ercItem = ERC_ITEM::Create( ERCE_PIN_NOT_CONNECTED ); ercItem->SetSheetSpecificPath( sheet ); ercItem->SetItemsSheetPaths( sheet ); ercItem->SetItems( pin );
SCH_MARKER* marker = new SCH_MARKER( ercItem, pin->GetPosition() ); screen->Append( marker );
ok = false; }
// If there are multiple pins in this SG, they might be indirectly connected (by netname)
// rather than directly connected (by wires). We want to flag dangling pins even if they
// join nets with another pin, as it's often a mistake
if( pins.size() > 1 ) { for( SCH_PIN* testPin : pins ) { // We only apply this test to power symbols, because other symbols have
// pins that are meant to be dangling, but the power symbols have pins
// that are *not* meant to be dangling.
if( testPin->GetLibPin()->GetParentSymbol()->IsPower() && testPin->ConnectedItems( sheet ).empty() && settings.IsTestEnabled( ERCE_PIN_NOT_CONNECTED ) ) { std::shared_ptr<ERC_ITEM> ercItem = ERC_ITEM::Create( ERCE_PIN_NOT_CONNECTED ); ercItem->SetSheetSpecificPath( sheet ); ercItem->SetItemsSheetPaths( sheet ); ercItem->SetItems( testPin );
SCH_MARKER* marker = new SCH_MARKER( ercItem, testPin->GetPosition() ); screen->Append( marker );
ok = false; } } } }
return ok;}
bool CONNECTION_GRAPH::ercCheckDanglingWireEndpoints( const CONNECTION_SUBGRAPH* aSubgraph ){ int err_count = 0; const SCH_SHEET_PATH& sheet = aSubgraph->m_sheet;
for( SCH_ITEM* item : aSubgraph->m_items ) { if( item->GetLayer() != LAYER_WIRE ) continue;
if( item->Type() == SCH_LINE_T ) { SCH_LINE* line = static_cast<SCH_LINE*>( item );
if( line->IsGraphicLine() ) continue;
auto report_error = [&]( VECTOR2I& location ) { std::shared_ptr<ERC_ITEM> ercItem = ERC_ITEM::Create( ERCE_UNCONNECTED_WIRE_ENDPOINT );
ercItem->SetItems( line ); ercItem->SetSheetSpecificPath( sheet ); ercItem->SetErrorMessage( _( "Unconnected wire endpoint" ) );
SCH_MARKER* marker = new SCH_MARKER( ercItem, location ); sheet.LastScreen()->Append( marker );
err_count++; };
if( line->IsStartDangling() ) report_error( line->GetConnectionPoints()[0] );
if( line->IsEndDangling() ) report_error( line->GetConnectionPoints()[1] ); } else if( item->Type() == SCH_BUS_WIRE_ENTRY_T ) { SCH_BUS_WIRE_ENTRY* entry = static_cast<SCH_BUS_WIRE_ENTRY*>( item );
auto report_error = [&]( VECTOR2I& location ) { std::shared_ptr<ERC_ITEM> ercItem = ERC_ITEM::Create( ERCE_UNCONNECTED_WIRE_ENDPOINT );
ercItem->SetItems( entry ); ercItem->SetSheetSpecificPath( sheet ); ercItem->SetErrorMessage( _( "Unconnected wire to bus entry" ) );
SCH_MARKER* marker = new SCH_MARKER( ercItem, location ); sheet.LastScreen()->Append( marker );
err_count++; };
if( entry->IsStartDangling() ) report_error( entry->GetConnectionPoints()[0] );
if( entry->IsEndDangling() ) report_error( entry->GetConnectionPoints()[1] ); }
}
return err_count > 0;}
bool CONNECTION_GRAPH::ercCheckFloatingWires( const CONNECTION_SUBGRAPH* aSubgraph ){ if( aSubgraph->m_driver ) return true;
const SCH_SHEET_PATH& sheet = aSubgraph->m_sheet; std::vector<SCH_ITEM*> wires;
// We've gotten this far, so we know we have no valid driver. All we need to do is check
// for a wire that we can place the error on.
for( SCH_ITEM* item : aSubgraph->m_items ) { if( item->Type() == SCH_LINE_T && item->GetLayer() == LAYER_WIRE ) wires.emplace_back( item ); else if( item->Type() == SCH_BUS_WIRE_ENTRY_T ) wires.emplace_back( item ); }
if( !wires.empty() ) { SCH_SCREEN* screen = aSubgraph->m_sheet.LastScreen();
std::shared_ptr<ERC_ITEM> ercItem = ERC_ITEM::Create( ERCE_WIRE_DANGLING ); ercItem->SetSheetSpecificPath( sheet ); ercItem->SetItems( wires[0], wires.size() > 1 ? wires[1] : nullptr, wires.size() > 2 ? wires[2] : nullptr, wires.size() > 3 ? wires[3] : nullptr );
SCH_MARKER* marker = new SCH_MARKER( ercItem, wires[0]->GetPosition() ); screen->Append( marker );
return false; }
return true;}
bool CONNECTION_GRAPH::ercCheckLabels( const CONNECTION_SUBGRAPH* aSubgraph ){ // Label connection rules:
// Any label without a no-connect needs to have at least 2 pins, otherwise it is invalid
// Local labels are flagged if they don't connect to any pins and don't have a no-connect
// Global labels are flagged if they appear only once, don't connect to any local labels,
// and don't have a no-connect marker
if( !aSubgraph->m_driver_connection ) return true;
// Buses are excluded from this test: many users create buses with only a single instance
// and it's not really a problem as long as the nets in the bus pass ERC
if( aSubgraph->m_driver_connection->IsBus() ) return true;
const SCH_SHEET_PATH& sheet = aSubgraph->m_sheet; ERC_SETTINGS& settings = m_schematic->ErcSettings(); bool ok = true; size_t pinCount = 0; bool has_nc = !!aSubgraph->m_no_connect;
std::map<KICAD_T, std::vector<SCH_TEXT*>> label_map;
auto hasPins = []( const CONNECTION_SUBGRAPH* aLocSubgraph ) -> size_t { return std::count_if( aLocSubgraph->m_items.begin(), aLocSubgraph->m_items.end(), []( const SCH_ITEM* item ) { return item->Type() == SCH_PIN_T; } ); };
auto reportError = [&]( SCH_TEXT* aText, int errCode ) { if( settings.IsTestEnabled( errCode ) ) { std::shared_ptr<ERC_ITEM> ercItem = ERC_ITEM::Create( errCode ); ercItem->SetSheetSpecificPath( sheet ); ercItem->SetItems( aText );
SCH_MARKER* marker = new SCH_MARKER( ercItem, aText->GetPosition() ); aSubgraph->m_sheet.LastScreen()->Append( marker ); } };
pinCount = hasPins( aSubgraph );
for( SCH_ITEM* item : aSubgraph->m_items ) { switch( item->Type() ) { case SCH_LABEL_T: case SCH_GLOBAL_LABEL_T: case SCH_HIER_LABEL_T: { SCH_TEXT* text = static_cast<SCH_TEXT*>( item );
label_map[item->Type()].push_back( text );
// Below, we'll create an ERC if the whole subgraph is unconnected. But, additionally,
// we want to error if an individual label in the subgraph is floating, even if it's
// connected to other valid things by way of another label on the same sheet.
if( text->IsDangling() ) { reportError( text, item->Type() == SCH_GLOBAL_LABEL_T ? ERCE_GLOBLABEL_DANGLING : ERCE_LABEL_NOT_CONNECTED ); return false; }
break; }
default: break; } }
if( label_map.empty() ) return true;
// No-connects on net neighbors will be noticed before, but to notice them on bus parents we
// need to walk the graph
for( auto& [ connection, subgraphs ] : aSubgraph->m_bus_parents ) { for( CONNECTION_SUBGRAPH* busParent : subgraphs ) { if( busParent->m_no_connect ) { has_nc = true; break; }
CONNECTION_SUBGRAPH* hp = busParent->m_hier_parent;
while( hp ) { if( hp->m_no_connect ) { has_nc = true; break; }
hp = hp->m_hier_parent; } } }
wxString netName = GetResolvedSubgraphName( aSubgraph );
wxCHECK_MSG( m_schematic, true, wxS( "Null m_schematic in CONNECTION_GRAPH::ercCheckLabels" ) );
// Labels that have multiple pins connected are not dangling (may be used for naming segments)
// so leave them without errors here
if( pinCount > 1 ) return true;
for( auto& [type, label_vec] : label_map ) { switch( type ) { case SCH_GLOBAL_LABEL_T: if( !settings.IsTestEnabled( ERCE_GLOBLABEL_DANGLING ) ) continue;
break; default: if( !settings.IsTestEnabled( ERCE_LABEL_NOT_CONNECTED ) ) continue;
break; }
for( SCH_TEXT* text : label_vec ) { size_t allPins = pinCount;
auto it = m_net_name_to_subgraphs_map.find( netName );
if( it != m_net_name_to_subgraphs_map.end() ) { for( const CONNECTION_SUBGRAPH* neighbor : it->second ) { if( neighbor == aSubgraph ) continue;
if( neighbor->m_no_connect ) has_nc = true;
allPins += hasPins( neighbor ); } }
if( allPins == 1 && !has_nc ) { reportError( text, type == SCH_GLOBAL_LABEL_T ? ERCE_GLOBLABEL_DANGLING : ERCE_LABEL_NOT_CONNECTED ); ok = false; }
if( allPins == 0 ) { reportError( text, type == SCH_GLOBAL_LABEL_T ? ERCE_GLOBLABEL_DANGLING : ERCE_LABEL_NOT_CONNECTED ); ok = false; } } }
return ok;}
int CONNECTION_GRAPH::ercCheckSingleGlobalLabel(){ int errors = 0;
std::map<wxString, std::tuple<int, const SCH_ITEM*, SCH_SHEET_PATH>> labelData;
for( const SCH_SHEET_PATH& sheet : m_sheetList ) { for( SCH_ITEM* item : sheet.LastScreen()->Items().OfType( SCH_GLOBAL_LABEL_T ) ) { SCH_TEXT* labelText = static_cast<SCH_TEXT*>( item ); wxString resolvedLabelText = EscapeString( labelText->GetShownText( &sheet, false ), CTX_NETNAME );
if( labelData.find( resolvedLabelText ) == labelData.end() ) { labelData[resolvedLabelText] = { 1, item, sheet }; } else { std::get<0>( labelData[resolvedLabelText] ) += 1; std::get<1>( labelData[resolvedLabelText] ) = nullptr; std::get<2>( labelData[resolvedLabelText] ) = sheet; } } }
for( const auto& label : labelData ) { if( std::get<0>( label.second ) == 1 ) { const SCH_SHEET_PATH& sheet = std::get<2>( label.second ); const SCH_ITEM* item = std::get<1>( label.second );
std::shared_ptr<ERC_ITEM> ercItem = ERC_ITEM::Create( ERCE_SINGLE_GLOBAL_LABEL ); ercItem->SetItems( std::get<1>( label.second ) ); ercItem->SetSheetSpecificPath( sheet ); ercItem->SetItemsSheetPaths( sheet );
SCH_MARKER* marker = new SCH_MARKER( ercItem, item->GetPosition() ); sheet.LastScreen()->Append( marker );
errors++; } }
return errors;}
int CONNECTION_GRAPH::ercCheckDirectiveLabels(){ int error_count = 0;
for( const SCH_SHEET_PATH& sheet : m_sheetList ) { for( SCH_ITEM* item : sheet.LastScreen()->Items().OfType( SCH_DIRECTIVE_LABEL_T ) ) { SCH_LABEL* label = static_cast<SCH_LABEL*>( item );
if( label->IsDangling() ) { std::shared_ptr<ERC_ITEM> ercItem = ERC_ITEM::Create( ERCE_LABEL_NOT_CONNECTED ); SCH_TEXT* text = static_cast<SCH_TEXT*>( item ); ercItem->SetSheetSpecificPath( sheet ); ercItem->SetItems( text );
SCH_MARKER* marker = new SCH_MARKER( ercItem, text->GetPosition() ); sheet.LastScreen()->Append( marker ); error_count++; } } }
return error_count;}
int CONNECTION_GRAPH::ercCheckHierSheets(){ int errors = 0;
ERC_SETTINGS& settings = m_schematic->ErcSettings();
for( const SCH_SHEET_PATH& sheet : m_sheetList ) { for( SCH_ITEM* item : sheet.LastScreen()->Items().OfType( SCH_SHEET_T ) ) { SCH_SHEET* parentSheet = static_cast<SCH_SHEET*>( item ); SCH_SHEET_PATH parentSheetPath = sheet;
parentSheetPath.push_back( parentSheet );
std::map<wxString, SCH_SHEET_PIN*> pins; std::map<wxString, SCH_HIERLABEL*> labels;
for( SCH_SHEET_PIN* pin : parentSheet->GetPins() ) { if( settings.IsTestEnabled( ERCE_HIERACHICAL_LABEL ) ) pins[ pin->GetShownText( &parentSheetPath, false ) ] = pin;
if( pin->IsDangling() && settings.IsTestEnabled( ERCE_PIN_NOT_CONNECTED ) ) { std::shared_ptr<ERC_ITEM> ercItem = ERC_ITEM::Create( ERCE_PIN_NOT_CONNECTED ); ercItem->SetItems( pin ); ercItem->SetSheetSpecificPath( sheet ); ercItem->SetItemsSheetPaths( sheet );
SCH_MARKER* marker = new SCH_MARKER( ercItem, pin->GetPosition() ); sheet.LastScreen()->Append( marker );
errors++; } }
if( settings.IsTestEnabled( ERCE_HIERACHICAL_LABEL ) ) { std::set<wxString> matchedPins;
for( SCH_ITEM* subItem : parentSheet->GetScreen()->Items() ) { if( subItem->Type() == SCH_HIER_LABEL_T ) { SCH_HIERLABEL* label = static_cast<SCH_HIERLABEL*>( subItem ); wxString labelText = label->GetShownText( &parentSheetPath, false );
if( !pins.count( labelText ) ) labels[ labelText ] = label; else matchedPins.insert( labelText ); } }
for( const wxString& matched : matchedPins ) pins.erase( matched );
for( const std::pair<const wxString, SCH_SHEET_PIN*>& unmatched : pins ) { wxString msg = wxString::Format( _( "Sheet pin %s has no matching hierarchical " "label inside the sheet" ), UnescapeString( unmatched.first ) );
std::shared_ptr<ERC_ITEM> ercItem = ERC_ITEM::Create( ERCE_HIERACHICAL_LABEL ); ercItem->SetItems( unmatched.second ); ercItem->SetErrorMessage( msg ); ercItem->SetSheetSpecificPath( sheet ); ercItem->SetItemsSheetPaths( sheet );
SCH_MARKER* marker = new SCH_MARKER( ercItem, unmatched.second->GetPosition() ); sheet.LastScreen()->Append( marker );
errors++; }
for( const std::pair<const wxString, SCH_HIERLABEL*>& unmatched : labels ) { wxString msg = wxString::Format( _( "Hierarchical label %s has no matching " "sheet pin in the parent sheet" ), UnescapeString( unmatched.first ) );
std::shared_ptr<ERC_ITEM> ercItem = ERC_ITEM::Create( ERCE_HIERACHICAL_LABEL ); ercItem->SetItems( unmatched.second ); ercItem->SetErrorMessage( msg ); ercItem->SetSheetSpecificPath( parentSheetPath ); ercItem->SetItemsSheetPaths( parentSheetPath );
SCH_MARKER* marker = new SCH_MARKER( ercItem, unmatched.second->GetPosition() ); parentSheet->GetScreen()->Append( marker );
errors++; } } } }
return errors;}
|