| /** |
| * This file has no copyright assigned and is placed in the Public Domain. |
| * This file is part of the mingw-w64 runtime package. |
| * No warranty is given; refer to the file DISCLAIMER.PD within this package. |
| */ |
| |
| #include <oscalls.h> |
| #include <internal.h> |
| #include <process.h> |
| #include <signal.h> |
| #include <math.h> |
| #include <stdlib.h> |
| #include <tchar.h> |
| #include <sect_attribs.h> |
| #include <locale.h> |
| #include <corecrt_startup.h> |
| |
| #if defined(__SEH__) && (!defined(__clang__) || __clang_major__ >= 7) |
| #define SEH_INLINE_ASM |
| #endif |
| |
| extern IMAGE_DOS_HEADER __ImageBase; |
| |
| extern void _fpreset (void); |
| |
| int *__cdecl __p__commode(void); |
| |
| #undef _fmode |
| extern int _fmode; |
| #undef _commode |
| extern int _commode; |
| extern int _dowildcard; |
| |
| extern _CRTIMP void __cdecl _initterm(_PVFV *, _PVFV *); |
| |
| static int __cdecl check_managed_app (void); |
| |
| extern _PIFV __xi_a[]; |
| extern _PIFV __xi_z[]; |
| extern _PVFV __xc_a[]; |
| extern _PVFV __xc_z[]; |
| |
| |
| /* TLS initialization hook. */ |
| extern const PIMAGE_TLS_CALLBACK __dyn_tls_init_callback; |
| |
| extern int __mingw_app_type; |
| |
| static int argc; |
| extern void __main(void); |
| static _TCHAR **argv; |
| static _TCHAR **envp; |
| |
| static int mainret=0; |
| static int managedapp; |
| static int has_cctor = 0; |
| extern LPTOP_LEVEL_EXCEPTION_FILTER __mingw_oldexcpt_handler; |
| |
| extern void _pei386_runtime_relocator (void); |
| long CALLBACK _gnu_exception_handler (EXCEPTION_POINTERS * exception_data); |
| static void duplicate_ppstrings (int ac, _TCHAR ***av); |
| |
| extern int _MINGW_INSTALL_DEBUG_MATHERR; |
| |
| #ifdef __MINGW_SHOW_INVALID_PARAMETER_EXCEPTION |
| #define __UNUSED_PARAM_1(x) x |
| #else |
| #define __UNUSED_PARAM_1 __UNUSED_PARAM |
| #endif |
| static void |
| __mingw_invalidParameterHandler (const wchar_t * __UNUSED_PARAM_1(expression), |
| const wchar_t * __UNUSED_PARAM_1(function), |
| const wchar_t * __UNUSED_PARAM_1(file), |
| unsigned int __UNUSED_PARAM_1(line), |
| uintptr_t __UNUSED_PARAM(pReserved)) |
| { |
| #ifdef __MINGW_SHOW_INVALID_PARAMETER_EXCEPTION |
| wprintf(L"Invalid parameter detected in function %s. File: %s Line: %d\n", function, file, line); |
| wprintf(L"Expression: %s\n", expression); |
| #endif |
| } |
| |
| static int __tmainCRTStartup (void); |
| |
| int WinMainCRTStartup (void); |
| |
| __attribute__((used)) /* required due to GNU LD bug: https://sourceware.org/bugzilla/show_bug.cgi?id=30300 */ |
| int WinMainCRTStartup (void) |
| { |
| int ret = 255; |
| #ifdef SEH_INLINE_ASM |
| asm ("\t.l_startw:\n"); |
| #endif |
| __mingw_app_type = 1; |
| ret = __tmainCRTStartup (); |
| #ifdef SEH_INLINE_ASM |
| asm ("\tnop\n" |
| "\t.l_endw: nop\n" |
| #ifdef __arm__ |
| "\t.seh_handler __C_specific_handler, %except\n" |
| #else |
| "\t.seh_handler __C_specific_handler, @except\n" |
| #endif |
| "\t.seh_handlerdata\n" |
| "\t.long 1\n" |
| "\t.rva .l_startw, .l_endw, _gnu_exception_handler ,.l_endw\n" |
| "\t.text"); |
| #endif |
| return ret; |
| } |
| |
| int mainCRTStartup (void); |
| |
| #if defined(__x86_64__) && !defined(__SEH__) |
| int __mingw_init_ehandler (void); |
| #endif |
| |
| __attribute__((used)) /* required due to GNU LD bug: https://sourceware.org/bugzilla/show_bug.cgi?id=30300 */ |
| int mainCRTStartup (void) |
| { |
| int ret = 255; |
| #ifdef SEH_INLINE_ASM |
| asm ("\t.l_start:\n"); |
| #endif |
| __mingw_app_type = 0; |
| ret = __tmainCRTStartup (); |
| #ifdef SEH_INLINE_ASM |
| asm ("\tnop\n" |
| "\t.l_end: nop\n" |
| #ifdef __arm__ |
| "\t.seh_handler __C_specific_handler, %except\n" |
| #else |
| "\t.seh_handler __C_specific_handler, @except\n" |
| #endif |
| "\t.seh_handlerdata\n" |
| "\t.long 1\n" |
| "\t.rva .l_start, .l_end, _gnu_exception_handler ,.l_end\n" |
| "\t.text"); |
| #endif |
| return ret; |
| } |
| |
| static |
| #if defined(__i386__) || defined(_X86_) |
| /* We need to make sure that we align the stack to 16 bytes for the sake of SSE |
| opts in main or in functions called from main. */ |
| __attribute__((force_align_arg_pointer)) |
| #endif |
| __declspec(noinline) int |
| __tmainCRTStartup (void) |
| { |
| void *lock_free = NULL; |
| void *fiberid = ((PNT_TIB)NtCurrentTeb())->StackBase; |
| BOOL nested = FALSE; |
| _startupinfo startinfo; |
| int ret = 0; |
| while((lock_free = InterlockedCompareExchangePointer (&__native_startup_lock, |
| fiberid, NULL)) != 0) |
| { |
| if (lock_free == fiberid) |
| { |
| nested = TRUE; |
| break; |
| } |
| Sleep(1000); |
| } |
| if (__native_startup_state == __initializing) |
| { |
| _amsg_exit (31); |
| } |
| else if (__native_startup_state == __uninitialized) |
| { |
| __native_startup_state = __initializing; |
| |
| _pei386_runtime_relocator (); |
| #if defined(__x86_64__) && !defined(__SEH__) |
| __mingw_init_ehandler (); |
| #endif |
| __mingw_oldexcpt_handler = SetUnhandledExceptionFilter (_gnu_exception_handler); |
| _set_invalid_parameter_handler (__mingw_invalidParameterHandler); |
| _fpreset (); |
| |
| managedapp = check_managed_app (); |
| if (__mingw_app_type) |
| __set_app_type (_GUI_APP); |
| else |
| __set_app_type (_CONSOLE_APP); |
| |
| *__p__fmode () = _fmode; |
| *__p__commode () = _commode; |
| |
| #ifdef _UNICODE |
| ret = _wsetargv (); |
| #else |
| ret = _setargv (); |
| #endif |
| if (ret < 0) |
| _amsg_exit (8); /* _RT_SPACEARG */ |
| |
| if (_MINGW_INSTALL_DEBUG_MATHERR == 1) |
| __setusermatherr (_matherr); |
| |
| if (__globallocalestatus == -1) |
| _configthreadlocale (-1); |
| |
| if (_initterm_e (__xi_a, __xi_z) != 0) |
| return 255; |
| |
| startinfo.newmode = _newmode; |
| #ifdef _UNICODE |
| ret = __wgetmainargs (&argc, &argv, &envp, _dowildcard, &startinfo); |
| #else |
| ret = __getmainargs (&argc, &argv, &envp, _dowildcard, &startinfo); |
| #endif |
| if (ret < 0) |
| _amsg_exit (8); /* _RT_SPACEARG */ |
| |
| _initterm (__xc_a, __xc_z); |
| __main (); /* C++ initialization. */ |
| |
| __native_startup_state = __initialized; |
| } |
| else |
| has_cctor = 1; |
| if (! nested) |
| (VOID)InterlockedExchangePointer (&__native_startup_lock, NULL); |
| |
| if (__dyn_tls_init_callback != NULL) |
| __dyn_tls_init_callback (NULL, DLL_THREAD_ATTACH, NULL); |
| |
| duplicate_ppstrings (argc, &argv); |
| #ifdef _UNICODE |
| __winitenv = envp; |
| #else |
| __initenv = envp; |
| #endif |
| mainret = _tmain (argc, argv, envp); |
| if (!managedapp) |
| exit (mainret); |
| |
| if (has_cctor == 0) |
| _cexit (); |
| |
| return mainret; |
| } |
| |
| extern int __mingw_initltsdrot_force; |
| extern int __mingw_initltsdyn_force; |
| extern int __mingw_initltssuo_force; |
| |
| static int __cdecl |
| check_managed_app (void) |
| { |
| PIMAGE_DOS_HEADER pDOSHeader; |
| PIMAGE_NT_HEADERS pPEHeader; |
| PIMAGE_OPTIONAL_HEADER32 pNTHeader32; |
| PIMAGE_OPTIONAL_HEADER64 pNTHeader64; |
| |
| /* Force to be linked. */ |
| __mingw_initltsdrot_force=1; |
| __mingw_initltsdyn_force=1; |
| __mingw_initltssuo_force=1; |
| |
| pDOSHeader = (PIMAGE_DOS_HEADER) &__ImageBase; |
| if (pDOSHeader->e_magic != IMAGE_DOS_SIGNATURE) |
| return 0; |
| |
| pPEHeader = (PIMAGE_NT_HEADERS)((char *)pDOSHeader + pDOSHeader->e_lfanew); |
| if (pPEHeader->Signature != IMAGE_NT_SIGNATURE) |
| return 0; |
| |
| pNTHeader32 = (PIMAGE_OPTIONAL_HEADER32) &pPEHeader->OptionalHeader; |
| switch (pNTHeader32->Magic) |
| { |
| case IMAGE_NT_OPTIONAL_HDR32_MAGIC: |
| if (pNTHeader32->NumberOfRvaAndSizes <= IMAGE_DIRECTORY_ENTRY_COM_DESCRIPTOR) |
| return 0; |
| return !! pNTHeader32->DataDirectory[IMAGE_DIRECTORY_ENTRY_COM_DESCRIPTOR].VirtualAddress; |
| case IMAGE_NT_OPTIONAL_HDR64_MAGIC: |
| pNTHeader64 = (PIMAGE_OPTIONAL_HEADER64)pNTHeader32; |
| if (pNTHeader64->NumberOfRvaAndSizes <= IMAGE_DIRECTORY_ENTRY_COM_DESCRIPTOR) |
| return 0; |
| return !! pNTHeader64->DataDirectory[IMAGE_DIRECTORY_ENTRY_COM_DESCRIPTOR].VirtualAddress; |
| } |
| return 0; |
| } |
| |
| static void duplicate_ppstrings (int ac, _TCHAR ***av) |
| { |
| _TCHAR **avl; |
| int i; |
| _TCHAR **n = (_TCHAR **) malloc (sizeof (_TCHAR *) * (ac + 1)); |
| |
| avl=*av; |
| for (i=0; i < ac; i++) |
| { |
| size_t l = sizeof (_TCHAR) * (_tcslen (avl[i]) + 1); |
| n[i] = (_TCHAR *) malloc (l); |
| memcpy (n[i], avl[i], l); |
| } |
| n[i] = NULL; |
| *av = n; |
| } |
| |
| int __cdecl atexit (_PVFV func) |
| { |
| /* |
| * msvcrt def file renames the real atexit() function to _crt_atexit(). |
| * UCRT provides atexit() function only under name _crt_atexit(). |
| * So redirect call to _crt_atexit() function. |
| */ |
| return _crt_atexit(func); |
| } |
| |
| char __mingw_module_is_dll = 0; |