/* * This program source code file is part of KiCad, a free EDA CAD application. * * Copyright (C) 2019-2023 KiCad Developers, see AUTHORS.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 */ #include #include #include // helper function to sort pins by pin num static bool sort_by_pin_number( const LIB_PIN* ref, const LIB_PIN* tst ); /** * Check a lib symbol to find incorrect settings * Pins not on a valid grid * Pins duplicated * Conflict with pins at same location * Incorrect Power Symbols * illegal reference prefix (cannot ends by a digit or a '?') * @param aSymbol is the library symbol to check * @param aMessages is a room to store error messages * @param aGridForPins (in IU) is the grid to test pin positions ( >= 25 mils ) * should be 25, 50 or 100 mils (convered to IUs) * @param aUnitsProvider a frame to format coordinates in messages */ void CheckLibSymbol( LIB_SYMBOL* aSymbol, std::vector& aMessages, int aGridForPins, EDA_DRAW_FRAME* aUnitsProvider ) { if( !aSymbol ) return; wxString msg; // Test reference prefix validity: // if the symbol is saved in a library, the prefix should not ends by a digit or a '?' // but it is acceptable if the symbol is saved to aschematic wxString reference_base = aSymbol->GetReferenceField().GetText(); wxString illegal_end( wxT( "0123456789?" ) ); wxUniChar last_char = reference_base.Last(); if( illegal_end.Find( last_char ) != wxNOT_FOUND ) { msg.Printf( _( "Warning: reference prefix
prefix ending by '%s' can create" " issues if saved in a symbol library" ), illegal_end ); msg += wxT( "

" ); aMessages.push_back( msg ); } LIB_PINS pinList; aSymbol->GetPins( pinList ); // Test for duplicates: // Sort pins by pin num, so 2 duplicate pins // (pins with the same number) will be consecutive in list sort( pinList.begin(), pinList.end(), sort_by_pin_number ); // The minimal grid size allowed to place a pin is 25 mils // the best grid size is 50 mils, but 25 mils is still usable // this is because all aSymbols are using a 50 mils grid to place pins, and therefore // the wires must be on the 50 mils grid // So raise an error if a pin is not on a 25 (or bigger :50 or 100) mils grid const int min_grid_size = schIUScale.MilsToIU( 25 ); const int clamped_grid_size = ( aGridForPins < min_grid_size ) ? min_grid_size : aGridForPins; for( unsigned ii = 1; ii < pinList.size(); ii++ ) { LIB_PIN* pin = pinList[ii - 1]; LIB_PIN* next = pinList[ii]; if( pin->GetNumber() != next->GetNumber() ) continue; // Pins are not duplicated only if they are in different convert bodies // (but GetConvert() == 0 means commun to all convert bodies) if( pin->GetConvert() != 0 && next->GetConvert() != 0 ) { if( pin->GetConvert() != next->GetConvert() ) continue; } wxString pinName; wxString nextName; if( pin->GetName() != "~" && !pin->GetName().IsEmpty() ) pinName = " '" + pin->GetName() + "'"; if( next->GetName() != "~" && !next->GetName().IsEmpty() ) nextName = " '" + next->GetName() + "'"; if( aSymbol->HasConversion() && next->GetConvert() ) { if( pin->GetUnit() == 0 || next->GetUnit() == 0 ) { msg.Printf( _( "Duplicate pin %s %s at location (%.3f, %.3f)" " conflicts with pin %s%s at location (%.3f, %.3f)" " of converted." ), next->GetNumber(), nextName, aUnitsProvider->MessageTextFromValue( next->GetPosition().x ), aUnitsProvider->MessageTextFromValue( -next->GetPosition().y ), pin->GetNumber(), pin->GetName(), aUnitsProvider->MessageTextFromValue( pin->GetPosition().x ), aUnitsProvider->MessageTextFromValue( -pin->GetPosition().y ) ); } else { msg.Printf( _( "Duplicate pin %s %s at location (%.3f, %.3f)" " conflicts with pin %s%s at location (%.3f, %.3f)" " in units %s and %s of converted." ), next->GetNumber(), nextName, aUnitsProvider->MessageTextFromValue( next->GetPosition().x ), aUnitsProvider->MessageTextFromValue( -next->GetPosition().y ), pin->GetNumber(), pinName, aUnitsProvider->MessageTextFromValue( pin->GetPosition().x ), aUnitsProvider->MessageTextFromValue( -pin->GetPosition().y ), aSymbol->GetUnitReference( next->GetUnit() ), aSymbol->GetUnitReference( pin->GetUnit() ) ); } } else { if( pin->GetUnit() == 0 || next->GetUnit() == 0 ) { msg.Printf( _( "Duplicate pin %s %s at location (%s, %s)" " conflicts with pin %s%s at location (%s, %s)." ), next->GetNumber(), nextName, aUnitsProvider->MessageTextFromValue( next->GetPosition().x ), aUnitsProvider->MessageTextFromValue( -next->GetPosition().y ), pin->GetNumber(), pinName, aUnitsProvider->MessageTextFromValue( pin->GetPosition().x ), aUnitsProvider->MessageTextFromValue( -pin->GetPosition().y ) ); } else { msg.Printf( _( "Duplicate pin %s %s at location (%s, %s)" " conflicts with pin %s%s at location (%s, %s)" " in units %s and %s." ), next->GetNumber(), nextName, aUnitsProvider->MessageTextFromValue( next->GetPosition().x ), aUnitsProvider->MessageTextFromValue( -next->GetPosition().y ), pin->GetNumber(), pinName, aUnitsProvider->MessageTextFromValue( pin->GetPosition().x ), aUnitsProvider->MessageTextFromValue( -pin->GetPosition().y ), aSymbol->GetUnitReference( next->GetUnit() ), aSymbol->GetUnitReference( pin->GetUnit() ) ); } } msg += wxT( "

" ); aMessages.push_back( msg ); } // Test for a valid power aSymbol. // A valid power aSymbol has only one unit, no convert and one pin. // And this pin should be PT_POWER_IN (invisible to be automatically connected) // or PT_POWER_OUT for a power flag if( aSymbol->IsPower() ) { if( aSymbol->GetUnitCount() != 1 ) { msg.Printf( _( "A Power Symbol should have only one unit

" ) ); aMessages.push_back( msg ); } if( aSymbol->HasConversion() ) { msg.Printf( _( "A Power Symbol should have no convert option

" ) ); aMessages.push_back( msg ); } if( pinList.size() != 1 ) { msg.Printf( _( "A Power Symbol should have only one pin

" ) ); aMessages.push_back( msg ); } LIB_PIN* pin = pinList[0]; if( pin->GetType() != ELECTRICAL_PINTYPE::PT_POWER_IN && pin->GetType() != ELECTRICAL_PINTYPE::PT_POWER_OUT ) { msg.Printf( _( "Suspicious Power Symbol
" "Only an input or output power pin has meaning

" ) ); aMessages.push_back( msg ); } if( pin->GetType() == ELECTRICAL_PINTYPE::PT_POWER_IN && pin->IsVisible() ) { msg.Printf( _( "Suspicious Power Symbol
" "Only invisible input power pins are automatically connected

" ) ); aMessages.push_back( msg ); } } for( LIB_PIN* pin : pinList ) { wxString pinName = pin->GetName(); if( pinName.IsEmpty() || pinName == "~" ) pinName = ""; else pinName = "'" + pinName + "'"; if( !aSymbol->IsPower() && pin->GetType() == ELECTRICAL_PINTYPE::PT_POWER_IN && !pin->IsVisible() ) { // hidden power pin if( aSymbol->HasConversion() && pin->GetConvert() ) { if( aSymbol->GetUnitCount() <= 1 ) { msg.Printf( _( "Info: Hidden power pin %s %s at location (%s, %s)" " of converted." ), pin->GetNumber(), pinName, aUnitsProvider->MessageTextFromValue( pin->GetPosition().x ), aUnitsProvider->MessageTextFromValue( -pin->GetPosition().y ) ); } else { msg.Printf( _( "Info: Hidden power pin %s %s at location (%s, %s)" " in unit %c of converted." ), pin->GetNumber(), pinName, aUnitsProvider->MessageTextFromValue( pin->GetPosition().x ), aUnitsProvider->MessageTextFromValue( -pin->GetPosition().y ), 'A' + pin->GetUnit() - 1 ); } } else { if( aSymbol->GetUnitCount() <= 1 ) { msg.Printf( _( "Info: Hidden power pin %s %s at location (%s, %s)." ), pin->GetNumber(), pinName, aUnitsProvider->MessageTextFromValue( pin->GetPosition().x ), aUnitsProvider->MessageTextFromValue( -pin->GetPosition().y ) ); } else { msg.Printf( _( "Info: Hidden power pin %s %s at location (%s, %s)" " in unit %c." ), pin->GetNumber(), pinName, aUnitsProvider->MessageTextFromValue( pin->GetPosition().x ), aUnitsProvider->MessageTextFromValue( -pin->GetPosition().y ), 'A' + pin->GetUnit() - 1 ); } } msg += wxT( "
" ); msg += _( "(Hidden power pins will drive their pin names on to any connected nets.)" ); msg += wxT( "

" ); aMessages.push_back( msg ); } if( ( (pin->GetPosition().x % clamped_grid_size) != 0 ) || ( (pin->GetPosition().y % clamped_grid_size) != 0 ) ) { // pin is off grid if( aSymbol->HasConversion() && pin->GetConvert() ) { if( aSymbol->GetUnitCount() <= 1 ) { msg.Printf( _( "Off grid pin %s %s at location (%s, %s)" " of converted." ), pin->GetNumber(), pinName, aUnitsProvider->MessageTextFromValue( pin->GetPosition().x ), aUnitsProvider->MessageTextFromValue( -pin->GetPosition().y ) ); } else { msg.Printf( _( "Off grid pin %s %s at location (%.3s, %.3s)" " in unit %c of converted." ), pin->GetNumber(), pinName, aUnitsProvider->MessageTextFromValue( pin->GetPosition().x ), aUnitsProvider->MessageTextFromValue( -pin->GetPosition().y ), 'A' + pin->GetUnit() - 1 ); } } else { if( aSymbol->GetUnitCount() <= 1 ) { msg.Printf( _( "Off grid pin %s %s at location (%s, %s)." ), pin->GetNumber(), pinName, aUnitsProvider->MessageTextFromValue( pin->GetPosition().x ), aUnitsProvider->MessageTextFromValue( -pin->GetPosition().y ) ); } else { msg.Printf( _( "Off grid pin %s %s at location (%s, %s)" " in unit %c." ), pin->GetNumber(), pinName, aUnitsProvider->MessageTextFromValue( pin->GetPosition().x ), aUnitsProvider->MessageTextFromValue( -pin->GetPosition().y ), 'A' + pin->GetUnit() - 1 ); } } msg += wxT( "

" ); aMessages.push_back( msg ); } } } bool sort_by_pin_number( const LIB_PIN* ref, const LIB_PIN* tst ) { // Use number as primary key int test = ref->GetNumber().Cmp( tst->GetNumber() ); // Use DeMorgan variant as secondary key if( test == 0 ) test = ref->GetConvert() - tst->GetConvert(); // Use unit as tertiary key if( test == 0 ) test = ref->GetUnit() - tst->GetUnit(); return test < 0; }