171 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			171 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| 
 | |
| #define WIN32_LEAN_AND_MEAN
 | |
| #include <Windows.h>
 | |
| #include <stdlib.h>
 | |
| 
 | |
| // Fix from from https://stackoverflow.com/a/34655235.
 | |
| //
 | |
| // __iob_func required by the MySQL we use,
 | |
| // but no longer exists in the VS 14.0+ crt.
 | |
| 
 | |
| #pragma comment(lib, "DbgHelp.lib")
 | |
| #pragma warning(disable:4091) // 'typedef ': ignored on left of '' when no variable is declared
 | |
| #include <DbgHelp.h>
 | |
| #include <corecrt_wstdio.h>
 | |
| 
 | |
| #define GET_CURRENT_CONTEXT(c, contextFlags) \
 | |
|   do { \
 | |
| 	c.ContextFlags = contextFlags; \
 | |
| 	__asm    call x \
 | |
| 	__asm x: pop eax \
 | |
| 	__asm    mov c.Eip, eax \
 | |
| 	__asm    mov c.Ebp, ebp \
 | |
| 	__asm    mov c.Esp, esp \
 | |
|   } while(0);
 | |
| 
 | |
| 
 | |
| FILE * __cdecl __iob_func(void)
 | |
| {
 | |
| 	CONTEXT c = { 0 };
 | |
| 	STACKFRAME64 s = { 0 };
 | |
| 	DWORD imageType;
 | |
| 	HANDLE hThread = GetCurrentThread();
 | |
| 	HANDLE hProcess = GetCurrentProcess();
 | |
| 
 | |
| 	GET_CURRENT_CONTEXT(c, CONTEXT_FULL);
 | |
| 
 | |
| 	imageType = IMAGE_FILE_MACHINE_I386;
 | |
| 	s.AddrPC.Offset = c.Eip;
 | |
| 	s.AddrPC.Mode = AddrModeFlat;
 | |
| 	s.AddrFrame.Offset = c.Ebp;
 | |
| 	s.AddrFrame.Mode = AddrModeFlat;
 | |
| 	s.AddrStack.Offset = c.Esp;
 | |
| 	s.AddrStack.Mode = AddrModeFlat;
 | |
| 
 | |
| 	if (!StackWalk64(imageType, hProcess, hThread, &s, &c, NULL, SymFunctionTableAccess64, SymGetModuleBase64, NULL))
 | |
| 	{
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	if (s.AddrReturn.Offset == 0)
 | |
| 	{
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	{
 | |
| 		unsigned char const * assembly = (unsigned char const *)(s.AddrReturn.Offset);
 | |
| 
 | |
| 		if (*assembly == 0x83 && *(assembly + 1) == 0xC0 && (*(assembly + 2) == 0x20 || *(assembly + 2) == 0x40))
 | |
| 		{
 | |
| 			if (*(assembly + 2) == 32)
 | |
| 			{
 | |
| 				return (FILE*)((unsigned char *)stdout - 32);
 | |
| 			}
 | |
| 			if (*(assembly + 2) == 64)
 | |
| 			{
 | |
| 				return (FILE*)((unsigned char *)stderr - 64);
 | |
| 			}
 | |
| 
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			return stdin;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return NULL;
 | |
| }
 | |
| 
 | |
| // Adapted from dosmap.c in Visual Studio 12.0 CRT sources.
 | |
| //
 | |
| // The _dosmaperr function is required by the MySQL lib we use,
 | |
| // but no longer exists in the VS 14.0+ crt.
 | |
| 
 | |
| static struct errentry
 | |
| {
 | |
| 	DWORD oscode;      // OS return value
 | |
| 	int   errnocode;   // System V error code
 | |
| 
 | |
| } errtable[] =
 | |
| {
 | |
| 	{ ERROR_INVALID_FUNCTION,       EINVAL    },  /* 1 */
 | |
| 	{ ERROR_FILE_NOT_FOUND,         ENOENT    },  /* 2 */
 | |
| 	{ ERROR_PATH_NOT_FOUND,         ENOENT    },  /* 3 */
 | |
| 	{ ERROR_TOO_MANY_OPEN_FILES,    EMFILE    },  /* 4 */
 | |
| 	{ ERROR_ACCESS_DENIED,          EACCES    },  /* 5 */
 | |
| 	{ ERROR_INVALID_HANDLE,         EBADF     },  /* 6 */
 | |
| 	{ ERROR_ARENA_TRASHED,          ENOMEM    },  /* 7 */
 | |
| 	{ ERROR_NOT_ENOUGH_MEMORY,      ENOMEM    },  /* 8 */
 | |
| 	{ ERROR_INVALID_BLOCK,          ENOMEM    },  /* 9 */
 | |
| 	{ ERROR_BAD_ENVIRONMENT,        E2BIG     },  /* 10 */
 | |
| 	{ ERROR_BAD_FORMAT,             ENOEXEC   },  /* 11 */
 | |
| 	{ ERROR_INVALID_ACCESS,         EINVAL    },  /* 12 */
 | |
| 	{ ERROR_INVALID_DATA,           EINVAL    },  /* 13 */
 | |
| 	{ ERROR_INVALID_DRIVE,          ENOENT    },  /* 15 */
 | |
| 	{ ERROR_CURRENT_DIRECTORY,      EACCES    },  /* 16 */
 | |
| 	{ ERROR_NOT_SAME_DEVICE,        EXDEV     },  /* 17 */
 | |
| 	{ ERROR_NO_MORE_FILES,          ENOENT    },  /* 18 */
 | |
| 	{ ERROR_LOCK_VIOLATION,         EACCES    },  /* 33 */
 | |
| 	{ ERROR_BAD_NETPATH,            ENOENT    },  /* 53 */
 | |
| 	{ ERROR_NETWORK_ACCESS_DENIED,  EACCES    },  /* 65 */
 | |
| 	{ ERROR_BAD_NET_NAME,           ENOENT    },  /* 67 */
 | |
| 	{ ERROR_FILE_EXISTS,            EEXIST    },  /* 80 */
 | |
| 	{ ERROR_CANNOT_MAKE,            EACCES    },  /* 82 */
 | |
| 	{ ERROR_FAIL_I24,               EACCES    },  /* 83 */
 | |
| 	{ ERROR_INVALID_PARAMETER,      EINVAL    },  /* 87 */
 | |
| 	{ ERROR_NO_PROC_SLOTS,          EAGAIN    },  /* 89 */
 | |
| 	{ ERROR_DRIVE_LOCKED,           EACCES    },  /* 108 */
 | |
| 	{ ERROR_BROKEN_PIPE,            EPIPE     },  /* 109 */
 | |
| 	{ ERROR_DISK_FULL,              ENOSPC    },  /* 112 */
 | |
| 	{ ERROR_INVALID_TARGET_HANDLE,  EBADF     },  /* 114 */
 | |
| 	{ ERROR_INVALID_HANDLE,         EINVAL    },  /* 124 */
 | |
| 	{ ERROR_WAIT_NO_CHILDREN,       ECHILD    },  /* 128 */
 | |
| 	{ ERROR_CHILD_NOT_COMPLETE,     ECHILD    },  /* 129 */
 | |
| 	{ ERROR_DIRECT_ACCESS_HANDLE,   EBADF     },  /* 130 */
 | |
| 	{ ERROR_NEGATIVE_SEEK,          EINVAL    },  /* 131 */
 | |
| 	{ ERROR_SEEK_ON_DEVICE,         EACCES    },  /* 132 */
 | |
| 	{ ERROR_DIR_NOT_EMPTY,          ENOTEMPTY },  /* 145 */
 | |
| 	{ ERROR_NOT_LOCKED,             EACCES    },  /* 158 */
 | |
| 	{ ERROR_BAD_PATHNAME,           ENOENT    },  /* 161 */
 | |
| 	{ ERROR_MAX_THRDS_REACHED,      EAGAIN    },  /* 164 */
 | |
| 	{ ERROR_LOCK_FAILED,            EACCES    },  /* 167 */
 | |
| 	{ ERROR_ALREADY_EXISTS,         EEXIST    },  /* 183 */
 | |
| 	{ ERROR_FILENAME_EXCED_RANGE,   ENOENT    },  /* 206 */
 | |
| 	{ ERROR_NESTING_NOT_ALLOWED,    EAGAIN    },  /* 215 */
 | |
| 	{ ERROR_NOT_ENOUGH_QUOTA,       ENOMEM    }   /* 1816 */
 | |
| };
 | |
| 
 | |
| // The following two constants must be the minimum and maximum
 | |
| // values in the (contiguous) range of Exec Failure errors.
 | |
| #define MIN_EXEC_ERROR ERROR_INVALID_STARTING_CODESEG
 | |
| #define MAX_EXEC_ERROR ERROR_INFLOOP_IN_RELOC_CHAIN
 | |
| 
 | |
| // These are the low and high value in the range of errors that are
 | |
| // access violations
 | |
| #define MIN_EACCES_RANGE ERROR_WRITE_PROTECT
 | |
| #define MAX_EACCES_RANGE ERROR_SHARING_BUFFER_EXCEEDED
 | |
| 
 | |
| void _dosmaperr(DWORD oserrno)
 | |
| {
 | |
| 	_doserrno = oserrno;
 | |
| 
 | |
| 	// Check the table for the OS error code
 | |
| 	for (size_t i = 0; i < _countof(errtable); ++i)
 | |
| 	{
 | |
| 		if (oserrno == errtable[i].oscode)
 | |
| 		{
 | |
| 			errno = errtable[i].errnocode;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// The error code wasn't in the table.  We check for a range of
 | |
| 	// EACCES errors or exec failure errors (ENOEXEC).  Otherwise
 | |
| 	// EINVAL is returned.
 | |
| 	if (oserrno >= MIN_EACCES_RANGE && oserrno <= MAX_EACCES_RANGE)
 | |
| 		errno = EACCES;
 | |
| 	else if (oserrno >= MIN_EXEC_ERROR && oserrno <= MAX_EXEC_ERROR)
 | |
| 		errno = ENOEXEC;
 | |
| 	else
 | |
| 		errno = EINVAL;
 | |
| }
 |