/** * @file pcbnew/dialogs/dialog_netlist.cpp */ /* * This program source code file is part of KiCad, a free EDA CAD application. * * Copyright (C) 1992-2013 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 */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #define NETLIST_SILENTMODE_KEY wxT("SilentMode") #define NETLIST_FULLMESSAGES_KEY wxT("NetlistReportAllMsg") #define NETLIST_DELETESINGLEPADNETS_KEY wxT("NetlistDeleteSinglePadNets") void PCB_EDIT_FRAME::InstallNetlistFrame( wxDC* DC ) { /* Setup the netlist file name to the last netlist file read, * or the board file name if the last filename is empty or last file not existing. */ wxFileName fn = GetLastNetListRead(); wxString lastNetlistName = GetLastNetListRead(); if( !fn.FileExists() ) { fn = GetBoard()->GetFileName(); fn.SetExt( NetlistFileExtension ); lastNetlistName = fn.GetFullPath(); } DIALOG_NETLIST dlg( this, DC, lastNetlistName ); dlg.ShowModal(); // Save project settings if needed. // Project settings are saved in the corresponding .pro file bool configChanged = lastNetlistName != GetLastNetListRead(); if( dlg.UseCmpFileForFpNames() != GetUseCmpFileForFpNames() ) { SetUseCmpFileForFpNames( dlg.UseCmpFileForFpNames() ); configChanged = true; } if( configChanged && !GetBoard()->GetFileName().IsEmpty() && IsOK( NULL, _( "The project configuration has changed. Do you want to save it?" ) ) ) { wxFileName fn = Prj().AbsolutePath( GetBoard()->GetFileName() ); fn.SetExt( ProjectFileExtension ); wxString pro_name = fn.GetFullPath(); Prj().ConfigSave( Kiface().KifaceSearch(), GROUP_PCB, GetProjectFileParameters(), pro_name ); } } DIALOG_NETLIST::DIALOG_NETLIST( PCB_EDIT_FRAME* aParent, wxDC * aDC, const wxString & aNetlistFullFilename ) : DIALOG_NETLIST_FBP( aParent ) { m_parent = aParent; m_dc = aDC; m_config = Kiface().KifaceSettings(); m_silentMode = m_config->Read( NETLIST_SILENTMODE_KEY, 0l ); m_reportAll = m_config->Read( NETLIST_FULLMESSAGES_KEY, 1l ); bool tmp = m_config->Read( NETLIST_DELETESINGLEPADNETS_KEY, 0l ); m_rbSingleNets->SetSelection( tmp == 0 ? 0 : 1); m_NetlistFilenameCtrl->SetValue( aNetlistFullFilename ); m_cmpNameSourceOpt->SetSelection( m_parent->GetUseCmpFileForFpNames() ? 1 : 0 ); m_checkBoxSilentMode->SetValue( m_silentMode ); m_checkBoxFullMessages->SetValue( m_reportAll ); GetSizer()->SetSizeHints( this ); } DIALOG_NETLIST::~DIALOG_NETLIST() { m_config->Write( NETLIST_SILENTMODE_KEY, (long) m_silentMode ); m_config->Write( NETLIST_FULLMESSAGES_KEY, (long) m_reportAll ); m_config->Write( NETLIST_DELETESINGLEPADNETS_KEY, (long) m_rbSingleNets->GetSelection() ); } void DIALOG_NETLIST::OnOpenNetlistClick( wxCommandEvent& event ) { wxString lastPath = wxFileName( Prj().GetProjectFullName() ).GetPath(); wxString lastNetlistRead = m_parent->GetLastNetListRead(); if( !lastNetlistRead.IsEmpty() && !wxFileName::FileExists( lastNetlistRead ) ) { lastNetlistRead = wxEmptyString; } else { wxFileName fn = lastNetlistRead; lastPath = fn.GetPath(); lastNetlistRead = fn.GetFullName(); } wxLogDebug( wxT( "Last net list read path '%s', file name '%s'." ), GetChars( lastPath ), GetChars( lastNetlistRead ) ); wxFileDialog FilesDialog( this, _( "Select Netlist" ), lastPath, lastNetlistRead, NetlistFileWildcard, wxFD_DEFAULT_STYLE | wxFD_FILE_MUST_EXIST ); if( FilesDialog.ShowModal() != wxID_OK ) return; m_NetlistFilenameCtrl->SetValue( FilesDialog.GetPath() ); } void DIALOG_NETLIST::OnReadNetlistFileClick( wxCommandEvent& event ) { wxString msg; wxString netlistFileName = m_NetlistFilenameCtrl->GetValue(); wxString cmpFileName; if( UseCmpFileForFpNames() ) { wxFileName fn = m_NetlistFilenameCtrl->GetValue(); fn.SetExt( ComponentFileExtension ); cmpFileName = fn.GetFullPath(); } // Give the user a chance to bail out when making changes from a netlist. if( !m_checkDryRun->GetValue() && !m_silentMode && !m_parent->GetBoard()->IsEmpty() && !IsOK( NULL, _( "The changes made by reading the netlist cannot be undone. Are you " "sure you want to read the netlist?" ) ) ) return; wxBusyCursor busy; m_MessageWindow->Clear(); msg.Printf( _( "Reading netlist file \"%s\".\n" ), GetChars( netlistFileName ) ); m_MessageWindow->AppendText( msg ); if( !cmpFileName.IsEmpty() ) { msg.Printf( _( "Using component footprint link file \"%s\".\n" ), GetChars( cmpFileName ) ); m_MessageWindow->AppendText( msg ); } if( m_Select_By_Timestamp->GetSelection() == 1 ) { msg.Printf( _( "Using time stamps to select footprints in file \"%s\".\n" ), GetChars( cmpFileName ) ); m_MessageWindow->AppendText( msg ); } WX_TEXT_CTRL_REPORTER reporter( m_MessageWindow ); reporter.SetReportAll( m_reportAll ); m_parent->ReadPcbNetlist( netlistFileName, cmpFileName, &reporter, m_ChangeExistingFootprintCtrl->GetSelection() == 1, m_DeleteBadTracks->GetSelection() == 1, m_RemoveExtraFootprintsCtrl->GetSelection() == 1, m_Select_By_Timestamp->GetSelection() == 1, m_rbSingleNets->GetSelection() == 1, m_checkDryRun->GetValue() ); } void DIALOG_NETLIST::OnTestFootprintsClick( wxCommandEvent& event ) { if( m_parent->GetBoard()->m_Modules == NULL ) { DisplayInfoMessage( this, _( "No modules" ) ); return; } // Lists of duplicates, missing references and not in netlist footprints: std::vector duplicate; wxArrayString missing; std::vector notInNetlist; wxString netlistFilename = m_NetlistFilenameCtrl->GetValue(); wxString cmpFilename; if( UseCmpFileForFpNames() ) { wxFileName fn = m_NetlistFilenameCtrl->GetValue(); fn.SetExt( ComponentFileExtension ); cmpFilename = fn.GetFullPath(); } if( !verifyFootprints( netlistFilename, cmpFilename, duplicate, missing, notInNetlist ) ) return; #define ERR_CNT_MAX 100 // Max number of errors to output in dialog // to avoid a too long message list wxString list; // The messages to display m_parent->SetLastNetListRead( netlistFilename ); int err_cnt = 0; // Search for duplicate footprints. if( duplicate.size() == 0 ) list << wxT("

") << _( "No duplicate." ) << wxT("

"); else { list << wxT("

") << _( "Duplicates:" ) << wxT("

"); for( unsigned ii = 0; ii < duplicate.size(); ii++ ) { MODULE* module = duplicate[ii]; if( module->GetReference().IsEmpty() ) list << wxT("
") << wxT("[noref)"); else list << wxT("
") << module->GetReference(); list << wxT(" (") << module->GetValue() << wxT(")"); list << wxT(" @ "); list << CoordinateToString( module->GetPosition().x ), list << wxT(", ") << CoordinateToString( module->GetPosition().y ), err_cnt++; if( ERR_CNT_MAX < err_cnt ) break; } } // Search for missing modules on board. if( missing.size() == 0 ) list << wxT("

") << _( "No missing modules." ) << wxT("

"); else { list << wxT("

") << _( "Missing:" ) << wxT("

"); for( unsigned ii = 0; ii < missing.size(); ii += 2 ) { list << wxT("
") << missing[ii]; list << wxT(" (") << missing[ii+1] << wxT(")"); err_cnt++; if( ERR_CNT_MAX < err_cnt ) break; } } // Search for modules found on board but not in net list. if( notInNetlist.size() == 0 ) list << wxT( "

" ) << _( "No extra modules." ) << wxT( "

" ); else { list << wxT( "

" ) << _( "Not in Netlist:" ) << wxT( "

" ); for( unsigned ii = 0; ii < notInNetlist.size(); ii++ ) { MODULE* module = notInNetlist[ii]; if( module->GetReference().IsEmpty() ) list << wxT( "
" ) << wxT( "[noref)" ); else list << wxT( "
" ) << module->GetReference() ; list << wxT( " (" ) << module->GetValue() << wxT( ")" ); list << wxT( " @ " ); list << CoordinateToString( module->GetPosition().x ), list << wxT( ", " ) << CoordinateToString( module->GetPosition().y ), err_cnt++; if( ERR_CNT_MAX < err_cnt ) break; } } if( ERR_CNT_MAX < err_cnt ) { list << wxT( "

" ) << _( "Too many errors: some are skipped" ) << wxT( "

" ); } HTML_MESSAGE_BOX dlg( this, _( "Check Modules" ) ); dlg.AddHTML_Text( list ); dlg.ShowModal(); } /*! * wxEVT_COMMAND_BUTTON_CLICKED event handler for ID_COMPILE_RATSNEST */ void DIALOG_NETLIST::OnCompileRatsnestClick( wxCommandEvent& event ) { m_parent->Compile_Ratsnest( m_dc, true ); } /*! * wxEVT_COMMAND_BUTTON_CLICKED event handler for wxID_CANCEL */ void DIALOG_NETLIST::OnCancelClick( wxCommandEvent& event ) { EndModal( wxID_CANCEL ); } void DIALOG_NETLIST::OnSaveMessagesToFile( wxCommandEvent& aEvent ) { wxFileName fn; if( !m_parent->GetLastNetListRead().IsEmpty() ) { fn = m_parent->GetLastNetListRead(); fn.SetExt( wxT( "txt" ) ); } else { fn = wxPathOnly( Prj().GetProjectFullName() ); } wxFileDialog dlg( this, _( "Save contents of message window" ), fn.GetPath(), fn.GetName(), TextWildcard, wxFD_SAVE | wxFD_OVERWRITE_PROMPT ); if( dlg.ShowModal() != wxID_OK ) return; fn = dlg.GetPath(); if( fn.GetExt().IsEmpty() ) fn.SetExt( wxT( "txt" ) ); wxFile f( fn.GetFullPath(), wxFile::write ); if( !f.IsOpened() ) { wxString msg; msg.Printf( _( "Cannot write message contents to file \"%s\"." ), GetChars( fn.GetFullPath() ) ); wxMessageBox( msg, _( "File Write Error" ), wxOK | wxICON_ERROR, this ); return; } f.Write( m_MessageWindow->GetValue() ); } void DIALOG_NETLIST::OnUpdateUISaveMessagesToFile( wxUpdateUIEvent& aEvent ) { aEvent.Enable( !m_MessageWindow->IsEmpty() ); } void DIALOG_NETLIST::OnUpdateUIValidNetlistFile( wxUpdateUIEvent& aEvent ) { aEvent.Enable( !m_NetlistFilenameCtrl->GetValue().IsEmpty() ); } bool DIALOG_NETLIST::verifyFootprints( const wxString& aNetlistFilename, const wxString & aCmpFilename, std::vector< MODULE* >& aDuplicates, wxArrayString& aMissing, std::vector< MODULE* >& aNotInNetlist ) { wxString msg; MODULE* module; MODULE* nextModule; NETLIST netlist; wxBusyCursor dummy; // Shows an hourglass while calculating. NETLIST_READER* netlistReader; COMPONENT* component; try { netlistReader = NETLIST_READER::GetNetlistReader( &netlist, aNetlistFilename, aCmpFilename ); if( netlistReader == NULL ) { msg.Printf( _( "Cannot open netlist file \"%s\"." ), GetChars( aNetlistFilename ) ); wxMessageBox( msg, _( "Netlist Load Error." ), wxOK | wxICON_ERROR ); return false; } std::auto_ptr< NETLIST_READER > nlr( netlistReader ); netlistReader->LoadNetlist(); } catch( const IO_ERROR& ioe ) { msg.Printf( _( "Error loading netlist file:\n%s" ), ioe.errorText.GetData() ); wxMessageBox( msg, _( "Netlist Load Error" ), wxOK | wxICON_ERROR ); return false; } #if defined( DEBUG ) { m_MessageWindow->Clear(); WX_TEXT_CTRL_REPORTER rpt( m_MessageWindow ); STRING_FORMATTER sf; netlist.Format( "netlist_stuff", &sf, 0 ); rpt.Report( FROM_UTF8( sf.GetString().c_str() ) ); } #endif BOARD* pcb = m_parent->GetBoard(); // Search for duplicate footprints. module = pcb->m_Modules; for( ; module != NULL; module = module->Next() ) { nextModule = module->Next(); for( ; nextModule != NULL; nextModule = nextModule->Next() ) { if( module->GetReference().CmpNoCase( nextModule->GetReference() ) == 0 ) { aDuplicates.push_back( module ); break; } } } // Search for component footprints in the netlist but not on the board. for( unsigned ii = 0; ii < netlist.GetCount(); ii++ ) { component = netlist.GetComponent( ii ); module = pcb->FindModuleByReference( component->GetReference() ); if( module == NULL ) { aMissing.Add( component->GetReference() ); aMissing.Add( component->GetValue() ); } } // Search for component footprints found on board but not in netlist. module = pcb->m_Modules; for( ; module != NULL; module = module->Next() ) { component = netlist.GetComponentByReference( module->GetReference() ); if( component == NULL ) aNotInNetlist.push_back( module ); } return true; }