KiCad PCB EDA Suite
Loading...
Searching...
No Matches
library_manager.h
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
18 * along with this program. If not, see <https://www.gnu.org/licenses/>.
19 */
20
21#ifndef LIBRARY_MANAGER_H
22#define LIBRARY_MANAGER_H
23
24#include <future>
25#include <memory>
26#include <shared_mutex>
27
28#include <kicommon.h>
30#include <io/io_base.h>
31
32
33class LIBRARY_MANAGER;
35class PROJECT;
36
37
46
49{
51 std::optional<LIBRARY_ERROR> error;
52};
53
54
57{
58 std::unique_ptr<IO_BASE> plugin;
59 const LIBRARY_TABLE_ROW* row = nullptr;
61
62 // For an entry in the process-wide globalLibs() cache, the adapter that loaded it and so owns
63 // the lifetime of row. Its destructor evicts the entry; identity is by adapter, not by the
64 // reusable row pointer.
66
67 int modify_hash = -1;
68 std::vector<wxString> available_fields_cache;
69};
70
71
77{
78public:
87
89
90 LIBRARY_MANAGER& Manager() const;
91
93 virtual LIBRARY_TABLE_TYPE Type() const = 0;
94
97
99 std::optional<LIBRARY_TABLE*> ProjectTable() const;
100
101 std::optional<wxString> FindLibraryByURI( const wxString& aURI ) const;
102
104 std::vector<wxString> GetLibraryNames() const;
105
112 bool HasLibrary( const wxString& aNickname, bool aCheckEnabled = false ) const;
113
115 bool DeleteLibrary( const wxString& aNickname );
116
117 std::optional<wxString> GetLibraryDescription( const wxString& aNickname ) const;
118
120 std::vector<LIBRARY_TABLE_ROW*> Rows( LIBRARY_TABLE_SCOPE aScope = LIBRARY_TABLE_SCOPE::BOTH,
121 bool aIncludeInvalid = false ) const;
122
124 std::optional<LIBRARY_TABLE_ROW*> GetRow( const wxString& aNickname,
126
128 std::optional<LIBRARY_TABLE_ROW*> FindRowByURI( const wxString& aUri,
130
132 virtual void ProjectChanged();
133
135 void GlobalTablesChanged( std::initializer_list<LIBRARY_TABLE_TYPE> aChangedTables = {} );
136
143 void ProjectTablesChanged( std::initializer_list<LIBRARY_TABLE_TYPE> aChangedTables = {} );
144
149 void ProjectTablesReloaded( std::initializer_list<LIBRARY_TABLE_TYPE> aChangedTables = {} );
150
151 void CheckTableRow( LIBRARY_TABLE_ROW& aRow );
152
154 void AsyncLoad();
155
156 virtual std::optional<LIB_STATUS> LoadOne( LIB_DATA* aLib ) = 0;
157
159 std::optional<float> AsyncLoadProgress() const;
160
161 void BlockUntilLoaded();
162
165 void AbortAsyncLoad();
166
167 bool IsLibraryLoaded( const wxString& aNickname );
168
170 std::optional<LIB_STATUS> GetLibraryStatus( const wxString& aNickname ) const;
171
173 std::vector<std::pair<wxString, LIB_STATUS>> GetLibraryStatuses() const;
174
176 wxString GetLibraryLoadErrors() const;
177
178 void ReloadLibraryEntry( const wxString& aNickname,
180
183 std::optional<LIB_STATUS> LoadLibraryEntry( const wxString& aNickname );
184
186 virtual bool IsWritable( const wxString& aNickname ) const;
187
189 bool CreateLibrary( const wxString& aNickname );
190
191 virtual bool SupportsConfigurationDialog( const wxString& aNickname ) const { return false; }
192
193 virtual void ShowConfigurationDialog( const wxString& aNickname, wxWindow* aParent ) const {};
194
195 virtual std::optional<LIBRARY_ERROR> LibraryError( const wxString& aNickname ) const;
196
197protected:
198 virtual std::map<wxString, LIB_DATA>& globalLibs() = 0;
199 virtual std::map<wxString, LIB_DATA>& globalLibs() const = 0;
200 virtual std::shared_mutex& globalLibsMutex() = 0;
201 virtual std::shared_mutex& globalLibsMutex() const = 0;
202
206 virtual void enumerateLibrary( LIB_DATA* aLib, const wxString& aUri ) = 0;
207
208 static wxString getUri( const LIBRARY_TABLE_ROW* aRow );
209
210 std::optional<const LIB_DATA*> fetchIfLoaded( const wxString& aNickname ) const;
211
212 std::optional<LIB_DATA*> fetchIfLoaded( const wxString& aNickname );
213
215 LIBRARY_RESULT<LIB_DATA*> loadIfNeeded( const wxString& aNickname );
216
217 LIBRARY_RESULT<LIB_DATA*> loadFromScope( const wxString& aNickname,
218 LIBRARY_TABLE_SCOPE aScope,
219 std::map<wxString, LIB_DATA>& aTarget,
220 std::shared_mutex& aMutex );
221
228
230 void abortLoad();
231
239 void resetProjectCache();
240
243
244 virtual IO_BASE* plugin( const LIB_DATA* aRow ) = 0;
245
247
248 // The actual library content is held in an associated IO plugin
249 std::map<wxString, LIB_DATA> m_libraries;
250
251 mutable std::shared_mutex m_librariesMutex;
252
253 std::atomic_bool m_abort;
254 std::vector<std::future<void>> m_futures;
255
256 std::atomic<size_t> m_loadCount{ 0 };
257 std::atomic<size_t> m_loadTotal{ 0 };
258 std::mutex m_loadMutex;
259};
260
261
263{
264public:
266
268
271
272 static wxString DefaultGlobalTablePath( LIBRARY_TABLE_TYPE aType );
273
274 static wxString StockTablePath( LIBRARY_TABLE_TYPE aType );
275
283 static wxString StockTableTokenizedURI( LIBRARY_TABLE_TYPE aType );
284
295 static wxString StockTableReferenceURI( LIBRARY_TABLE_TYPE aType );
296
297 static bool IsTableValid( const wxString& aPath );
298
300 static bool GlobalTablesValid();
301
303 static std::vector<LIBRARY_TABLE_TYPE> InvalidGlobalTables();
304
305 static bool CreateGlobalTable( LIBRARY_TABLE_TYPE aType, bool aPopulateDefaultLibraries );
306
308 void LoadGlobalTables( std::initializer_list<LIBRARY_TABLE_TYPE> aTablesToLoad = {} );
309
311 void LoadProjectTables( std::initializer_list<LIBRARY_TABLE_TYPE> aTablesToLoad = {} );
312
313 void ReloadTables( LIBRARY_TABLE_SCOPE aScope, std::initializer_list<LIBRARY_TABLE_TYPE> aTablesToLoad = {} );
314
316 void ProjectChanged();
317
321 void AbortAsyncLoads();
322
323 void RegisterAdapter( LIBRARY_TABLE_TYPE aType,
324 std::unique_ptr<LIBRARY_MANAGER_ADAPTER>&& aAdapter );
325
326 bool RemoveAdapter( LIBRARY_TABLE_TYPE aType, LIBRARY_MANAGER_ADAPTER* aAdapter );
327
328 std::optional<LIBRARY_MANAGER_ADAPTER*> Adapter( LIBRARY_TABLE_TYPE aType ) const;
329
338 std::optional<LIBRARY_TABLE*> Table( LIBRARY_TABLE_TYPE aType,
339 LIBRARY_TABLE_SCOPE aScope );
340
348 std::vector<LIBRARY_TABLE_ROW*> Rows( LIBRARY_TABLE_TYPE aType,
350 bool aIncludeInvalid = false ) const;
351
359 std::optional<LIBRARY_TABLE_ROW*> GetRow( LIBRARY_TABLE_TYPE aType, const wxString &aNickname,
361
362 std::optional<LIBRARY_TABLE_ROW*> FindRowByURI( LIBRARY_TABLE_TYPE aType, const wxString &aUri,
364
365 void ReloadLibraryEntry( LIBRARY_TABLE_TYPE aType, const wxString& aNickname,
367
370 std::optional<LIB_STATUS> LoadLibraryEntry( LIBRARY_TABLE_TYPE aType,
371 const wxString& aNickname );
372
373 void LoadProjectTables( const wxString& aProjectPath,
374 std::initializer_list<LIBRARY_TABLE_TYPE> aTablesToLoad = {} );
375
385 std::optional<wxString> GetFullURI( LIBRARY_TABLE_TYPE aType, const wxString& aNickname,
386 bool aSubstituted = false );
387
388 static wxString GetFullURI( const LIBRARY_TABLE_ROW* aRow, bool aSubstituted = false );
389
390 static wxString ExpandURI( const wxString& aShortURI, const PROJECT& aProject );
391
392 static bool UrisAreEquivalent( const wxString& aURI1, const wxString& aURI2 );
393
403 static bool IsPcmManagedRow( const LIBRARY_TABLE_ROW& aRow );
404
408 void ApplyLibOverrides( LIBRARY_TABLE& aTable );
409
418 void SetLibOverride( const wxString& aTablePath, const wxString& aNickname,
419 bool aDisabled, bool aHidden );
420
422 void ClearLibOverride( const wxString& aTablePath, const wxString& aNickname );
423
424private:
425 void loadTables( const wxString& aTablePath, LIBRARY_TABLE_SCOPE aScope,
426 std::vector<LIBRARY_TABLE_TYPE> aTablesToLoad = {} );
427
428 void loadNestedTables( LIBRARY_TABLE& aTable );
429
431 void applyLibOverrides( LIBRARY_TABLE& aTable );
432
433 static wxString tableFileName( LIBRARY_TABLE_TYPE aType );
434
435 void createEmptyTable( LIBRARY_TABLE_TYPE aType, LIBRARY_TABLE_SCOPE aScope );
436
437 std::map<LIBRARY_TABLE_TYPE, std::unique_ptr<LIBRARY_TABLE>> m_tables;
438
440 std::map<wxString, std::unique_ptr<LIBRARY_TABLE>> m_childTables;
441
442 // TODO: support multiple projects
443 std::map<LIBRARY_TABLE_TYPE, std::unique_ptr<LIBRARY_TABLE>> m_projectTables;
444
445 std::map<LIBRARY_TABLE_TYPE, std::unique_ptr<LIBRARY_MANAGER_ADAPTER>> m_adapters;
446
447 mutable std::mutex m_adaptersMutex;
448
449 typedef std::tuple<LIBRARY_TABLE_TYPE, LIBRARY_TABLE_SCOPE, wxString> ROW_CACHE_KEY;
450
451 std::map<ROW_CACHE_KEY, LIBRARY_TABLE_ROW*> m_rowCache;
452 mutable std::mutex m_rowCacheMutex;
453};
454
455#endif //LIBRARY_MANAGER_H
The interface used by the classes that actually can load IO plugins for the different parts of KiCad ...
std::optional< float > AsyncLoadProgress() const
Returns async load progress between 0.0 and 1.0, or nullopt if load is not in progress.
virtual std::optional< LIB_STATUS > LoadOne(LIB_DATA *aLib)=0
void resetProjectCache()
Aborts pending loads and resets every project-scope cache entry in place (plugin and row cleared,...
void ReloadLibraryEntry(const wxString &aNickname, LIBRARY_TABLE_SCOPE aScope=LIBRARY_TABLE_SCOPE::BOTH)
virtual std::shared_mutex & globalLibsMutex() const =0
std::optional< LIB_STATUS > GetLibraryStatus(const wxString &aNickname) const
Returns the status of a loaded library, or nullopt if the library hasn't been loaded (yet)
std::optional< LIBRARY_TABLE * > ProjectTable() const
Retrieves the project library table for this adapter type, or nullopt if one doesn't exist.
void AbortAsyncLoad()
Aborts any async load in progress; blocks until fully done aborting.
virtual void enumerateLibrary(LIB_DATA *aLib, const wxString &aUri)=0
Override in derived class to perform library-specific enumeration.
LIBRARY_TABLE * GlobalTable() const
Retrieves the global library table for this adapter type.
virtual std::shared_mutex & globalLibsMutex()=0
void evictOwnedGlobalEntries()
Erases this adapter's own entries (LIB_DATA::global_owner == this) from the process-wide globalLibs()...
LIBRARY_MANAGER_ADAPTER(LIBRARY_MANAGER &aManager)
Constructs a type-specific adapter into the library manager.
bool IsLibraryLoaded(const wxString &aNickname)
std::vector< LIBRARY_TABLE_ROW * > Rows(LIBRARY_TABLE_SCOPE aScope=LIBRARY_TABLE_SCOPE::BOTH, bool aIncludeInvalid=false) const
Like LIBRARY_MANAGER::Rows but filtered to the LIBRARY_TABLE_TYPE of this adapter.
std::optional< LIBRARY_TABLE_ROW * > FindRowByURI(const wxString &aUri, LIBRARY_TABLE_SCOPE aScope=LIBRARY_TABLE_SCOPE::BOTH) const
Like LIBRARY_MANAGER::FindRowByURI but filtered to the LIBRARY_TABLE_TYPE of this adapter.
virtual std::map< wxString, LIB_DATA > & globalLibs()=0
virtual LIBRARY_TABLE_TYPE Type() const =0
The type of library table this adapter works with.
bool DeleteLibrary(const wxString &aNickname)
Deletes the given library from disk if it exists; returns true if deleted.
LIBRARY_RESULT< LIB_DATA * > loadIfNeeded(const wxString &aNickname)
Fetches a loaded library, triggering a load of that library if it isn't loaded yet.
wxString GetLibraryLoadErrors() const
Returns all library load errors as newline-separated strings for display.
std::optional< wxString > FindLibraryByURI(const wxString &aURI) const
std::shared_mutex m_librariesMutex
LIBRARY_MANAGER & Manager() const
void abortLoad()
Aborts any async load in progress; blocks until fully done aborting.
std::optional< wxString > GetLibraryDescription(const wxString &aNickname) const
virtual LIBRARY_RESULT< IO_BASE * > createPlugin(const LIBRARY_TABLE_ROW *row)=0
Creates a concrete plugin for the given row.
void GlobalTablesChanged(std::initializer_list< LIBRARY_TABLE_TYPE > aChangedTables={})
Notify the adapter that the global library tables have changed.
bool HasLibrary(const wxString &aNickname, bool aCheckEnabled=false) const
Test for the existence of aNickname in the library tables.
std::optional< LIBRARY_TABLE_ROW * > GetRow(const wxString &aNickname, LIBRARY_TABLE_SCOPE aScope=LIBRARY_TABLE_SCOPE::BOTH) const
Like LIBRARY_MANAGER::GetRow but filtered to the LIBRARY_TABLE_TYPE of this adapter.
virtual void ShowConfigurationDialog(const wxString &aNickname, wxWindow *aParent) const
std::map< wxString, LIB_DATA > m_libraries
virtual IO_BASE * plugin(const LIB_DATA *aRow)=0
std::vector< wxString > GetLibraryNames() const
Returns a list of library nicknames that are available (skips any that failed to load)
LIBRARY_RESULT< LIB_DATA * > loadFromScope(const wxString &aNickname, LIBRARY_TABLE_SCOPE aScope, std::map< wxString, LIB_DATA > &aTarget, std::shared_mutex &aMutex)
std::vector< std::future< void > > m_futures
virtual bool SupportsConfigurationDialog(const wxString &aNickname) const
static wxString getUri(const LIBRARY_TABLE_ROW *aRow)
bool CreateLibrary(const wxString &aNickname)
Creates the library (i.e. saves to disk) for the given row if it exists.
virtual bool IsWritable(const wxString &aNickname) const
Return true if the given nickname exists and is not a read-only library.
std::vector< std::pair< wxString, LIB_STATUS > > GetLibraryStatuses() const
Returns a list of all library nicknames and their status (even if they failed to load)
std::atomic< size_t > m_loadTotal
virtual std::map< wxString, LIB_DATA > & globalLibs() const =0
std::atomic< size_t > m_loadCount
LIBRARY_MANAGER & m_manager
std::optional< LIB_STATUS > LoadLibraryEntry(const wxString &aNickname)
Synchronously loads the named library to LOADED state.
std::optional< const LIB_DATA * > fetchIfLoaded(const wxString &aNickname) const
static wxString StockTableTokenizedURI(LIBRARY_TABLE_TYPE aType)
std::mutex m_rowCacheMutex
std::map< wxString, std::unique_ptr< LIBRARY_TABLE > > m_childTables
Map of full URI to table object for tables that are referenced by global or project tables.
std::map< ROW_CACHE_KEY, LIBRARY_TABLE_ROW * > m_rowCache
std::mutex m_adaptersMutex
static wxString DefaultGlobalTablePath(LIBRARY_TABLE_TYPE aType)
static std::vector< LIBRARY_TABLE_TYPE > InvalidGlobalTables()
static bool GlobalTablesValid()
std::map< LIBRARY_TABLE_TYPE, std::unique_ptr< LIBRARY_TABLE > > m_projectTables
LIBRARY_MANAGER(const LIBRARY_MANAGER &)=delete
static bool CreateGlobalTable(LIBRARY_TABLE_TYPE aType, bool aPopulateDefaultLibraries)
void LoadGlobalTables(std::initializer_list< LIBRARY_TABLE_TYPE > aTablesToLoad={})
(Re)loads the global library tables in the given list, or all tables if no list is given
std::map< LIBRARY_TABLE_TYPE, std::unique_ptr< LIBRARY_TABLE > > m_tables
LIBRARY_MANAGER & operator=(const LIBRARY_MANAGER &)=delete
std::tuple< LIBRARY_TABLE_TYPE, LIBRARY_TABLE_SCOPE, wxString > ROW_CACHE_KEY
static wxString StockTableReferenceURI(LIBRARY_TABLE_TYPE aType)
static wxString StockTablePath(LIBRARY_TABLE_TYPE aType)
static bool IsTableValid(const wxString &aPath)
std::map< LIBRARY_TABLE_TYPE, std::unique_ptr< LIBRARY_MANAGER_ADAPTER > > m_adapters
Container for project specific data.
Definition project.h:62
void ProjectChanged() override
#define KICOMMON_API
Definition kicommon.h:27
LOAD_STATUS
Status of a library load managed by a library adapter.
tl::expected< ResultType, LIBRARY_ERROR > LIBRARY_RESULT
LIBRARY_TABLE_TYPE
LIBRARY_TABLE_SCOPE
Storage for an actual loaded library (including library content owned by the plugin)
std::vector< wxString > available_fields_cache
LIB_STATUS status
std::unique_ptr< IO_BASE > plugin
const LIBRARY_MANAGER_ADAPTER * global_owner
const LIBRARY_TABLE_ROW * row
The overall status of a loaded or loading library.
std::optional< LIBRARY_ERROR > error
LOAD_STATUS load_status