From 8a0c225efa7365a6d05e5849c64d641ef7537e4d Mon Sep 17 00:00:00 2001 From: Jeff Young Date: Thu, 6 Apr 2023 15:33:49 +0100 Subject: [PATCH] Unify most special-case processing to the clearance if-then-else. This also allows us to support hole-to-hole clearance when dragging a via near another via on the same net. Fixes https://gitlab.com/kicad/code/kicad/issues/12781 --- pcbnew/router/pns_item.cpp | 76 +++++++++++++++++-------------- pcbnew/router/pns_kicad_iface.cpp | 2 + 2 files changed, 44 insertions(+), 34 deletions(-) diff --git a/pcbnew/router/pns_item.cpp b/pcbnew/router/pns_item.cpp index 4bbf81203d..999b0f9c9c 100644 --- a/pcbnew/router/pns_item.cpp +++ b/pcbnew/router/pns_item.cpp @@ -103,20 +103,6 @@ bool ITEM::collideSimple( const ITEM* aHead, const NODE* aNode, if( aHead->m_kind == LINE_T ) lineWidthH = static_cast( aHead )->Width() / 2; - // same nets? no collision! - if( aCtx && aCtx->options.m_differentNetsOnly - && m_net == aHead->m_net && m_net >= 0 && aHead->m_net >= 0 ) - { - return false; - } - - // a pad associated with a "free" pin (NIC) doesn't have a net until it has been used - if( aCtx && aCtx->options.m_differentNetsOnly - && ( IsFreePad() || aHead->IsFreePad() ) ) - { - return false; - } - // check if we are not on completely different layers first if( !m_layers.Overlaps( aHead->m_layers ) ) return false; @@ -128,34 +114,56 @@ bool ITEM::collideSimple( const ITEM* aHead, const NODE* aNode, collisionsFound |= holeI->collideSimple( aHead, aNode, aCtx ); if( holeH && holeI ) - collisionsFound |= holeI->collideSimple( holeH, aNode, aCtx ); - - int clearance; - - if( aCtx && aCtx->options.m_overrideClearance >= 0 ) { - clearance = aCtx->options.m_overrideClearance; - } - else if( aNode->GetRuleResolver()->IsKeepout( this, aHead ) ) - { - clearance = 0; // keepouts are exact boundary; no clearance - } - else - { - clearance = aNode->GetClearance( this, aHead ); + if( aCtx ) + { + // Hole to hole collsions are never net-specific + COLLISION_SEARCH_OPTIONS holeToHoleOptions( aCtx->options ); + holeToHoleOptions.m_differentNetsOnly = false; + COLLISION_SEARCH_CONTEXT holeToHoleCtx( aCtx->obstacles, holeToHoleOptions ); + collisionsFound |= holeI->collideSimple( holeH, aNode, &holeToHoleCtx ); + } + else + { + collisionsFound |= holeI->collideSimple( holeH, aNode, nullptr ); + } } // fixme: this f***ing singleton must go... ROUTER* router = ROUTER::GetInstance(); ROUTER_IFACE* iface = router ? router->GetInterface() : nullptr; + bool differentNetsOnly = aCtx && aCtx->options.m_differentNetsOnly; + int clearance; - if( iface ) + if( differentNetsOnly && m_net == aHead->m_net && m_net >= 0 && aHead->m_net >= 0 ) { - if( !iface->IsFlashedOnLayer( this, aHead->Layer() ) ) - return collisionsFound; - - if( !iface->IsFlashedOnLayer( aHead, Layer() ) ) - return collisionsFound; + // same nets? no clearance! + clearance = -1; + } + else if( differentNetsOnly && ( IsFreePad() || aHead->IsFreePad() ) ) + { + // a pad associated with a "free" pin (NIC) doesn't have a net until it has been used + clearance = -1; + } + else if( aNode->GetRuleResolver()->IsKeepout( this, aHead ) ) + { + clearance = 0; // keepouts are exact boundary; no clearance + } + else if( iface && !iface->IsFlashedOnLayer( this, aHead->Layer() ) ) + { + clearance = -1; + } + else if( iface && !iface->IsFlashedOnLayer( aHead, Layer() ) ) + { + clearance = -1; + } + else if( aCtx && aCtx->options.m_overrideClearance >= 0 ) + { + clearance = aCtx->options.m_overrideClearance; + } + else + { + clearance = aNode->GetClearance( this, aHead ); } if( clearance >= 0 ) diff --git a/pcbnew/router/pns_kicad_iface.cpp b/pcbnew/router/pns_kicad_iface.cpp index 25e923a42a..e18bb20f36 100644 --- a/pcbnew/router/pns_kicad_iface.cpp +++ b/pcbnew/router/pns_kicad_iface.cpp @@ -334,6 +334,7 @@ bool PNS_PCBNEW_RULE_RESOLVER::QueryConstraint( PNS::CONSTRAINT_TYPE aType, case PNS::ITEM::VIA_T: parentA = &m_dummyVias[0]; break; case PNS::ITEM::SEGMENT_T: parentA = &m_dummyTracks[0]; break; case PNS::ITEM::LINE_T: parentA = &m_dummyTracks[0]; break; + case PNS::ITEM::HOLE_T: parentB = &m_dummyVias[0]; break; default: break; } @@ -352,6 +353,7 @@ bool PNS_PCBNEW_RULE_RESOLVER::QueryConstraint( PNS::CONSTRAINT_TYPE aType, case PNS::ITEM::VIA_T: parentB = &m_dummyVias[1]; break; case PNS::ITEM::SEGMENT_T: parentB = &m_dummyTracks[1]; break; case PNS::ITEM::LINE_T: parentB = &m_dummyTracks[1]; break; + case PNS::ITEM::HOLE_T: parentB = &m_dummyVias[1]; break; default: break; }