351 lines
10 KiB
C++
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();
|
|
}
|