/* * This program source code file is part of KiCad, a free EDA CAD application. * * Copyright (C) 2018 CERN * @author Jon Evans * * 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 . */ #include #include #include #include #include #include #include #include #include #include /** * * Buses can be defined in multiple ways. A bus vector consists of a prefix and * a numeric range of suffixes: * * BUS_NAME[M..N] * * For example, the bus A[3..0] will contain nets A3, A2, A1, and A0. * The BUS_NAME is required. M and N must be integers but do not need to be in * any particular order -- A[0..3] produces the same result. * * Like net names, bus names cannot contain whitespace. * * A bus group is just a grouping of signals, separated by spaces, some * of which may be bus vectors. Bus groups can have names, but do not need to. * * MEMORY{A[15..0] D[7..0] RW CE OE} * * In named bus groups, the net names are expanded as . * In the above example, the nets would be named like MEMORY.A15, MEMORY.D0, etc. * * {USB_DP USB_DN} * * In the above example, the bus is unnamed and so the underlying net names are * just USB_DP and USB_DN. * */ SCH_CONNECTION::SCH_CONNECTION( SCH_ITEM* aParent, SCH_SHEET_PATH aPath ) : m_sheet( aPath ), m_parent( aParent ), m_driver( nullptr ), m_graph( nullptr ) { Reset(); } SCH_CONNECTION::SCH_CONNECTION( CONNECTION_GRAPH* aGraph ) : m_sheet( SCH_SHEET_PATH() ), m_parent( nullptr ), m_driver( nullptr ), m_graph( aGraph ) { Reset(); } bool SCH_CONNECTION::operator==( const SCH_CONNECTION& aOther ) const { // NOTE: Not comparing m_dirty or net/bus/subgraph codes if( ( aOther.m_driver == m_driver ) && ( aOther.m_type == m_type ) && ( aOther.m_name == m_name ) && ( aOther.m_sheet == m_sheet ) ) { return true; } return false; } void SCH_CONNECTION::SetDriver( SCH_ITEM* aItem ) { m_driver = aItem; recacheName(); for( const std::shared_ptr& member : m_members ) member->SetDriver( aItem ); } void SCH_CONNECTION::SetSheet( SCH_SHEET_PATH aSheet ) { m_sheet = aSheet; recacheName(); for( const std::shared_ptr& member : m_members ) member->SetSheet( aSheet ); } bool SCH_CONNECTION::operator!=( const SCH_CONNECTION& aOther ) const { return !( aOther == *this ); } void SCH_CONNECTION::ConfigureFromLabel( const wxString& aLabel ) { m_members.clear(); m_name = aLabel; m_local_name = aLabel; wxString prefix; std::vector members; wxString unescaped = UnescapeString( aLabel ); if( NET_SETTINGS::ParseBusVector( unescaped, &prefix, &members ) ) { m_type = CONNECTION_TYPE::BUS; m_vector_prefix = prefix; long i = 0; for( const wxString& vector_member : members ) { auto member = std::make_shared( m_parent, m_sheet ); member->m_type = CONNECTION_TYPE::NET; member->m_prefix = m_prefix; member->m_local_name = vector_member; member->m_vector_index = i++; member->SetName( vector_member ); member->SetGraph( m_graph ); m_members.push_back( member ); } } else if( NET_SETTINGS::ParseBusGroup( unescaped, &prefix, &members ) ) { m_type = CONNECTION_TYPE::BUS_GROUP; m_bus_prefix = prefix; // Named bus groups generate a net prefix, unnamed ones don't if( !prefix.IsEmpty() ) prefix += wxT( "." ); for( const wxString& group_member : members ) { // Handle alias inside bus group member list if( auto alias = m_graph->GetBusAlias( group_member ) ) { for( const wxString& alias_member : alias->Members() ) { auto member = std::make_shared< SCH_CONNECTION >( m_parent, m_sheet ); member->SetPrefix( prefix ); member->SetGraph( m_graph ); member->ConfigureFromLabel( alias_member ); m_members.push_back( member ); } } else { auto member = std::make_shared< SCH_CONNECTION >( m_parent, m_sheet ); member->SetPrefix( prefix ); member->SetGraph( m_graph ); member->ConfigureFromLabel( group_member ); m_members.push_back( member ); } } } else { m_type = CONNECTION_TYPE::NET; } recacheName(); } void SCH_CONNECTION::Reset() { m_type = CONNECTION_TYPE::NONE; m_name.Empty(); m_local_name.Empty(); m_cached_name.Empty(); m_cached_name_with_path.Empty(); m_prefix.Empty(); m_bus_prefix.Empty(); m_suffix .Empty(); m_lastDriver = m_driver; m_driver = nullptr; m_members.clear(); m_dirty = true; m_net_code = 0; m_bus_code = 0; m_subgraph_code = 0; m_vector_start = 0; m_vector_end = 0; m_vector_index = 0; m_vector_prefix.Empty(); } void SCH_CONNECTION::Clone( SCH_CONNECTION& aOther ) { m_graph = aOther.m_graph; m_type = aOther.Type(); m_lastDriver = aOther.GetLastDriver(); m_driver = aOther.Driver(); m_sheet = aOther.Sheet(); m_name = aOther.m_name; // Note: m_local_name is not cloned m_prefix = aOther.Prefix(); m_bus_prefix = aOther.BusPrefix(); m_suffix = aOther.Suffix(); m_members = aOther.Members(); m_net_code = aOther.NetCode(); m_bus_code = aOther.BusCode(); m_vector_start = aOther.VectorStart(); m_vector_end = aOther.VectorEnd(); // Note: m_vector_index is not cloned m_vector_prefix = aOther.VectorPrefix(); // Note: subgraph code isn't cloned, it should remain with the original object recacheName(); } bool SCH_CONNECTION::IsDriver() const { wxASSERT( Parent() ); switch( Parent()->Type() ) { case SCH_LABEL_T: case SCH_GLOBAL_LABEL_T: case SCH_HIER_LABEL_T: case SCH_SHEET_PIN_T: case SCH_SHEET_T: case LIB_PIN_T: return true; case SCH_PIN_T: { auto pin = static_cast( Parent() ); // Only annotated components should drive nets return ( pin->IsPowerConnection() || pin->GetParentComponent()->IsAnnotated( &m_sheet ) ); } default: return false; } } bool SCH_CONNECTION::HasDriverChanged() const { return m_driver != m_lastDriver; } void SCH_CONNECTION::ClearDriverChanged() { m_lastDriver = m_driver; } wxString SCH_CONNECTION::Name( bool aIgnoreSheet ) const { wxASSERT( !m_cached_name.IsEmpty() ); return aIgnoreSheet ? m_cached_name : m_cached_name_with_path; } void SCH_CONNECTION::recacheName() { m_cached_name = m_name.IsEmpty() ? "" : m_prefix + m_name + m_suffix; bool prepend_path = true; if( !Parent() || m_type == CONNECTION_TYPE::NONE ) prepend_path = false; if( m_driver ) { switch( m_driver->Type() ) { case SCH_GLOBAL_LABEL_T: case SCH_PIN_T: // Pins are either power connections or belong to a uniquely-annotated // component, so they don't need a path if they are driving the subgraph prepend_path = false; break; default: break; } } m_cached_name_with_path = prepend_path ? m_sheet.PathHumanReadable() + m_cached_name : m_cached_name; } void SCH_CONNECTION::SetPrefix( const wxString& aPrefix ) { m_prefix = aPrefix; recacheName(); for( const auto& m : Members() ) m->SetPrefix( aPrefix ); } void SCH_CONNECTION::SetSuffix( const wxString& aSuffix ) { m_suffix = aSuffix; recacheName(); for( const auto& m : Members() ) m->SetSuffix( aSuffix ); } void SCH_CONNECTION::AppendInfoToMsgPanel( MSG_PANEL_ITEMS& aList ) const { wxString msg, group_name; std::vector group_members; aList.push_back( MSG_PANEL_ITEM( _( "Connection Name" ), UnescapeString( Name() ), BROWN ) ); // NOTE(JE) Disabling this for now, because net codes are generated in the netlist exporter // in order to avoid sort costs. It may make sense to just tear out net codes from the // CONNECTION_GRAPH entirely in the future, as they are mostly only useful for netlist exports. #if 0 if( !IsBus() ) { msg.Printf( "%d", m_net_code ); aList.push_back( MSG_PANEL_ITEM( _( "Net Code" ), msg, BROWN ) ); } #endif if( auto alias = m_graph->GetBusAlias( m_name ) ) { msg.Printf( _( "Bus Alias %s Members" ), m_name ); wxString members; for( const auto& member : alias->Members() ) members << member << " "; aList.push_back( MSG_PANEL_ITEM( msg, members, RED ) ); } else if( NET_SETTINGS::ParseBusGroup( m_name, &group_name, &group_members ) ) { for( const auto& group_member : group_members ) { if( auto group_alias = m_graph->GetBusAlias( group_member ) ) { msg.Printf( _( "Bus Alias %s Members" ), group_alias->GetName() ); wxString members; for( const auto& member : group_alias->Members() ) members << member << " "; aList.push_back( MSG_PANEL_ITEM( msg, members, RED ) ); } } } #if defined(DEBUG) // These messages are not flagged as translatable, because they are only debug messages if( !ADVANCED_CFG::GetCfg().m_realTimeConnectivity || !CONNECTION_GRAPH::m_allowRealTime ) return; if( IsBus() ) { msg.Printf( "%d", m_bus_code ); aList.push_back( MSG_PANEL_ITEM( "Bus Code", msg, BROWN ) ); } msg.Printf( "%d", m_subgraph_code ); aList.push_back( MSG_PANEL_ITEM( "Subgraph Code", msg, BROWN ) ); if( auto driver = Driver() ) { msg.Printf( "%s at %p", driver->GetSelectMenuText( EDA_UNITS::MILLIMETRES ), driver ); aList.push_back( MSG_PANEL_ITEM( "Connection Source", msg, RED ) ); } #endif } bool SCH_CONNECTION::IsBusLabel( const wxString& aLabel ) { const wxString& unescaped = UnescapeString( aLabel ); return NET_SETTINGS::ParseBusVector( unescaped, nullptr, nullptr ) || NET_SETTINGS::ParseBusGroup( unescaped, nullptr, nullptr ); } bool SCH_CONNECTION::MightBeBusLabel( const wxString& aLabel ) { // Weak heuristic for performance reasons. Stronger test will be used for connectivity wxString label = UnescapeString( aLabel ); return label.Contains( wxT( "[" ) ) || label.Contains( wxT( "{" ) ); } const std::vector< std::shared_ptr< SCH_CONNECTION > > SCH_CONNECTION::AllMembers() const { std::vector< std::shared_ptr< SCH_CONNECTION > > ret( m_members ); for( const auto& member : m_members ) if( member->IsBus() ) ret.insert( ret.end(), member->Members().begin(), member->Members().end() ); return ret; } static bool isSuperSub( wxChar c ) { return c == '_' || c == '^'; }; wxString SCH_CONNECTION::PrintBusForUI( const wxString& aGroup ) { size_t groupLen = aGroup.length(); size_t i = 0; wxString ret; int braceNesting = 0; int tildeNesting = 0; // Parse prefix // for( ; i < groupLen; ++i ) { if( isSuperSub( aGroup[i] ) && i + 1 < groupLen && aGroup[i+1] == '{' ) { braceNesting++; i++; continue; } else if( aGroup[i] == '~' ) { if( tildeNesting ) { tildeNesting = 0; continue; } else { tildeNesting++; } } else if( aGroup[i] == '}' ) { braceNesting--; continue; } ret += aGroup[i]; if( aGroup[i] == '{' ) break; } // Parse members // i++; // '{' character for( ; i < groupLen; ++i ) { if( isSuperSub( aGroup[i] ) && i + 1 < groupLen && aGroup[i+1] == '{' ) { braceNesting++; i++; continue; } else if( aGroup[i] == '~' ) { if( tildeNesting ) { tildeNesting = 0; continue; } else { tildeNesting++; } } else if( aGroup[i] == '}' ) { braceNesting--; continue; } ret += aGroup[i]; if( aGroup[i] == '}' ) break; } return ret; } bool SCH_CONNECTION::IsSubsetOf( SCH_CONNECTION* aOther ) const { if( !aOther->IsBus() ) return false; for( const auto& member : aOther->Members() ) { if( member->FullLocalName() == FullLocalName() ) return true; } return false; } bool SCH_CONNECTION::IsMemberOfBus( SCH_CONNECTION* aOther ) const { if( !aOther->IsBus() ) return false; auto me = Name( true ); for( const auto& m : aOther->Members() ) if( m->Name( true ) == me ) return true; return false; }