KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
drc_test_provider.cpp
Go to the documentation of this file.
1/*
2 * This program source code file is part of KiCad, a free EDA CAD application.
3 *
4 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
5 *
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License
8 * as published by the Free Software Foundation; either version 2
9 * of the License, or (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, you may find one here:
18 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
19 * or you may search the http://www.gnu.org website for the version 2 license,
20 * or you may write to the Free Software Foundation, Inc.,
21 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
22 */
23
24#include "pcb_table.h"
25#include <drc/drc_item.h>
27#include <pcb_track.h>
28#include <footprint.h>
29#include <pad.h>
30#include <zone.h>
31#include <pcb_text.h>
32
33
34// A list of all basic (ie: non-compound) board geometry items
35std::vector<KICAD_T> DRC_TEST_PROVIDER::s_allBasicItems;
37
38
40{
41 for( DRC_TEST_PROVIDER* provider : m_providers )
42 delete provider;
43}
44
45
48 m_drcEngine( nullptr )
49{
50}
51
52
54{
55 if( s_allBasicItems.size() == 0 )
56 {
57 for( int i = 0; i < MAX_STRUCT_TYPE_ID; i++ )
58 {
59 if( i != PCB_FOOTPRINT_T && i != PCB_GROUP_T )
60 {
61 s_allBasicItems.push_back( (KICAD_T) i );
62
63 if( i != PCB_ZONE_T )
64 s_allBasicItemsButZones.push_back( (KICAD_T) i );
65 }
66 }
67 }
68}
69
70
71const wxString DRC_TEST_PROVIDER::GetName() const { return wxT( "<no name test>" ); }
72const wxString DRC_TEST_PROVIDER::GetDescription() const { return wxEmptyString; }
73
74
75void DRC_TEST_PROVIDER::reportViolation( std::shared_ptr<DRC_ITEM>& item,
76 const VECTOR2I& aMarkerPos, int aMarkerLayer,
77 DRC_CUSTOM_MARKER_HANDLER* aCustomHandler )
78{
79 std::lock_guard<std::mutex> lock( m_statsMutex );
80 if( item->GetViolatingRule() )
81 accountCheck( item->GetViolatingRule() );
82
83 item->SetViolatingTest( this );
84 m_drcEngine->ReportViolation( item, aMarkerPos, aMarkerLayer, aCustomHandler );
85}
86
87
88bool DRC_TEST_PROVIDER::reportProgress( size_t aCount, size_t aSize, size_t aDelta )
89{
90 if( ( aCount % aDelta ) == 0 || aCount == aSize - 1 )
91 {
92 if( !m_drcEngine->ReportProgress( static_cast<double>( aCount ) / aSize ) )
93 return false;
94 }
95
96 return true;
97}
98
99
100bool DRC_TEST_PROVIDER::reportPhase( const wxString& aMessage )
101{
102 reportAux( aMessage );
103 return m_drcEngine->ReportPhase( aMessage );
104}
105
106
107void DRC_TEST_PROVIDER::reportAux( const wxChar* fmt, ... )
108{
109 va_list vargs;
110 va_start( vargs, fmt );
111 wxString str;
112 str.PrintfV( fmt, vargs );
113 va_end( vargs );
114 m_drcEngine->ReportAux( str );
115}
116
117
119{
120 auto it = m_stats.find( ruleToTest );
121
122 if( it == m_stats.end() )
123 m_stats[ ruleToTest ] = 1;
124 else
125 m_stats[ ruleToTest ] += 1;
126}
127
128
129void DRC_TEST_PROVIDER::accountCheck( const DRC_CONSTRAINT& constraintToTest )
130{
131 accountCheck( constraintToTest.GetParentRule() );
132}
133
134
136{
137 if( !m_isRuleDriven )
138 return;
139
140 m_drcEngine->ReportAux( wxT( "Rule hit statistics: " ) );
141
142 for( const std::pair<const DRC_RULE* const, int>& stat : m_stats )
143 {
144 if( stat.first )
145 {
146 m_drcEngine->ReportAux( wxString::Format( wxT( " - rule '%s': %d hits " ),
147 stat.first->m_Name,
148 stat.second ) );
149 }
150 }
151}
152
153
154int DRC_TEST_PROVIDER::forEachGeometryItem( const std::vector<KICAD_T>& aTypes, const LSET& aLayers,
155 const std::function<bool( BOARD_ITEM*)>& aFunc )
156{
157 BOARD *brd = m_drcEngine->GetBoard();
158 std::bitset<MAX_STRUCT_TYPE_ID> typeMask;
159 int n = 0;
160
161 if( aTypes.size() == 0 )
162 {
163 for( int i = 0; i < MAX_STRUCT_TYPE_ID; i++ )
164 typeMask[ i ] = true;
165 }
166 else
167 {
168 for( KICAD_T aType : aTypes )
169 typeMask[ aType ] = true;
170 }
171
172 for( PCB_TRACK* item : brd->Tracks() )
173 {
174 if( (item->GetLayerSet() & aLayers).any() )
175 {
176 if( typeMask[ PCB_TRACE_T ] && item->Type() == PCB_TRACE_T )
177 {
178 aFunc( item );
179 n++;
180 }
181 else if( typeMask[ PCB_VIA_T ] && item->Type() == PCB_VIA_T )
182 {
183 aFunc( item );
184 n++;
185 }
186 else if( typeMask[ PCB_ARC_T ] && item->Type() == PCB_ARC_T )
187 {
188 aFunc( item );
189 n++;
190 }
191 }
192 }
193
194 for( BOARD_ITEM* item : brd->Drawings() )
195 {
196 if( (item->GetLayerSet() & aLayers).any() )
197 {
198 if( typeMask[ PCB_DIMENSION_T ] && BaseType( item->Type() ) == PCB_DIMENSION_T )
199 {
200 if( !aFunc( item ) )
201 return n;
202
203 n++;
204 }
205 else if( typeMask[ PCB_SHAPE_T ] && item->Type() == PCB_SHAPE_T )
206 {
207 if( !aFunc( item ) )
208 return n;
209
210 n++;
211 }
212 else if( typeMask[ PCB_TEXT_T ] && item->Type() == PCB_TEXT_T )
213 {
214 if( !aFunc( item ) )
215 return n;
216
217 n++;
218 }
219 else if( typeMask[ PCB_TEXTBOX_T ] && item->Type() == PCB_TEXTBOX_T )
220 {
221 if( !aFunc( item ) )
222 return n;
223
224 n++;
225 }
226 else if( typeMask[ PCB_TARGET_T ] && item->Type() == PCB_TARGET_T )
227 {
228 if( !aFunc( item ) )
229 return n;
230
231 n++;
232 }
233 else if( item->Type() == PCB_TABLE_T )
234 {
235 if( typeMask[ PCB_TABLE_T ] )
236 {
237 if( !aFunc( item ) )
238 return n;
239
240 n++;
241 }
242
243 if( typeMask[ PCB_TABLECELL_T ] )
244 {
245 for( PCB_TABLECELL* cell : static_cast<PCB_TABLE*>( item )->GetCells() )
246 {
247 if( !aFunc( cell ) )
248 return n;
249
250 n++;
251 }
252 }
253 }
254 }
255 }
256
257 if( typeMask[ PCB_ZONE_T ] )
258 {
259 for( ZONE* item : brd->Zones() )
260 {
261 if( ( item->GetLayerSet() & aLayers ).any() )
262 {
263 if( !aFunc( item ) )
264 return n;
265
266 n++;
267 }
268 }
269 }
270
271 for( FOOTPRINT* footprint : brd->Footprints() )
272 {
273 if( typeMask[ PCB_FIELD_T ] )
274 {
275 for( PCB_FIELD* field : footprint->GetFields() )
276 {
277 if( ( field->GetLayerSet() & aLayers ).any() )
278 {
279 if( !aFunc( field ) )
280 return n;
281
282 n++;
283 }
284 }
285 }
286
287 if( typeMask[ PCB_PAD_T ] )
288 {
289 for( PAD* pad : footprint->Pads() )
290 {
291 // Careful: if a pad has a hole then it pierces all layers
292 if( pad->HasHole() || ( pad->GetLayerSet() & aLayers ).any() )
293 {
294 if( !aFunc( pad ) )
295 return n;
296
297 n++;
298 }
299 }
300 }
301
302 for( BOARD_ITEM* dwg : footprint->GraphicalItems() )
303 {
304 if( (dwg->GetLayerSet() & aLayers).any() )
305 {
306 if( typeMask[ PCB_DIMENSION_T ] && BaseType( dwg->Type() ) == PCB_DIMENSION_T )
307 {
308 if( !aFunc( dwg ) )
309 return n;
310
311 n++;
312 }
313 else if( typeMask[ PCB_TEXT_T ] && dwg->Type() == PCB_TEXT_T )
314 {
315 if( !aFunc( dwg ) )
316 return n;
317
318 n++;
319 }
320 else if( typeMask[ PCB_TEXTBOX_T ] && dwg->Type() == PCB_TEXTBOX_T )
321 {
322 if( !aFunc( dwg ) )
323 return n;
324
325 n++;
326 }
327 else if( typeMask[ PCB_SHAPE_T ] && dwg->Type() == PCB_SHAPE_T )
328 {
329 if( !aFunc( dwg ) )
330 return n;
331
332 n++;
333 }
334 }
335 }
336
337 if( typeMask[ PCB_ZONE_T ] )
338 {
339 for( ZONE* zone : footprint->Zones() )
340 {
341 if( (zone->GetLayerSet() & aLayers).any() )
342 {
343 if( !aFunc( zone ) )
344 return n;
345
346 n++;
347 }
348 }
349 }
350
351 if( typeMask[ PCB_FOOTPRINT_T ] )
352 {
353 if( !aFunc( footprint ) )
354 return n;
355
356 n++;
357 }
358 }
359
360 return n;
361}
362
363
365{
366 if( const PCB_FIELD* field = dynamic_cast<const PCB_FIELD*>( aItem ) )
367 {
368 if( !field->IsVisible() )
369 return true;
370 }
371
372 return false;
373}
374
375
376wxString DRC_TEST_PROVIDER::formatMsg( const wxString& aFormatString, const wxString& aSource,
377 double aConstraint, double aActual )
378{
379 wxString constraint_str = MessageTextFromValue( aConstraint );
380 wxString actual_str = MessageTextFromValue( aActual );
381
382 if( constraint_str == actual_str )
383 {
384 // Use more precise formatting if the message-text strings were equal.
385 constraint_str = StringFromValue( aConstraint, true );
386 actual_str = StringFromValue( aActual, true );
387 }
388
389 return wxString::Format( aFormatString, aSource, constraint_str, actual_str );
390}
391
392wxString DRC_TEST_PROVIDER::formatMsg( const wxString& aFormatString, const wxString& aSource,
393 const EDA_ANGLE& aConstraint, const EDA_ANGLE& aActual )
394{
395 wxString constraint_str = MessageTextFromValue( aConstraint );
396 wxString actual_str = MessageTextFromValue( aActual );
397
398 if( constraint_str == actual_str )
399 {
400 // Use more precise formatting if the message-text strings were equal.
401 constraint_str = StringFromValue( aConstraint, true );
402 actual_str = StringFromValue( aActual, true );
403 }
404
405 return wxString::Format( aFormatString, aSource, constraint_str, actual_str );
406}
constexpr EDA_IU_SCALE pcbIUScale
Definition: base_units.h:108
A base class for any item which can be embedded within the BOARD container class, and therefore insta...
Definition: board_item.h:78
Information pertinent to a Pcbnew printed circuit board.
Definition: board.h:297
const ZONES & Zones() const
Definition: board.h:342
const FOOTPRINTS & Footprints() const
Definition: board.h:338
const TRACKS & Tracks() const
Definition: board.h:336
const DRAWINGS & Drawings() const
Definition: board.h:340
DRC_RULE * GetParentRule() const
Definition: drc_rule.h:156
void ReportViolation(const std::shared_ptr< DRC_ITEM > &aItem, const VECTOR2I &aPos, int aMarkerLayer, DRC_CUSTOM_MARKER_HANDLER *aCustomHandler=nullptr)
BOARD * GetBoard() const
Definition: drc_engine.h:96
bool ReportProgress(double aProgress)
void ReportAux(const wxString &aStr)
bool ReportPhase(const wxString &aMessage)
std::vector< DRC_TEST_PROVIDER * > m_providers
Represent a DRC "provider" which runs some DRC functions over a BOARD and spits out DRC_ITEM and posi...
wxString formatMsg(const wxString &aFormatString, const wxString &aSource, double aConstraint, double aActual)
static std::vector< KICAD_T > s_allBasicItemsButZones
virtual bool reportPhase(const wxString &aStageName)
virtual void reportViolation(std::shared_ptr< DRC_ITEM > &item, const VECTOR2I &aMarkerPos, int aMarkerLayer, DRC_CUSTOM_MARKER_HANDLER *aCustomHandler=nullptr)
int forEachGeometryItem(const std::vector< KICAD_T > &aTypes, const LSET &aLayers, const std::function< bool(BOARD_ITEM *)> &aFunc)
static std::vector< KICAD_T > s_allBasicItems
virtual const wxString GetDescription() const
DRC_ENGINE * m_drcEngine
bool isInvisibleText(const BOARD_ITEM *aItem) const
std::unordered_map< const DRC_RULE *, int > m_stats
void reportAux(const wxString &aMsg)
virtual void accountCheck(const DRC_RULE *ruleToTest)
virtual const wxString GetName() const
virtual void reportRuleStatistics()
virtual bool reportProgress(size_t aCount, size_t aSize, size_t aDelta=1)
LSET is a set of PCB_LAYER_IDs.
Definition: lset.h:37
Definition: pad.h:54
wxString MessageTextFromValue(double aValue, bool aAddUnitLabel=true, EDA_DATA_TYPE aType=EDA_DATA_TYPE::DISTANCE) const
A lower-precision version of StringFromValue().
wxString StringFromValue(double aValue, bool aAddUnitLabel=false, EDA_DATA_TYPE aType=EDA_DATA_TYPE::DISTANCE) const
Converts aValue in internal units into a united string.
Handle a list of polygons defining a copper zone.
Definition: zone.h:74
std::function< void(PCB_MARKER *aMarker)> DRC_CUSTOM_MARKER_HANDLER
Definition: drc_engine.h:69
EDA_UNITS
Definition: eda_units.h:46
constexpr KICAD_T BaseType(const KICAD_T aType)
Return the underlying type of the given type.
Definition: typeinfo.h:250
KICAD_T
The set of class identification values stored in EDA_ITEM::m_structType.
Definition: typeinfo.h:78
@ PCB_SHAPE_T
class PCB_SHAPE, a segment not on copper layers
Definition: typeinfo.h:88
@ PCB_VIA_T
class PCB_VIA, a via (like a track segment on a copper layer)
Definition: typeinfo.h:97
@ MAX_STRUCT_TYPE_ID
Definition: typeinfo.h:238
@ PCB_GROUP_T
class PCB_GROUP, a set of BOARD_ITEMs
Definition: typeinfo.h:110
@ PCB_TEXTBOX_T
class PCB_TEXTBOX, wrapped text on a layer
Definition: typeinfo.h:93
@ PCB_ZONE_T
class ZONE, a copper pour area
Definition: typeinfo.h:107
@ PCB_TEXT_T
class PCB_TEXT, text on a layer
Definition: typeinfo.h:92
@ PCB_FIELD_T
class PCB_FIELD, text associated with a footprint property
Definition: typeinfo.h:90
@ PCB_TARGET_T
class PCB_TARGET, a target (graphic item)
Definition: typeinfo.h:106
@ PCB_TABLECELL_T
class PCB_TABLECELL, PCB_TEXTBOX for use in tables
Definition: typeinfo.h:95
@ PCB_FOOTPRINT_T
class FOOTPRINT, a footprint
Definition: typeinfo.h:86
@ PCB_PAD_T
class PAD, a pad in a footprint
Definition: typeinfo.h:87
@ PCB_ARC_T
class PCB_ARC, an arc track segment on a copper layer
Definition: typeinfo.h:98
@ PCB_DIMENSION_T
class PCB_DIMENSION_BASE: abstract dimension meta-type
Definition: typeinfo.h:100
@ PCB_TABLE_T
class PCB_TABLE, table of PCB_TABLECELLs
Definition: typeinfo.h:94
@ PCB_TRACE_T
class PCB_TRACK, a track segment (segment on a copper layer)
Definition: typeinfo.h:96