2007-12-01 03:42:52 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* This program source code file is part of KICAD, a free EDA CAD application.
|
|
|
|
*
|
2010-09-18 17:55:08 +00:00
|
|
|
* Copyright (C) 2004-2007 Jean-Pierre Charras, jean-pierre.charras@gipsa-lab.inpg.fr
|
2007-12-01 03:42:52 +00:00
|
|
|
* Copyright (C) 2007 Dick Hollenbeck, dick@softplc.com
|
|
|
|
* Copyright (C) 2007 Kicad Developers, see change_log.txt for contributors.
|
|
|
|
*
|
|
|
|
* 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 2
|
|
|
|
* 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, you may find one here:
|
|
|
|
* http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
|
|
|
|
* or you may search the http://www.gnu.org website for the version 2 license,
|
|
|
|
* or you may write to the Free Software Foundation, Inc.,
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
|
|
|
*/
|
|
|
|
|
2007-08-08 20:51:08 +00:00
|
|
|
/****************************/
|
|
|
|
/* DRC control */
|
|
|
|
/****************************/
|
2007-06-05 12:10:51 +00:00
|
|
|
|
|
|
|
#include "fctsys.h"
|
|
|
|
#include "common.h"
|
2009-02-04 15:25:03 +00:00
|
|
|
#include "class_drawpanel.h"
|
2009-07-30 11:04:07 +00:00
|
|
|
#include "pcbnew.h"
|
|
|
|
#include "wxPcbStruct.h"
|
2007-06-05 12:10:51 +00:00
|
|
|
#include "autorout.h"
|
|
|
|
#include "trigo.h"
|
2009-02-04 15:25:03 +00:00
|
|
|
#include "gestfich.h"
|
2009-10-28 11:48:47 +00:00
|
|
|
#include "class_board_design_settings.h"
|
2007-06-05 12:10:51 +00:00
|
|
|
|
|
|
|
#include "protos.h"
|
|
|
|
|
2007-11-27 01:34:35 +00:00
|
|
|
#include "drc_stuff.h"
|
2009-06-18 13:30:52 +00:00
|
|
|
#include "dialog_drc.h"
|
2007-11-27 01:34:35 +00:00
|
|
|
|
|
|
|
|
2007-12-01 03:42:52 +00:00
|
|
|
void DRC::ShowDialog()
|
|
|
|
{
|
|
|
|
if( !m_ui )
|
|
|
|
{
|
2009-06-18 13:30:52 +00:00
|
|
|
m_ui = new DIALOG_DRC_CONTROL( this, m_mainWindow );
|
2007-12-04 18:23:38 +00:00
|
|
|
updatePointers();
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2007-12-04 18:23:38 +00:00
|
|
|
// copy data retained in this DRC object into the m_ui DrcPanel:
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2010-09-15 14:53:33 +00:00
|
|
|
PutValueInLocalUnits( *m_ui->m_SetTrackMinWidthCtrl,
|
|
|
|
m_pcb->GetBoardDesignSettings()->m_TrackMinWidth,
|
2009-06-18 13:30:52 +00:00
|
|
|
m_mainWindow->m_InternalUnits );;
|
2010-09-15 14:53:33 +00:00
|
|
|
PutValueInLocalUnits( *m_ui->m_SetViaMinSizeCtrl,
|
|
|
|
m_pcb->GetBoardDesignSettings()->m_ViasMinSize,
|
2009-06-18 13:30:52 +00:00
|
|
|
m_mainWindow->m_InternalUnits );;
|
2010-09-15 14:53:33 +00:00
|
|
|
PutValueInLocalUnits( *m_ui->m_SetMicroViakMinSizeCtrl,
|
|
|
|
m_pcb->GetBoardDesignSettings()->m_MicroViasMinSize,
|
2009-06-18 13:30:52 +00:00
|
|
|
m_mainWindow->m_InternalUnits );;
|
2007-12-01 03:42:52 +00:00
|
|
|
|
2007-12-04 18:23:38 +00:00
|
|
|
m_ui->m_CreateRptCtrl->SetValue( m_doCreateRptFile );
|
|
|
|
m_ui->m_RptFilenameCtrl->SetValue( m_rptFilename );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
updatePointers();
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2009-03-23 19:54:15 +00:00
|
|
|
m_ui->Show( true );
|
2007-12-01 03:42:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-12-04 18:23:38 +00:00
|
|
|
void DRC::DestroyDialog( int aReason )
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
2007-12-03 05:14:51 +00:00
|
|
|
if( m_ui )
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
2007-12-04 18:23:38 +00:00
|
|
|
if( aReason == wxID_OK )
|
|
|
|
{
|
2008-02-09 08:34:45 +00:00
|
|
|
// if user clicked OK, save his choices in this DRC object.
|
2010-09-15 14:53:33 +00:00
|
|
|
m_doCreateRptFile = m_ui->m_CreateRptCtrl->GetValue();
|
|
|
|
m_rptFilename = m_ui->m_RptFilenameCtrl->GetValue();
|
2007-12-04 18:23:38 +00:00
|
|
|
}
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2007-12-03 05:14:51 +00:00
|
|
|
m_ui->Destroy();
|
|
|
|
m_ui = 0;
|
2007-12-01 03:42:52 +00:00
|
|
|
}
|
|
|
|
}
|
2007-06-05 12:10:51 +00:00
|
|
|
|
2007-11-27 01:34:35 +00:00
|
|
|
|
2007-12-01 03:42:52 +00:00
|
|
|
DRC::DRC( WinEDA_PcbFrame* aPcbWindow )
|
|
|
|
{
|
|
|
|
m_mainWindow = aPcbWindow;
|
2009-03-23 19:54:15 +00:00
|
|
|
m_pcb = aPcbWindow->GetBoard();
|
|
|
|
m_ui = 0;
|
2007-12-01 03:42:52 +00:00
|
|
|
|
2008-02-09 08:34:45 +00:00
|
|
|
// establish initial values for everything:
|
2009-06-27 06:26:41 +00:00
|
|
|
m_doPad2PadTest = true; // enable pad to pad clearance tests
|
|
|
|
m_doUnconnectedTest = true; // enable unconnected tests
|
|
|
|
m_doZonesTest = true; // enable zone to items clearance tests
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2009-03-23 19:54:15 +00:00
|
|
|
m_doCreateRptFile = false;
|
2007-11-27 01:34:35 +00:00
|
|
|
|
2007-12-01 03:42:52 +00:00
|
|
|
// m_rptFilename set to empty by its constructor
|
|
|
|
|
2010-09-18 08:29:40 +00:00
|
|
|
m_currentMarker = NULL;
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2009-03-23 19:54:15 +00:00
|
|
|
m_segmAngle = 0;
|
2007-12-01 03:42:52 +00:00
|
|
|
m_segmLength = 0;
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2007-12-01 03:42:52 +00:00
|
|
|
m_xcliplo = 0;
|
|
|
|
m_ycliplo = 0;
|
|
|
|
m_xcliphi = 0;
|
|
|
|
m_ycliphi = 0;
|
|
|
|
}
|
|
|
|
|
2009-03-23 19:54:15 +00:00
|
|
|
|
2007-12-04 18:23:38 +00:00
|
|
|
DRC::~DRC()
|
|
|
|
{
|
2008-02-09 08:34:45 +00:00
|
|
|
// maybe someday look at pointainer.h <- google for "pointainer.h"
|
2009-03-23 19:54:15 +00:00
|
|
|
for( unsigned i = 0; i<m_unconnected.size(); ++i )
|
2007-12-04 18:23:38 +00:00
|
|
|
delete m_unconnected[i];
|
|
|
|
}
|
2007-12-01 03:42:52 +00:00
|
|
|
|
2009-03-23 19:54:15 +00:00
|
|
|
|
2008-01-20 19:55:22 +00:00
|
|
|
/*********************************************/
|
2007-12-01 03:42:52 +00:00
|
|
|
int DRC::Drc( TRACK* aRefSegm, TRACK* aList )
|
2008-01-20 19:55:22 +00:00
|
|
|
/*********************************************/
|
2007-06-05 12:10:51 +00:00
|
|
|
{
|
2007-12-01 03:42:52 +00:00
|
|
|
updatePointers();
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2008-12-04 04:28:11 +00:00
|
|
|
if( !doTrackDrc( aRefSegm, aList, true ) )
|
2007-08-08 20:51:08 +00:00
|
|
|
{
|
2007-12-01 05:37:44 +00:00
|
|
|
wxASSERT( m_currentMarker );
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2009-04-17 08:51:02 +00:00
|
|
|
m_currentMarker->DisplayInfo( m_mainWindow );
|
2007-12-01 03:42:52 +00:00
|
|
|
return BAD_DRC;
|
2007-08-08 20:51:08 +00:00
|
|
|
}
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2007-12-01 03:42:52 +00:00
|
|
|
return OK_DRC;
|
|
|
|
}
|
2007-08-08 20:51:08 +00:00
|
|
|
|
|
|
|
|
2008-01-31 20:53:44 +00:00
|
|
|
/**************************************************************/
|
2009-03-23 19:54:15 +00:00
|
|
|
int DRC::Drc( ZONE_CONTAINER* aArea, int CornerIndex )
|
2008-01-31 20:53:44 +00:00
|
|
|
/*************************************************************/
|
2009-03-23 19:54:15 +00:00
|
|
|
|
2008-01-20 19:55:22 +00:00
|
|
|
/**
|
|
|
|
* Function Drc
|
2008-01-31 20:53:44 +00:00
|
|
|
* tests the outline segment starting at CornerIndex and returns the result and displays the error
|
2008-01-20 19:55:22 +00:00
|
|
|
* in the status panel only if one exists.
|
|
|
|
* Test Edge inside other areas
|
|
|
|
* Test Edge too close other areas
|
2008-01-31 20:53:44 +00:00
|
|
|
* @param aEdge The areaparent which contains the corner.
|
|
|
|
* @param CornerIndex The starting point of the segment to test.
|
2008-01-20 19:55:22 +00:00
|
|
|
* @return int - BAD_DRC (1) if DRC error or OK_DRC (0) if OK
|
|
|
|
*/
|
|
|
|
{
|
|
|
|
updatePointers();
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2009-03-23 19:54:15 +00:00
|
|
|
if( !doEdgeZoneDrc( aArea, CornerIndex ) )
|
2008-01-20 19:55:22 +00:00
|
|
|
{
|
|
|
|
wxASSERT( m_currentMarker );
|
2009-04-17 08:51:02 +00:00
|
|
|
m_currentMarker->DisplayInfo( m_mainWindow );
|
2008-01-20 19:55:22 +00:00
|
|
|
return BAD_DRC;
|
|
|
|
}
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2008-01-20 19:55:22 +00:00
|
|
|
return OK_DRC;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-06-18 13:30:52 +00:00
|
|
|
/**
|
|
|
|
* Function RunTests
|
|
|
|
* will actually run all the tests specified with a previous call to
|
|
|
|
* SetSettings()
|
|
|
|
*/
|
2010-09-15 14:53:33 +00:00
|
|
|
void DRC::RunTests( wxTextCtrl* aMessages )
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
2009-02-26 09:33:14 +00:00
|
|
|
// Ensure ratsnest is up to date:
|
2009-05-24 18:28:36 +00:00
|
|
|
if( (m_pcb->m_Status_Pcb & LISTE_RATSNEST_ITEM_OK) == 0 )
|
2009-06-27 06:26:41 +00:00
|
|
|
{
|
2009-09-10 15:22:26 +00:00
|
|
|
if( aMessages )
|
|
|
|
{
|
2010-09-15 14:53:33 +00:00
|
|
|
aMessages->AppendText( _( "Compile ratsnest...\n" ) );
|
2009-09-10 15:22:26 +00:00
|
|
|
wxSafeYield();
|
|
|
|
}
|
|
|
|
|
2009-02-26 09:33:14 +00:00
|
|
|
m_mainWindow->Compile_Ratsnest( NULL, true );
|
2009-06-27 06:26:41 +00:00
|
|
|
}
|
2009-02-26 09:33:14 +00:00
|
|
|
|
2007-12-04 18:23:38 +00:00
|
|
|
// someone should have cleared the two lists before calling this.
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
if( !testNetClasses() )
|
|
|
|
{
|
|
|
|
// testing the netclasses is a special case because if the netclasses
|
|
|
|
// do not pass the g_DesignSettings checks, then every member of a net
|
|
|
|
// class (a NET) will cause its items such as tracks, vias, and pads
|
|
|
|
// to also fail. So quit after *all* netclass errors have been reported.
|
|
|
|
if( aMessages )
|
2010-09-15 14:53:33 +00:00
|
|
|
aMessages->AppendText( _( "Aborting\n" ) );
|
2009-09-10 15:22:26 +00:00
|
|
|
|
|
|
|
// update the m_ui listboxes
|
|
|
|
updatePointers();
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2007-12-01 03:42:52 +00:00
|
|
|
// test pad to pad clearances, nothing to do with tracks, vias or zones.
|
|
|
|
if( m_doPad2PadTest )
|
2009-06-27 06:26:41 +00:00
|
|
|
{
|
2009-09-10 15:22:26 +00:00
|
|
|
if( aMessages )
|
|
|
|
{
|
2010-09-15 14:53:33 +00:00
|
|
|
aMessages->AppendText( _( "Pad clearances...\n" ) );
|
2009-09-10 15:22:26 +00:00
|
|
|
wxSafeYield();
|
|
|
|
}
|
|
|
|
|
2007-12-01 03:42:52 +00:00
|
|
|
testPad2Pad();
|
2009-06-27 06:26:41 +00:00
|
|
|
}
|
2007-12-01 03:42:52 +00:00
|
|
|
|
2007-12-01 05:37:44 +00:00
|
|
|
// test track and via clearances to other tracks, pads, and vias
|
2009-09-10 15:22:26 +00:00
|
|
|
if( aMessages )
|
|
|
|
{
|
2010-09-15 14:53:33 +00:00
|
|
|
aMessages->AppendText( _( "Track clearances...\n" ) );
|
2009-09-10 15:22:26 +00:00
|
|
|
wxSafeYield();
|
|
|
|
}
|
|
|
|
|
2007-12-01 03:42:52 +00:00
|
|
|
testTracks();
|
|
|
|
|
2008-12-04 20:53:11 +00:00
|
|
|
// Before testing segments and unconnected, refill all zones:
|
2009-02-26 09:33:14 +00:00
|
|
|
// this is a good caution, because filled areas can be outdated.
|
2009-09-10 15:22:26 +00:00
|
|
|
if( aMessages )
|
|
|
|
{
|
2010-09-15 14:53:33 +00:00
|
|
|
aMessages->AppendText( _( "Fill zones...\n" ) );
|
2009-09-10 15:22:26 +00:00
|
|
|
wxSafeYield();
|
|
|
|
}
|
2008-12-04 20:53:11 +00:00
|
|
|
m_mainWindow->Fill_All_Zones( false );
|
|
|
|
|
2008-02-09 08:34:45 +00:00
|
|
|
// test zone clearances to other zones, pads, tracks, and vias
|
2010-09-15 14:53:33 +00:00
|
|
|
if( aMessages && m_doZonesTest )
|
2009-09-10 15:22:26 +00:00
|
|
|
{
|
2010-09-15 14:53:33 +00:00
|
|
|
aMessages->AppendText( _( "Test zones...\n" ) );
|
2009-09-10 15:22:26 +00:00
|
|
|
wxSafeYield();
|
|
|
|
}
|
|
|
|
|
2008-03-18 04:04:17 +00:00
|
|
|
testZones( m_doZonesTest );
|
2007-12-01 03:42:52 +00:00
|
|
|
|
2008-02-09 08:34:45 +00:00
|
|
|
// find and gather unconnected pads.
|
2007-12-01 03:42:52 +00:00
|
|
|
if( m_doUnconnectedTest )
|
2009-06-27 06:26:41 +00:00
|
|
|
{
|
2009-09-10 15:22:26 +00:00
|
|
|
if( aMessages )
|
|
|
|
{
|
2010-09-15 14:53:33 +00:00
|
|
|
aMessages->AppendText( _( "Unconnected pads...\n" ) );
|
2009-09-10 15:22:26 +00:00
|
|
|
wxSafeYield();
|
|
|
|
}
|
|
|
|
|
2007-12-01 03:42:52 +00:00
|
|
|
testUnconnected();
|
2009-06-27 06:26:41 +00:00
|
|
|
}
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2007-12-04 18:23:38 +00:00
|
|
|
// update the m_ui listboxes
|
2007-12-03 05:14:51 +00:00
|
|
|
updatePointers();
|
2009-09-10 15:22:26 +00:00
|
|
|
|
|
|
|
if( aMessages )
|
|
|
|
{
|
|
|
|
// no newline on this one because it is last, don't want the window
|
|
|
|
// to unnecessarily scroll.
|
2010-09-15 14:53:33 +00:00
|
|
|
aMessages->AppendText( _( "Finished" ) );
|
2009-09-10 15:22:26 +00:00
|
|
|
}
|
2007-12-03 05:14:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/***************************************************************/
|
|
|
|
void DRC::ListUnconnectedPads()
|
|
|
|
/***************************************************************/
|
|
|
|
{
|
|
|
|
testUnconnected();
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2007-12-04 18:23:38 +00:00
|
|
|
// update the m_ui listboxes
|
|
|
|
updatePointers();
|
2007-12-01 03:42:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-12-03 05:14:51 +00:00
|
|
|
void DRC::updatePointers()
|
|
|
|
{
|
|
|
|
// update my pointers, m_mainWindow is the only unchangable one
|
2009-01-05 05:21:35 +00:00
|
|
|
m_pcb = m_mainWindow->GetBoard();
|
2007-12-04 18:23:38 +00:00
|
|
|
|
2009-03-23 19:54:15 +00:00
|
|
|
if( m_ui ) // Use diag list boxes only in DRC dialog
|
2008-02-09 08:34:45 +00:00
|
|
|
{
|
|
|
|
m_ui->m_ClearanceListBox->SetList( new DRC_LIST_MARKERS( m_pcb ) );
|
|
|
|
m_ui->m_UnconnectedListBox->SetList( new DRC_LIST_UNCONNECTED( &m_unconnected ) );
|
|
|
|
}
|
2007-12-03 05:14:51 +00:00
|
|
|
}
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2007-12-03 05:14:51 +00:00
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
bool DRC::doNetClass( NETCLASS* nc, wxString& msg )
|
|
|
|
{
|
|
|
|
bool ret = true;
|
|
|
|
|
2010-01-31 20:01:46 +00:00
|
|
|
const BOARD_DESIGN_SETTINGS& g = *m_pcb->GetBoardDesignSettings();
|
2009-09-10 15:22:26 +00:00
|
|
|
|
2010-09-15 14:53:33 +00:00
|
|
|
#define FmtVal( x ) GetChars( ReturnStringFromValue( g_UserUnit, x, PCB_INTERNAL_UNIT ) )
|
2009-09-10 15:22:26 +00:00
|
|
|
|
2010-01-31 20:01:46 +00:00
|
|
|
#if 0 // set to 1 when (if...) BOARD_DESIGN_SETTINGS has a m_MinClearance value
|
2009-10-21 19:16:25 +00:00
|
|
|
if( nc->GetClearance() < g.m_MinClearance )
|
2009-09-10 15:22:26 +00:00
|
|
|
{
|
2010-09-15 14:53:33 +00:00
|
|
|
msg.Printf( _( "NETCLASS: '%s' has Clearance:%s which is less than global:%s" ),
|
|
|
|
GetChars( nc->GetName() ),
|
|
|
|
FmtVal( nc->GetClearance() ),
|
|
|
|
FmtVal( g.m_TrackClearance )
|
|
|
|
);
|
2009-09-10 15:22:26 +00:00
|
|
|
|
|
|
|
m_currentMarker = fillMarker( DRCE_NETCLASS_CLEARANCE, msg, m_currentMarker );
|
|
|
|
m_pcb->Add( m_currentMarker );
|
|
|
|
m_currentMarker = 0;
|
|
|
|
ret = false;
|
|
|
|
}
|
2009-10-21 19:16:25 +00:00
|
|
|
#endif
|
2009-09-10 15:22:26 +00:00
|
|
|
|
|
|
|
if( nc->GetTrackWidth() < g.m_TrackMinWidth )
|
|
|
|
{
|
2010-09-15 14:53:33 +00:00
|
|
|
msg.Printf( _( "NETCLASS: '%s' has TrackWidth:%s which is less than global:%s" ),
|
|
|
|
GetChars( nc->GetName() ),
|
|
|
|
FmtVal( nc->GetTrackWidth() ),
|
|
|
|
FmtVal( g.m_TrackMinWidth )
|
|
|
|
);
|
2009-09-10 15:22:26 +00:00
|
|
|
|
|
|
|
m_currentMarker = fillMarker( DRCE_NETCLASS_TRACKWIDTH, msg, m_currentMarker );
|
|
|
|
m_pcb->Add( m_currentMarker );
|
|
|
|
m_currentMarker = 0;
|
|
|
|
ret = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if( nc->GetViaDiameter() < g.m_ViasMinSize )
|
|
|
|
{
|
2010-09-15 14:53:33 +00:00
|
|
|
msg.Printf( _( "NETCLASS: '%s' has Via Dia:%s which is less than global:%s" ),
|
|
|
|
GetChars( nc->GetName() ),
|
|
|
|
FmtVal( nc->GetViaDiameter() ),
|
|
|
|
FmtVal( g.m_ViasMinSize )
|
|
|
|
);
|
2009-09-10 15:22:26 +00:00
|
|
|
|
|
|
|
m_currentMarker = fillMarker( DRCE_NETCLASS_VIASIZE, msg, m_currentMarker );
|
|
|
|
m_pcb->Add( m_currentMarker );
|
|
|
|
m_currentMarker = 0;
|
|
|
|
ret = false;
|
|
|
|
}
|
|
|
|
|
2009-10-21 19:16:25 +00:00
|
|
|
if( nc->GetViaDrill() < g.m_ViasMinDrill )
|
2009-09-10 15:22:26 +00:00
|
|
|
{
|
2010-09-15 14:53:33 +00:00
|
|
|
msg.Printf( _( "NETCLASS: '%s' has Via Drill:%s which is less than global:%s" ),
|
|
|
|
GetChars( nc->GetName() ),
|
|
|
|
FmtVal( nc->GetViaDrill() ),
|
|
|
|
FmtVal( g.m_ViasMinDrill )
|
|
|
|
);
|
2009-09-10 15:22:26 +00:00
|
|
|
|
|
|
|
m_currentMarker = fillMarker( DRCE_NETCLASS_VIADRILLSIZE, msg, m_currentMarker );
|
|
|
|
m_pcb->Add( m_currentMarker );
|
|
|
|
m_currentMarker = 0;
|
|
|
|
ret = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if( nc->GetuViaDiameter() < g.m_MicroViasMinSize )
|
|
|
|
{
|
2010-09-15 14:53:33 +00:00
|
|
|
msg.Printf( _( "NETCLASS: '%s' has uVia Dia:%s which is less than global:%s" ),
|
|
|
|
GetChars( nc->GetName() ),
|
|
|
|
FmtVal( nc->GetuViaDiameter() ),
|
|
|
|
FmtVal( g.m_MicroViasMinSize )
|
|
|
|
);
|
2009-09-10 15:22:26 +00:00
|
|
|
|
|
|
|
m_currentMarker = fillMarker( DRCE_NETCLASS_uVIASIZE, msg, m_currentMarker );
|
|
|
|
m_pcb->Add( m_currentMarker );
|
|
|
|
m_currentMarker = 0;
|
|
|
|
ret = false;
|
|
|
|
}
|
|
|
|
|
2009-10-21 19:16:25 +00:00
|
|
|
if( nc->GetuViaDrill() < g.m_MicroViasMinDrill )
|
2009-09-10 15:22:26 +00:00
|
|
|
{
|
2010-09-15 14:53:33 +00:00
|
|
|
msg.Printf( _( "NETCLASS: '%s' has uVia Drill:%s which is less than global:%s" ),
|
|
|
|
GetChars( nc->GetName() ),
|
|
|
|
FmtVal( nc->GetuViaDrill() ),
|
|
|
|
FmtVal( g.m_MicroViasMinDrill )
|
|
|
|
);
|
2009-09-10 15:22:26 +00:00
|
|
|
|
|
|
|
m_currentMarker = fillMarker( DRCE_NETCLASS_uVIADRILLSIZE, msg, m_currentMarker );
|
|
|
|
m_pcb->Add( m_currentMarker );
|
|
|
|
m_currentMarker = 0;
|
|
|
|
ret = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool DRC::testNetClasses()
|
|
|
|
{
|
2010-09-15 14:53:33 +00:00
|
|
|
bool ret = true;
|
2009-09-10 15:22:26 +00:00
|
|
|
|
|
|
|
NETCLASSES& netclasses = m_pcb->m_NetClasses;
|
|
|
|
|
|
|
|
wxString msg; // construct this only once here, not in a loop, since somewhat expensive.
|
|
|
|
|
|
|
|
if( !doNetClass( netclasses.GetDefault(), msg ) )
|
|
|
|
ret = false;
|
|
|
|
|
|
|
|
for( NETCLASSES::const_iterator i = netclasses.begin(); i != netclasses.end(); ++i )
|
|
|
|
{
|
|
|
|
NETCLASS* nc = i->second;
|
|
|
|
|
|
|
|
if( !doNetClass( nc, msg ) )
|
|
|
|
ret = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-12-01 03:42:52 +00:00
|
|
|
void DRC::testTracks()
|
|
|
|
{
|
2009-03-23 19:54:15 +00:00
|
|
|
for( TRACK* segm = m_pcb->m_Track; segm && segm->Next(); segm = segm->Next() )
|
2007-08-08 20:51:08 +00:00
|
|
|
{
|
2008-12-04 04:28:11 +00:00
|
|
|
if( !doTrackDrc( segm, segm->Next(), true ) )
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
|
|
|
wxASSERT( m_currentMarker );
|
2008-02-09 08:34:45 +00:00
|
|
|
m_pcb->Add( m_currentMarker );
|
|
|
|
m_currentMarker = 0;
|
2007-12-01 03:42:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2007-11-27 01:34:35 +00:00
|
|
|
|
2007-12-01 03:42:52 +00:00
|
|
|
|
2009-03-23 19:54:15 +00:00
|
|
|
/***********************/
|
2007-12-01 03:42:52 +00:00
|
|
|
void DRC::testPad2Pad()
|
2009-03-23 19:54:15 +00:00
|
|
|
/***********************/
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
2008-03-18 04:04:17 +00:00
|
|
|
std::vector<D_PAD*> sortedPads;
|
|
|
|
|
|
|
|
CreateSortedPadListByXCoord( m_pcb, &sortedPads );
|
2007-12-01 03:42:52 +00:00
|
|
|
|
|
|
|
// find the max size of the pads (used to stop the test)
|
2009-03-23 19:54:15 +00:00
|
|
|
int max_size = 0;
|
2008-03-18 04:04:17 +00:00
|
|
|
|
2009-03-23 19:54:15 +00:00
|
|
|
for( unsigned i = 0; i<sortedPads.size(); ++i )
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
2008-03-18 04:04:17 +00:00
|
|
|
D_PAD* pad = sortedPads[i];
|
|
|
|
|
2010-09-18 17:55:08 +00:00
|
|
|
if( pad->m_ShapeMaxRadius > max_size ) // m_ShapeMaxRadius is the radius value of the circle containing the pad
|
|
|
|
max_size = pad->m_ShapeMaxRadius;
|
2007-12-01 03:42:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Test the pads
|
2008-03-18 04:04:17 +00:00
|
|
|
D_PAD** listEnd = &sortedPads[ sortedPads.size() ];
|
|
|
|
|
2009-03-23 19:54:15 +00:00
|
|
|
for( unsigned i = 0; i<sortedPads.size(); ++i )
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
2008-03-18 04:04:17 +00:00
|
|
|
D_PAD* pad = sortedPads[i];
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2010-09-15 14:53:33 +00:00
|
|
|
int x_limit = max_size + pad->GetClearance() +
|
2010-09-18 17:55:08 +00:00
|
|
|
pad->m_ShapeMaxRadius + pad->GetPosition().x;
|
2009-09-10 15:22:26 +00:00
|
|
|
|
|
|
|
if( !doPadToPadsDrc( pad, &sortedPads[i], listEnd, x_limit ) )
|
2007-11-27 01:34:35 +00:00
|
|
|
{
|
2007-12-01 03:42:52 +00:00
|
|
|
wxASSERT( m_currentMarker );
|
|
|
|
m_pcb->Add( m_currentMarker );
|
2008-02-09 08:34:45 +00:00
|
|
|
m_currentMarker = 0;
|
2007-11-27 01:34:35 +00:00
|
|
|
}
|
2007-12-01 03:42:52 +00:00
|
|
|
}
|
|
|
|
}
|
2007-11-27 01:34:35 +00:00
|
|
|
|
|
|
|
|
2007-12-01 03:42:52 +00:00
|
|
|
void DRC::testUnconnected()
|
|
|
|
{
|
2009-05-24 18:28:36 +00:00
|
|
|
if( (m_pcb->m_Status_Pcb & LISTE_RATSNEST_ITEM_OK) == 0 )
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
|
|
|
wxClientDC dc( m_mainWindow->DrawPanel );
|
|
|
|
m_mainWindow->Compile_Ratsnest( &dc, TRUE );
|
|
|
|
}
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2009-05-28 08:42:24 +00:00
|
|
|
if( m_pcb->GetRatsnestsCount() == 0 )
|
2007-12-01 03:42:52 +00:00
|
|
|
return;
|
2007-11-27 01:34:35 +00:00
|
|
|
|
2009-05-28 08:42:24 +00:00
|
|
|
for( unsigned ii = 0; ii < m_pcb->GetRatsnestsCount(); ++ii )
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
2009-05-28 08:42:24 +00:00
|
|
|
RATSNEST_ITEM* rat = &m_pcb->m_FullRatsnest[ii];
|
2009-05-24 18:28:36 +00:00
|
|
|
if( (rat->m_Status & CH_ACTIF) == 0 )
|
2007-12-01 03:42:52 +00:00
|
|
|
continue;
|
2007-08-08 20:51:08 +00:00
|
|
|
|
2009-05-24 18:28:36 +00:00
|
|
|
D_PAD* padStart = rat->m_PadStart;
|
|
|
|
D_PAD* padEnd = rat->m_PadEnd;
|
2007-11-27 01:34:35 +00:00
|
|
|
|
2009-07-06 18:02:26 +00:00
|
|
|
DRC_ITEM* uncItem = new DRC_ITEM( DRCE_UNCONNECTED_PADS,
|
2009-03-23 19:54:15 +00:00
|
|
|
padStart->MenuText( m_pcb ), padEnd->MenuText( m_pcb ),
|
|
|
|
padStart->GetPosition(), padEnd->GetPosition() );
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2007-12-04 18:23:38 +00:00
|
|
|
m_unconnected.push_back( uncItem );
|
2007-12-01 03:42:52 +00:00
|
|
|
}
|
|
|
|
}
|
2007-11-27 01:34:35 +00:00
|
|
|
|
2009-03-23 19:54:15 +00:00
|
|
|
|
2009-01-24 19:30:39 +00:00
|
|
|
/**********************************************/
|
2009-03-23 19:54:15 +00:00
|
|
|
void DRC::testZones( bool adoTestFillSegments )
|
2009-01-24 19:30:39 +00:00
|
|
|
/**********************************************/
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
2008-02-09 08:34:45 +00:00
|
|
|
// Test copper areas for valide netcodes
|
|
|
|
// if a netcode is < 0 the netname was not found when reading a netlist
|
2009-01-24 19:30:39 +00:00
|
|
|
// if a netcode is == 0 the netname is void, and the zone is not connected.
|
|
|
|
// This is allowed, but i am not sure this is a good idea
|
2008-02-09 08:34:45 +00:00
|
|
|
for( int ii = 0; ii < m_pcb->GetAreaCount(); ii++ )
|
|
|
|
{
|
|
|
|
ZONE_CONTAINER* Area_To_Test = m_pcb->GetArea( ii );
|
2009-03-23 19:54:15 +00:00
|
|
|
if( !Area_To_Test->IsOnCopperLayer() )
|
2008-09-26 19:51:36 +00:00
|
|
|
continue;
|
2009-01-24 19:30:39 +00:00
|
|
|
if( Area_To_Test->GetNet() < 0 )
|
2008-02-09 08:34:45 +00:00
|
|
|
{
|
|
|
|
m_currentMarker = fillMarker( Area_To_Test,
|
2009-03-23 19:54:15 +00:00
|
|
|
DRCE_NON_EXISTANT_NET_FOR_ZONE_OUTLINE, m_currentMarker );
|
2008-02-09 08:34:45 +00:00
|
|
|
m_pcb->Add( m_currentMarker );
|
|
|
|
m_currentMarker = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test copper areas outlines, and create markers when needed
|
|
|
|
m_pcb->Test_Drc_Areas_Outlines_To_Areas_Outlines( NULL, true );
|
2008-01-10 20:53:41 +00:00
|
|
|
|
2007-12-01 03:42:52 +00:00
|
|
|
TRACK* zoneSeg;
|
|
|
|
|
2008-12-04 04:28:11 +00:00
|
|
|
if( !adoTestFillSegments )
|
|
|
|
return;
|
|
|
|
|
2010-09-18 08:29:40 +00:00
|
|
|
// m_pcb->m_Zone is fully obsolete. Keep this test for compatibility
|
|
|
|
// with old designs. Will be removed on day
|
2008-12-04 04:28:11 +00:00
|
|
|
for( zoneSeg = m_pcb->m_Zone; zoneSeg && zoneSeg->Next(); zoneSeg = zoneSeg->Next() )
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
|
|
|
// Test zoneSeg with other zone segments and with all pads
|
2008-12-04 04:28:11 +00:00
|
|
|
if( !doTrackDrc( zoneSeg, zoneSeg->Next(), true ) )
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
|
|
|
wxASSERT( m_currentMarker );
|
|
|
|
m_pcb->Add( m_currentMarker );
|
2008-02-09 08:34:45 +00:00
|
|
|
m_currentMarker = 0;
|
2007-12-01 03:42:52 +00:00
|
|
|
}
|
|
|
|
|
2008-12-04 04:28:11 +00:00
|
|
|
// Pads already tested: disable pad test
|
2007-12-01 03:42:52 +00:00
|
|
|
|
2008-12-04 04:28:11 +00:00
|
|
|
bool rc = doTrackDrc( zoneSeg, m_pcb->m_Track, false );
|
2007-12-01 03:42:52 +00:00
|
|
|
if( !rc )
|
|
|
|
{
|
|
|
|
wxASSERT( m_currentMarker );
|
|
|
|
m_pcb->Add( m_currentMarker );
|
2008-02-09 08:34:45 +00:00
|
|
|
m_currentMarker = 0;
|
2007-12-01 03:42:52 +00:00
|
|
|
}
|
2007-08-08 20:51:08 +00:00
|
|
|
}
|
2007-12-01 03:42:52 +00:00
|
|
|
}
|
2007-08-08 20:51:08 +00:00
|
|
|
|
2007-12-01 03:42:52 +00:00
|
|
|
|
|
|
|
/***********************************************************************/
|
2008-12-04 04:28:11 +00:00
|
|
|
bool DRC::doTrackDrc( TRACK* aRefSeg, TRACK* aStart, bool testPads )
|
2007-12-01 03:42:52 +00:00
|
|
|
/***********************************************************************/
|
2007-06-05 12:10:51 +00:00
|
|
|
{
|
2010-09-15 14:53:33 +00:00
|
|
|
TRACK* track;
|
|
|
|
int dx, dy; // utilise pour calcul des dim x et dim y des segments
|
|
|
|
int layerMask;
|
|
|
|
int net_code_ref;
|
|
|
|
wxPoint shape_pos;
|
2009-09-10 15:22:26 +00:00
|
|
|
|
2010-09-15 14:53:33 +00:00
|
|
|
NETCLASS* netclass = aRefSeg->GetNetClass();
|
2007-08-08 20:51:08 +00:00
|
|
|
|
2010-09-18 08:29:40 +00:00
|
|
|
/* In order to make some calculations more easier or faster,
|
|
|
|
* pads and tracks coordinates will be made relative to the reference segment origin
|
|
|
|
*/
|
|
|
|
wxPoint origin = aRefSeg->m_Start; // origin will be the origin of other coordinates
|
2007-12-01 03:42:52 +00:00
|
|
|
|
2010-09-18 08:29:40 +00:00
|
|
|
m_segmEnd.x = dx = aRefSeg->m_End.x - origin.x;
|
|
|
|
m_segmEnd.y = dy = aRefSeg->m_End.y - origin.y;
|
2007-12-01 03:42:52 +00:00
|
|
|
|
|
|
|
layerMask = aRefSeg->ReturnMaskLayer();
|
|
|
|
net_code_ref = aRefSeg->GetNet();
|
|
|
|
|
|
|
|
m_segmAngle = 0;
|
2007-12-04 20:44:41 +00:00
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
// Phase 0 : Test vias
|
2008-12-04 04:28:11 +00:00
|
|
|
if( aRefSeg->Type() == TYPE_VIA )
|
2007-12-04 20:44:41 +00:00
|
|
|
{
|
2009-09-10 15:22:26 +00:00
|
|
|
// test if the via size is smaller than minimum
|
2009-06-18 13:30:52 +00:00
|
|
|
if( aRefSeg->Shape() == VIA_MICROVIA )
|
|
|
|
{
|
2010-01-13 13:43:36 +00:00
|
|
|
if( aRefSeg->m_Width < netclass->GetuViaMinDiameter() )
|
2009-06-18 13:30:52 +00:00
|
|
|
{
|
|
|
|
m_currentMarker = fillMarker( aRefSeg, NULL,
|
|
|
|
DRCE_TOO_SMALL_MICROVIA, m_currentMarker );
|
|
|
|
return false;
|
2009-09-10 15:22:26 +00:00
|
|
|
}
|
2009-06-18 13:30:52 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2010-01-13 13:43:36 +00:00
|
|
|
if( aRefSeg->m_Width < netclass->GetViaMinDiameter() )
|
2009-06-18 13:30:52 +00:00
|
|
|
{
|
|
|
|
m_currentMarker = fillMarker( aRefSeg, NULL,
|
|
|
|
DRCE_TOO_SMALL_VIA, m_currentMarker );
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2009-09-10 15:22:26 +00:00
|
|
|
|
2008-02-09 08:34:45 +00:00
|
|
|
// test if via's hole is bigger than its diameter
|
2009-03-23 19:54:15 +00:00
|
|
|
// This test is necessary since the via hole size and width can be modified
|
2009-09-10 15:22:26 +00:00
|
|
|
// and a default via hole can be bigger than some vias sizes
|
2008-02-23 04:53:44 +00:00
|
|
|
if( aRefSeg->GetDrillValue() > aRefSeg->m_Width )
|
2007-12-04 20:44:41 +00:00
|
|
|
{
|
2008-02-09 08:34:45 +00:00
|
|
|
m_currentMarker = fillMarker( aRefSeg, NULL,
|
2009-03-23 19:54:15 +00:00
|
|
|
DRCE_VIA_HOLE_BIGGER, m_currentMarker );
|
2007-12-04 20:44:41 +00:00
|
|
|
return false;
|
|
|
|
}
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2009-03-23 19:54:15 +00:00
|
|
|
// For microvias: test if they are blind vias and only between 2 layers
|
2008-02-09 08:34:45 +00:00
|
|
|
// because they are used for very small drill size and are drill by laser
|
|
|
|
// and **only** one layer can be drilled
|
2008-01-12 20:31:56 +00:00
|
|
|
if( aRefSeg->Shape() == VIA_MICROVIA )
|
|
|
|
{
|
2009-03-23 19:54:15 +00:00
|
|
|
int layer1, layer2;
|
2008-02-09 08:34:45 +00:00
|
|
|
bool err = true;
|
2009-09-10 15:22:26 +00:00
|
|
|
|
2009-03-23 19:54:15 +00:00
|
|
|
( (SEGVIA*) aRefSeg )->ReturnLayerPair( &layer1, &layer2 );
|
|
|
|
if( layer1> layer2 )
|
|
|
|
EXCHG( layer1, layer2 );
|
|
|
|
|
2008-02-09 08:34:45 +00:00
|
|
|
// test:
|
2009-12-07 03:46:13 +00:00
|
|
|
if( layer1 == LAYER_N_BACK && layer2 == LAYER_N_2 )
|
2009-03-23 19:54:15 +00:00
|
|
|
err = false;
|
2010-09-15 14:53:33 +00:00
|
|
|
if( layer1 == (m_pcb->GetBoardDesignSettings()->GetCopperLayerCount() - 2 )
|
|
|
|
&& layer2 == LAYER_N_FRONT )
|
2009-03-23 19:54:15 +00:00
|
|
|
err = false;
|
|
|
|
if( err )
|
2008-02-09 08:34:45 +00:00
|
|
|
{
|
|
|
|
m_currentMarker = fillMarker( aRefSeg, NULL,
|
2009-03-23 19:54:15 +00:00
|
|
|
DRCE_MICRO_VIA_INCORRECT_LAYER_PAIR, m_currentMarker );
|
2008-02-09 08:34:45 +00:00
|
|
|
return false;
|
|
|
|
}
|
2008-01-12 20:31:56 +00:00
|
|
|
}
|
2007-12-04 20:44:41 +00:00
|
|
|
}
|
2009-06-18 13:30:52 +00:00
|
|
|
else // This is a track segment
|
|
|
|
{
|
2010-01-13 13:43:36 +00:00
|
|
|
if( aRefSeg->m_Width < netclass->GetTrackMinWidth() )
|
2009-06-18 13:30:52 +00:00
|
|
|
{
|
|
|
|
m_currentMarker = fillMarker( aRefSeg, NULL,
|
|
|
|
DRCE_TOO_SMALL_TRACK_WIDTH, m_currentMarker );
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2008-02-09 08:34:45 +00:00
|
|
|
|
|
|
|
// for a non horizontal or vertical segment Compute the segment angle
|
|
|
|
// in tenths of degrees and its length
|
2007-12-01 03:42:52 +00:00
|
|
|
if( dx || dy )
|
2007-08-08 20:51:08 +00:00
|
|
|
{
|
2007-12-01 03:42:52 +00:00
|
|
|
// Compute the segment angle in 0,1 degrees
|
|
|
|
m_segmAngle = ArcTangente( dy, dx );
|
2007-11-27 01:34:35 +00:00
|
|
|
|
2007-12-01 03:42:52 +00:00
|
|
|
// Compute the segment length: we build an equivalent rotated segment,
|
2008-02-09 08:34:45 +00:00
|
|
|
// this segment is horizontal, therefore dx = length
|
|
|
|
RotatePoint( &dx, &dy, m_segmAngle ); // dx = length, dy = 0
|
2007-12-01 03:42:52 +00:00
|
|
|
}
|
2007-11-27 01:34:35 +00:00
|
|
|
|
2007-12-01 03:42:52 +00:00
|
|
|
m_segmLength = dx;
|
|
|
|
|
|
|
|
/******************************************/
|
|
|
|
/* Phase 1 : test DRC track to pads : */
|
|
|
|
/******************************************/
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2010-09-18 17:55:08 +00:00
|
|
|
// Use a dummy pad to test DRC tracks versus holes, for pads not on all copper layers
|
|
|
|
// but having a hole
|
|
|
|
D_PAD dummypad( (MODULE*) NULL ); // construct this once outside following loop
|
|
|
|
dummypad.m_Masque_Layer = ALL_CU_LAYERS; // Ensure the hole is on all layers
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2007-12-01 03:42:52 +00:00
|
|
|
// Compute the min distance to pads
|
2008-12-04 04:28:11 +00:00
|
|
|
if( testPads )
|
|
|
|
{
|
2009-06-06 18:08:49 +00:00
|
|
|
for( unsigned ii = 0; ii<m_pcb->GetPadsCount(); ++ii )
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
2009-06-11 14:26:17 +00:00
|
|
|
D_PAD* pad = m_pcb->m_NetInfo->GetPad( ii );
|
2008-12-04 04:28:11 +00:00
|
|
|
|
|
|
|
/* No problem if pads are on an other layer,
|
|
|
|
* But if a drill hole exists (a pad on a single layer can have a hole!)
|
|
|
|
* we must test the hole
|
2007-12-01 03:42:52 +00:00
|
|
|
*/
|
2008-12-04 04:28:11 +00:00
|
|
|
if( (pad->m_Masque_Layer & layerMask ) == 0 )
|
|
|
|
{
|
2010-09-18 17:55:08 +00:00
|
|
|
/* We must test the pad hole. In order to use the function checkClearanceSegmToPad(),
|
2008-12-04 04:28:11 +00:00
|
|
|
* a pseudo pad is used, with a shape and a size like the hole
|
|
|
|
*/
|
|
|
|
if( pad->m_Drill.x == 0 )
|
|
|
|
continue;
|
|
|
|
|
2010-09-18 17:55:08 +00:00
|
|
|
dummypad.m_Size = pad->m_Drill;
|
|
|
|
dummypad.SetPosition( pad->GetPosition() );
|
|
|
|
dummypad.m_PadShape = pad->m_DrillShape;
|
|
|
|
dummypad.m_Orient = pad->m_Orient;
|
|
|
|
dummypad.ComputeShapeMaxRadius(); // compute the radius of the circle containing this pad
|
|
|
|
m_padToTestPos.x = dummypad.GetPosition().x - origin.x;
|
|
|
|
m_padToTestPos.y = dummypad.GetPosition().y - origin.y;
|
2008-12-04 04:28:11 +00:00
|
|
|
|
2010-09-18 17:55:08 +00:00
|
|
|
if( !checkClearanceSegmToPad( &dummypad, aRefSeg->m_Width,
|
2010-09-15 14:53:33 +00:00
|
|
|
netclass->GetClearance() ) )
|
2008-12-04 04:28:11 +00:00
|
|
|
{
|
|
|
|
m_currentMarker = fillMarker( aRefSeg, pad,
|
2009-03-23 19:54:15 +00:00
|
|
|
DRCE_TRACK_NEAR_THROUGH_HOLE, m_currentMarker );
|
2008-12-04 04:28:11 +00:00
|
|
|
return false;
|
|
|
|
}
|
2007-12-01 03:42:52 +00:00
|
|
|
continue;
|
2008-12-04 04:28:11 +00:00
|
|
|
}
|
2007-12-01 03:42:52 +00:00
|
|
|
|
2008-12-04 04:28:11 +00:00
|
|
|
/* The pad must be in a net (i.e pt_pad->GetNet() != 0 )
|
|
|
|
* but no problem if the pad netcode is the current netcode (same net)
|
|
|
|
*/
|
2009-03-23 19:54:15 +00:00
|
|
|
if( pad->GetNet() // the pad must be connected
|
|
|
|
&& net_code_ref == pad->GetNet() ) // the pad net is the same as current net -> Ok
|
2008-12-04 04:28:11 +00:00
|
|
|
continue;
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2008-12-04 04:28:11 +00:00
|
|
|
// DRC for the pad
|
|
|
|
shape_pos = pad->ReturnShapePos();
|
2010-09-18 17:55:08 +00:00
|
|
|
m_padToTestPos.x = shape_pos.x - origin.x;
|
|
|
|
m_padToTestPos.y = shape_pos.y - origin.y;
|
2009-09-10 15:22:26 +00:00
|
|
|
|
2010-09-18 17:55:08 +00:00
|
|
|
if( !checkClearanceSegmToPad( pad, aRefSeg->m_Width, aRefSeg->GetClearance( pad ) ) )
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
2008-02-09 08:34:45 +00:00
|
|
|
m_currentMarker = fillMarker( aRefSeg, pad,
|
2009-03-23 19:54:15 +00:00
|
|
|
DRCE_TRACK_NEAR_PAD, m_currentMarker );
|
2007-12-01 03:42:52 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************/
|
|
|
|
/* Phase 2: test DRC with other track segments */
|
|
|
|
/***********************************************/
|
|
|
|
|
|
|
|
// At this point the reference segment is the X axis
|
|
|
|
|
|
|
|
// Test the reference segment with other track segments
|
2009-03-23 19:54:15 +00:00
|
|
|
for( track = aStart; track; track = track->Next() )
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
|
|
|
// coord des extremites du segment teste dans le repere modifie
|
|
|
|
int x0;
|
|
|
|
int y0;
|
|
|
|
int xf;
|
2008-02-09 08:34:45 +00:00
|
|
|
int yf;
|
|
|
|
|
2007-12-01 03:42:52 +00:00
|
|
|
// No problem if segments have the same net code:
|
|
|
|
if( net_code_ref == track->GetNet() )
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// No problem if segment are on different layers :
|
|
|
|
if( ( layerMask & track->ReturnMaskLayer() ) == 0 )
|
|
|
|
continue;
|
|
|
|
|
2008-02-09 08:34:45 +00:00
|
|
|
// the minimum distance = clearance plus half the reference track
|
2007-12-01 03:42:52 +00:00
|
|
|
// width plus half the other track's width
|
2010-04-08 11:33:43 +00:00
|
|
|
int w_dist = aRefSeg->GetClearance( track );
|
2010-09-15 14:53:33 +00:00
|
|
|
w_dist += (aRefSeg->m_Width + track->m_Width) / 2;
|
2007-12-01 03:42:52 +00:00
|
|
|
|
|
|
|
// If the reference segment is a via, we test it here
|
2008-12-04 04:28:11 +00:00
|
|
|
if( aRefSeg->Type() == TYPE_VIA )
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
|
|
|
int angle = 0; // angle du segment a tester;
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2010-09-18 08:29:40 +00:00
|
|
|
dx = track->m_End.x - track->m_Start.x;
|
|
|
|
dy = track->m_End.y - track->m_Start.y;
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2010-09-18 08:29:40 +00:00
|
|
|
x0 = aRefSeg->m_Start.x - track->m_Start.x;
|
|
|
|
y0 = aRefSeg->m_Start.y - track->m_Start.y;
|
2007-12-01 03:42:52 +00:00
|
|
|
|
2008-12-04 04:28:11 +00:00
|
|
|
if( track->Type() == TYPE_VIA )
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
2010-09-18 08:29:40 +00:00
|
|
|
// Test distance between two vias, i.e. two circles, trivial case
|
2007-12-01 03:42:52 +00:00
|
|
|
if( (int) hypot( x0, y0 ) < w_dist )
|
|
|
|
{
|
2008-02-09 08:34:45 +00:00
|
|
|
m_currentMarker = fillMarker( aRefSeg, track,
|
2009-03-23 19:54:15 +00:00
|
|
|
DRCE_VIA_NEAR_VIA, m_currentMarker );
|
2007-12-01 03:42:52 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else // test via to segment
|
|
|
|
{
|
|
|
|
// Compute l'angle
|
|
|
|
angle = ArcTangente( dy, dx );
|
|
|
|
|
|
|
|
// Compute new coordinates ( the segment become horizontal)
|
|
|
|
RotatePoint( &dx, &dy, angle );
|
|
|
|
RotatePoint( &x0, &y0, angle );
|
|
|
|
|
|
|
|
if( !checkMarginToCircle( x0, y0, w_dist, dx ) )
|
|
|
|
{
|
2008-02-26 21:12:31 +00:00
|
|
|
m_currentMarker = fillMarker( track, aRefSeg,
|
2009-03-23 19:54:15 +00:00
|
|
|
DRCE_VIA_NEAR_TRACK, m_currentMarker );
|
2007-12-01 03:42:52 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2008-02-26 21:12:31 +00:00
|
|
|
/* We compute x0,y0, xf,yf = starting and ending point coordinates for
|
|
|
|
* the segment to test in the new axis : the new X axis is the
|
|
|
|
* reference segment. We must translate and rotate the segment to test
|
|
|
|
*/
|
2010-09-18 08:29:40 +00:00
|
|
|
x0 = track->m_Start.x - origin.x;
|
|
|
|
y0 = track->m_Start.y - origin.y;
|
2007-12-01 03:42:52 +00:00
|
|
|
|
2010-09-18 08:29:40 +00:00
|
|
|
xf = track->m_End.x - origin.x;
|
|
|
|
yf = track->m_End.y - origin.y;
|
2007-12-01 03:42:52 +00:00
|
|
|
|
|
|
|
RotatePoint( &x0, &y0, m_segmAngle );
|
|
|
|
RotatePoint( &xf, &yf, m_segmAngle );
|
|
|
|
|
2008-12-04 04:28:11 +00:00
|
|
|
if( track->Type() == TYPE_VIA )
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
|
|
|
if( checkMarginToCircle( x0, y0, w_dist, m_segmLength ) )
|
|
|
|
continue;
|
2008-02-09 08:34:45 +00:00
|
|
|
|
|
|
|
m_currentMarker = fillMarker( aRefSeg, track,
|
2009-03-23 19:54:15 +00:00
|
|
|
DRCE_TRACK_NEAR_VIA, m_currentMarker );
|
2007-12-01 03:42:52 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* We have changed axis:
|
|
|
|
* the reference segment is Horizontal.
|
|
|
|
* 3 cases : the segment to test can be parallel, perpendicular or have an other direction
|
|
|
|
*/
|
|
|
|
if( y0 == yf ) // parallel segments
|
|
|
|
{
|
|
|
|
if( abs( y0 ) >= w_dist )
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if( x0 > xf )
|
2009-03-23 19:54:15 +00:00
|
|
|
EXCHG( x0, xf ); /* pour que x0 <= xf */
|
2007-12-01 03:42:52 +00:00
|
|
|
|
2009-03-23 19:54:15 +00:00
|
|
|
if( x0 > (-w_dist) && x0 < (m_segmLength + w_dist) ) /* possible error drc */
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
|
|
|
/* Fine test : we consider the rounded shape of the ends */
|
|
|
|
if( x0 >= 0 && x0 <= m_segmLength )
|
|
|
|
{
|
2008-02-09 08:34:45 +00:00
|
|
|
m_currentMarker = fillMarker( aRefSeg, track,
|
2009-03-23 19:54:15 +00:00
|
|
|
DRCE_TRACK_ENDS1, m_currentMarker );
|
2007-12-01 03:42:52 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if( !checkMarginToCircle( x0, y0, w_dist, m_segmLength ) )
|
|
|
|
{
|
2008-02-09 08:34:45 +00:00
|
|
|
m_currentMarker = fillMarker( aRefSeg, track,
|
2009-03-23 19:54:15 +00:00
|
|
|
DRCE_TRACK_ENDS2, m_currentMarker );
|
2007-12-01 03:42:52 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if( xf > (-w_dist) && xf < (m_segmLength + w_dist) )
|
|
|
|
{
|
|
|
|
/* Fine test : we consider the rounded shape of the ends */
|
|
|
|
if( xf >= 0 && xf <= m_segmLength )
|
|
|
|
{
|
2008-02-09 08:34:45 +00:00
|
|
|
m_currentMarker = fillMarker( aRefSeg, track,
|
2009-03-23 19:54:15 +00:00
|
|
|
DRCE_TRACK_ENDS3, m_currentMarker );
|
2007-12-01 03:42:52 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if( !checkMarginToCircle( xf, yf, w_dist, m_segmLength ) )
|
|
|
|
{
|
2008-02-09 08:34:45 +00:00
|
|
|
m_currentMarker = fillMarker( aRefSeg, track,
|
2009-03-23 19:54:15 +00:00
|
|
|
DRCE_TRACK_ENDS4, m_currentMarker );
|
2007-12-01 03:42:52 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if( x0 <=0 && xf >= 0 )
|
|
|
|
{
|
2008-02-09 08:34:45 +00:00
|
|
|
m_currentMarker = fillMarker( aRefSeg, track,
|
2009-03-23 19:54:15 +00:00
|
|
|
DRCE_TRACK_UNKNOWN1, m_currentMarker );
|
2007-12-01 03:42:52 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if( x0 == xf ) // perpendicular segments
|
|
|
|
{
|
|
|
|
if( ( x0 <= (-w_dist) ) || ( x0 >= (m_segmLength + w_dist) ) )
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// Test if segments are crossing
|
|
|
|
if( y0 > yf )
|
|
|
|
EXCHG( y0, yf );
|
|
|
|
if( (y0 < 0) && (yf > 0) )
|
|
|
|
{
|
2008-02-09 08:34:45 +00:00
|
|
|
m_currentMarker = fillMarker( aRefSeg, track,
|
2009-03-23 19:54:15 +00:00
|
|
|
DRCE_TRACKS_CROSSING, m_currentMarker );
|
2007-12-01 03:42:52 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// At this point the drc error is due to an end near a reference segm end
|
|
|
|
if( !checkMarginToCircle( x0, y0, w_dist, m_segmLength ) )
|
|
|
|
{
|
2008-02-09 08:34:45 +00:00
|
|
|
m_currentMarker = fillMarker( aRefSeg, track,
|
2009-03-23 19:54:15 +00:00
|
|
|
DRCE_ENDS_PROBLEM1, m_currentMarker );
|
2007-12-01 03:42:52 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if( !checkMarginToCircle( xf, yf, w_dist, m_segmLength ) )
|
|
|
|
{
|
2008-02-09 08:34:45 +00:00
|
|
|
m_currentMarker = fillMarker( aRefSeg, track,
|
2009-03-23 19:54:15 +00:00
|
|
|
DRCE_ENDS_PROBLEM2, m_currentMarker );
|
2007-12-01 03:42:52 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2009-09-10 15:22:26 +00:00
|
|
|
else // segments quelconques entre eux
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
|
|
|
// calcul de la "surface de securite du segment de reference
|
|
|
|
// First rought 'and fast) test : the track segment is like a rectangle
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2007-12-01 03:42:52 +00:00
|
|
|
m_xcliplo = m_ycliplo = -w_dist;
|
2008-02-09 08:34:45 +00:00
|
|
|
m_xcliphi = m_segmLength + w_dist;
|
2007-12-01 03:42:52 +00:00
|
|
|
m_ycliphi = w_dist;
|
|
|
|
|
2008-02-09 08:34:45 +00:00
|
|
|
// A fine test is needed because a serment is not exactly a
|
2007-12-01 03:42:52 +00:00
|
|
|
// rectangle, it has rounded ends
|
|
|
|
if( !checkLine( x0, y0, xf, yf ) )
|
|
|
|
{
|
|
|
|
/* 2eme passe : the track has rounded ends.
|
2008-02-09 08:34:45 +00:00
|
|
|
* we must a fine test for each rounded end and the
|
|
|
|
* rectangular zone
|
2007-12-01 03:42:52 +00:00
|
|
|
*/
|
|
|
|
|
2008-02-09 08:34:45 +00:00
|
|
|
m_xcliplo = 0;
|
2007-12-01 03:42:52 +00:00
|
|
|
m_xcliphi = m_segmLength;
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2007-12-01 03:42:52 +00:00
|
|
|
if( !checkLine( x0, y0, xf, yf ) )
|
|
|
|
{
|
2008-02-09 08:34:45 +00:00
|
|
|
m_currentMarker = fillMarker( aRefSeg, track,
|
2009-03-23 19:54:15 +00:00
|
|
|
DRCE_ENDS_PROBLEM3, m_currentMarker );
|
2007-12-01 03:42:52 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
else // The drc error is due to the starting or the ending point of the reference segment
|
|
|
|
{
|
|
|
|
// Test the starting and the ending point
|
|
|
|
int angle, rx0, ry0, rxf, ryf;
|
|
|
|
x0 = track->m_Start.x;
|
|
|
|
y0 = track->m_Start.y;
|
|
|
|
|
|
|
|
xf = track->m_End.x;
|
|
|
|
yf = track->m_End.y;
|
|
|
|
|
|
|
|
dx = xf - x0;
|
|
|
|
dy = yf - y0;
|
|
|
|
|
|
|
|
/* Compute the segment orientation (angle) en 0,1 degre */
|
|
|
|
angle = ArcTangente( dy, dx );
|
|
|
|
|
|
|
|
/* Compute the segment lenght: dx = longueur */
|
|
|
|
RotatePoint( &dx, &dy, angle );
|
|
|
|
|
|
|
|
/* Comute the reference segment coordinates relatives to a
|
2008-02-09 08:34:45 +00:00
|
|
|
* X axis = current tested segment
|
2007-12-01 03:42:52 +00:00
|
|
|
*/
|
|
|
|
rx0 = aRefSeg->m_Start.x - x0;
|
|
|
|
ry0 = aRefSeg->m_Start.y - y0;
|
|
|
|
rxf = aRefSeg->m_End.x - x0;
|
|
|
|
ryf = aRefSeg->m_End.y - y0;
|
|
|
|
|
|
|
|
RotatePoint( &rx0, &ry0, angle );
|
|
|
|
RotatePoint( &rxf, &ryf, angle );
|
|
|
|
if( !checkMarginToCircle( rx0, ry0, w_dist, dx ) )
|
|
|
|
{
|
2008-02-09 08:34:45 +00:00
|
|
|
m_currentMarker = fillMarker( aRefSeg, track,
|
2009-03-23 19:54:15 +00:00
|
|
|
DRCE_ENDS_PROBLEM4, m_currentMarker );
|
2007-12-01 03:42:52 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if( !checkMarginToCircle( rxf, ryf, w_dist, dx ) )
|
|
|
|
{
|
2008-02-09 08:34:45 +00:00
|
|
|
m_currentMarker = fillMarker( aRefSeg, track,
|
2009-03-23 19:54:15 +00:00
|
|
|
DRCE_ENDS_PROBLEM5, m_currentMarker );
|
2007-12-01 03:42:52 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
|
|
|
bool DRC::doPadToPadsDrc( D_PAD* aRefPad, LISTE_PAD* aStart, LISTE_PAD* aEnd,
|
2009-09-10 15:22:26 +00:00
|
|
|
int x_limit )
|
2007-12-01 03:42:52 +00:00
|
|
|
/*****************************************************************************/
|
|
|
|
{
|
2010-09-18 17:55:08 +00:00
|
|
|
int layerMask = aRefPad->m_Masque_Layer & ALL_CU_LAYERS;
|
|
|
|
|
|
|
|
// used to test DRC pad to holes: this dummypad is a pad hole to test
|
|
|
|
// pad to pad hole DRC, using pad to pad DRC test.
|
|
|
|
// this dummy pad is a circle or an oval.
|
|
|
|
static D_PAD dummypad( (MODULE*) NULL );
|
|
|
|
dummypad.m_Masque_Layer = ALL_CU_LAYERS; // za hole is on all layers
|
|
|
|
dummypad.m_LocalClearance = 1; /* Use the minimal local clerance value for the dummy pad
|
|
|
|
* the clearance of the active pad will be used
|
|
|
|
* as minimum distance to a hole
|
|
|
|
* (a value = 0 means use netclass value)
|
|
|
|
*/
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2009-03-23 19:54:15 +00:00
|
|
|
for( LISTE_PAD* pad_list = aStart; pad_list<aEnd; ++pad_list )
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
|
|
|
D_PAD* pad = *pad_list;
|
|
|
|
if( pad == aRefPad )
|
|
|
|
continue;
|
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
// We can stop the test when pad->m_Pos.x > x_limit
|
|
|
|
// because the list is sorted by X values
|
|
|
|
if( pad->m_Pos.x > x_limit )
|
2007-12-01 03:42:52 +00:00
|
|
|
break;
|
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
// No problem if pads are on different copper layers,
|
|
|
|
// but their hole (if any ) can create RDC error because they are on all
|
|
|
|
// copper layers, so we test them
|
2007-12-01 03:42:52 +00:00
|
|
|
if( (pad->m_Masque_Layer & layerMask ) == 0 )
|
2009-03-23 19:54:15 +00:00
|
|
|
{
|
2009-09-10 15:22:26 +00:00
|
|
|
// if holes are in the same location and have the same size and shape,
|
|
|
|
// this can be accepted
|
2009-03-23 19:54:15 +00:00
|
|
|
if( pad->GetPosition() == aRefPad->GetPosition()
|
|
|
|
&& pad->m_Drill == aRefPad->m_Drill
|
|
|
|
&& pad->m_DrillShape == aRefPad->m_DrillShape )
|
|
|
|
{
|
|
|
|
if( aRefPad->m_DrillShape == PAD_CIRCLE )
|
|
|
|
continue;
|
2010-09-18 17:55:08 +00:00
|
|
|
if( pad->m_Orient == aRefPad->m_Orient ) // for oval holes: must also have the same orientation
|
2009-03-23 19:54:15 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Here, we must test clearance between holes and pads
|
|
|
|
* dummypad size and shape is adjusted to pad drill size and shape
|
|
|
|
*/
|
2010-09-18 17:55:08 +00:00
|
|
|
if( pad->m_Drill.x ) // pad under testing has a hole, test this hole against pad reference
|
2009-03-23 19:54:15 +00:00
|
|
|
{
|
2010-09-15 14:53:33 +00:00
|
|
|
dummypad.SetPosition( pad->GetPosition() );
|
2009-03-23 19:54:15 +00:00
|
|
|
dummypad.m_Size = pad->m_Drill;
|
2010-09-18 17:55:08 +00:00
|
|
|
dummypad.m_PadShape = (pad->m_DrillShape == PAD_OVAL) ? PAD_OVAL : PAD_CIRCLE;
|
2009-03-23 19:54:15 +00:00
|
|
|
dummypad.m_Orient = pad->m_Orient;
|
2010-09-18 17:55:08 +00:00
|
|
|
dummypad.ComputeShapeMaxRadius(); // compute the radius of the circle containing this pad
|
2009-09-10 15:22:26 +00:00
|
|
|
if( !checkClearancePadToPad( aRefPad, &dummypad ) )
|
2009-03-23 19:54:15 +00:00
|
|
|
{
|
|
|
|
// here we have a drc error on pad!
|
|
|
|
m_currentMarker = fillMarker( pad, aRefPad,
|
|
|
|
DRCE_HOLE_NEAR_PAD, m_currentMarker );
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2009-09-10 15:22:26 +00:00
|
|
|
|
2009-03-23 19:54:15 +00:00
|
|
|
if( aRefPad->m_Drill.x ) // pad reference has a hole
|
|
|
|
{
|
2010-09-15 14:53:33 +00:00
|
|
|
dummypad.SetPosition( aRefPad->GetPosition() );
|
2009-03-23 19:54:15 +00:00
|
|
|
dummypad.m_Size = aRefPad->m_Drill;
|
2010-09-18 17:55:08 +00:00
|
|
|
dummypad.m_PadShape = (aRefPad->m_DrillShape == PAD_OVAL) ? PAD_OVAL : PAD_CIRCLE;
|
2009-03-23 19:54:15 +00:00
|
|
|
dummypad.m_Orient = aRefPad->m_Orient;
|
2010-09-18 17:55:08 +00:00
|
|
|
dummypad.ComputeShapeMaxRadius(); // compute the radius of the circle containing this pad
|
2009-09-10 15:22:26 +00:00
|
|
|
if( !checkClearancePadToPad( pad, &dummypad ) )
|
2009-03-23 19:54:15 +00:00
|
|
|
{
|
|
|
|
// here we have a drc erroron aRefPad!
|
|
|
|
m_currentMarker = fillMarker( aRefPad, pad,
|
|
|
|
DRCE_HOLE_NEAR_PAD, m_currentMarker );
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2007-12-01 03:42:52 +00:00
|
|
|
continue;
|
2009-03-23 19:54:15 +00:00
|
|
|
}
|
2007-12-01 03:42:52 +00:00
|
|
|
|
2010-09-18 17:55:08 +00:00
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
// The pad must be in a net (i.e pt_pad->GetNet() != 0 ),
|
|
|
|
// But no problem if pads have the same netcode (same net)
|
2009-03-23 19:54:15 +00:00
|
|
|
if( pad->GetNet() && ( aRefPad->GetNet() == pad->GetNet() ) )
|
2007-12-01 03:42:52 +00:00
|
|
|
continue;
|
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
// if pads are from the same footprint
|
|
|
|
if( pad->GetParent() == aRefPad->GetParent() )
|
|
|
|
{
|
|
|
|
// and have the same pad number ( equivalent pads )
|
|
|
|
|
|
|
|
// one can argue that this 2nd test is not necessary, that any
|
|
|
|
// two pads from a single module are acceptable. This 2nd test
|
|
|
|
// should eventually be a configuration option.
|
|
|
|
if( pad->m_NumPadName == aRefPad->m_NumPadName )
|
|
|
|
continue;
|
|
|
|
}
|
2007-12-01 03:42:52 +00:00
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
if( !checkClearancePadToPad( aRefPad, pad ) )
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
|
|
|
// here we have a drc error!
|
2008-02-09 08:34:45 +00:00
|
|
|
m_currentMarker = fillMarker( aRefPad, pad,
|
2009-03-23 19:54:15 +00:00
|
|
|
DRCE_PAD_NEAR_PAD1, m_currentMarker );
|
2007-12-01 03:42:52 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-09-07 13:28:11 +00:00
|
|
|
// Rotate a vector by an angle
|
2009-09-10 15:22:26 +00:00
|
|
|
wxPoint rotate( wxPoint p, int angle )
|
|
|
|
{
|
2009-09-07 13:28:11 +00:00
|
|
|
wxPoint n;
|
2010-09-15 14:53:33 +00:00
|
|
|
double theta = M_PI * (double) angle / 1800.0;
|
|
|
|
|
|
|
|
n.x = wxRound( (double) p.x * cos( theta ) - (double) p.y * sin( theta ) );
|
2009-12-15 21:11:05 +00:00
|
|
|
n.y = wxRound( p.x * sin( theta ) + p.y * cos( theta ) );
|
2009-09-07 13:28:11 +00:00
|
|
|
return n;
|
|
|
|
}
|
|
|
|
|
2009-12-15 21:11:05 +00:00
|
|
|
|
2010-09-18 17:55:08 +00:00
|
|
|
/* test DRC between 2 pads.
|
|
|
|
* this function can be also used to test DRC between a pas and a hole,
|
|
|
|
* because a hole is like a round pad.
|
|
|
|
*/
|
2009-09-10 15:22:26 +00:00
|
|
|
bool DRC::checkClearancePadToPad( D_PAD* aRefPad, D_PAD* aPad )
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
2010-09-18 17:55:08 +00:00
|
|
|
int dist;
|
2009-03-23 19:54:15 +00:00
|
|
|
|
2007-12-01 03:42:52 +00:00
|
|
|
int pad_angle;
|
|
|
|
|
2010-09-18 17:55:08 +00:00
|
|
|
// Get the clerance between the 2 pads. this is the min distance between aRefPad and aPad
|
2009-09-10 15:22:26 +00:00
|
|
|
int dist_min = aRefPad->GetClearance( aPad );
|
|
|
|
|
2010-09-18 17:55:08 +00:00
|
|
|
// relativePadPos is the aPad shape position relative to the aRefPad shape position
|
|
|
|
wxPoint relativePadPos = aPad->ReturnShapePos() - aRefPad->ReturnShapePos();
|
2007-12-01 03:42:52 +00:00
|
|
|
|
2010-09-18 17:55:08 +00:00
|
|
|
dist = (int) hypot( relativePadPos.x, relativePadPos.y );
|
2007-12-01 03:42:52 +00:00
|
|
|
|
2010-09-18 17:55:08 +00:00
|
|
|
// return true if clearance between aRefPad and aPad is >= dist_min, else false
|
2007-12-01 03:42:52 +00:00
|
|
|
bool diag = true;
|
|
|
|
|
2009-09-10 15:22:26 +00:00
|
|
|
// Quick test: Clearance is OK if the bounding circles are further away than "dist_min"
|
2010-09-18 17:55:08 +00:00
|
|
|
if( (dist - aRefPad->m_ShapeMaxRadius - aPad->m_ShapeMaxRadius) >= dist_min )
|
2007-12-04 18:23:38 +00:00
|
|
|
goto exit;
|
2007-12-01 03:42:52 +00:00
|
|
|
|
2010-09-15 14:53:33 +00:00
|
|
|
/* Here, pads are near and DRC depend on the pad shapes
|
2010-09-18 08:29:40 +00:00
|
|
|
* We must compare distance using a fine shape analysis
|
|
|
|
* Because a circle or oval shape is the easier shape to test, try to have
|
|
|
|
* aRefPad shape type = PAD_CIRCLE or PAD_OVAL. Swap aRefPad and aPad if needed
|
|
|
|
*/
|
2007-12-04 18:23:38 +00:00
|
|
|
bool swap_pads;
|
|
|
|
swap_pads = false;
|
2008-01-05 17:30:56 +00:00
|
|
|
if( (aRefPad->m_PadShape != PAD_CIRCLE) && (aPad->m_PadShape == PAD_CIRCLE) )
|
2007-12-01 03:42:52 +00:00
|
|
|
swap_pads = true;
|
2008-01-05 17:30:56 +00:00
|
|
|
else if( (aRefPad->m_PadShape != PAD_OVAL) && (aPad->m_PadShape == PAD_OVAL) )
|
2007-12-01 03:42:52 +00:00
|
|
|
swap_pads = true;
|
|
|
|
|
|
|
|
if( swap_pads )
|
|
|
|
{
|
|
|
|
EXCHG( aRefPad, aPad );
|
2010-09-18 17:55:08 +00:00
|
|
|
relativePadPos = -relativePadPos;
|
2007-12-01 03:42:52 +00:00
|
|
|
}
|
|
|
|
|
2010-09-15 14:53:33 +00:00
|
|
|
/* Because pad exchange, aRefPad shape is PAD_CIRCLE or PAD_OVAL,
|
|
|
|
* if one of the 2 pads was a PAD_CIRCLE or PAD_OVAL.
|
|
|
|
* Therefore, if aRefPad is a PAD_RECT or a PAD_TRAPEZOID,
|
2010-09-18 08:29:40 +00:00
|
|
|
* aPad is also a PAD_RECT or a PAD_TRAPEZOID
|
2010-09-15 14:53:33 +00:00
|
|
|
*/
|
2007-12-01 03:42:52 +00:00
|
|
|
switch( aRefPad->m_PadShape )
|
|
|
|
{
|
2010-09-18 17:55:08 +00:00
|
|
|
case PAD_CIRCLE:
|
|
|
|
/* One can use checkClearanceSegmToPad to test clearance
|
|
|
|
* aRefPad is like a track segment with a null lenght and a witdth = m_Size.x
|
|
|
|
*/
|
2009-03-23 19:54:15 +00:00
|
|
|
m_segmLength = 0;
|
|
|
|
m_segmAngle = 0;
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2010-09-18 08:29:40 +00:00
|
|
|
m_segmEnd.x = m_segmEnd.y = 0;
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2010-09-18 17:55:08 +00:00
|
|
|
m_padToTestPos.x = relativePadPos.x;
|
|
|
|
m_padToTestPos.y = relativePadPos.y;
|
|
|
|
diag = checkClearanceSegmToPad( aPad, aRefPad->m_Size.x, dist_min );
|
2007-12-01 03:42:52 +00:00
|
|
|
break;
|
|
|
|
|
2008-01-05 17:30:56 +00:00
|
|
|
case PAD_RECT:
|
2010-09-18 17:55:08 +00:00
|
|
|
RotatePoint( &relativePadPos, aRefPad->m_Orient );
|
2010-09-15 14:53:33 +00:00
|
|
|
|
|
|
|
// pad_angle = pad orient relative to the aRefPad orient
|
|
|
|
pad_angle = aRefPad->m_Orient + aPad->m_Orient;
|
2007-12-01 03:42:52 +00:00
|
|
|
NORMALIZE_ANGLE_POS( pad_angle );
|
2008-01-05 17:30:56 +00:00
|
|
|
if( aPad->m_PadShape == PAD_RECT )
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
|
|
|
wxSize size = aPad->m_Size;
|
2010-09-15 14:53:33 +00:00
|
|
|
|
|
|
|
// The trivial case is if both rects are rotated by multiple of 90 deg
|
|
|
|
// Most of time this is the case, and the test is fast
|
|
|
|
if( ( (aRefPad->m_Orient == 0) || (aRefPad->m_Orient == 900)
|
2010-09-18 08:29:40 +00:00
|
|
|
|| (aRefPad->m_Orient == 1800) || (aRefPad->m_Orient == 2700) )
|
2010-09-15 14:53:33 +00:00
|
|
|
&& ( (aPad->m_Orient == 0) || (aPad->m_Orient == 900) || (aPad->m_Orient == 1800)
|
|
|
|
|| (aPad->m_Orient == 2700) ) )
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
|
|
|
if( (pad_angle == 900) || (pad_angle == 2700) )
|
|
|
|
{
|
|
|
|
EXCHG( size.x, size.y );
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test DRC:
|
2009-03-23 19:54:15 +00:00
|
|
|
diag = false;
|
2007-12-01 03:42:52 +00:00
|
|
|
|
2010-09-18 17:55:08 +00:00
|
|
|
relativePadPos.x = ABS( relativePadPos.x );
|
|
|
|
relativePadPos.y = ABS( relativePadPos.y );
|
2007-12-01 03:42:52 +00:00
|
|
|
|
2010-09-18 17:55:08 +00:00
|
|
|
if( ( relativePadPos.x - ( (size.x + aRefPad->m_Size.x) / 2 ) ) >= dist_min )
|
2007-12-01 03:42:52 +00:00
|
|
|
diag = true;
|
|
|
|
|
2010-09-18 17:55:08 +00:00
|
|
|
if( ( relativePadPos.y - ( (size.y + aRefPad->m_Size.y) / 2 ) ) >= dist_min )
|
2007-12-01 03:42:52 +00:00
|
|
|
diag = true;
|
|
|
|
}
|
2010-09-15 14:53:33 +00:00
|
|
|
else // al least on pad has any other orient. Test is more tricky
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
2009-09-07 13:28:11 +00:00
|
|
|
/* Use TestForIntersectionOfStraightLineSegments() for all 4 edges (segments).*/
|
2009-09-10 15:22:26 +00:00
|
|
|
|
|
|
|
/* Test if one center point is contained in the other and thus the pads overlap.
|
2009-09-07 13:28:11 +00:00
|
|
|
* This case is not covered by the following check if one pad is
|
|
|
|
* completely contained in the other (because edges don't intersect)!
|
|
|
|
*/
|
2010-09-15 14:53:33 +00:00
|
|
|
if( ( (dist < aPad->m_Size.x) && (dist < aPad->m_Size.y) )
|
|
|
|
|| ( (dist < aRefPad->m_Size.x) && (dist < aRefPad->m_Size.y) ) )
|
|
|
|
{
|
2009-09-07 13:28:11 +00:00
|
|
|
diag = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Vectors from center to corner
|
2010-09-15 14:53:33 +00:00
|
|
|
wxPoint aPad_c2c = wxPoint( aPad->m_Size.x / 2, aPad->m_Size.y / 2 );
|
|
|
|
wxPoint aRefPad_c2c = wxPoint( aRefPad->m_Size.x / 2, aRefPad->m_Size.y / 2 );
|
|
|
|
|
|
|
|
for( int i = 0; i<4; i++ ) // for all edges in aPad
|
|
|
|
{
|
|
|
|
wxPoint p11 = aPad->ReturnShapePos() + rotate( aPad_c2c, aPad->m_Orient );
|
2009-09-07 13:28:11 +00:00
|
|
|
|
|
|
|
// flip the center-to-corner vector
|
2010-09-15 14:53:33 +00:00
|
|
|
if( i % 2 == 0 )
|
|
|
|
{
|
2009-09-07 13:28:11 +00:00
|
|
|
aPad_c2c.x = -aPad_c2c.x;
|
2010-09-15 14:53:33 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2009-09-07 13:28:11 +00:00
|
|
|
aPad_c2c.y = -aPad_c2c.y;
|
|
|
|
}
|
2010-09-15 14:53:33 +00:00
|
|
|
wxPoint p12 = aPad->ReturnShapePos() + rotate( aPad_c2c, aPad->m_Orient );
|
|
|
|
|
|
|
|
for( int j = 0; j<4; j++ ) // for all edges in aRefPad
|
|
|
|
{
|
|
|
|
wxPoint p21 = aRefPad->ReturnShapePos() + rotate( aRefPad_c2c,
|
|
|
|
aRefPad->m_Orient );
|
2009-09-10 15:22:26 +00:00
|
|
|
|
2009-09-07 13:28:11 +00:00
|
|
|
// flip the center-to-corner vector
|
2010-09-15 14:53:33 +00:00
|
|
|
if( j % 2 == 0 )
|
|
|
|
{
|
2009-09-07 13:28:11 +00:00
|
|
|
aRefPad_c2c.x = -aRefPad_c2c.x;
|
2010-09-15 14:53:33 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2009-09-07 13:28:11 +00:00
|
|
|
aRefPad_c2c.y = -aRefPad_c2c.y;
|
|
|
|
}
|
2010-09-15 14:53:33 +00:00
|
|
|
wxPoint p22 = aRefPad->ReturnShapePos() + rotate( aRefPad_c2c,
|
|
|
|
aRefPad->m_Orient );
|
|
|
|
|
|
|
|
int x, y;
|
|
|
|
double d;
|
|
|
|
int intersect = TestForIntersectionOfStraightLineSegments( p11.x,
|
|
|
|
p11.y,
|
|
|
|
p12.x,
|
|
|
|
p12.y,
|
|
|
|
p21.x,
|
|
|
|
p21.y,
|
|
|
|
p22.x,
|
|
|
|
p22.y,
|
|
|
|
&x,
|
|
|
|
&y,
|
|
|
|
&d );
|
2009-09-10 15:22:26 +00:00
|
|
|
if( intersect || (d< dist_min) )
|
|
|
|
{
|
2010-09-15 14:53:33 +00:00
|
|
|
diag = false;
|
2009-09-07 13:28:11 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2007-12-01 03:42:52 +00:00
|
|
|
}
|
2010-09-15 14:53:33 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// TODO: Pad -> other shape! (PAD_TRAPEZOID)
|
2007-12-01 03:42:52 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2008-01-05 17:30:56 +00:00
|
|
|
case PAD_OVAL: /* an oval pad is like a track segment */
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
2010-09-15 14:53:33 +00:00
|
|
|
/* Create a track segment with same dimensions as the oval aRefPad
|
|
|
|
* and use checkClearanceSegmToPad function to test aPad to aRefPad clearance
|
2010-09-18 17:55:08 +00:00
|
|
|
*/
|
2007-12-01 03:42:52 +00:00
|
|
|
int segm_width;
|
2010-09-18 17:55:08 +00:00
|
|
|
m_segmAngle = aRefPad->m_Orient; // Segment orient.
|
|
|
|
if( aRefPad->m_Size.y < aRefPad->m_Size.x ) // Build an horizontal equiv segment
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
2009-03-23 19:54:15 +00:00
|
|
|
segm_width = aRefPad->m_Size.y;
|
|
|
|
m_segmLength = aRefPad->m_Size.x - aRefPad->m_Size.y;
|
2007-12-01 03:42:52 +00:00
|
|
|
}
|
2010-09-15 14:53:33 +00:00
|
|
|
else // Vertical oval: build an horizontal equiv segment and rotate 90.0 deg
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
2009-03-23 19:54:15 +00:00
|
|
|
segm_width = aRefPad->m_Size.x;
|
|
|
|
m_segmLength = aRefPad->m_Size.y - aRefPad->m_Size.x;
|
2007-12-01 03:42:52 +00:00
|
|
|
m_segmAngle += 900;
|
|
|
|
}
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2010-09-18 17:55:08 +00:00
|
|
|
/* the start point must be 0,0 and currently relativePadPos
|
2010-09-15 14:53:33 +00:00
|
|
|
* is relative the center of pad coordinate */
|
|
|
|
wxPoint segstart;
|
2010-09-18 17:55:08 +00:00
|
|
|
segstart.x = -m_segmLength / 2; // Start point coordinate of the horizontal equivalent segment
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2010-09-18 17:55:08 +00:00
|
|
|
RotatePoint( &segstart, m_segmAngle ); // True start point coordinate of the equivalent segment
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2010-09-15 14:53:33 +00:00
|
|
|
// move pad position relative to the segment origin
|
2010-09-18 17:55:08 +00:00
|
|
|
m_padToTestPos = relativePadPos - segstart;
|
|
|
|
|
2010-09-15 14:53:33 +00:00
|
|
|
// Calculate segment end
|
2010-09-18 08:29:40 +00:00
|
|
|
m_segmEnd.x = -2 * segstart.x;
|
|
|
|
m_segmEnd.y = -2 * segstart.y; // end of segment coordinate
|
2010-09-18 17:55:08 +00:00
|
|
|
diag = checkClearanceSegmToPad( aPad, segm_width, dist_min );
|
2007-12-01 03:42:52 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2009-03-23 19:54:15 +00:00
|
|
|
case PAD_TRAPEZOID:
|
2007-12-01 03:42:52 +00:00
|
|
|
default:
|
|
|
|
/* TODO...*/
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2007-12-04 18:23:38 +00:00
|
|
|
exit: // the only way out (hopefully) for simpler debugging
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2007-12-01 03:42:52 +00:00
|
|
|
return diag;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-09-18 17:55:08 +00:00
|
|
|
/* test if distance between a segment is > aMinDist
|
|
|
|
* segment start point is assumed in (0,0) and segment start point in m_segmEnd
|
|
|
|
* and have aSegmentWidth.
|
|
|
|
*/
|
|
|
|
bool DRC::checkClearanceSegmToPad( const D_PAD* aPad, int aSegmentWidth, int aMinDist )
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
2010-09-18 08:29:40 +00:00
|
|
|
wxSize padHalfsize; // half the dimension of the pad
|
2010-09-18 17:55:08 +00:00
|
|
|
int orient;
|
|
|
|
int x0, y0, xf, yf;
|
|
|
|
int seuil;
|
|
|
|
int deltay;
|
2007-12-01 03:42:52 +00:00
|
|
|
|
2010-09-18 17:55:08 +00:00
|
|
|
int segmHalfWidth = aSegmentWidth / 2;
|
|
|
|
seuil = segmHalfWidth + aMinDist;
|
|
|
|
padHalfsize.x = aPad->m_Size.x >> 1;
|
|
|
|
padHalfsize.y = aPad->m_Size.y >> 1;
|
2007-12-01 03:42:52 +00:00
|
|
|
|
2010-09-18 17:55:08 +00:00
|
|
|
if( aPad->m_PadShape == PAD_CIRCLE )
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
2010-09-18 17:55:08 +00:00
|
|
|
/* Easy case: just test the distance between segment and pad centre
|
|
|
|
* calculate pad coordinates in the X,Y axis with X axis = segment to test
|
|
|
|
*/
|
2010-09-18 08:29:40 +00:00
|
|
|
RotatePoint( &m_padToTestPos.x, &m_padToTestPos.y, m_segmAngle );
|
2010-09-18 17:55:08 +00:00
|
|
|
return checkMarginToCircle( m_padToTestPos.x, m_padToTestPos.y,
|
|
|
|
seuil + padHalfsize.x, m_segmLength );
|
2007-12-01 03:42:52 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2010-09-18 17:55:08 +00:00
|
|
|
/* calculate the bounding box of the pad, including the clearance and the segment width
|
|
|
|
* if the line from 0 to m_segmEnd does not intersect this bounding box,
|
|
|
|
* the clearance is always OK
|
|
|
|
* But if intersect, a better analysis of the pad shape must be done.
|
|
|
|
*/
|
2010-09-18 08:29:40 +00:00
|
|
|
m_xcliplo = m_padToTestPos.x - seuil - padHalfsize.x;
|
|
|
|
m_ycliplo = m_padToTestPos.y - seuil - padHalfsize.y;
|
|
|
|
m_xcliphi = m_padToTestPos.x + seuil + padHalfsize.x;
|
|
|
|
m_ycliphi = m_padToTestPos.y + seuil + padHalfsize.y;
|
2007-12-01 03:42:52 +00:00
|
|
|
|
|
|
|
x0 = y0 = 0;
|
|
|
|
|
2010-09-18 08:29:40 +00:00
|
|
|
xf = m_segmEnd.x;
|
|
|
|
yf = m_segmEnd.y;
|
2007-12-01 03:42:52 +00:00
|
|
|
|
2010-09-18 17:55:08 +00:00
|
|
|
orient = aPad->m_Orient;
|
2007-12-01 03:42:52 +00:00
|
|
|
|
2010-09-18 08:29:40 +00:00
|
|
|
RotatePoint( &x0, &y0, m_padToTestPos.x, m_padToTestPos.y, -orient );
|
|
|
|
RotatePoint( &xf, &yf, m_padToTestPos.x, m_padToTestPos.y, -orient );
|
2007-12-01 03:42:52 +00:00
|
|
|
|
|
|
|
if( checkLine( x0, y0, xf, yf ) )
|
|
|
|
return true;
|
|
|
|
|
2010-09-18 17:55:08 +00:00
|
|
|
/* segment intersects the bounding box. But there is not always a DRC error.
|
|
|
|
* A fine analysis of the pad shape must be done.
|
|
|
|
*/
|
|
|
|
switch( aPad->m_PadShape )
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
|
2008-01-05 17:30:56 +00:00
|
|
|
case PAD_OVAL:
|
2010-09-18 17:55:08 +00:00
|
|
|
/* an oval is a complex shape, but is a rectangle and 2 circles
|
|
|
|
* these 3 basic shapes are more easy to test.
|
|
|
|
*/
|
|
|
|
/* We use a vertical oval shape. for horizontal ovals, swap x and y size and rotate the shape*/
|
2010-09-18 08:29:40 +00:00
|
|
|
if( padHalfsize.x > padHalfsize.y )
|
2007-12-01 03:42:52 +00:00
|
|
|
{
|
2010-09-18 08:29:40 +00:00
|
|
|
EXCHG( padHalfsize.x, padHalfsize.y );
|
2007-12-01 03:42:52 +00:00
|
|
|
orient += 900;
|
|
|
|
if( orient >= 3600 )
|
|
|
|
orient -= 3600;
|
|
|
|
}
|
2010-09-18 08:29:40 +00:00
|
|
|
deltay = padHalfsize.y - padHalfsize.x;
|
2007-12-01 03:42:52 +00:00
|
|
|
|
2010-09-18 17:55:08 +00:00
|
|
|
// ici: padHalfsize.x = rayon, delta = dist centre cercles a centre pad
|
2007-12-01 03:42:52 +00:00
|
|
|
|
2010-09-18 17:55:08 +00:00
|
|
|
// Test the rectangle area between the two circles
|
2010-09-18 08:29:40 +00:00
|
|
|
m_xcliplo = m_padToTestPos.x - seuil - padHalfsize.x;
|
2010-09-18 17:55:08 +00:00
|
|
|
m_ycliplo = m_padToTestPos.y - segmHalfWidth - deltay;
|
2010-09-18 08:29:40 +00:00
|
|
|
m_xcliphi = m_padToTestPos.x + seuil + padHalfsize.x;
|
2010-09-18 17:55:08 +00:00
|
|
|
m_ycliphi = m_padToTestPos.y + segmHalfWidth + deltay;
|
2007-12-01 03:42:52 +00:00
|
|
|
if( !checkLine( x0, y0, xf, yf ) )
|
|
|
|
return false;
|
|
|
|
|
2010-09-18 17:55:08 +00:00
|
|
|
// test the first circle
|
|
|
|
x0 = m_padToTestPos.x; // x0,y0 = centre of the upper circle of the oval shape
|
2010-09-18 08:29:40 +00:00
|
|
|
y0 = m_padToTestPos.y + deltay;
|
2010-09-18 17:55:08 +00:00
|
|
|
|
|
|
|
// Calculate the actual position of the circle, given the pad orientation:
|
2010-09-18 08:29:40 +00:00
|
|
|
RotatePoint( &x0, &y0, m_padToTestPos.x, m_padToTestPos.y, orient );
|
2007-12-01 03:42:52 +00:00
|
|
|
|
2010-09-18 17:55:08 +00:00
|
|
|
// Calculate the actual position of the circle in the new X,Y axis:
|
|
|
|
RotatePoint( &x0, &y0, m_segmAngle );
|
2010-09-18 08:29:40 +00:00
|
|
|
if( !checkMarginToCircle( x0, y0, padHalfsize.x + seuil, m_segmLength ) )
|
2007-12-01 03:42:52 +00:00
|
|
|
return false;
|
|
|
|
|
2010-09-18 17:55:08 +00:00
|
|
|
// test the second circle
|
|
|
|
x0 = m_padToTestPos.x; // x0,y0 = centre of the lower circle of the oval shape
|
2010-09-18 08:29:40 +00:00
|
|
|
y0 = m_padToTestPos.y - deltay;
|
|
|
|
RotatePoint( &x0, &y0, m_padToTestPos.x, m_padToTestPos.y, orient );
|
2007-12-01 03:42:52 +00:00
|
|
|
RotatePoint( &x0, &y0, m_segmAngle );
|
|
|
|
|
2010-09-18 08:29:40 +00:00
|
|
|
if( !checkMarginToCircle( x0, y0, padHalfsize.x + seuil, m_segmLength ) )
|
2007-12-01 03:42:52 +00:00
|
|
|
return false;
|
|
|
|
break;
|
|
|
|
|
2008-01-05 17:30:56 +00:00
|
|
|
case PAD_RECT: /* 2 rectangle + 4 1/4 cercles a tester */
|
2007-12-01 03:42:52 +00:00
|
|
|
/* Test du rectangle dimx + seuil, dimy */
|
2010-09-18 08:29:40 +00:00
|
|
|
m_xcliplo = m_padToTestPos.x - padHalfsize.x - seuil;
|
|
|
|
m_ycliplo = m_padToTestPos.y - padHalfsize.y;
|
|
|
|
m_xcliphi = m_padToTestPos.x + padHalfsize.x + seuil;
|
|
|
|
m_ycliphi = m_padToTestPos.y + padHalfsize.y;
|
2007-12-01 03:42:52 +00:00
|
|
|
|
|
|
|
if( !checkLine( x0, y0, xf, yf ) )
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Test du rectangle dimx , dimy + seuil */
|
2010-09-18 08:29:40 +00:00
|
|
|
m_xcliplo = m_padToTestPos.x - padHalfsize.x;
|
|
|
|
m_ycliplo = m_padToTestPos.y - padHalfsize.y - seuil;
|
|
|
|
m_xcliphi = m_padToTestPos.x + padHalfsize.x;
|
|
|
|
m_ycliphi = m_padToTestPos.y + padHalfsize.y + seuil;
|
2007-12-01 03:42:52 +00:00
|
|
|
|
|
|
|
if( !checkLine( x0, y0, xf, yf ) )
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* test des 4 cercles ( surface d'solation autour des sommets */
|
|
|
|
/* test du coin sup. gauche du pad */
|
2010-09-18 08:29:40 +00:00
|
|
|
x0 = m_padToTestPos.x - padHalfsize.x;
|
|
|
|
y0 = m_padToTestPos.y - padHalfsize.y;
|
|
|
|
RotatePoint( &x0, &y0, m_padToTestPos.x, m_padToTestPos.y, orient );
|
2007-12-01 03:42:52 +00:00
|
|
|
RotatePoint( &x0, &y0, m_segmAngle );
|
|
|
|
if( !checkMarginToCircle( x0, y0, seuil, m_segmLength ) )
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* test du coin sup. droit du pad */
|
2010-09-18 08:29:40 +00:00
|
|
|
x0 = m_padToTestPos.x + padHalfsize.x;
|
|
|
|
y0 = m_padToTestPos.y - padHalfsize.y;
|
|
|
|
RotatePoint( &x0, &y0, m_padToTestPos.x, m_padToTestPos.y, orient );
|
2007-12-01 03:42:52 +00:00
|
|
|
RotatePoint( &x0, &y0, m_segmAngle );
|
|
|
|
if( !checkMarginToCircle( x0, y0, seuil, m_segmLength ) )
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* test du coin inf. gauche du pad */
|
2010-09-18 08:29:40 +00:00
|
|
|
x0 = m_padToTestPos.x - padHalfsize.x;
|
|
|
|
y0 = m_padToTestPos.y + padHalfsize.y;
|
|
|
|
RotatePoint( &x0, &y0, m_padToTestPos.x, m_padToTestPos.y, orient );
|
2007-12-01 03:42:52 +00:00
|
|
|
RotatePoint( &x0, &y0, m_segmAngle );
|
|
|
|
if( !checkMarginToCircle( x0, y0, seuil, m_segmLength ) )
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* test du coin inf. droit du pad */
|
2010-09-18 08:29:40 +00:00
|
|
|
x0 = m_padToTestPos.x + padHalfsize.x;
|
|
|
|
y0 = m_padToTestPos.y + padHalfsize.y;
|
|
|
|
RotatePoint( &x0, &y0, m_padToTestPos.x, m_padToTestPos.y, orient );
|
2007-12-01 03:42:52 +00:00
|
|
|
RotatePoint( &x0, &y0, m_segmAngle );
|
|
|
|
if( !checkMarginToCircle( x0, y0, seuil, m_segmLength ) )
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
2010-09-15 14:53:33 +00:00
|
|
|
|
|
|
|
case PAD_TRAPEZOID: //TODO
|
|
|
|
break;
|
2007-12-01 03:42:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2009-03-23 19:54:15 +00:00
|
|
|
|
2007-12-01 03:42:52 +00:00
|
|
|
/**********************************************************************/
|
|
|
|
bool DRC::checkMarginToCircle( int cx, int cy, int radius, int longueur )
|
|
|
|
/**********************************************************************/
|
|
|
|
{
|
|
|
|
if( abs( cy ) > radius )
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if( (cx >= -radius ) && ( cx <= (longueur + radius) ) )
|
|
|
|
{
|
|
|
|
if( (cx >= 0) && (cx <= longueur) )
|
|
|
|
return false;
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2007-12-01 03:42:52 +00:00
|
|
|
if( cx > longueur )
|
|
|
|
cx -= longueur;
|
2008-02-09 08:34:45 +00:00
|
|
|
|
2007-12-01 03:42:52 +00:00
|
|
|
if( hypot( cx, cy ) < radius )
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**********************************************/
|
|
|
|
/* int Tst_Ligne(int x1,int y1,int x2,int y2) */
|
|
|
|
/**********************************************/
|
|
|
|
|
|
|
|
static inline int USCALE( unsigned arg, unsigned num, unsigned den )
|
|
|
|
{
|
|
|
|
int ii;
|
|
|
|
|
|
|
|
ii = (int) ( ( (double) arg * num ) / den );
|
|
|
|
return ii;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#define WHEN_OUTSIDE return true
|
|
|
|
#define WHEN_INSIDE
|
|
|
|
|
|
|
|
bool DRC::checkLine( int x1, int y1, int x2, int y2 )
|
|
|
|
{
|
|
|
|
int temp;
|
|
|
|
|
|
|
|
if( x1 > x2 )
|
|
|
|
{
|
|
|
|
EXCHG( x1, x2 );
|
|
|
|
EXCHG( y1, y2 );
|
|
|
|
}
|
|
|
|
if( (x2 < m_xcliplo) || (x1 > m_xcliphi) )
|
|
|
|
{
|
|
|
|
WHEN_OUTSIDE;
|
|
|
|
}
|
|
|
|
if( y1 < y2 )
|
|
|
|
{
|
|
|
|
if( (y2 < m_ycliplo) || (y1 > m_ycliphi) )
|
|
|
|
{
|
|
|
|
WHEN_OUTSIDE;
|
|
|
|
}
|
|
|
|
if( y1 < m_ycliplo )
|
|
|
|
{
|
|
|
|
temp = USCALE( (x2 - x1), (m_ycliplo - y1), (y2 - y1) );
|
|
|
|
if( (x1 += temp) > m_xcliphi )
|
|
|
|
{
|
|
|
|
WHEN_OUTSIDE;
|
|
|
|
}
|
|
|
|
y1 = m_ycliplo;
|
|
|
|
WHEN_INSIDE;
|
|
|
|
}
|
|
|
|
if( y2 > m_ycliphi )
|
|
|
|
{
|
|
|
|
temp = USCALE( (x2 - x1), (y2 - m_ycliphi), (y2 - y1) );
|
|
|
|
if( (x2 -= temp) < m_xcliplo )
|
|
|
|
{
|
|
|
|
WHEN_OUTSIDE;
|
|
|
|
}
|
|
|
|
y2 = m_ycliphi;
|
|
|
|
WHEN_INSIDE;
|
|
|
|
}
|
|
|
|
if( x1 < m_xcliplo )
|
|
|
|
{
|
|
|
|
temp = USCALE( (y2 - y1), (m_xcliplo - x1), (x2 - x1) );
|
2008-02-09 08:34:45 +00:00
|
|
|
y1 += temp;
|
2009-03-23 19:54:15 +00:00
|
|
|
x1 = m_xcliplo;
|
2007-12-01 03:42:52 +00:00
|
|
|
WHEN_INSIDE;
|
|
|
|
}
|
|
|
|
if( x2 > m_xcliphi )
|
|
|
|
{
|
|
|
|
temp = USCALE( (y2 - y1), (x2 - m_xcliphi), (x2 - x1) );
|
2008-02-09 08:34:45 +00:00
|
|
|
y2 -= temp;
|
2009-03-23 19:54:15 +00:00
|
|
|
x2 = m_xcliphi;
|
2007-12-01 03:42:52 +00:00
|
|
|
WHEN_INSIDE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if( (y1 < m_ycliplo) || (y2 > m_ycliphi) )
|
|
|
|
{
|
|
|
|
WHEN_OUTSIDE;
|
|
|
|
}
|
|
|
|
if( y1 > m_ycliphi )
|
|
|
|
{
|
|
|
|
temp = USCALE( (x2 - x1), (y1 - m_ycliphi), (y1 - y2) );
|
|
|
|
if( (x1 += temp) > m_xcliphi )
|
|
|
|
{
|
|
|
|
WHEN_OUTSIDE;
|
|
|
|
}
|
|
|
|
y1 = m_ycliphi;
|
|
|
|
WHEN_INSIDE;
|
2007-08-08 20:51:08 +00:00
|
|
|
}
|
2007-12-01 03:42:52 +00:00
|
|
|
if( y2 < m_ycliplo )
|
2007-08-08 20:51:08 +00:00
|
|
|
{
|
2007-12-01 03:42:52 +00:00
|
|
|
temp = USCALE( (x2 - x1), (m_ycliplo - y2), (y1 - y2) );
|
|
|
|
if( (x2 -= temp) < m_xcliplo )
|
|
|
|
{
|
|
|
|
WHEN_OUTSIDE;
|
|
|
|
}
|
|
|
|
y2 = m_ycliplo;
|
|
|
|
WHEN_INSIDE;
|
2007-08-08 20:51:08 +00:00
|
|
|
}
|
2007-12-01 03:42:52 +00:00
|
|
|
if( x1 < m_xcliplo )
|
2007-08-08 20:51:08 +00:00
|
|
|
{
|
2007-12-01 03:42:52 +00:00
|
|
|
temp = USCALE( (y1 - y2), (m_xcliplo - x1), (x2 - x1) );
|
|
|
|
y1 -= temp;
|
2009-03-23 19:54:15 +00:00
|
|
|
x1 = m_xcliplo;
|
2007-12-01 03:42:52 +00:00
|
|
|
WHEN_INSIDE;
|
|
|
|
}
|
|
|
|
if( x2 > m_xcliphi )
|
|
|
|
{
|
|
|
|
temp = USCALE( (y1 - y2), (x2 - m_xcliphi), (x2 - x1) );
|
|
|
|
y2 += temp;
|
2009-03-23 19:54:15 +00:00
|
|
|
x2 = m_xcliphi;
|
2007-12-01 03:42:52 +00:00
|
|
|
WHEN_INSIDE;
|
2007-08-08 20:51:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-03-23 19:54:15 +00:00
|
|
|
if( ( (x2 + x1) / 2 <= m_xcliphi ) && ( (x2 + x1) / 2 >= m_xcliplo ) \
|
|
|
|
&& ( (y2 + y1) / 2 <= m_ycliphi ) && ( (y2 + y1) / 2 >= m_ycliplo ) )
|
2007-08-08 20:51:08 +00:00
|
|
|
{
|
2007-12-01 03:42:52 +00:00
|
|
|
return false;
|
2007-08-08 20:51:08 +00:00
|
|
|
}
|
|
|
|
else
|
2007-12-01 03:42:52 +00:00
|
|
|
return true;
|
2007-06-05 12:10:51 +00:00
|
|
|
}
|