2011-10-11 13:38:13 +00:00
|
|
|
/*
|
|
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
|
|
*
|
2013-09-25 19:09:57 +00:00
|
|
|
* Copyright (C) 2013 Jean-Pierre Charras, jp.charras at wanadoo.fr
|
|
|
|
* Copyright (C) 2013 Wayne Stambaugh <stambaughw@verizon.net>
|
|
|
|
* Copyright (C) 1992-2013 KiCad Developers, see AUTHORS.txt for contributors.
|
2011-10-11 13:38:13 +00:00
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 2
|
|
|
|
* of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, you may find one here:
|
|
|
|
* http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
|
|
|
|
* or you may search the http://www.gnu.org website for the version 2 license,
|
|
|
|
* or you may write to the Free Software Foundation, Inc.,
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
|
|
|
*/
|
|
|
|
|
2011-10-07 14:41:30 +00:00
|
|
|
/**
|
|
|
|
* @file eeschema/netlist.cpp
|
|
|
|
*/
|
2007-05-06 16:03:28 +00:00
|
|
|
|
2012-01-23 04:33:36 +00:00
|
|
|
#include <fctsys.h>
|
|
|
|
#include <wxEeschemaStruct.h>
|
2009-09-25 18:49:04 +00:00
|
|
|
|
2012-01-23 04:33:36 +00:00
|
|
|
#include <netlist.h>
|
2013-09-25 19:09:57 +00:00
|
|
|
#include <class_netlist_object.h>
|
2012-01-23 04:33:36 +00:00
|
|
|
#include <protos.h>
|
|
|
|
#include <class_library.h>
|
|
|
|
#include <lib_pin.h>
|
|
|
|
#include <sch_junction.h>
|
|
|
|
#include <sch_component.h>
|
|
|
|
#include <sch_line.h>
|
|
|
|
#include <sch_no_connect.h>
|
|
|
|
#include <sch_text.h>
|
|
|
|
#include <sch_sheet.h>
|
|
|
|
#include <algorithm>
|
2009-04-05 20:49:15 +00:00
|
|
|
|
2010-11-10 15:30:12 +00:00
|
|
|
#include <boost/foreach.hpp>
|
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
#define IS_WIRE false
|
|
|
|
#define IS_BUS true
|
2013-04-01 05:47:44 +00:00
|
|
|
|
2009-07-12 15:29:42 +00:00
|
|
|
// Buffer to build the list of items used in netlist and erc calculations
|
2013-09-27 12:30:35 +00:00
|
|
|
NETLIST_OBJECT_LIST s_NetObjectslist( true );
|
2009-04-05 20:49:15 +00:00
|
|
|
|
2010-08-03 05:11:05 +00:00
|
|
|
//#define NETLIST_DEBUG
|
2007-05-06 16:03:28 +00:00
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
NETLIST_OBJECT_LIST::~NETLIST_OBJECT_LIST()
|
|
|
|
{
|
|
|
|
if( m_isOwner )
|
2013-09-27 12:30:35 +00:00
|
|
|
FreeList();
|
|
|
|
else
|
|
|
|
Clear();
|
2013-09-25 19:09:57 +00:00
|
|
|
}
|
2007-09-21 04:40:12 +00:00
|
|
|
|
2011-10-07 14:41:30 +00:00
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
/*
|
|
|
|
* Delete all objects in list and clear list
|
|
|
|
* (free memory used to store info about NETLIST_OBJECT items)
|
|
|
|
*/
|
2013-09-27 12:30:35 +00:00
|
|
|
void NETLIST_OBJECT_LIST::FreeList()
|
2007-09-21 04:40:12 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
std::vector<NETLIST_OBJECT*>::iterator iter;
|
|
|
|
|
|
|
|
for( iter = begin(); iter != end(); iter++ )
|
2007-09-21 04:40:12 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
NETLIST_OBJECT* item = *iter;
|
|
|
|
delete item;
|
2007-09-21 04:40:12 +00:00
|
|
|
}
|
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
clear();
|
|
|
|
}
|
2007-09-20 21:06:49 +00:00
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
void NETLIST_OBJECT_LIST::SortListbyNetcode()
|
2011-12-12 14:02:37 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
sort( this->begin(), this->end(), NETLIST_OBJECT_LIST::sortItemsbyNetcode );
|
2011-12-12 14:02:37 +00:00
|
|
|
}
|
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
void NETLIST_OBJECT_LIST::SortListbySheet()
|
2007-05-06 16:03:28 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
sort( this->begin(), this->end(), NETLIST_OBJECT_LIST::sortItemsBySheet );
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
2007-09-20 21:06:49 +00:00
|
|
|
|
2009-11-04 20:46:53 +00:00
|
|
|
/*
|
|
|
|
* Build net list connection table.
|
2013-09-27 12:30:35 +00:00
|
|
|
* Initializes s_NetObjectslist
|
2007-09-20 21:06:49 +00:00
|
|
|
*/
|
2013-09-27 12:30:35 +00:00
|
|
|
NETLIST_OBJECT_LIST * SCH_EDIT_FRAME::BuildNetListBase()
|
2007-05-06 16:03:28 +00:00
|
|
|
{
|
2009-12-02 21:44:03 +00:00
|
|
|
wxBusyCursor Busy;
|
2007-09-20 21:06:49 +00:00
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
// Creates the flattened sheet list:
|
|
|
|
SCH_SHEET_LIST aSheets;
|
2009-07-12 15:29:42 +00:00
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
// Build netlist info
|
2013-09-27 12:30:35 +00:00
|
|
|
bool success = s_NetObjectslist.BuildNetListInfo( aSheets );
|
2007-09-20 21:06:49 +00:00
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
if( !success )
|
2011-10-12 14:03:43 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
SetStatusText( _("No Objects" ) );
|
2013-09-27 12:30:35 +00:00
|
|
|
return &s_NetObjectslist;
|
2011-10-12 14:03:43 +00:00
|
|
|
}
|
2009-07-12 15:29:42 +00:00
|
|
|
|
2013-04-14 11:02:19 +00:00
|
|
|
/* The new %zu specification is needed to properly format a size_t
|
|
|
|
* value (returned by size(), here) */
|
2013-09-25 19:09:57 +00:00
|
|
|
wxString msg;
|
2007-09-20 21:06:49 +00:00
|
|
|
|
2013-09-27 12:30:35 +00:00
|
|
|
msg.Printf( _( "Net count = %zu" ), s_NetObjectslist.size() );
|
2013-09-25 19:09:57 +00:00
|
|
|
SetStatusText( msg );
|
2013-09-27 12:30:35 +00:00
|
|
|
|
|
|
|
return &s_NetObjectslist;
|
2013-09-25 19:09:57 +00:00
|
|
|
}
|
2007-09-20 21:06:49 +00:00
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
/* the master function of NETLIST_OBJECT_LIST class.
|
|
|
|
* Build the list of connected objects (pins, labels ...) and
|
|
|
|
* all info needed to generate netlists or run ERC diags
|
|
|
|
*/
|
|
|
|
bool NETLIST_OBJECT_LIST::BuildNetListInfo( SCH_SHEET_LIST& aSheets )
|
|
|
|
{
|
2013-09-27 12:30:35 +00:00
|
|
|
s_NetObjectslist.SetOwner( true );
|
|
|
|
s_NetObjectslist.FreeList();
|
2007-09-20 21:06:49 +00:00
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
SCH_SHEET_PATH* sheet;
|
|
|
|
|
|
|
|
// Fill list with connected items from the flattened sheet list
|
|
|
|
for( sheet = aSheets.GetFirst(); sheet != NULL;
|
|
|
|
sheet = aSheets.GetNext() )
|
|
|
|
{
|
|
|
|
for( SCH_ITEM* item = sheet->LastScreen()->GetDrawItems(); item; item = item->Next() )
|
|
|
|
{
|
|
|
|
item->GetNetListItem( *this, sheet );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if( size() == 0 )
|
|
|
|
return false;
|
2007-09-20 21:06:49 +00:00
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
// Sort objects by Sheet
|
|
|
|
SortListbySheet();
|
|
|
|
|
2013-09-29 18:24:38 +00:00
|
|
|
sheet = &(GetItem( 0 )->m_SheetPath);
|
2013-09-27 12:30:35 +00:00
|
|
|
m_lastNetCode = m_lastBusNetCode = 1;
|
2008-03-20 01:50:21 +00:00
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
for( unsigned ii = 0, istart = 0; ii < size(); ii++ )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
NETLIST_OBJECT* net_item = GetItem( ii );
|
2011-10-07 14:41:30 +00:00
|
|
|
|
2013-09-29 18:24:38 +00:00
|
|
|
if( net_item->m_SheetPath != *sheet ) // Sheet change
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2013-09-29 18:24:38 +00:00
|
|
|
sheet = &(net_item->m_SheetPath);
|
2009-07-15 07:10:07 +00:00
|
|
|
istart = ii;
|
2007-09-20 21:06:49 +00:00
|
|
|
}
|
|
|
|
|
2009-07-15 07:10:07 +00:00
|
|
|
switch( net_item->m_Type )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2009-07-12 15:29:42 +00:00
|
|
|
case NET_ITEM_UNSPECIFIED:
|
2009-11-04 20:46:53 +00:00
|
|
|
wxMessageBox( wxT( "BuildNetListBase() error" ) );
|
2009-07-12 15:29:42 +00:00
|
|
|
break;
|
2009-11-04 20:46:53 +00:00
|
|
|
|
2007-09-20 21:06:49 +00:00
|
|
|
case NET_PIN:
|
|
|
|
case NET_PINLABEL:
|
|
|
|
case NET_SHEETLABEL:
|
|
|
|
case NET_NOCONNECT:
|
2009-07-15 07:10:07 +00:00
|
|
|
if( net_item->GetNet() != 0 )
|
2009-11-04 20:46:53 +00:00
|
|
|
break;
|
2007-09-20 21:06:49 +00:00
|
|
|
|
|
|
|
case NET_SEGMENT:
|
2013-09-25 19:09:57 +00:00
|
|
|
// Test connections point to point type without bus.
|
2009-07-15 07:10:07 +00:00
|
|
|
if( net_item->GetNet() == 0 )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2013-09-27 12:30:35 +00:00
|
|
|
net_item->SetNet( m_lastNetCode );
|
|
|
|
m_lastNetCode++;
|
2007-09-20 21:06:49 +00:00
|
|
|
}
|
2011-10-07 14:41:30 +00:00
|
|
|
|
2013-09-27 12:30:35 +00:00
|
|
|
pointToPointConnect( net_item, IS_WIRE, istart );
|
2007-09-20 21:06:49 +00:00
|
|
|
break;
|
|
|
|
|
2011-10-07 14:41:30 +00:00
|
|
|
case NET_JUNCTION:
|
2013-09-25 19:09:57 +00:00
|
|
|
// Control of the junction outside BUS.
|
2009-07-15 07:10:07 +00:00
|
|
|
if( net_item->GetNet() == 0 )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2013-09-27 12:30:35 +00:00
|
|
|
net_item->SetNet( m_lastNetCode );
|
|
|
|
m_lastNetCode++;
|
2007-09-20 21:06:49 +00:00
|
|
|
}
|
2011-10-07 14:41:30 +00:00
|
|
|
|
2013-09-27 12:30:35 +00:00
|
|
|
segmentToPointConnect( net_item, IS_WIRE, istart );
|
2007-09-20 21:06:49 +00:00
|
|
|
|
2009-11-04 20:46:53 +00:00
|
|
|
/* Control of the junction, on BUS. */
|
2009-07-15 07:10:07 +00:00
|
|
|
if( net_item->m_BusNetCode == 0 )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2013-09-27 12:30:35 +00:00
|
|
|
net_item->m_BusNetCode = m_lastBusNetCode;
|
|
|
|
m_lastBusNetCode++;
|
2007-09-20 21:06:49 +00:00
|
|
|
}
|
2011-10-07 14:41:30 +00:00
|
|
|
|
2013-09-27 12:30:35 +00:00
|
|
|
segmentToPointConnect( net_item, IS_BUS, istart );
|
2007-09-20 21:06:49 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case NET_LABEL:
|
2008-02-12 21:12:46 +00:00
|
|
|
case NET_HIERLABEL:
|
2008-03-20 01:50:21 +00:00
|
|
|
case NET_GLOBLABEL:
|
2013-09-25 19:09:57 +00:00
|
|
|
// Test connections type junction without bus.
|
2009-07-15 07:10:07 +00:00
|
|
|
if( net_item->GetNet() == 0 )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2013-09-27 12:30:35 +00:00
|
|
|
net_item->SetNet( m_lastNetCode );
|
|
|
|
m_lastNetCode++;
|
2007-09-20 21:06:49 +00:00
|
|
|
}
|
2011-10-07 14:41:30 +00:00
|
|
|
|
2013-09-27 12:30:35 +00:00
|
|
|
segmentToPointConnect( net_item, IS_WIRE, istart );
|
2007-09-20 21:06:49 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case NET_SHEETBUSLABELMEMBER:
|
2009-07-15 07:10:07 +00:00
|
|
|
if( net_item->m_BusNetCode != 0 )
|
2009-11-04 20:46:53 +00:00
|
|
|
break;
|
2007-09-20 21:06:49 +00:00
|
|
|
|
|
|
|
case NET_BUS:
|
2009-11-04 20:46:53 +00:00
|
|
|
/* Control type connections point to point mode bus */
|
2009-07-15 07:10:07 +00:00
|
|
|
if( net_item->m_BusNetCode == 0 )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2013-09-27 12:30:35 +00:00
|
|
|
net_item->m_BusNetCode = m_lastBusNetCode;
|
|
|
|
m_lastBusNetCode++;
|
2007-09-20 21:06:49 +00:00
|
|
|
}
|
2011-10-07 14:41:30 +00:00
|
|
|
|
2013-09-27 12:30:35 +00:00
|
|
|
pointToPointConnect( net_item, IS_BUS, istart );
|
2007-09-20 21:06:49 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case NET_BUSLABELMEMBER:
|
2008-02-12 21:12:46 +00:00
|
|
|
case NET_HIERBUSLABELMEMBER:
|
2008-03-20 01:50:21 +00:00
|
|
|
case NET_GLOBBUSLABELMEMBER:
|
2009-11-04 20:46:53 +00:00
|
|
|
/* Control connections similar has on BUS */
|
2009-07-15 07:10:07 +00:00
|
|
|
if( net_item->GetNet() == 0 )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2013-09-27 12:30:35 +00:00
|
|
|
net_item->m_BusNetCode = m_lastBusNetCode;
|
|
|
|
m_lastBusNetCode++;
|
2007-09-20 21:06:49 +00:00
|
|
|
}
|
2011-10-07 14:41:30 +00:00
|
|
|
|
2013-09-27 12:30:35 +00:00
|
|
|
segmentToPointConnect( net_item, IS_BUS, istart );
|
2007-09-20 21:06:49 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-07-12 15:29:42 +00:00
|
|
|
#if defined(NETLIST_DEBUG) && defined(DEBUG)
|
2008-02-12 21:12:46 +00:00
|
|
|
std::cout << "\n\nafter sheet local\n\n";
|
2013-09-25 19:09:57 +00:00
|
|
|
DumpNetTable();
|
2008-03-20 01:50:21 +00:00
|
|
|
#endif
|
2007-09-21 04:40:12 +00:00
|
|
|
|
2009-11-04 20:46:53 +00:00
|
|
|
/* Updating the Bus Labels Netcode connected by Bus */
|
2013-09-27 12:30:35 +00:00
|
|
|
connectBusLabels();
|
2007-09-20 21:06:49 +00:00
|
|
|
|
2009-11-04 20:46:53 +00:00
|
|
|
/* Group objects by label. */
|
2013-09-25 19:09:57 +00:00
|
|
|
for( unsigned ii = 0; ii < size(); ii++ )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
switch( GetItem( ii )->m_Type )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
|
|
|
case NET_PIN:
|
|
|
|
case NET_SHEETLABEL:
|
|
|
|
case NET_SEGMENT:
|
2011-10-07 14:41:30 +00:00
|
|
|
case NET_JUNCTION:
|
2007-09-20 21:06:49 +00:00
|
|
|
case NET_BUS:
|
|
|
|
case NET_NOCONNECT:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NET_LABEL:
|
2008-03-20 01:50:21 +00:00
|
|
|
case NET_GLOBLABEL:
|
2007-09-20 21:06:49 +00:00
|
|
|
case NET_PINLABEL:
|
|
|
|
case NET_BUSLABELMEMBER:
|
2008-03-20 01:50:21 +00:00
|
|
|
case NET_GLOBBUSLABELMEMBER:
|
2013-09-25 19:09:57 +00:00
|
|
|
labelConnect( GetItem( ii ) );
|
2007-09-20 21:06:49 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case NET_SHEETBUSLABELMEMBER:
|
2008-03-20 01:50:21 +00:00
|
|
|
case NET_HIERLABEL:
|
|
|
|
case NET_HIERBUSLABELMEMBER:
|
2007-09-20 21:06:49 +00:00
|
|
|
break;
|
2009-11-04 20:46:53 +00:00
|
|
|
|
2009-07-12 15:29:42 +00:00
|
|
|
case NET_ITEM_UNSPECIFIED:
|
|
|
|
break;
|
2007-09-20 21:06:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-07-12 15:29:42 +00:00
|
|
|
#if defined(NETLIST_DEBUG) && defined(DEBUG)
|
2008-02-12 21:12:46 +00:00
|
|
|
std::cout << "\n\nafter sheet global\n\n";
|
2013-09-25 19:09:57 +00:00
|
|
|
DumpNetTable();
|
2008-03-20 01:50:21 +00:00
|
|
|
#endif
|
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
// Connection between hierarchy sheets
|
2013-09-27 12:30:35 +00:00
|
|
|
for( unsigned ii = 0; ii < size(); ii++ )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
if( GetItem( ii )->m_Type == NET_SHEETLABEL
|
|
|
|
|| GetItem( ii )->m_Type == NET_SHEETBUSLABELMEMBER )
|
2013-09-27 12:30:35 +00:00
|
|
|
sheetLabelConnect( GetItem( ii ) );
|
2007-09-20 21:06:49 +00:00
|
|
|
}
|
2008-03-20 01:50:21 +00:00
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
// Sort objects by NetCode
|
|
|
|
SortListbyNetcode();
|
2007-09-20 21:06:49 +00:00
|
|
|
|
2009-07-12 15:29:42 +00:00
|
|
|
#if defined(NETLIST_DEBUG) && defined(DEBUG)
|
2010-08-03 05:11:05 +00:00
|
|
|
std::cout << "\n\nafter qsort()\n";
|
2013-09-25 19:09:57 +00:00
|
|
|
DumpNetTable();
|
2008-03-20 01:50:21 +00:00
|
|
|
#endif
|
|
|
|
|
2009-11-04 20:46:53 +00:00
|
|
|
/* Compress numbers of Netcode having consecutive values. */
|
2013-09-25 19:09:57 +00:00
|
|
|
int NetCode = 0;
|
2013-09-27 12:30:35 +00:00
|
|
|
m_lastNetCode = 0;
|
2011-10-07 14:41:30 +00:00
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
for( unsigned ii = 0; ii < size(); ii++ )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2013-09-27 12:30:35 +00:00
|
|
|
if( GetItem( ii )->GetNet() != m_lastNetCode )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2008-03-20 01:50:21 +00:00
|
|
|
NetCode++;
|
2013-09-27 12:30:35 +00:00
|
|
|
m_lastNetCode = GetItem( ii )->GetNet();
|
2007-09-20 21:06:49 +00:00
|
|
|
}
|
2011-10-07 14:41:30 +00:00
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
GetItem( ii )->SetNet( NetCode );
|
2007-09-20 21:06:49 +00:00
|
|
|
}
|
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
// Set the minimal connection info:
|
2013-09-27 12:30:35 +00:00
|
|
|
setUnconnectedFlag();
|
2010-06-23 17:00:12 +00:00
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
// find the best label object to give the best net name to each net
|
2013-09-27 12:30:35 +00:00
|
|
|
findBestNetNameForEachNet();
|
2013-09-25 19:09:57 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Helper function to give a priority to sort labels:
|
|
|
|
// NET_PINLABEL and NET_GLOBLABEL are global labels
|
|
|
|
// and the priority is hight
|
|
|
|
static int getPriority( const NETLIST_OBJECT* Objet )
|
|
|
|
{
|
|
|
|
switch( Objet->m_Type )
|
|
|
|
{
|
|
|
|
case NET_PIN: return 1;
|
|
|
|
case NET_LABEL: return 2;
|
|
|
|
case NET_HIERLABEL: return 3;
|
|
|
|
case NET_PINLABEL: return 4;
|
|
|
|
case NET_GLOBLABEL: return 5;
|
|
|
|
default: break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-09-27 12:30:35 +00:00
|
|
|
/* function evalLabelsPriority used by findBestNetNameForEachNet()
|
2013-09-25 19:09:57 +00:00
|
|
|
* evalLabelsPriority calculates the priority of alabel1 and aLabel2
|
2013-09-27 12:30:35 +00:00
|
|
|
* return true if alabel1 has a highter priority than aLabel2
|
2013-09-25 19:09:57 +00:00
|
|
|
*/
|
|
|
|
static bool evalLabelsPriority( const NETLIST_OBJECT* aLabel1,
|
|
|
|
const NETLIST_OBJECT* aLabel2 )
|
|
|
|
{
|
|
|
|
int priority1 = getPriority( aLabel1 );
|
|
|
|
int priority2 = getPriority( aLabel2 );
|
|
|
|
|
|
|
|
if( priority1 != priority2 )
|
2013-09-27 12:30:35 +00:00
|
|
|
return priority1 > priority2;
|
2013-09-25 19:09:57 +00:00
|
|
|
|
|
|
|
// Objects have here the same priority, therefore they have the same type.
|
|
|
|
|
|
|
|
// for global labels, we select the best candidate by alphabetic order
|
|
|
|
// because they have no sheetpath as prefix name
|
|
|
|
// for other labels, we select them before by sheet deep order
|
|
|
|
// because the actual name is /sheetpath/label
|
|
|
|
// and for a given path length, by alphabetic order
|
|
|
|
|
|
|
|
if( aLabel1->m_Type == NET_PINLABEL || aLabel1->m_Type == NET_GLOBLABEL )
|
|
|
|
return aLabel1->m_Label.Cmp( aLabel2->m_Label );
|
|
|
|
|
|
|
|
// not global: names are prefixed by their sheetpath
|
|
|
|
// use name defined in higher hierarchical sheet
|
|
|
|
// (i.e. shorter path because paths are /<timestamp1>/<timestamp2>/...
|
|
|
|
// and timestamp = 8 letters.
|
2013-09-29 18:24:38 +00:00
|
|
|
if( aLabel1->m_SheetPath.Path().Length() != aLabel2->m_SheetPath.Path().Length() )
|
|
|
|
return aLabel1->m_SheetPath.Path().Length() < aLabel2->m_SheetPath.Path().Length();
|
2013-09-25 19:09:57 +00:00
|
|
|
|
|
|
|
// Sheet paths have the same length: use alphabetic label name order
|
|
|
|
// For labels on sheets having an equivalent deep in hierarchy, use
|
|
|
|
// alphabetic label name order:
|
|
|
|
if( aLabel1->m_Label.Cmp( aLabel2->m_Label ) != 0 )
|
2013-09-27 12:30:35 +00:00
|
|
|
return aLabel1->m_Label.Cmp( aLabel2->m_Label ) < 0;
|
2013-09-25 19:09:57 +00:00
|
|
|
|
2013-09-29 18:24:38 +00:00
|
|
|
return aLabel1->m_SheetPath.PathHumanReadable().Cmp(
|
|
|
|
aLabel2->m_SheetPath.PathHumanReadable() ) < 0;
|
2010-06-23 17:00:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-11-12 16:36:43 +00:00
|
|
|
/**
|
2013-09-27 12:30:35 +00:00
|
|
|
* Function findBestNetNameForEachNet
|
2010-06-23 17:00:12 +00:00
|
|
|
* fill the .m_NetNameCandidate member of each item of aNetItemBuffer
|
|
|
|
* with a reference to the "best" NETLIST_OBJECT usable to give a name to the net
|
|
|
|
* If no suitable object found, .m_NetNameCandidate is filled with 0.
|
|
|
|
* The "best" NETLIST_OBJECT is a NETLIST_OBJECT that have the type label
|
|
|
|
* and by priority order:
|
|
|
|
* the label is global or local
|
|
|
|
* the label is in the first sheet in a hierarchy (the root sheet has the most priority)
|
|
|
|
* alphabetic order.
|
|
|
|
*/
|
2013-09-27 12:30:35 +00:00
|
|
|
void NETLIST_OBJECT_LIST::findBestNetNameForEachNet()
|
2010-06-23 17:00:12 +00:00
|
|
|
{
|
|
|
|
int netcode = 0; // current netcode for tested items
|
2013-09-25 19:09:57 +00:00
|
|
|
unsigned idxstart = 0; // index of the first item of this net
|
|
|
|
NETLIST_OBJECT* item;
|
|
|
|
NETLIST_OBJECT* candidate;
|
|
|
|
|
|
|
|
// Pass 1: find the best name for labelled nets:
|
|
|
|
item = NULL;
|
|
|
|
candidate = NULL;
|
|
|
|
for( unsigned ii = 0; ii <= size(); ii++ )
|
2010-06-23 17:00:12 +00:00
|
|
|
{
|
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
if( ii == size() ) // last item already found
|
2010-06-23 17:00:12 +00:00
|
|
|
netcode = -2;
|
|
|
|
else
|
2013-09-25 19:09:57 +00:00
|
|
|
item = GetItem( ii );
|
2011-10-07 14:41:30 +00:00
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
if( netcode != item->GetNet() ) // End of net found
|
2010-06-23 17:00:12 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
if( candidate ) // One or more labels exists, find the best
|
2010-06-23 17:00:12 +00:00
|
|
|
{
|
|
|
|
for (unsigned jj = idxstart; jj < ii; jj++ )
|
2013-09-25 19:09:57 +00:00
|
|
|
GetItem( jj )->SetNetNameCandidate( candidate );
|
2010-06-23 17:00:12 +00:00
|
|
|
}
|
2011-10-07 14:41:30 +00:00
|
|
|
|
2010-06-23 17:00:12 +00:00
|
|
|
if( netcode == -2 )
|
|
|
|
break;
|
2011-10-07 14:41:30 +00:00
|
|
|
|
2010-06-23 17:00:12 +00:00
|
|
|
netcode = item->GetNet();
|
2013-09-25 19:09:57 +00:00
|
|
|
candidate = NULL;
|
2010-06-23 17:00:12 +00:00
|
|
|
idxstart = ii;
|
|
|
|
}
|
2011-10-07 14:41:30 +00:00
|
|
|
|
2010-06-23 17:00:12 +00:00
|
|
|
switch( item->m_Type )
|
|
|
|
{
|
|
|
|
case NET_HIERLABEL:
|
|
|
|
case NET_LABEL:
|
|
|
|
case NET_PINLABEL:
|
2010-07-14 13:24:36 +00:00
|
|
|
case NET_GLOBLABEL:
|
2013-09-25 19:09:57 +00:00
|
|
|
// A candidate is found: select the better between the previous
|
|
|
|
// and this one
|
|
|
|
if( candidate == NULL )
|
|
|
|
candidate = item;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if( evalLabelsPriority( item, candidate ) )
|
2013-09-27 12:30:35 +00:00
|
|
|
// item has a highter priority than candidate
|
|
|
|
// so update the best candidate
|
2013-09-25 19:09:57 +00:00
|
|
|
candidate = item;
|
|
|
|
}
|
2010-06-23 17:00:12 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
// Pass 2: find the best name for not labelled nets:
|
|
|
|
// The "default" net name is Net-<<Ref cmp>_Pad<num pad>>
|
|
|
|
// (see NETLIST_OBJECT::GetShortNetName())
|
|
|
|
// therefore the "best" is the short net name alphabetically classed first
|
|
|
|
// (to avoid net names changes when the net is not modified,
|
|
|
|
// even if components are moved or deleted and undelete or replaced, as long
|
|
|
|
// the reference is kept)
|
2013-09-27 17:58:58 +00:00
|
|
|
|
|
|
|
// Build the list of items with no net names
|
|
|
|
NETLIST_OBJECT_LIST list;
|
|
|
|
for( unsigned ii = 0; ii < size(); ii++ )
|
|
|
|
{
|
|
|
|
item = GetItem( ii );
|
|
|
|
if( !item->HasNetNameCandidate() )
|
|
|
|
list.push_back( item );
|
|
|
|
}
|
|
|
|
|
|
|
|
if( list.size() == 0 )
|
|
|
|
return;
|
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
idxstart = 0;
|
|
|
|
candidate = NULL;
|
2013-09-27 17:58:58 +00:00
|
|
|
netcode = list.GetItemNet( 0 );
|
|
|
|
|
|
|
|
for( unsigned ii = 0; ii <= list.size(); ii++ )
|
2010-07-14 13:24:36 +00:00
|
|
|
{
|
2013-09-27 17:58:58 +00:00
|
|
|
if( ii < list.size() )
|
|
|
|
item = list.GetItem( ii );
|
2011-10-07 14:41:30 +00:00
|
|
|
|
2013-09-27 17:58:58 +00:00
|
|
|
if( netcode != item->GetNet() || ii >= list.size() ) // End of net found
|
2010-06-23 17:00:12 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
if( candidate )
|
2010-07-14 13:24:36 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
for (unsigned jj = idxstart; jj < ii; jj++ )
|
2013-09-27 17:58:58 +00:00
|
|
|
{
|
|
|
|
NETLIST_OBJECT* obj = list.GetItem( jj );
|
|
|
|
obj->SetNetNameCandidate( candidate );
|
|
|
|
}
|
2010-07-14 13:24:36 +00:00
|
|
|
}
|
2013-09-25 19:09:57 +00:00
|
|
|
|
2013-09-27 17:58:58 +00:00
|
|
|
if( ii >= list.size() )
|
2013-09-25 19:09:57 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
netcode = item->GetNet();
|
|
|
|
candidate = NULL;
|
|
|
|
idxstart = ii;
|
2010-07-14 13:24:36 +00:00
|
|
|
}
|
2013-09-25 19:09:57 +00:00
|
|
|
|
2013-09-30 06:36:40 +00:00
|
|
|
// Examine all pins of the net to find the best candidate,
|
|
|
|
// i.e. the first net name candidate, by alphabetic order
|
|
|
|
// the net names are names bu_ilt by GetShortNetName
|
|
|
|
// (Net-<{reference}-Pad{pad number}> like Net-<U3-Pad5>
|
|
|
|
// Not named nets do not have usually a lot of members.
|
|
|
|
// Many have only 2 members(a pad and a non connection symbol)
|
2013-09-27 17:58:58 +00:00
|
|
|
if( item->m_Type == NET_PIN )
|
2010-07-14 13:24:36 +00:00
|
|
|
{
|
2013-09-27 17:58:58 +00:00
|
|
|
// A candidate is found, however components which are not in
|
|
|
|
// netlist are not candidate because some have their reference
|
2013-09-30 06:36:40 +00:00
|
|
|
// changed each time the netlist is built (power components)
|
|
|
|
// and anyway obviously they are not a good candidate
|
|
|
|
SCH_COMPONENT* link = item->GetComponentParent();
|
|
|
|
if( link && link->IsInNetlist() )
|
2010-11-08 14:27:02 +00:00
|
|
|
{
|
2013-09-27 17:58:58 +00:00
|
|
|
// select the better between the previous and this one
|
|
|
|
item->SetNetNameCandidate( item ); // Needed to calculate GetShortNetName
|
2013-09-30 06:36:40 +00:00
|
|
|
|
2013-09-27 17:58:58 +00:00
|
|
|
if( candidate == NULL )
|
2013-09-25 19:09:57 +00:00
|
|
|
candidate = item;
|
2013-09-27 17:58:58 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
if( item->GetShortNetName().Cmp( candidate->GetShortNetName() ) < 0 )
|
|
|
|
candidate = item;
|
|
|
|
}
|
2010-11-08 14:27:02 +00:00
|
|
|
}
|
2010-07-14 13:24:36 +00:00
|
|
|
}
|
2010-06-23 17:00:12 +00:00
|
|
|
}
|
|
|
|
}
|
2007-09-20 21:06:49 +00:00
|
|
|
|
2011-10-11 13:38:13 +00:00
|
|
|
|
2009-11-04 20:46:53 +00:00
|
|
|
/*
|
2013-09-25 19:09:57 +00:00
|
|
|
* Propagate net codes from a parent sheet to an include sheet,
|
|
|
|
* from a pin sheet connection
|
2009-11-04 20:46:53 +00:00
|
|
|
*/
|
2013-09-27 12:30:35 +00:00
|
|
|
void NETLIST_OBJECT_LIST::sheetLabelConnect( NETLIST_OBJECT* SheetLabel )
|
2007-05-06 16:03:28 +00:00
|
|
|
{
|
2007-10-13 06:18:44 +00:00
|
|
|
if( SheetLabel->GetNet() == 0 )
|
2007-09-20 21:06:49 +00:00
|
|
|
return;
|
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
for( unsigned ii = 0; ii < size(); ii++ )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
NETLIST_OBJECT* ObjetNet = GetItem( ii );
|
2011-10-11 13:38:13 +00:00
|
|
|
|
2013-09-29 18:24:38 +00:00
|
|
|
if( ObjetNet->m_SheetPath != SheetLabel->m_SheetPathInclude )
|
2009-11-04 20:46:53 +00:00
|
|
|
continue; //use SheetInclude, not the sheet!!
|
2008-03-20 01:50:21 +00:00
|
|
|
|
2011-10-11 13:38:13 +00:00
|
|
|
if( (ObjetNet->m_Type != NET_HIERLABEL ) && (ObjetNet->m_Type != NET_HIERBUSLABELMEMBER ) )
|
2007-09-20 21:06:49 +00:00
|
|
|
continue;
|
2008-03-20 01:50:21 +00:00
|
|
|
|
2009-07-12 15:29:42 +00:00
|
|
|
if( ObjetNet->GetNet() == SheetLabel->GetNet() )
|
2009-11-04 20:46:53 +00:00
|
|
|
continue; //already connected.
|
2008-03-20 01:50:21 +00:00
|
|
|
|
2010-06-24 18:31:43 +00:00
|
|
|
if( ObjetNet->m_Label.CmpNoCase( SheetLabel->m_Label ) != 0 )
|
2009-11-04 20:46:53 +00:00
|
|
|
continue; //different names.
|
2007-09-20 21:06:49 +00:00
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
// Propagate Netcode having all the objects of the same Netcode.
|
2009-07-12 15:29:42 +00:00
|
|
|
if( ObjetNet->GetNet() )
|
2013-09-25 19:09:57 +00:00
|
|
|
propageNetCode( ObjetNet->GetNet(), SheetLabel->GetNet(), IS_WIRE );
|
2007-09-20 21:06:49 +00:00
|
|
|
else
|
2009-07-12 15:29:42 +00:00
|
|
|
ObjetNet->SetNet( SheetLabel->GetNet() );
|
2007-09-20 21:06:49 +00:00
|
|
|
}
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
2007-09-20 21:06:49 +00:00
|
|
|
|
2009-11-04 20:46:53 +00:00
|
|
|
/*
|
2013-09-25 19:09:57 +00:00
|
|
|
* Analyzes the labels type bus member (<BUS_NAME><member_number>
|
|
|
|
* Propagate net codes between the corresponding labels (ie when
|
|
|
|
* the <member_number> is the same) when they are connected
|
|
|
|
* uqsually by their BusNetCode
|
2013-09-27 12:30:35 +00:00
|
|
|
* Uses and updates the variable m_lastNetCode
|
2007-09-20 21:06:49 +00:00
|
|
|
*/
|
2013-09-27 12:30:35 +00:00
|
|
|
void NETLIST_OBJECT_LIST::connectBusLabels()
|
2007-05-06 16:03:28 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
for( unsigned ii = 0; ii < size(); ii++ )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
NETLIST_OBJECT* Label = GetItem( ii );
|
2011-10-07 14:41:30 +00:00
|
|
|
|
|
|
|
if( (Label->m_Type == NET_SHEETBUSLABELMEMBER)
|
|
|
|
|| (Label->m_Type == NET_BUSLABELMEMBER)
|
|
|
|
|| (Label->m_Type == NET_HIERBUSLABELMEMBER) )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2007-10-13 06:18:44 +00:00
|
|
|
if( Label->GetNet() == 0 )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2013-09-27 12:30:35 +00:00
|
|
|
Label->SetNet( m_lastNetCode );
|
|
|
|
m_lastNetCode++;
|
2007-09-20 21:06:49 +00:00
|
|
|
}
|
2008-03-20 01:50:21 +00:00
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
for( unsigned jj = ii + 1; jj < size(); jj++ )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
NETLIST_OBJECT* LabelInTst = GetItem( jj );
|
2007-09-20 21:06:49 +00:00
|
|
|
if( (LabelInTst->m_Type == NET_SHEETBUSLABELMEMBER)
|
|
|
|
|| (LabelInTst->m_Type == NET_BUSLABELMEMBER)
|
2008-02-12 21:12:46 +00:00
|
|
|
|| (LabelInTst->m_Type == NET_HIERBUSLABELMEMBER) )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
|
|
|
if( LabelInTst->m_BusNetCode != Label->m_BusNetCode )
|
|
|
|
continue;
|
2008-03-20 01:50:21 +00:00
|
|
|
|
2007-09-20 21:06:49 +00:00
|
|
|
if( LabelInTst->m_Member != Label->m_Member )
|
|
|
|
continue;
|
2008-03-20 01:50:21 +00:00
|
|
|
|
2007-10-13 06:18:44 +00:00
|
|
|
if( LabelInTst->GetNet() == 0 )
|
|
|
|
LabelInTst->SetNet( Label->GetNet() );
|
2007-09-20 21:06:49 +00:00
|
|
|
else
|
2013-09-25 19:09:57 +00:00
|
|
|
propageNetCode( LabelInTst->GetNet(), Label->GetNet(), IS_WIRE );
|
2007-09-20 21:06:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
2008-08-22 14:48:30 +00:00
|
|
|
|
|
|
|
|
2009-11-04 20:46:53 +00:00
|
|
|
/*
|
2013-09-25 19:09:57 +00:00
|
|
|
* propageNetCode propagates the net code NewNetCode to all elements
|
|
|
|
* having previously the net code OldNetCode
|
|
|
|
* If IsBus == false, m_Netcode is used to propagate the new net code
|
|
|
|
* If IsBus == true, m_BusNetCode is used to propagate the new net code
|
2007-09-20 21:06:49 +00:00
|
|
|
*/
|
2013-09-25 19:09:57 +00:00
|
|
|
void NETLIST_OBJECT_LIST::propageNetCode( int aOldNetCode, int aNewNetCode, bool aIsBus )
|
2007-05-06 16:03:28 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
if( aOldNetCode == aNewNetCode )
|
2007-09-20 21:06:49 +00:00
|
|
|
return;
|
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
if( aIsBus == false ) // Propagate NetCode
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
for( unsigned jj = 0; jj < size(); jj++ )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
NETLIST_OBJECT* objet = GetItem( jj );
|
2011-10-07 14:41:30 +00:00
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
if( objet->GetNet() == aOldNetCode )
|
|
|
|
objet->SetNet( aNewNetCode );
|
2007-09-20 21:06:49 +00:00
|
|
|
}
|
|
|
|
}
|
2013-09-25 19:09:57 +00:00
|
|
|
else // Propagate BusNetCode
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
for( unsigned jj = 0; jj < size(); jj++ )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
NETLIST_OBJECT* objet = GetItem( jj );
|
2011-10-07 14:41:30 +00:00
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
if( objet->m_BusNetCode == aOldNetCode )
|
|
|
|
objet->m_BusNetCode = aNewNetCode;
|
2007-09-20 21:06:49 +00:00
|
|
|
}
|
|
|
|
}
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
2007-09-20 21:06:49 +00:00
|
|
|
|
2009-11-04 20:46:53 +00:00
|
|
|
/*
|
|
|
|
* Check if Ref element is connected to other elements of the list of objects
|
|
|
|
* in the schematic, by mode point
|
|
|
|
* A point (end superimposed)
|
2008-03-20 01:50:21 +00:00
|
|
|
*
|
2009-11-04 20:46:53 +00:00
|
|
|
* If IsBus:
|
|
|
|
* The connection involves elements such as bus
|
|
|
|
* (Or BUS or BUSLABEL JUNCTION)
|
|
|
|
* Otherwise
|
|
|
|
* The connection involves elements such as non-bus
|
|
|
|
* (Other than BUS or BUSLABEL)
|
2008-03-20 01:50:21 +00:00
|
|
|
*
|
2009-11-04 20:46:53 +00:00
|
|
|
* The Ref object must have a valid Netcode.
|
2008-03-20 01:50:21 +00:00
|
|
|
*
|
2009-11-04 20:46:53 +00:00
|
|
|
* The list of objects is SUPPOSED class by SheetPath Croissants,
|
|
|
|
* And research is done from the start element, 1st element
|
|
|
|
* Leaf schema
|
|
|
|
* (There can be no physical connection between elements of different sheets)
|
2007-09-20 21:06:49 +00:00
|
|
|
*/
|
2013-09-27 12:30:35 +00:00
|
|
|
void NETLIST_OBJECT_LIST::pointToPointConnect( NETLIST_OBJECT* aRef, bool aIsBus,
|
2013-09-25 19:09:57 +00:00
|
|
|
int start )
|
2007-05-06 16:03:28 +00:00
|
|
|
{
|
2009-07-12 15:29:42 +00:00
|
|
|
int netCode;
|
2007-09-20 21:06:49 +00:00
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
if( aIsBus == false ) // Objects other than BUS and BUSLABELS
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
netCode = aRef->GetNet();
|
2011-10-07 14:41:30 +00:00
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
for( unsigned i = start; i < size(); i++ )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
NETLIST_OBJECT* item = GetItem( i );
|
2011-10-07 14:41:30 +00:00
|
|
|
|
2013-09-29 18:24:38 +00:00
|
|
|
if( item->m_SheetPath != aRef->m_SheetPath ) //used to be > (why?)
|
2008-02-12 21:12:46 +00:00
|
|
|
continue;
|
2008-03-20 01:50:21 +00:00
|
|
|
|
2009-07-12 15:29:42 +00:00
|
|
|
switch( item->m_Type )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
|
|
|
case NET_SEGMENT:
|
|
|
|
case NET_PIN:
|
|
|
|
case NET_LABEL:
|
2008-02-12 21:12:46 +00:00
|
|
|
case NET_HIERLABEL:
|
2008-03-20 01:50:21 +00:00
|
|
|
case NET_GLOBLABEL:
|
2007-09-20 21:06:49 +00:00
|
|
|
case NET_SHEETLABEL:
|
|
|
|
case NET_PINLABEL:
|
2011-10-07 14:41:30 +00:00
|
|
|
case NET_JUNCTION:
|
2007-09-20 21:06:49 +00:00
|
|
|
case NET_NOCONNECT:
|
2013-09-25 19:09:57 +00:00
|
|
|
if( aRef->m_Start == item->m_Start
|
|
|
|
|| aRef->m_Start == item->m_End
|
|
|
|
|| aRef->m_End == item->m_Start
|
|
|
|
|| aRef->m_End == item->m_End )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2009-07-12 15:29:42 +00:00
|
|
|
if( item->GetNet() == 0 )
|
|
|
|
item->SetNet( netCode );
|
2007-09-20 21:06:49 +00:00
|
|
|
else
|
2013-09-25 19:09:57 +00:00
|
|
|
propageNetCode( item->GetNet(), netCode, IS_WIRE );
|
2007-09-20 21:06:49 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NET_BUS:
|
|
|
|
case NET_BUSLABELMEMBER:
|
|
|
|
case NET_SHEETBUSLABELMEMBER:
|
2008-02-12 21:12:46 +00:00
|
|
|
case NET_HIERBUSLABELMEMBER:
|
2008-03-20 01:50:21 +00:00
|
|
|
case NET_GLOBBUSLABELMEMBER:
|
2009-07-12 15:29:42 +00:00
|
|
|
case NET_ITEM_UNSPECIFIED:
|
2007-09-20 21:06:49 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2009-11-04 20:46:53 +00:00
|
|
|
else /* Object type BUS, BUSLABELS, and junctions. */
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
netCode = aRef->m_BusNetCode;
|
2011-10-07 14:41:30 +00:00
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
for( unsigned i = start; i < size(); i++ )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
NETLIST_OBJECT* item = GetItem( i );
|
2011-10-07 14:41:30 +00:00
|
|
|
|
2013-09-29 18:24:38 +00:00
|
|
|
if( item->m_SheetPath != aRef->m_SheetPath )
|
2008-02-12 21:12:46 +00:00
|
|
|
continue;
|
2007-09-20 21:06:49 +00:00
|
|
|
|
2009-07-12 15:29:42 +00:00
|
|
|
switch( item->m_Type )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2009-07-12 15:29:42 +00:00
|
|
|
case NET_ITEM_UNSPECIFIED:
|
2007-09-20 21:06:49 +00:00
|
|
|
case NET_SEGMENT:
|
|
|
|
case NET_PIN:
|
|
|
|
case NET_LABEL:
|
2008-02-12 21:12:46 +00:00
|
|
|
case NET_HIERLABEL:
|
2008-03-20 01:50:21 +00:00
|
|
|
case NET_GLOBLABEL:
|
2007-09-20 21:06:49 +00:00
|
|
|
case NET_SHEETLABEL:
|
|
|
|
case NET_PINLABEL:
|
|
|
|
case NET_NOCONNECT:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NET_BUS:
|
|
|
|
case NET_BUSLABELMEMBER:
|
|
|
|
case NET_SHEETBUSLABELMEMBER:
|
2008-02-12 21:12:46 +00:00
|
|
|
case NET_HIERBUSLABELMEMBER:
|
2008-03-20 01:50:21 +00:00
|
|
|
case NET_GLOBBUSLABELMEMBER:
|
2011-10-07 14:41:30 +00:00
|
|
|
case NET_JUNCTION:
|
2013-09-25 19:09:57 +00:00
|
|
|
if( aRef->m_Start == item->m_Start
|
|
|
|
|| aRef->m_Start == item->m_End
|
|
|
|
|| aRef->m_End == item->m_Start
|
|
|
|
|| aRef->m_End == item->m_End )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2009-07-12 15:29:42 +00:00
|
|
|
if( item->m_BusNetCode == 0 )
|
|
|
|
item->m_BusNetCode = netCode;
|
2007-09-20 21:06:49 +00:00
|
|
|
else
|
2013-09-25 19:09:57 +00:00
|
|
|
propageNetCode( item->m_BusNetCode, netCode, IS_BUS );
|
2007-09-20 21:06:49 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
2013-09-25 19:09:57 +00:00
|
|
|
* Search connections betweena junction and segments
|
|
|
|
* Propagate the junction net code to objects connected by this junction.
|
|
|
|
* The junction must have a valid net code
|
2010-11-07 20:06:07 +00:00
|
|
|
* The list of objects is expected sorted by sheets.
|
2013-09-25 19:09:57 +00:00
|
|
|
* Search is done from index aIdxStart to the last element of list
|
2007-09-20 21:06:49 +00:00
|
|
|
*/
|
2013-09-27 12:30:35 +00:00
|
|
|
void NETLIST_OBJECT_LIST::segmentToPointConnect( NETLIST_OBJECT* aJonction,
|
2013-09-25 19:09:57 +00:00
|
|
|
bool aIsBus, int aIdxStart )
|
2007-05-06 16:03:28 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
for( unsigned i = aIdxStart; i < size(); i++ )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
NETLIST_OBJECT* segment = GetItem( i );
|
2009-07-12 15:29:42 +00:00
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
// if different sheets, obviously no physical connection between elements.
|
2013-09-29 18:24:38 +00:00
|
|
|
if( segment->m_SheetPath != aJonction->m_SheetPath )
|
2008-02-12 21:12:46 +00:00
|
|
|
continue;
|
2007-09-20 21:06:49 +00:00
|
|
|
|
2013-04-01 05:47:44 +00:00
|
|
|
if( aIsBus == IS_WIRE )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
if( segment->m_Type != NET_SEGMENT )
|
2007-09-20 21:06:49 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
if( segment->m_Type != NET_BUS )
|
2007-09-20 21:06:49 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2013-09-27 12:30:35 +00:00
|
|
|
if( IsPointOnSegment( segment->m_Start, segment->m_End, aJonction->m_Start ) )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
// Propagation Netcode has all the objects of the same Netcode.
|
2013-04-01 05:47:44 +00:00
|
|
|
if( aIsBus == IS_WIRE )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
if( segment->GetNet() )
|
|
|
|
propageNetCode( segment->GetNet(), aJonction->GetNet(), aIsBus );
|
2007-09-20 21:06:49 +00:00
|
|
|
else
|
2013-09-25 19:09:57 +00:00
|
|
|
segment->SetNet( aJonction->GetNet() );
|
2007-09-20 21:06:49 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
if( segment->m_BusNetCode )
|
|
|
|
propageNetCode( segment->m_BusNetCode, aJonction->m_BusNetCode, aIsBus );
|
2007-09-20 21:06:49 +00:00
|
|
|
else
|
2013-09-25 19:09:57 +00:00
|
|
|
segment->m_BusNetCode = aJonction->m_BusNetCode;
|
2007-09-20 21:06:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
2007-09-20 21:06:49 +00:00
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
/*
|
|
|
|
* This function merges the net codes of groups of objects already connected
|
|
|
|
* to labels (wires, bus, pins ... ) when 2 labels are equivalents
|
|
|
|
* (i.e. group objects connected by labels)
|
|
|
|
*/
|
|
|
|
void NETLIST_OBJECT_LIST::labelConnect( NETLIST_OBJECT* aLabelRef )
|
2007-05-06 16:03:28 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
if( aLabelRef->GetNet() == 0 )
|
2007-09-20 21:06:49 +00:00
|
|
|
return;
|
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
for( unsigned i = 0; i < size(); i++ )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
NETLIST_OBJECT* item = GetItem( i );
|
|
|
|
|
|
|
|
if( item->GetNet() == aLabelRef->GetNet() )
|
2007-09-20 21:06:49 +00:00
|
|
|
continue;
|
2011-10-07 14:41:30 +00:00
|
|
|
|
2013-09-29 18:24:38 +00:00
|
|
|
if( item->m_SheetPath != aLabelRef->m_SheetPath )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
if( item->m_Type != NET_PINLABEL && item->m_Type != NET_GLOBLABEL
|
|
|
|
&& item->m_Type != NET_GLOBBUSLABELMEMBER )
|
2008-03-20 01:50:21 +00:00
|
|
|
continue;
|
2011-10-07 14:41:30 +00:00
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
if( (item->m_Type == NET_GLOBLABEL
|
|
|
|
|| item->m_Type == NET_GLOBBUSLABELMEMBER)
|
|
|
|
&& item->m_Type != aLabelRef->m_Type )
|
2008-03-20 01:50:21 +00:00
|
|
|
//global labels only connect other global labels.
|
2007-09-20 21:06:49 +00:00
|
|
|
continue;
|
|
|
|
}
|
2008-08-22 14:48:30 +00:00
|
|
|
|
2010-11-08 14:27:02 +00:00
|
|
|
// NET_HIERLABEL are used to connect sheets.
|
2013-09-25 19:09:57 +00:00
|
|
|
// NET_LABEL are local to a sheet
|
|
|
|
// NET_GLOBLABEL are global.
|
2010-11-08 14:27:02 +00:00
|
|
|
// NET_PINLABEL is a kind of global label (generated by a power pin invisible)
|
2013-09-25 19:09:57 +00:00
|
|
|
if( item->IsLabelType() )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
if( item->m_Label.CmpNoCase( aLabelRef->m_Label ) != 0 )
|
2007-09-20 21:06:49 +00:00
|
|
|
continue;
|
2008-08-22 14:48:30 +00:00
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
if( item->GetNet() )
|
|
|
|
propageNetCode( item->GetNet(), aLabelRef->GetNet(), IS_WIRE );
|
2007-09-20 21:06:49 +00:00
|
|
|
else
|
2013-09-25 19:09:57 +00:00
|
|
|
item->SetNet( aLabelRef->GetNet() );
|
2007-09-20 21:06:49 +00:00
|
|
|
}
|
|
|
|
}
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
2008-08-22 14:48:30 +00:00
|
|
|
|
2013-09-27 12:30:35 +00:00
|
|
|
/* Set the m_ConnectionType member of items in list
|
2013-09-25 19:09:57 +00:00
|
|
|
* depending on the connection type:
|
|
|
|
* UNCONNECTED, PAD_CONNECT or NOCONNECT_SYMBOL_PRESENT
|
|
|
|
* The list is expected sorted by order of net code,
|
|
|
|
* i.e. items having the same net code are grouped
|
2007-09-20 21:06:49 +00:00
|
|
|
*/
|
2013-09-27 12:30:35 +00:00
|
|
|
void NETLIST_OBJECT_LIST::setUnconnectedFlag()
|
2007-05-06 16:03:28 +00:00
|
|
|
{
|
2009-07-12 15:29:42 +00:00
|
|
|
NETLIST_OBJECT* NetItemRef;
|
|
|
|
unsigned NetStart, NetEnd;
|
2011-10-12 15:34:52 +00:00
|
|
|
NET_CONNECTION_T StateFlag;
|
2007-09-20 21:06:49 +00:00
|
|
|
|
2009-07-12 15:29:42 +00:00
|
|
|
NetStart = NetEnd = 0;
|
2008-12-10 16:49:53 +00:00
|
|
|
StateFlag = UNCONNECTED;
|
2013-09-25 19:09:57 +00:00
|
|
|
for( unsigned ii = 0; ii < size(); ii++ )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
NetItemRef = GetItem( ii );
|
2009-07-12 15:29:42 +00:00
|
|
|
if( NetItemRef->m_Type == NET_NOCONNECT && StateFlag != PAD_CONNECT )
|
|
|
|
StateFlag = NOCONNECT_SYMBOL_PRESENT;
|
2007-09-20 21:06:49 +00:00
|
|
|
|
2009-11-04 20:46:53 +00:00
|
|
|
/* Analysis of current net. */
|
2009-07-12 15:29:42 +00:00
|
|
|
unsigned idxtoTest = ii + 1;
|
2007-09-20 21:06:49 +00:00
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
if( ( idxtoTest >= size() )
|
|
|
|
|| ( NetItemRef->GetNet() != GetItem( idxtoTest )->GetNet() ) )
|
2008-03-20 01:50:21 +00:00
|
|
|
{
|
2013-09-27 12:30:35 +00:00
|
|
|
/* Net analysis to update m_ConnectionType */
|
2009-07-12 15:29:42 +00:00
|
|
|
NetEnd = idxtoTest;
|
2007-09-20 21:06:49 +00:00
|
|
|
|
2013-09-27 12:30:35 +00:00
|
|
|
/* set m_ConnectionType member to StateFlag for all items of
|
2009-11-04 20:46:53 +00:00
|
|
|
* this net: */
|
2009-07-12 15:29:42 +00:00
|
|
|
for( unsigned kk = NetStart; kk < NetEnd; kk++ )
|
2013-09-27 12:30:35 +00:00
|
|
|
GetItem( kk )->m_ConnectionType = StateFlag;
|
2007-09-20 21:06:49 +00:00
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
if( idxtoTest >= size() )
|
2007-09-20 21:06:49 +00:00
|
|
|
return;
|
|
|
|
|
2009-07-12 15:29:42 +00:00
|
|
|
/* Start Analysis next Net */
|
2008-12-10 16:49:53 +00:00
|
|
|
StateFlag = UNCONNECTED;
|
2009-07-12 15:29:42 +00:00
|
|
|
NetStart = idxtoTest;
|
2007-09-20 21:06:49 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2009-11-04 20:46:53 +00:00
|
|
|
/* test the current item: if this is a pin and if the reference item
|
|
|
|
* is also a pin, then 2 pins are connected, so set StateFlag to
|
|
|
|
* PAD_CONNECT (can be already done) Of course, if the current
|
|
|
|
* item is a no connect symbol, set StateFlag to
|
|
|
|
* NOCONNECT_SYMBOL_PRESENT to inhibit error diags. However if
|
|
|
|
* StateFlag is already set to PAD_CONNECT this state is kept (the
|
|
|
|
* no connect symbol was surely an error and an ERC will report this)
|
2009-07-12 15:29:42 +00:00
|
|
|
*/
|
|
|
|
for( ; ; idxtoTest++ )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2013-09-25 19:09:57 +00:00
|
|
|
if( ( idxtoTest >= size() )
|
|
|
|
|| ( NetItemRef->GetNet() != GetItem( idxtoTest )->GetNet() ) )
|
2007-09-20 21:06:49 +00:00
|
|
|
break;
|
|
|
|
|
2013-09-25 19:09:57 +00:00
|
|
|
switch( GetItem( idxtoTest )->m_Type )
|
2007-09-20 21:06:49 +00:00
|
|
|
{
|
2009-07-12 15:29:42 +00:00
|
|
|
case NET_ITEM_UNSPECIFIED:
|
2009-11-04 20:46:53 +00:00
|
|
|
wxMessageBox( wxT( "BuildNetListBase() error" ) );
|
2009-07-12 15:29:42 +00:00
|
|
|
break;
|
2009-11-04 20:46:53 +00:00
|
|
|
|
2007-09-20 21:06:49 +00:00
|
|
|
case NET_SEGMENT:
|
|
|
|
case NET_LABEL:
|
2008-02-12 21:12:46 +00:00
|
|
|
case NET_HIERLABEL:
|
2008-03-20 01:50:21 +00:00
|
|
|
case NET_GLOBLABEL:
|
2007-09-20 21:06:49 +00:00
|
|
|
case NET_SHEETLABEL:
|
|
|
|
case NET_PINLABEL:
|
|
|
|
case NET_BUS:
|
|
|
|
case NET_BUSLABELMEMBER:
|
|
|
|
case NET_SHEETBUSLABELMEMBER:
|
2008-02-12 21:12:46 +00:00
|
|
|
case NET_HIERBUSLABELMEMBER:
|
2008-03-20 01:50:21 +00:00
|
|
|
case NET_GLOBBUSLABELMEMBER:
|
2011-10-07 14:41:30 +00:00
|
|
|
case NET_JUNCTION:
|
2007-09-20 21:06:49 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case NET_PIN:
|
|
|
|
if( NetItemRef->m_Type == NET_PIN )
|
2008-01-05 17:30:56 +00:00
|
|
|
StateFlag = PAD_CONNECT;
|
2011-10-07 14:41:30 +00:00
|
|
|
|
2007-09-20 21:06:49 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case NET_NOCONNECT:
|
2008-01-05 17:30:56 +00:00
|
|
|
if( StateFlag != PAD_CONNECT )
|
2009-07-12 15:29:42 +00:00
|
|
|
StateFlag = NOCONNECT_SYMBOL_PRESENT;
|
2011-10-07 14:41:30 +00:00
|
|
|
|
2007-09-20 21:06:49 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
2013-09-25 19:09:57 +00:00
|
|
|
|