Check for collisions on common layers.

Fixes https://gitlab.com/kicad/code/kicad/issues/13438
This commit is contained in:
Jeff Young 2023-01-08 20:31:55 +00:00
parent d5ad962b7c
commit 6ed90134c7
1 changed files with 28 additions and 17 deletions

View File

@ -456,12 +456,10 @@ bool collidesWithArea( BOARD_ITEM* aItem, PCB_EXPR_CONTEXT* aCtx, ZONE* aArea )
{ {
if( aCtx->HasErrorCallback() ) if( aCtx->HasErrorCallback() )
aCtx->ReportError( _( "Footprint has no front courtyard." ) ); aCtx->ReportError( _( "Footprint has no front courtyard." ) );
return false;
} }
else else if( areaOutline.Collide( &courtyard.Outline( 0 ) ) )
{ {
return areaOutline.Collide( &courtyard.Outline( 0 ) ); return true;
} }
} }
@ -473,12 +471,10 @@ bool collidesWithArea( BOARD_ITEM* aItem, PCB_EXPR_CONTEXT* aCtx, ZONE* aArea )
{ {
if( aCtx->HasErrorCallback() ) if( aCtx->HasErrorCallback() )
aCtx->ReportError( _( "Footprint has no back courtyard." ) ); aCtx->ReportError( _( "Footprint has no back courtyard." ) );
return false;
} }
else else if( areaOutline.Collide( &courtyard.Outline( 0 ) ) )
{ {
return areaOutline.Collide( &courtyard.Outline( 0 ) ); return true;
} }
} }
@ -619,7 +615,7 @@ static void intersectsAreaFunc( LIBEVAL::CONTEXT* aCtx, void* self )
[item, arg, context]() -> double [item, arg, context]() -> double
{ {
BOARD* board = item->GetBoard(); BOARD* board = item->GetBoard();
PCB_LAYER_ID layer = context->GetLayer(); PCB_LAYER_ID aLayer = context->GetLayer();
BOX2I itemBBox = item->GetBoundingBox(); BOX2I itemBBox = item->GetBoundingBox();
if( searchAreas( board, arg->AsString(), context, if( searchAreas( board, arg->AsString(), context,
@ -628,25 +624,40 @@ static void intersectsAreaFunc( LIBEVAL::CONTEXT* aCtx, void* self )
if( !aArea || aArea == item || aArea->GetParent() == item ) if( !aArea || aArea == item || aArea->GetParent() == item )
return false; return false;
if( !( aArea->GetLayerSet() & item->GetLayerSet() ).any() ) LSET commonLayers = aArea->GetLayerSet() & item->GetLayerSet();
if( !commonLayers.any() )
return false; return false;
if( !aArea->GetBoundingBox().Intersects( itemBBox ) ) if( !aArea->GetBoundingBox().Intersects( itemBBox ) )
return false; return false;
std::unique_lock<std::mutex> cacheLock( board->m_CachesMutex ); std::unique_lock<std::mutex> cacheLock( board->m_CachesMutex );
PTR_PTR_LAYER_CACHE_KEY key = { aArea, item, layer }; LSET testLayers;
auto i = board->m_IntersectsAreaCache.find( key ); if( aLayer != UNDEFINED_LAYER )
testLayers.set( aLayer );
else
testLayers = commonLayers;
if( i != board->m_IntersectsAreaCache.end() ) for( PCB_LAYER_ID layer : testLayers.UIOrder() )
return i->second; {
PTR_PTR_LAYER_CACHE_KEY key = { aArea, item, layer };
bool collides = collidesWithArea( item, context, aArea ); auto i = board->m_IntersectsAreaCache.find( key );
board->m_IntersectsAreaCache[ key ] = collides; if( i != board->m_IntersectsAreaCache.end() && i->second )
return true;
return collides; bool collides = collidesWithArea( item, context, aArea );
board->m_IntersectsAreaCache[ key ] = collides;
if( collides )
return true;
}
return false;
} ) ) } ) )
{ {
return 1.0; return 1.0;