2020-09-13 10:37:20 +00:00
|
|
|
/*
|
|
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
|
|
*
|
2024-01-04 11:49:06 +00:00
|
|
|
* Copyright (C) 2020-2024 KiCad Developers, see AUTHORS.txt for contributors.
|
2020-09-13 10:37:20 +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
|
|
|
|
*/
|
|
|
|
|
2020-09-11 13:41:45 +00:00
|
|
|
#include <drc/drc_engine.h>
|
|
|
|
#include <drc/drc_item.h>
|
2020-09-11 15:40:36 +00:00
|
|
|
#include <drc/drc_test_provider.h>
|
2021-06-11 21:07:02 +00:00
|
|
|
#include <pcb_track.h>
|
2020-11-12 20:19:22 +00:00
|
|
|
#include <footprint.h>
|
|
|
|
#include <pad.h>
|
2020-11-11 23:05:59 +00:00
|
|
|
#include <zone.h>
|
2020-10-04 23:34:59 +00:00
|
|
|
#include <pcb_text.h>
|
2020-06-13 23:28:08 +00:00
|
|
|
|
2020-10-25 20:24:47 +00:00
|
|
|
|
|
|
|
// A list of all basic (ie: non-compound) board geometry items
|
|
|
|
std::vector<KICAD_T> DRC_TEST_PROVIDER::s_allBasicItems;
|
2020-10-27 17:09:27 +00:00
|
|
|
std::vector<KICAD_T> DRC_TEST_PROVIDER::s_allBasicItemsButZones;
|
2020-10-25 20:24:47 +00:00
|
|
|
|
|
|
|
|
2020-12-29 19:05:09 +00:00
|
|
|
DRC_TEST_PROVIDER_REGISTRY::~DRC_TEST_PROVIDER_REGISTRY()
|
|
|
|
{
|
|
|
|
for( DRC_TEST_PROVIDER* provider : m_providers )
|
|
|
|
delete provider;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-09-11 15:04:11 +00:00
|
|
|
DRC_TEST_PROVIDER::DRC_TEST_PROVIDER() :
|
2022-09-19 16:09:59 +00:00
|
|
|
UNITS_PROVIDER( pcbIUScale, EDA_UNITS::MILLIMETRES ),
|
|
|
|
m_drcEngine( nullptr )
|
2020-06-13 23:28:08 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2020-09-12 19:28:22 +00:00
|
|
|
|
2022-03-11 20:13:47 +00:00
|
|
|
void DRC_TEST_PROVIDER::Init()
|
|
|
|
{
|
|
|
|
if( s_allBasicItems.size() == 0 )
|
|
|
|
{
|
|
|
|
for( int i = 0; i < MAX_STRUCT_TYPE_ID; i++ )
|
|
|
|
{
|
|
|
|
if( i != PCB_FOOTPRINT_T && i != PCB_GROUP_T )
|
|
|
|
{
|
|
|
|
s_allBasicItems.push_back( (KICAD_T) i );
|
|
|
|
|
2023-03-30 11:49:23 +00:00
|
|
|
if( i != PCB_ZONE_T )
|
2022-03-11 20:13:47 +00:00
|
|
|
s_allBasicItemsButZones.push_back( (KICAD_T) i );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-03-11 21:16:52 +00:00
|
|
|
const wxString DRC_TEST_PROVIDER::GetName() const { return wxT( "<no name test>" ); }
|
2022-03-13 16:09:06 +00:00
|
|
|
const wxString DRC_TEST_PROVIDER::GetDescription() const { return wxEmptyString; }
|
2020-06-13 23:28:08 +00:00
|
|
|
|
2020-09-12 19:28:22 +00:00
|
|
|
|
2021-07-26 19:35:12 +00:00
|
|
|
void DRC_TEST_PROVIDER::reportViolation( std::shared_ptr<DRC_ITEM>& item,
|
2022-08-15 16:59:34 +00:00
|
|
|
const VECTOR2I& aMarkerPos, int aMarkerLayer )
|
2020-06-17 22:36:54 +00:00
|
|
|
{
|
2023-09-21 02:50:16 +00:00
|
|
|
std::lock_guard<std::mutex> lock( m_statsMutex );
|
2021-01-23 00:09:18 +00:00
|
|
|
if( item->GetViolatingRule() )
|
|
|
|
accountCheck( item->GetViolatingRule() );
|
|
|
|
|
2020-09-14 17:54:14 +00:00
|
|
|
item->SetViolatingTest( this );
|
2021-12-26 13:47:00 +00:00
|
|
|
m_drcEngine->ReportViolation( item, aMarkerPos, aMarkerLayer );
|
2020-06-17 22:36:54 +00:00
|
|
|
}
|
|
|
|
|
2020-09-12 19:28:22 +00:00
|
|
|
|
2024-01-04 11:49:06 +00:00
|
|
|
bool DRC_TEST_PROVIDER::reportProgress( size_t aCount, size_t aSize, size_t aDelta )
|
2020-06-17 22:36:54 +00:00
|
|
|
{
|
2020-09-18 19:57:54 +00:00
|
|
|
if( ( aCount % aDelta ) == 0 || aCount == aSize - 1 )
|
|
|
|
{
|
2024-01-04 11:49:06 +00:00
|
|
|
if( !m_drcEngine->ReportProgress( static_cast<double>( aCount ) / aSize ) )
|
2020-09-18 19:57:54 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
2020-06-17 22:36:54 +00:00
|
|
|
}
|
|
|
|
|
2020-09-12 19:28:22 +00:00
|
|
|
|
2020-09-18 19:57:54 +00:00
|
|
|
bool DRC_TEST_PROVIDER::reportPhase( const wxString& aMessage )
|
2020-06-17 22:36:54 +00:00
|
|
|
{
|
2020-09-15 11:06:15 +00:00
|
|
|
reportAux( aMessage );
|
2020-09-18 19:57:54 +00:00
|
|
|
return m_drcEngine->ReportPhase( aMessage );
|
2020-06-18 16:55:22 +00:00
|
|
|
}
|
2020-06-17 22:36:54 +00:00
|
|
|
|
2020-09-12 19:28:22 +00:00
|
|
|
|
2023-04-16 11:35:00 +00:00
|
|
|
void DRC_TEST_PROVIDER::reportAux( const wxChar* fmt, ... )
|
2020-06-18 16:55:22 +00:00
|
|
|
{
|
|
|
|
va_list vargs;
|
|
|
|
va_start( vargs, fmt );
|
|
|
|
wxString str;
|
|
|
|
str.PrintfV( fmt, vargs );
|
|
|
|
va_end( vargs );
|
|
|
|
m_drcEngine->ReportAux( str );
|
2020-06-17 22:36:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-09-11 15:04:11 +00:00
|
|
|
void DRC_TEST_PROVIDER::accountCheck( const DRC_RULE* ruleToTest )
|
2020-06-17 22:36:54 +00:00
|
|
|
{
|
|
|
|
auto it = m_stats.find( ruleToTest );
|
2020-09-11 15:04:11 +00:00
|
|
|
|
2020-06-17 22:36:54 +00:00
|
|
|
if( it == m_stats.end() )
|
2020-07-29 22:02:36 +00:00
|
|
|
m_stats[ ruleToTest ] = 1;
|
2020-06-17 22:36:54 +00:00
|
|
|
else
|
2020-09-14 17:54:14 +00:00
|
|
|
m_stats[ ruleToTest ] += 1;
|
2020-06-17 22:36:54 +00:00
|
|
|
}
|
2020-07-27 13:32:37 +00:00
|
|
|
|
2020-09-12 19:28:22 +00:00
|
|
|
|
2020-09-11 15:04:11 +00:00
|
|
|
void DRC_TEST_PROVIDER::accountCheck( const DRC_CONSTRAINT& constraintToTest )
|
2020-08-11 22:17:43 +00:00
|
|
|
{
|
|
|
|
accountCheck( constraintToTest.GetParentRule() );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-09-11 15:04:11 +00:00
|
|
|
void DRC_TEST_PROVIDER::reportRuleStatistics()
|
2020-07-27 13:32:37 +00:00
|
|
|
{
|
2020-09-07 19:06:31 +00:00
|
|
|
if( !m_isRuleDriven )
|
|
|
|
return;
|
|
|
|
|
2022-03-11 21:16:52 +00:00
|
|
|
m_drcEngine->ReportAux( wxT( "Rule hit statistics: " ) );
|
2020-09-11 15:04:11 +00:00
|
|
|
|
2020-09-12 19:28:22 +00:00
|
|
|
for( const std::pair<const DRC_RULE* const, int>& stat : m_stats )
|
2020-07-27 13:32:37 +00:00
|
|
|
{
|
2020-09-14 17:54:14 +00:00
|
|
|
if( stat.first )
|
|
|
|
{
|
2022-03-11 21:16:52 +00:00
|
|
|
m_drcEngine->ReportAux( wxString::Format( wxT( " - rule '%s': %d hits " ),
|
2020-09-14 17:54:14 +00:00
|
|
|
stat.first->m_Name,
|
|
|
|
stat.second ) );
|
|
|
|
}
|
2020-07-27 13:32:37 +00:00
|
|
|
}
|
|
|
|
}
|
2020-08-11 22:15:50 +00:00
|
|
|
|
2020-09-12 19:28:22 +00:00
|
|
|
|
|
|
|
int DRC_TEST_PROVIDER::forEachGeometryItem( const std::vector<KICAD_T>& aTypes, LSET aLayers,
|
|
|
|
const std::function<bool( BOARD_ITEM*)>& aFunc )
|
2020-08-11 22:15:50 +00:00
|
|
|
{
|
|
|
|
BOARD *brd = m_drcEngine->GetBoard();
|
|
|
|
std::bitset<MAX_STRUCT_TYPE_ID> typeMask;
|
|
|
|
int n = 0;
|
|
|
|
|
|
|
|
if( aTypes.size() == 0 )
|
|
|
|
{
|
|
|
|
for( int i = 0; i < MAX_STRUCT_TYPE_ID; i++ )
|
2020-09-12 19:28:22 +00:00
|
|
|
typeMask[ i ] = true;
|
2020-08-11 22:15:50 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-09-12 19:28:22 +00:00
|
|
|
for( KICAD_T aType : aTypes )
|
|
|
|
typeMask[ aType ] = true;
|
2020-08-11 22:15:50 +00:00
|
|
|
}
|
|
|
|
|
2021-06-11 21:07:02 +00:00
|
|
|
for( PCB_TRACK* item : brd->Tracks() )
|
2020-08-11 22:15:50 +00:00
|
|
|
{
|
2020-08-25 17:42:52 +00:00
|
|
|
if( (item->GetLayerSet() & aLayers).any() )
|
2020-08-11 22:15:50 +00:00
|
|
|
{
|
2020-08-25 17:42:52 +00:00
|
|
|
if( typeMask[ PCB_TRACE_T ] && item->Type() == PCB_TRACE_T )
|
|
|
|
{
|
|
|
|
aFunc( item );
|
|
|
|
n++;
|
|
|
|
}
|
|
|
|
else if( typeMask[ PCB_VIA_T ] && item->Type() == PCB_VIA_T )
|
|
|
|
{
|
|
|
|
aFunc( item );
|
|
|
|
n++;
|
|
|
|
}
|
|
|
|
else if( typeMask[ PCB_ARC_T ] && item->Type() == PCB_ARC_T )
|
|
|
|
{
|
|
|
|
aFunc( item );
|
|
|
|
n++;
|
|
|
|
}
|
2020-08-11 22:15:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-09-11 15:04:11 +00:00
|
|
|
for( BOARD_ITEM* item : brd->Drawings() )
|
2020-08-11 22:15:50 +00:00
|
|
|
{
|
2020-08-25 17:42:52 +00:00
|
|
|
if( (item->GetLayerSet() & aLayers).any() )
|
2020-08-11 22:15:50 +00:00
|
|
|
{
|
2021-12-04 23:52:00 +00:00
|
|
|
if( typeMask[ PCB_DIMENSION_T ] && BaseType( item->Type() ) == PCB_DIMENSION_T )
|
2020-08-25 17:42:52 +00:00
|
|
|
{
|
|
|
|
if( !aFunc( item ) )
|
|
|
|
return n;
|
|
|
|
|
|
|
|
n++;
|
|
|
|
}
|
2020-10-04 14:19:33 +00:00
|
|
|
else if( typeMask[ PCB_SHAPE_T ] && item->Type() == PCB_SHAPE_T )
|
2020-08-25 17:42:52 +00:00
|
|
|
{
|
|
|
|
if( !aFunc( item ) )
|
|
|
|
return n;
|
|
|
|
|
|
|
|
n++;
|
|
|
|
}
|
2022-03-09 00:47:37 +00:00
|
|
|
else if( typeMask[ PCB_TEXT_T ] && item->Type() == PCB_TEXT_T )
|
|
|
|
{
|
|
|
|
if( !aFunc( item ) )
|
|
|
|
return n;
|
|
|
|
|
|
|
|
n++;
|
|
|
|
}
|
|
|
|
else if( typeMask[ PCB_TEXTBOX_T ] && item->Type() == PCB_TEXTBOX_T )
|
2020-08-25 17:42:52 +00:00
|
|
|
{
|
|
|
|
if( !aFunc( item ) )
|
|
|
|
return n;
|
|
|
|
|
|
|
|
n++;
|
|
|
|
}
|
|
|
|
else if( typeMask[ PCB_TARGET_T ] && item->Type() == PCB_TARGET_T )
|
|
|
|
{
|
|
|
|
if( !aFunc( item ) )
|
|
|
|
return n;
|
|
|
|
|
|
|
|
n++;
|
|
|
|
}
|
2020-08-11 22:15:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-11 23:05:59 +00:00
|
|
|
if( typeMask[ PCB_ZONE_T ] )
|
2020-08-11 22:15:50 +00:00
|
|
|
{
|
2020-11-11 23:05:59 +00:00
|
|
|
for( ZONE* item : brd->Zones() )
|
2020-08-11 22:15:50 +00:00
|
|
|
{
|
2020-11-12 23:50:33 +00:00
|
|
|
if( ( item->GetLayerSet() & aLayers ).any() )
|
2020-08-25 17:42:52 +00:00
|
|
|
{
|
|
|
|
if( !aFunc( item ) )
|
|
|
|
return n;
|
|
|
|
|
|
|
|
n++;
|
|
|
|
}
|
2020-08-11 22:15:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-13 15:15:52 +00:00
|
|
|
for( FOOTPRINT* footprint : brd->Footprints() )
|
2020-08-11 22:15:50 +00:00
|
|
|
{
|
2023-06-24 14:48:27 +00:00
|
|
|
if( typeMask[ PCB_FIELD_T ] )
|
2020-08-11 22:15:50 +00:00
|
|
|
{
|
2023-06-15 19:37:07 +00:00
|
|
|
for( PCB_FIELD* field : footprint->GetFields() )
|
2020-08-25 17:42:52 +00:00
|
|
|
{
|
2023-06-15 19:37:07 +00:00
|
|
|
if( ( field->GetLayerSet() & aLayers ).any() )
|
|
|
|
{
|
|
|
|
if( !aFunc( field ) )
|
|
|
|
return n;
|
2020-08-25 17:42:52 +00:00
|
|
|
|
2023-06-15 19:37:07 +00:00
|
|
|
n++;
|
|
|
|
}
|
2020-08-25 17:42:52 +00:00
|
|
|
}
|
2020-08-11 22:15:50 +00:00
|
|
|
}
|
|
|
|
|
2020-10-15 10:36:37 +00:00
|
|
|
if( typeMask[ PCB_PAD_T ] )
|
2020-08-11 22:15:50 +00:00
|
|
|
{
|
2020-11-12 23:50:33 +00:00
|
|
|
for( PAD* pad : footprint->Pads() )
|
2020-08-11 22:15:50 +00:00
|
|
|
{
|
2021-01-31 00:30:40 +00:00
|
|
|
// Careful: if a pad has a hole then it pierces all layers
|
2022-07-15 15:14:11 +00:00
|
|
|
if( pad->HasHole() || ( pad->GetLayerSet() & aLayers ).any() )
|
2020-08-25 17:42:52 +00:00
|
|
|
{
|
|
|
|
if( !aFunc( pad ) )
|
|
|
|
return n;
|
|
|
|
|
|
|
|
n++;
|
|
|
|
}
|
2020-08-11 22:15:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-12 23:50:33 +00:00
|
|
|
for( BOARD_ITEM* dwg : footprint->GraphicalItems() )
|
2020-08-11 22:15:50 +00:00
|
|
|
{
|
2020-08-25 17:42:52 +00:00
|
|
|
if( (dwg->GetLayerSet() & aLayers).any() )
|
2020-08-11 22:15:50 +00:00
|
|
|
{
|
2021-12-04 23:52:00 +00:00
|
|
|
if( typeMask[ PCB_DIMENSION_T ] && BaseType( dwg->Type() ) == PCB_DIMENSION_T )
|
|
|
|
{
|
|
|
|
if( !aFunc( dwg ) )
|
|
|
|
return n;
|
|
|
|
|
|
|
|
n++;
|
|
|
|
}
|
2023-03-30 11:49:23 +00:00
|
|
|
else if( typeMask[ PCB_TEXT_T ] && dwg->Type() == PCB_TEXT_T )
|
2022-03-09 00:47:37 +00:00
|
|
|
{
|
|
|
|
if( !aFunc( dwg ) )
|
|
|
|
return n;
|
|
|
|
|
|
|
|
n++;
|
|
|
|
}
|
2023-03-30 11:49:23 +00:00
|
|
|
else if( typeMask[ PCB_TEXTBOX_T ] && dwg->Type() == PCB_TEXTBOX_T )
|
2020-08-25 17:42:52 +00:00
|
|
|
{
|
|
|
|
if( !aFunc( dwg ) )
|
|
|
|
return n;
|
|
|
|
|
|
|
|
n++;
|
|
|
|
}
|
2023-03-30 11:49:23 +00:00
|
|
|
else if( typeMask[ PCB_SHAPE_T ] && dwg->Type() == PCB_SHAPE_T )
|
2020-08-25 17:42:52 +00:00
|
|
|
{
|
|
|
|
if( !aFunc( dwg ) )
|
|
|
|
return n;
|
|
|
|
|
|
|
|
n++;
|
|
|
|
}
|
2020-08-11 22:15:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-30 11:49:23 +00:00
|
|
|
if( typeMask[ PCB_ZONE_T ] )
|
2020-08-11 22:15:50 +00:00
|
|
|
{
|
2020-11-12 23:50:33 +00:00
|
|
|
for( ZONE* zone : footprint->Zones() )
|
2020-08-11 22:15:50 +00:00
|
|
|
{
|
2020-10-15 10:36:37 +00:00
|
|
|
if( (zone->GetLayerSet() & aLayers).any() )
|
2020-08-25 17:42:52 +00:00
|
|
|
{
|
2020-10-25 15:02:49 +00:00
|
|
|
if( !aFunc( zone ) )
|
2020-08-25 17:42:52 +00:00
|
|
|
return n;
|
|
|
|
|
|
|
|
n++;
|
|
|
|
}
|
2020-08-11 22:15:50 +00:00
|
|
|
}
|
|
|
|
}
|
2020-10-25 15:02:49 +00:00
|
|
|
|
2020-11-13 12:21:02 +00:00
|
|
|
if( typeMask[ PCB_FOOTPRINT_T ] )
|
2020-10-25 15:02:49 +00:00
|
|
|
{
|
2020-11-12 23:50:33 +00:00
|
|
|
if( !aFunc( footprint ) )
|
2020-10-25 15:02:49 +00:00
|
|
|
return n;
|
|
|
|
|
|
|
|
n++;
|
|
|
|
}
|
2020-08-11 22:15:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return n;
|
|
|
|
}
|
2020-10-02 18:58:12 +00:00
|
|
|
|
|
|
|
|
|
|
|
bool DRC_TEST_PROVIDER::isInvisibleText( const BOARD_ITEM* aItem ) const
|
|
|
|
{
|
2023-06-24 14:48:27 +00:00
|
|
|
if( const PCB_TEXT* text = dynamic_cast<const PCB_TEXT*>( aItem ) )
|
2020-10-02 18:58:12 +00:00
|
|
|
{
|
|
|
|
if( !text->IsVisible() )
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
2020-12-29 19:05:09 +00:00
|
|
|
}
|
2022-10-06 20:52:17 +00:00
|
|
|
|
|
|
|
|
|
|
|
wxString DRC_TEST_PROVIDER::formatMsg( const wxString& aFormatString, const wxString& aSource,
|
2023-10-28 12:01:36 +00:00
|
|
|
double aConstraint, double aActual )
|
2022-10-06 20:52:17 +00:00
|
|
|
{
|
|
|
|
wxString constraint_str = MessageTextFromValue( aConstraint );
|
|
|
|
wxString actual_str = MessageTextFromValue( aActual );
|
|
|
|
|
|
|
|
if( constraint_str == actual_str )
|
|
|
|
{
|
|
|
|
// Use more precise formatting if the message-text strings were equal.
|
2023-01-04 00:02:25 +00:00
|
|
|
constraint_str = StringFromValue( aConstraint, true );
|
|
|
|
actual_str = StringFromValue( aActual, true );
|
2022-10-06 20:52:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return wxString::Format( aFormatString, aSource, constraint_str, actual_str );
|
2023-06-06 15:09:34 +00:00
|
|
|
}
|