KiCad PCB EDA Suite
Loading...
Searching...
No Matches
single_top.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 (C) 2014 SoftPLC Corporation, Dick Hollenbeck <[email protected]>
5 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
6 *
7 * This program is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU General Public License
9 * as published by the Free Software Foundation; either version 2
10 * of the License, or (at your option) any later version.
11 *
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU 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, you may find one here:
19 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
20 * or you may search the http://www.gnu.org website for the version 2 license,
21 * or you may write to the Free Software Foundation, Inc.,
22 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
23 */
24
25
26/*
27
28 This is a program launcher for a single KIFACE DSO. It only mimics a KIWAY,
29 not actually implements one, since only a single DSO is supported by it.
30
31 It is compiled multiple times, once for each standalone program and as such
32 gets different compiler command line supplied #defines from CMake.
33
34*/
35
36
37#include <typeinfo>
38#include <wx/cmdline.h>
39#include <wx/dialog.h>
40#include <wx/filename.h>
41#include <wx/stdpaths.h>
42#include <wx/snglinst.h>
43#include <wx/html/htmlwin.h>
44
45#include <kiway.h>
46#include <build_version.h>
47#include <pgm_base.h>
48#include <app_monitor.h>
49#include <kiway_player.h>
50#include <macros.h>
51#include <confirm.h>
53
57#include <paths.h>
58
59#include <kiplatform/app.h>
61
62#include <git2.h>
63#include <thread_pool.h>
64
67
68#ifdef KICAD_USE_SENTRY
69#include <sentry.h>
70#endif
71
72#ifdef KICAD_IPC_API
73#include <api/api_server.h>
74#endif
75
76// Only a single KIWAY is supported in this single_top top level component,
77// which is dedicated to loading only a single DSO.
79
80
81// implement a PGM_BASE and a wxApp side by side:
82
86static struct PGM_SINGLE_TOP : public PGM_BASE
87{
88 bool OnPgmInit();
89
90 void OnPgmExit()
91 {
92 // Abort and wait on any background jobs
93 GetKiCadThreadPool().purge();
94 GetKiCadThreadPool().wait();
95
96 Kiway.OnKiwayEnd();
97
98#ifdef KICAD_IPC_API
99 m_api_server.reset();
100#endif
101
103 {
105 m_settings_manager->Save();
106 }
107
108 // Destroy everything in PGM_BASE, especially wxSingleInstanceCheckerImpl
109 // earlier than wxApp and earlier than static destruction would.
111 git_libgit2_shutdown();
112 }
113
114 void MacOpenFile( const wxString& aFileName ) override
115 {
116 wxFileName filename( aFileName );
117
118 if( filename.FileExists() )
119 {
120 #if 0
121 // this pulls in EDA_DRAW_FRAME type info, which we don't want in
122 // the single_top link image.
123 KIWAY_PLAYER* frame = dynamic_cast<KIWAY_PLAYER*>( App().GetTopWindow() );
124 #else
125 KIWAY_PLAYER* frame = (KIWAY_PLAYER*) App().GetTopWindow();
126 #endif
127 if( frame )
128 {
129 if( wxWindow* blocking_win = frame->Kiway().GetBlockingDialog() )
130 blocking_win->Close( true );
131
132 frame->OpenProjectFiles( std::vector<wxString>( 1, aFileName ) );
133 }
134 }
135 }
136
138
139
140// A module to allow Html module initialization/cleanup
141// When a wxHtmlWindow is used *only* in a dll/so module, the Html text is displayed
142// as plain text.
143// This helper class is just used to force wxHtmlWinParser initialization
144// see https://groups.google.com/forum/#!topic/wx-users/FF0zv5qGAT0
145class HtmlModule: public wxModule
146{
147public:
149 virtual bool OnInit() override { AddDependency( CLASSINFO( wxHtmlWinParser ) ); return true; };
150 virtual void OnExit() override {};
151
152private:
154};
155
157
158
159#ifdef NDEBUG
160// Define a custom assertion handler
161void CustomAssertHandler( const wxString& file,
162 int line,
163 const wxString& func,
164 const wxString& cond,
165 const wxString& msg )
166{
167 Pgm().HandleAssert( file, line, func, cond, msg );
168}
169#endif
170
171
176struct APP_SINGLE_TOP : public wxApp
177{
178 APP_SINGLE_TOP() : wxApp()
179 {
180 SetPgm( &program );
181
182 // Init the environment each platform wants
184 }
185
186
187 bool OnInit() override
188 {
189#ifdef NDEBUG
190 // These checks generate extra assert noise
191 wxSizerFlags::DisableConsistencyChecks();
192 wxDISABLE_DEBUG_SUPPORT();
193 wxSetAssertHandler( CustomAssertHandler );
194#endif
195
196 // Perform platform-specific init tasks
197 if( !KIPLATFORM::APP::Init() )
198 return false;
199
200#ifndef DEBUG
201 // Enable logging traces to the console in release build.
202 // This is usually disabled, but it can be useful for users to run to help
203 // debug issues and other problems.
204 if( wxGetEnv( wxS( "KICAD_ENABLE_WXTRACE" ), nullptr ) )
205 {
206 wxLog::EnableLogging( true );
207 wxLog::SetLogLevel( wxLOG_Trace );
208 }
209#endif
210
211 // Force wxHtmlWinParser initialization when a wxHtmlWindow is used only
212 // in a shared library (.so or .dll file)
213 // Otherwise the Html text is displayed as plain text.
214 HtmlModule html_init;
215
216 try
217 {
218 return program.OnPgmInit();
219 }
220 catch( ... )
221 {
222 Pgm().HandleException( std::current_exception() );
223 }
224
225 program.OnPgmExit();
226
227 return false;
228 }
229
230 int OnExit() override
231 {
232 program.OnPgmExit();
233 return wxApp::OnExit();
234 }
235
236 int OnRun() override
237 {
238 int ret = -1;
239
240 try
241 {
242 ret = wxApp::OnRun();
243 }
244 catch(...)
245 {
246 Pgm().HandleException( std::current_exception() );
247 }
248
249 return ret;
250 }
251
252 int FilterEvent( wxEvent& aEvent ) override
253 {
254 if( aEvent.GetEventType() == wxEVT_SHOW )
255 {
256 wxShowEvent& event = static_cast<wxShowEvent&>( aEvent );
257 wxDialog* dialog = dynamic_cast<wxDialog*>( event.GetEventObject() );
258
259 std::vector<void*>& dlgs = Pgm().m_ModalDialogs;
260
261 if( dialog )
262 {
263 if( event.IsShown() && dialog->IsModal() )
264 {
265 dlgs.push_back( dialog );
266 }
267 // Under GTK, sometimes the modal flag is cleared before hiding
268 else if( !event.IsShown() && !dlgs.empty() )
269 {
270 // If we close the expected dialog, remove it from our stack
271 if( dlgs.back() == dialog )
272 dlgs.pop_back();
273 // If an out-of-order, remove all dialogs added after the closed one
274 else if( auto it = std::find( dlgs.begin(), dlgs.end(), dialog ); it != dlgs.end() )
275 dlgs.erase( it, dlgs.end() );
276 }
277 }
278 }
279
280 return Event_Skip;
281 }
282
283 void OnUnhandledException() override
284 {
285 Pgm().HandleException( std::current_exception(), true );
286 }
287
288#if defined( DEBUG )
296 virtual bool OnExceptionInMainLoop() override
297 {
298 try
299 {
300 throw;
301 }
302 catch( ... )
303 {
304 Pgm().HandleException( std::current_exception() );
305 }
306
307 return false; // continue on. Return false to abort program
308 }
309#endif
310
311#ifdef __WXMAC__
312
320 void MacOpenFile( const wxString& aFileName ) override
321 {
322 Pgm().MacOpenFile( aFileName );
323 }
324
325#endif
326};
327
328IMPLEMENT_APP( APP_SINGLE_TOP )
329
330
332{
333#if defined(DEBUG)
334 wxString absoluteArgv0 = wxStandardPaths::Get().GetExecutablePath();
335
336 if( !wxIsAbsolutePath( absoluteArgv0 ) )
337 {
338 wxLogError( wxT( "No meaningful argv[0]" ) );
339 return false;
340 }
341#endif
342
343 // Initialize the git library before trying to initialize individual programs
344 int gitInit = git_libgit2_init();
345
346 if( gitInit < 0 )
347 {
348 const git_error* err = git_error_last();
349 wxString msg = wxS( "Failed to initialize git library" );
350
351 if( err && err->message )
352 msg += wxS( ": " ) + wxString::FromUTF8( err->message );
353
354 wxLogError( msg );
355 return false;
356 }
357
358 if( !InitPgm( false ) )
359 {
360 // Clean up
361 OnPgmExit();
362 return false;
363 }
364
365#if !defined(BUILD_KIWAY_DLL)
366
367 // Only bitmap2component and pcb_calculator use this code currently, as they
368 // are not split to use single_top as a link image separate from a *.kiface.
369 // i.e. they are single part link images so don't need to load a *.kiface.
370
371 // Get the getter, it is statically linked into this binary image.
372 KIFACE_GETTER_FUNC* ki_getter = &KIFACE_GETTER;
373
374 int kiface_version;
375
376 // Get the KIFACE.
377 KIFACE* kiface = ki_getter( &kiface_version, KIFACE_VERSION, this );
378
379 // Trick the KIWAY into thinking it loaded a KIFACE, by recording the KIFACE
380 // in the KIWAY. It needs to be there for KIWAY::OnKiwayEnd() anyways.
381 Kiway.set_kiface( KIWAY::KifaceType( TOP_FRAME ), kiface );
382#endif
383
384 // Tell the settings manager about the current Kiway
385 GetSettingsManager().SetKiway( &Kiway );
386
387 GetSettingsManager().RegisterSettings( new KICAD_SETTINGS );
388
389
390 if( const COMMON_SETTINGS* cfg = Pgm().GetCommonSettings() )
391 {
392 if( cfg->m_Appearance.app_theme == APP_THEME::DARK )
394 else if( cfg->m_Appearance.app_theme == APP_THEME::AUTO )
396 }
397
398#ifdef KICAD_IPC_API
399 // Create the API server thread once the app event loop exists
400 m_api_server = std::make_unique<KICAD_API_SERVER>();
401#endif
402
403 // Use KIWAY to create a top window, which registers its existence also.
404 // "TOP_FRAME" is a macro that is passed on compiler command line from CMake,
405 // and is one of the types in FRAME_T.
406 KIWAY_PLAYER* frame = Kiway.Player( TOP_FRAME, true );
407
408 if( frame == nullptr )
409 {
410 // Clean up
411 OnPgmExit();
412 return false;
413 }
414
415 Kiway.SetTop( frame );
416
417 STARTWIZARD startWizard;
418 startWizard.CheckAndRun( frame );
419
420 // Load library tables after startup wizard
421 GetLibraryManager().LoadGlobalTables();
422
423 // Preload libraries, since for single-top this won't have been done earlier
424 if( KIFACE* topFrame = Kiway.KiFACE( KIWAY::KifaceType( TOP_FRAME ) ) )
425 topFrame->PreloadLibraries( &Kiway );
426
428
429 App().SetTopWindow( frame ); // wxApp gets a face.
430 App().SetAppDisplayName( frame->GetAboutTitle() );
431
432 wxString relaunchDisplayName = frame->GetAboutTitle() + " " + GetMajorMinorVersion();
434
435 // Allocate a slice of time to show the frame and update wxWidgets widgets
436 // (especially setting valid sizes) after creating frame and before calling
437 // OpenProjectFiles() that can update/use some widgets.
438 // The 2 calls to wxSafeYield are needed on wxGTK for best results.
439 wxSafeYield();
440 HideSplash();
441 frame->Show();
442 wxSafeYield();
443
444 // Now after the frame processing, the rest of the positional args are files
445 std::vector<wxString> fileArgs;
446
447
448 static const wxCmdLineEntryDesc desc[] = {
449 { wxCMD_LINE_PARAM, nullptr, nullptr, "File to load", wxCMD_LINE_VAL_STRING,
450 wxCMD_LINE_PARAM_MULTIPLE | wxCMD_LINE_PARAM_OPTIONAL },
451 { wxCMD_LINE_NONE, nullptr, nullptr, nullptr, wxCMD_LINE_VAL_NONE, 0 }
452 };
453
454 wxCmdLineParser parser( App().argc, App().argv );
455 parser.SetDesc( desc );
456 parser.Parse( false );
457
458 if( parser.GetParamCount() )
459 {
460 /*
461 gerbview handles multiple project data files, i.e. gerber files on
462 cmd line. Others currently do not, they handle only one. For common
463 code simplicity we simply pass all the arguments in however, each
464 program module can do with them what they want, ignore, complain
465 whatever. We don't establish policy here, as this is a multi-purpose
466 launcher.
467 */
468
469 for( size_t i = 0; i < parser.GetParamCount(); i++ )
470 fileArgs.push_back( parser.GetParam( i ) );
471
472 // special attention to a single argument: argv[1] (==argSet[0])
473 if( fileArgs.size() == 1 )
474 {
475 wxFileName argv1( fileArgs[0] );
476
477#if defined(PGM_DATA_FILE_EXT)
478 // PGM_DATA_FILE_EXT, if present, may be different for each compile,
479 // it may come from CMake on the compiler command line, but often does not.
480 // This facility is mostly useful for those program footprints
481 // supporting a single argv[1].
482 if( !argv1.GetExt() )
483 argv1.SetExt( wxT( PGM_DATA_FILE_EXT ) );
484#endif
485 argv1.MakeAbsolute();
486
487 fileArgs[0] = argv1.GetFullPath();
488 }
489
490 frame->OpenProjectFiles( fileArgs );
491 }
492
493#ifdef KICAD_IPC_API
494 m_api_server->SetReadyToReply();
495#endif
496
497 return true;
498}
wxString GetMajorMinorVersion()
Get only the major and minor version in a string major.minor.
wxString GetAboutTitle() const
virtual bool OnInit() override
virtual void OnExit() override
wxDECLARE_DYNAMIC_CLASS(HtmlModule)
KIWAY & Kiway() const
Return a reference to the KIWAY that this object has an opportunity to participate in.
A wxFrame capable of the OpenProjectFiles function, meaning it can load a portion of a KiCad project.
virtual bool OpenProjectFiles(const std::vector< wxString > &aFileList, int aCtl=0)
Open a project or set of files given by aFileList.
A minimalistic software bus for communications between various DLLs/DSOs (DSOs) within the same KiCad...
Definition kiway.h:315
static FACE_T KifaceType(FRAME_T aFrameType)
A simple mapping function which returns the FACE_T which is known to implement aFrameType.
Definition kiway.cpp:343
wxWindow * GetBlockingDialog()
Gets the window pointer to the blocking dialog (to send it signals)
Definition kiway.cpp:688
static const wxString & GetExecutablePath()
Definition paths.cpp:671
virtual COMMON_SETTINGS * GetCommonSettings() const
Definition pgm_base.cpp:541
virtual wxApp & App()
Return a bare naked wxApp which may come from wxPython, SINGLE_TOP, or kicad.exe.
Definition pgm_base.cpp:211
void PreloadDesignBlockLibraries(KIWAY *aKiway)
Starts a background job to preload the global and project design block libraries.
Definition pgm_base.cpp:887
virtual void MacOpenFile(const wxString &aFileName)=0
Specific to MacOSX (not used under Linux or Windows).
std::unique_ptr< SETTINGS_MANAGER > m_settings_manager
Definition pgm_base.h:411
void Destroy()
Definition pgm_base.cpp:190
void HandleException(std::exception_ptr aPtr, bool aUnhandled=false)
A exception handler to be used at the top level if exceptions bubble up that for.
Definition pgm_base.cpp:802
std::vector< void * > m_ModalDialogs
Definition pgm_base.h:392
bool InitPgm(bool aHeadless=false, bool aIsUnitTest=false)
Initialize this program.
Definition pgm_base.cpp:327
void HandleAssert(const wxString &aFile, int aLine, const wxString &aFunc, const wxString &aCond, const wxString &aMsg)
A common assert handler to be used between single_top and kicad.
Definition pgm_base.cpp:839
virtual SETTINGS_MANAGER & GetSettingsManager() const
Definition pgm_base.h:130
void HideSplash()
Definition pgm_base.cpp:316
virtual LIBRARY_MANAGER & GetLibraryManager() const
Definition pgm_base.h:132
void SaveCommonSettings()
Save the program (process) settings subset which are stored .kicad_common.
Definition pgm_base.cpp:532
void CheckAndRun(wxWindow *parent)
This file is part of the common library.
KIFACE * KIFACE_GETTER_FUNC(int *aKIFACEversion, int aKIWAYversion, PGM_BASE *aProgram)
Point to the one and only KIFACE export.
Definition kiway.h:553
#define KIFACE_GETTER
Definition kiway.h:113
#define KIFACE_VERSION
Definition kiway.h:112
#define KFCTL_STANDALONE
Running as a standalone Top.
Definition kiway.h:163
This file contains miscellaneous commonly used macros and functions.
bool Init()
Perform application-specific initialization tasks.
Definition unix/app.cpp:40
void EnableDarkMode(bool aForce)
Definition unix/app.cpp:58
void Init()
Perform environment initialization tasks.
void SetAppDetailsForWindow(wxWindow *aWindow, const wxString &aRelaunchCommand, const wxString &aRelaunchDisplayName)
Sets the relaunch command for taskbar pins, this is intended for Windows.
void SetPgm(PGM_BASE *pgm)
PGM_BASE & Pgm()
The global program "get" accessor.
see class PGM_BASE
PGM_SINGLE_TOP program
wxIMPLEMENT_DYNAMIC_CLASS(HtmlModule, wxModule)
KIWAY Kiway(KFCTL_STANDALONE)
Implement a bare naked wxApp (so that we don't become dependent on functionality in a wxApp derivativ...
int OnRun() override
int OnExit() override
void OnUnhandledException() override
bool OnInit() override
int FilterEvent(wxEvent &aEvent) override
Implement a participant in the KIWAY alchemy.
Definition kiway.h:156
Implement PGM_BASE with its own OnPgmInit() and OnPgmExit().
void MacOpenFile(const wxString &aFileName) override
Specific to MacOSX (not used under Linux or Windows).
IFACE KIFACE_BASE kiface("pcb_test_frame", KIWAY::FACE_PCB)
thread_pool & GetKiCadThreadPool()
Get a reference to the current thread pool.