Fix a few issues with hierarchical propagation

This commit is contained in:
Jon Evans 2019-03-18 18:38:06 -04:00 committed by Wayne Stambaugh
parent 3c714f1d8c
commit e98debfeb2
4 changed files with 251 additions and 84 deletions

View File

@ -50,6 +50,13 @@ bool CONNECTION_SUBGRAPH::ResolveDrivers( bool aCreateMarkers )
m_driver = nullptr; 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( auto item : m_drivers ) for( auto item : m_drivers )
{ {
int item_priority = 0; int item_priority = 0;
@ -57,8 +64,8 @@ bool CONNECTION_SUBGRAPH::ResolveDrivers( bool aCreateMarkers )
switch( item->Type() ) switch( item->Type() )
{ {
case SCH_SHEET_PIN_T: item_priority = 2; break; case SCH_SHEET_PIN_T: item_priority = 2; break;
case SCH_LABEL_T: item_priority = 3; break; case SCH_HIERARCHICAL_LABEL_T: item_priority = 3; break;
case SCH_HIERARCHICAL_LABEL_T: item_priority = 4; break; case SCH_LABEL_T: item_priority = 4; break;
case SCH_PIN_CONNECTION_T: case SCH_PIN_CONNECTION_T:
{ {
auto pin_connection = static_cast<SCH_PIN_CONNECTION*>( item ); auto pin_connection = static_cast<SCH_PIN_CONNECTION*>( item );
@ -90,6 +97,9 @@ bool CONNECTION_SUBGRAPH::ResolveDrivers( bool aCreateMarkers )
} }
} }
if( highest_priority >= 4 )
m_strong_driver = true;
if( candidates.size() ) if( candidates.size() )
{ {
if( candidates.size() > 1 ) if( candidates.size() > 1 )
@ -109,9 +119,27 @@ bool CONNECTION_SUBGRAPH::ResolveDrivers( bool aCreateMarkers )
return name_a < name_b; return name_a < name_b;
} ); } );
} }
if( highest_priority == 2 )
{
// We have multiple options, and they are all hierarchical
// sheet pins. Let's prefer outputs over inputs.
for( auto c : candidates )
{
auto p = static_cast<SCH_SHEET_PIN*>( c );
if( p->GetShape() == NET_OUTPUT )
{
m_driver = c;
break;
}
}
}
} }
m_driver = candidates[0]; if( !m_driver )
m_driver = candidates[0];
} }
// For power connections, we allow multiple drivers // For power connections, we allow multiple drivers
@ -641,6 +669,56 @@ void CONNECTION_GRAPH::buildConnectionGraph()
while( threadsFinished < parallelThreadCount ) while( threadsFinished < parallelThreadCount )
std::this_thread::sleep_for( std::chrono::milliseconds( 10 ) ); std::this_thread::sleep_for( std::chrono::milliseconds( 10 ) );
// 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_subgraphs )
subgraph->m_dirty = true;
for( auto subgraph : m_subgraphs )
{
if( !subgraph->m_dirty )
continue;
subgraph->m_dirty = false;
if( !subgraph->m_driver || subgraph->m_strong_driver )
continue;
auto name = subgraph->m_driver->Connection( subgraph->m_sheet )->Name();
unsigned suffix = 1;
for( auto candidate : m_subgraphs )
{
if( !candidate->m_dirty )
continue;
if( candidate == subgraph || !candidate->m_driver || candidate->m_strong_driver )
continue;
auto conn = candidate->m_driver->Connection( candidate->m_sheet );
auto check_name = conn->Name();
if( check_name == name )
{
auto new_name = wxString::Format( _( "%s%u" ), name, suffix );
#ifdef CONNECTIVITY_DEBUG
wxLogDebug( "Subgraph %ld and %ld both have name %s. Changing %ld to %s.",
subgraph->m_code, candidate->m_code, name,
candidate->m_code, new_name );
#endif
conn->SetSuffix( wxString::Format( _( "%u" ), suffix ) );
candidate->m_dirty = false;
suffix++;
}
}
}
// Generate net codes // Generate net codes
for( auto subgraph : m_subgraphs ) for( auto subgraph : m_subgraphs )
@ -721,6 +799,10 @@ void CONNECTION_GRAPH::buildConnectionGraph()
connections_to_check.push_back( std::make_shared<SCH_CONNECTION>( *connection ) ); connections_to_check.push_back( std::make_shared<SCH_CONNECTION>( *connection ) );
} }
// Look for "neighbors" for subgraphs that have hierarchical connections.
// These are usually other subgraphs that have local labels on the
// same sheet and so should be connected together.
for( unsigned i = 0; i < connections_to_check.size(); i++ ) for( unsigned i = 0; i < connections_to_check.size(); i++ )
{ {
auto member = connections_to_check[i]; auto member = connections_to_check[i];
@ -984,109 +1066,176 @@ void CONNECTION_GRAPH::buildConnectionGraph()
* 3) Recurse down onto any subsheets connected to the SSSG. * 3) Recurse down onto any subsheets connected to the SSSG.
*/ */
for( auto item : subgraph->m_items ) std::vector<CONNECTION_SUBGRAPH*> child_subgraphs;
child_subgraphs.push_back( subgraph );
for( unsigned i = 0; i < child_subgraphs.size(); i++ )
{ {
if( item->Type() == SCH_SHEET_PIN_T ) // child_subgraphs[i] now refers to the "parent" subgraph that we
// are descending the hierarchy with. If there are multiple levels
// of hierarchy, those will get pushed onto child_subgraphs below.
for( auto item : child_subgraphs[i]->m_items )
{ {
auto sp = static_cast<SCH_SHEET_PIN*>( item ); if( item->Type() == SCH_SHEET_PIN_T )
auto sp_name = sp->GetText();
auto subsheet = sheet;
subsheet.push_back( sp->GetParent() );
for( auto candidate : m_subgraphs )
{ {
if( !candidate->m_dirty ) auto sp = static_cast<SCH_SHEET_PIN*>( item );
continue; auto sp_name = sp->GetText();
auto subsheet = child_subgraphs[i]->m_sheet;
subsheet.push_back( sp->GetParent() );
if( candidate->m_sheet == subsheet && candidate->m_driver ) #ifdef CONNECTIVITY_DEBUG
wxLogDebug( wxT("Propagating sheet pin %s on %s with connection %s to subsheet %s"),
sp_name, child_subgraphs[i]->m_sheet.PathHumanReadable(),
connection->Name(), subsheet.PathHumanReadable() );
#endif
for( auto candidate : m_subgraphs )
{ {
auto driver = candidate->m_driver; if( !candidate->m_dirty )
continue;
if( ( driver->Type() == SCH_HIERARCHICAL_LABEL_T ) && if( candidate->m_sheet == subsheet && candidate->m_driver )
( static_cast<SCH_HIERLABEL*>( driver )->GetText() == sp_name ) )
{ {
// We found a subgraph that is a subsheet child of SCH_ITEM* hier_label = nullptr;
// our top-level subgraph, so let's mark it
candidate->m_dirty = false; for( auto d : candidate->m_drivers )
auto type = driver->Connection( subsheet )->Type();
// Directly update subsheet net connections
for( auto c_item : candidate->m_items )
{ {
auto c = c_item->Connection( subsheet ); if( ( d->Type() == SCH_HIERARCHICAL_LABEL_T ) &&
( static_cast<SCH_HIERLABEL*>( d )->GetText() == sp_name ) )
wxASSERT( c ); hier_label = d;
if( ( connection->IsBus() && c->IsNet() ) ||
( connection->IsNet() && c->IsBus() ) )
{
continue;
}
c->Clone( *connection );
} }
// Now propagate to subsheet neighbors if( hier_label )
for( auto& kv : candidate->m_neighbor_map )
{ {
auto member = kv.first; #ifdef CONNECTIVITY_DEBUG
std::shared_ptr<SCH_CONNECTION> top_level_conn; wxLogDebug( "Found child %s", static_cast<SCH_HIERLABEL*>( hier_label )->GetText() );
#endif
if( type == CONNECTION_BUS_GROUP ) // We found a subgraph that is a subsheet child of
// our top-level subgraph, so let's mark it
candidate->m_dirty = false;
auto type = hier_label->Connection( subsheet )->Type();
bool candidate_has_sheet_pins = false;
// Directly update subsheet net connections
for( auto c_item : candidate->m_items )
{ {
// Bus group: match parent by name if( c_item->Type() == SCH_SHEET_PIN_T )
for( auto parent_member : connection->Members() ) candidate_has_sheet_pins = true;
auto c = c_item->Connection( subsheet );
wxASSERT( c );
if( ( connection->IsBus() && c->IsNet() ) ||
( connection->IsNet() && c->IsBus() ) )
{ {
if( parent_member->IsNet() && continue;
parent_member->Name( true ) == member->Name( true ) ) }
c->Clone( *connection );
}
// Now propagate to subsheet neighbors
for( auto& kv : candidate->m_neighbor_map )
{
auto member = kv.first;
std::shared_ptr<SCH_CONNECTION> top_level_conn;
#ifdef CONNECTIVITY_DEBUG
wxLogDebug( "Found child neighbor from member %s",
member->Name() );
#endif
if( type == CONNECTION_BUS_GROUP )
{
// Bus group: match parent by name
for( auto parent_member : connection->Members() )
{ {
top_level_conn = parent_member; if( parent_member->IsNet() &&
} parent_member->Name( true ) == member->Name( true ) )
else if( parent_member->IsBus() )
{
for( auto& sub_member : parent_member->Members() )
{ {
if( sub_member->Name( true ) == member->Name( true ) ) top_level_conn = parent_member;
top_level_conn = sub_member; }
else if( parent_member->IsBus() )
{
for( auto& sub_member : parent_member->Members() )
{
if( sub_member->Name( true ) == member->Name( true ) )
top_level_conn = sub_member;
}
} }
} }
} }
} else if( type == CONNECTION_BUS )
else if( type == CONNECTION_BUS )
{
// Bus vector: match parent by index
for( auto parent_member : connection->Members() )
{ {
if( parent_member->VectorIndex() == member->VectorIndex() ) // Bus vector: match parent by index
top_level_conn = parent_member; for( auto parent_member : connection->Members() )
{
if( parent_member->VectorIndex() == member->VectorIndex() )
top_level_conn = parent_member;
}
}
else
{
top_level_conn = connection;
}
// If top_level_conn was not found, probably it's
// an ERC error and will be caught by ERC
if( !top_level_conn )
{
continue;
}
for( auto neighbor : kv.second )
{
#ifdef CONNECTIVITY_DEBUG
wxLogDebug( "Propagating to neighbor driven by %s",
neighbor->m_driver->GetSelectMenuText( MILLIMETRES ) );
#endif
bool neighbor_has_sheet_pins = false;
for( auto n_item : neighbor->m_items )
{
auto c = n_item->Connection( subsheet );
wxASSERT( c );
c->Clone( *top_level_conn );
if( n_item->Type() == SCH_SHEET_PIN_T )
neighbor_has_sheet_pins = true;
}
if( neighbor_has_sheet_pins )
{
#ifdef CONNECTIVITY_DEBUG
wxLogDebug( "Neighbor driven by %s has subsheet pins",
neighbor->m_driver->GetSelectMenuText( MILLIMETRES ) );
#endif
child_subgraphs.push_back( neighbor );
}
} }
} }
else
// Now, check to see if the candidate also has
// sheet pin members. If so, add to the queue.
if( candidate_has_sheet_pins)
{ {
top_level_conn = connection; #ifdef CONNECTIVITY_DEBUG
} wxLogDebug( "Candidate %s has subsheet pins",
candidate->m_driver->GetSelectMenuText( MILLIMETRES ) );
// If top_level_conn was not found, probably it's #endif
// an ERC error and will be caught by ERC child_subgraphs.push_back( candidate );
if( !top_level_conn )
{
continue;
}
for( auto neighbor : kv.second )
{
for( auto n_item : neighbor->m_items )
{
auto c = n_item->Connection( subsheet );
wxASSERT( c );
c->Clone( *top_level_conn );
}
} }
} }
} }

View File

@ -86,6 +86,9 @@ public:
/// True if this subgraph contains multiple power ports to join in one net /// True if this subgraph contains multiple power ports to join in one net
bool m_multiple_power_ports; bool m_multiple_power_ports;
/// True if the driver is "strong": a label or power object
bool m_strong_driver;
/// No-connect item in graph, if any /// No-connect item in graph, if any
SCH_ITEM* m_no_connect; SCH_ITEM* m_no_connect;

View File

@ -188,6 +188,7 @@ void SCH_CONNECTION::Reset()
m_type = CONNECTION_NONE; m_type = CONNECTION_NONE;
m_name = "<NO NET>"; m_name = "<NO NET>";
m_prefix = ""; m_prefix = "";
m_suffix = "";
m_driver = nullptr; m_driver = nullptr;
m_members.clear(); m_members.clear();
m_dirty = true; m_dirty = true;
@ -208,6 +209,8 @@ void SCH_CONNECTION::Clone( SCH_CONNECTION& aOther )
m_sheet = aOther.Sheet(); m_sheet = aOther.Sheet();
m_name = aOther.Name( true ); m_name = aOther.Name( true );
m_prefix = aOther.Prefix(); m_prefix = aOther.Prefix();
// Don't clone suffix, it will be rolled into the name
//m_suffix = aOther.Suffix();
m_members = aOther.Members(); m_members = aOther.Members();
m_net_code = aOther.NetCode(); m_net_code = aOther.NetCode();
m_bus_code = aOther.BusCode(); m_bus_code = aOther.BusCode();
@ -242,7 +245,7 @@ bool SCH_CONNECTION::IsDriver() const
wxString SCH_CONNECTION::Name( bool aIgnoreSheet ) const wxString SCH_CONNECTION::Name( bool aIgnoreSheet ) const
{ {
wxString ret = m_name; wxString ret = m_name + m_suffix;
if( !Parent() || m_type == CONNECTION_NONE ) if( !Parent() || m_type == CONNECTION_NONE )
return ret; return ret;
@ -277,7 +280,7 @@ void SCH_CONNECTION::AppendInfoToMsgPanel( MSG_PANEL_ITEMS& aList ) const
wxString msg, group_name; wxString msg, group_name;
std::vector<wxString> group_members; std::vector<wxString> group_members;
aList.push_back( MSG_PANEL_ITEM( _( "Connection Name" ), m_name, BROWN ) ); aList.push_back( MSG_PANEL_ITEM( _( "Connection Name" ), Name(), BROWN ) );
msg.Printf( "%d", m_net_code ); msg.Printf( "%d", m_net_code );
aList.push_back( MSG_PANEL_ITEM( _( "Net Code" ), msg, BROWN ) ); aList.push_back( MSG_PANEL_ITEM( _( "Net Code" ), msg, BROWN ) );

View File

@ -152,11 +152,21 @@ public:
return m_prefix; return m_prefix;
} }
wxString Suffix() const
{
return m_suffix;
}
void SetPrefix( wxString aPrefix ) void SetPrefix( wxString aPrefix )
{ {
m_prefix = aPrefix; m_prefix = aPrefix;
} }
void SetSuffix( wxString aSuffix )
{
m_suffix = aSuffix;
}
CONNECTION_TYPE Type() const CONNECTION_TYPE Type() const
{ {
return m_type; return m_type;
@ -318,6 +328,8 @@ private:
///< Prefix if connection is member of a labeled bus group (or "" if not) ///< Prefix if connection is member of a labeled bus group (or "" if not)
wxString m_prefix; wxString m_prefix;
wxString m_suffix; ///< Name suffix (used only for disambiguation)
int m_net_code; // TODO(JE) remove if unused int m_net_code; // TODO(JE) remove if unused
int m_bus_code; // TODO(JE) remove if unused int m_bus_code; // TODO(JE) remove if unused