/* * This program source code file is part of KiCad, a free EDA CAD application. * * Copyright (C) 2011 Jean-Pierre Charras, * Copyright (C) 2013-2016 SoftPLC Corporation, Dick Hollenbeck * Copyright (C) 1992-2016 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 */ /** * @file footprint_info.cpp */ /** No. concurrent threads doing "http(s) GET". More than 6 is not significantly faster, less than 6 is likely slower. Main thread is in this count, so if set to 1 then no temp threads are created. */ #define READER_THREADS 6 /* * Functions to read footprint libraries and fill m_footprints by available footprints names * and their documentation (comments and keywords) */ #include #include #include #include #include #include #include #include #include #include #include #include /* static wxString ToHTMLFragment( const IO_ERROR* aDerivative ) { @todo 1) change up IO_ERROR so it keeps linenumbers, source file name and error message in separate strings. 2) Add a summarizing virtual member like virtual wxString What() to combine all portions of an IO_ERROR's text into a single wxString. 3) Do same for PARSE_ERROR. 4) Add a "reason or error category" to IO_ERROR and thereby also PARSE_ERROR? msg += " for( int i=0; i" ); for ( unsigned ii = 0; ii < strings_list->GetCount(); ii++ ) { msg += wxT( "
  • " ); msg += strings_list->Item( ii ) + wxT( "
  • " ); } msg += wxT( "" ); m_htmlWindow->AppendToPage( msg ); delete strings_list; } */ void FOOTPRINT_INFO::load() { FP_LIB_TABLE* fptable = m_owner->GetTable(); wxASSERT( fptable ); std::unique_ptr footprint( fptable->FootprintLoad( m_nickname, m_fpname ) ); if( footprint.get() == NULL ) // Should happen only with malformed/broken libraries { m_pad_count = 0; m_unique_pad_count = 0; } else { m_pad_count = footprint->GetPadCount( DO_NOT_INCLUDE_NPTH ); m_unique_pad_count = footprint->GetUniquePadCount( DO_NOT_INCLUDE_NPTH ); m_keywords = footprint->GetKeywords(); m_doc = footprint->GetDescription(); // tell ensure_loaded() I'm loaded. m_loaded = true; } } void FOOTPRINT_LIST::loader_job( const wxString* aNicknameList, int aJobZ ) { for( int i=0; iFootprintEnumerate( nickname ); for( unsigned ni=0; ni nicknames; // do all of them nicknames = aTable->GetLogicalLibs(); // Even though the PLUGIN API implementation is the place for the // locale toggling, in order to keep LOCAL_IO::C_count at 1 or greater // for the duration of all helper threads, we increment by one here via instantiation. // Only done here because of the multi-threaded nature of this code. // Without this C_count skips in and out of "equal to zero" and causes // needless locale toggling among the threads, based on which of them // are in a PLUGIN::FootprintLoad() function. And that is occasionally // none of them. LOCALE_IO top_most_nesting; // Something which will not invoke a thread copy constructor, one of many ways obviously: typedef boost::ptr_vector< boost::thread > MYTHREADS; MYTHREADS threads; unsigned jobz = (nicknames.size() + READER_THREADS - 1) / READER_THREADS; // Give each thread JOBZ nicknames to process. The last portion of, or if the entire // size() is small, I'll do myself. for( unsigned i=0; i= nicknames.size() ) // on the last iteration of this for(;;) { jobz = nicknames.size() - i; // Only a little bit to do, I'll do it myself on current thread. // I am part of the READER_THREADS count. loader_job( &nicknames[i], jobz ); } else { // Delegate the job to a temporary thread created here. threads.push_back( new boost::thread( &FOOTPRINT_LIST::loader_job, this, &nicknames[i], jobz ) ); } i += jobz; } // Wait for all the worker threads to complete, it does not matter in what order // we wait for them as long as a full sweep is made. Think of the great race, // everyone must finish. for( unsigned i=0; i! ? centric output, possibly with // recommendations for remedy of errors. Add numeric error codes // to PARSE_ERROR, and switch on them for remedies, etc. Full // access is provided to everything in every exception! HTML_MESSAGE_BOX dlg( aWindow, _( "Load Error" ) ); dlg.MessageSet( _( "Errors were encountered loading footprints:" ) ); wxString msg; for( unsigned i = 0; i" ) + m_errors[i].errorText + wxT( "

    " ); } dlg.AddHTML_Text( msg ); dlg.ShowModal(); }