1 /////////////////////////////////////////////////////////////////////////////
3 // Purpose: WinMain/DllMain
4 // Author: Julian Smart
8 // Copyright: (c) Julian Smart
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 // ============================================================================
14 // ============================================================================
16 // ----------------------------------------------------------------------------
18 // ----------------------------------------------------------------------------
20 // For compilers that support precompilation, includes "wx.h".
21 #include "wx/wxprec.h"
29 #include "wx/cmdline.h"
30 #include "wx/scopeguard.h"
32 #include "wx/msw/private.h"
33 #include "wx/msw/seh.h"
35 #if wxUSE_ON_FATAL_EXCEPTION
36 #include "wx/datetime.h"
37 #include "wx/msw/crashrpt.h"
38 #endif // wxUSE_ON_FATAL_EXCEPTION
41 // there is no ExitProcess() under CE but exiting the main thread has the
44 #define ExitProcess ExitThread
49 // BC++ has to be special: its run-time expects the DLL entry point to be
50 // named DllEntryPoint instead of the (more) standard DllMain
51 #define DllMain DllEntryPoint
52 #endif // __BORLANDC__
54 #if defined(__WXMICROWIN__)
55 #define HINSTANCE HANDLE
58 // defined in common/init.cpp
59 extern int wxEntryReal(int& argc
, wxChar
**argv
);
61 // ============================================================================
62 // implementation: various entry points
63 // ============================================================================
67 #if wxUSE_ON_FATAL_EXCEPTION && defined(__VISUALC__) && !defined(__WXWINCE__)
68 // VC++ (at least from 4.0 up to version 7.1) is incredibly broken in that
69 // a "catch ( ... )" will *always* catch SEH exceptions in it even though
70 // it should have never been the case... to prevent such catches from
71 // stealing the exceptions from our wxGlobalSEHandler which is only called
72 // if the exception is not handled elsewhere, we have to also call it from
73 // a special SEH translator function which is called by VC CRT when a Win32
76 // this warns that /EHa (async exceptions) should be used when using
77 // _set_se_translator but, in fact, this doesn't seem to change anything
78 // with VC++ up to 7.1 -- to be confirmed with VC++ 8
80 #pragma warning(disable:4535)
83 // note that the SE translator must be called wxSETranslator!
84 #define DisableAutomaticSETranslator() _set_se_translator(wxSETranslator)
86 #define DisableAutomaticSETranslator()
87 #endif // __VISUALC__/!__VISUALC__
89 // ----------------------------------------------------------------------------
90 // wrapper wxEntry catching all Win32 exceptions occurring in a wx program
91 // ----------------------------------------------------------------------------
93 // wrap real wxEntry in a try-except block to be able to call
94 // OnFatalException() if necessary
95 #if wxUSE_ON_FATAL_EXCEPTION
97 // global pointer to exception information, only valid inside OnFatalException,
98 // used by wxStackWalker and wxCrashReport
99 extern EXCEPTION_POINTERS
*wxGlobalSEInformation
= NULL
;
101 // flag telling us whether the application wants to handle exceptions at all
102 static bool gs_handleExceptions
= false;
104 static void wxFatalExit()
106 // use the same exit code as abort()
110 unsigned long wxGlobalSEHandler(EXCEPTION_POINTERS
*pExcPtrs
)
112 if ( gs_handleExceptions
&& wxTheApp
)
114 // store the pointer to exception info
115 wxGlobalSEInformation
= pExcPtrs
;
117 // give the user a chance to do something special about this
120 wxTheApp
->OnFatalException();
122 wxSEH_IGNORE
// ignore any exceptions inside the exception handler
124 wxGlobalSEInformation
= NULL
;
126 // this will execute our handler and terminate the process
127 return EXCEPTION_EXECUTE_HANDLER
;
130 return EXCEPTION_CONTINUE_SEARCH
;
135 void wxSETranslator(unsigned int WXUNUSED(code
), EXCEPTION_POINTERS
*ep
)
137 switch ( wxGlobalSEHandler(ep
) )
140 wxFAIL_MSG( _T("unexpected wxGlobalSEHandler() return value") );
143 case EXCEPTION_EXECUTE_HANDLER
:
144 // if wxApp::OnFatalException() had been called we should exit the
145 // application -- but we shouldn't kill our host when we're a DLL
151 case EXCEPTION_CONTINUE_SEARCH
:
152 // we're called for each "catch ( ... )" and if we (re)throw from
153 // here, the catch handler body is not executed, so the effect is
154 // as if had inhibited translation of SE to C++ ones because the
155 // handler will never see any structured exceptions
160 #endif // __VISUALC__
162 bool wxHandleFatalExceptions(bool doit
)
164 // assume this can only be called from the main thread
165 gs_handleExceptions
= doit
;
167 #if wxUSE_CRASHREPORT
170 // try to find a place where we can put out report file later
171 wxChar fullname
[MAX_PATH
];
172 if ( !::GetTempPath(WXSIZEOF(fullname
), fullname
) )
174 wxLogLastError(_T("GetTempPath"));
176 // when all else fails...
177 wxStrcpy(fullname
, _T("c:\\"));
180 // use PID and date to make the report file name more unique
181 wxString name
= wxString::Format
184 wxTheApp
? wxTheApp
->GetAppName().c_str()
186 wxDateTime::Now().Format(_T("%Y%m%dT%H%M%S")).c_str(),
187 ::GetCurrentProcessId()
190 wxStrncat(fullname
, name
, WXSIZEOF(fullname
) - wxStrlen(fullname
) - 1);
192 wxCrashReport::SetFileName(fullname
);
194 #endif // wxUSE_CRASHREPORT
199 int wxEntry(int& argc
, wxChar
**argv
)
201 DisableAutomaticSETranslator();
205 return wxEntryReal(argc
, argv
);
210 #else // !wxUSE_ON_FATAL_EXCEPTION
212 #if defined(__VISUALC__) && !defined(__WXWINCE__)
215 wxSETranslator(unsigned int WXUNUSED(code
), EXCEPTION_POINTERS
* WXUNUSED(ep
))
217 // see wxSETranslator() version for wxUSE_ON_FATAL_EXCEPTION above
221 #endif // __VISUALC__
223 int wxEntry(int& argc
, wxChar
**argv
)
225 DisableAutomaticSETranslator();
227 return wxEntryReal(argc
, argv
);
230 #endif // wxUSE_ON_FATAL_EXCEPTION/!wxUSE_ON_FATAL_EXCEPTION
234 #if wxUSE_GUI && defined(__WXMSW__)
236 #if wxUSE_UNICODE && !defined(__WXWINCE__)
237 #define NEED_UNICODE_CHECK
240 #ifdef NEED_UNICODE_CHECK
242 // check whether Unicode is available
243 static bool wxIsUnicodeAvailable()
245 static const wchar_t *ERROR_STRING
= L
"wxWidgets Fatal Error";
247 if ( wxGetOsVersion() != wxWINDOWS_NT
)
249 // we need to be built with MSLU support
250 #if !wxUSE_UNICODE_MSLU
251 // note that we can use MessageBoxW() as it's implemented even under
252 // Win9x - OTOH, we can't use wxGetTranslation() because the file APIs
253 // used by wxLocale are not
257 L
"This program uses Unicode and requires Windows NT/2000/XP.\n"
265 #else // wxUSE_UNICODE_MSLU
266 // and the MSLU DLL must also be available
267 HMODULE hmod
= ::LoadLibraryA("unicows.dll");
273 L
"This program uses Unicode and requires unicows.dll to work "
274 L
"under current operating system.\n"
276 L
"Please install unicows.dll and relaunch the program.",
283 // this is not really necessary but be tidy
286 // finally do the last check: has unicows.lib initialized correctly?
287 hmod
= ::LoadLibraryW(L
"unicows.dll");
293 L
"This program uses Unicode but is not using unicows.dll\n"
294 L
"correctly and so cannot work under current operating system.\n"
295 L
"Please contact the program author for an updated version.\n"
306 #endif // !wxUSE_UNICODE_MSLU
312 #endif // NEED_UNICODE_CHECK
314 // ----------------------------------------------------------------------------
315 // Windows-specific wxEntry
316 // ----------------------------------------------------------------------------
318 // helper function used to clean up in wxEntry() just below
320 // notice that argv elements are supposed to be allocated using malloc() while
321 // argv array itself is allocated with new
322 static void wxFreeArgs(int argc
, wxChar
**argv
)
324 for ( int i
= 0; i
< argc
; i
++ )
332 WXDLLEXPORT
int wxEntry(HINSTANCE hInstance
,
333 HINSTANCE
WXUNUSED(hPrevInstance
),
334 wxCmdLineArgType
WXUNUSED(pCmdLine
),
337 // the first thing to do is to check if we're trying to run an Unicode
338 // program under Win9x w/o MSLU emulation layer - if so, abort right now
339 // as it has no chance to work and has all chances to crash
340 #ifdef NEED_UNICODE_CHECK
341 if ( !wxIsUnicodeAvailable() )
343 #endif // NEED_UNICODE_CHECK
346 // remember the parameters Windows gave us
347 wxSetInstance(hInstance
);
348 wxApp::m_nCmdShow
= nCmdShow
;
350 // parse the command line: we can't use pCmdLine in Unicode build so it is
351 // simpler to never use it at all (this also results in a more correct
354 // break the command line in words
357 const wxChar
*cmdLine
= ::GetCommandLine();
360 args
= wxCmdLineParser::ConvertStringToArgs(cmdLine
);
364 // WinCE doesn't insert the program itself, so do it ourselves.
365 args
.Insert(wxGetFullModuleName(), 0);
368 int argc
= args
.GetCount();
370 // +1 here for the terminating NULL
371 wxChar
**argv
= new wxChar
*[argc
+ 1];
372 for ( int i
= 0; i
< argc
; i
++ )
374 argv
[i
] = wxStrdup(args
[i
]);
377 // argv[] must be NULL-terminated
380 wxON_BLOCK_EXIT2(wxFreeArgs
, argc
, argv
);
382 return wxEntry(argc
, argv
);
385 #endif // wxUSE_GUI && __WXMSW__
387 // ----------------------------------------------------------------------------
389 // ----------------------------------------------------------------------------
393 HINSTANCE wxhInstance
= 0;
395 extern "C" HINSTANCE
wxGetInstance()
400 void wxSetInstance(HINSTANCE hInst
)