From 5349f47f95d5048168e2e4f3a68592af9c4a98cd Mon Sep 17 00:00:00 2001 From: Jon Evans Date: Sun, 6 Jun 2021 14:57:41 -0400 Subject: [PATCH] PNS: Prevent walkaround churn in unsolvable situations Fixes https://gitlab.com/kicad/code/kicad/-/issues/8558 --- pcbnew/router/pns_walkaround.cpp | 19 +++++++++++++++---- 1 file changed, 15 insertions(+), 4 deletions(-) diff --git a/pcbnew/router/pns_walkaround.cpp b/pcbnew/router/pns_walkaround.cpp index 444e2cb2ea..f541817217 100644 --- a/pcbnew/router/pns_walkaround.cpp +++ b/pcbnew/router/pns_walkaround.cpp @@ -60,10 +60,11 @@ WALKAROUND::WALKAROUND_STATUS WALKAROUND::singleStep( LINE& aPath, bool aWinding if( !current_obs ) return DONE; + VECTOR2I initialLast = aPath.CPoint( -1 ); + SHAPE_LINE_CHAIN path_walk; - bool s_cw = aPath.Walkaround( current_obs->m_hull, path_walk, - aWindingDirection ); + bool s_cw = aPath.Walkaround( current_obs->m_hull, path_walk, aWindingDirection ); PNS_DBG( Dbg(), BeginGroup, "hull/walk" ); char name[128]; @@ -76,11 +77,21 @@ WALKAROUND::WALKAROUND_STATUS WALKAROUND::singleStep( LINE& aPath, bool aWinding PNS_DBG( Dbg(), Message, wxString::Format( "Stat cw %d", !!s_cw ) ); PNS_DBGN( Dbg(), EndGroup ); - current_obs = nearestObstacle( LINE( aPath, path_walk ) ); - path_walk.Simplify(); aPath.SetShape( path_walk ); + // If the end of the line is inside an obstacle, additional walkaround iterations are not + // going to help. Also, if one walkaround step didn't produce a new last point, additional + // steps won't either. Exit now to prevent pegging the iteration limiter and causing lag. + if( initialLast == path_walk.CLastPoint() || + ( current_obs && current_obs->m_hull.PointInside( initialLast ) && + !current_obs->m_hull.PointOnEdge( initialLast ) ) ) + { + return ALMOST_DONE; + } + + current_obs = nearestObstacle( LINE( aPath, path_walk ) ); + return IN_PROGRESS; }