Fix DRC performance with multi-layer keepout zones.

The main issue was a parameter mismatch which caused On^2 behaviour
for zone layers.

But there are several other performance optimizations here, along
with status bar updating for zones while running the dissallow test.

Fixes https://gitlab.com/kicad/code/kicad/issues/8521
This commit is contained in:
Jeff Young 2021-06-02 14:00:11 +01:00
parent b6de4da686
commit 00ed75b891
11 changed files with 116 additions and 56 deletions

View File

@ -736,7 +736,7 @@ void DRC_ENGINE::RunTests( EDA_UNITS aUnits, bool aReportAllTrackErrors, bool aT
m_board->m_CopperZoneRTrees[ zone ] = std::make_unique<DRC_RTREE>(); m_board->m_CopperZoneRTrees[ zone ] = std::make_unique<DRC_RTREE>();
for( int layer : zone->GetLayerSet().Seq() ) for( PCB_LAYER_ID layer : zone->GetLayerSet().Seq() )
{ {
if( IsCopperLayer( layer ) ) if( IsCopperLayer( layer ) )
m_board->m_CopperZoneRTrees[ zone ]->Insert( zone, layer ); m_board->m_CopperZoneRTrees[ zone ]->Insert( zone, layer );

View File

@ -81,16 +81,17 @@ public:
/** /**
* Function Insert() * Function Insert()
* Inserts an item into the tree. * Inserts an item into the tree on a particular layer with an optional worst clearance.
*/ */
void Insert( BOARD_ITEM* aItem, int aWorstClearance = 0, int aLayer = UNDEFINED_LAYER ) void Insert( BOARD_ITEM* aItem, PCB_LAYER_ID aLayer, int aWorstClearance = 0 )
{ {
std::vector<SHAPE*> subshapes; wxASSERT( aLayer != UNDEFINED_LAYER );
auto addLayer = if( aItem->Type() == PCB_FP_TEXT_T && !static_cast<FP_TEXT*>( aItem )->IsVisible() )
[&]( PCB_LAYER_ID layer ) return;
{
std::shared_ptr<SHAPE> shape = aItem->GetEffectiveShape( layer ); std::vector<SHAPE*> subshapes;
std::shared_ptr<SHAPE> shape = aItem->GetEffectiveShape( ToLAYER_ID( aLayer ) );
subshapes.clear(); subshapes.clear();
if( shape->HasIndexableSubshapes() ) if( shape->HasIndexableSubshapes() )
@ -106,36 +107,11 @@ public:
const int mmin[2] = { bbox.GetX(), bbox.GetY() }; const int mmin[2] = { bbox.GetX(), bbox.GetY() };
const int mmax[2] = { bbox.GetRight(), bbox.GetBottom() }; const int mmax[2] = { bbox.GetRight(), bbox.GetBottom() };
ITEM_WITH_SHAPE* itemShape = new ITEM_WITH_SHAPE( aItem, subshape, shape );
m_tree[layer]->Insert( mmin, mmax, new ITEM_WITH_SHAPE( aItem, subshape, m_tree[aLayer]->Insert( mmin, mmax, itemShape );
shape ) );
m_count++; m_count++;
} }
};
if( aItem->Type() == PCB_FP_TEXT_T && !static_cast<FP_TEXT*>( aItem )->IsVisible() )
return;
if( aLayer != UNDEFINED_LAYER )
{
addLayer( (PCB_LAYER_ID) aLayer );
}
else
{
LSET layers = aItem->GetLayerSet();
// Special-case pad holes which pierce all the copper layers
if( aItem->Type() == PCB_PAD_T )
{
PAD* pad = static_cast<PAD*>( aItem );
if( pad->GetDrillSizeX() > 0 && pad->GetDrillSizeY() > 0 )
layers |= LSET::AllCuMask();
}
for( int layer : layers.Seq() )
addLayer( (PCB_LAYER_ID) layer );
}
} }
/** /**
@ -261,7 +237,7 @@ public:
* position. * position.
*/ */
bool QueryColliding( EDA_RECT aBox, SHAPE* aRefShape, PCB_LAYER_ID aLayer, int aClearance, bool QueryColliding( EDA_RECT aBox, SHAPE* aRefShape, PCB_LAYER_ID aLayer, int aClearance,
int* aActual = nullptr, VECTOR2I* aPos = nullptr ) const int* aActual, VECTOR2I* aPos ) const
{ {
aBox.Inflate( aClearance ); aBox.Inflate( aClearance );
@ -308,6 +284,32 @@ public:
return false; return false;
} }
/**
* Quicker version of above that just reports a raw yes/no.
*/
bool QueryColliding( EDA_RECT aBox, SHAPE* aRefShape, PCB_LAYER_ID aLayer ) const
{
int min[2] = { aBox.GetX(), aBox.GetY() };
int max[2] = { aBox.GetRight(), aBox.GetBottom() };
bool collision = false;
auto visit =
[&]( ITEM_WITH_SHAPE* aItem ) -> bool
{
if( aRefShape->Collide( aItem->shape, 0 ) )
{
collision = true;
return false;
}
return true;
};
this->m_tree[aLayer]->Search( min, max, visit );
return collision;
}
typedef std::pair<PCB_LAYER_ID, PCB_LAYER_ID> LAYER_PAIR; typedef std::pair<PCB_LAYER_ID, PCB_LAYER_ID> LAYER_PAIR;
struct PAIR_INFO struct PAIR_INFO

View File

@ -165,7 +165,23 @@ bool DRC_TEST_PROVIDER_COPPER_CLEARANCE::Run()
if( !reportProgress( ii++, count, delta ) ) if( !reportProgress( ii++, count, delta ) )
return false; return false;
m_copperTree.Insert( item, m_largestClearance ); LSET layers = item->GetLayerSet();
// Special-case pad holes which pierce all the copper layers
if( item->Type() == PCB_PAD_T )
{
PAD* pad = static_cast<PAD*>( item );
if( pad->GetDrillSizeX() > 0 && pad->GetDrillSizeY() > 0 )
layers |= LSET::AllCuMask();
}
for( PCB_LAYER_ID layer : layers.Seq() )
{
if( IsCopperLayer( layer ) )
m_copperTree.Insert( item, layer, m_largestClearance );
}
return true; return true;
}; };

View File

@ -329,7 +329,12 @@ bool test::DRC_TEST_PROVIDER_DIFF_PAIR_COUPLING::Run()
auto addToTree = auto addToTree =
[&copperTree]( BOARD_ITEM *item ) -> bool [&copperTree]( BOARD_ITEM *item ) -> bool
{ {
copperTree.Insert( item ); for( PCB_LAYER_ID layer : item->GetLayerSet().Seq() )
{
if( IsCopperLayer( layer ) )
copperTree.Insert( item, layer );
}
return true; return true;
}; };

View File

@ -68,6 +68,12 @@ bool DRC_TEST_PROVIDER_DISALLOW::Run()
if( !reportPhase( _( "Checking keepouts & disallow constraints..." ) ) ) if( !reportPhase( _( "Checking keepouts & disallow constraints..." ) ) )
return false; // DRC cancelled return false; // DRC cancelled
// Zones can be expensive (particularly when multi-layer), so we run the progress bar on them
BOARD* board = m_drcEngine->GetBoard();
int boardZoneCount = board->Zones().size();
int delta = std::max( 1, boardZoneCount / board->GetCopperLayerCount() );
int ii = 0;
auto doCheckItem = auto doCheckItem =
[&]( BOARD_ITEM* item ) [&]( BOARD_ITEM* item )
{ {
@ -123,6 +129,12 @@ bool DRC_TEST_PROVIDER_DISALLOW::Run()
if( zone->GetIsRuleArea() ) if( zone->GetIsRuleArea() )
return true; return true;
if( item->Type() == PCB_ZONE_T )
{
if( !reportProgress( ii++, boardZoneCount, delta ) )
return false; // DRC cancelled
}
} }
item->ClearFlags( HOLE_PROXY ); item->ClearFlags( HOLE_PROXY );

View File

@ -208,7 +208,13 @@ bool DRC_TEST_PROVIDER_EDGE_CLEARANCE::Run()
forEachGeometryItem( s_allBasicItemsButZones, LSET::AllCuMask(), queryBoardGeometryItems ); forEachGeometryItem( s_allBasicItemsButZones, LSET::AllCuMask(), queryBoardGeometryItems );
for( const std::unique_ptr<PCB_SHAPE>& edge : edges ) for( const std::unique_ptr<PCB_SHAPE>& edge : edges )
edgesTree.Insert( edge.get(), m_largestClearance ); {
for( PCB_LAYER_ID layer : { Edge_Cuts, Margin } )
{
if( edge->IsOnLayer( layer ) )
edgesTree.Insert( edge.get(), layer, m_largestClearance );
}
}
wxString val; wxString val;
wxGetEnv( "WXTRACE", &val ); wxGetEnv( "WXTRACE", &val );

View File

@ -151,7 +151,7 @@ bool DRC_TEST_PROVIDER_HOLE_TO_HOLE::Run()
// We only care about drilled (ie: round) holes // We only care about drilled (ie: round) holes
if( pad->GetDrillSize().x && pad->GetDrillSize().x == pad->GetDrillSize().y ) if( pad->GetDrillSize().x && pad->GetDrillSize().x == pad->GetDrillSize().y )
m_holeTree.Insert( item, m_largestClearance, F_Cu ); m_holeTree.Insert( item, F_Cu, m_largestClearance );
} }
else if( item->Type() == PCB_VIA_T ) else if( item->Type() == PCB_VIA_T )
{ {
@ -159,7 +159,7 @@ bool DRC_TEST_PROVIDER_HOLE_TO_HOLE::Run()
// We only care about mechanically drilled (ie: non-laser) holes // We only care about mechanically drilled (ie: non-laser) holes
if( via->GetViaType() == VIATYPE::THROUGH ) if( via->GetViaType() == VIATYPE::THROUGH )
m_holeTree.Insert( item, m_largestClearance, F_Cu ); m_holeTree.Insert( item, F_Cu, m_largestClearance );
} }
return true; return true;

View File

@ -112,7 +112,12 @@ bool DRC_TEST_PROVIDER_SILK_CLEARANCE::Run()
auto addToSilkTree = auto addToSilkTree =
[&silkTree]( BOARD_ITEM* item ) -> bool [&silkTree]( BOARD_ITEM* item ) -> bool
{ {
silkTree.Insert( item ); for( PCB_LAYER_ID layer : { F_SilkS, B_SilkS } )
{
if( item->IsOnLayer( layer ) )
silkTree.Insert( item, layer );
}
return true; return true;
}; };
@ -129,7 +134,9 @@ bool DRC_TEST_PROVIDER_SILK_CLEARANCE::Run()
if( !reportProgress( ii++, targets, delta ) ) if( !reportProgress( ii++, targets, delta ) )
return false; return false;
targetTree.Insert( item ); for( PCB_LAYER_ID layer : item->GetLayerSet().Seq() )
targetTree.Insert( item, layer );
return true; return true;
}; };

View File

@ -104,14 +104,24 @@ bool DRC_TEST_PROVIDER_SILK_TO_MASK::Run()
auto addMaskToTree = auto addMaskToTree =
[&maskTree]( BOARD_ITEM *item ) -> bool [&maskTree]( BOARD_ITEM *item ) -> bool
{ {
maskTree.Insert( item ); for( PCB_LAYER_ID layer : { F_Mask, B_Mask } )
{
if( item->IsOnLayer( layer ) )
maskTree.Insert( item, layer );
}
return true; return true;
}; };
auto addSilkToTree = auto addSilkToTree =
[&silkTree]( BOARD_ITEM *item ) -> bool [&silkTree]( BOARD_ITEM *item ) -> bool
{ {
silkTree.Insert( item ); for( PCB_LAYER_ID layer : { F_SilkS, B_SilkS } )
{
if( item->IsOnLayer( layer ) )
silkTree.Insert( item, layer );
}
return true; return true;
}; };

View File

@ -540,11 +540,13 @@ static void insideArea( LIBEVAL::CONTEXT* aCtx, void* self )
DRC_RTREE* zoneRTree = board->m_CopperZoneRTrees[ zone ].get(); DRC_RTREE* zoneRTree = board->m_CopperZoneRTrees[ zone ].get();
std::vector<SHAPE*> shapes;
if( zoneRTree ) if( zoneRTree )
{ {
for( PCB_LAYER_ID layer : area->GetLayerSet().Seq() ) for( PCB_LAYER_ID layer : area->GetLayerSet().Seq() )
{ {
if( zoneRTree->QueryColliding( itemBBox, &areaOutline, layer, 0 ) ) if( zoneRTree->QueryColliding( itemBBox, &areaOutline, layer ) )
return true; return true;
} }
} }

View File

@ -275,7 +275,7 @@ void TRACKS_CLEANER::cleanup( bool aDeleteDuplicateVias, bool aDeleteNullSegment
for( TRACK* track : m_brd->Tracks() ) for( TRACK* track : m_brd->Tracks() )
{ {
track->ClearFlags( IS_DELETED | SKIP_STRUCT ); track->ClearFlags( IS_DELETED | SKIP_STRUCT );
rtree.Insert( track ); rtree.Insert( track, track->GetLayer() );
} }
std::set<BOARD_ITEM*> toRemove; std::set<BOARD_ITEM*> toRemove;