KiCad PCB EDA Suite
Loading...
Searching...
No Matches
drc_test_provider_physical_clearance.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.
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 <common.h>
25#include <macros.h>
27#include <footprint.h>
28#include <pad.h>
29#include <pcb_track.h>
30#include <pcb_shape.h>
31#include <zone.h>
32#include <advanced_config.h>
34#include <geometry/seg.h>
36#include <drc/drc_rtree.h>
37#include <drc/drc_item.h>
39
40/*
41 Physical clearance tests.
42
43 Errors generated:
44 - DRCE_PHYSICAL_CLEARANCE
45 - DRCE_PHYSICAL_HOLE_CLEARANCE
46*/
47
49{
50public:
54
56
57 virtual bool Run() override;
58
59 virtual const wxString GetName() const override { return wxT( "physical_clearance" ); };
60
61private:
62 int testItemAgainstItem( BOARD_ITEM* aItem, SHAPE* aItemShape, PCB_LAYER_ID aLayer,
63 BOARD_ITEM* aOther );
64
65 void testItemAgainstZones( BOARD_ITEM* aItem, PCB_LAYER_ID aLayer );
66
67 void testShapeLineChain( const SHAPE_LINE_CHAIN& aOutline, int aLineWidth, PCB_LAYER_ID aLayer,
68 BOARD_ITEM* aParentItem, DRC_CONSTRAINT& aConstraint );
69
70 void testZoneLayer( ZONE* aZone, PCB_LAYER_ID aLayer, DRC_CONSTRAINT& aConstraint );
71
72private:
74};
75
76
78{
79 m_board = m_drcEngine->GetBoard();
80 m_itemTree.clear();
81
82 int errorMax = m_board->GetDesignSettings().m_MaxError;
83 LSET boardCopperLayers = LSET::AllCuMask( m_board->GetCopperLayerCount() );
84
85 if( m_board->m_DRCMaxPhysicalClearance <= 0 )
86 {
87 REPORT_AUX( wxT( "No physical clearance constraints found. Tests not run." ) );
88 return true; // continue with other tests
89 }
90
91 size_t progressDelta = 250;
92 size_t count = 0;
93 size_t ii = 0;
94
95 if( !reportPhase( _( "Gathering physical items..." ) ) )
96 return false; // DRC cancelled
97
98 static const std::vector<KICAD_T> itemTypes = {
101 PCB_PAD_T,
107 };
108
109 static const LSET courtyards( { F_CrtYd, B_CrtYd } );
110
111 //
112 // Generate a count for use in progress reporting.
113 //
114
116 [&]( BOARD_ITEM* item ) -> bool
117 {
118 if( isInvisibleText( item ) )
119 return true;
120
121 ++count;
122 return true;
123 } );
124
125 //
126 // Generate a BOARD_ITEM RTree.
127 //
128
130 [&]( BOARD_ITEM* item ) -> bool
131 {
132 if( isInvisibleText( item ) )
133 return true;
134
135 if( !reportProgress( ii++, count, progressDelta ) )
136 return false;
137
138 LSET layers = item->GetLayerSet();
139
140 // Special-case holes and edge-cuts which pierce all physical layers
141 if( item->HasHole() )
142 {
143 if( layers.Contains( F_Cu ) )
144 layers |= LSET( LSET::FrontBoardTechMask() ).set( F_CrtYd );
145
146 if( layers.Contains( B_Cu ) )
147 layers |= LSET( LSET::BackBoardTechMask() ).set( B_CrtYd );
148
149 if( layers.Contains( F_Cu ) && layers.Contains( B_Cu ) )
150 layers |= boardCopperLayers;
151 }
152 else if( item->Type() == PCB_FOOTPRINT_T )
153 {
154 layers = courtyards;
155 }
156 else if( item->IsOnLayer( Edge_Cuts ) )
157 {
158 layers |= LSET::PhysicalLayersMask() | courtyards;
159 }
160
161 for( PCB_LAYER_ID layer : layers )
162 m_itemTree.Insert( item, layer, m_board->m_DRCMaxPhysicalClearance, ATOMIC_TABLES );
163
164 return true;
165 } );
166
167 std::unordered_map<PTR_PTR_CACHE_KEY, LSET> checkedPairs;
168 progressDelta = 100;
169 ii = 0;
170
171 //
172 // Run clearance checks -between- items.
173 //
174
175 if( !m_drcEngine->IsErrorLimitExceeded( DRCE_CLEARANCE )
176 || !m_drcEngine->IsErrorLimitExceeded( DRCE_HOLE_CLEARANCE ) )
177 {
178 if( !reportPhase( _( "Checking physical clearances..." ) ) )
179 return false; // DRC cancelled
180
182 [&]( BOARD_ITEM* item ) -> bool
183 {
184 if( isInvisibleText( item ) )
185 return true;
186
187 if( !reportProgress( ii++, count, progressDelta ) )
188 return false;
189
190 LSET layers = item->GetLayerSet();
191
192 if( item->Type() == PCB_FOOTPRINT_T )
193 layers = courtyards;
194
195 for( PCB_LAYER_ID layer : layers )
196 {
197 std::shared_ptr<SHAPE> itemShape = item->GetEffectiveShape( layer );
198
199 m_itemTree.QueryColliding( item, layer, layer,
200 // Filter:
201 [&]( BOARD_ITEM* other ) -> bool
202 {
203 if( item->Type() == PCB_TABLECELL_T && item->GetParent() == other )
204 return false;
205
206 BOARD_ITEM* a = item;
207 BOARD_ITEM* b = other;
208
209 // store canonical order so we don't collide in both
210 // directions (a:b and b:a)
211 if( static_cast<void*>( a ) > static_cast<void*>( b ) )
212 std::swap( a, b );
213
214 auto it = checkedPairs.find( { a, b } );
215
216 if( it != checkedPairs.end() && it->second.test( layer ) )
217 {
218 return false;
219 }
220 else
221 {
222 checkedPairs[ { a, b } ].set( layer );
223 return true;
224 }
225 },
226 // Visitor:
227 [&]( BOARD_ITEM* other ) -> bool
228 {
229 if( testItemAgainstItem( item, itemShape.get(), layer,
230 other ) > 0 )
231 {
232 BOARD_ITEM* a = item;
233 BOARD_ITEM* b = other;
234
235 // store canonical order
236 if( static_cast<void*>( a ) > static_cast<void*>( b ) )
237 std::swap( a, b );
238
239 // Once we record one DRC for error for physical clearance
240 // we don't need to record more
241 checkedPairs[ { a, b } ].set();
242 }
243
244 return !m_drcEngine->IsCancelled();
245 },
246 m_board->m_DRCMaxPhysicalClearance );
247
248 testItemAgainstZones( item, layer );
249 }
250
251 return true;
252 } );
253 }
254
255 progressDelta = 100;
256 count = 0;
257 ii = 0;
258
259 //
260 // Generate a count for progress reporting.
261 //
262
263 forEachGeometryItem( { PCB_ZONE_T, PCB_SHAPE_T }, boardCopperLayers,
264 [&]( BOARD_ITEM* item ) -> bool
265 {
266 ZONE* zone = dynamic_cast<ZONE*>( item );
267
268 if( zone && zone->GetIsRuleArea() )
269 return true; // Continue with other items
270
271 count += ( item->GetLayerSet() & boardCopperLayers ).count();
272
273 return true;
274 } );
275
276 //
277 // Run clearance checks -within- polygonal items.
278 //
279
280 forEachGeometryItem( { PCB_ZONE_T, PCB_SHAPE_T }, boardCopperLayers,
281 [&]( BOARD_ITEM* item ) -> bool
282 {
283 PCB_SHAPE* shape = dynamic_cast<PCB_SHAPE*>( item );
284 ZONE* zone = dynamic_cast<ZONE*>( item );
285
286 if( zone && zone->GetIsRuleArea() )
287 return true; // Continue with other items
288
289 for( PCB_LAYER_ID layer : item->GetLayerSet() )
290 {
291 if( IsCopperLayer( layer ) )
292 {
293 if( !reportProgress( ii++, count, progressDelta ) )
294 return false;
295
296 DRC_CONSTRAINT c = m_drcEngine->EvalRules( PHYSICAL_CLEARANCE_CONSTRAINT, item, nullptr,
297 layer );
298
299 if( shape )
300 {
301 switch( shape->GetShape() )
302 {
303 case SHAPE_T::POLY:
304 testShapeLineChain( shape->GetPolyShape().Outline( 0 ), shape->GetWidth(), layer,
305 item, c );
306 break;
307
308 case SHAPE_T::BEZIER:
309 {
310 SHAPE_LINE_CHAIN asPoly;
311
312 shape->RebuildBezierToSegmentsPointsList( errorMax );
313
314 for( const VECTOR2I& pt : shape->GetBezierPoints() )
315 asPoly.Append( pt );
316
317 testShapeLineChain( asPoly, shape->GetWidth(), layer, item, c );
318 break;
319 }
320
321 case SHAPE_T::ARC:
322 {
323 SHAPE_LINE_CHAIN asPoly;
324
325 VECTOR2I center = shape->GetCenter();
326 EDA_ANGLE angle = -shape->GetArcAngle();
327 double r = shape->GetRadius();
328 int steps = GetArcToSegmentCount( r, errorMax, angle );
329
330 asPoly.Append( shape->GetStart() );
331
332 for( int step = 1; step <= steps; ++step )
333 {
334 EDA_ANGLE rotation = ( angle * step ) / steps;
335 VECTOR2I pt = shape->GetStart();
336
337 RotatePoint( pt, center, rotation );
338 asPoly.Append( pt );
339 }
340
341 testShapeLineChain( asPoly, shape->GetWidth(), layer, item, c );
342 break;
343 }
344
345 // Simple shapes can't create self-intersections, and I'm not sure a user
346 // would want a report that one side of their rectangle was too close to
347 // the other side.
349 case SHAPE_T::SEGMENT:
350 case SHAPE_T::CIRCLE:
351 break;
352
353 default:
355 }
356 }
357
358 if( zone )
359 testZoneLayer( static_cast<ZONE*>( item ), layer, c );
360 }
361
362 if( m_drcEngine->IsCancelled() )
363 return false;
364 }
365
366 return !m_drcEngine->IsCancelled();
367 } );
368
369 return !m_drcEngine->IsCancelled();
370}
371
372
374 int aLineWidth, PCB_LAYER_ID aLayer,
375 BOARD_ITEM* aParentItem,
376 DRC_CONSTRAINT& aConstraint )
377{
378 // We don't want to collide with neighboring segments forming a curve until the concavity
379 // approaches 180 degrees.
380 double angleTolerance = DEG2RAD( 180.0 - ADVANCED_CFG::GetCfg().m_SliverAngleTolerance );
381 int epsilon = m_board->GetDesignSettings().GetDRCEpsilon();
382 int count = aOutline.SegmentCount();
383 int clearance = aConstraint.GetValue().Min();
384
385 if( aConstraint.GetSeverity() == RPT_SEVERITY_IGNORE || clearance - epsilon <= 0 )
386 return;
387
388 // Trigonometry is not cheap; cache seg angles
389 std::vector<double> angles;
390 angles.reserve( count );
391
392 auto angleDiff =
393 []( double a, double b ) -> double
394 {
395 if( a > b )
396 std::swap( a, b );
397
398 double diff = b - a;
399
400 if( diff > M_PI )
401 return 2 * M_PI - diff;
402 else
403 return diff;
404 };
405
406 for( int ii = 0; ii < count; ++ii )
407 {
408 const SEG& seg = aOutline.CSegment( ii );
409
410 // NB: don't store angles of really short segments (which could point anywhere)
411
412 if( seg.SquaredLength() > SEG::Square( epsilon * 2 ) )
413 {
414 angles.push_back( EDA_ANGLE( seg.B - seg.A ).AsRadians() );
415 }
416 else if( ii > 0 )
417 {
418 angles.push_back( angles.back() );
419 }
420 else
421 {
422 for( int jj = 1; jj < count; ++jj )
423 {
424 const SEG& following = aOutline.CSegment( jj );
425
426 if( following.SquaredLength() > SEG::Square( epsilon * 2 ) || jj == count - 1 )
427 {
428 angles.push_back( EDA_ANGLE( following.B - following.A ).AsRadians() );
429 break;
430 }
431 }
432 }
433 }
434
435 // Find collisions before reporting so that we can condense them into fewer reports.
436 std::vector< std::pair<VECTOR2I, int> > collisions;
437
438 for( int ii = 0; ii < count; ++ii )
439 {
440 const SEG seg = aOutline.CSegment( ii );
441 double segAngle = angles[ ii ];
442
443 // Exclude segments on either side of us until we reach the angle tolerance
444 int firstCandidate = ii + 1;
445 int lastCandidate = count - 1;
446
447 while( firstCandidate < count )
448 {
449 if( angleDiff( segAngle, angles[ firstCandidate ] ) < angleTolerance )
450 firstCandidate++;
451 else
452 break;
453 }
454
455 if( aOutline.IsClosed() )
456 {
457 if( ii > 0 )
458 lastCandidate = ii - 1;
459
460 while( lastCandidate != std::min( firstCandidate, count - 1 ) )
461 {
462 if( angleDiff( segAngle, angles[ lastCandidate ] ) < angleTolerance )
463 lastCandidate = ( lastCandidate == 0 ) ? count - 1 : lastCandidate - 1;
464 else
465 break;
466 }
467 }
468
469 // Now run the collision between seg and each candidate seg in the candidate range.
470 if( lastCandidate < ii )
471 lastCandidate = count - 1;
472
473 for( int jj = firstCandidate; jj <= lastCandidate; ++jj )
474 {
475 const SEG candidate = aOutline.CSegment( jj );
476 int actual;
477
478 if( seg.Collide( candidate, clearance + aLineWidth - epsilon, &actual ) )
479 {
480 VECTOR2I firstPoint = seg.NearestPoint( candidate );
481 VECTOR2I secondPoint = candidate.NearestPoint( seg );
482 VECTOR2I pos = ( firstPoint + secondPoint ) / 2;
483
484 if( !collisions.empty() && ( pos - collisions.back().first ).EuclideanNorm() < clearance * 2 )
485 {
486 if( actual < collisions.back().second )
487 {
488 collisions.back().first = pos;
489 collisions.back().second = actual;
490 }
491
492 continue;
493 }
494
495 collisions.push_back( { pos, actual } );
496 }
497 }
498 }
499
500 for( const std::pair<VECTOR2I, int>& collision : collisions )
501 {
502 std::shared_ptr<DRC_ITEM> drcItem = DRC_ITEM::Create( DRCE_CLEARANCE );
503 VECTOR2I pt = collision.first;
504
505 if( FOOTPRINT* parentFP = aParentItem->GetParentFootprint() )
506 {
507 RotatePoint( pt, parentFP->GetOrientation() );
508 pt += parentFP->GetPosition();
509 }
510
511 wxString msg = formatMsg( _( "Internal clearance violation (%s clearance %s; actual %s)" ),
512 aConstraint.GetName(),
513 clearance,
514 collision.second );
515
516 drcItem->SetErrorMessage( msg );
517 drcItem->SetItems( aParentItem );
518 drcItem->SetViolatingRule( aConstraint.GetParentRule() );
519
520 reportViolation( drcItem, pt, aLayer );
521 }
522}
523
524
526 DRC_CONSTRAINT& aConstraint )
527{
528 int epsilon = m_board->GetDesignSettings().GetDRCEpsilon();
529 int clearance = aConstraint.GetValue().Min();
530
531 if( aConstraint.GetSeverity() == RPT_SEVERITY_IGNORE || clearance - epsilon <= 0 )
532 return;
533
535
536 // Turn fractured fill into outlines and holes
537 fill.Simplify();
538
539 for( int outlineIdx = 0; outlineIdx < fill.OutlineCount(); ++outlineIdx )
540 {
541 SHAPE_LINE_CHAIN* firstOutline = &fill.Outline( outlineIdx );
542
543 //
544 // Step one: outline to outline clearance violations
545 //
546
547 for( int ii = outlineIdx + 1; ii < fill.OutlineCount(); ++ii )
548 {
549 SHAPE_LINE_CHAIN* secondOutline = &fill.Outline( ii );
550
551 for( int jj = 0; jj < secondOutline->SegmentCount(); ++jj )
552 {
553 SEG secondSeg = secondOutline->Segment( jj );
554 int actual;
555 VECTOR2I pos;
556
557 if( firstOutline->Collide( secondSeg, clearance - epsilon, &actual, &pos ) )
558 {
559 std::shared_ptr<DRC_ITEM> drcItem = DRC_ITEM::Create( DRCE_CLEARANCE );
560 wxString msg = formatMsg( _( "(%s clearance %s; actual %s)" ),
561 aConstraint.GetName(),
562 clearance,
563 actual );
564
565 drcItem->SetErrorMessage( drcItem->GetErrorText() + wxS( " " ) + msg );
566 drcItem->SetItems( aZone );
567 drcItem->SetViolatingRule( aConstraint.GetParentRule() );
568 reportViolation( drcItem, pos, aLayer );
569 }
570 }
571
572 if( m_drcEngine->IsCancelled() )
573 return;
574 }
575
576 //
577 // Step two: interior hole clearance violations
578 //
579
580 for( int holeIdx = 0; holeIdx < fill.HoleCount( outlineIdx ); ++holeIdx )
581 {
582 testShapeLineChain( fill.Hole( outlineIdx, holeIdx ), 0, aLayer, aZone, aConstraint );
583
584 if( m_drcEngine->IsCancelled() )
585 return;
586 }
587 }
588}
589
590
592 PCB_LAYER_ID aLayer, BOARD_ITEM* aOther )
593{
594 bool testClearance = !m_drcEngine->IsErrorLimitExceeded( DRCE_CLEARANCE );
595 bool testHoles = !m_drcEngine->IsErrorLimitExceeded( DRCE_HOLE_CLEARANCE );
596 DRC_CONSTRAINT constraint;
597 int clearance = 0;
598 int actual;
599 int violations = 0;
600 VECTOR2I pos;
601 LSET boardCopperLayers = LSET::AllCuMask( m_board->GetCopperLayerCount() );
602
603 std::shared_ptr<SHAPE> otherShapeStorage = aOther->GetEffectiveShape( aLayer );
604 SHAPE* otherShape = otherShapeStorage.get();
605
606 if( testClearance )
607 {
608 constraint = m_drcEngine->EvalRules( PHYSICAL_CLEARANCE_CONSTRAINT, aItem, aOther, aLayer );
609 clearance = constraint.GetValue().Min();
610 }
611
612 if( constraint.GetSeverity() != RPT_SEVERITY_IGNORE && clearance > 0 )
613 {
614 // Collide (and generate violations) based on a well-defined order so that exclusion
615 // checking against previously-generated violations will work.
616 if( aItem->m_Uuid > aOther->m_Uuid )
617 {
618 std::swap( aItem, aOther );
619 std::swap( aItemShape, otherShape );
620 }
621
622 if( aItemShape->Collide( otherShape, clearance, &actual, &pos ) )
623 {
624 std::shared_ptr<DRC_ITEM> drcItem = DRC_ITEM::Create( DRCE_CLEARANCE );
625 wxString msg = formatMsg( _( "(%s clearance %s; actual %s)" ),
626 constraint.GetName(),
627 clearance,
628 actual );
629
630 drcItem->SetErrorMessage( drcItem->GetErrorText() + wxS( " " ) + msg );
631 drcItem->SetItems( aItem, aOther );
632 drcItem->SetViolatingRule( constraint.GetParentRule() );
633 reportTwoShapeGeometry( drcItem, pos, aItemShape, otherShape, aLayer, actual );
634 ++violations;
635 }
636 }
637
638 if( testHoles )
639 {
640 std::shared_ptr<SHAPE_SEGMENT> itemHoleShape;
641 std::shared_ptr<SHAPE_SEGMENT> otherHoleShape;
642 clearance = 0;
643
644 if( aItem->Type() == PCB_VIA_T )
645 {
646 LSET layers = aItem->GetLayerSet();
647
648 if( layers.Contains( F_Cu ) )
649 layers |= LSET( LSET::FrontBoardTechMask() ).set( F_CrtYd );
650
651 if( layers.Contains( B_Cu ) )
652 layers |= LSET( LSET::BackBoardTechMask() ).set( B_CrtYd );
653
654 if( layers.Contains( F_Cu ) && layers.Contains( B_Cu ) )
655 layers |= boardCopperLayers;
656
657 wxCHECK_MSG( layers.Contains( aLayer ), violations,
658 wxT( "Bug! Vias should only be checked for layers on which they exist" ) );
659
660 itemHoleShape = aItem->GetEffectiveHoleShape();
661 }
662 else if( aItem->HasHole() )
663 {
664 itemHoleShape = aItem->GetEffectiveHoleShape();
665 }
666
667 if( aOther->Type() == PCB_VIA_T )
668 {
669 LSET layers = aOther->GetLayerSet();
670
671 if( layers.Contains( F_Cu ) )
672 layers |= LSET( LSET::FrontBoardTechMask() ).set( F_CrtYd );
673
674 if( layers.Contains( B_Cu ) )
675 layers |= LSET( LSET::BackBoardTechMask() ).set( B_CrtYd );
676
677 if( layers.Contains( F_Cu ) && layers.Contains( B_Cu ) )
678 layers |= boardCopperLayers;
679
680 wxCHECK_MSG( layers.Contains( aLayer ), violations,
681 wxT( "Bug! Vias should only be checked for layers on which they exist" ) );
682
683 otherHoleShape = aOther->GetEffectiveHoleShape();
684 }
685 else if( aOther->HasHole() )
686 {
687 otherHoleShape = aOther->GetEffectiveHoleShape();
688 }
689
690 if( itemHoleShape || otherHoleShape )
691 {
692 constraint = m_drcEngine->EvalRules( PHYSICAL_HOLE_CLEARANCE_CONSTRAINT, aOther, aItem, aLayer );
693 clearance = constraint.GetValue().Min();
694 }
695
696 if( constraint.GetSeverity() != RPT_SEVERITY_IGNORE && clearance > 0 )
697 {
698 if( itemHoleShape && itemHoleShape->Collide( otherShape, clearance, &actual, &pos ) )
699 {
700 std::shared_ptr<DRC_ITEM> drcItem = DRC_ITEM::Create( DRCE_HOLE_CLEARANCE );
701 wxString msg = formatMsg( _( "(%s clearance %s; actual %s)" ),
702 constraint.GetName(),
703 clearance ,
704 actual );
705
706 drcItem->SetErrorMessage( drcItem->GetErrorText() + wxS( " " ) + msg );
707 drcItem->SetItems( aItem, aOther );
708 drcItem->SetViolatingRule( constraint.GetParentRule() );
709 reportTwoShapeGeometry( drcItem, pos, itemHoleShape.get(), otherShape, aLayer, actual );
710 ++violations;
711 }
712
713 if( otherHoleShape && otherHoleShape->Collide( aItemShape, clearance, &actual, &pos ) )
714 {
715 std::shared_ptr<DRC_ITEM> drcItem = DRC_ITEM::Create( DRCE_HOLE_CLEARANCE );
716 wxString msg = formatMsg( _( "(%s clearance %s; actual %s)" ),
717 constraint.GetName(),
718 clearance,
719 actual );
720
721 drcItem->SetErrorMessage( drcItem->GetErrorText() + wxS( " " ) + msg );
722 drcItem->SetItems( aItem, aOther );
723 drcItem->SetViolatingRule( constraint.GetParentRule() );
724 reportTwoShapeGeometry( drcItem, pos, otherHoleShape.get(), aItemShape, aLayer, actual );
725 ++violations;
726 }
727 }
728 }
729
730 return violations;
731}
732
733
735 PCB_LAYER_ID aLayer )
736{
737 for( ZONE* zone : m_board->m_DRCZones )
738 {
739 if( !zone->GetLayerSet().test( aLayer ) )
740 continue;
741
742 BOX2I itemBBox = aItem->GetBoundingBox();
743 BOX2I worstCaseBBox = itemBBox;
744
745 worstCaseBBox.Inflate( m_board->m_DRCMaxClearance );
746
747 if( !worstCaseBBox.Intersects( zone->GetBoundingBox() ) )
748 continue;
749
750 bool testClearance = !m_drcEngine->IsErrorLimitExceeded( DRCE_CLEARANCE );
751 bool testHoles = !m_drcEngine->IsErrorLimitExceeded( DRCE_HOLE_CLEARANCE );
752
753 if( !testClearance && !testHoles )
754 return;
755
756 DRC_RTREE* zoneRTree = m_board->m_CopperZoneRTreeCache[ zone ].get();
757 DRC_CONSTRAINT constraint;
758 bool colliding;
759 int clearance = -1;
760 int actual;
761 VECTOR2I pos;
762
763 if( testClearance )
764 {
765 constraint = m_drcEngine->EvalRules( PHYSICAL_CLEARANCE_CONSTRAINT, aItem, zone, aLayer );
766 clearance = constraint.GetValue().Min();
767 }
768
769 if( constraint.GetSeverity() != RPT_SEVERITY_IGNORE && clearance > 0 )
770 {
771 std::shared_ptr<SHAPE> itemShape = aItem->GetEffectiveShape( aLayer );
772
773 if( aItem->Type() == PCB_PAD_T )
774 {
775 PAD* pad = static_cast<PAD*>( aItem );
776
777 if( !pad->FlashLayer( aLayer ) )
778 {
779 if( pad->GetDrillSize().x == 0 && pad->GetDrillSize().y == 0 )
780 continue;
781
782 std::shared_ptr<SHAPE_SEGMENT> hole = pad->GetEffectiveHoleShape();
783 int size = hole->GetWidth();
784
785 itemShape = std::make_shared<SHAPE_SEGMENT>( hole->GetSeg(), size );
786 }
787 }
788
789 if( IsCopperLayer( aLayer ) && zoneRTree )
790 {
791 colliding = zoneRTree->QueryColliding( itemBBox, itemShape.get(), aLayer, clearance,
792 &actual, &pos );
793 }
794 else
795 {
796 colliding = zone->Outline()->Collide( itemShape.get(), clearance, &actual, &pos );
797 }
798
799 if( colliding )
800 {
801 std::shared_ptr<DRC_ITEM> drcItem = DRC_ITEM::Create( DRCE_CLEARANCE );
802 wxString msg = formatMsg( _( "(%s clearance %s; actual %s)" ),
803 constraint.GetName(),
804 clearance,
805 actual );
806
807 drcItem->SetErrorMessage( drcItem->GetErrorText() + wxS( " " ) + msg );
808 drcItem->SetItems( aItem, zone );
809 drcItem->SetViolatingRule( constraint.GetParentRule() );
810 reportTwoItemGeometry( drcItem, pos, aItem, zone, aLayer, actual );
811 }
812 }
813
814 if( testHoles )
815 {
816 std::shared_ptr<SHAPE_SEGMENT> holeShape;
817
818 if( aItem->Type() == PCB_VIA_T )
819 {
820 if( aItem->GetLayerSet().Contains( aLayer ) )
821 holeShape = aItem->GetEffectiveHoleShape();
822 }
823 else if( aItem->HasHole() )
824 {
825 holeShape = aItem->GetEffectiveHoleShape();
826 }
827
828 if( holeShape )
829 {
830 constraint = m_drcEngine->EvalRules( PHYSICAL_HOLE_CLEARANCE_CONSTRAINT, aItem, zone, aLayer );
831 clearance = constraint.GetValue().Min();
832
833 if( constraint.GetSeverity() != RPT_SEVERITY_IGNORE && clearance > 0 )
834 {
835 if( IsCopperLayer( aLayer ) && zoneRTree )
836 {
837 colliding = zoneRTree->QueryColliding( itemBBox, holeShape.get(), aLayer,
838 clearance, &actual, &pos );
839 }
840 else
841 {
842 colliding = zone->Outline()->Collide( holeShape.get(), clearance, &actual, &pos );
843 }
844
845 if( colliding )
846 {
847 std::shared_ptr<DRC_ITEM> drcItem = DRC_ITEM::Create( DRCE_HOLE_CLEARANCE );
848 wxString msg = formatMsg( _( "(%s clearance %s; actual %s)" ),
849 constraint.GetName(),
850 clearance,
851 actual );
852
853 drcItem->SetErrorMessage( drcItem->GetErrorText() + wxS( " " ) + msg );
854 drcItem->SetItems( aItem, zone );
855 drcItem->SetViolatingRule( constraint.GetParentRule() );
856
857 std::shared_ptr<SHAPE> zoneShape = zone->GetEffectiveShape( aLayer );
858 reportTwoShapeGeometry( drcItem, pos, holeShape.get(), zoneShape.get(), aLayer, actual );
859 }
860 }
861 }
862 }
863
864 if( m_drcEngine->IsCancelled() )
865 return;
866 }
867}
868
869
870namespace detail
871{
873}
BOX2< VECTOR2I > BOX2I
Definition box2.h:922
static const ADVANCED_CFG & GetCfg()
Get the singleton instance's config, which is shared by all consumers.
BASE_SET & set(size_t pos)
Definition base_set.h:116
A base class for any item which can be embedded within the BOARD container class, and therefore insta...
Definition board_item.h:79
virtual bool IsOnLayer(PCB_LAYER_ID aLayer) const
Test to see if this object is on the given layer.
Definition board_item.h:314
virtual std::shared_ptr< SHAPE > GetEffectiveShape(PCB_LAYER_ID aLayer=UNDEFINED_LAYER, FLASHING aFlash=FLASHING::DEFAULT) const
Some pad shapes can be complex (rounded/chamfered rectangle), even without considering custom shapes.
FOOTPRINT * GetParentFootprint() const
virtual LSET GetLayerSet() const
Return a std::bitset of all layers on which the item physically resides.
Definition board_item.h:252
BOARD_ITEM_CONTAINER * GetParent() const
Definition board_item.h:210
virtual std::shared_ptr< SHAPE_SEGMENT > GetEffectiveHoleShape() const
virtual bool HasHole() const
Definition board_item.h:156
constexpr BOX2< Vec > & Inflate(coord_type dx, coord_type dy)
Inflates the rectangle horizontally by dx and vertically by dy.
Definition box2.h:558
constexpr bool Intersects(const BOX2< Vec > &aRect) const
Definition box2.h:311
wxString GetName() const
Definition drc_rule.h:174
SEVERITY GetSeverity() const
Definition drc_rule.h:187
const MINOPTMAX< int > & GetValue() const
Definition drc_rule.h:166
DRC_RULE * GetParentRule() const
Definition drc_rule.h:170
static std::shared_ptr< DRC_ITEM > Create(int aErrorCode)
Constructs a DRC_ITEM for the given error code.
Definition drc_item.cpp:384
Implement an R-tree for fast spatial and layer indexing of connectable items.
Definition drc_rtree.h:50
int QueryColliding(BOARD_ITEM *aRefItem, PCB_LAYER_ID aRefLayer, PCB_LAYER_ID aTargetLayer, std::function< bool(BOARD_ITEM *)> aFilter=nullptr, std::function< bool(BOARD_ITEM *)> aVisitor=nullptr, int aClearance=0) const
This is a fast test which essentially does bounding-box overlap given a worst-case clearance.
Definition drc_rtree.h:217
void testZoneLayer(ZONE *aZone, PCB_LAYER_ID aLayer, DRC_CONSTRAINT &aConstraint)
virtual ~DRC_TEST_PROVIDER_PHYSICAL_CLEARANCE()=default
void testItemAgainstZones(BOARD_ITEM *aItem, PCB_LAYER_ID aLayer)
virtual bool Run() override
Run this provider against the given PCB with configured options (if any).
void testShapeLineChain(const SHAPE_LINE_CHAIN &aOutline, int aLineWidth, PCB_LAYER_ID aLayer, BOARD_ITEM *aParentItem, DRC_CONSTRAINT &aConstraint)
int testItemAgainstItem(BOARD_ITEM *aItem, SHAPE *aItemShape, PCB_LAYER_ID aLayer, BOARD_ITEM *aOther)
virtual bool reportPhase(const wxString &aStageName)
void reportTwoShapeGeometry(std::shared_ptr< DRC_ITEM > &aDrcItem, const VECTOR2I &aMarkerPos, const SHAPE *aShape1, const SHAPE *aShape2, PCB_LAYER_ID aLayer, int aDistance)
int forEachGeometryItem(const std::vector< KICAD_T > &aTypes, const LSET &aLayers, const std::function< bool(BOARD_ITEM *)> &aFunc)
void reportTwoItemGeometry(std::shared_ptr< DRC_ITEM > &aDrcItem, const VECTOR2I &aMarkerPos, const BOARD_ITEM *aItem1, const BOARD_ITEM *aItem2, PCB_LAYER_ID aLayer, int aDistance)
void reportViolation(std::shared_ptr< DRC_ITEM > &item, const VECTOR2I &aMarkerPos, int aMarkerLayer, const std::function< void(PCB_MARKER *)> &aPathGenerator=[](PCB_MARKER *){})
bool isInvisibleText(const BOARD_ITEM *aItem) const
wxString formatMsg(const wxString &aFormatString, const wxString &aSource, double aConstraint, double aActual, EDA_DATA_TYPE aDataType=EDA_DATA_TYPE::DISTANCE)
virtual bool reportProgress(size_t aCount, size_t aSize, size_t aDelta=1)
double AsRadians() const
Definition eda_angle.h:120
virtual const BOX2I GetBoundingBox() const
Return the orthogonal bounding box of this object for display purposes.
Definition eda_item.cpp:110
const KIID m_Uuid
Definition eda_item.h:516
KICAD_T Type() const
Returns the type of object.
Definition eda_item.h:110
EDA_ANGLE GetArcAngle() const
SHAPE_POLY_SET & GetPolyShape()
Definition eda_shape.h:336
int GetRadius() const
SHAPE_T GetShape() const
Definition eda_shape.h:168
void RebuildBezierToSegmentsPointsList(int aMaxError)
Rebuild the m_bezierPoints vertex list that approximate the Bezier curve by a list of segments.
const VECTOR2I & GetStart() const
Return the starting point of the graphic.
Definition eda_shape.h:173
const std::vector< VECTOR2I > & GetBezierPoints() const
Definition eda_shape.h:320
wxString SHAPE_T_asString() const
LSET is a set of PCB_LAYER_IDs.
Definition lset.h:37
static const LSET & FrontBoardTechMask()
Return a mask holding technical layers used in a board fabrication (no CU layer) on front side.
Definition lset.cpp:652
static const LSET & AllLayersMask()
Definition lset.cpp:624
static LSET AllCuMask()
return AllCuMask( MAX_CU_LAYERS );
Definition lset.cpp:591
static const LSET & PhysicalLayersMask()
Return a mask holding all layers which are physically realized.
Definition lset.cpp:680
static const LSET & BackBoardTechMask()
Return a mask holding technical layers used in a board fabrication (no CU layer) on Back side.
Definition lset.cpp:638
bool Contains(PCB_LAYER_ID aLayer) const
See if the layer set contains a PCB layer.
Definition lset.h:63
T Min() const
Definition minoptmax.h:33
Definition pad.h:54
VECTOR2I GetCenter() const override
This defaults to the center of the bounding box if not overridden.
Definition pcb_shape.h:81
int GetWidth() const override
Definition seg.h:42
VECTOR2I A
Definition seg.h:49
VECTOR2I B
Definition seg.h:50
const VECTOR2I NearestPoint(const VECTOR2I &aP) const
Compute a point on the segment (this) that is closest to point aP.
Definition seg.cpp:604
static SEG::ecoord Square(int a)
Definition seg.h:123
bool Collide(const SEG &aSeg, int aClearance, int *aActual=nullptr) const
Definition seg.cpp:536
ecoord SquaredLength() const
Definition seg.h:348
Represent a polyline containing arcs as well as line segments: A chain of connected line and/or arc s...
bool IsClosed() const override
virtual bool Collide(const VECTOR2I &aP, int aClearance=0, int *aActual=nullptr, VECTOR2I *aLocation=nullptr) const override
Check if point aP lies closer to us than aClearance.
SEG Segment(int aIndex) const
Return a copy of the aIndex-th segment in the line chain.
void Append(int aX, int aY, bool aAllowDuplication=false)
Append a new point at the end of the line chain.
int SegmentCount() const
Return the number of segments in this line chain.
const SEG CSegment(int aIndex) const
Return a constant copy of the aIndex segment in the line chain.
Represent a set of closed polygons.
bool Collide(const SHAPE *aShape, int aClearance=0, int *aActual=nullptr, VECTOR2I *aLocation=nullptr) const override
Check if the boundary of shape (this) lies closer to the shape aShape than aClearance,...
int HoleCount(int aOutline) const
Returns the number of holes in a given outline.
void Simplify()
Simplify the polyset (merges overlapping polys, eliminates degeneracy/self-intersections)
SHAPE_LINE_CHAIN & Outline(int aIndex)
Return the reference to aIndex-th outline in the set.
SHAPE_LINE_CHAIN & Hole(int aOutline, int aHole)
Return the reference to aHole-th hole in the aIndex-th outline.
int OutlineCount() const
Return the number of outlines in the set.
SHAPE_POLY_SET CloneDropTriangulation() const
An abstract shape on 2D plane.
Definition shape.h:126
virtual bool Collide(const VECTOR2I &aP, int aClearance=0, int *aActual=nullptr, VECTOR2I *aLocation=nullptr) const
Check if the boundary of shape (this) lies closer to the point aP than aClearance,...
Definition shape.h:181
Handle a list of polygons defining a copper zone.
Definition zone.h:74
bool GetIsRuleArea() const
Accessors to parameters used in Rule Area zones:
Definition zone.h:704
const std::shared_ptr< SHAPE_POLY_SET > & GetFilledPolysList(PCB_LAYER_ID aLayer) const
Definition zone.h:600
const BOX2I GetBoundingBox() const override
Definition zone.cpp:616
SHAPE_POLY_SET * Outline()
Definition zone.h:335
virtual LSET GetLayerSet() const override
Return a std::bitset of all layers on which the item physically resides.
Definition zone.h:136
virtual std::shared_ptr< SHAPE > GetEffectiveShape(PCB_LAYER_ID aLayer=UNDEFINED_LAYER, FLASHING aFlash=FLASHING::DEFAULT) const override
Some pad shapes can be complex (rounded/chamfered rectangle), even without considering custom shapes.
Definition zone.cpp:1560
The common library.
@ DRCE_HOLE_CLEARANCE
Definition drc_item.h:55
@ DRCE_CLEARANCE
Definition drc_item.h:44
#define ATOMIC_TABLES
Definition drc_rtree.h:43
@ PHYSICAL_HOLE_CLEARANCE_CONSTRAINT
Definition drc_rule.h:78
@ PHYSICAL_CLEARANCE_CONSTRAINT
Definition drc_rule.h:77
#define REPORT_AUX(s)
#define _(s)
@ SEGMENT
Definition eda_shape.h:45
@ RECTANGLE
Use RECTANGLE instead of RECT to avoid collision in a Windows header.
Definition eda_shape.h:46
a few functions useful in geometry calculations.
int GetArcToSegmentCount(int aRadius, int aErrorMax, const EDA_ANGLE &aArcAngle)
bool IsCopperLayer(int aLayerId)
Test whether a layer is a copper layer.
Definition layer_ids.h:677
PCB_LAYER_ID
A quick note on layer IDs:
Definition layer_ids.h:60
@ F_CrtYd
Definition layer_ids.h:116
@ Edge_Cuts
Definition layer_ids.h:112
@ B_Cu
Definition layer_ids.h:65
@ B_CrtYd
Definition layer_ids.h:115
@ F_Cu
Definition layer_ids.h:64
This file contains miscellaneous commonly used macros and functions.
#define UNIMPLEMENTED_FOR(type)
Definition macros.h:96
static DRC_REGISTER_TEST_PROVIDER< DRC_TEST_PROVIDER_ANNULAR_WIDTH > dummy
@ RPT_SEVERITY_IGNORE
const double epsilon
VECTOR2I center
int clearance
int actual
#define M_PI
void RotatePoint(int *pX, int *pY, const EDA_ANGLE &aAngle)
Calculate the new point of coord coord pX, pY, for a rotation center 0, 0.
Definition trigo.cpp:229
double DEG2RAD(double deg)
Definition trigo.h:166
@ 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
@ 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:108
@ 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_BARCODE_T
class PCB_BARCODE, a barcode (graphic item)
Definition typeinfo.h:101
@ 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
VECTOR2< int32_t > VECTOR2I
Definition vector2d.h:695