/*** *crt0dat.c - 32-bit C run-time initialization/termination routines * * Copyright (c) Microsoft Corporation. All rights reserved. * *Purpose: * This module contains the routines _cinit, exit, and _exit * for C run-time startup and termination. _cinit and exit * are called from the _astart code in crt0.asm. * This module also defines several data variables used by the * runtime. * * [NOTE: Lock segment definitions are at end of module.] * *******************************************************************************/ #include #include #include #include #include #include #include #include #include #include #include #include #include #include /* define errno */ /* define umask */ int _umaskval = 0; /* define version info variables */ _CRTIMP unsigned int _osplatform = 0; _CRTIMP unsigned int _osver = 0; _CRTIMP unsigned int _winver = 0; _CRTIMP unsigned int _winmajor = 0; _CRTIMP unsigned int _winminor = 0; /* argument vector and environment */ _CRTIMP int __argc = 0; _CRTIMP char **__argv = NULL; _CRTIMP wchar_t **__wargv = NULL; _CRTIMP char **_environ = NULL; _CRTIMP char **__initenv = NULL; _CRTIMP wchar_t **_wenviron = NULL; _CRTIMP wchar_t **__winitenv = NULL; _CRTIMP char *_pgmptr = NULL; /* ptr to program name */ _CRTIMP wchar_t *_wpgmptr = NULL; /* ptr to wide program name */ void (__cdecl * _aexit_rtn)(int) = _exit; /* RT message return procedure */ /* callable exit flag */ char _exitflag = 0; /* * flag indicating if C runtime termination has been done. set if exit, * _exit, _cexit or _c_exit has been called. checked when _CRTDLL_INIT * is called with DLL_PROCESS_DETACH. */ int _C_Termination_Done = FALSE; int _C_Exit_Done = FALSE; #ifndef CRTDLL /* * Flag checked by getenv() and _putenv() to determine if the environment has * been initialized. */ int __env_initialized; #endif /* CRTDLL */ #ifdef _MBCS /* * Flag to ensure multibyte ctype table is only initialized once */ int __mbctype_initialized; #endif /* _MBCS */ /* * NOTE: THE USE OF THE POINTERS DECLARED BELOW DEPENDS ON THE PROPERTIES * OF C COMMUNAL VARIABLES. SPECIFICALLY, THEY ARE NON-NULL IFF THERE EXISTS * A DEFINITION ELSEWHERE INITIALIZING THEM TO NON-NULL VALUES. */ /* * pointers to initialization functions */ #ifndef CRTDLL const _PVFI _FPinit; /* floating point init. */ #endif /* CRTDLL */ /* * pointers to initialization sections */ extern _CRTALLOC(".CRT$XIA") _PIFV __xi_a[]; extern _CRTALLOC(".CRT$XIZ") _PIFV __xi_z[]; /* C initializers */ extern _CRTALLOC(".CRT$XCA") _PVFV __xc_a[]; extern _CRTALLOC(".CRT$XCZ") _PVFV __xc_z[]; /* C++ initializers */ extern _CRTALLOC(".CRT$XPA") _PVFV __xp_a[]; extern _CRTALLOC(".CRT$XPZ") _PVFV __xp_z[]; /* C pre-terminators */ extern _CRTALLOC(".CRT$XTA") _PVFV __xt_a[]; extern _CRTALLOC(".CRT$XTZ") _PVFV __xt_z[]; /* C terminators */ /* * pointers to the start and finish of the _onexit/atexit table * NOTE - the pointers are stored encoded. */ _PVFV *__onexitbegin; _PVFV *__onexitend; /* * Pointer to callback function to initialize any dynamically initialized * __declspec(thread) variables. This relies on a subtle aspect of C. * The pointer is defined here uninitialized. It is defined initialized in * tlsdyn.c. If user code uses dynamically initialized __declspec(thread) * variables, then compiler-injected dependencies cause tlsdyn.obj to be * linked. In that case, the non-zero definition of __dyn_tls_init_callback * in tlsdyn.obj will take precedence, and the startup code will execute the * callback. This use of multiple definitions is only legal in C, not C++. */ #ifndef CRTDLL const PIMAGE_TLS_CALLBACK __dyn_tls_init_callback; #endif /* CRTDLL */ /* * static (internal) functions that walk a table of function pointers, * calling each entry between the two pointers, skipping NULL entries * * _initterm needs to be exported for CRT DLL so that C++ initializers in the * client EXE / DLLs can be initialized. * * _initterm_e calls function pointers that return a nonzero error code to * indicate an initialization failed fatally. */ #ifdef CRTDLL void __cdecl _initterm(_PVFV *, _PVFV *); #else /* CRTDLL */ static void __cdecl _initterm(_PVFV *, _PVFV *); #endif /* CRTDLL */ int __cdecl _initterm_e(_PIFV *, _PIFV *); #ifdef CRTDLL /* Copied from dllstuff\crtexe.c. */ /*** *check_managed_app() - Check for a managed executable * *Purpose: * Determine if the EXE the startup code is linked into is a managed app * by looking for the COM Runtime Descriptor in the Image Data Directory * of the PE or PE+ header. * *Entry: * None * *Exit: * 1 if managed app, 0 if not. * *Exceptions: * *******************************************************************************/ static int __cdecl check_managed_app ( void ) { PIMAGE_DOS_HEADER pDOSHeader; PIMAGE_NT_HEADERS pPEHeader; PIMAGE_OPTIONAL_HEADER32 pNTHeader32; PIMAGE_OPTIONAL_HEADER64 pNTHeader64; pDOSHeader = (PIMAGE_DOS_HEADER)GetModuleHandleA(NULL); 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: /* PE header */ /* prefast assumes we are overflowing __ImageBase */ #pragma warning(push) #pragma warning(disable:26000) if ( pNTHeader32->NumberOfRvaAndSizes <= IMAGE_DIRECTORY_ENTRY_COM_DESCRIPTOR ) return 0; #pragma warning(pop) return !! pNTHeader32 -> DataDirectory[IMAGE_DIRECTORY_ENTRY_COM_DESCRIPTOR] . VirtualAddress; case IMAGE_NT_OPTIONAL_HDR64_MAGIC: /* PE+ header */ pNTHeader64 = (PIMAGE_OPTIONAL_HEADER64)pNTHeader32; if ( pNTHeader64->NumberOfRvaAndSizes <= IMAGE_DIRECTORY_ENTRY_COM_DESCRIPTOR ) return 0; return !! pNTHeader64 -> DataDirectory[IMAGE_DIRECTORY_ENTRY_COM_DESCRIPTOR] . VirtualAddress; } /* Not PE or PE+, so not managed */ return 0; } #endif /* CRTDLL */ /*** *_cinit - C initialization * *Purpose: * This routine performs the shared DOS and Windows initialization. * The following order of initialization must be preserved - * * 1. Check for devices for file handles 0 - 2 * 2. Integer divide interrupt vector setup * 3. General C initializer routines * *Entry: * No parameters: Called from __crtstart and assumes data * set up correctly there. * *Exit: * Initializes C runtime data. * Returns 0 if all .CRT$XI internal initializations succeeded, else * the _RT_* fatal error code encountered. * *Exceptions: * *******************************************************************************/ #ifndef CRTDLL extern void __cdecl _initp_misc_cfltcvt_tab(); #endif /* CRTDLL */ int __cdecl _cinit ( int initFloatingPrecision ) { int initret; /* * initialize floating point package, if present */ #ifdef CRTDLL _fpmath(initFloatingPrecision); #else /* CRTDLL */ if (_FPinit != NULL && _IsNonwritableInCurrentImage((PBYTE)&_FPinit)) { (*_FPinit)(initFloatingPrecision); } _initp_misc_cfltcvt_tab(); #endif /* CRTDLL */ /* * do initializations */ initret = _initterm_e( __xi_a, __xi_z ); if ( initret != 0 ) return initret; #ifdef _RTC atexit(_RTC_Terminate); #endif /* _RTC */ /* * do C++ initializations */ _initterm( __xc_a, __xc_z ); #ifndef CRTDLL /* * If we have any dynamically initialized __declspec(thread) * variables, then invoke their initialization for the thread on * which the DLL is being loaded, by calling __dyn_tls_init through * a callback defined in tlsdyn.obj. We can't rely on the OS * calling __dyn_tls_init with DLL_PROCESS_ATTACH because, on * Win2K3 and before, that call happens before the CRT is * initialized. */ if (__dyn_tls_init_callback != NULL && _IsNonwritableInCurrentImage((PBYTE)&__dyn_tls_init_callback)) { __dyn_tls_init_callback(NULL, DLL_THREAD_ATTACH, NULL); } #endif /* CRTDLL */ return 0; } /*** *exit(status), _exit(status), _cexit(void), _c_exit(void) - C termination * *Purpose: * * Entry points: * * exit(code): Performs all the C termination functions * and terminates the process with the return code * supplied by the user. * * _exit(code): Performs a quick exit routine that does not * do certain 'high-level' exit processing. The _exit * routine terminates the process with the return code * supplied by the user. * * _cexit(): Performs the same C lib termination processing * as exit(code) but returns control to the caller * when done (i.e., does NOT terminate the process). * * _c_exit(): Performs the same C lib termination processing * as _exit(code) but returns control to the caller * when done (i.e., does NOT terminate the process). * * Termination actions: * * exit(), _cexit(): * * 1. call user's terminator routines * 2. call C runtime preterminators * * _exit(), _c_exit(): * * 3. call C runtime terminators * 4. return to DOS or caller * * Notes: * * The termination sequence is complicated due to the multiple entry * points sharing the common code body while having different entry/exit * sequences. * * Multi-thread notes: * * 1. exit() should NEVER be called when mthread locks are held. * The exit() routine can make calls that try to get mthread locks. * * 2. _exit()/_c_exit() can be called from anywhere, with or without locks held. * Thus, _exit() can NEVER try to get locks (otherwise, deadlock * may occur). _exit() should always 'work' (i.e., the process * should always terminate successfully). * * 3. Only one thread is allowed into the exit code (see _lockexit() * and _unlockexit() routines). * *Entry: * exit(), _exit() * int status - exit status (0-255) * * _cexit(), _c_exit() * * *Exit: * exit(), _exit() * * * _cexit(), _c_exit() * Return to caller * *Uses: * *Exceptions: * *******************************************************************************/ /* worker routine prototype */ static void __cdecl doexit (int code, int quick, int retcaller); void __cdecl exit ( int code ) { doexit(code, 0, 0); /* full term, kill process */ } void __cdecl _exit ( int code ) { doexit(code, 1, 0); /* quick term, kill process */ } void __cdecl _cexit ( void ) { doexit(0, 0, 1); /* full term, return to caller */ } void __cdecl _c_exit ( void ) { doexit(0, 1, 1); /* quick term, return to caller */ } /*** *_amsg_exit(rterrnum) - Fast exit fatal errors * *Purpose: * Exit the program with error code of 255 and appropriate error * message. * *Entry: * int rterrnum - error message number (amsg_exit only). * *Exit: * Calls exit() (for integer divide-by-0) or _exit() indirectly * through _aexit_rtn [amsg_exit]. * For multi-thread: calls _exit() function * *Exceptions: * *******************************************************************************/ void __cdecl _amsg_exit ( int rterrnum ) { void(__cdecl *ertn)(int); _FF_MSGBANNER(); /* write run-time error banner */ _NMSG_WRITE(rterrnum); /* write message */ ertn = (void(__cdecl*)(int)) _decode_pointer(_aexit_rtn); ertn(255); /* normally _exit(255) */ } #ifdef _DEBUG /*** * __freeCrtMemory() * * Purpose: * To free as much as CRT memory as possible. This helps in keeping CRT leaks as * minimum. * *******************************************************************************/ void __cdecl __freeCrtMemory() { void **pptr; for (pptr = _wenviron; pptr != NULL && *pptr != NULL; ++pptr) { _free_crt(*pptr); } _free_crt(_wenviron); _wenviron = NULL; pptr = _environ; for (; pptr != NULL && *pptr != NULL; ++pptr) { _free_crt(*pptr); } _free_crt(_environ); _environ = NULL; _free_crt(__wargv); _free_crt(__argv); _free_crt(_decode_pointer(__onexitbegin)); __wargv = NULL; __argv = NULL; __onexitbegin = (_PVFV *)_encoded_null(); if (InterlockedDecrement(&(__ptmbcinfo->refcount)) == 0 && __ptmbcinfo != &__initialmbcinfo) { _free_crt(__ptmbcinfo); __ptmbcinfo = &__initialmbcinfo; } InterlockedIncrement(&(__ptmbcinfo->refcount)); } #endif /* _DEBUG */ static void __cdecl doexit ( int code, int quick, int retcaller ) { #ifdef _DEBUG static int fExit = 0; #endif /* _DEBUG */ #ifdef CRTDLL if (!retcaller && check_managed_app()) { /* Only if the EXE is managed then we call CorExitProcess. Native cleanup is done in .cctor of the EXE If the Exe is Native then native clean up should be done before calling (Cor)ExitProcess. */ __crtCorExitProcess(code); } #endif /* CRTDLL */ _lockexit(); /* assure only 1 thread in exit path */ __TRY if (_C_Exit_Done != TRUE) { _C_Termination_Done = TRUE; /* save callable exit flag (for use by terminators) */ _exitflag = (char) retcaller; /* 0 = term, !0 = callable exit */ if (!quick) { /* * do _onexit/atexit() terminators * (if there are any) * * These terminators MUST be executed in reverse order (LIFO)! * * NOTE: * This code assumes that __onexitbegin points * to the first valid onexit() entry and that * __onexitend points past the last valid entry. * If __onexitbegin == __onexitend, the table * is empty and there are no routines to call. */ _PVFV * onexitbegin = (_PVFV *)_decode_pointer(__onexitbegin); _PVFV * onexitend = (_PVFV *)_decode_pointer(__onexitend); if (onexitbegin) { while ( --onexitend >= onexitbegin ) /* * if current table entry is non-NULL, * call thru it. */ if ( *onexitend != NULL && *onexitend != _encoded_null()) (*(_PVFV)_decode_pointer(*onexitend))(); } #ifndef CRTDLL /* * do pre-terminators */ _initterm(__xp_a, __xp_z); #endif /* CRTDLL */ } #ifndef CRTDLL /* * do terminators */ _initterm(__xt_a, __xt_z); #endif /* CRTDLL */ #ifdef _DEBUG /* Dump all memory leaks */ if (!fExit && _CrtSetDbgFlag(_CRTDBG_REPORT_FLAG) & _CRTDBG_LEAK_CHECK_DF) { fExit = 1; #ifndef CRTDLL __freeCrtMemory(); _CrtDumpMemoryLeaks(); #endif /* CRTDLL */ } #endif /* _DEBUG */ } /* return to OS or to caller */ __FINALLY if (retcaller) _unlockexit(); /* unlock the exit code path */ __END_TRY_FINALLY if (retcaller) return; _C_Exit_Done = TRUE; _unlockexit(); /* unlock the exit code path */ __crtExitProcess(code); } #ifdef CRTDLL /*** * __crtdll_callstaticterminators * *Purpose: * call terminators. This is called from CRT_INIT when dll entrypoint is * called with DLL_PROCESS_DETACH. We can't call the terminators from exit * as there may be some dll that may need crt functionality during * DLL_PROCESS_DETACH * *******************************************************************************/ void __crtdll_callstaticterminators(void) { /* * do pre-terminators */ _initterm(__xp_a, __xp_z); /* * do terminators */ _initterm(__xt_a, __xt_z); } #endif /* CRTDLL */ /*** * __crtCorExitProcess - CRT wrapper for CorExitProcess * __crtExitProcess - CRT wrapper for ExitProcess * *Purpose: * If we're part of a managed app, then call the CorExitProcess, * otherwise call ExitProcess. For managed apps, calling ExitProcess can * be problematic, because it doesn't give the managed FinalizerThread a * chance to clean up. * * To determine if we're a managed app, we check if mscoree.dll is loaded. * Then, if CorExitProcess is available, we call it. * *Entry: * int status - exit code * *Exit: * Does not return * *Exceptions: * *******************************************************************************/ typedef void (WINAPI * PFN_EXIT_PROCESS)(UINT uExitCode); void __cdecl __crtCorExitProcess ( int status ) { HMODULE hmod; PFN_EXIT_PROCESS pfn; hmod = GetModuleHandle("mscoree.dll"); if (hmod != NULL) { pfn = (PFN_EXIT_PROCESS)GetProcAddress(hmod, "CorExitProcess"); if (pfn != NULL) { pfn(status); } } /* * Either mscoree.dll isn't loaded, * or CorExitProcess isn't exported from mscoree.dll, * or CorExitProcess returned (should never happen). * Just call return. */ } void __cdecl __crtExitProcess ( int status ) { __crtCorExitProcess(status); /* * Either mscoree.dll isn't loaded, * or CorExitProcess isn't exported from mscoree.dll, * or CorExitProcess returned (should never happen). * Just call ExitProcess. */ ExitProcess(status); } /*** * _lockexit - Aquire the exit code lock * *Purpose: * Makes sure only one thread is in the exit code at a time. * If a thread is already in the exit code, it must be allowed * to continue. All other threads must pend. * * Notes: * * (1) It is legal for a thread that already has the lock to * try and get it again(!). That is, consider the following * sequence: * * (a) program calls exit() * (b) thread locks exit code * (c) user onexit() routine calls _exit() * (d) same thread tries to lock exit code * * Since _exit() must ALWAYS be able to work (i.e., can be called * from anywhere with no regard for locking), we must make sure the * program does not deadlock at step (d) above. * * (2) If a thread executing exit() or _exit() aquires the exit lock, * other threads trying to get the lock will pend forever. That is, * since exit() and _exit() terminate the process, there is not need * for them to unlock the exit code path. * * (3) Note that onexit()/atexit() routines call _lockexit/_unlockexit * to protect mthread access to the onexit table. * * (4) The 32-bit OS semaphore calls DO allow a single thread to acquire * the same lock multiple times* thus, this version is straight forward. * *Entry: * *Exit: * Calling thread has exit code path locked on return. * *Exceptions: * *******************************************************************************/ void __cdecl _lockexit ( void ) { _mlock(_EXIT_LOCK1); } /*** * _unlockexit - Release exit code lock * *Purpose: * [See _lockexit() description above.] * * This routine is called by _cexit(), _c_exit(), and onexit()/atexit(). * The exit() and _exit() routines never unlock the exit code path since * they are terminating the process. * *Entry: * Exit code path is unlocked. * *Exit: * *Exceptions: * *******************************************************************************/ void __cdecl _unlockexit ( void ) { _munlock(_EXIT_LOCK1); } /*** *_init_pointers() - initialize global function pointers with OS encoded values * *Purpose: * *Entry: * *Exit: * *Exceptions: * *******************************************************************************/ extern void __cdecl _initp_eh_hooks(void*); extern void __cdecl _initp_heap_handler(void*); extern void __cdecl _initp_misc_initcrit(void*); extern void __cdecl _initp_misc_invarg(void*); extern void __cdecl _initp_misc_purevirt(void*); extern void __cdecl _initp_misc_rand_s(void*); extern void __cdecl _initp_misc_winsig(void*); extern void __cdecl _initp_misc_winxfltr(void*); void __cdecl _init_pointers() { void *enull = _encoded_null(); _initp_heap_handler(enull); _initp_misc_initcrit(enull); _initp_misc_invarg(enull); _initp_misc_purevirt(enull); _initp_misc_rand_s(enull); _initp_misc_winsig(enull); _initp_misc_winxfltr(enull); _initp_eh_hooks(enull); /* startup\crt0dat.c */ _aexit_rtn = (void(__cdecl*)(int)) _encode_pointer(_exit); } /*** * static void _initterm(_PVFV * pfbegin, _PVFV * pfend) - call entries in * function pointer table * *Purpose: * Walk a table of function pointers, calling each entry, as follows: * * 1. walk from beginning to end, pfunctbl is assumed to point * to the beginning of the table, which is currently a null entry, * as is the end entry. * 2. skip NULL entries * 3. stop walking when the end of the table is encountered * *Entry: * _PVFV *pfbegin - pointer to the beginning of the table (first * valid entry). * _PVFV *pfend - pointer to the end of the table (after last * valid entry). * *Exit: * No return value * *Notes: * This routine must be exported in the CRT DLL model so that the client * EXE and client DLL(s) can call it to initialize their C++ constructors. * *Exceptions: * If either pfbegin or pfend is NULL, or invalid, all bets are off! * *******************************************************************************/ #ifdef CRTDLL void __cdecl _initterm ( #else /* CRTDLL */ static void __cdecl _initterm ( #endif /* CRTDLL */ _PVFV * pfbegin, _PVFV * pfend ) { /* * walk the table of function pointers from the bottom up, until * the end is encountered. Do not skip the first entry. The initial * value of pfbegin points to the first valid entry. Do not try to * execute what pfend points to. Only entries before pfend are valid. */ while ( pfbegin < pfend ) { /* * if current table entry is non-NULL, call thru it. */ if ( *pfbegin != NULL ) (**pfbegin)(); ++pfbegin; } } /*** * static int _initterm_e(_PIFV * pfbegin, _PIFV * pfend) - call entries in * function pointer table, return error code on any failure * *Purpose: * Walk a table of function pointers in the same way as _initterm, but * here the functions return an error code. If an error is returned, it * will be a nonzero value equal to one of the _RT_* codes. * *Entry: * _PIFV *pfbegin - pointer to the beginning of the table (first * valid entry). * _PIFV *pfend - pointer to the end of the table (after last * valid entry). * *Exit: * No return value * *Notes: * This routine must be exported in the CRT DLL model so that the client * EXE and client DLL(s) can call it. * *Exceptions: * If either pfbegin or pfend is NULL, or invalid, all bets are off! * *******************************************************************************/ int __cdecl _initterm_e ( _PIFV * pfbegin, _PIFV * pfend ) { int ret = 0; /* * walk the table of function pointers from the bottom up, until * the end is encountered. Do not skip the first entry. The initial * value of pfbegin points to the first valid entry. Do not try to * execute what pfend points to. Only entries before pfend are valid. */ while ( pfbegin < pfend && ret == 0) { /* * if current table entry is non-NULL, call thru it. */ if ( *pfbegin != NULL ) ret = (**pfbegin)(); ++pfbegin; } return ret; } /*** *errno_t _get_osplatform() - get _osplatform * *Purpose: * Get the value of _osplatform * *Entry: * unsigned int *pValue - pointer where to store the value * *Exit: * The error code * *Exceptions: * Input parameters are validated. Refer to the validation section of the function. * *******************************************************************************/ errno_t _get_osplatform(unsigned int *pValue) { /* validation section */ _VALIDATE_RETURN_ERRCODE(pValue != NULL, EINVAL); _BEGIN_SECURE_CRT_DEPRECATION_DISABLE _VALIDATE_RETURN_ERRCODE(_osplatform!=0, EINVAL); *pValue = _osplatform; _END_SECURE_CRT_DEPRECATION_DISABLE return 0; } /*** *errno_t _get_osver() - get _osver * *Purpose: * Get the value of _osver * *Entry: * unsigned int *pValue - pointer where to store the value * *Exit: * The error code * *Exceptions: * Input parameters are validated. Refer to the validation section of the function. * *******************************************************************************/ errno_t _get_osver(unsigned int *pValue) { /* validation section */ _VALIDATE_RETURN_ERRCODE(pValue != NULL, EINVAL); _BEGIN_SECURE_CRT_DEPRECATION_DISABLE _VALIDATE_RETURN_ERRCODE(_osplatform!=0, EINVAL); *pValue = _osver; _END_SECURE_CRT_DEPRECATION_DISABLE return 0; } /*** *errno_t _get_winver() - get _winver * *Purpose: * Get the value of _winver * *Entry: * unsigned int *pValue - pointer where to store the value * *Exit: * The error code * *Exceptions: * Input parameters are validated. Refer to the validation section of the function. * *******************************************************************************/ errno_t _get_winver(unsigned int *pValue) { /* validation section */ _VALIDATE_RETURN_ERRCODE(pValue != NULL, EINVAL); _BEGIN_SECURE_CRT_DEPRECATION_DISABLE _VALIDATE_RETURN_ERRCODE(_osplatform!=0, EINVAL); *pValue = _winver; _END_SECURE_CRT_DEPRECATION_DISABLE return 0; } /*** *errno_t _get_winmajor() - get _winver * *Purpose: * Get the value of _winmajor * *Entry: * unsigned int *pValue - pointer where to store the value * *Exit: * The error code * *Exceptions: * Input parameters are validated. Refer to the validation section of the function. * *******************************************************************************/ errno_t _get_winmajor(unsigned int *pValue) { /* validation section */ _VALIDATE_RETURN_ERRCODE(pValue != NULL, EINVAL); _BEGIN_SECURE_CRT_DEPRECATION_DISABLE _VALIDATE_RETURN_ERRCODE(_osplatform!=0, EINVAL); *pValue = _winmajor; _END_SECURE_CRT_DEPRECATION_DISABLE return 0; } /*** *errno_t _get_winminor() - get _winminor * *Purpose: * Get the value of _winminor * *Entry: * unsigned int *pValue - pointer where to store the value * *Exit: * The error code * *Exceptions: * Input parameters are validated. Refer to the validation section of the function. * *******************************************************************************/ errno_t _get_winminor(unsigned int *pValue) { /* validation section */ _VALIDATE_RETURN_ERRCODE(pValue != NULL, EINVAL); _BEGIN_SECURE_CRT_DEPRECATION_DISABLE _VALIDATE_RETURN_ERRCODE(_osplatform!=0, EINVAL); *pValue = _winminor; _END_SECURE_CRT_DEPRECATION_DISABLE return 0; } /*** *errno_t _get_wpgmptr() - get _wpgmptr, i.e. the module (EXE) wide name * *Purpose: * Get the value of _wpgmptr * *Entry: * wchar_t **pValue - pointer where to store the value * *Exit: * The error code * *Exceptions: * Input parameters are validated. Refer to the validation section of the function. * *******************************************************************************/ errno_t _get_wpgmptr(wchar_t **pValue) { /* validation section */ _VALIDATE_RETURN_ERRCODE(pValue != NULL, EINVAL); _BEGIN_SECURE_CRT_DEPRECATION_DISABLE _VALIDATE_RETURN_ERRCODE(_wpgmptr != NULL, EINVAL); *pValue = _wpgmptr; _END_SECURE_CRT_DEPRECATION_DISABLE return 0; } /*** *errno_t _get_pgmptr() - get _pgmptr, i.e. the module (EXE) name * *Purpose: * Get the value of _pgmptr * *Entry: * char **pValue - pointer where to store the value * *Exit: * The error code * *Exceptions: * Input parameters are validated. Refer to the validation section of the function. * *******************************************************************************/ errno_t _get_pgmptr(char **pValue) { /* validation section */ _VALIDATE_RETURN_ERRCODE(pValue != NULL, EINVAL); _BEGIN_SECURE_CRT_DEPRECATION_DISABLE _VALIDATE_RETURN_ERRCODE(_pgmptr != NULL, EINVAL); *pValue = _pgmptr; _END_SECURE_CRT_DEPRECATION_DISABLE return 0; }