/* * KiRouter - a push-and-(sometimes-)shove PCB router * * Copyright (C) 2013-2014 CERN * Copyright (C) 2016-2020 KiCad Developers, see AUTHORS.txt for contributors. * Author: Tomasz Wlostowski * * This program is free software: you can redistribute it and/or modify it * under the terms of the GNU General Public License as published by the * Free Software Foundation, either version 3 of the License, or (at your * option) any later version. * * This program is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * General Public License for more details. * * You should have received a copy of the GNU General Public License along * with this program. If not, see . */ #include "pns_node.h" #include "pns_item.h" #include "pns_line.h" #include "pns_router.h" typedef VECTOR2I::extended_type ecoord; namespace PNS { bool ITEM::collideSimple( const ITEM* aOther, int aClearance, bool aNeedMTV, VECTOR2I* aMTV, const NODE* aParentNode, bool aDifferentNetsOnly ) const { const ROUTER_IFACE* iface = ROUTER::GetInstance()->GetInterface(); const SHAPE* shapeA = Shape(); const SHAPE* shapeB = aOther->Shape(); // same nets? no collision! if( aDifferentNetsOnly && m_net == aOther->m_net && m_net >= 0 && aOther->m_net >= 0 ) return false; // check if we are not on completely different layers first if( !m_layers.Overlaps( aOther->m_layers ) ) return false; if( !aOther->Layers().IsMultilayer() && !iface->IsOnLayer( this, aOther->Layer() ) ) { if( !AlternateShape() ) { wxLogError( "Missing expected Alternate shape for %s at %d %d", m_parent->GetClass(), Anchor( 0 ).x, Anchor( 0 ).y ); } else { shapeA = AlternateShape(); Mark( MK_ALT_SHAPE ); } } if( !Layers().IsMultilayer() && !iface->IsOnLayer( aOther, Layer() ) ) { if( !aOther->AlternateShape() ) { wxLogError( "Missing expected Alternate shape for %s at %d %d", aOther->Parent()->GetClass(), aOther->Anchor( 0 ).x, aOther->Anchor( 0 ).y ); } else { shapeB = aOther->AlternateShape(); aOther->Mark( MK_ALT_SHAPE ); } } if( aNeedMTV ) return shapeA->Collide( shapeB, aClearance, aMTV ); else return shapeA->Collide( shapeB, aClearance ); } bool ITEM::Collide( const ITEM* aOther, int aClearance, bool aNeedMTV, VECTOR2I* aMTV, const NODE* aParentNode, bool aDifferentNetsOnly ) const { if( collideSimple( aOther, aClearance, aNeedMTV, aMTV, aParentNode, aDifferentNetsOnly ) ) return true; // special case for "head" line with a via attached at the end. if( aOther->m_kind == LINE_T ) { const LINE* line = static_cast( aOther ); int clearance = aClearance - line->Width() / 2; if( line->EndsWithVia() ) { return collideSimple( &line->Via(), clearance, aNeedMTV, aMTV, aParentNode, aDifferentNetsOnly ); } } return false; } std::string ITEM::KindStr() const { switch( m_kind ) { case ARC_T: return "arc"; case LINE_T: return "line"; case SEGMENT_T: return "segment"; case VIA_T: return "via"; case JOINT_T: return "joint"; case SOLID_T: return "solid"; case DIFF_PAIR_T: return "diff-pair"; default: return "unknown"; } } ITEM::~ITEM() { } }