/* * This program source code file is part of KiCad, a free EDA CAD application. * * Copyright (C) 2021-2024 KiCad Developers. * * 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 */ #include #include #include #include #include #include #include #include #include #include #include #include #include /* Library parity test. Errors generated: - DRCE_LIB_FOOTPRINT_ISSUES - DRCE_LIB_FOOTPRINT_MISMATCH */ class DRC_TEST_PROVIDER_LIBRARY_PARITY : public DRC_TEST_PROVIDER { public: DRC_TEST_PROVIDER_LIBRARY_PARITY() { m_isRuleDriven = false; } virtual ~DRC_TEST_PROVIDER_LIBRARY_PARITY() { } virtual bool Run() override; virtual const wxString GetName() const override { return wxT( "library_parity" ); }; virtual const wxString GetDescription() const override { return wxT( "Performs board footprint vs library integity checks" ); } }; // // The TEST*() macros have two modes: // In "Report" mode (aReporter != nullptr) all properties are checked and reported on. // In "DRC" mode (aReporter == nulltpr) properties are only checked until a difference is found. // #define TEST( a, b, msg ) \ do { \ if( a != b ) \ { \ diff = true; \ \ if( aReporter && wxString( msg ).length() ) \ aReporter->Report( msg ); \ } \ \ if( diff && !aReporter ) \ return diff; \ } while (0) #define EPSILON 1 #define TEST_PT( a, b, msg ) \ do { \ if( abs( a.x - b.x ) > EPSILON \ || abs( a.y - b.y ) > EPSILON ) \ { \ diff = true; \ \ if( aReporter && wxString( msg ).length() ) \ aReporter->Report( msg ); \ } \ \ if( diff && !aReporter ) \ return diff; \ } while (0) #define EPSILON_D 0.000001 #define TEST_D( a, b, msg ) \ do { \ if( abs( a - b ) > EPSILON_D ) \ { \ diff = true; \ \ if( aReporter && wxString( msg ).length() ) \ aReporter->Report( msg ); \ } \ \ if( diff && !aReporter ) \ return diff; \ } while (0) #define ITEM_DESC( item ) ( item )->GetItemDescription( &g_unitsProvider ) #define PAD_DESC( pad ) wxString::Format( _( "Pad %s" ), ( pad )->GetNumber() ) UNITS_PROVIDER g_unitsProvider( pcbIUScale, EDA_UNITS::MILLIMETRES ); bool primitiveNeedsUpdate( const std::shared_ptr& a, const std::shared_ptr& b ) { REPORTER* aReporter = nullptr; bool diff = false; TEST( a->GetShape(), b->GetShape(), "" ); switch( a->GetShape() ) { case SHAPE_T::RECTANGLE: { BOX2I aRect( a->GetStart(), a->GetEnd() - a->GetStart() ); BOX2I bRect( b->GetStart(), b->GetEnd() - b->GetStart() ); aRect.Normalize(); bRect.Normalize(); TEST_PT( aRect.GetOrigin(), bRect.GetOrigin(), "" ); TEST_PT( aRect.GetEnd(), bRect.GetEnd(), "" ); break; } case SHAPE_T::SEGMENT: case SHAPE_T::CIRCLE: TEST_PT( a->GetStart(), b->GetStart(), "" ); TEST_PT( a->GetEnd(), b->GetEnd(), "" ); break; case SHAPE_T::ARC: TEST_PT( a->GetStart(), b->GetStart(), "" ); TEST_PT( a->GetEnd(), b->GetEnd(), "" ); // Arc center is calculated and so may have round-off errors when parents are // differentially rotated. if( ( a->GetCenter() - b->GetCenter() ).EuclideanNorm() > pcbIUScale.mmToIU( 0.0005 ) ) return true; break; case SHAPE_T::BEZIER: TEST_PT( a->GetStart(), b->GetStart(), "" ); TEST_PT( a->GetEnd(), b->GetEnd(), "" ); TEST_PT( a->GetBezierC1(), b->GetBezierC1(), "" ); TEST_PT( a->GetBezierC2(), b->GetBezierC2(), "" ); break; case SHAPE_T::POLY: TEST( a->GetPolyShape().TotalVertices(), b->GetPolyShape().TotalVertices(), "" ); for( int ii = 0; ii < a->GetPolyShape().TotalVertices(); ++ii ) TEST_PT( a->GetPolyShape().CVertex( ii ), b->GetPolyShape().CVertex( ii ), "" ); break; default: UNIMPLEMENTED_FOR( a->SHAPE_T_asString() ); } TEST( a->GetStroke(), b->GetStroke(), "" ); TEST( a->IsFilled(), b->IsFilled(), "" ); return diff; } bool padHasOverrides( const PAD* a, const PAD* b, REPORTER* aReporter ) { bool diff = false; TEST( a->GetLocalClearance(), b->GetLocalClearance(), wxString::Format( _( "%s has clearance override." ), PAD_DESC( a ) ) ); TEST( a->GetLocalSolderMaskMargin(), b->GetLocalSolderMaskMargin(), wxString::Format( _( "%s has solder mask expansion override." ), PAD_DESC( a ) ) ); TEST( a->GetLocalSolderPasteMargin(), b->GetLocalSolderPasteMargin(), wxString::Format( _( "%s has solder paste clearance override." ), PAD_DESC( a ) ) ); TEST_D( a->GetLocalSolderPasteMarginRatio(), b->GetLocalSolderPasteMarginRatio(), wxString::Format( _( "%s has solder paste clearance override." ), PAD_DESC( a ) ) ); TEST( a->GetZoneConnection(), b->GetZoneConnection(), wxString::Format( _( "%s has zone connection override." ), PAD_DESC( a ) ) ); TEST( a->GetThermalGap(), b->GetThermalGap(), wxString::Format( _( "%s has thermal relief gap override." ), PAD_DESC( a ) ) ); TEST( a->GetThermalSpokeWidth(), b->GetThermalSpokeWidth(), wxString::Format( _( "%s has thermal relief spoke width override." ), PAD_DESC( a ) ) ); TEST_D( a->GetThermalSpokeAngle().AsDegrees(), b->GetThermalSpokeAngle().AsDegrees(), wxString::Format( _( "%s has thermal relief spoke angle override." ), PAD_DESC( a ) ) ); TEST( a->GetCustomShapeInZoneOpt(), b->GetCustomShapeInZoneOpt(), wxString::Format( _( "%s has zone knockout setting override." ), PAD_DESC( a ) ) ); return diff; } bool padNeedsUpdate( const PAD* a, const PAD* b, REPORTER* aReporter ) { bool diff = false; TEST( a->GetPadToDieLength(), b->GetPadToDieLength(), wxString::Format( _( "%s pad to die length differs." ), PAD_DESC( a ) ) ); TEST_PT( a->GetFPRelativePosition(), b->GetFPRelativePosition(), wxString::Format( _( "%s position differs." ), PAD_DESC( a ) ) ); TEST( a->GetNumber(), b->GetNumber(), wxString::Format( _( "%s has different numbers." ), PAD_DESC( a ) ) ); // These are assigned from the schematic and not from the library // TEST( a->GetPinFunction(), b->GetPinFunction() ); // TEST( a->GetPinType(), b->GetPinType() ); bool layerSettingsDiffer = a->GetRemoveUnconnected() != b->GetRemoveUnconnected(); // NB: KeepTopBottom is undefined if RemoveUnconnected is NOT set. if( a->GetRemoveUnconnected() ) layerSettingsDiffer |= a->GetKeepTopBottom() != b->GetKeepTopBottom(); // Trim layersets to the current board before comparing LSET enabledLayers = a->GetBoard()->GetEnabledLayers(); LSET aLayers = a->GetLayerSet() & enabledLayers; LSET bLayers = b->GetLayerSet() & enabledLayers; if( layerSettingsDiffer || aLayers != bLayers ) { diff = true; if( aReporter ) aReporter->Report( wxString::Format( _( "%s layers differ." ), PAD_DESC( a ) ) ); else return true; } TEST( a->GetShape(), b->GetShape(), wxString::Format( _( "%s pad shape type differs." ), PAD_DESC( a ) ) ); TEST( a->GetAttribute(), b->GetAttribute(), wxString::Format( _( "%s pad type differs." ), PAD_DESC( a ) ) ); TEST( a->GetProperty(), b->GetProperty(), wxString::Format( _( "%s fabrication property differs." ), PAD_DESC( a ) ) ); // The pad orientation, for historical reasons is the pad rotation + parent rotation. TEST_D( ( a->GetOrientation() - a->GetParentFootprint()->GetOrientation() ).Normalize().AsDegrees(), ( b->GetOrientation() - b->GetParentFootprint()->GetOrientation() ).Normalize().AsDegrees(), wxString::Format( _( "%s orientation differs." ), PAD_DESC( a ) ) ); TEST( a->GetSize(), b->GetSize(), wxString::Format( _( "%s size differs." ), PAD_DESC( a ) ) ); TEST( a->GetDelta(), b->GetDelta(), wxString::Format( _( "%s trapezoid delta differs." ), PAD_DESC( a ) ) ); if( a->GetRoundRectCornerRadius() != b->GetRoundRectCornerRadius() || a->GetRoundRectRadiusRatio() != b->GetRoundRectRadiusRatio() ) { diff = true; if( aReporter ) aReporter->Report( wxString::Format( _( "%s rounded corners differ." ), PAD_DESC( a ) ) ); else return true; } if( a->GetChamferRectRatio() != b->GetChamferRectRatio() || a->GetChamferPositions() != b->GetChamferPositions() ) { diff = true; if( aReporter ) aReporter->Report( wxString::Format( _( "%s chamfered corners differ." ), PAD_DESC( a ) ) ); else return true; } TEST_PT( a->GetOffset(), b->GetOffset(), wxString::Format( _( "%s shape offset from hole differs." ), PAD_DESC( a ) ) ); TEST( a->GetDrillShape(), b->GetDrillShape(), wxString::Format( _( "%s drill shape differs." ), PAD_DESC( a ) ) ); TEST( a->GetDrillSize(), b->GetDrillSize(), wxString::Format( _( "%s drill size differs." ), PAD_DESC( a ) ) ); // Clearance and zone connection overrides are as likely to be set at the board level as in // the library. // // If we ignore them and someone *does* change one of them in the library, then stale // footprints won't be caught. // // On the other hand, if we report them then boards that override at the board level are // going to be VERY noisy. // // So we just do it when we have a reporter. if( aReporter && padHasOverrides( a, b, aReporter ) ) diff = true; bool primitivesDiffer = false; if( a->GetPrimitives().size() != b->GetPrimitives().size() ) { primitivesDiffer = true; } else { for( size_t ii = 0; ii < a->GetPrimitives().size(); ++ii ) { if( primitiveNeedsUpdate( a->GetPrimitives()[ii], b->GetPrimitives()[ii] ) ) { primitivesDiffer = true; break; } } } if( primitivesDiffer ) { diff = true; if( aReporter ) aReporter->Report( wxString::Format( _( "%s shape primitives differ." ), PAD_DESC( a ) ) ); else return true; } return diff; } bool shapeNeedsUpdate( const PCB_SHAPE& curr_shape, const PCB_SHAPE& ref_shape ) { // curr_shape and ref_shape are expected to be normalized, for a more reliable test. REPORTER* aReporter = nullptr; bool diff = false; TEST( curr_shape.GetShape(), ref_shape.GetShape(), "" ); switch( curr_shape.GetShape() ) { case SHAPE_T::RECTANGLE: { BOX2I aRect( curr_shape.GetStart(), curr_shape.GetEnd() - curr_shape.GetStart() ); BOX2I bRect( ref_shape.GetStart(), ref_shape.GetEnd() - ref_shape.GetStart() ); aRect.Normalize(); bRect.Normalize(); TEST_PT( aRect.GetOrigin(), bRect.GetOrigin(), "" ); TEST_PT( aRect.GetEnd(), bRect.GetEnd(), "" ); break; } case SHAPE_T::SEGMENT: case SHAPE_T::CIRCLE: TEST_PT( curr_shape.GetStart(), ref_shape.GetStart(), "" ); TEST_PT( curr_shape.GetEnd(), ref_shape.GetEnd(), "" ); break; case SHAPE_T::ARC: TEST_PT( curr_shape.GetStart(), ref_shape.GetStart(), "" ); TEST_PT( curr_shape.GetEnd(), ref_shape.GetEnd(), "" ); // Arc center is calculated and so may have round-off errors when parents are // differentially rotated. if( ( curr_shape.GetCenter() - ref_shape.GetCenter() ).EuclideanNorm() > pcbIUScale.mmToIU( 0.0005 ) ) return true; break; case SHAPE_T::BEZIER: TEST_PT( curr_shape.GetStart(), ref_shape.GetStart(), "" ); TEST_PT( curr_shape.GetEnd(), ref_shape.GetEnd(), "" ); TEST_PT( curr_shape.GetBezierC1(), ref_shape.GetBezierC1(), "" ); TEST_PT( curr_shape.GetBezierC2(), ref_shape.GetBezierC2(), "" ); break; case SHAPE_T::POLY: TEST( curr_shape.GetPolyShape().TotalVertices(), ref_shape.GetPolyShape().TotalVertices(), "" ); for( int ii = 0; ii < curr_shape.GetPolyShape().TotalVertices(); ++ii ) TEST_PT( curr_shape.GetPolyShape().CVertex( ii ), ref_shape.GetPolyShape().CVertex( ii ), "" ); break; default: UNIMPLEMENTED_FOR( curr_shape.SHAPE_T_asString() ); } if( curr_shape.IsOnCopperLayer() ) TEST( curr_shape.GetStroke(), ref_shape.GetStroke(), "" ); TEST( curr_shape.IsFilled(), ref_shape.IsFilled(), "" ); TEST( curr_shape.GetLayer(), ref_shape.GetLayer(), "" ); return diff; } bool zoneNeedsUpdate( const ZONE* a, const ZONE* b, REPORTER* aReporter ) { bool diff = false; TEST( a->GetCornerSmoothingType(), b->GetCornerSmoothingType(), wxString::Format( _( "%s corner smoothing setting differs." ), ITEM_DESC( a ) ) ); TEST( a->GetCornerRadius(), b->GetCornerRadius(), wxString::Format( _( "%s corner smoothing radius differs." ), ITEM_DESC( a ) ) ); TEST( a->GetZoneName(), b->GetZoneName(), wxString::Format( _( "%s name differs." ), ITEM_DESC( a ) ) ); TEST( a->GetAssignedPriority(), b->GetAssignedPriority(), wxString::Format( _( "%s priority differs." ), ITEM_DESC( a ) ) ); TEST( a->GetIsRuleArea(), b->GetIsRuleArea(), wxString::Format( _( "%s keep-out property differs." ), ITEM_DESC( a ) ) ); TEST( a->GetDoNotAllowCopperPour(), b->GetDoNotAllowCopperPour(), wxString::Format( _( "%s keep out copper fill setting differs." ), ITEM_DESC( a ) ) ); TEST( a->GetDoNotAllowFootprints(), b->GetDoNotAllowFootprints(), wxString::Format( _( "%s keep out footprints setting differs." ), ITEM_DESC( a ) ) ); TEST( a->GetDoNotAllowPads(), b->GetDoNotAllowPads(), wxString::Format( _( "%s keep out pads setting differs." ), ITEM_DESC( a ) ) ); TEST( a->GetDoNotAllowTracks(), b->GetDoNotAllowTracks(), wxString::Format( _( "%s keep out tracks setting differs." ), ITEM_DESC( a ) ) ); TEST( a->GetDoNotAllowVias(), b->GetDoNotAllowVias(), wxString::Format( _( "%s keep out vias setting differs." ), ITEM_DESC( a ) ) ); TEST( a->GetLayerSet(), b->GetLayerSet(), wxString::Format( _( "%s layers differ." ), ITEM_DESC( a ) ) ); TEST( a->GetPadConnection(), b->GetPadConnection(), wxString::Format( _( "%s pad connection property differs." ), ITEM_DESC( a ) ) ); TEST( a->GetLocalClearance(), b->GetLocalClearance(), wxString::Format( _( "%s local clearance differs." ), ITEM_DESC( a ) ) ); TEST( a->GetThermalReliefGap(), b->GetThermalReliefGap(), wxString::Format( _( "%s thermal relief gap differs." ), ITEM_DESC( a ) ) ); TEST( a->GetThermalReliefSpokeWidth(), b->GetThermalReliefSpokeWidth(), wxString::Format( _( "%s thermal relief spoke width differs." ), ITEM_DESC( a ) ) ); TEST( a->GetMinThickness(), b->GetMinThickness(), wxString::Format( _( "%s min thickness differs." ), ITEM_DESC( a ) ) ); TEST( a->GetIslandRemovalMode(), b->GetIslandRemovalMode(), wxString::Format( _( "%s remove islands setting differs." ), ITEM_DESC( a ) ) ); TEST( a->GetMinIslandArea(), b->GetMinIslandArea(), wxString::Format( _( "%s minimum island size setting differs." ), ITEM_DESC( a ) ) ); TEST( a->GetFillMode(), b->GetFillMode(), wxString::Format( _( "%s fill type differs." ), ITEM_DESC( a ) ) ); TEST( a->GetHatchThickness(), b->GetHatchThickness(), wxString::Format( _( "%s hatch width differs." ), ITEM_DESC( a ) ) ); TEST( a->GetHatchGap(), b->GetHatchGap(), wxString::Format( _( "%s hatch gap differs." ), ITEM_DESC( a ) ) ); TEST_D( a->GetHatchOrientation().AsDegrees(), b->GetHatchOrientation().AsDegrees(), wxString::Format( _( "%s hatch orientation differs." ), ITEM_DESC( a ) ) ); TEST( a->GetHatchSmoothingLevel(), b->GetHatchSmoothingLevel(), wxString::Format( _( "%s hatch smoothing level differs." ), ITEM_DESC( a ) ) ); TEST( a->GetHatchSmoothingValue(), b->GetHatchSmoothingValue(), wxString::Format( _( "%s hatch smoothing amount differs." ), ITEM_DESC( a ) ) ); TEST( a->GetHatchHoleMinArea(), b->GetHatchHoleMinArea(), wxString::Format( _( "%s minimum hatch hole setting differs." ), ITEM_DESC( a ) ) ); // This is just a display property // TEST( a->GetHatchBorderAlgorithm(), b->GetHatchBorderAlgorithm() ); TEST( a->Outline()->TotalVertices(), b->Outline()->TotalVertices(), wxString::Format( _( "%s outline corner count differs." ), ITEM_DESC( a ) ) ); bool cornersDiffer = false; for( int ii = 0; ii < a->Outline()->TotalVertices(); ++ii ) { if( a->Outline()->CVertex( ii ) != b->Outline()->CVertex( ii ) ) { diff = true; cornersDiffer = true; break; } } if( cornersDiffer && aReporter ) aReporter->Report( wxString::Format( _( "%s corners differ." ), ITEM_DESC( a ) ) ); return diff; } bool FOOTPRINT::FootprintNeedsUpdate( const FOOTPRINT* aLibFP, REPORTER* aReporter ) { UNITS_PROVIDER unitsProvider( pcbIUScale, EDA_UNITS::MILLIMETRES ); wxASSERT( aLibFP ); bool diff = false; // To avoid issues when comparing the footprint on board and the footprint in library // use the footprint from lib flipped, rotated and at same position as this. // And using the footprint from lib with same changes as this minimize the issues // due to rounding and shape modifications std::unique_ptr temp( static_cast( aLibFP->Clone() ) ); temp->SetParentGroup( nullptr ); temp->SetParent( GetBoard() ); // Needed to know the copper layer count; if( IsFlipped() != temp->IsFlipped() ) temp->Flip( { 0, 0 }, false ); if( GetOrientation() != temp->GetOrientation() ) temp->SetOrientation( GetOrientation() ); if( GetPosition() != temp->GetPosition() ) temp->SetPosition( GetPosition() ); for( BOARD_ITEM* item : temp->GraphicalItems() ) item->Normalize(); // This temporary footprint must not have a parent when it goes out of scope because it // must not trigger the IncrementTimestamp call in ~FOOTPRINT. temp->SetParent( nullptr ); aLibFP = temp.get(); TEST( GetLibDescription(), aLibFP->GetLibDescription(), _( "Footprint descriptions differ." ) ); TEST( GetKeywords(), aLibFP->GetKeywords(), _( "Footprint keywords differ." ) ); #define TEST_ATTR( a, b, attr, msg ) TEST( ( a & attr ), ( b & attr ), msg ) TEST_ATTR( GetAttributes(), aLibFP->GetAttributes(), (FP_THROUGH_HOLE | FP_SMD), _( "Footprint types differ." ) ); TEST_ATTR( GetAttributes(), aLibFP->GetAttributes(), FP_ALLOW_SOLDERMASK_BRIDGES, wxString::Format( _( "'%s' settings differ." ), _( "Allow bridged solder mask apertures between pads" ) ) ); TEST_ATTR( GetAttributes(), aLibFP->GetAttributes(), FP_BOARD_ONLY, wxString::Format( _( "'%s' settings differ." ), _( "Not in schematic" ) ) ); TEST_ATTR( GetAttributes(), aLibFP->GetAttributes(), FP_EXCLUDE_FROM_POS_FILES, wxString::Format( _( "'%s' settings differ." ), _( "Exclude from position files" ) ) ); // this test is skipped: EXCLUDE_FROM_BOM attribute is related to a given design, // not to a lib footprint. EXCLUDE_FROM_BOM must be tested only in Schematic Parity #if 0 TEST_ATTR( GetAttributes(), aLibFP->GetAttributes(), FP_EXCLUDE_FROM_BOM, wxString::Format( _( "'%s' settings differ." ), _( "Exclude from bill of materials" ) ) ); #endif TEST_ATTR( GetAttributes(), aLibFP->GetAttributes(), FP_ALLOW_MISSING_COURTYARD, wxString::Format( _( "'%s' settings differ." ), _( "Exempt From Courtyard Requirement" ) ) ); // this test is skipped: Do No Place attribute is related to a given design, // not to a lib footprint. DNP must be tested only in Schematic Parity #if 0 TEST_ATTR( GetAttributes(), aLibFP->GetAttributes(), FP_DNP, wxString::Format( _( "'%s' settings differ." ), _( "Do not populate" ) ) ); #endif // Clearance and zone connection overrides are as likely to be set at the board level as in // the library. // // If we ignore them and someone *does* change one of them in the library, then stale // footprints won't be caught. // // On the other hand, if we report them then boards that override at the board level are // going to be VERY noisy. // // For now we report them if there's a reporter, but we DON'T generate DRC errors on them. if( aReporter ) { TEST( GetLocalClearance(), aLibFP->GetLocalClearance(), _( "Pad clearance overridden." ) ); TEST( GetLocalSolderMaskMargin(), aLibFP->GetLocalSolderMaskMargin(), _( "Solder mask expansion overridden." ) ); TEST( GetLocalSolderPasteMargin(), aLibFP->GetLocalSolderPasteMargin(), _( "Solder paste absolute clearance overridden." ) ); TEST_D( GetLocalSolderPasteMarginRatio(), aLibFP->GetLocalSolderPasteMarginRatio(), _( "Solder paste relative clearance overridden." ) ); TEST( GetZoneConnection(), aLibFP->GetZoneConnection(), _( "Zone connection overridden." ) ); } TEST( GetNetTiePadGroups().size(), aLibFP->GetNetTiePadGroups().size(), _( "Net tie pad groups differ." ) ); for( size_t ii = 0; ii < GetNetTiePadGroups().size(); ++ii ) { TEST( GetNetTiePadGroups()[ii], aLibFP->GetNetTiePadGroups()[ii], _( "Net tie pad groups differ." ) ); } #define REPORT( msg ) { if( aReporter ) aReporter->Report( msg ); } #define CHECKPOINT { if( diff && !aReporter ) return diff; } // Text items are really problematic. We don't want to test the reference, but after that // it gets messy. // // What about the value? Depends on whether or not it's a singleton part. // // And what about other texts? They might be added only to instances on the board, or even // changed for instances on the board. Or they might want to be tested for equality. // // Currently we punt and ignore all the text items. // Drawings and pads are also somewhat problematic as there's no guarantee that they'll be // in the same order in the two footprints. Rather than building some sophisticated hashing // algorithm we use the footprint sorting functions to attempt to sort them in the same // order. // However FOOTPRINT::cmp_drawings uses PCB_SHAPE coordinates and other infos, so we have // already normalized graphic items in model footprint from library, so we need to normalize // graphic items in the footprint to test (*this). So normalize them using a copy of this FOOTPRINT dummy( *this ); dummy.SetParentGroup( nullptr ); dummy.SetParent( nullptr ); for( BOARD_ITEM* item : dummy.GraphicalItems() ) item->Normalize(); std::set aShapes; std::copy_if( dummy.GraphicalItems().begin(), dummy.GraphicalItems().end(), std::inserter( aShapes, aShapes.begin() ), []( BOARD_ITEM* item ) { return item->Type() == PCB_SHAPE_T; } ); std::set bShapes; std::copy_if( aLibFP->GraphicalItems().begin(), aLibFP->GraphicalItems().end(), std::inserter( bShapes, bShapes.begin() ), []( BOARD_ITEM* item ) { return item->Type() == PCB_SHAPE_T; } ); if( aShapes.size() != bShapes.size() ) { diff = true; REPORT( _( "Graphic item count differs." ) ); } else { for( auto aIt = aShapes.begin(), bIt = bShapes.begin(); aIt != aShapes.end(); aIt++, bIt++ ) { // aShapes and bShapes are the tested footprint PCB_SHAPE and the model PCB_SHAPE. // These shapes are already normalized. PCB_SHAPE* curr_shape = static_cast( *aIt ); PCB_SHAPE* test_shape = static_cast( *bIt ); if( shapeNeedsUpdate( *curr_shape, *test_shape ) ) { diff = true; REPORT( wxString::Format( _( "%s differs." ), ITEM_DESC( *aIt ) ) ); } } } CHECKPOINT; std::set aPads( Pads().begin(), Pads().end() ); std::set bPads( aLibFP->Pads().begin(), aLibFP->Pads().end() ); if( aPads.size() != bPads.size() ) { diff = true; REPORT( _( "Pad count differs." ) ); } else { for( auto aIt = aPads.begin(), bIt = bPads.begin(); aIt != aPads.end(); aIt++, bIt++ ) { if( padNeedsUpdate( *aIt, *bIt, aReporter ) ) diff = true; else if( aReporter && padHasOverrides( *aIt, *bIt, aReporter ) ) diff = true; } } CHECKPOINT; std::set aZones( Zones().begin(), Zones().end() ); std::set bZones( aLibFP->Zones().begin(), aLibFP->Zones().end() ); if( aZones.size() != bZones.size() ) { diff = true; REPORT( _( "Rule area count differs." ) ); } else { for( auto aIt = aZones.begin(), bIt = bZones.begin(); aIt != aZones.end(); aIt++, bIt++ ) diff |= zoneNeedsUpdate( *aIt, *bIt, aReporter ); } return diff; } bool DRC_TEST_PROVIDER_LIBRARY_PARITY::Run() { BOARD* board = m_drcEngine->GetBoard(); PROJECT* project = board->GetProject(); if( !project ) { reportAux( _( "No project loaded, skipping library parity tests." ) ); return true; // Continue with other tests } if( !reportPhase( _( "Loading footprint library table..." ) ) ) return false; // DRC cancelled std::map> libFootprintCache; FP_LIB_TABLE* libTable = PROJECT_PCB::PcbFootprintLibs( project ); wxString msg; int ii = 0; const int progressDelta = 250; if( !reportPhase( _( "Checking board footprints against library..." ) ) ) return false; for( FOOTPRINT* footprint : board->Footprints() ) { if( m_drcEngine->IsErrorLimitExceeded( DRCE_LIB_FOOTPRINT_ISSUES ) && m_drcEngine->IsErrorLimitExceeded( DRCE_LIB_FOOTPRINT_MISMATCH ) ) { return true; // Continue with other tests } if( !reportProgress( ii++, (int) board->Footprints().size(), progressDelta ) ) return false; // DRC cancelled LIB_ID fpID = footprint->GetFPID(); wxString libName = fpID.GetLibNickname(); wxString fpName = fpID.GetLibItemName(); const LIB_TABLE_ROW* libTableRow = nullptr; if( libName.IsEmpty() ) { // Not much we can do here continue; } try { libTableRow = libTable->FindRow( libName ); } catch( const IO_ERROR& ) { } if( !libTableRow ) { if( !m_drcEngine->IsErrorLimitExceeded( DRCE_LIB_FOOTPRINT_ISSUES ) ) { std::shared_ptr drcItem = DRC_ITEM::Create( DRCE_LIB_FOOTPRINT_ISSUES ); msg.Printf( _( "The current configuration does not include the library '%s'." ), libName ); drcItem->SetErrorMessage( msg ); drcItem->SetItems( footprint ); reportViolation( drcItem, footprint->GetCenter(), UNDEFINED_LAYER ); } continue; } else if( !libTable->HasLibrary( libName, true ) ) { if( !m_drcEngine->IsErrorLimitExceeded( DRCE_LIB_FOOTPRINT_ISSUES ) ) { std::shared_ptr drcItem = DRC_ITEM::Create( DRCE_LIB_FOOTPRINT_ISSUES ); msg.Printf( _( "The library '%s' is not enabled in the current configuration." ), libName ); drcItem->SetErrorMessage( msg ); drcItem->SetItems( footprint ); reportViolation( drcItem, footprint->GetCenter(), UNDEFINED_LAYER ); } continue; } auto cacheIt = libFootprintCache.find( fpID ); std::shared_ptr libFootprint; if( cacheIt != libFootprintCache.end() ) { libFootprint = cacheIt->second; } else { try { libFootprint.reset( libTable->FootprintLoad( libName, fpName, true ) ); if( libFootprint ) libFootprintCache[ fpID ] = libFootprint; } catch( const IO_ERROR& ) { } } if( !libFootprint ) { if( !m_drcEngine->IsErrorLimitExceeded( DRCE_LIB_FOOTPRINT_ISSUES ) ) { std::shared_ptr drcItem = DRC_ITEM::Create( DRCE_LIB_FOOTPRINT_ISSUES ); msg.Printf( _( "Footprint '%s' not found in library '%s'." ), fpName, libName ); drcItem->SetErrorMessage( msg ); drcItem->SetItems( footprint ); reportViolation( drcItem, footprint->GetCenter(), UNDEFINED_LAYER ); } } else if( footprint->FootprintNeedsUpdate( libFootprint.get() ) ) { if( !m_drcEngine->IsErrorLimitExceeded( DRCE_LIB_FOOTPRINT_MISMATCH ) ) { std::shared_ptr drcItem = DRC_ITEM::Create( DRCE_LIB_FOOTPRINT_MISMATCH ); msg.Printf( _( "Footprint '%s' does not match copy in library '%s'." ), fpName, libName ); drcItem->SetErrorMessage( msg ); drcItem->SetItems( footprint ); reportViolation( drcItem, footprint->GetCenter(), UNDEFINED_LAYER ); } } } return true; } namespace detail { static DRC_REGISTER_TEST_PROVIDER dummy; }