KiCad PCB EDA Suite
Loading...
Searching...
No Matches
footprint_library_adapter.cpp
Go to the documentation of this file.
1/*
2 * This program source code file is part of KiCad, a free EDA CAD application.
3 *
4 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
5 * @author Jon Evans <[email protected]>
6 *
7 * This program is free software: you can redistribute it and/or modify it
8 * under the terms of the GNU General Public License as published by the
9 * Free Software Foundation, either version 3 of the License, or (at your
10 * option) any later version.
11 *
12 * This program is distributed in the hope that it will be useful, but
13 * WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * General Public License for more details.
16 *
17 * You should have received a copy of the GNU General Public License along
18 * with this program. If not, see <http://www.gnu.org/licenses/>.
19 */
20
22
23#include <chrono>
24#include <env_vars.h>
25#include <footprint_info_impl.h>
26#include <thread_pool.h>
27#include <trace_helpers.h>
28#include <footprint.h>
29
30#include <magic_enum.hpp>
31#include <wx/hash.h>
32#include <wx/log.h>
33
34using namespace std::chrono_literals;
35
36
38
40
42
44
45
50
51
53{
54 return ENV_VAR::GetVersionedEnvVarName( wxS( "FOOTPRINT_DIR" ) );
55}
56
57
59{
60 wxArrayString namesAS;
61 std::map<std::string, UTF8> options = aLib->row->GetOptionsMap();
62 PCB_IO* plugin = pcbplugin( aLib );
63 wxString uri = getUri( aLib->row );
64 wxString nickname = aLib->row->Nickname();
65
66 // FootprintEnumerate populates the plugin's internal FP_CACHE with parsed footprints
67 plugin->FootprintEnumerate( namesAS, uri, false, &options );
68
69 std::vector<std::unique_ptr<FOOTPRINT>> footprints;
70 footprints.reserve( namesAS.size() );
71
72 // For plugins with internal caches (like kicad_sexpr), GetEnumeratedFootprint returns
73 // a borrowed pointer and ClearCachedFootprints handles cleanup. For other plugins,
74 // GetEnumeratedFootprint allocates new memory that we must delete after cloning.
75 const bool pluginCaches = plugin->CachesEnumeratedFootprints();
76
77 for( const wxString& footprintName : namesAS )
78 {
79 try
80 {
81 const FOOTPRINT* cached = plugin->GetEnumeratedFootprint( uri, footprintName, &options );
82
83 if( !cached )
84 continue;
85
86 FOOTPRINT* footprint = static_cast<FOOTPRINT*>( cached->Duplicate( IGNORE_PARENT_GROUP ) );
87 footprint->SetParent( nullptr );
88
89 // For non-caching plugins, delete the allocated footprint now that we've cloned it
90 if( !pluginCaches )
91 delete cached;
92
93 LIB_ID id = footprint->GetFPID();
94 id.SetLibNickname( nickname );
95 footprint->SetFPID( id );
96 footprints.emplace_back( footprint );
97 }
98 catch( IO_ERROR& e )
99 {
100 wxLogTrace( traceLibraries, "FP: %s:%s enumeration error: %s",
101 nickname, footprintName, e.What() );
102 }
103 }
104
105 {
106 std::unique_lock lock( PreloadedFootprintsMutex );
107 PreloadedFootprints.Get()[nickname] = std::move( footprints );
108 }
109
110 // Clear the plugin's FP_CACHE now that we've copied footprints to PreloadedFootprints.
111 // This eliminates the double-caching that was consuming ~1.2GB of extra RAM.
112 plugin->ClearCachedFootprints( uri );
113}
114
115
116std::optional<LIB_STATUS> FOOTPRINT_LIBRARY_ADAPTER::LoadOne( LIB_DATA* aLib )
117{
119
120 std::map<std::string, UTF8> options = aLib->row->GetOptionsMap();
121
122 try
123 {
124 wxArrayString dummyList;
125 pcbplugin( aLib )->FootprintEnumerate( dummyList, getUri( aLib->row ), false, &options );
127 }
128 catch( IO_ERROR& e )
129 {
131 aLib->status.error = LIBRARY_ERROR( { e.What() } );
132 wxLogTrace( traceLibraries, "FP: %s: plugin threw exception: %s", aLib->row->Nickname(), e.What() );
133 }
134
135 return aLib->status;
136}
137
138
139std::optional<LIB_STATUS> FOOTPRINT_LIBRARY_ADAPTER::LoadOne( const wxString& nickname )
140{
142
143 if( result.has_value() )
144 return LoadOne( *result );
145
146 return LIB_STATUS{
147 .load_status = LOAD_STATUS::LOAD_ERROR,
148 .error = LIBRARY_ERROR( { result.error() } )
149 };
150}
151
152
153std::vector<FOOTPRINT*> FOOTPRINT_LIBRARY_ADAPTER::GetFootprints( const wxString& aNickname, bool aBestEfforts )
154{
155 std::vector<FOOTPRINT*> footprints;
156
157 std::shared_lock lock( PreloadedFootprintsMutex );
158 auto it = PreloadedFootprints.Get().find( aNickname );
159
160 if( it == PreloadedFootprints.Get().end() )
161 return footprints;
162
163 footprints.reserve( it->second.size() );
164
165 for( const auto& fp : it->second )
166 footprints.push_back( fp.get() );
167
168 return footprints;
169}
170
171
172
173std::vector<wxString> FOOTPRINT_LIBRARY_ADAPTER::GetFootprintNames( const wxString& aNickname, bool aBestEfforts )
174{
175 // TODO(JE) can we kill wxArrayString in internal API?
176 wxArrayString namesAS;
177 std::vector<wxString> names;
178
179 if( std::optional<const LIB_DATA*> maybeLib = fetchIfLoaded( aNickname ) )
180 {
181 const LIB_DATA* lib = *maybeLib;
182 std::map<std::string, UTF8> options = lib->row->GetOptionsMap();
183
184 try
185 {
186 pcbplugin( lib )->FootprintEnumerate( namesAS, getUri( lib->row ), true, &options );
187 }
188 catch( IO_ERROR& e )
189 {
190 wxLogTrace( traceLibraries, "FP: Exception enumerating library %s: %s", lib->row->Nickname(), e.What() );
191 }
192 }
193
194 for( const wxString& name : namesAS )
195 names.emplace_back( name );
196
197 return names;
198}
199
200
201long long FOOTPRINT_LIBRARY_ADAPTER::GenerateTimestamp( const wxString* aNickname )
202{
203 long long hash = 0;
204
205 if( aNickname )
206 {
207 wxCHECK( HasLibrary( *aNickname, true ), hash );
208
209 if( std::optional<const LIB_DATA*> r = fetchIfLoaded( *aNickname ); r.has_value() )
210 {
211 PCB_IO* plugin = dynamic_cast<PCB_IO*>( ( *r )->plugin.get() );
212 wxCHECK( plugin, hash );
213 return plugin->GetLibraryTimestamp( LIBRARY_MANAGER::GetFullURI( ( *r )->row, true ) )
214 + wxHashTable::MakeKey( *aNickname );
215 }
216 }
217
218 for( const wxString& nickname : GetLibraryNames() )
219 {
220 if( std::optional<const LIB_DATA*> r = fetchIfLoaded( nickname ); r.has_value() )
221 {
222 wxCHECK2( ( *r )->plugin->IsPCB_IO(), continue );
223 PCB_IO* plugin = static_cast<PCB_IO*>( ( *r )->plugin.get() );
224 hash += plugin->GetLibraryTimestamp( LIBRARY_MANAGER::GetFullURI( ( *r )->row, true ) )
225 + wxHashTable::MakeKey( nickname );
226 }
227 }
228
229 return hash;
230}
231
232
233bool FOOTPRINT_LIBRARY_ADAPTER::FootprintExists( const wxString& aNickname, const wxString& aName )
234{
235 if( std::optional<const LIB_DATA*> maybeLib = fetchIfLoaded( aNickname ) )
236 {
237 const LIB_DATA* lib = *maybeLib;
238 std::map<std::string, UTF8> options = lib->row->GetOptionsMap();
239 return pcbplugin( lib )->FootprintExists( getUri( lib->row ), aName, &options );
240 }
241
242 return false;
243}
244
245
246FOOTPRINT* FOOTPRINT_LIBRARY_ADAPTER::LoadFootprint( const wxString& aNickname, const wxString& aName, bool aKeepUUID )
247{
248 // First check if the footprint is in PreloadedFootprints and clone from there.
249 // This avoids re-parsing the file and keeps FP_CACHE from being repopulated.
250 {
251 std::shared_lock lock( PreloadedFootprintsMutex );
252 auto libIt = PreloadedFootprints.Get().find( aNickname );
253
254 if( libIt != PreloadedFootprints.Get().end() )
255 {
256 for( const auto& fp : libIt->second )
257 {
258 if( fp->GetFPID().GetLibItemName() == UTF8( aName ) )
259 {
261
262 if( aKeepUUID )
263 copy = static_cast<FOOTPRINT*>( fp->Clone() );
264 else
265 copy = static_cast<FOOTPRINT*>( fp->Duplicate( IGNORE_PARENT_GROUP ) );
266
267 copy->SetParent( nullptr );
268 return copy;
269 }
270 }
271 }
272 }
273
274 // Footprint not found in PreloadedFootprints, fall back to plugin.
275 // This re-parses the file but is needed for footprints not yet enumerated.
276 if( std::optional<const LIB_DATA*> lib = fetchIfLoaded( aNickname ) )
277 {
278 try
279 {
280 if( FOOTPRINT* footprint = pcbplugin( *lib )->FootprintLoad( getUri( ( *lib )->row ), aName, aKeepUUID ) )
281 {
282 LIB_ID id = footprint->GetFPID();
283 id.SetLibNickname( ( *lib )->row->Nickname() );
284 footprint->SetFPID( id );
285 return footprint;
286 }
287 }
288 catch( const IO_ERROR& ioe )
289 {
290 wxLogTrace( traceLibraries, "LoadFootprint: error loading %s:%s: %s", aNickname, aName, ioe.What() );
291 }
292 }
293 else
294 {
295 wxLogTrace( traceLibraries, "LoadFootprint: requested library %s not loaded", aNickname );
296 }
297
298 return nullptr;
299}
300
301
303{
304 wxString nickname = aFootprintId.GetLibNickname();
305 wxString footprintName = aFootprintId.GetLibItemName();
306
307 if( nickname.size() )
308 return LoadFootprint( nickname, footprintName, aKeepUUID );
309
310 // nickname is empty, sequentially search (alphabetically) all libs/nicks for first match:
311 for( const wxString& library : GetLibraryNames() )
312 {
313 // FootprintLoad() returns NULL on not found, does not throw exception
314 // unless there's an IO_ERROR.
315 if( FOOTPRINT* ret = LoadFootprint( library, footprintName, aKeepUUID ) )
316 return ret;
317 }
318
319 return nullptr;
320}
321
322
324 const FOOTPRINT* aFootprint,
325 bool aOverwrite )
326{
327 wxCHECK( aFootprint, SAVE_SKIPPED );
328
329 if( std::optional<const LIB_DATA*> lib = fetchIfLoaded( aNickname ) )
330 {
331 if( !aOverwrite )
332 {
333 wxString fpname = aFootprint->GetFPID().GetLibItemName();
334
335 try
336 {
337 FOOTPRINT* existing = pcbplugin( *lib )->FootprintLoad( getUri( ( *lib )->row ), fpname, false );
338
339 if( existing )
340 {
341 delete existing;
342 return SAVE_SKIPPED;
343 }
344 }
345 catch( IO_ERROR& e )
346 {
347 wxLogTrace( traceLibraries, "SaveFootprint: error checking for existing footprint %s: %s",
348 aFootprint->GetFPIDAsString(), e.What() );
349 return SAVE_SKIPPED;
350 }
351 }
352
353 try
354 {
355 pcbplugin( *lib )->FootprintSave( getUri( ( *lib )->row ), aFootprint );
356 }
357 catch( IO_ERROR& e )
358 {
359 wxLogTrace( traceLibraries, "SaveFootprint: error saving %s: %s",
360 aFootprint->GetFPIDAsString(), e.What() );
361 return SAVE_SKIPPED;
362 }
363
364 return SAVE_OK;
365 }
366 else
367 {
368 wxLogTrace( traceLibraries, "SaveFootprint: requested library %s not loaded", aNickname );
369 return SAVE_SKIPPED;
370 }
371}
372
373void FOOTPRINT_LIBRARY_ADAPTER::DeleteFootprint( const wxString& aNickname, const wxString& aFootprintName )
374{
375 if( std::optional<const LIB_DATA*> lib = fetchIfLoaded( aNickname ) )
376 {
377 try
378 {
379 pcbplugin( *lib )->FootprintDelete( getUri( ( *lib )->row ), aFootprintName );
380 }
381 catch( IO_ERROR& e )
382 {
383 wxLogTrace( traceLibraries, "DeleteFootprint: error deleting %s:%s: %s", aNickname,
384 aFootprintName, e.What() );
385 }
386 }
387 else
388 {
389 wxLogTrace( traceLibraries, "DeleteFootprint: requested library %s not loaded", aNickname );
390 }
391}
392
393
395{
396 {
397 std::shared_lock lock( m_librariesMutex );
398
399 if( auto it = m_libraries.find( aLib ); it != m_libraries.end() )
400 return it->second.plugin->IsLibraryWritable( getUri( it->second.row ) );
401 }
402
403 {
404 std::shared_lock lock( GlobalLibraryMutex );
405
406 if( auto it = GlobalLibraries.Get().find( aLib ); it != GlobalLibraries.Get().end() )
407 return it->second.plugin->IsLibraryWritable( getUri( it->second.row ) );
408 }
409
410 return false;
411}
412
413
415{
417
418 if( type == PCB_IO_MGR::NESTED_TABLE )
419 {
420 wxString msg;
421 wxFileName fileName( row->URI() );
422
423 if( fileName.FileExists() )
424 return tl::unexpected( LIBRARY_TABLE_OK() );
425 else
426 msg = wxString::Format( _( "Nested table '%s' not found." ), row->URI() );
427
428 return tl::unexpected( LIBRARY_ERROR( msg ) );
429 }
430 else if( type == PCB_IO_MGR::PCB_FILE_UNKNOWN )
431 {
432 wxLogTrace( traceLibraries, "FP: Plugin type %s is unknown!", row->Type() );
433 wxString msg = wxString::Format( _( "Unknown library type %s " ), row->Type() );
434 return tl::unexpected( LIBRARY_ERROR( msg ) );
435 }
436
438 wxCHECK( plugin, tl::unexpected( LIBRARY_ERROR( _( "Internal error" ) ) ) );
439
440 wxLogTrace( traceLibraries, "FP: Library %s (%s) plugin created", row->Nickname(),
441 magic_enum::enum_name( row->Scope() ) );
442
443 return plugin;
444}
445
446
448{
449 // Note: can't use dynamic_cast across compile units on Mac
450 wxCHECK( aRow->plugin->IsPCB_IO(), nullptr );
451 PCB_IO* ret = static_cast<PCB_IO*>( aRow->plugin.get() );
452 return ret;
453}
const char * name
virtual void SetParent(EDA_ITEM *aParent)
Definition eda_item.h:113
bool IsFootprintLibWritable(const wxString &aNickname)
Return true if the library given by aNickname is writable.
static std::shared_mutex GlobalLibraryMutex
void DeleteFootprint(const wxString &aNickname, const wxString &aFootprintName)
Deletes the aFootprintName from the library given by aNickname.
std::vector< FOOTPRINT * > GetFootprints(const wxString &aNickname, bool aBestEfforts=false)
Retrieves a list of footprints contained in a given loaded library.
void enumerateLibrary(LIB_DATA *aLib) override
Override in derived class to perform library-specific enumeration.
IO_BASE * plugin(const LIB_DATA *aRow) override
SAVE_T SaveFootprint(const wxString &aNickname, const FOOTPRINT *aFootprint, bool aOverwrite=true)
Write aFootprint to an existing library given by aNickname.
std::vector< wxString > GetFootprintNames(const wxString &aNickname, bool aBestEfforts=false)
Retrieves a list of footprint names contained in a given loaded library.
FOOTPRINT * LoadFootprint(const wxString &aNickname, const wxString &aName, bool aKeepUUID)
Load a FOOTPRINT having aName from the library given by aNickname.
static LEAK_AT_EXIT< std::map< wxString, LIB_DATA > > GlobalLibraries
static PCB_IO * pcbplugin(const LIB_DATA *aRow)
FOOTPRINT * LoadFootprintWithOptionalNickname(const LIB_ID &aFootprintId, bool aKeepUUID)
Load a footprint having aFootprintId with possibly an empty nickname.
LIBRARY_RESULT< IO_BASE * > createPlugin(const LIBRARY_TABLE_ROW *row) override
Creates a concrete plugin for the given row.
FOOTPRINT_LIBRARY_ADAPTER(LIBRARY_MANAGER &aManager)
long long GenerateTimestamp(const wxString *aNickname)
Generates a filesystem timestamp / hash value for library(ies)
std::optional< LIB_STATUS > LoadOne(LIB_DATA *aLib) override
Loads or reloads the given library, if it exists.
SAVE_T
The set of return values from SaveSymbol() below.
static LEAK_AT_EXIT< std::map< wxString, std::vector< std::unique_ptr< FOOTPRINT > > > > PreloadedFootprints
Storage for preloaded footprints, indexed by library nickname.
bool FootprintExists(const wxString &aNickname, const wxString &aName)
static std::shared_mutex PreloadedFootprintsMutex
void SetFPID(const LIB_ID &aFPID)
Definition footprint.h:352
wxString GetFPIDAsString() const
Definition footprint.h:357
const LIB_ID & GetFPID() const
Definition footprint.h:351
BOARD_ITEM * Duplicate(bool addToParentGroup, BOARD_COMMIT *aCommit=nullptr) const override
Create a copy of this BOARD_ITEM.
virtual bool IsPCB_IO() const
Work-around for lack of dynamic_cast across compile units on Mac.
Definition io_base.h:84
Hold an error message and may be used when throwing exceptions containing meaningful error messages.
virtual const wxString What() const
A composite of Problem() and Where()
A wrapper for static data that should not be destroyed at program exit.
LIBRARY_MANAGER_ADAPTER(LIBRARY_MANAGER &aManager)
Constructs a type-specific adapter into the library manager.
LIBRARY_RESULT< LIB_DATA * > loadIfNeeded(const wxString &aNickname)
Fetches a loaded library, triggering a load of that library if it isn't loaded yet.
std::shared_mutex m_librariesMutex
bool HasLibrary(const wxString &aNickname, bool aCheckEnabled=false) const
Test for the existence of aNickname in the library tables.
std::map< wxString, LIB_DATA > m_libraries
std::vector< wxString > GetLibraryNames() const
Returns a list of library nicknames that are available (skips any that failed to load)
static wxString getUri(const LIBRARY_TABLE_ROW *aRow)
std::optional< const LIB_DATA * > fetchIfLoaded(const wxString &aNickname) const
std::optional< wxString > GetFullURI(LIBRARY_TABLE_TYPE aType, const wxString &aNickname, bool aSubstituted=false) const
Return the full location specifying URI for the LIB, either in original UI form or in environment var...
LIBRARY_TABLE_SCOPE Scope() const
std::map< std::string, UTF8 > GetOptionsMap() const
const wxString & Type() const
const wxString & URI() const
const wxString & Nickname() const
A logical library item identifier and consists of various portions much like a URI.
Definition lib_id.h:49
int SetLibNickname(const UTF8 &aLibNickname)
Override the logical library name portion of the LIB_ID to aLibNickname.
Definition lib_id.cpp:100
const UTF8 & GetLibItemName() const
Definition lib_id.h:102
const UTF8 & GetLibNickname() const
Return the logical library name portion of a LIB_ID.
Definition lib_id.h:87
static PCB_FILE_T EnumFromStr(const wxString &aFileType)
Return the PCB_FILE_T from the corresponding plugin type name: "kicad", "legacy", etc.
PCB_FILE_T
The set of file types that the PCB_IO_MGR knows about, and for which there has been a plugin written,...
Definition pcb_io_mgr.h:56
@ PCB_FILE_UNKNOWN
0 is not a legal menu id on Mac
Definition pcb_io_mgr.h:57
static PCB_IO * FindPlugin(PCB_FILE_T aFileType)
Return a #PLUGIN which the caller can use to import, export, save, or load design documents.
A base class that BOARD loading and saving plugins should derive from.
Definition pcb_io.h:70
virtual void FootprintEnumerate(wxArrayString &aFootprintNames, const wxString &aLibraryPath, bool aBestEfforts, const std::map< std::string, UTF8 > *aProperties=nullptr)
Return a list of footprint names contained within the library at aLibraryPath.
Definition pcb_io.cpp:95
virtual bool FootprintExists(const wxString &aLibraryPath, const wxString &aFootprintName, const std::map< std::string, UTF8 > *aProperties=nullptr)
Check for the existence of a footprint.
Definition pcb_io.cpp:136
virtual void FootprintDelete(const wxString &aLibraryPath, const wxString &aFootprintName, const std::map< std::string, UTF8 > *aProperties=nullptr)
Delete aFootprintName from the library at aLibraryPath.
Definition pcb_io.cpp:160
virtual FOOTPRINT * FootprintLoad(const wxString &aLibraryPath, const wxString &aFootprintName, bool aKeepUUID=false, const std::map< std::string, UTF8 > *aProperties=nullptr)
Load a footprint having aFootprintName from the aLibraryPath containing a library format that this PC...
Definition pcb_io.cpp:144
virtual void FootprintSave(const wxString &aLibraryPath, const FOOTPRINT *aFootprint, const std::map< std::string, UTF8 > *aProperties=nullptr)
Write aFootprint to an existing library located at aLibraryPath.
Definition pcb_io.cpp:152
An 8 bit string that is assuredly encoded in UTF8, and supplies special conversion support to and fro...
Definition utf8.h:71
#define _(s)
#define IGNORE_PARENT_GROUP
Definition eda_item.h:55
Functions related to environment variables, including help functions.
const wxChar *const traceLibraries
Flag to enable library table and library manager tracing.
tl::expected< ResultType, LIBRARY_ERROR > LIBRARY_RESULT
KICOMMON_API wxString GetVersionedEnvVarName(const wxString &aBaseName)
Construct a versioned environment variable based on this KiCad major version.
Definition env_vars.cpp:77
Storage for an actual loaded library (including library content owned by the plugin)
LIB_STATUS status
std::unique_ptr< IO_BASE > plugin
const LIBRARY_TABLE_ROW * row
The overall status of a loaded or loading library.
std::optional< LIBRARY_ERROR > error
LOAD_STATUS load_status
wxString result
Test unit parsing edge cases and error handling.
wxLogTrace helper definitions.