2020-09-30 16:53:49 +02:00

351 lines
10 KiB
C++

/***
*error.cpp - RTC support
*
* Copyright (c) 1998-2001, Microsoft Corporation. All rights reserved.
*
*
*Revision History:
* 07-28-98 JWM Module incorporated into CRTs (from KFrei)
* 11-03-98 KBF added throw() to eliminate C++ EH code
* 05-11-99 KBF Error if RTC support define not enabled
* 05-26-99 KBF Added -RTCu stuff, _RTC_ prefix on all non-statics
* 11-30-99 PML Compile /Wp64 clean.
* 03-19-01 KBF Fix buffer overruns (vs7#227306), eliminate all /GS
* checks (vs7#224261).
* 03-26-01 PML Use GetVersionExA, not GetVersionEx (vs7#230286)
*
****/
#ifndef _RTC
#error RunTime Check support not enabled!
#endif
#include "rtcpriv.h"
#pragma intrinsic(strcpy)
#pragma intrinsic(strcat)
#pragma intrinsic(strlen)
static int __cdecl _IsDebuggerPresent();
int _RTC_ErrorLevels[_RTC_ILLEGAL] = {1,1,1,1};
static const char *_RTC_ErrorMessages[_RTC_ILLEGAL+1] =
{
"The value of ESP was not properly saved across a function "
"call. This is usually a result of calling a function "
"declared with one calling convention with a function "
"pointer declared with a different calling convention.\n\r",
"A cast to a smaller data type has caused a loss of data. "
"If this was intentional, you should mask the source of "
"the cast with the appropriate bitmask. For example: \n\r"
"\tchar c = (i & 0xFF);\n\r"
"Changing the code in this way will not affect the quality of the resulting optimized code.\n\r",
"Stack memory was corrupted\n\r",
"A local variable was used before it was initialized\n\r",
#ifdef _RTC_ADVMEM
"Referencing invalid memory\n\r",
"Referencing memory across different blocks\n\r",
#endif
"Unknown Runtime Check Error\n\r"
};
static const BOOL _RTC_NoFalsePositives[_RTC_ILLEGAL+1] =
{
TRUE, // ESP was trashed
FALSE, // Shortening convert
TRUE, // Stack corruption
TRUE, // Uninitialized use
#ifdef _RTC_ADVMEM
TRUE, // Invalid memory reference
FALSE, // Different memory blocks
#endif
TRUE // Illegal
};
// returns TRUE if debugger understands, FALSE if not
static BOOL
DebuggerProbe( DWORD dwLevelRequired ) throw()
{
EXCEPTION_VISUALCPP_DEBUG_INFO info;
BYTE bDebuggerListening = FALSE;
info.dwType = EXCEPTION_DEBUGGER_PROBE;
info.DebuggerProbe.dwLevelRequired = dwLevelRequired;
info.DebuggerProbe.pbDebuggerPresent = &bDebuggerListening;
__try
{
HelloVC( info );
}
__except(EXCEPTION_CONTINUE_EXECUTION)
{
}
return (BOOL)bDebuggerListening;
}
// returns TRUE if debugger reported it (or was ignored), FALSE if runtime needs to report it
static int
DebuggerRuntime( DWORD dwErrorNumber, BOOL bRealBug, PVOID pvReturnAddr, LPCWSTR pwMessage ) throw()
{
EXCEPTION_VISUALCPP_DEBUG_INFO info;
BYTE bDebuggerListening = FALSE;
info.dwType = EXCEPTION_DEBUGGER_RUNTIMECHECK;
info.RuntimeError.dwRuntimeNumber = dwErrorNumber;
info.RuntimeError.bRealBug = bRealBug;
info.RuntimeError.pvReturnAddress = pvReturnAddr;
info.RuntimeError.pbDebuggerPresent = &bDebuggerListening;
info.RuntimeError.pwRuntimeMessage = pwMessage;
__try
{
HelloVC( info );
}
__except(EXCEPTION_CONTINUE_EXECUTION)
{
}
return (BOOL)bDebuggerListening;
}
static void
failwithmessage(void *retaddr, int crttype, int errnum, const char *msg)
{
_RTC_error_fn fn = _RTC_GetErrorFunc(retaddr);
bool dobreak;
if (DebuggerProbe( EXCEPTION_DEBUGGER_RUNTIMECHECK ))
{
wchar_t *buf = (wchar_t*)_alloca(sizeof(wchar_t) * (strlen(msg) + 2));
int i;
for (i = 0; msg[i]; i++)
buf[i] = msg[i];
buf[i] = 0;
if (DebuggerRuntime(errnum, _RTC_NoFalsePositives[errnum], retaddr, buf))
return;
dobreak = false;
} else
dobreak = true;
if (!fn || (dobreak && _IsDebuggerPresent()))
DebugBreak();
else
{
char *srcName = (char*)_alloca(sizeof(char) * 513);
int lineNum;
char *moduleName;
_RTC_GetSrcLine(((DWORD)(uintptr_t)retaddr)-5, srcName, 512, &lineNum, &moduleName);
// We're just running - report it like the user setup (or the default way)
// If we don't recognize this type, it defaults to an error
if (fn(crttype, srcName, lineNum, moduleName,
"Run-Time Check Failure #%d - %s", errnum, msg) == 1)
DebugBreak();
}
}
void __cdecl
_RTC_Failure(void *retaddr, int errnum)
{
int crttype;
const char *msg;
if (errnum < _RTC_ILLEGAL && errnum >= 0) {
crttype = _RTC_ErrorLevels[errnum];
msg = _RTC_ErrorMessages[errnum];
} else {
crttype = 1;
msg = _RTC_ErrorMessages[_RTC_ILLEGAL];
errnum = _RTC_ILLEGAL;
}
// If we're running inside a debugger, raise an exception
if (crttype != _RTC_ERRTYPE_IGNORE)
{
failwithmessage(retaddr, crttype, errnum, msg);
}
}
static
char *IntToString(int i)
{
static char buf[15];
bool neg = i < 0;
int pos = 14;
buf[14] = 0;
do {
buf[--pos] = i % 10 + '0';
i /= 10;
} while (i);
if (neg)
buf[--pos] = '-';
return &buf[pos];
}
void __cdecl
_RTC_MemFailure(void *retaddr, int errnum, const void *assign)
{
char *srcName = (char*)_alloca(sizeof(char) * 513);
int lineNum;
char *moduleName;
int crttype = _RTC_ErrorLevels[errnum];
if (crttype == _RTC_ERRTYPE_IGNORE)
return;
_RTC_GetSrcLine(((DWORD)(uintptr_t)assign)-5, srcName, 512, &lineNum, &moduleName);
if (!lineNum)
_RTC_Failure(retaddr, errnum);
else
{
char *msg = (char*)_alloca(strlen(_RTC_ErrorMessages[errnum]) +
strlen(srcName) + strlen(moduleName) +
150);
strcpy(msg, _RTC_ErrorMessages[errnum]);
strcat(msg, "Invalid pointer was assigned at\n\rFile:\t");
strcat(msg, srcName);
strcat(msg, "\n\rLine:\t");
strcat(msg, IntToString(lineNum));
strcat(msg, "\n\rModule:\t");
strcat(msg, moduleName);
failwithmessage(retaddr, crttype, errnum, msg);
}
}
void __cdecl
_RTC_StackFailure(void *retaddr, const char *varname)
{
int crttype = _RTC_ErrorLevels[_RTC_CORRUPT_STACK];
if (crttype != _RTC_ERRTYPE_IGNORE)
{
char *msg = (char*)_alloca(strlen(varname) + 80);
strcpy(msg, "Stack around the variable '");
strcat(msg, varname);
strcat(msg, "' was corrupted.");
failwithmessage(retaddr, crttype, _RTC_CORRUPT_STACK, msg);
}
}
void __cdecl
_RTC_UninitUse(const char *varname)
{
int crttype = _RTC_ErrorLevels[_RTC_UNINIT_LOCAL_USE];
if (crttype != _RTC_ERRTYPE_IGNORE)
{
char *msg = (char*)_alloca(strlen(varname) + 80);
if (varname)
{
strcpy(msg, "The variable '");
strcat(msg, varname);
strcat(msg, "' is being used without being defined.");
} else
{
strcpy(msg, "A variable is being used without being defined.");
}
failwithmessage(_ReturnAddress(), crttype, _RTC_UNINIT_LOCAL_USE, msg);
}
}
/* The rest of this file just implements "IsDebuggerPresent" functionality */
#pragma pack (push, 1)
typedef struct _TIB {
PVOID ExceptionList;
PVOID StackLimit;
PVOID StackBase;
PVOID SubSystemTib;
PVOID Something1;
PVOID ArbitraryUserPointer;
struct _TIB* Self;
WORD Flags;
WORD Win16MutextCount;
PVOID DebugContext;
DWORD CurrentPriority;
DWORD MessageQueueSelector;
PVOID* TlsSlots; // most likely an array
} TIB;
#pragma pack (pop)
//
// Define function to return the current Thread Environment Block
// AXPMOD - v-caseyc 9/22/98 We use and intrinsic function _rdteb to get the Thread
// Information Block
#if defined(_M_ALPHA)
void *_rdteb(void);
#pragma intrinsic(_rdteb)
static _inline TIB* GetCurrentTib() { return (TIB*) _rdteb(); }
#else // If not _M_ALPHA (AXPMOD)
#pragma warning (disable:4035)
#define OffsetTib 0x18
static _inline TIB* GetCurrentTib() { __asm mov eax, fs:[OffsetTib] }
#pragma warning (default:4035)
#endif // _M_ALPHA (End AXPMOD)
#define DLL_NOT_FOUND_EXCEPTION (0xc0000135L)
typedef BOOL (WINAPI* NT_IS_DEBUGGER_PRESENT) ();
static NT_IS_DEBUGGER_PRESENT FnIsDebuggerPresent = NULL;
static PVOID
WinGetDebugContext()
{
return GetCurrentTib()->DebugContext;
}
// here's the Win95 version of IsDebuggerPresent
static BOOL WINAPI
Win95IsDebuggerPresent()
{
if (WinGetDebugContext ()) {
return TRUE;
} else {
return FALSE;
}
}
static BOOL
Initialize()
{
HINSTANCE hInst = NULL;
hInst = LoadLibrary ("Kernel32.dll");
FnIsDebuggerPresent =
(NT_IS_DEBUGGER_PRESENT) GetProcAddress (hInst, "IsDebuggerPresent");
if (!FnIsDebuggerPresent) {
OSVERSIONINFOA *VersionInfo = (OSVERSIONINFOA*)_alloca(sizeof(OSVERSIONINFOA));
VersionInfo->dwOSVersionInfoSize = sizeof (OSVERSIONINFOA);
if (GetVersionExA (VersionInfo) &&
VersionInfo->dwPlatformId == VER_PLATFORM_WIN32_WINDOWS &&
VersionInfo->dwMajorVersion == 4)
FnIsDebuggerPresent = Win95IsDebuggerPresent;
}
return !!(FnIsDebuggerPresent);
}
// This is a version of IsDebuggerPresent () that works for all Win32 platforms.
static int __cdecl
_IsDebuggerPresent()
{
static BOOL fInited = FALSE;
if (!fInited) {
if (!Initialize())
RaiseException (DLL_NOT_FOUND_EXCEPTION, 0, 0, NULL);
fInited = TRUE;
}
return FnIsDebuggerPresent();
}