KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
pns_shove.cpp
Go to the documentation of this file.
1/*
2 * KiRouter - a push-and-(sometimes-)shove PCB router
3 *
4 * Copyright (C) 2013-2014 CERN
5 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
6 * Author: Tomasz Wlostowski <tomasz.wlostowski@cern.ch>
7 *
8 * This program is free software: you can redistribute it and/or modify it
9 * under the terms of the GNU General Public License as published by the
10 * Free Software Foundation, either version 3 of the License, or (at your
11 * option) any later version.
12 *
13 * This program is distributed in the hope that it will be useful, but
14 * WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 * General Public License for more details.
17 *
18 * You should have received a copy of the GNU General Public License along
19 * with this program. If not, see <http://www.gnu.org/licenses/>.
20 */
21
22#include <deque>
23#include <cassert>
24#include <math/box2.h>
25
26#include <wx/log.h>
27
28#include "pns_arc.h"
29#include "pns_line.h"
30#include "pns_node.h"
31#include "pns_debug_decorator.h"
32#include "pns_walkaround.h"
33#include "pns_shove.h"
34#include "pns_solid.h"
35#include "pns_optimizer.h"
36#include "pns_via.h"
37#include "pns_utils.h"
38#include "pns_router.h"
39#include "pns_topology.h"
40
41#include "time_limit.h"
42
43// fixme - move all logger calls to debug decorator
44
46
47namespace PNS {
48
49void SHOVE::replaceItems( ITEM* aOld, std::unique_ptr< ITEM > aNew )
50{
51 OPT_BOX2I changed_area = ChangedArea( aOld, aNew.get() );
52
53 if( changed_area )
54 m_affectedArea = m_affectedArea ? m_affectedArea->Merge( *changed_area ) : *changed_area;
55
56 ROOT_LINE_ENTRY *re = nullptr;
58
59 if( aOld->OfKind( ITEM::VIA_T ) )
60 {
61 VIA* vold = static_cast<VIA*>( aOld );
62 VIA* vnew = static_cast<VIA*>( aNew.get() );
63 re = touchRootLine( vold );
64 re->newVia = vnew;
65 newId = static_cast<VIA*>( aNew.get() )->Uid();
66
67 PNS_DBG( Dbg(), Message,
68 wxString::Format( "replace-via node=%p vold=%p [%d %d]-> vnew=%p [%d %d] nid %llu", m_currentNode, aOld,
69 vold->Pos().x, vold->Pos().y, aNew.get(), vnew->Pos().x,
70 vnew->Pos().y, newId ) );
71 }
72
73 m_currentNode->Replace( aOld, std::move( aNew ) );
74
75 if( re )
76 m_rootLineHistory[ newId ] = re;
77}
78
79
80SHOVE::ROOT_LINE_ENTRY* SHOVE::replaceLine( LINE& aOld, LINE& aNew, bool aIncludeInChangedArea, bool aAllowRedundantSegments, NODE* aNode )
81{
82 if( aIncludeInChangedArea )
83 {
84 OPT_BOX2I changed_area = ChangedArea( aOld, aNew );
85
86 if( changed_area )
87 {
88 SHAPE_RECT r( *changed_area );
89 PNS_DBG( Dbg(), AddShape, &r, BLUE, 0, wxT( "shove-changed-area" ) );
90
91 m_affectedArea = m_affectedArea ? m_affectedArea->Merge( *changed_area )
92 : *changed_area;
93 }
94 }
95
96 int old_sc = aOld.SegmentCount();
97 int old_lc = aOld.LinkCount();
98
99 if( aOld.EndsWithVia() )
100 {
101 LINKED_ITEM* viaLink = nullptr;
102 for( LINKED_ITEM* lnk : aOld.Links() )
103 {
104 if( lnk->OfKind( ITEM::VIA_T ) )
105 {
106 viaLink = lnk;
107 break;
108 }
109 }
110 if( viaLink )
111 aOld.Unlink( viaLink );
112 }
113
114 bool foundPredecessor = false;
115 ROOT_LINE_ENTRY *rootEntry = nullptr;
116
117 // Keep track of the 'root lines', i.e. the unmodified (pre-shove) versions
118 // of the affected tracks in a map. The optimizer can then query the pre-shove shape
119 // for each shoved line and perform additional constraint checks (i.e. prevent overzealous
120 // optimizations)
121
122 // Check if the shoved line already has an ancestor (e.g. line from a previous shove
123 // iteration/cursor movement)
124 for( LINKED_ITEM* link : aOld.Links() )
125 {
126 auto oldLineIter = m_rootLineHistory.find( link->Uid() );
127
128 if( oldLineIter != m_rootLineHistory.end() )
129 {
130 rootEntry = oldLineIter->second;
131 foundPredecessor = true;
132 break;
133 }
134 }
135
136 // If found, use it, otherwise, create new entry in the map (we have a genuine new 'root' line)
137 if( !foundPredecessor )
138 {
139 if( ! rootEntry )
140 {
141 rootEntry = new ROOT_LINE_ENTRY( aOld.Clone() );
142 }
143
144 for( LINKED_ITEM* link : aOld.Links() )
145 {
146 m_rootLineHistory[link->Uid()] = rootEntry;
147 }
148 }
149
150 // Now update the NODE (calling Replace invalidates the Links() in a LINE)
151 if( aNode )
152 aNode->Replace( aOld, aNew, aAllowRedundantSegments );
153 else
154 m_currentNode->Replace( aOld, aNew, aAllowRedundantSegments );
155
156
157 // point the Links() of the new line to its oldest ancestor
158 for( LINKED_ITEM* link : aNew.Links() )
159 {
160 m_rootLineHistory[ link->Uid() ] = rootEntry;
161 }
162
163 rootEntry->newLine = aNew;
164
165 return rootEntry;
166}
167
168
169int SHOVE::getClearance( const ITEM* aA, const ITEM* aB ) const
170{
171 if( m_forceClearance >= 0 )
172 return m_forceClearance;
173
174 int clearance = m_currentNode->GetClearance( aA, aB, false );
175
176 if( aA->HasHole() )
177 clearance = std::max( clearance, m_currentNode->GetClearance( aA->Hole(), aB, false ) );
178
179 if( aB->HasHole() )
180 clearance = std::max( clearance, m_currentNode->GetClearance( aA, aB->Hole(), false ) );
181
182 return clearance;
183}
184
185
186void SHOVE::sanityCheck( LINE* aOld, LINE* aNew )
187{
188 assert( aOld->CPoint( 0 ) == aNew->CPoint( 0 ) );
189 assert( aOld->CPoint( -1 ) == aNew->CPoint( -1 ) );
190}
191
192
193SHOVE::SHOVE( NODE* aWorld, ROUTER* aRouter ) :
194 ALGO_BASE( aRouter )
195{
197 m_forceClearance = -1;
198 m_root = aWorld;
199 m_currentNode = aWorld;
201
202 // Initialize other temporary variables:
203 m_draggedVia = nullptr;
204 m_iter = 0;
205 m_multiLineMode = false;
206 m_headsModified = false;
210}
211
212
214{
215}
216
217
218LINE SHOVE::assembleLine( const LINKED_ITEM* aSeg, int* aIndex, bool aPreCleanup )
219{
220 LINE cur = m_currentNode->AssembleLine( const_cast<LINKED_ITEM*>( aSeg ), aIndex, true );
221
222 if( aPreCleanup )
223 {
224 LINE cleaned;
225
226 if (preShoveCleanup( &cur, &cleaned ) )
227 return cleaned;
228 }
229
230 return cur;
231}
232
233
234// A dumb function that checks if the shoved line is shoved the right way, e.g. visually
235// "outwards" of the line/via applying pressure on it. Unfortunately there's no mathematical
236// concept of orientation of an open curve, so we use some primitive heuristics: if the shoved
237// line wraps around the start of the "pusher", it's likely shoved in wrong direction.
238
239// Update: there's no concept of an orientation of an open curve, but nonetheless Tom's dumb
240// as.... (censored)
241// Two open curves put together make a closed polygon... Tom should learn high school geometry!
242bool SHOVE::checkShoveDirection( const LINE& aCurLine, const LINE& aObstacleLine,
243 const LINE& aShovedLine ) const
244{
245 SHAPE_LINE_CHAIN::POINT_INSIDE_TRACKER checker( aCurLine.CPoint( 0) );
246 checker.AddPolyline( aObstacleLine.CLine() );
247 checker.AddPolyline( aShovedLine.CLine().Reverse() );
248
249 bool inside = checker.IsInside();
250
251 return !inside;
252}
253
254
255/*
256 * Push aObstacleLine away from aCurLine's via by the clearance distance, returning the result
257 * in aResultLine.
258 *
259 * Must be called only when aCurLine itself is on another layer (or has no segments) so that it
260 * can be ignored.
261 */
262bool SHOVE::shoveLineFromLoneVia( const LINE& aCurLine, const LINE& aObstacleLine,
263 LINE& aResultLine )
264{
265 // Build a hull for aCurLine's via and re-walk aObstacleLine around it.
266
267 int obstacleLineWidth = aObstacleLine.Width();
268 const VIA& via = aCurLine.Via();
269 int clearance = getClearance( &via, &aObstacleLine );
270 HOLE* viaHole = via.Hole();
271 int holeClearance = getClearance( viaHole, &aObstacleLine );
272
273 if( holeClearance + via.Drill() / 2 > clearance + via.Diameter( aObstacleLine.Layer() ) / 2 )
274 clearance = holeClearance + via.Drill() / 2 - via.Diameter( aObstacleLine.Layer() ) / 2;
275
276 SHAPE_LINE_CHAIN hull = aCurLine.Via().Hull( clearance, obstacleLineWidth, aCurLine.Layer() );
277 SHAPE_LINE_CHAIN path_cw;
278 SHAPE_LINE_CHAIN path_ccw;
279
280 if( ! aObstacleLine.Walkaround( hull, path_cw, true ) )
281 return false;
282
283 if( ! aObstacleLine.Walkaround( hull, path_ccw, false ) )
284 return false;
285
286 const SHAPE_LINE_CHAIN& shortest = path_ccw.Length() < path_cw.Length() ? path_ccw : path_cw;
287
288 if( shortest.PointCount() < 2 )
289 return false;
290
291 if( aObstacleLine.CPoint( -1 ) != shortest.CPoint( -1 ) )
292 return false;
293
294 if( aObstacleLine.CPoint( 0 ) != shortest.CPoint( 0 ) )
295 return false;
296
297 aResultLine.SetShape( shortest );
298
299 if( aResultLine.Collide( &aCurLine, m_currentNode, aResultLine.Layer() ) )
300 return false;
301
302 return true;
303}
304
305
306/*
307 * Re-walk aObstacleLine around the given set of hulls, returning the result in aResultLine.
308 */
309bool SHOVE::shoveLineToHullSet( const LINE& aCurLine, const LINE& aObstacleLine, LINE& aResultLine,
310 const HULL_SET& aHulls, bool aPermitAdjustingStart,
311 bool aPermitAdjustingEnd )
312{
313 const int c_ENDPOINT_ON_HULL_THRESHOLD = 1000;
314 int attempt;
315 bool permitAdjustingEndpoints = aPermitAdjustingStart || aPermitAdjustingEnd;
316
317 PNS_DBG( Dbg(), BeginGroup, "shove-details", 1 );
318
319 for( attempt = 0; attempt < 4; attempt++ )
320 {
321 bool invertTraversal = ( attempt >= 2 );
322 bool clockwise = attempt % 2;
323 int vFirst = -1, vLast = -1;
324 SHAPE_LINE_CHAIN obs = aObstacleLine.CLine();
325 LINE l( aObstacleLine );
327
328 if( permitAdjustingEndpoints && l.SegmentCount() >= 1 )
329 {
330 auto minDistP = [&]( VECTOR2I pref, int& mdist, int& minhull ) -> VECTOR2I
331 {
332 int min_dist = std::numeric_limits<int>::max();
333 VECTOR2I nearestP;
334
335 for( int i = 0; i < (int) aHulls.size(); i++ )
336 {
337 const SHAPE_LINE_CHAIN& hull =
338 aHulls[invertTraversal ? aHulls.size() - 1 - i : i];
339 int dist;
340 const VECTOR2I p = hull.NearestPoint( pref, true );
341
342 if( hull.PointInside( pref ) )
343 dist = 0;
344 else
345 dist = ( p - pref ).EuclideanNorm();
346
347 if( dist < c_ENDPOINT_ON_HULL_THRESHOLD && dist < min_dist )
348 {
349 bool reject = false;
350
351 if( !reject )
352 {
353 min_dist = dist;
354 nearestP = p;
355 minhull = invertTraversal ? aHulls.size() - 1 - i : i;
356 }
357 }
358 }
359 mdist = min_dist;
360 return nearestP;
361 };
362
363 int minDist0, minDist1, minhull0, minhull1 ;
364 VECTOR2I p0 = minDistP( l.CPoint( 0 ), minDist0, minhull0 );
365 VECTOR2I p1 = minDistP( l.CPoint( -1 ), minDist1, minhull1 );
366
367 PNS_DBG( Dbg(), Message, wxString::Format( "mindists : %d %d hulls %d %d\n", minDist0, minDist1, minhull0, minhull1 ) );
368
369 if( minDist1 < c_ENDPOINT_ON_HULL_THRESHOLD && aPermitAdjustingEnd )
370 {
371 l.Line().Append( p1 );
372 obs = l.CLine();
373 path = l.CLine();
374 }
375
376 if( minDist0 < c_ENDPOINT_ON_HULL_THRESHOLD && aPermitAdjustingStart )
377 {
378 l.Line().Insert( 0, p0 );
379 obs = l.CLine();
380 path = l.CLine();
381 }
382 }
383
384
385 bool failWalk = false;
386
387 for( int i = 0; i < (int) aHulls.size(); i++ )
388 {
389 const SHAPE_LINE_CHAIN& hull = aHulls[invertTraversal ? aHulls.size() - 1 - i : i];
390
391 PNS_DBG( Dbg(), AddShape, &hull, YELLOW, 10000, wxString::Format( "hull[%d]", i ) );
392 PNS_DBG( Dbg(), AddShape, &path, WHITE, l.Width(), wxString::Format( "path[%d]", i ) );
393 PNS_DBG( Dbg(), AddShape, &obs, LIGHTGRAY, aObstacleLine.Width(), wxString::Format( "obs[%d]", i ) );
394
395 if( !l.Walkaround( hull, path, clockwise ) )
396 {
397 PNS_DBG( Dbg(), Message, wxString::Format( wxT( "Fail-Walk %s %s %d\n" ),
398 hull.Format().c_str(),
399 l.CLine().Format().c_str(),
400 clockwise? 1 : 0) );
401
402 failWalk = true;
403 break;
404 }
405
406 path.Simplify2();
407 l.SetShape( path );
408 }
409
410 if( failWalk )
411 continue;
412
413 for( int i = 0; i < std::min( path.PointCount(), obs.PointCount() ); i++ )
414 {
415 if( path.CPoint( i ) != obs.CPoint( i ) )
416 {
417 vFirst = i;
418 break;
419 }
420 }
421
422 int k = obs.PointCount() - 1;
423
424 for( int i = path.PointCount() - 1; i >= 0 && k >= 0; i--, k-- )
425 {
426 if( path.CPoint( i ) != obs.CPoint( k ) )
427 {
428 vLast = i;
429 break;
430 }
431 }
432
433 if( ( vFirst < 0 || vLast < 0 ) && !path.CompareGeometry( obs ) )
434 {
435 PNS_DBG( Dbg(), Message, wxString::Format( wxT( "attempt %d fail vfirst-last" ),
436 attempt ) );
437 continue;
438 }
439
440 if( path.CPoint( -1 ) != obs.CPoint( -1 ) || path.CPoint( 0 ) != obs.CPoint( 0 ) )
441 {
442 PNS_DBG( Dbg(), Message, wxString::Format( wxT( "attempt %d fail vend-start\n" ),
443 attempt ) );
444 continue;
445 }
446
447 if( !checkShoveDirection( aCurLine, aObstacleLine, l ) )
448 {
449 PNS_DBG( Dbg(), Message, wxString::Format( wxT( "attempt %d fail direction-check" ),
450 attempt ) );
451 aResultLine.SetShape( l.CLine() );
452 continue;
453 }
454
455 if( path.SelfIntersecting() )
456 {
457 PNS_DBG( Dbg(), Message, wxString::Format( wxT( "attempt %d fail self-intersect" ),
458 attempt ) );
459 continue;
460 }
461
462 bool colliding = l.Collide( &aCurLine, m_currentNode, l.Layer() );
463
464 #if 0
465 if(( aCurLine.Marker() & MK_HEAD ) && !colliding )
466 {
467 const JOINT* jtStart = m_currentNode->FindJoint( aCurLine.CPoint( 0 ), &aCurLine );
468
469 for( ITEM* item : jtStart->LinkList() )
470 {
471 if( item->Collide( &l, m_currentNode, l.Layer() ) )
472 colliding = true;
473 }
474 }
475 #endif
476
477 if( colliding )
478 {
479 PNS_DBG( Dbg(), Message, wxString::Format( wxT( "attempt %d fail coll-check" ),
480 attempt ) );
481 continue;
482 }
483
484 aResultLine.SetShape( l.CLine() );
485
486 PNS_DBGN( Dbg(), EndGroup );
487
488 return true;
489 }
490
491 PNS_DBGN( Dbg(), EndGroup );
492
493 return false;
494}
495
496
497/*
498 * Push aObstacleLine line away from aCurLine by the clearance distance, and return the result in
499 * aResultLine.
500 */
501bool SHOVE::ShoveObstacleLine( const LINE& aCurLine, const LINE& aObstacleLine,
502 LINE& aResultLine )
503{
504 const int cHullFailureExpansionFactor = 1000;
505 int extraHullExpansion = 0;
506
507 bool voeStart = false, voeEnd = false;
508 const JOINT* jtStart = nullptr;
509 const JOINT* jtEnd = nullptr;
510
511 if( aObstacleLine.PointCount() >= 2 )
512 {
513 jtStart = m_currentNode->FindJoint( aObstacleLine.CPoint( 0 ), &aObstacleLine );
514 jtEnd = m_currentNode->FindJoint( aObstacleLine.CPoint( -1 ), &aObstacleLine );
515 }
516
517 if( jtStart )
518 voeStart = jtStart->Via() != nullptr;
519 if( jtEnd )
520 voeEnd = jtEnd->Via() != nullptr;
521
522 aResultLine.ClearLinks();
523 bool viaOnEnd = aCurLine.EndsWithVia();
524
525 LINE obstacleLine( aObstacleLine );
526 std::optional<VIA> obsVia;
527
528 if( obstacleLine.EndsWithVia() )
529 {
530 obsVia = aObstacleLine.Via();
531 obstacleLine.RemoveVia();
532 }
533
534 PNS_DBG( Dbg(), Message, wxString::Format( wxT( "shove process-single: voe-cur %d voe-obs %d" ),
535 aCurLine.EndsWithVia()?1:0, aObstacleLine.EndsWithVia()?1:0 ) );
536
537
538 if( viaOnEnd && ( !aCurLine.LayersOverlap( &obstacleLine ) || aCurLine.SegmentCount() == 0 ) )
539 {
540 // Shove obstacleLine to the hull of aCurLine's via.
541 return shoveLineFromLoneVia( aCurLine, obstacleLine, aResultLine );
542 }
543 else
544 {
545 // Build a set of hulls around the segments of aCurLine. Hulls are at the clearance
546 // distance + obstacleLine's linewidth so that when re-walking obstacleLine along the
547 // hull it will be at the appropriate clearance.
548
549 int obstacleLineWidth = obstacleLine.Width();
550 int clearance = getClearance( &aCurLine, &obstacleLine );
551 int currentLineSegmentCount = aCurLine.SegmentCount();
552
553 /*PNS_DBG( Dbg(), Message, wxString::Format( wxT( "shove process-single: cur net %d obs %d cl %d" ),
554 m_router->GetInterface()->GetNetCode( aCurLine.Net() ),
555 m_router->GetInterface()->GetNetCode( obstacleLine.Net() ),
556 clearance ) );*/
557
558 for( int attempt = 0; attempt < 3; attempt++ )
559 {
560 HULL_SET hulls;
561
562 hulls.reserve( currentLineSegmentCount + 1 );
563
564 for( int i = 0; i < currentLineSegmentCount; i++ )
565 {
566 SEGMENT seg( aCurLine, aCurLine.CSegment( i ) );
567
568 // Arcs need additional clearance to ensure the hulls are always bigger than the arc
569 if( aCurLine.CLine().IsArcSegment( i ) )
570 {
571 PNS_DBG( Dbg(), Message, wxString::Format( wxT( "shove add-extra-clearance %d" ),
574 }
575
576 SHAPE_LINE_CHAIN hull = seg.Hull( clearance + extraHullExpansion, obstacleLineWidth, obstacleLine.Layer() );
577
578 hulls.push_back( hull );
579 }
580
581 if( viaOnEnd )
582 {
583 const VIA& via = aCurLine.Via();
584 int viaClearance = getClearance( &via, &obstacleLine );
585 HOLE* viaHole = via.Hole();
586 int holeClearance = getClearance( viaHole, &obstacleLine );
587 int layer = aObstacleLine.Layer();
588
589 if( holeClearance + via.Drill() / 2 > viaClearance + via.Diameter( layer ) / 2 )
590 {
591 viaClearance = holeClearance + via.Drill() / 2 - via.Diameter( layer ) / 2;
592 }
593
594 hulls.push_back( aCurLine.Via().Hull( viaClearance, obstacleLineWidth, layer ) );
595 }
596
597 bool permitMovingStart = (attempt >= 2) && !voeStart;
598 bool permitMovingEnd = (attempt >= 2) && !voeEnd;
599
600 if (shoveLineToHullSet( aCurLine, obstacleLine, aResultLine, hulls, permitMovingStart, permitMovingEnd ) )
601 {
602 if( obsVia )
603 aResultLine.AppendVia( *obsVia );
604
605 return true;
606 }
607
608 extraHullExpansion += cHullFailureExpansionFactor;
609 }
610 }
611
612 return false;
613}
614
615
616/*
617 * TODO describe....
618 */
620{
621 int segIndex;
622
623 LINE obstacleLine = assembleLine( aObstacleSeg, &segIndex, true );
624 LINE shovedLine( obstacleLine );
625 SEGMENT tmp( *aObstacleSeg );
626
627 if( obstacleLine.HasLockedSegments() )
628 {
629 PNS_DBG(Dbg(), Message, "try walk (locked segments)");
630 return SH_TRY_WALK;
631 }
632
633 bool shoveOK = ShoveObstacleLine( aCurrent, obstacleLine, shovedLine );
634
635 const double extensionWalkThreshold = 1.0;
636
637 double obsLen = obstacleLine.CLine().Length();
638 double shovedLen = shovedLine.CLine().Length();
639 double extensionFactor = 0.0;
640
641 if( obsLen != 0.0f )
642 extensionFactor = shovedLen / obsLen - 1.0;
643
644 //if( extensionFactor > extensionWalkThreshold )
645 // return SH_TRY_WALK;
646
647 assert( obstacleLine.LayersOverlap( &shovedLine ) );
648
649 if( Dbg() )
650 {
651 PNS_DBG( Dbg(), AddItem, aObstacleSeg, BLUE, 0, wxT( "colliding-segment" ) );
652 PNS_DBG( Dbg(), AddItem, &aCurrent, RED, 10000, wxString::Format( "current-line [links %d l %d v %d]", aCurrent.LinkCount(), aCurrent.Layer(), aCurrent.EndsWithVia() ) );
653 PNS_DBG( Dbg(), AddItem, &obstacleLine, GREEN, 10000, wxString::Format( "obstacle-line [links %d l %d v %d]", obstacleLine.LinkCount(), obstacleLine.Layer(), obstacleLine.EndsWithVia() ) );
654 PNS_DBG( Dbg(), AddItem, &shovedLine, BLUE, 10000, wxT( "shoved-line" ) );
655 }
656
657 if( shoveOK )
658 {
659 int rank = aCurrent.Rank();
660
661 shovedLine.SetRank( rank - 1 );
662 shovedLine.Line().Simplify2();
663
664 unwindLineStack( &obstacleLine );
665
666 replaceLine( obstacleLine, shovedLine, true, false );
667
668 if( !pushLineStack( shovedLine ) )
669 return SH_INCOMPLETE;
670
671 return SH_OK;
672 }
673
674 return SH_INCOMPLETE;
675}
676
677
678/*
679 * TODO describe....
680 */
682{
683 int segIndex;
684 LINE obstacleLine = assembleLine( aObstacleArc, &segIndex );
685 LINE shovedLine( obstacleLine );
686 ARC tmp( *aObstacleArc );
687
688 if( obstacleLine.HasLockedSegments() )
689 return SH_TRY_WALK;
690
691 bool shoveOK = ShoveObstacleLine( aCurrent, obstacleLine, shovedLine );
692
693 const double extensionWalkThreshold = 1.0;
694
695 double obsLen = obstacleLine.CLine().Length();
696 double shovedLen = shovedLine.CLine().Length();
697 double extensionFactor = 0.0;
698
699 if( obsLen != 0.0f )
700 extensionFactor = shovedLen / obsLen - 1.0;
701
702 if( extensionFactor > extensionWalkThreshold )
703 return SH_TRY_WALK;
704
705 assert( obstacleLine.LayersOverlap( &shovedLine ) );
706
707 PNS_DBG( Dbg(), AddItem, &tmp, WHITE, 10000, wxT( "obstacle-arc" ) );
708 PNS_DBG( Dbg(), AddItem, &aCurrent, RED, 10000, wxT( "current-line" ) );
709 PNS_DBG( Dbg(), AddItem, &obstacleLine, GREEN, 10000, wxT( "obstacle-line" ) );
710 PNS_DBG( Dbg(), AddItem, &shovedLine, BLUE, 10000, wxT( "shoved-line" ) );
711
712 if( shoveOK )
713 {
714 int rank = aCurrent.Rank();
715 shovedLine.SetRank( rank - 1 );
716
717 replaceLine( obstacleLine, shovedLine, true, false );
718
719 if( !pushLineStack( shovedLine ) )
720 return SH_INCOMPLETE;
721 }
722
723 return SH_OK;
724}
725
726
727/*
728 * TODO describe....
729 */
730SHOVE::SHOVE_STATUS SHOVE::onCollidingLine( LINE& aCurrent, LINE& aObstacle, int aNextRank )
731{
732 LINE shovedLine( aObstacle );
733
734 bool shoveOK = ShoveObstacleLine( aCurrent, aObstacle, shovedLine );
735
736 PNS_DBG( Dbg(), AddItem, &aObstacle, RED, 100000, wxT( "obstacle-line" ) );
737 PNS_DBG( Dbg(), AddItem, &aCurrent, GREEN, 150000, wxT( "current-line" ) );
738 PNS_DBG( Dbg(), AddItem, &shovedLine, BLUE, 200000, wxT( "shoved-line" ) );
739
740 if( shoveOK )
741 {
742#if 0
743 if( shovedLine.Marker() & MK_HEAD )
744 {
745 if( m_multiLineMode )
746 return SH_INCOMPLETE;
747
748 m_newHead = shovedLine;
749 }
750#endif
751 replaceLine( aObstacle, shovedLine, true, false );
752
753 shovedLine.SetRank( aNextRank );
754
755 if( !pushLineStack( shovedLine ) )
756 return SH_INCOMPLETE;
757
758 return SH_OK;
759 }
760
761 return SH_INCOMPLETE;
762}
763
764
765/*
766 * TODO describe....
767 */
768SHOVE::SHOVE_STATUS SHOVE::onCollidingSolid( LINE& aCurrent, ITEM* aObstacle, OBSTACLE& aObstacleInfo )
769{
770 LINE walkaroundLine( aCurrent );
771
772 if( aCurrent.EndsWithVia() )
773 {
774 VIA vh = aCurrent.Via();
775 VIA* via = nullptr;
776 const JOINT* jtStart = m_currentNode->FindJoint( vh.Pos(), &aCurrent );
777
778 if( !jtStart )
779 return SH_INCOMPLETE;
780
781 for( ITEM* item : jtStart->LinkList() )
782 {
783 if( item->OfKind( ITEM::VIA_T ) )
784 {
785 via = (VIA*) item;
786 break;
787 }
788 }
789
790 // TODO(JE) viastacks -- can aObstacle be a via?
791 if( via && via->Collide( aObstacle, m_currentNode, aObstacle->Layer() ) )
792 return onCollidingVia( aObstacle, via, aObstacleInfo, aObstacle->Rank() - 1 );
793 }
794
795 TOPOLOGY topo( m_currentNode );
796 TOPOLOGY::CLUSTER cluster = topo.AssembleCluster( aObstacle, aCurrent.Layers().Start(), 10.0 );
797
798 PNS_DBG( Dbg(), BeginGroup, "walk-cluster", 1 );
799
800 for( ITEM* item : cluster.m_items )
801 PNS_DBG( Dbg(), AddItem, item, RED, 10000, wxT( "cl-item" ) );
802
803 PNS_DBGN( Dbg(), EndGroup );
804
805 WALKAROUND walkaround( m_currentNode, Router() );
806 walkaround.SetDebugDecorator( Dbg() );
807 walkaround.SetSolidsOnly( false );
808 walkaround.RestrictToCluster( true, cluster );
810 walkaround.SetIterationLimit( Settings().WalkaroundIterationLimit() ); // fixme: make configurable
811
812 int currentRank = aCurrent.Rank();
813 int nextRank;
814
815 bool success = false;
816
817 PNS_DBG( Dbg(), AddItem, &aCurrent, RED, 10000, wxT( "current-line" ) );
818
819 for( int attempt = 0; attempt < 2; attempt++ )
820 {
821 if( attempt == 1 || Settings().JumpOverObstacles() )
822 nextRank = currentRank - 1;
823 else
824 nextRank = currentRank + 10000;
825
826 WALKAROUND::RESULT status = walkaround.Route( aCurrent );
827
828 if( status.status[ WALKAROUND::WP_SHORTEST ] != WALKAROUND::ST_DONE )//fixme policies!
829 continue;
830
831 walkaroundLine = status.lines[ WALKAROUND::WP_SHORTEST ];
832
833 walkaroundLine.ClearLinks();
834 walkaroundLine.Unmark();
835 walkaroundLine.Line().Simplify2();
836
837 if( walkaroundLine.HasLoops() )
838 continue;
839
840 PNS_DBG( Dbg(), AddItem, &walkaroundLine, BLUE, 10000, wxT( "walk-line" ) );
841
842#if 0
843 if( aCurrent.Marker() & MK_HEAD )
844 {
845 walkaroundLine.Mark( MK_HEAD );
846
847 if( m_multiLineMode )
848 continue;
849
850 m_newHead = walkaroundLine;
851 }
852#endif
853
854
855 if( !m_lineStack.empty() )
856 {
857 LINE lastLine = m_lineStack.front();
858
859 if( lastLine.Collide( &walkaroundLine, m_currentNode, lastLine.Layer() ) )
860 {
861 LINE dummy( lastLine );
862
863 if( ShoveObstacleLine( walkaroundLine, lastLine, dummy ) )
864 {
865 success = true;
866 break;
867 }
868 }
869 else
870 {
871 success = true;
872 break;
873 }
874 }
875 }
876
877 if(!success)
878 return SH_INCOMPLETE;
879
880 replaceLine( aCurrent, walkaroundLine, true, false );
881 walkaroundLine.SetRank( nextRank );
882
883
884 popLineStack();
885
886 if( !pushLineStack( walkaroundLine ) )
887 return SH_INCOMPLETE;
888
889 return SH_OK;
890}
891
892
893void SHOVE::pruneRootLines( NODE *aRemovedNode )
894{
895 PNS_DBG( Dbg(), Message, wxString::Format("prune called" ) );
896
897 NODE::ITEM_VECTOR added, removed;
898 aRemovedNode->GetUpdatedItems( removed, added );
899
900 for( const ITEM* item : added )
901 {
902 if( item->OfKind( ITEM::LINKED_ITEM_MASK_T ) )
903 {
904 const LINKED_ITEM* litem = static_cast<const LINKED_ITEM*>( item );
905
906 m_rootLineHistory.erase( litem->Uid() );
907 }
908 }
909}
910
911
912/*
913 * Pops NODE stackframes which no longer collide with aHeadSet. Optionally sets aDraggedVia
914 * to the dragged via of the last unpopped state.
915 */
917{
918 while( m_nodeStack.size() > 1 )
919 {
920 SPRINGBACK_TAG& spTag = m_nodeStack.back();
921
922 // Prevent the springback algo from erasing NODEs that might contain items used by the ROUTER_TOOL/LINE_PLACER.
923 // I noticed this can happen for the endItem provided to LINE_PLACER::Move() and cause a nasty crash.
925 break;
926
927 std::optional<OBSTACLE> obs = spTag.m_node->CheckColliding( aHeadSet );
928
929 if( !obs && !spTag.m_locked )
930 {
931 int i;
932
933 PNS_DBG( Dbg(), Message, wxString::Format( "pop-sp node=%p depth=%d", spTag.m_node, spTag.m_node->Depth() ) );
934
935 pruneRootLines( spTag.m_node );
936
937
938 delete spTag.m_node;
939 m_nodeStack.pop_back();
940 }
941 else
942 {
943 break;
944 }
945 }
946
947 if( m_nodeStack.empty() )
948 return m_root;
949
950 SPRINGBACK_TAG& spTag = m_nodeStack.back();
951
952 for( int i = 0; i < spTag.m_draggedVias.size(); i++ )
953 {
954 if (spTag.m_draggedVias[i].valid)
955 {
956 m_headLines[i].prevVia = m_headLines[i].theVia = spTag.m_draggedVias[i];
957 m_headLines[i].geometryModified = true;
958 PNS_DBG( Dbg(), Message,
959 wxString::Format( "restore-springback-via depth=%d %d %d %d %d ",
960 spTag.m_node->Depth(),
961 (int) m_nodeStack.size(),
962 m_headLines[i].theVia->pos.x,
963 m_headLines[i].theVia->pos.y,
964 m_headLines[i].theVia->layers.Start(),
965 m_headLines[i].theVia->layers.End() ) );
966 }
967 }
968
969 return m_nodeStack.back().m_node;
970}
971
972
973/*
974 * Push the current NODE on to the stack. aDraggedVia is the dragged via *before* the push
975 * (which will be restored in the event the stackframe is popped).
976 */
977bool SHOVE::pushSpringback( NODE* aNode, const OPT_BOX2I& aAffectedArea )
978{
980 OPT_BOX2I prev_area;
981
982 if( !m_nodeStack.empty() )
983 prev_area = m_nodeStack.back().m_affectedArea;
984
985 st.m_draggedVias.resize( m_headLines.size() );
986 int n = 0;
987
988 for( HEAD_LINE_ENTRY& head : m_headLines )
989 {
990 if ( head.theVia )
991 {
992 VIA_HANDLE vhandle = *head.theVia;
993
994 PNS_DBG( Dbg(), Message,
995 wxString::Format( "push-sp via depth=%d %d %d %d %d ", aNode->Depth(), vhandle.pos.x,
996 vhandle.pos.y,
997 vhandle.layers.Start(),
998 vhandle.layers.End() ) );
999
1000 st.m_draggedVias[n] = vhandle;
1001 }
1002
1003 n++;
1004 }
1005
1006 st.m_node = aNode;
1007
1008 if( aAffectedArea )
1009 {
1010 if( prev_area )
1011 st.m_affectedArea = prev_area->Merge( *aAffectedArea );
1012 else
1013 st.m_affectedArea = aAffectedArea;
1014 }
1015 else
1016 {
1017 st.m_affectedArea = prev_area;
1018 }
1019
1020 st.m_seq = ( m_nodeStack.empty() ? 1 : m_nodeStack.back().m_seq + 1 );
1021 st.m_locked = false;
1022
1023 m_nodeStack.push_back( st );
1024
1025 PNS_DBG( Dbg(), Message, wxString::Format( "push-sp depth=%d node=%p", st.m_node->Depth(), st.m_node ) );
1026
1027 return true;
1028}
1029
1030
1031/*
1032 * Push or shove a via by at least aForce. (The via might be pushed or shoved slightly further
1033 * to keep it from landing on an existing joint.)
1034 */
1035SHOVE::SHOVE_STATUS SHOVE::pushOrShoveVia( VIA* aVia, const VECTOR2I& aForce, int aNewRank, bool aDontUnwindStack )
1036{
1037 LINE_PAIR_VEC draggedLines;
1038 VECTOR2I p0( aVia->Pos() );
1039 const JOINT* jt = m_currentNode->FindJoint( p0, aVia );
1040 VECTOR2I p0_pushed( p0 + aForce );
1041
1042 PNS_DBG( Dbg(), Message, wxString::Format( wxT( "via force [%d %d]\n" ), aForce.x, aForce.y ) );
1043
1044 // nothing to do...
1045 if ( aForce.x == 0 && aForce.y == 0 )
1046 return SH_OK;
1047
1048 if( !jt )
1049 {
1050 PNS_DBG( Dbg(), Message, wxT( "weird, can't find the center-of-via joint\n" ) );
1051 return SH_INCOMPLETE;
1052 }
1053
1054 if( Settings().ShoveVias() == false || aVia->IsLocked() )
1055 return SH_TRY_WALK;
1056
1057 if( jt->IsLocked() )
1058 return SH_INCOMPLETE;
1059
1060 // make sure pushed via does not overlap with any existing joint
1061 while( true )
1062 {
1063 const JOINT* jt_next = m_currentNode->FindJoint( p0_pushed, aVia );
1064
1065 if( !jt_next )
1066 break;
1067
1068 p0_pushed += aForce.Resize( 2 );
1069 }
1070
1071 std::unique_ptr<VIA> pushedVia = Clone( *aVia );
1072 pushedVia->SetPos( p0_pushed );
1073 pushedVia->Mark( aVia->Marker() );
1074
1075 for( ITEM* item : jt->LinkList() )
1076 {
1077 if( item->OfKind( ITEM::SEGMENT_T | ITEM::ARC_T ) )
1078 {
1079 LINKED_ITEM* li = static_cast<LINKED_ITEM*>( item );
1080 LINE_PAIR lp;
1081 int segIndex;
1082
1083 lp.first = assembleLine( li, &segIndex );
1084
1085 if( lp.first.HasLockedSegments() )
1086 return SH_TRY_WALK;
1087
1088 assert( segIndex == 0 || ( segIndex == ( lp.first.SegmentCount() - 1 ) ) );
1089
1090 if( segIndex == 0 )
1091 lp.first.Reverse();
1092
1093 lp.second = lp.first;
1094 lp.second.ClearLinks();
1095 lp.second.DragCorner( p0_pushed, lp.second.CLine().Find( p0 ) );
1096 lp.second.Line().Simplify2();
1097 draggedLines.push_back( lp );
1098 }
1099 }
1100
1101 pushedVia->SetRank( aNewRank );
1102 PNS_DBG( Dbg(), Message, wxString::Format("via rank %d, fanout %d\n", pushedVia->Rank(), (int) draggedLines.size() ) );
1103
1104 PNS_DBG( Dbg(), AddPoint, aVia->Pos(), LIGHTGREEN, 100000, "via-pre" );
1105 PNS_DBG( Dbg(), AddPoint, pushedVia->Pos(), LIGHTRED, 100000, "via-post" );
1106
1107 VIA *v2 = pushedVia.get();
1108
1109 if( !aDontUnwindStack )
1110 unwindLineStack( aVia );
1111
1112 replaceItems( aVia, std::move( pushedVia ) );
1113
1114 if( draggedLines.empty() ) // stitching via? make sure the router won't forget about it
1115 {
1116 LINE tmpLine;
1117 tmpLine.LinkVia( v2 );
1118 if( !pushLineStack( tmpLine ) )
1119 return SH_INCOMPLETE;
1120 }
1121
1122 int n = 0;
1123 for( LINE_PAIR lp : draggedLines )
1124 {
1125 if( !aDontUnwindStack )
1126 unwindLineStack( &lp.first );
1127
1128 PNS_DBG( Dbg(), Message, wxString::Format("fan %d/%d\n", n, (int) draggedLines.size() ) );
1129 n++;
1130
1131 if( lp.second.SegmentCount() )
1132 {
1133 lp.second.ClearLinks();
1134 ROOT_LINE_ENTRY* rootEntry = replaceLine( lp.first, lp.second, true, true );
1135
1136 lp.second.LinkVia( v2 );
1137
1138 if( !aDontUnwindStack )
1139 unwindLineStack( &lp.second );
1140
1141 lp.second.SetRank( aNewRank );
1142
1143 if( rootEntry )
1144 rootEntry->newLine = lp.second; // fixme: it's inelegant
1145
1146
1147 PNS_DBG( Dbg(), Message, wxString::Format("PushViaF %p %d eov %d\n", &lp.second, lp.second.SegmentCount(), lp.second.EndsWithVia()?1:0 ) );
1148
1149 if( !pushLineStack( lp.second ) ) //, true ) ) // WHY?
1150 return SH_INCOMPLETE;
1151 }
1152 else
1153 {
1154 m_currentNode->Remove( lp.first );
1155 }
1156
1157 PNS_DBG( Dbg(), AddItem, &lp.first, LIGHTGREEN, 10000, wxT( "fan-pre" ) );
1158 PNS_DBG( Dbg(), AddItem, &lp.second, LIGHTRED, 10000, wxT( "fan-post" ) );
1159 }
1160
1161 return SH_OK;
1162}
1163
1164
1165/*
1166 * Calculate the minimum translation vector required to resolve a collision with a via and
1167 * shove the via by that distance.
1168 */
1169SHOVE::SHOVE_STATUS SHOVE::onCollidingVia( ITEM* aCurrent, VIA* aObstacleVia, OBSTACLE& aObstacleInfo, int aNextRank )
1170{
1171 assert( aObstacleVia );
1172
1173 int clearance = getClearance( aCurrent, aObstacleVia );
1174 VECTOR2I mtv;
1175 int rank = -1;
1176
1177 bool lineCollision = false;
1178 bool viaCollision = false;
1179 bool solidCollision = false;
1180 VECTOR2I mtvLine, mtvVia, mtvSolid;
1181
1182 PNS_DBG( Dbg(), BeginGroup, "push-via-by-line", 1 );
1183
1184 if( aCurrent->OfKind( ITEM::LINE_T ) )
1185 {
1186 VIA vtmp ( *aObstacleVia );
1187 int layer = aCurrent->Layer();
1188
1189 if( aObstacleInfo.m_maxFanoutWidth > 0
1190 && aObstacleInfo.m_maxFanoutWidth > aObstacleVia->Diameter( layer ) )
1191 {
1192 vtmp.SetDiameter( layer, aObstacleInfo.m_maxFanoutWidth );
1193 }
1194
1195 LINE* currentLine = (LINE*) aCurrent;
1196
1197 PNS_DBG( Dbg(), AddItem, currentLine, LIGHTRED, 10000, wxT( "current-line" ) );
1198
1199 if( currentLine->EndsWithVia() )
1200 {
1201 PNS_DBG( Dbg(), AddItem, &currentLine->Via(), LIGHTRED, 10000, wxT( "current-line-via" ) );
1202 }
1203
1204 PNS_DBG( Dbg(), AddItem, vtmp.Clone(), LIGHTRED, 100000, wxT( "orig-via" ) );
1205
1206 lineCollision = vtmp.Shape( layer )->Collide( currentLine->Shape( -1 ),
1207 clearance + currentLine->Width() / 2,
1208 &mtvLine );
1209
1210 // Check the via if present. Via takes priority.
1211 if( currentLine->EndsWithVia() )
1212 {
1213 const VIA& currentVia = currentLine->Via();
1214 int viaClearance = getClearance( &currentVia, &vtmp );
1215 VECTOR2I layerMtv;
1216
1217 for( int viaLayer : currentVia.RelevantShapeLayers( &vtmp ) )
1218 {
1219 viaCollision |= currentVia.Shape( viaLayer )->Collide( vtmp.Shape( viaLayer ),
1220 viaClearance,
1221 &layerMtv );
1222
1223 if( layerMtv.SquaredEuclideanNorm() > mtvVia.SquaredEuclideanNorm() )
1224 mtvVia = layerMtv;
1225 }
1226 }
1227 }
1228 else if( aCurrent->OfKind( ITEM::SOLID_T ) )
1229 {
1230 PNS_DBG( Dbg(), Message, wxT("collidee-is-solid" ) );
1231 // TODO(JE) if this case is real, handle via stacks
1232 solidCollision = aCurrent->Shape( -1 )->Collide( aObstacleVia->Shape( -1 ), clearance,
1233 &mtvSolid );
1234 //PNS_DBGN( Dbg(), EndGroup );
1235
1236 // TODO: is this possible at all? We don't shove solids.
1237 //return SH_INCOMPLETE;
1238 }
1239
1240 // fixme: we may have a sign issue in Collide(CIRCLE, LINE_CHAIN)
1241 if( viaCollision )
1242 mtv = -mtvVia;
1243 else if ( lineCollision )
1244 mtv = -mtvLine;
1245 else if ( solidCollision )
1246 mtv = -mtvSolid;
1247 else
1248 mtv = VECTOR2I(0, 0);
1249
1250 SHOVE::SHOVE_STATUS st = pushOrShoveVia( aObstacleVia, -mtv, aNextRank );
1251 PNS_DBG( Dbg(), Message, wxString::Format("push-or-shove-via st %d", st ) );
1252
1253 PNS_DBGN( Dbg(), EndGroup );
1254
1255 return st;
1256}
1257
1258
1259/*
1260 * TODO describe....
1261 */
1262SHOVE::SHOVE_STATUS SHOVE::onReverseCollidingVia( LINE& aCurrent, VIA* aObstacleVia, OBSTACLE& aObstacleInfo )
1263{
1264 int n = 0;
1265
1266 if( aCurrent.EndsWithVia() )
1267 {
1268 const VECTOR2I p0 = aCurrent.Via().Pos();
1269 const VECTOR2I p1 = aObstacleVia->Pos();
1270
1271 int layer = aCurrent.Layer();
1272 int dist = (p0 - p1).EuclideanNorm() - aCurrent.Via().Diameter( layer ) / 2
1273 - aObstacleVia->Diameter( layer ) / 2;
1274
1275 int clearance = getClearance( &aCurrent.Via(), aObstacleVia );
1276
1277 SHAPE_LINE_CHAIN hull = aObstacleVia->Hull( clearance, aCurrent.Width(), layer );
1278
1279 bool epInsideHull = hull.PointInside( p0 );
1280
1281 PNS_DBG( Dbg(), AddShape, &hull, LIGHTYELLOW, 100000, wxT( "obstacle-via-hull" ) );
1282 PNS_DBG( Dbg(), Message, wxString::Format("via2via coll check dist %d cl %d delta %d pi %d\n", dist, clearance, dist - clearance, epInsideHull ? 1 : 0) );
1283
1284 bool viaCollision = false;
1285
1286 for( int viaLayer : aCurrent.Via().RelevantShapeLayers( aObstacleVia ) )
1287 {
1288 viaCollision |=
1289 aCurrent.Via().Shape( viaLayer )->Collide( aObstacleVia->Shape( viaLayer ),
1290 clearance );
1291 }
1292
1293 if( viaCollision )
1294 {
1295 return onCollidingVia( &aCurrent, aObstacleVia, aObstacleInfo, aCurrent.Rank() - 1 );
1296 }
1297 }
1298
1299 LINE cur( aCurrent );
1300 cur.ClearLinks();
1301
1302 const JOINT* jt = m_currentNode->FindJoint( aObstacleVia->Pos(), aObstacleVia );
1303 LINE shoved( aCurrent );
1304 shoved.ClearLinks();
1305
1306 cur.RemoveVia();
1307 unwindLineStack( &aCurrent );
1308
1309 for( ITEM* item : jt->LinkList() )
1310 {
1311 if( item->OfKind( ITEM::SEGMENT_T | ITEM::ARC_T ) && item->LayersOverlap( &aCurrent ) )
1312 {
1313 LINKED_ITEM* li = static_cast<LINKED_ITEM*>( item );
1314 LINE head = assembleLine( li );
1315
1316 head.AppendVia( *aObstacleVia );
1317
1318 bool shoveOK = ShoveObstacleLine( head, cur, shoved );
1319
1320 if( !shoveOK )
1321 {
1322 PNS_DBG( Dbg(), BeginGroup, "on-reverse-via-fail-shove", m_iter );
1323 PNS_DBG( Dbg(), AddItem, aObstacleVia, LIGHTRED, 100000, wxT( "obstacle-via" ) );
1324 PNS_DBG( Dbg(), AddItem, &aCurrent, LIGHTGREEN, 10000, wxT( "current-line" ) );
1325 PNS_DBG( Dbg(), AddItem, &shoved, LIGHTRED, 10000, wxT( "shoved-line" ) );
1326
1327 if( aCurrent.EndsWithVia() )
1328 {
1329 PNS_DBG( Dbg(), AddItem, &aCurrent.Via(), LIGHTGREEN, 100000, wxT( "current-line-via" ) );
1330 }
1331
1332 PNS_DBGN( Dbg(), EndGroup );
1333
1334 return SH_INCOMPLETE;
1335 }
1336
1337 cur.SetShape( shoved.CLine() );
1338 n++;
1339 }
1340 }
1341
1342 if( !n )
1343 {
1344 PNS_DBG( Dbg(), BeginGroup, "on-reverse-via-fail-lonevia", m_iter );
1345 PNS_DBG( Dbg(), AddItem, aObstacleVia, LIGHTRED, 100000, wxT( "the-via" ) );
1346 PNS_DBG( Dbg(), AddItem, &aCurrent, LIGHTGREEN, 10000, wxT( "current-line" ) );
1347 PNS_DBGN( Dbg(), EndGroup );
1348
1349 LINE head( aCurrent );
1350 head.Line().Clear();
1351 head.AppendVia( *aObstacleVia );
1352 head.ClearLinks();
1353
1354 bool shoveOK = ShoveObstacleLine( head, aCurrent, shoved );
1355
1356 if( !shoveOK )
1357 return SH_INCOMPLETE;
1358
1359 cur.SetShape( shoved.CLine() );
1360 }
1361
1362 if( aCurrent.EndsWithVia() )
1363 shoved.AppendVia( aCurrent.Via() );
1364
1365 PNS_DBG( Dbg(), BeginGroup, "on-reverse-via", m_iter );
1366 PNS_DBG( Dbg(), AddItem, aObstacleVia, YELLOW, 0, wxT( "rr-the-via" ) );
1367 PNS_DBG( Dbg(), AddItem, &aCurrent, BLUE, 0, wxT( "rr-current-line" ) );
1368 PNS_DBG( Dbg(), AddItem, &shoved, GREEN, 0, wxT( "rr-shoved-line" ) );
1369 PNS_DBGN( Dbg(), EndGroup );
1370
1371 int currentRank = aCurrent.Rank();
1372 unwindLineStack( &aCurrent );
1373 replaceLine( aCurrent, shoved, true, false );
1374
1375 if( !pushLineStack( shoved ) )
1376 return SH_INCOMPLETE;
1377
1378 shoved.SetRank( currentRank );
1379
1380 return SH_OK;
1381}
1382
1383
1385{
1386 int d = 0;
1387
1388 for( std::vector<LINE>::iterator i = m_lineStack.begin(); i != m_lineStack.end() ; )
1389 {
1390 if( i->ContainsLink( aSeg ) )
1391 {
1392
1393// note to my future self: if we have a "tadpole" in the stack, keep track of the via even if the parent line has been deleted.
1394// otherwise - the via will be ignored in the case of collisions with tracks on another layer. Can happen pretty often in densely packed PCBs.
1395 if( i->EndsWithVia() && !aSeg->OfKind( ITEM::VIA_T ) )
1396 {
1397 VIA* via = nullptr;
1398
1399 for( LINKED_ITEM* l : i->Links() )
1400 {
1401 if( l->OfKind( ITEM::VIA_T ) )
1402 {
1403 via = static_cast<VIA*>( l );
1404 }
1405 }
1406
1407 if( via )
1408 {
1409 i->ClearLinks();
1410 i->Line().Clear();
1411 i->LinkVia( via );
1412 }
1413 i++;
1414 }
1415 else
1416 {
1417 i = m_lineStack.erase( i );
1418 }
1419 }
1420 else
1421 {
1422 i++;
1423 }
1424
1425 d++;
1426 }
1427
1428 for( std::vector<LINE>::iterator i = m_optimizerQueue.begin(); i != m_optimizerQueue.end() ; )
1429 {
1430 if( i->ContainsLink( aSeg ) && !aSeg->OfKind( ITEM::VIA_T ) )
1431 i = m_optimizerQueue.erase( i );
1432 else
1433 i++;
1434 }
1435}
1436
1437
1438void SHOVE::unwindLineStack( const ITEM* aItem )
1439{
1440 if( aItem->OfKind( ITEM::SEGMENT_T | ITEM::ARC_T ) )
1441 {
1442 unwindLineStack( static_cast<const LINKED_ITEM*>( aItem ) );
1443 }
1444 else if( aItem->OfKind( ITEM::LINE_T ) )
1445 {
1446 const LINE* l = static_cast<const LINE*>( aItem );
1447
1448 for( const LINKED_ITEM* seg : l->Links() )
1449 unwindLineStack( seg );
1450 }
1451}
1452
1453
1454bool SHOVE::pushLineStack( const LINE& aL, bool aKeepCurrentOnTop )
1455{
1456 if( !aL.IsLinked() && aL.SegmentCount() != 0 )
1457 {
1458 PNS_DBG( Dbg(), AddItem, &aL, BLUE, 10000, wxT( "push line stack failed" ) );
1459
1460 return false;
1461 }
1462
1463 if( aKeepCurrentOnTop && m_lineStack.size() > 0)
1464 {
1465 m_lineStack.insert( m_lineStack.begin() + m_lineStack.size() - 1, aL );
1466 }
1467 else
1468 {
1469 m_lineStack.push_back( aL );
1470 }
1471
1472
1474 m_optimizerQueue.push_back( aL );
1475
1476 return true;
1477}
1478
1479
1481{
1482 int pre = m_optimizerQueue.size();
1483 for( std::vector<LINE>::iterator i = m_optimizerQueue.begin(); i != m_optimizerQueue.end(); )
1484 {
1485 bool found = false;
1486
1487 for( LINKED_ITEM* s : aLine.Links() )
1488 {
1489 PNS_DBG( Dbg(), Message,
1490 wxString::Format( "cur lc %d lnk %p cnt %d", i->LinkCount(), s, aLine.LinkCount() ) );
1491
1492 if( i->ContainsLink( s ) && !s->OfKind( ITEM::VIA_T ) )
1493 {
1494
1495 i = m_optimizerQueue.erase( i );
1496 found = true;
1497 break;
1498 }
1499 }
1500
1501 if( !found )
1502 i++;
1503 }
1504
1505 return true;
1506}
1507
1509{
1510 LINE& l = m_lineStack.back();
1512 m_lineStack.pop_back();
1513}
1514
1515
1516bool SHOVE::fixupViaCollisions( const LINE* aCurrent, OBSTACLE& obs )
1517{
1518 int layer = aCurrent->Layer();
1519
1520 // if the current obstacle is a via, consider also the lines connected to it
1521 // if their widths are larger or equal than the via diameter, the shove algorithm
1522 // will very likely fail in the subsequent iterations (as our base assumption is track
1523 // ends can never move on their own, only dragged by force-propagated vias
1524
1525 // our colliding item is a via: just find the max width of the traces connected to it
1526 if( obs.m_item->OfKind( ITEM::VIA_T ) )
1527 {
1528 const VIA* v = static_cast<const VIA*>( obs.m_item );
1529 int maxw = 0;
1530 const JOINT* jv = m_currentNode->FindJoint( v->Pos(), v );
1531
1532 ITEM_SET links( jv->CLinks() );
1533
1534 for( ITEM* link : links )
1535 {
1536 if( link->OfKind( ITEM::SEGMENT_T ) ) // consider segments ...
1537 {
1538 const SEGMENT* seg = static_cast<const SEGMENT*>( link );
1539 maxw = std::max( seg->Width(), maxw );
1540 }
1541 else if( link->OfKind( ITEM::ARC_T ) ) // ... or arcs
1542 {
1543 const ARC* arc = static_cast<const ARC*>( link );
1544 maxw = std::max( arc->Width(), maxw );
1545 }
1546 }
1547
1548 obs.m_maxFanoutWidth = 0;
1549
1550 if( maxw > 0 && maxw >= v->Diameter( layer ) )
1551 {
1552 obs.m_maxFanoutWidth = maxw + 1;
1553 PNS_DBG( Dbg(), Message,
1554 wxString::Format( "Fixup via: new-w %d via-w %d", maxw, v->Diameter( layer ) ) );
1555
1556 return true;
1557 }
1558 return false;
1559 }
1560
1561
1562 // our colliding item is a segment. check if it has a via on either of the ends.
1563 if( !obs.m_item->OfKind( ITEM::SEGMENT_T ) )
1564 return false;
1565
1566 const SEGMENT* s = static_cast<const SEGMENT*>( obs.m_item );
1567 int sl = s->Layer();
1568
1569 const JOINT* ja = m_currentNode->FindJoint( s->Seg().A, s );
1570 const JOINT* jb = m_currentNode->FindJoint( s->Seg().B, s );
1571
1572 VIA* vias[] = { ja->Via(), jb->Via() };
1573
1574 for( int i = 0; i < 2; i++ )
1575 {
1576 VIA* v = vias[i];
1577
1578 // via diameter is larger than the segment width - cool, the force propagation algo
1579 // will be able to deal with it, no need to intervene
1580 if( !v || v->Diameter( sl ) > s->Width() )
1581 continue;
1582
1583 VIA vtest( *v );
1584 vtest.SetDiameter( sl, s->Width() );
1585
1586 // enlarge the via to the width of the segment
1587 if( vtest.Collide( aCurrent, m_currentNode, aCurrent->Layer() ) )
1588 {
1589 // if colliding, drop the segment in the shove iteration loop and force-propagate the via instead
1590 obs.m_item = v;
1591 obs.m_maxFanoutWidth = s->Width() + 1;
1592 return true;
1593 }
1594 }
1595 return false;
1596}
1597
1598bool SHOVE::patchTadpoleVia( ITEM* nearest, LINE& current )
1599{
1600 if (current.CLine().PointCount() < 1 )
1601 return false;
1602
1603// PNS_DBG(Dbg(), Message, wxString::Format( "cp %d %d", current.CLine().CPoint(-1).x, current.CLine().CPoint(-1).y ) );
1604
1605 auto jtViaEnd = m_currentNode->FindJoint( current.CLine().CPoint(-1), &current );
1606
1607// PNS_DBG(Dbg(), Message, wxString::Format( "jt %p", jtViaEnd ) );
1608
1609 if ( !jtViaEnd )
1610 return false;
1611
1612 auto viaEnd = jtViaEnd->Via();
1613
1614 if (! viaEnd )
1615 return false;
1616
1617 bool colliding = m_currentNode->CheckColliding( viaEnd ).has_value();
1618
1619// PNS_DBG(Dbg(), Message, wxString::Format( "patch-tadpole viaEnd %p colliding %d", viaEnd, colliding?1:0 ) );
1620
1621 if( viaEnd && !current.EndsWithVia() && colliding )
1622 {
1623 current.LinkVia( viaEnd );
1624 }
1625
1626 return false;
1627}
1628
1629/*
1630 * Resolve the next collision.
1631 */
1633{
1634 LINE currentLine = m_lineStack.back();
1635 NODE::OPT_OBSTACLE nearest;
1636 SHOVE_STATUS st = SH_NULL;
1637
1638 ROUTER_IFACE* iface = Router()->GetInterface();
1639
1640 if( Dbg() )
1641 Dbg()->SetIteration( aIter );
1642
1643 PNS_DBG( Dbg(), AddItem, &currentLine, RED, currentLine.Width(),
1644 wxString::Format( wxT( "current sc=%d net=%s evia=%d" ),
1645 currentLine.SegmentCount(),
1646 iface->GetNetName( currentLine.Net() ),
1647 currentLine.EndsWithVia() ? 1 : 0 ) );
1648
1650 {
1652 opts.m_kindMask = search_order;
1653 opts.m_filter = [ this ] ( const ITEM* item ) -> bool
1654 {
1655 bool rv = true;
1656
1658 {
1659 const LINKED_ITEM* litem = static_cast<const LINKED_ITEM*>( item );
1660 ROOT_LINE_ENTRY* ent = this->findRootLine( litem );
1661
1662 if( !ent && m_defaultPolicy & SHP_IGNORE )
1663 rv = false;
1664
1665 if( ent && ent->policy & SHP_IGNORE )
1666 rv = false;
1667 }
1668 else
1669 {
1671 rv = false;
1672 }
1673
1674 return rv;
1675 };
1676
1677 nearest = m_currentNode->NearestObstacle( &currentLine, opts );
1678
1679 if( nearest )
1680 {
1681 PNS_DBG( Dbg(), AddShape, nearest->m_item->Shape( currentLine.Layer() ), YELLOW, 10000,
1682 wxString::Format( "nearest %p %s rank %d",
1683 nearest->m_item,
1684 nearest->m_item->KindStr(),
1685 nearest->m_item->Rank() ) );
1686 }
1687
1688 if( nearest )
1689 break;
1690 }
1691
1692 if( !nearest )
1693 {
1694 m_lineStack.pop_back();
1695 PNS_DBG( Dbg(), Message, wxT( "no-nearest-item ") );
1696 return SH_OK;
1697 }
1698
1699 bool viaFixup = fixupViaCollisions( &currentLine, *nearest );
1700
1701 PNS_DBG( Dbg(), Message, wxString::Format( wxT( "iter %d: via-fixup %d" ), aIter, viaFixup?1:0 ) );
1702
1703
1704 ITEM* ni = nearest->m_item;
1705
1706 UNITS_PROVIDER up( pcbIUScale, EDA_UNITS::MM );
1707 PNS_DBG( Dbg(), Message, wxString::Format( wxT( "NI: %s (%s) %p %d" ),
1708 ni->Format(),
1709 ni->Parent() ? ni->Parent()->GetItemDescription( &up, false )
1710 : wxString( wxT( "null" ) ),
1711 ni,
1712 ni->OwningNode()->Depth() ) );
1713
1714 unwindLineStack( ni );
1715
1716 if( !ni->OfKind( ITEM::SOLID_T ) && ni->Rank() >= 0 && ni->Rank() > currentLine.Rank() )
1717 {
1718 // Collision with a higher-ranking object (ie: one that we've already shoved)
1719 //
1720 switch( ni->Kind() )
1721 {
1722 case ITEM::VIA_T:
1723 {
1724 PNS_DBG( Dbg(), BeginGroup, wxString::Format( wxT( "iter %d: reverse-collide-via" ), aIter ), 0 );
1725
1726 patchTadpoleVia( ni, currentLine );
1727
1728 // TODO(JE) viastacks -- via-via collisions here?
1729 if( currentLine.EndsWithVia()
1730 && nearest->m_item->Collide( &currentLine.Via(), m_currentNode,
1731 nearest->m_item->Layer() ) )
1732 {
1733 PNS_DBG( Dbg(), AddItem, nearest->m_item, YELLOW, 100000, wxT("v2v nearesti" ) );
1734 //PNS_DBG( Dbg(), AddItem, nearest->m_head,RED, 100000, wxString::Format("v2v nearesth force=%d,%d" ) );
1735
1736 st = onCollidingVia( &currentLine, (VIA*) ni, *nearest, ni->Rank() + 1 );
1737
1738 //ni->SetRank( currentLine.Rank() );
1739 }
1740 else
1741 {
1742 st = onReverseCollidingVia( currentLine, (VIA*) ni, *nearest );
1743 }
1744
1745 PNS_DBGN( Dbg(), EndGroup );
1746
1747 break;
1748 }
1749
1750 case ITEM::SEGMENT_T:
1751 {
1752 PNS_DBG( Dbg(), BeginGroup, wxString::Format( wxT( "iter %d: reverse-collide-segment" ),
1753 aIter ), 0 );
1754
1755 PNS_DBG( Dbg(), AddItem, ni, YELLOW, 100000, wxT("head" ) );
1756
1757 LINE revLine = assembleLine( static_cast<SEGMENT*>( ni ) );
1758
1759 popLineStack();
1760 unwindLineStack( &revLine );
1761 patchTadpoleVia( ni, currentLine );
1762
1763 if( currentLine.EndsWithVia()
1764 && currentLine.Via().Collide( (SEGMENT*) ni, m_currentNode, currentLine.Layer() ) )
1765 {
1766 VIA_HANDLE vh;
1767 vh.layers = currentLine.Via().Layers();
1768 vh.net = currentLine.Via().Net();
1769 vh.pos = currentLine.Via().Pos();
1770 vh.valid = true;
1771 auto rvia = m_currentNode->FindViaByHandle( vh );
1772 if ( !rvia )
1773 st = SH_INCOMPLETE;
1774 else
1775 st = onCollidingVia( &revLine, rvia, *nearest, revLine.Rank() + 1 );
1776 }
1777 else
1778 st = onCollidingLine( revLine, currentLine, revLine.Rank() + 1 );
1779
1780
1781 if( !pushLineStack( revLine ) )
1782 {
1783 return SH_INCOMPLETE;
1784 }
1785
1786
1787 PNS_DBGN( Dbg(), EndGroup );
1788
1789 break;
1790 }
1791
1792 case ITEM::ARC_T:
1793 {
1794 //TODO(snh): Handle Arc shove separate from track
1795 PNS_DBG( Dbg(), BeginGroup, wxString::Format( wxT( "iter %d: reverse-collide-arc " ), aIter ), 0 );
1796 LINE revLine = assembleLine( static_cast<ARC*>( ni ) );
1797
1798 popLineStack();
1799 st = onCollidingLine( revLine, currentLine, revLine.Rank() - 1 );
1800
1801 PNS_DBGN( Dbg(), EndGroup );
1802
1803 if( !pushLineStack( revLine ) )
1804 return SH_INCOMPLETE;
1805
1806 break;
1807 }
1808
1809 default:
1810 assert( false );
1811 }
1812 }
1813 else
1814 {
1815 // Collision with a lower-ranking object or a solid
1816 //
1817 switch( ni->Kind() )
1818 {
1819 case ITEM::SEGMENT_T:
1820 PNS_DBG( Dbg(), BeginGroup, wxString::Format( wxT( "iter %d: collide-segment " ), aIter ), 0 );
1821
1822 st = onCollidingSegment( currentLine, (SEGMENT*) ni );
1823
1824 if( st == SH_TRY_WALK )
1825 st = onCollidingSolid( currentLine, ni, *nearest );
1826
1827 PNS_DBGN( Dbg(), EndGroup );
1828
1829 break;
1830
1831 //TODO(snh): Customize Arc collide
1832 case ITEM::ARC_T:
1833 PNS_DBG( Dbg(), BeginGroup, wxString::Format( wxT( "iter %d: collide-arc " ), aIter ), 0 );
1834
1835 st = onCollidingArc( currentLine, static_cast<ARC*>( ni ) );
1836
1837 if( st == SH_TRY_WALK )
1838 st = onCollidingSolid( currentLine, ni, *nearest );
1839
1840 PNS_DBGN( Dbg(), EndGroup );
1841
1842 break;
1843
1844 case ITEM::VIA_T:
1845 PNS_DBG( Dbg(), BeginGroup, wxString::Format( wxT( "iter %d: collide-via (fixup: %d)" ), aIter, 0 ), 0 );
1846 st = onCollidingVia( &currentLine, (VIA*) ni, *nearest, currentLine.Rank() - 1 );
1847
1848 if( st == SH_TRY_WALK )
1849 st = onCollidingSolid( currentLine, ni, *nearest );
1850
1851 PNS_DBGN( Dbg(), EndGroup );
1852
1853 break;
1854
1855 case ITEM::HOLE_T:
1856 case ITEM::SOLID_T:
1857 PNS_DBG( Dbg(), BeginGroup, wxString::Format( wxT( "iter %d: walk-solid " ), aIter ), 0);
1858 st = onCollidingSolid( currentLine, ni, *nearest );
1859
1860 PNS_DBGN( Dbg(), EndGroup );
1861
1862 break;
1863
1864 default:
1865 break;
1866 }
1867 }
1868
1869 return st;
1870}
1871
1872
1873/*
1874 * Resolve collisions.
1875 * Each iteration pushes the next colliding object out of the way. Iterations are continued as
1876 * long as they propagate further collisions, or until the iteration timeout or max iteration
1877 * count is reached.
1878 */
1880{
1881 SHOVE_STATUS st = SH_OK;
1882
1884
1885 PNS_DBG( Dbg(), Message, wxString::Format( "ShoveStart [root: %d jts, current: %d jts]",
1886 m_root->JointCount(),
1888
1889 int iterLimit = Settings().ShoveIterationLimit();
1890 TIME_LIMIT timeLimit = Settings().ShoveTimeLimit();
1891
1892 m_iter = 0;
1893
1894 timeLimit.Restart();
1895
1896 if( m_lineStack.empty() && m_draggedVia )
1897 {
1898 // If we're shoving a free via then push a proxy LINE (with the via on the end) onto
1899 // the stack.
1901 }
1902
1903 while( !m_lineStack.empty() )
1904 {
1905 PNS_DBG( Dbg(), Message, wxString::Format( "iter %d: node %p stack %d ", m_iter,
1906 m_currentNode, (int) m_lineStack.size() ) );
1907
1908 st = shoveIteration( m_iter );
1909
1910 m_iter++;
1911
1912 if( st == SH_INCOMPLETE || timeLimit.Expired() || m_iter >= iterLimit )
1913 {
1914 PNS_DBG( Dbg(), Message, wxString::Format( "Fail [time limit expired: %d iter %d iter limit %d",
1915 timeLimit.Expired()?1:0, m_iter, iterLimit ) );
1916 st = SH_INCOMPLETE;
1917 break;
1918 }
1919 }
1920
1921 return st;
1922}
1923
1924
1926{
1927 OPT_BOX2I area;
1928
1929 if( !m_nodeStack.empty() )
1930 area = m_nodeStack.back().m_affectedArea;
1931
1932 if( area && m_affectedArea)
1933 area->Merge( *m_affectedArea );
1934 else if( !area )
1935 area = m_affectedArea;
1936
1937 return area;
1938}
1939
1940
1942{
1943 for( const LINKED_ITEM* link : aLine.Links() )
1944 {
1945 auto it = m_rootLineHistory.find( link->Uid() );
1946
1947 if( it != m_rootLineHistory.end() )
1948 return it->second;
1949 }
1950
1951 return nullptr;
1952}
1953
1955{
1956 auto it = m_rootLineHistory.find( aItem->Uid() );
1957
1958 if( it != m_rootLineHistory.end() )
1959 return it->second;
1960
1961 return nullptr;
1962}
1963
1964
1966{
1967 for( const LINKED_ITEM* link : aLine.Links() )
1968 {
1969 auto it = m_rootLineHistory.find( link->Uid() );
1970
1971 if( it != m_rootLineHistory.end() )
1972 {
1973 PNS_DBG( Dbg(), Message, wxString::Format( wxT( "touch [found] uid=%llu type=%s"), link->Uid(), link->KindStr() ) );
1974
1975 return it->second;
1976 }
1977 }
1978
1979 auto rootEntry = new ROOT_LINE_ENTRY( aLine.Clone() );
1980
1981
1982 for( const LINKED_ITEM* link : aLine.Links() )
1983 {
1984 PNS_DBG( Dbg(), Message, wxString::Format( wxT( "touch [create] uid=%llu type=%s"), link->Uid(), link->KindStr() ) );
1985 m_rootLineHistory[link->Uid()] = rootEntry;
1986 }
1987
1988
1989 return rootEntry;
1990}
1991
1992
1994{
1995 auto it = m_rootLineHistory.find( aItem->Uid() );
1996
1997 if( it != m_rootLineHistory.end() )
1998 {
1999 PNS_DBG( Dbg(), Message, wxString::Format( wxT( "touch [create] uid=%llu"), aItem->Uid() ) );
2000 return it->second;
2001 }
2002
2003 auto rootEntry = new ROOT_LINE_ENTRY( nullptr );
2004
2005 PNS_DBG( Dbg(), Message, wxString::Format( wxT( "touch [create] uid=%llu"), aItem->Uid() ) );
2006 m_rootLineHistory[ aItem->Uid() ] = rootEntry;
2007
2008 return rootEntry;
2009}
2010
2011
2013{
2014 OPTIMIZER optimizer( aNode );
2015 int optFlags = 0;
2016 int n_passes = 0;
2017
2019
2021
2022 int maxWidth = 0;
2023
2024 for( LINE& line : m_optimizerQueue )
2025 maxWidth = std::max( line.Width(), maxWidth );
2026
2027 if( area )
2028 {
2029 area->Inflate( maxWidth );
2030 area = area->Intersect( VisibleViewArea() );
2031 }
2032
2033 switch( effort )
2034 {
2035 case OE_LOW:
2036 optFlags |= OPTIMIZER::MERGE_OBTUSE;
2037 n_passes = 1;
2038 break;
2039
2040 case OE_MEDIUM:
2041 optFlags |= OPTIMIZER::MERGE_SEGMENTS;
2042 n_passes = 2;
2043 break;
2044
2045 case OE_FULL:
2046 optFlags = OPTIMIZER::MERGE_SEGMENTS;
2047 n_passes = 2;
2048 break;
2049
2050 default:
2051 break;
2052 }
2053
2055
2056 if( area )
2057 {
2058 SHAPE_RECT r( *area );
2059
2060 PNS_DBG( Dbg(), AddShape, &r, BLUE, 0, wxT( "opt-area" ) );
2061
2062 optFlags |= OPTIMIZER::RESTRICT_AREA;
2063 optimizer.SetRestrictArea( *area, false );
2064 }
2065
2067
2068 // Smart Pads is incompatible with 90-degree mode for now
2069 if( Settings().SmartPads()
2070 && ( cornerMode == DIRECTION_45::MITERED_45 || cornerMode == DIRECTION_45::ROUNDED_45 ) )
2071 {
2072 optFlags |= OPTIMIZER::SMART_PADS;
2073 }
2074
2075
2076 optimizer.SetEffortLevel( optFlags & ~m_optFlagDisableMask );
2077 optimizer.SetCollisionMask( ITEM::ANY_T );
2078
2079 std::set<const ITEM*> itemsChk;
2080
2081 auto iface = Router()->GetInterface();
2082
2083 for( int pass = 0; pass < n_passes; pass++ )
2084 {
2085 std::reverse( m_optimizerQueue.begin(), m_optimizerQueue.end() );
2086
2087 PNS_DBG( Dbg(), Message, wxString::Format( wxT( "optimize %d lines, pass %d"), (int)m_optimizerQueue.size(), (int)pass ) );
2088
2089 for( int i = 0; i < m_optimizerQueue.size(); i++ )
2090 {
2091 LINE& lineToOpt = m_optimizerQueue[i];
2092 LINE* rootLine = nullptr;
2093 auto rootEntry = findRootLine( lineToOpt );
2094
2095 if( rootEntry )
2096 {
2097 rootLine = rootEntry->rootLine;
2098
2099 if( rootEntry->policy & SHP_DONT_OPTIMIZE )
2100 continue;
2101 if( rootEntry->isHead )
2102 continue;
2103 }
2104
2105 LINE optimized;
2106 if( optimizer.Optimize( &lineToOpt, &optimized, rootLine ) )
2107 {
2108 assert( optimized.LinkCount() == 0 );
2109
2110 //PNS_DBG( Dbg(), AddShape, &lineToOpt.CLine(), BLUE, 0, wxT( "shove-pre-opt" ) );
2111 //if( rootLine )
2112 // PNS_DBG( Dbg(), AddItem, rootLine, RED, 0, wxT( "shove-root-opt" ) );
2113
2114 replaceLine( lineToOpt, optimized, false, aNode );
2115 m_optimizerQueue[i] = optimized; // keep links in the lines in the queue up to date
2116
2117 //PNS_DBG( Dbg(), AddShape, &optimized.CLine(), GREEN, 0, wxT( "shove-post-opt" ) );
2118 }
2119 }
2120 }
2121}
2122
2123
2125{
2126 return m_currentNode ? m_currentNode : m_root; //m_nodeStack.empty() ? m_root : m_nodeStack.back().m_node;
2127}
2128
2129
2131{
2132 SPRINGBACK_TAG sp;
2133 sp.m_node = aNode;
2134 sp.m_locked = true;
2135
2136 m_nodeStack.push_back(sp);
2137
2138 PNS_DBG( Dbg(), Message, wxString::Format( "addLockedSPNode node=%p stack=%d\n", sp.m_node, (int) m_nodeStack.size() ) );
2139
2140 return true;
2141}
2142
2143
2145{
2146 bool found = false;
2147
2148 auto iter = m_nodeStack.begin();
2149
2150 while( iter != m_nodeStack.end() )
2151 {
2152 if ( iter->m_node == aNode )
2153 {
2154 found = true;
2155 break;
2156 }
2157
2158 iter++;
2159 }
2160
2161 if( !found )
2162 return false;
2163
2164 auto start = iter;
2165
2166 aNode->KillChildren();
2167 m_nodeStack.erase( start, m_nodeStack.end() );
2168
2169 if( !m_nodeStack.empty() )
2170 m_currentNode = m_nodeStack.back().m_node;
2171 else
2173
2174 return true;
2175}
2176
2177
2179{
2180 if( m_nodeStack.empty() )
2181 return false;
2182
2183 while( !m_nodeStack.back().m_locked && m_nodeStack.size() > 1 )
2184 m_nodeStack.pop_back();
2185
2186 m_currentNode = m_nodeStack.back().m_node;
2187
2188 return m_nodeStack.back().m_locked;
2189}
2190
2191
2193{
2194 auto iter = m_nodeStack.begin();
2195
2196 while( iter != m_nodeStack.end() )
2197 {
2198 if ( iter->m_node == aNode )
2199 {
2200 iter->m_locked = false;
2201 break;
2202 }
2203
2204 iter++;
2205 }
2206}
2207
2208
2210{
2211 m_optFlagDisableMask = aMask;
2212}
2213
2214
2216{
2218}
2219
2220
2222{
2223 m_defaultPolicy = aPolicy;
2224}
2225
2226
2227void SHOVE::SetShovePolicy( const LINKED_ITEM* aItem, int aPolicy )
2228{
2229 auto rl = touchRootLine( aItem );
2230 rl->policy = aPolicy;
2231}
2232
2233void SHOVE::SetShovePolicy( const LINE& aLine, int aPolicy )
2234{
2235 auto rl = touchRootLine( aLine );
2236 rl->policy = aPolicy;
2237}
2238
2239
2241{
2242 m_headLines.clear();
2243}
2244
2245
2246void SHOVE::AddHeads( const LINE& aHead, int aPolicy )
2247{
2248 m_headLines.push_back( SHOVE::HEAD_LINE_ENTRY( aHead, aPolicy ) );
2249}
2250
2251
2252void SHOVE::AddHeads( VIA_HANDLE aHead, VECTOR2I aNewPos, int aPolicy )
2253{
2254 SHOVE::HEAD_LINE_ENTRY ent( aHead, aPolicy );
2255 ent.viaNewPos = aNewPos;
2256 ent.prevVia = aHead;
2257 ent.theVia = aHead;
2258 m_headLines.push_back( ent );
2259}
2260
2261void removeHead( NODE *aNode, LINE& head )
2262{
2263 for (auto lnk : head.Links() )
2264 {
2265 if( lnk->BelongsTo( aNode ) )
2266 aNode->Remove( lnk );
2267 }
2268}
2269
2271{
2272 auto iface = Router()->GetInterface();
2273
2274 NODE::ITEM_VECTOR removed, added;
2275
2276 m_currentNode->GetUpdatedItems( removed, added );
2277
2278 for( auto& item : added )
2279 {
2280 auto rootEntry = findRootLine( static_cast<LINKED_ITEM*>( item ) );
2281 if( rootEntry && rootEntry->isHead )
2282 {
2283 m_currentNode->Remove( item );
2284 }
2285 }
2286}
2287
2288
2289void SHOVE::reconstructHeads( bool aShoveFailed )
2290{
2291 int i = 0;
2292 auto iface = Router()->GetInterface();
2293
2294 PNS_DBG( Dbg(), Message, wxString::Format("reconstructing %zu heads", m_headLines.size() ) );
2295
2296 for( auto& headEntry : m_headLines )
2297 {
2298 if( headEntry.origHead )
2299 {
2300 auto rootEntry = findRootLine( *headEntry.origHead );
2301
2302 PNS_DBG( Dbg(), Message, wxString::Format("orig LinkC=%d RE=%p", headEntry.origHead->LinkCount(), rootEntry ) );
2303
2304 assert( rootEntry );
2305 assert( rootEntry->rootLine );
2306
2307 if( rootEntry->newLine )
2308 {
2309 headEntry.newHead = rootEntry->newLine;
2310 headEntry.geometryModified = !rootEntry->newLine->CLine().CompareGeometry( rootEntry->rootLine->CLine() );
2311
2312 wxString msg = wxString::Format(
2313 "head %d/%d [net %-20s]: root %p, lc-root %d, lc-new %d\n", i, (int) m_headLines.size(),
2314 iface->GetNetName( rootEntry->rootLine->Net() ).c_str().AsChar(), rootEntry->rootLine, rootEntry->rootLine->LinkCount(), headEntry.newHead->LinkCount() );
2315 PNS_DBG( Dbg(), AddItem, rootEntry->rootLine, CYAN, 0, msg );
2316 PNS_DBG( Dbg(), Message, msg );
2317
2318 }
2319 else
2320 {
2321 wxString msg = wxString::Format(
2322 "head %d/%d [net %-20s]: unmodified, lc-orig %d\n", i, (int) m_headLines.size(),
2323 iface->GetNetName( headEntry.origHead->Net() ).c_str().AsChar(),
2324 headEntry.origHead->LinkCount() );
2325 PNS_DBG( Dbg(), Message, msg );
2326 }
2327
2328 i++;
2329 } else {
2330 auto rootEntry = findRootLine( headEntry.draggedVia );
2331
2332 if( rootEntry->newVia )
2333 {
2334 headEntry.geometryModified = true;
2335 headEntry.theVia = VIA_HANDLE( rootEntry->newVia->Pos(), rootEntry->newVia->Layers(), rootEntry->newVia->Net() );
2336 auto chk = m_currentNode->FindViaByHandle( *headEntry.theVia );
2337 wxString msg = wxString::Format( "[modif] via orig %p chk %p\n", headEntry.draggedVia, chk );
2338
2339 PNS_DBG( Dbg(), Message, msg );
2340 assert( chk != nullptr );
2341 }
2342 else
2343 {
2344 headEntry.theVia = VIA_HANDLE( rootEntry->oldVia->Pos(), rootEntry->oldVia->Layers(), rootEntry->oldVia->Net() );
2345 auto chk = m_currentNode->FindViaByHandle( *headEntry.theVia );
2346 wxString msg = wxString::Format( "[unmodif] via orig %p chk %p\n", headEntry.draggedVia, chk );
2347 PNS_DBG( Dbg(), Message, msg );
2348 assert( chk != nullptr );
2349
2350 }
2351
2352
2353 }
2354
2355 m_headsModified |= headEntry.geometryModified;
2356 }
2357}
2358
2359
2360
2362{
2363 //COLLISION_SEARCH_CONTEXT ctx;
2364
2365 //ctx.options.m_differentNetsOnly = false;
2366 //ctx.options.m_kindMask = ITEM::SEGMENT_T; // fixme arcs
2367
2368 SHAPE_LINE_CHAIN orig( aOld->CLine() );
2369
2370 int vc_prev = orig.PointCount();
2371 orig.Simplify2();
2372 int vc_post = orig.PointCount();
2373
2374 *aNew = *aOld;
2375
2376 PNS_DBG( Dbg(), Message, wxString::Format( "**** PreshoveCleanup %d -> %d\n", vc_prev, vc_post ) );
2377
2378 if( vc_prev != vc_post )
2379 {
2380 aNew->ClearLinks();
2381 aNew->SetShape( orig );
2382 replaceLine( *aOld, *aNew );
2383 return true;
2384 }
2385
2386 return false;
2387}
2388
2389// new algo
2391{
2392 SHOVE_STATUS st = SH_OK;
2393
2394 m_multiLineMode = false;
2395 int currentHeadId = 0;
2396 int totalHeads = m_headLines.size();
2397
2398 m_headsModified = false;
2399 m_lineStack.clear();
2400 m_optimizerQueue.clear();
2401
2402 ITEM_SET headSet;
2403
2404 PNS_DBG( Dbg(), Message, wxString::Format("shove run (heads: %d, currentNode=%p, depth=%d)", (int) m_headLines.size(), m_currentNode, m_currentNode->Depth() ) );
2405
2406 for( auto& l : m_headLines )
2407 {
2408 if( l.theVia )
2409 {
2410 PNS_DBG( Dbg(), Message, wxString::Format("process head-via [%d %d] node=%p", l.theVia->pos.x, l.theVia->pos.y, m_currentNode ) );
2411 auto realVia = m_currentNode->FindViaByHandle( *l.theVia );
2412 assert( realVia != nullptr );
2413 headSet.Add( realVia->Clone() );
2414 }
2415 else
2416 {
2417 headSet.Add( *l.origHead->Clone() );
2418 }
2419 }
2420
2421 // Pop NODEs containing previous shoves which are no longer necessary
2422 NODE* parent = reduceSpringback( headSet );
2423 m_currentNode = parent->Branch();
2425
2426
2427
2428
2429
2430
2431 //nodeStats( Dbg(), m_currentNode, "right-after-branch" );
2432
2433 auto iface = Router()->GetInterface();
2434
2435 // for ( auto& hq : m_headLines )
2436 // if( hq.oldHead )
2437 // m_currentNode->Remove( *hq.oldHead );
2438
2439
2440 // Push the via to its new location
2441 for( auto& headLineEntry : m_headLines )
2442 {
2443 //if( rootEntry->line ) // head already processed in previous steps
2444 //{
2445 // PNS_DBG( Dbg(), Message, wxString::Format( "RL found" ) );
2446
2447 //continue;
2448 //}
2450
2451 if( headLineEntry.theVia )
2452 {
2453 VIA* viaToDrag = m_currentNode->FindViaByHandle( *headLineEntry.theVia );
2454
2455 if( !viaToDrag )
2456 {
2457 st = SH_INCOMPLETE;
2458 break;
2459 }
2460
2461 auto viaRoot = touchRootLine( viaToDrag );
2462 viaRoot->oldVia = viaToDrag;
2463 headLineEntry.draggedVia = viaToDrag;
2464
2465 st = pushOrShoveVia( viaToDrag, ( headLineEntry.viaNewPos - viaToDrag->Pos() ), 0, true );
2466
2467 if( st != SH_OK )
2468 break;
2469 }
2470 else
2471 {
2472 // Create a new NODE to store this version of the world
2473 assert( headLineEntry.origHead->LinkCount() == 0 );
2474 m_currentNode->Add( *headLineEntry.origHead );
2475
2476 //nodeStats( Dbg(), m_currentNode, "add-head" );
2477
2478
2479
2480 PNS_DBG( Dbg(), Message,
2481 wxString::Format( "touchRoot ohlc %d roots %d re=%p\n",
2482 headLineEntry.origHead->LinkCount(),
2483 (int) m_rootLineHistory.size(),
2484 findRootLine( *headLineEntry.origHead ) ) );
2485
2486
2487 LINE head( *headLineEntry.origHead );
2488
2489 // empty head? nothing to shove...
2490 if( !head.SegmentCount() && !head.EndsWithVia() )
2491 {
2492 st = SH_INCOMPLETE;
2493 break;
2494 }
2495
2496 currentHeadId++;
2497
2498 if( head.PointCount() > 0 )
2499 {
2500 m_currentNode->LockJoint( head.CPoint( 0 ), &head, true );
2501 }
2502
2503 if( !head.EndsWithVia() )
2504 m_currentNode->LockJoint( head.CPoint( -1 ), &head, true );
2505
2506 SetShovePolicy( head, headLineEntry.policy );
2507
2508 //head.Mark( MK_HEAD );
2509 head.SetRank( 100000 ); //- 100 * currentHeadId );
2510
2511 if( head.EndsWithVia() )
2512 {
2513 auto headVia = Clone( head.Via() );
2514 headVia->SetRank( 100000 ); // - 100 * currentHeadId );
2515 headLineEntry.origHead->LinkVia( headVia.get() );
2516 head.LinkVia( headVia.get() );
2517 m_currentNode->Add( std::move( headVia ) );
2518 }
2519
2520 auto headRoot = touchRootLine( *headLineEntry.origHead );
2521 headRoot->isHead = true;
2522 headRoot->rootLine = new PNS::LINE( *headLineEntry.origHead );
2523 headRoot->policy = headLineEntry.policy;
2524 if( head.EndsWithVia() )
2525 {
2526 m_rootLineHistory[ headLineEntry.origHead->Via().Uid() ] = headRoot;
2527 }
2528
2529
2530 PNS_DBG( Dbg(), Message,
2531 wxString::Format( "headLC %d, rlLC %d oolc %d eov %d\n", head.LinkCount(),
2532 headRoot->rootLine->LinkCount(),
2533 headLineEntry.origHead->LinkCount(),
2534 head.EndsWithVia()?1:0 ) );
2535
2536 //auto rootEntry = findRootLine( &head );
2537
2538 PNS_DBG( Dbg(), Message,
2539 wxString::Format( "Shove heads %d/%d h-lc=%d net=%s Line=%d Policy=%s",
2540 currentHeadId, totalHeads, head.LinkCount(),
2541 iface->GetNetName( head.Net() ), headRoot->newLine ? 1 : 0,
2542 headRoot ? formatPolicy( headRoot->policy )
2543 : wxString( wxT( "default[ne]" ) ) ) );
2544
2545
2546 // nodeStats( Dbg(), m_currentNode, "pre-push-stack" );
2547
2548 if( !pushLineStack( head ) )
2549 {
2550 st = SH_INCOMPLETE;
2551 break;
2552 }
2553 }
2554
2555 st = shoveMainLoop();
2556
2557 //nodeStats( Dbg(), m_currentNode, "post-main-loop" );
2558
2559 if( st != SH_OK )
2560 break;
2561 };
2562
2563 PNS_DBG( Dbg(), Message,
2564 wxString::Format( "Shove status : %s after %d iterations, heads: %d",
2565 ( ( st == SH_OK || st == SH_HEAD_MODIFIED ) ? "OK" : "FAILURE" ),
2566 m_iter, (int) m_headLines.size() ) );
2567 if( st == SH_OK )
2568 {
2569 //nodeStats( Dbg(), m_currentNode, "pre-opt" );
2570
2572
2573 reconstructHeads( false );
2574 removeHeads();
2575
2576 // this must be called afrter reconstructHeads as it requires up-to-date via handles
2578 }
2579 else
2580 {
2581 //reconstructHeads( true );
2582
2583 for( auto& headEntry : m_headLines )
2584 {
2585 if( headEntry.prevVia )
2586 {
2587
2588 PNS_DBG( Dbg(), Message,
2589 wxString::Format( "Fail-restore via mod [%d, %d] orig [%d, %d]",
2590 headEntry.theVia->pos.x,
2591 headEntry.theVia->pos.y,
2592 headEntry.prevVia->pos.x,
2593 headEntry.prevVia->pos.y ) );
2594
2595 headEntry.theVia = headEntry.prevVia;
2596 headEntry.geometryModified = true;
2597 m_headsModified = true;
2598 }
2599 }
2600
2602
2603 delete m_currentNode;
2604 m_currentNode = parent;
2605 }
2606
2607 return st;
2608}
2609
2611 {
2617 SHP_DONT_OPTIMIZE = 0x10
2619
2620const wxString SHOVE::formatPolicy( int aPolicy )
2621{
2622 if( aPolicy == SHP_DEFAULT )
2623 return wxT( "default" );
2624
2625 wxString rv;
2626
2627 if( aPolicy & SHP_SHOVE )
2628 rv.Append( "shove ");
2629 if( aPolicy & SHP_WALK_FORWARD )
2630 rv.Append( "walk-forward ");
2631 if( aPolicy & SHP_WALK_FORWARD )
2632 rv.Append( "walk-back ");
2633 if( aPolicy & SHP_IGNORE )
2634 rv.Append( "ignore ");
2635 if( aPolicy & SHP_IGNORE )
2636 rv.Append( "dont-optimize ");
2637
2638 return rv;
2639}
2640
2641bool SHOVE::HeadsModified( int aIndex ) const
2642{
2643 if( aIndex < 0 )
2644 return m_headsModified;
2645 else
2646 return m_headLines[ aIndex ].geometryModified;
2647}
2648
2649const PNS::LINE SHOVE::GetModifiedHead( int aIndex ) const
2650{
2651 return *m_headLines[ aIndex ].newHead;
2652}
2653
2654const VIA_HANDLE SHOVE::GetModifiedHeadVia( int aIndex ) const
2655{
2656 return *m_headLines[ aIndex ].theVia;
2657}
2658
2659
2660
2661}
2662
constexpr EDA_IU_SCALE pcbIUScale
Definition: base_units.h:108
std::optional< BOX2I > OPT_BOX2I
Definition: box2.h:926
constexpr BOX2I KiROUND(const BOX2D &aBoxD)
Definition: box2.h:990
CORNER_MODE
Corner modes.
Definition: direction45.h:67
@ ROUNDED_45
H/V/45 with filleted corners.
Definition: direction45.h:69
@ MITERED_45
H/V/45 with mitered corners (default)
Definition: direction45.h:68
virtual wxString GetItemDescription(UNITS_PROVIDER *aUnitsProvider, bool aFull) const
Return a user-visible description string of this item.
Definition: eda_item.cpp:118
Base class for all P&S algorithms (shoving, walkaround, line placement, dragging, etc....
Definition: pns_algo_base.h:43
const BOX2I & VisibleViewArea() const
void SetDebugDecorator(DEBUG_DECORATOR *aDecorator)
Assign a debug decorator allowing this algo to draw extra graphics for visual debugging.
Definition: pns_algo_base.h:73
ROUTER * Router() const
Return current router settings.
Definition: pns_algo_base.h:54
ROUTING_SETTINGS & Settings() const
Return the logger object, allowing to dump geometry to a file.
DEBUG_DECORATOR * Dbg() const
Definition: pns_algo_base.h:78
int Width() const override
Definition: pns_arc.h:88
virtual void SetIteration(int iter)
void Add(const LINE &aLine)
Definition: pns_itemset.cpp:33
Base class for PNS router board items.
Definition: pns_item.h:98
BOARD_ITEM * Parent() const
Definition: pns_item.h:199
virtual const std::string Format() const
Definition: pns_item.cpp:336
virtual int Rank() const
Definition: pns_item.h:266
virtual const SHAPE * Shape(int aLayer) const
Return the geometrical shape of the item.
Definition: pns_item.h:242
const PNS_LAYER_RANGE & Layers() const
Definition: pns_item.h:212
virtual NET_HANDLE Net() const
Definition: pns_item.h:210
PnsKind Kind() const
Return the type (kind) of the item.
Definition: pns_item.h:173
std::set< int > RelevantShapeLayers(const ITEM *aOther) const
Returns the set of layers on which either this or the other item can have a unique shape.
Definition: pns_item.cpp:94
virtual int Layer() const
Definition: pns_item.h:216
PnsKind
< Supported item types
Definition: pns_item.h:102
@ SEGMENT_T
Definition: pns_item.h:107
@ LINKED_ITEM_MASK_T
Definition: pns_item.h:113
bool Collide(const ITEM *aHead, const NODE *aNode, int aLayer, COLLISION_SEARCH_CONTEXT *aCtx=nullptr) const
Check for a collision (clearance violation) with between us and item aOther.
Definition: pns_item.cpp:303
virtual const NODE * OwningNode() const
Definition: pns_item.cpp:352
bool OfKind(int aKindMask) const
Definition: pns_item.h:181
bool LayersOverlap(const ITEM *aOther) const
Return true if the set of layers spanned by aOther overlaps our layers.
Definition: pns_item.h:221
virtual HOLE * Hole() const
Definition: pns_item.h:304
virtual bool HasHole() const
Definition: pns_item.h:303
bool IsLocked() const
Definition: pns_item.h:278
virtual int Marker() const
Definition: pns_item.h:263
A 2D point on a given set of layers and belonging to a certain net, that links together a number of b...
Definition: pns_joint.h:43
const std::vector< ITEM * > & LinkList() const
Definition: pns_joint.h:303
VIA * Via() const
Definition: pns_joint.h:275
const ITEM_SET & CLinks() const
Definition: pns_joint.h:308
bool IsLocked() const
Definition: pns_joint.h:357
Represents a track on a PCB, connecting two non-trivial joints (that is, vias, pads,...
Definition: pns_line.h:62
const VECTOR2I & CPoint(int aIdx) const
Definition: pns_line.h:146
bool HasLoops() const
Definition: pns_line.cpp:1190
bool HasLockedSegments() const
Definition: pns_line.cpp:1300
int Rank() const override
Definition: pns_line.cpp:1121
void SetShape(const SHAPE_LINE_CHAIN &aLine)
Return the shape of the line.
Definition: pns_line.h:127
virtual void Mark(int aMarker) const override
Definition: pns_line.cpp:127
void LinkVia(VIA *aVia)
Definition: pns_line.cpp:1099
const SHAPE_LINE_CHAIN & CLine() const
Definition: pns_line.h:138
void RemoveVia()
Definition: pns_line.cpp:1319
void SetRank(int aRank) override
Definition: pns_line.cpp:1111
SHAPE_LINE_CHAIN & Line()
Definition: pns_line.h:137
const SHAPE * Shape(int aLayer) const override
Modifiable accessor to the underlying shape.
Definition: pns_line.h:134
virtual int Marker() const override
Definition: pns_line.cpp:146
void AppendVia(const VIA &aVia)
Definition: pns_line.cpp:1086
VIA & Via()
Definition: pns_line.h:198
int SegmentCount() const
Definition: pns_line.h:140
virtual void Unmark(int aMarker=-1) const override
Definition: pns_line.cpp:137
int PointCount() const
Definition: pns_line.h:141
bool Walkaround(SHAPE_LINE_CHAIN aObstacle, SHAPE_LINE_CHAIN &aPre, SHAPE_LINE_CHAIN &aWalk, SHAPE_LINE_CHAIN &aPost, bool aCw) const
Calculate a line tightly wrapping a convex hull of an obstacle object (aObstacle).
bool EndsWithVia() const
Definition: pns_line.h:190
const SEG CSegment(int aIdx) const
Set line width.
Definition: pns_line.h:147
virtual LINE * Clone() const override
Return a deep copy of the item.
Definition: pns_line.cpp:119
int Width() const
Return true if the line is geometrically identical as line aOther.
Definition: pns_line.h:157
UNIQ_ID Uid() const
Keep the router "world" - i.e.
Definition: pns_node.h:231
NODE * Branch()
Create a lightweight copy (called branch) of self that tracks the changes (added/removed items) wrs t...
Definition: pns_node.cpp:143
std::vector< ITEM * > ITEM_VECTOR
Definition: pns_node.h:242
int GetClearance(const ITEM *aA, const ITEM *aB, bool aUseClearanceEpsilon=true) const
Return the pre-set worst case clearance between any pair of items.
Definition: pns_node.cpp:129
void Replace(ITEM *aOldItem, std::unique_ptr< ITEM > aNewItem)
Replace an item with another one.
Definition: pns_node.cpp:869
void GetUpdatedItems(ITEM_VECTOR &aRemoved, ITEM_VECTOR &aAdded)
Return the list of items removed and added in this branch with respect to the root branch.
Definition: pns_node.cpp:1466
OPT_OBSTACLE CheckColliding(const ITEM *aItem, int aKindMask=ITEM::ANY_T)
Check if the item collides with anything else in the world, and if found, returns the obstacle.
Definition: pns_node.cpp:410
const JOINT * FindJoint(const VECTOR2I &aPos, int aLayer, NET_HANDLE aNet) const
Search for a joint at a given position, layer and belonging to given net.
Definition: pns_node.cpp:1265
std::optional< OBSTACLE > OPT_OBSTACLE
Definition: pns_node.h:241
int Depth() const
Definition: pns_node.h:281
bool Add(std::unique_ptr< SEGMENT > aSegment, bool aAllowRedundant=false)
Add an item to the current node.
Definition: pns_node.cpp:665
OPT_OBSTACLE NearestObstacle(const LINE *aLine, const COLLISION_SEARCH_OPTIONS &aOpts=COLLISION_SEARCH_OPTIONS())
Follow the line in search of an obstacle that is nearest to the starting to the line's starting point...
Definition: pns_node.cpp:286
void LockJoint(const VECTOR2I &aPos, const ITEM *aItem, bool aLock)
Definition: pns_node.cpp:1292
int JointCount() const
Return the number of nodes in the inheritance chain (wrs to the root node).
Definition: pns_node.h:275
VIA * FindViaByHandle(const VIA_HANDLE &handle) const
Definition: pns_node.cpp:1749
void KillChildren()
Definition: pns_node.cpp:1546
void ClearRanks(int aMarkerMask=MK_HEAD|MK_VIOLATION)
Definition: pns_node.cpp:1581
void Remove(ARC *aArc)
Remove an item from this branch.
Definition: pns_node.cpp:909
const LINE AssembleLine(LINKED_ITEM *aSeg, int *aOriginSegmentIndex=nullptr, bool aStopAtLockedJoints=false, bool aFollowLockedSegments=false)
Follow the joint map to assemble a line connecting two non-trivial joints starting from segment aSeg.
Definition: pns_node.cpp:1047
Perform various optimizations of the lines being routed, attempting to make the lines shorter and les...
Definition: pns_optimizer.h:95
void SetRestrictArea(const BOX2I &aArea, bool aStrict=true)
void SetCollisionMask(int aMask)
void SetEffortLevel(int aEffort)
static bool Optimize(LINE *aLine, int aEffortLevel, NODE *aWorld, const VECTOR2I &aV=VECTOR2I(0, 0))
@ LIMIT_CORNER_COUNT
Do not attempt to optimize if the resulting line's corner count is outside the predefined range.
@ SMART_PADS
Reroute pad exits.
@ MERGE_SEGMENTS
Reduce corner cost iteratively.
Definition: pns_optimizer.h:99
@ MERGE_OBTUSE
Reduce corner cost by merging obtuse segments.
virtual wxString GetNetName(PNS::NET_HANDLE aNet) const =0
virtual DEBUG_DECORATOR * GetDebugDecorator()=0
ROUTER_IFACE * GetInterface() const
Definition: pns_router.h:225
TIME_LIMIT ShoveTimeLimit() const
PNS_OPTIMIZATION_EFFORT OptimizerEffort() const
Set the optimizer effort. Bigger means cleaner traces, but slower routing.
DIRECTION_45::CORNER_MODE GetCornerMode() const
const SEG & Seg() const
Definition: pns_segment.h:93
int Width() const override
Definition: pns_segment.h:88
const SHAPE_LINE_CHAIN Hull(int aClearance, int aWalkaroundThickness, int aLayer=-1) const override
Definition: pns_line.cpp:573
void SetSpringbackDoNotTouchNode(const NODE *aNode)
Definition: pns_shove.cpp:2215
SHOVE_STATUS shoveIteration(int aIter)
Definition: pns_shove.cpp:1632
SHOVE_STATUS shoveMainLoop()
Definition: pns_shove.cpp:1879
std::vector< SHAPE_LINE_CHAIN > HULL_SET
Definition: pns_shove.h:110
void reconstructHeads(bool aShoveFailed)
Definition: pns_shove.cpp:2289
int m_iter
Definition: pns_shove.h:244
std::pair< LINE, LINE > LINE_PAIR
Definition: pns_shove.h:112
SHOVE_STATUS pushOrShoveVia(VIA *aVia, const VECTOR2I &aForce, int aNextRank, bool aDontUnwindStack=false)
Definition: pns_shove.cpp:1035
SHOVE_STATUS onReverseCollidingVia(LINE &aCurrent, VIA *aObstacleVia, OBSTACLE &aObstacleInfo)
Definition: pns_shove.cpp:1262
OPT_BOX2I m_affectedArea
Definition: pns_shove.h:252
std::vector< LINE > m_lineStack
Definition: pns_shove.h:233
void popLineStack()
Definition: pns_shove.cpp:1508
ROOT_LINE_ENTRY * findRootLine(const LINE &aLine)
Definition: pns_shove.cpp:1941
std::vector< SPRINGBACK_TAG > m_nodeStack
Definition: pns_shove.h:232
@ SHP_DONT_OPTIMIZE
Definition: pns_shove.h:65
@ SHP_IGNORE
Definition: pns_shove.h:64
@ SHP_DEFAULT
Definition: pns_shove.h:60
@ SHP_WALK_FORWARD
Definition: pns_shove.h:62
bool patchTadpoleVia(ITEM *nearest, LINE &current)
Definition: pns_shove.cpp:1598
SHOVE_STATUS onCollidingArc(LINE &aCurrent, ARC *aObstacleArc)
Definition: pns_shove.cpp:681
bool shoveLineToHullSet(const LINE &aCurLine, const LINE &aObstacleLine, LINE &aResultLine, const HULL_SET &aHulls, bool aPermitAdjustingStart=false, bool aPermitAdjustingEnd=false)
Definition: pns_shove.cpp:309
NODE * m_currentNode
Definition: pns_shove.h:240
SHOVE_STATUS Run()
Definition: pns_shove.cpp:2390
LINE assembleLine(const LINKED_ITEM *aSeg, int *aIndex=nullptr, bool aPreCleanup=false)
Definition: pns_shove.cpp:218
bool checkShoveDirection(const LINE &aCurLine, const LINE &aObstacleLine, const LINE &aShovedLine) const
Definition: pns_shove.cpp:242
int m_forceClearance
Definition: pns_shove.h:246
SHOVE_STATUS onCollidingSegment(LINE &aCurrent, SEGMENT *aObstacleSeg)
Definition: pns_shove.cpp:619
SHOVE(NODE *aWorld, ROUTER *aRouter)
Definition: pns_shove.cpp:193
void DisablePostShoveOptimizations(int aMask)
Definition: pns_shove.cpp:2209
bool RewindSpringbackTo(NODE *aNode)
Definition: pns_shove.cpp:2144
bool pushLineStack(const LINE &aL, bool aKeepCurrentOnTop=false)
Definition: pns_shove.cpp:1454
@ SH_INCOMPLETE
Definition: pns_shove.h:53
@ SH_HEAD_MODIFIED
Definition: pns_shove.h:54
@ SH_TRY_WALK
Definition: pns_shove.h:55
OPT_BOX2I totalAffectedArea() const
Definition: pns_shove.cpp:1925
bool RewindToLastLockedNode()
Definition: pns_shove.cpp:2178
std::deque< HEAD_LINE_ENTRY > m_headLines
Definition: pns_shove.h:235
int m_defaultPolicy
Definition: pns_shove.h:251
const wxString formatPolicy(int aPolicy)
Definition: pns_shove.cpp:2620
bool ShoveObstacleLine(const LINE &aCurLine, const LINE &aObstacleLine, LINE &aResultLine)
Definition: pns_shove.cpp:501
SHOVE_STATUS onCollidingVia(ITEM *aCurrent, VIA *aObstacleVia, OBSTACLE &aObstacleInfo, int aNextRank)
Definition: pns_shove.cpp:1169
void sanityCheck(LINE *aOld, LINE *aNew)
Definition: pns_shove.cpp:186
int m_restrictSpringbackTagId
Definition: pns_shove.h:242
void runOptimizer(NODE *aNode)
Definition: pns_shove.cpp:2012
ROOT_LINE_ENTRY * touchRootLine(const LINE &aLine)
Definition: pns_shove.cpp:1965
const PNS::LINE GetModifiedHead(int aIndex) const
Definition: pns_shove.cpp:2649
bool HeadsModified(int aIndex=-1) const
Definition: pns_shove.cpp:2641
void UnlockSpringbackNode(NODE *aNode)
Definition: pns_shove.cpp:2192
NODE * m_root
Definition: pns_shove.h:239
NODE * reduceSpringback(const ITEM_SET &aHeadSet)
Definition: pns_shove.cpp:916
bool AddLockedSpringbackNode(NODE *aNode)
Definition: pns_shove.cpp:2130
void SetDefaultShovePolicy(int aPolicy)
Definition: pns_shove.cpp:2221
void AddHeads(const LINE &aHead, int aPolicy=SHP_DEFAULT)
Definition: pns_shove.cpp:2246
void SetShovePolicy(const LINKED_ITEM *aItem, int aPolicy)
Definition: pns_shove.cpp:2227
void unwindLineStack(const LINKED_ITEM *aSeg)
Definition: pns_shove.cpp:1384
bool preShoveCleanup(LINE *aOld, LINE *aNew)
Definition: pns_shove.cpp:2361
NODE * CurrentNode()
Definition: pns_shove.cpp:2124
int getClearance(const ITEM *aA, const ITEM *aB) const
Definition: pns_shove.cpp:169
bool m_multiLineMode
Definition: pns_shove.h:247
std::vector< LINE > m_optimizerQueue
Definition: pns_shove.h:234
SHOVE_STATUS onCollidingSolid(LINE &aCurrent, ITEM *aObstacle, OBSTACLE &aObstacleInfo)
Definition: pns_shove.cpp:768
bool pushSpringback(NODE *aNode, const OPT_BOX2I &aAffectedArea)
Definition: pns_shove.cpp:977
int m_optFlagDisableMask
Definition: pns_shove.h:249
SHOVE_STATUS onCollidingLine(LINE &aCurrent, LINE &aObstacle, int aNextRank)
Definition: pns_shove.cpp:730
bool fixupViaCollisions(const LINE *aCurrent, OBSTACLE &obs)
Definition: pns_shove.cpp:1516
bool pruneLineFromOptimizerQueue(const LINE &aLine)
Definition: pns_shove.cpp:1480
void replaceItems(ITEM *aOld, std::unique_ptr< ITEM > aNew)
Definition: pns_shove.cpp:49
bool shoveLineFromLoneVia(const LINE &aCurLine, const LINE &aObstacleLine, LINE &aResultLine)
Definition: pns_shove.cpp:262
VIA * m_draggedVia
Definition: pns_shove.h:243
bool m_headsModified
Definition: pns_shove.h:245
const NODE * m_springbackDoNotTouchNode
Definition: pns_shove.h:241
void ClearHeads()
Definition: pns_shove.cpp:2240
std::unordered_map< LINKED_ITEM::UNIQ_ID, ROOT_LINE_ENTRY * > m_rootLineHistory
Definition: pns_shove.h:237
void pruneRootLines(NODE *aRemovedNode)
Definition: pns_shove.cpp:893
const VIA_HANDLE GetModifiedHeadVia(int aIndex) const
Definition: pns_shove.cpp:2654
ROOT_LINE_ENTRY * replaceLine(LINE &aOld, LINE &aNew, bool aIncludeInChangedArea=true, bool aAllowRedundantSegments=true, NODE *aNode=nullptr)
Definition: pns_shove.cpp:80
std::vector< LINE_PAIR > LINE_PAIR_VEC
Definition: pns_shove.h:113
void removeHeads()
Definition: pns_shove.cpp:2270
bool Expired() const
Definition: time_limit.cpp:39
const CLUSTER AssembleCluster(ITEM *aStart, int aLayer, double aAreaExpansionLimit=0.0, NET_HANDLE aExcludedNet=nullptr)
int Diameter(int aLayer) const
Definition: pns_via.h:196
void SetDiameter(int aLayer, int aDiameter)
Definition: pns_via.h:203
const VECTOR2I & Pos() const
Definition: pns_via.h:180
const SHAPE * Shape(int aLayer) const override
Return the geometrical shape of the item.
Definition: pns_via.h:234
const SHAPE_LINE_CHAIN Hull(int aClearance=0, int aWalkaroundThickness=0, int aLayer=-1) const override
Definition: pns_via.cpp:217
VIA * Clone() const override
Return a deep copy of the item.
Definition: pns_via.cpp:235
void SetIterationLimit(const int aIterLimit)
void RestrictToCluster(bool aEnabled, const TOPOLOGY::CLUSTER &aCluster)
void SetSolidsOnly(bool aSolidsOnly)
STATUS Route(const LINE &aInitialPath, LINE &aWalkPath, bool aOptimize=true)
void SetAllowedPolicies(std::vector< WALK_POLICY > aPolicies)
int Start() const
Definition: pns_layerset.h:86
int End() const
Definition: pns_layerset.h:91
VECTOR2I A
Definition: seg.h:49
VECTOR2I B
Definition: seg.h:50
static double DefaultAccuracyForPCB()
Definition: shape_arc.h:276
A dynamic state checking if a point lies within polygon with a dynamically built outline ( with each ...
void AddPolyline(const SHAPE_LINE_CHAIN &aPolyline)
bool PointInside(const VECTOR2I &aPt, int aAccuracy=0, bool aUseBBoxCache=false) const override
Check if point aP lies inside a closed shape.
Represent a polyline containing arcs as well as line segments: A chain of connected line and/or arc s...
const SHAPE_LINE_CHAIN Reverse() const
Reverse point order in the line chain.
int PointCount() const
Return the number of points (vertices) in this line chain.
void Clear()
Remove all points from the line chain.
SHAPE_LINE_CHAIN & Simplify2(bool aRemoveColinear=true)
void Append(int aX, int aY, bool aAllowDuplication=false)
Append a new point at the end of the line chain.
const VECTOR2I & CPoint(int aIndex) const
Return a reference to a given point in the line chain.
const VECTOR2I NearestPoint(const VECTOR2I &aP, bool aAllowInternalShapePoints=true) const
Find a point on the line chain that is closest to point aP.
const std::string Format(bool aCplusPlus=true) const override
bool IsArcSegment(size_t aSegment) const
void Insert(size_t aVertex, const VECTOR2I &aP)
long long int Length() const
Return length of the line chain in Euclidean metric.
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
constexpr extended_type SquaredEuclideanNorm() const
Compute the squared euclidean norm of the vector, which is defined as (x ** 2 + y ** 2).
Definition: vector2d.h:307
VECTOR2_TRAITS< int32_t >::extended_type extended_type
Definition: vector2d.h:73
VECTOR2< T > Resize(T aNewLength) const
Return a vector of the same direction, but length specified in aNewLength.
Definition: vector2d.h:385
@ LIGHTGREEN
Definition: color4d.h:63
@ WHITE
Definition: color4d.h:48
@ BLUE
Definition: color4d.h:56
@ LIGHTGRAY
Definition: color4d.h:47
@ LIGHTYELLOW
Definition: color4d.h:49
@ GREEN
Definition: color4d.h:57
@ CYAN
Definition: color4d.h:58
@ YELLOW
Definition: color4d.h:67
@ LIGHTRED
Definition: color4d.h:65
@ RED
Definition: color4d.h:59
Push and Shove diff pair dimensions (gap) settings dialog.
SHOVE_POLICY
Definition: pns_shove.cpp:2611
@ SHP_DEFAULT
Definition: pns_shove.cpp:2612
@ SHP_WALK_BACK
Definition: pns_shove.cpp:2615
@ SHP_IGNORE
Definition: pns_shove.cpp:2616
@ SHP_WALK_FORWARD
Definition: pns_shove.cpp:2614
@ SHP_SHOVE
Definition: pns_shove.cpp:2613
@ SHP_DONT_OPTIMIZE
Definition: pns_shove.cpp:2617
@ MK_HEAD
Definition: pns_item.h:43
OPT_BOX2I ChangedArea(const ITEM *aItemA, const ITEM *aItemB)
Definition: pns_utils.cpp:369
void removeHead(NODE *aNode, LINE &head)
Definition: pns_shove.cpp:2261
std::unique_ptr< typename std::remove_const< T >::type > Clone(const T &aItem)
Definition: pns_item.h:344
#define PNS_DBG(dbg, method,...)
#define PNS_DBGN(dbg, method)
VECTOR2I::extended_type ecoord
Definition: pns_shove.cpp:45
std::vector< FAB_LAYER_COLOR > dummy
std::function< bool(const ITEM *)> m_filter
Definition: pns_node.h:119
Hold an object colliding with another object, along with some useful data about the collision.
Definition: pns_node.h:87
int m_maxFanoutWidth
worst case (largest) width of the tracks connected to the item
Definition: pns_node.h:94
ITEM * m_item
Item found to be colliding with m_head.
Definition: pns_node.h:89
Definition: pns_shove.h:130
std::optional< VIA_HANDLE > theVia
Definition: pns_shove.h:145
std::optional< VIA_HANDLE > prevVia
Definition: pns_shove.h:144
VECTOR2I viaNewPos
Definition: pns_shove.h:147
Definition: pns_shove.h:116
int policy
Definition: pns_shove.h:125
std::optional< LINE > newLine
Definition: pns_shove.h:124
VIA * newVia
Definition: pns_shove.h:123
std::vector< VIA_HANDLE > m_draggedVias
Definition: pns_shove.h:163
std::set< ITEM * > m_items
Definition: pns_topology.h:47
VECTOR2I pos
Definition: pns_via.h:53
NET_HANDLE net
Definition: pns_via.h:55
PNS_LAYER_RANGE layers
Definition: pns_via.h:54
LINE lines[MaxWalkPolicies]
STATUS status[MaxWalkPolicies]
int clearance
VECTOR2I v2(1, 0)
VECTOR2< int32_t > VECTOR2I
Definition: vector2d.h:695