/***********************/ /* PCBNEW: netlist.cpp */ /***********************/ /* * Function to read a netlist. Whele readint a netlis: * - Load new footprints * - Initialise net info * - Test for missing or extra footprints * - Recalculate ratsnest * * Important remark: * When reading a netlist Pcbnew must identify existing footprints (link between existing footprints an components in netlist) * This identification can be from 2 fields: * - The reference (U2, R5 ..): this is the normal mode * - The Time Stamp (Signature Temporelle), useful after a full schematic reannotation * because references can be changed for the same schematic. * So when reading a netlist this identification ReadPcbNetlist() has aselection of the way to identify footprints. * If we want to fully reannotate a schematic this sequence must be used * SAVE your board !!! * Create and read the netlist (to ensure all info is correct, mainly references and time stamp) * Reannotate the schematic (references will be changed, but not time stamp) * Recreate and read the new netlist using the Time Stamp identification (that reinit the new references) */ #include "fctsys.h" #include "common.h" #include "class_drawpanel.h" #include "confirm.h" #include "kicad_string.h" #include "gestfich.h" #include "pcbnew.h" #include "wxPcbStruct.h" #include "dialog_netlist.h" #include "protos.h" // constants used by ReadPcbNetlist(): #define TESTONLY 1 #define READMODULE 0 /* Structures locales */ class MODULEtoLOAD { public: wxString m_LibName; wxString m_CmpName; wxString m_TimeStampPath; MODULEtoLOAD* m_Next; public: MODULEtoLOAD( const wxString& libname, const wxString& cmpname, const wxString& timestamp_path ) { m_LibName = libname; m_CmpName = cmpname; m_TimeStampPath = timestamp_path; m_Next = NULL; } ~MODULEtoLOAD() { }; MODULEtoLOAD* Next() const { return (MODULEtoLOAD*) m_Next; } void SetNext( MODULEtoLOAD* next ) { m_Next = next; } }; /* Fonctions locales : */ static void SortListModulesToLoadByLibname( int NbModules ); static int BuildFootprintsListFromNetlistFile( const wxString& aNetlistFullFilename, wxArrayString& aBufName ); static bool OpenNetlistFile( const wxString& aFullFileName ); static void AddToList( const wxString& NameLibCmp, const wxString& NameCmp, const wxString& TimeStampPath ); static int SetPadNetName( wxWindow* aFrame, char* Text, MODULE* Module, wxTextCtrl* aMessageWindow ); static int ReadListeModules( const wxString& CmpFullFileName, const wxString* RefCmp, const wxString* TimeStampPath, wxString& NameModule ); static MODULE* ReadNetModule( WinEDA_PcbFrame* aFrame, wxTextCtrl* aMessageWindow, const wxString& CmpFullFileName, char* Text, int* UseFichCmp, int TstOnly, bool Select_By_Timestamp, bool aChangeFootprint ); static void LoadListeModules( WinEDA_PcbFrame* aPcbFrame, wxDC* DC ); /* Variables locales */ static int s_NbNewModules; static MODULEtoLOAD* s_ModuleToLoad_List; FILE* source; /************************************************************************/ bool OpenNetlistFile( const wxString& aFullFileName ) /************************************************************************/ /** function OpenNetlistFile * used to open a netlist file */ { if( aFullFileName.IsEmpty() ) return FALSE; /* No filename: exit */ source = wxFopen( aFullFileName, wxT( "rt" ) ); if( source == 0 ) { wxString msg; msg.Printf( _( "Netlist file %s not found" ), aFullFileName.GetData() ); DisplayError( NULL, msg ); return FALSE; } return true; } /*******************************************************************************/ void ReadPcbNetlist( WinEDA_PcbFrame* aFrame, const wxString& aNetlistFullFilename, const wxString& aCmpFullFileName, wxTextCtrl* aMessageWindow, bool aChangeFootprint, bool aDeleteBadTracks, bool aDeleteExtraFootprints, bool aSelect_By_Timestamp ) /********************************************************************************/ /** Function ReadPcbNetlist * Update footprints (load missing footprints and delete on request extra footprints) * Update connectivity info ( Net Name list ) * Update Reference, value and "TIME STAMP" * @param aNetlistFullFilename = netlist file name (*.net) * @param aCmpFullFileName = cmp/footprint list file name (*.cmp) if not found, only the netmist will be used * * the format of the netlist is something like: # EESchema Netlist Version 1.0 generee le 18/5/2005-12:30:22 * ( * ( 40C08647 $noname R20 4,7K {Lib=R} * ( 1 VCC ) * ( 2 MODB_1 ) * ) * ( 40C0863F $noname R18 4,7_k {Lib=R} * ( 1 VCC ) * ( 2 MODA_1 ) * ) * } * #End */ { int LineNum, State, Comment; MODULE* Module = NULL; D_PAD* PtPad; char Line[256]; char* Text; int UseFichCmp = 1; if( !OpenNetlistFile( aNetlistFullFilename ) ) return; if( aMessageWindow ) { wxString msg; msg.Printf( _( "Reading Netlist \"%s\"" ), GetChars( aNetlistFullFilename ) ); aMessageWindow->AppendText( msg + wxT( "\n" ) ); } aFrame->GetScreen()->SetModify(); aFrame->GetBoard()->m_Status_Pcb = 0; State = 0; LineNum = 0; Comment = 0; s_NbNewModules = 0; wxBusyCursor dummy; // Shows an hourglass while calculating /* First, read the netlist: Build the list of footprints to load (new footprints) */ while( GetLine( source, Line, &LineNum ) ) { Text = StrPurge( Line ); if( Comment ) /* Comments in progress */ { if( ( Text = strchr( Text, '}' ) ) == NULL ) continue; Comment = 0; } if( *Text == '{' ) /* Start Comment */ { Comment = 1; if( ( Text = strchr( Text, '}' ) ) == NULL ) continue; } if( *Text == '(' ) State++; if( *Text == ')' ) State--; if( State == 2 ) { Module = ReadNetModule( aFrame, aMessageWindow, aCmpFullFileName, Text, &UseFichCmp, TESTONLY, aSelect_By_Timestamp, aChangeFootprint ); continue; } if( State >= 3 ) /* la ligne de description d'un pad est ici non analysee */ { State--; } } /* Load new footprints */ if( s_NbNewModules ) { LoadListeModules( aFrame, NULL ); // Free module list: MODULEtoLOAD* item, * next_item; for( item = s_ModuleToLoad_List; item != NULL; item = next_item ) { next_item = item->Next(); delete item; } s_ModuleToLoad_List = NULL; } /* Second read , All footprints are on board, one must update the schematic info (pad netnames) */ fseek( source, 0, SEEK_SET ); LineNum = 0; while( GetLine( source, Line, &LineNum ) ) { Text = StrPurge( Line ); if( Comment ) /* we are reading a commment */ { if( ( Text = strchr( Text, '}' ) ) == NULL ) /* this is the end of a comment */ continue; Comment = 0; } if( *Text == '{' ) /* this is the beginning of a comment */ { Comment = 1; if( ( Text = strchr( Text, '}' ) ) == NULL ) continue; } if( *Text == '(' ) State++; if( *Text == ')' ) State--; if( State == 2 ) { Module = ReadNetModule( aFrame, aMessageWindow, aCmpFullFileName, Text, &UseFichCmp, READMODULE, aSelect_By_Timestamp, aChangeFootprint ); if( Module == NULL ) // the module could not be created (perhaps footprint not found in library) { continue; } else /* clear pads netnames */ { PtPad = Module->m_Pads; for( ; PtPad != NULL; PtPad = (D_PAD*) PtPad->Next() ) { PtPad->SetNetname( wxEmptyString ); } } continue; } if( State >= 3 ) { if( Module ) { SetPadNetName( NULL, Text, Module, aMessageWindow ); } State--; } } fclose( source ); // Delete footprints not found in netlist: if( aDeleteExtraFootprints ) { wxArrayString ModuleListFromNetlist; /* Build list of modules in the netlist */ int NbModulesNetListe = BuildFootprintsListFromNetlistFile( aNetlistFullFilename, ModuleListFromNetlist ); if( NbModulesNetListe ) { MODULE* NextModule; Module = aFrame->GetBoard()->m_Modules; bool ask_for_confirmation = true; for( ; Module != NULL; Module = NextModule ) { int ii; NextModule = Module->Next(); if( (Module->m_ModuleStatus & MODULE_is_LOCKED) ) continue; for( ii = 0; ii < NbModulesNetListe; ii++ ) { if( Module->m_Reference->m_Text.CmpNoCase( ModuleListFromNetlist[ii] ) == 0 ) { break; /* Module trouve en netliste */ } } if( ii == NbModulesNetListe ) /* Module not found in netlist */ { if( ask_for_confirmation ) { ask_for_confirmation = false; if( !IsOK( NULL, _( "Ok to delete footprints not in netlist ?" ) ) ) break; } aFrame->Delete_Module( Module, NULL, false ); } } } } /* Rebuild the connectivity */ aFrame->Compile_Ratsnest( NULL, true ); if( aFrame->GetBoard()->m_Track ) { if( aDeleteBadTracks ) // Remove erroneous tracks { Netliste_Controle_piste( aFrame, NULL, true ); aFrame->Compile_Ratsnest( NULL, true ); } } aFrame->DrawPanel->Refresh(); aFrame->GetBoard()->DisplayInfo( aFrame ); } /****************************************************************************/ MODULE* ReadNetModule( WinEDA_PcbFrame* aFrame, wxTextCtrl* aMessageWindow, const wxString& aCmpFullFileName, char* Text, int* UseFichCmp, int TstOnly, bool aSelect_By_Timestamp, bool aChangeFootprint ) /****************************************************************************/ /* charge la description d'une empreinte, netliste type PCBNEW * et met a jour le module correspondant * * Si TstOnly == 0 si le module n'existe pas, il est charge * Si TstOnly != 0 si le module n'existe pas, il est ajoute a la liste des * modules a charger * Text contient la premiere ligne de la description * UseFichCmp est un flag * si != 0, le fichier des composants .CMP sera utilise * est remis a 0 si le fichier n'existe pas * * Analyse les lignes type: * ( 40C08647 $noname R20 4,7K {Lib=R} * ( 1 VCC ) * ( 2 MODB_1 ) */ { MODULE* Module; char* text; wxString TimeStampPath; wxString TextNameLibMod; wxString TextValeur; wxString TextCmpName; wxString NameLibCmp; int Error = 0; char Line[1024]; bool Found; strcpy( Line, Text ); TextValeur = wxT( "~" ); if( ( text = strtok( Line, " ()\t\n" ) ) == NULL ) Error = 1; else TimeStampPath = CONV_FROM_UTF8( text ); if( ( text = strtok( NULL, " ()\t\n" ) ) == NULL ) Error = 1; else TextNameLibMod = CONV_FROM_UTF8( text ); if( ( text = strtok( NULL, " ()\t\n" ) ) == NULL ) Error = 1; else TextCmpName = CONV_FROM_UTF8( text ); if( ( text = strtok( NULL, " ()\t\n" ) ) == NULL ) Error = -1; else TextValeur = CONV_FROM_UTF8( text ); if( Error > 0 ) return NULL; /* Tst si composant deja charge */ Module = aFrame->GetBoard()->m_Modules; MODULE* NextModule; for( Found = false; Module != NULL; Module = NextModule ) { NextModule = Module->Next(); if( aSelect_By_Timestamp ) /* identification by time stamp */ { if( TimeStampPath.CmpNoCase( Module->m_Path ) == 0 ) Found = true; } else /* identification by Reference */ { if( TextCmpName.CmpNoCase( Module->m_Reference->m_Text ) == 0 ) Found = true; } if( Found ) // test footprint matching for existing modules: current m_LibRef and module name in netlist must match { if( TstOnly != TESTONLY ) { NameLibCmp = TextNameLibMod; if( *UseFichCmp ) { if( aSelect_By_Timestamp ) { *UseFichCmp = ReadListeModules( aCmpFullFileName, NULL, &TimeStampPath, NameLibCmp ); } else { *UseFichCmp = ReadListeModules( aCmpFullFileName, &TextCmpName, NULL, NameLibCmp ); } } if( Module->m_LibRef.CmpNoCase( NameLibCmp ) != 0 ) // Module Mismatch: Current module and module specified in netlist are different. { if( aChangeFootprint ) // footprint exchange allowed: { MODULE* NewModule = aFrame->Get_Librairie_Module( wxEmptyString, NameLibCmp, true ); if( NewModule ) /* Change old module to the new module (and delete the old one)*/ Module = aFrame->Exchange_Module( NULL, Module, NewModule ); } else { wxString msg; msg.Printf( _( "Component \"%s\": Mismatch! module is [%s] and netlist said [%s]\n" ), TextCmpName.GetData(), Module->m_LibRef.GetData(), NameLibCmp.GetData() ); if( aMessageWindow ) aMessageWindow->AppendText( msg ); } } } break; } } if( Module == NULL ) /* a new module must be loaded from libs */ { if( *UseFichCmp ) { if( aSelect_By_Timestamp == 1 ) { *UseFichCmp = ReadListeModules( aCmpFullFileName, NULL, &TimeStampPath, NameLibCmp ); } else { *UseFichCmp = ReadListeModules( aCmpFullFileName, &TextCmpName, NULL, NameLibCmp ); } } if( TstOnly == TESTONLY ) AddToList( NameLibCmp, TextCmpName, TimeStampPath ); else { if( aMessageWindow ) { wxString msg; msg.Printf( _( "Component [%s] not found" ), TextCmpName.GetData() ); aMessageWindow->AppendText( msg + wxT( "\n" ) ); } } return NULL; /* Le module n'avait pas pu etre charge */ } /* Fields update ( reference, value and "Time Stamp") */ Module->m_Reference->m_Text = TextCmpName; Module->m_Value->m_Text = TextValeur; Module->m_Path = TimeStampPath; return Module; /* composant trouve */ } /*******************************************************************************************/ int SetPadNetName( wxWindow* frame, char* Text, MODULE* Module, wxTextCtrl* aMessageWindow ) /*******************************************************************************************/ /** Function SetPadNetName * Update a pad netname in a given footprint * @param Text = Text from netlist (format: (pad = net) ) * @param Module = the given footprint * @param aMessageWindow = a wxTextCtrl to print error and warning message (can be NULL) */ { D_PAD* pad; char* TextPinName, * TextNetName; int Error = 0; bool trouve; char Line[256]; wxString Msg; if( Module == NULL ) return 0; strcpy( Line, Text ); if( ( TextPinName = strtok( Line, " ()\t\n" ) ) == NULL ) Error = 1; if( ( TextNetName = strtok( NULL, " ()\t\n" ) ) == NULL ) Error = 1; if( Error ) return 0; /* recherche du pad */ pad = Module->m_Pads; trouve = FALSE; for( ; pad != NULL; pad = (D_PAD*) pad->Next() ) { if( strnicmp( TextPinName, pad->m_Padname, 4 ) == 0 ) /* trouve */ { trouve = true; if( *TextNetName != '?' ) pad->SetNetname( CONV_FROM_UTF8( TextNetName ) ); else pad->SetNetname( wxEmptyString ); } } if( !trouve ) { if( aMessageWindow ) { wxString pin_name = CONV_FROM_UTF8( TextPinName ); Msg.Printf( _( "Module [%s]: Pad [%s] not found" ), Module->m_Reference->m_Text.GetData(), pin_name.GetData() ); aMessageWindow->AppendText( Msg + wxT( "\n" ) ); } } return trouve; } /*****************************************************/ MODULE* WinEDA_PcbFrame::ListAndSelectModuleName( void ) /*****************************************************/ /** * build and shows a list of existing modules on board * The user can slect a module from this list * @return a pointer to the selected module or NULL */ { int ii, jj; MODULE* Module; if( GetBoard()->m_Modules == NULL ) { DisplayError( this, _( "No Modules" ) ); return 0; } WinEDAListBox listbox( this, _( "Components" ), NULL, wxEmptyString ); Module = (MODULE*) GetBoard()->m_Modules; for( ; Module != NULL; Module = (MODULE*) Module->Next() ) { listbox.Append( Module->m_Reference->m_Text ); } ii = listbox.ShowModal(); if( ii < 0 ) { Module = NULL; } else /* Search for the selected footprint */ { wxString ref = listbox.GetTextSelection(); Module = (MODULE*) GetBoard()->m_Modules; for( jj = 0; Module != NULL; Module = (MODULE*) Module->Next(), jj++ ) { if( Module->m_Reference->m_Text.Cmp( ref ) == 0 ) break; } } return Module; } /*****************************************************************************************/ void TestFor_Duplicate_Missing_And_Extra_Footprints( wxWindow* aFrame, const wxString& aNetlistFullFilename, BOARD* aPcb ) /******************************************************************************************/ /** Function TestFor_Duplicate_Missing_And_Extra_Footprints * Build a list from the given board and netlist : * 1 - for duplicate footprints on board * 2 - for missing footprints * 3 - for footprints not in netlist * @param aFrame = current active frame * @param aNetlistFullFilename = the given netlist * @param aPcb = the given board */ #define MAX_LEN_TXT 32 { int ii; MODULE* Module, * pt_aux; int NbModulesNetListe, nberr = 0; WinEDA_TextFrame* List; wxArrayString ModuleListFromNetlist; if( aPcb->m_Modules == NULL ) { DisplayInfoMessage( aFrame, _( "No modules" ), 10 ); return; } /* Construction de la liste des references des modules de la netliste */ NbModulesNetListe = BuildFootprintsListFromNetlistFile( aNetlistFullFilename, ModuleListFromNetlist ); if( NbModulesNetListe < 0 ) return; /* File not found */ if( NbModulesNetListe == 0 ) { DisplayError( aFrame, _( "No modules in NetList" ), 10 ); return; } List = new WinEDA_TextFrame( aFrame, _( "Check Modules" ) ); /* recherche des doubles */ List->Append( _( "Duplicates" ) ); Module = aPcb->m_Modules; for( ; Module != NULL; Module = Module->Next() ) { pt_aux = Module->Next(); for( ; pt_aux != NULL; pt_aux = pt_aux->Next() ) { if( Module->m_Reference->m_Text.CmpNoCase( pt_aux->m_Reference->m_Text ) == 0 ) { List->Append( Module->m_Reference->m_Text ); nberr++; break; } } } /* recherche des manquants par rapport a la netliste*/ List->Append( _( "Lack:" ) ); for( ii = 0; ii < NbModulesNetListe; ii++ ) { Module = (MODULE*) aPcb->m_Modules; for( ; Module != NULL; Module = Module->Next() ) { if( Module->m_Reference->m_Text.CmpNoCase( ModuleListFromNetlist[ii] ) == 0 ) { break; } } if( Module == NULL ) // Module missing, not found in board { List->Append( ModuleListFromNetlist[ii] ); nberr++; } } /* recherche des modules supplementaires (i.e. Non en Netliste) */ List->Append( _( "Not in Netlist:" ) ); Module = (MODULE*) aPcb->m_Modules; for( ; Module != NULL; Module = Module->Next() ) { for( ii = 0; ii < NbModulesNetListe; ii++ ) { if( Module->m_Reference->m_Text.CmpNoCase( ModuleListFromNetlist[ii] ) == 0 ) { break; /* Module trouve en netliste */ } } if( ii == NbModulesNetListe ) /* Module not found in netlist */ { List->Append( Module->m_Reference->m_Text ); nberr++; } } List->ShowModal(); List->Destroy(); } /***************************************************************************************/ int BuildFootprintsListFromNetlistFile( const wxString& aNetlistFullFilename, wxArrayString& aBufName ) /***************************************************************************************/ /** Function BuildFootprintsListFromNetlistFile * Fill BufName with footprints names read from the netlist. * @param aNetlistFullFilename = netlist file name * @param BufName = wxArrayString to fill with footprint names * @return Footprint count, or -1 if netlist file cannot opened */ { int textlen; int nb_modules_lus; int State, LineNum, Comment; char Line[1024], * Text, * LibModName; if( !OpenNetlistFile( aNetlistFullFilename ) ) return -1; State = 0; LineNum = 0; Comment = 0; nb_modules_lus = 0; textlen = MAX_LEN_TXT; while( GetLine( source, Line, &LineNum ) ) { Text = StrPurge( Line ); if( Comment ) /* Commentaires en cours */ { if( ( Text = strchr( Text, '}' ) ) == NULL ) continue; Comment = 0; } if( *Text == '{' ) /* Commentaires */ { Comment = 1; if( ( Text = strchr( Text, '}' ) ) == NULL ) continue; } if( *Text == '(' ) State++; if( *Text == ')' ) State--; if( State == 2 ) { int Error = 0; if( strtok( Line, " ()\t\n" ) == NULL ) Error = 1; /* TimeStamp */ if( ( LibModName = strtok( NULL, " ()\t\n" ) ) == NULL ) Error = 1; /* nom Lib */ /* Lecture du nom (reference) du composant: */ if( ( Text = strtok( NULL, " ()\t\n" ) ) == NULL ) Error = 1; nb_modules_lus++; aBufName.Add( CONV_FROM_UTF8( Text ) ); continue; } if( State >= 3 ) { State--; /* Lecture 1 ligne relative au Pad */ } } fclose( source ); return nb_modules_lus; } /*****************************************************************************************/ int ReadListeModules( const wxString& CmpFullFileName, const wxString* RefCmp, const wxString* TimeStamp, wxString& NameModule ) /*****************************************************************************************/ /* * Lit le fichier .CMP donnant l'equivalence Modules / Composants * Retourne: * Si ce fichier existe retourne : * 1 et le nom module dans NameModule * -1 si module non trouve en fichier * sinon 0; * * parametres d'appel: * RefCmp (NULL si selection par TimeStamp) * TimeStamp (signature temporelle si elle existe, NULL sinon) * pointeur sur le buffer recevant le nom du module * * Exemple de fichier: * * Cmp-Mod V01 Genere par PcbNew le 29/10/2003-13:11:6 * * BeginCmp * TimeStamp = /322D3011; * Reference = BUS1; * ValeurCmp = BUSPC; * IdModule = BUS_PC; * EndCmp * * BeginCmp * TimeStamp = /32307DE2/AA450F67; * Reference = C1; * ValeurCmp = 47uF; * IdModule = CP6; * EndCmp * */ { wxString refcurrcmp, timestamp, idmod; char ia[1024]; char* ptcar; FILE* FichCmp; if( (RefCmp == NULL) && (TimeStamp == 0) ) return 0; FichCmp = wxFopen( CmpFullFileName, wxT( "rt" ) ); if( FichCmp == NULL ) { wxString msg; msg.Printf( _( "File <%s> not found, use Netlist for lib module selection" ), CmpFullFileName.GetData() ); DisplayError( NULL, msg, 20 ); return 0; } while( fgets( ia, sizeof(ia), FichCmp ) != NULL ) { if( strnicmp( ia, "BeginCmp", 8 ) != 0 ) continue; /* Ici une description de 1 composant commence */ refcurrcmp.Empty(); idmod.Empty(); timestamp.Empty(); while( fgets( ia, sizeof(ia), FichCmp ) != NULL ) { if( strnicmp( ia, "EndCmp", 6 ) == 0 ) break; if( strnicmp( ia, "Reference =", 11 ) == 0 ) { ptcar = ia + 11; ptcar = strtok( ptcar, " =;\t\n" ); if( ptcar ) refcurrcmp = CONV_FROM_UTF8( ptcar ); continue; } if( strnicmp( ia, "IdModule =", 11 ) == 0 ) { ptcar = ia + 11; ptcar = strtok( ptcar, " =;\t\n" ); if( ptcar ) idmod = CONV_FROM_UTF8( ptcar ); continue; } if( strnicmp( ia, "TimeStamp =", 11 ) == 0 ) { ptcar = ia + 11; ptcar = strtok( ptcar, " =;\t\n" ); if( ptcar ) timestamp = CONV_FROM_UTF8( ptcar ); } } /* Fin lecture 1 descr composant */ /* Test du Composant lu en fichier: est-il le bon */ if( RefCmp ) { if( RefCmp->CmpNoCase( refcurrcmp ) == 0 ) //Found! { fclose( FichCmp ); NameModule = idmod; return 1; } } else if( TimeStamp ) { if( TimeStamp->CmpNoCase( timestamp ) == 0 && !timestamp.IsEmpty() ) //Found { fclose( FichCmp ); NameModule = idmod; return 1; } } } fclose( FichCmp ); return -1; } /***********************************************************************************/ void AddToList( const wxString& NameLibCmp, const wxString& CmpName, const wxString& path ) /************************************************************************************/ /* Fontion copiant en memoire de travail les caracteristiques * des nouveaux modules */ { MODULEtoLOAD* NewMod; NewMod = new MODULEtoLOAD( NameLibCmp, CmpName, path ); NewMod->SetNext( s_ModuleToLoad_List ); s_ModuleToLoad_List = NewMod; s_NbNewModules++; } /***************************************************************/ void LoadListeModules( WinEDA_PcbFrame* aPcbFrame, wxDC* DC ) /***************************************************************/ /* Routine de chargement des nouveaux modules en une seule lecture des * librairies * Si un module vient d'etre charge il est duplique, ce qui evite une lecture * inutile de la librairie */ { MODULEtoLOAD* ref, * cmp; int ii; MODULE* Module = NULL; wxPoint OldPos = aPcbFrame->GetScreen()->m_Curseur; if( s_NbNewModules == 0 ) return; SortListModulesToLoadByLibname( s_NbNewModules ); ref = cmp = s_ModuleToLoad_List; // Calculate the footprint "best" position: if( aPcbFrame->SetBoardBoundaryBoxFromEdgesOnly() ) { aPcbFrame->GetScreen()->m_Curseur.x = aPcbFrame->GetBoard()->m_BoundaryBox.GetRight() + 5000; aPcbFrame->GetScreen()->m_Curseur.y = aPcbFrame->GetBoard()->m_BoundaryBox.GetBottom() + 10000; } else { aPcbFrame->GetScreen()->m_Curseur = wxPoint( 0, 0 ); } for( ii = 0; ii < s_NbNewModules; ii++, cmp = cmp->Next() ) { if( (ii == 0) || ( ref->m_LibName != cmp->m_LibName) ) { /* New footprint : must be loaded from a library */ Module = aPcbFrame->Get_Librairie_Module( wxEmptyString, cmp->m_LibName, FALSE ); ref = cmp; if( Module == NULL ) { wxString msg; msg.Printf( _( "Component [%s]: footprint <%s> not found" ), cmp->m_CmpName.GetData(), cmp->m_LibName.GetData() ); DisplayError( NULL, msg ); continue; } aPcbFrame->Place_Module( Module, DC ); /* Update schematic links : reference "Time Stamp" and schematic hierarchical path */ Module->m_Reference->m_Text = cmp->m_CmpName; Module->m_TimeStamp = GetTimeStamp(); Module->m_Path = cmp->m_TimeStampPath; } else { /* Footprint already loaded from a library, duplicate it (faster) */ MODULE* newmodule; if( Module == NULL ) continue; /* module non existant en libr */ newmodule = new MODULE( aPcbFrame->GetBoard() ); newmodule->Copy( Module ); aPcbFrame->GetBoard()->Add( newmodule, ADD_APPEND ); Module = newmodule; Module->m_Reference->m_Text = cmp->m_CmpName; Module->m_TimeStamp = GetTimeStamp(); Module->m_Path = cmp->m_TimeStampPath; } } aPcbFrame->GetScreen()->m_Curseur = OldPos; } /* Routine utilisee par qsort pour le tri des modules a charger */ static int SortByLibName( MODULEtoLOAD** ref, MODULEtoLOAD** cmp ) { int ii = (*ref)->m_LibName.CmpNoCase( (*cmp)->m_LibName ); return ii; } /*************************************************/ void SortListModulesToLoadByLibname( int NbModules ) /**************************************************/ /* Rearrage la liste des modules List par ordre alphabetique des noms lib des modules */ { MODULEtoLOAD** base_list, * item; int ii; base_list = (MODULEtoLOAD**) MyMalloc( NbModules * sizeof(MODULEtoLOAD*) ); for( ii = 0, item = s_ModuleToLoad_List; ii < NbModules; ii++ ) { base_list[ii] = item; item = item->Next(); } qsort( base_list, NbModules, sizeof(MODULEtoLOAD*), ( int( * ) ( const void*, const void* ) )SortByLibName ); // Reconstruction du chainage: s_ModuleToLoad_List = *base_list; for( ii = 0; ii < NbModules - 1; ii++ ) { item = base_list[ii]; item->SetNext( base_list[ii + 1] ); } // Dernier item: Pnext = NULL: item = base_list[ii]; item->SetNext( NULL ); free( base_list ); }