- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我正在研究小型 Windows 异常处理引擎,试图从系统中收集最多的信息,包括 C++ 异常 RTTI。
在 MSVS 2015 编译的 32 位 VectoredExceptionHandler 中,我成功地获得了指向所抛出类型的 RTTI 的 std::type_info 指针。它可以很容易地在 ((_ThrowInfo*) ExceptionPointers->ExceptionRecord->ExceptionInformation[2])->pCatchableTypeArray->arrayOfCatchableTypes[0]
中找到(参见 classic article of Raymond Chen ,MS 的 ehdata.h
文件和许多其他文件中的一些定义)。该方法基于获取pCatchableTypeArray
MSVC 内置成员 _ThrowInfo
由编译器构建的结构数据。
但是在64位环境下,_ThrowInfo
不包含直接的 RTTI:不幸的是,pCatchableTypeArray
一片空白。在反汇编窗口中,我看到即使在调用 _CxxThrowException
之前它也是 NULL , 主要 MS throw 处理程序。我搜索了许多关于 MSVC 中使用的新 64 位异常处理机制的文章,但没有关于 RTTI 的信息。但也许我错过了一些东西。
是否有任何方法可以获得在 64 位 MSVC 环境中工作的 vector 异常处理程序中抛出的 C++ 异常的 std::type_info(或简单的类型名称)?
这是转储 32 位和 64 位异常信息的输出:
32 位(RTTI 成功):
VectoredExceptionHandler(): Start
exc->ExceptionCode = 0xE06D7363
exc->ExceptionAddress = 0x74E2C54F
exc->NumberParameters = 3
exc->ExceptionInformation[0] = 0x19930520 (sig)
exc->ExceptionInformation[1] = 0x004FFD9C (object)
exc->ExceptionInformation[2] = 0x003AD85C (throwInfo)
exc->ExceptionInformation[3] = 0x005B18F8 (module)
throwInfo->attributes = 0x00000000
throwInfo->pmfnUnwind = 0x00000000
throwInfo->pForwardCompat = 0x00000000
throwInfo->pCatchableTypeArray = 0x003AD870
object = 0x004FFD9C
throwInfo = 0x003AD85C
module = 0x00000000
throwInfo->pCatchableTypeArray = 0x003AD870
cArray = 0x003AD870
cArray->arrayOfCatchableTypes[0] = 0x003AD878
cType = 0x003AD878
cType->pType = 0x003AFA70
type = 0x003AFA70
type->name() = "struct `int __cdecl main(void)'::`2'::meow_exception"
cType->sizeOrOffset = 4
VectoredExceptionHandler(): End
main(): catch (meow_exception { 3 })
VectoredExceptionHandler(): Start
exc->ExceptionCode = 0xE06D7363
exc->ExceptionAddress = 0x000007FEFCE0A06D
exc->NumberParameters = 4
exc->ExceptionInformation[0] = 0x0000000019930520 (sig)
exc->ExceptionInformation[1] = 0x000000000025FBE0 (object)
exc->ExceptionInformation[2] = 0x000000013FC52AB0 (throwInfo)
exc->ExceptionInformation[3] = 0x000000013FBE0000 (module)
module = 0x000000013FBE0000
throwInfo->attributes = 0x00000000
throwInfo->pmfnUnwind = 0x0000000000000000
throwInfo->pForwardCompat = 0x0000000000072AD0
throwInfo->pCatchableTypeArray = 0x0000000000000000
VectoredExceptionHandler(): End
main(): catch (meow_exception { 3 })
#include <stdio.h>
#include <typeinfo>
#include <windows.h>
//--------------------------------------------------------------------------------------------------
const unsigned EXCEPTION_CPP_MICROSOFT = 0xE06D7363, // '?msc'
EXCEPTION_CPP_MICROSOFT_EH_MAGIC_NUMBER1 = 0x19930520, // '?msc' version magic, see ehdata.h
EXCEPTION_OUTPUT_DEBUG_STRING = 0x40010006, // OutputDebugString() call
EXCEPTION_THREAD_NAME = 0x406D1388; // Passing name of thread to the debugger
void OutputDebugPrintf (const char* format, ...);
//--------------------------------------------------------------------------------------------------
long WINAPI VectoredExceptionHandler (EXCEPTION_POINTERS* pointers)
{
const EXCEPTION_RECORD* exc = pointers->ExceptionRecord;
if (exc->ExceptionCode == EXCEPTION_OUTPUT_DEBUG_STRING ||
exc->ExceptionCode == EXCEPTION_THREAD_NAME)
return EXCEPTION_CONTINUE_SEARCH;
OutputDebugPrintf ("\n%s(): Start\n\n", __func__);
OutputDebugPrintf ("exc->ExceptionCode = 0x%X\n", exc->ExceptionCode);
OutputDebugPrintf ("exc->ExceptionAddress = 0x%p\n", exc->ExceptionAddress);
if (exc->ExceptionInformation[0] == EXCEPTION_CPP_MICROSOFT_EH_MAGIC_NUMBER1 &&
exc->NumberParameters >= 3)
{
OutputDebugPrintf ("exc->NumberParameters = %u\n", exc->NumberParameters);
OutputDebugPrintf ("exc->ExceptionInformation[0] = 0x%p (sig)\n", (void*) exc->ExceptionInformation[0]);
OutputDebugPrintf ("exc->ExceptionInformation[1] = 0x%p (object)\n", (void*) exc->ExceptionInformation[1]);
OutputDebugPrintf ("exc->ExceptionInformation[2] = 0x%p (throwInfo)\n", (void*) exc->ExceptionInformation[2]);
OutputDebugPrintf ("exc->ExceptionInformation[3] = 0x%p (module)\n", (void*) exc->ExceptionInformation[3]);
OutputDebugPrintf ("\n");
HMODULE module = (exc->NumberParameters >= 4)? (HMODULE) exc->ExceptionInformation[3] : NULL;
if (module)
{
OutputDebugPrintf ("module = 0x%p\n", module);
OutputDebugPrintf ("\n");
}
const _ThrowInfo* throwInfo = (const _ThrowInfo*) exc->ExceptionInformation[2];
if (throwInfo)
{
OutputDebugPrintf ("throwInfo->attributes = 0x%08X\n", throwInfo->attributes);
OutputDebugPrintf ("throwInfo->pmfnUnwind = 0x%p\n", throwInfo->pmfnUnwind);
OutputDebugPrintf ("throwInfo->pForwardCompat = 0x%p\n", throwInfo->pForwardCompat);
OutputDebugPrintf ("throwInfo->pCatchableTypeArray = 0x%p\n", throwInfo->pCatchableTypeArray);
OutputDebugPrintf ("\n");
}
if (throwInfo && throwInfo->pCatchableTypeArray)
{
#define RVA_TO_VA_(type, addr) ( (type) ((uintptr_t) module + (uintptr_t) (addr)) )
OutputDebugPrintf ("object = 0x%p\n", (void*) exc->ExceptionInformation[1]);
OutputDebugPrintf ("throwInfo = 0x%p\n", (void*) throwInfo);
OutputDebugPrintf ("module = 0x%p\n", (void*) module);
OutputDebugPrintf ("\n");
const _CatchableTypeArray* cArray = RVA_TO_VA_(const _CatchableTypeArray*, throwInfo->pCatchableTypeArray);
OutputDebugPrintf ("throwInfo->pCatchableTypeArray = 0x%p\n", (void*) throwInfo->pCatchableTypeArray);
OutputDebugPrintf ("cArray = 0x%p\n\n", (void*) cArray);
const _CatchableType* cType = RVA_TO_VA_(const _CatchableType*, cArray->arrayOfCatchableTypes[0]);
OutputDebugPrintf ("cArray->arrayOfCatchableTypes[0] = 0x%p\n", (void*) cArray->arrayOfCatchableTypes[0]);
OutputDebugPrintf ("cType = 0x%p\n\n", (void*) cType);
const std::type_info* type = RVA_TO_VA_(const std::type_info*, cType->pType);
OutputDebugPrintf ("cType->pType = 0x%p\n", (void*) cType->pType);
OutputDebugPrintf ("type = 0x%p\n\n", (void*) type);
OutputDebugPrintf ("type->name() = \"%s\"\n", type->name());
OutputDebugPrintf ("cType->sizeOrOffset = %zu\n\n", (size_t) cType->sizeOrOffset);
#undef RVA_TO_VA_
}
}
OutputDebugPrintf ("%s(): End\n", __func__);
return EXCEPTION_CONTINUE_SEARCH;
}
//--------------------------------------------------------------------------------------------------
void OutputDebugPrintf (const char* format, ...)
{
static char buf [1024] = "";
va_list arg; va_start (arg, format);
_vsnprintf_s (buf, sizeof (buf) - 1, _TRUNCATE, format, arg);
va_end (arg);
OutputDebugString (buf);
printf ("%s", buf);
}
//--------------------------------------------------------------------------------------------------
int main()
{
OutputDebugPrintf ("\n%s(): Start\n", __func__);
AddVectoredExceptionHandler (1, VectoredExceptionHandler);
struct meow_exception { int code = 3; };
try
{
throw meow_exception();
}
catch (const meow_exception& e)
{
OutputDebugPrintf ("\n%s(): catch (meow_exception { %d })\n", __func__, e.code);
}
catch (...)
{
OutputDebugPrintf ("\n%s(): catch (...)\n", __func__);
}
OutputDebugPrintf ("\n%s(): End\n", __func__);
return 0;
}
// Microsoft (R) C/C++ Optimizing Compiler Version 19.00.24213.1 (part of VS 2015 SP3)
cl /c code.cpp /EHsc /W4
link code.obj kernel32.lib /machine:x86 /subsystem:console /debug
最佳答案
为了解决这个问题,我深入研究并发现了一些关于 MSVC 64 位模式的有趣的事情。我发现我们不能依赖 64 位模式下的内部编译器预定义类型,因为其中一些是错误的。
我对比了一些编译器预定义的内部结构的定义,比如_ThrowInfo
和 _CatchableType
, 编译器生成的程序集列表正在运行 /FAs
命令行开关。
以下是从程序集文件中提取的这些结构的实例(以下是 MSVC 2015 64 位版本):
;---------------------------------------------------------------------------------------
; Listing generated by Microsoft Optimizing Compiler Version 19.00.24213.1
; Simplified: many lines skipped, some sections reordered etc -- Ded
;---------------------------------------------------------------------------------------
main proc
; struct meow_exception { int code = 3; };
;
; try
; {
; throw meow_exception();
...
lea rdx, OFFSET FLAT:_TI1?AUmeow_exception@?1??main@@YAHXZ@ ; lea &_ThrowInfo
lea rcx, QWORD PTR $T1[rsp]
call _CxxThrowException
;---------------------------------------------------------------------------------------
_TI1?AUmeow_exception@?1??main@@YAHXZ@ ; _ThrowInfo
DD 0
DD 0
DD 0
DD imagerel _CTA1?AUmeow_exception@?1??main@@YAHXZ@ ; &_CatchableTypeArray
;---------------------------------------------------------------------------------------
_CTA1?AUmeow_exception@?1??main@@YAHXZ@ ; _CatchableTypeArray
DD 1
DD imagerel _CT??_R0?AUmeow_exception@?1??main@@YAHXZ@@84 ; &_CatchableType
;---------------------------------------------------------------------------------------
_CT??_R0?AUmeow_exception@?1??main@@YAHXZ@@84 ; _CatchableType
DD 0
DD imagerel ??_R0?AUmeow_exception@?1??main@@YAHXZ@@8 ; &_TypeDescriptor
DD 0
DD 0ffffffffh
ORG $+4
DD 04h
DD 0
;---------------------------------------------------------------------------------------
??_R0?AUmeow_exception@?1??main@@YAHXZ@@8 ; _TypeDescriptor (aka std::type_info)
DQ FLAT:??_7type_info@@6B@
DQ 0
DB '.?AUmeow_exception@?1??main@@YAHXZ@', 0 ; Mangled type name
;---------------------------------------------------------------------------------------
FLAT
修饰符代替了
imagerel
,
DD
中的
DQ
代替了
_TypeDescriptor
)。
ehdata.h
的预定义类型进行比较。文件(例如,参见 MSVC 2013 中的
well-known source by Geoff Chappell 或
C:\Program Files (x86)\Microsoft Visual Studio 12.0\VC\crt\src\ehdata.h
文件;不幸的是,该文件在 MSVC 2015 运行时源中不存在):
typedef const struct _s__ThrowInfo
{
unsigned int attributes;
_PMFN pmfnUnwind; // this is a pointer!
int (__cdecl *pForwardCompat) (...); // this is a pointer too!
_CatchableTypeArray *pCatchableTypeArray; // this is a pointer too!
}
_ThrowInfo;
typedef const struct _s__CatchableType
{
unsigned int properties;
_TypeDescriptor *pType; // this is a pointer too!
_PMD thisDisplacement;
int sizeOrOffset;
_PMFN copyFunction; // this is a pointer too!
}
_CatchableType;
imagerel
地址修饰符,这些是指 RVA。
这些 RVA 是 32 位的 并定义为 32 位
DD
关键词。
_ThrowInfo
或
_CatchableType
上面)
不对应于汇编器二进制布局。 从 C++ 端看,这些结构的大小更大,而且字段偏移量也是错误的。
namespace CORRECT
{
struct ThrowInfo
{
__int32 attributes;
__int32 pmfnUnwind; // now this is 32-bit RVA
__int32 pForwardCompat; // now this is 32-bit RVA
__int32 pCatchableTypeArray; // now this is 32-bit RVA
};
struct CatchableType
{
__int32 properties;
__int32 pType; // now this is 32-bit RVA
_PMD thisDisplacement;
__int32 sizeOrOffset;
__int32 copyFunction; // now this is 32-bit RVA
};
}
_ThrowInfo
的内容和
_CatchableType
使用内部定义和我自己的定义。结果如下(MSVC 2015 64 位):
exc->ExceptionCode = 0xE06D7363
exc->ExceptionAddress = 0x000007FEFD69A06D
exc->NumberParameters = 4
exc->ExceptionInformation[0] = 0x0000000019930520 (sig)
exc->ExceptionInformation[1] = 0x00000000002BF8B0 (object)
exc->ExceptionInformation[2] = 0x000000013F9C4210 (throwInfo)
exc->ExceptionInformation[3] = 0x000000013F950000 (module)
Built-in: _ThrowInfo, size 28
_throwInfo->attributes = 0x00000000 [ofs: 0, size: 4, type: unsigned int]
_throwInfo->pmfnUnwind = 0x00000000 [ofs: 4, size: 8, type: void (__cdecl*)(void * __ptr64)]
_throwInfo->pForwardCompat = 0x00074230 [ofs: 12, size: 8, type: int (__cdecl*)(void)]
_throwInfo->pCatchableTypeArray = 0x00000000 [ofs: 20, size: 8, type: struct _s__CatchableTypeArray const * __ptr64]
Custom: CORRECT::ThrowInfo, size 16
throwInfo->attributes = 0x00000000 [ofs: 0, size: 4, type: int]
throwInfo->pmfnUnwind = 0x00000000 [ofs: 4, size: 4, type: int]
throwInfo->pForwardCompat = 0x00000000 [ofs: 8, size: 4, type: int]
throwInfo->pCatchableTypeArray = 0x00074230 [ofs: 12, size: 4, type: int]
throwInfo->pCatchableTypeArray = 0x0000000000074230
cArray = 0x000000013F9C4230
Built-in: _CatchableType, size 36
_cType->properties = 0x00000000 [ofs: 0, size: 4, type: unsigned int]
_cType->pType = 0x00075D58 [ofs: 4, size: 8, type: struct _TypeDescriptor * __ptr64]
_cType->thisDisplacement.mdisp = 0xFFFFFFFF [ofs: 12, size: 4, type: int]
_cType->thisDisplacement.pdisp = 0x00000000 [ofs: 16, size: 4, type: int]
_cType->thisDisplacement.vdisp = 0x00000004 [ofs: 20, size: 4, type: int]
_cType->sizeOrOffset = 0x00000000 [ofs: 24, size: 4, type: int]
_cType->copyFunction = 0x00000000 [ofs: 28, size: 8, type: void (__cdecl*)(void * __ptr64)]
Custom: CORRECT::CatchableType, size 28
cType->properties = 0x00000000 [ofs: 0, size: 4, type: int]
cType->pType = 0x00075D58 [ofs: 4, size: 4, type: int]
cType->thisDisplacement.mdisp = 0x00000000 [ofs: 8, size: 4, type: int]
cType->thisDisplacement.pdisp = 0xFFFFFFFF [ofs: 12, size: 4, type: int]
cType->thisDisplacement.vdisp = 0x00000000 [ofs: 16, size: 4, type: int]
cType->sizeOrOffset = 0x00000004 [ofs: 20, size: 4, type: int]
cType->copyFunction = 0x00000000 [ofs: 24, size: 4, type: int]
cArray->arrayOfCatchableTypes[0] = 0x0000000000074240
cType = 0x000000013F9C4240
cType->pType = 0x0000000000075D58
type = 0x000000013F9C5D58
type->name() = "struct `int __cdecl main(void)'::`2'::meow_exception"
cType->sizeOrOffset = 4
CORRECT::
定义,很容易获得所需的正确 RTTI。
C:\Program Files (x86)\Microsoft Visual Studio 12.0\VC\crt\src\ehdata.h
来自 MSVC 2013 运行时源的文件包含条件预处理器指令
#ifdef _EH_RELATIVE_OFFSETS
替换指向
__int32
的指针抵消。但是预定义的内部编译器类型总是包含 64 位错误的指针。
#define _EH_RELATIVE_OFFSETS
并使用上述
ehdata.h
)。之后,不要忘记通过添加
ImageBase
手动将 RVA 转换为通用 C++ 指针。地址照常。但是
不应信任此类结构中的指针成员 以及包含此类指针的定义,因为
它们不反射(reflect)真正的 64 位二进制布局。
#include <stdio.h>
#include <typeinfo>
#include <stdexcept>
#include <windows.h>
//------------------------------------------------------------------------------------------------------------------------------
//! These definitions are based on assembly listings produded by the compiler (/FAs) rather than built-in ones
//! @{
#pragma pack (push, 4)
namespace CORRECT
{
struct CatchableType
{
__int32 properties;
__int32 pType;
_PMD thisDisplacement;
__int32 sizeOrOffset;
__int32 copyFunction;
};
struct ThrowInfo
{
__int32 attributes;
__int32 pmfnUnwind;
__int32 pForwardCompat;
__int32 pCatchableTypeArray;
};
}
#pragma pack (pop)
//! @}
//------------------------------------------------------------------------------------------------------------------------------
const unsigned EXCEPTION_CPP_MICROSOFT = 0xE06D7363, // '?msc'
EXCEPTION_CPP_MICROSOFT_EH_MAGIC_NUMBER1 = 0x19930520, // '?msc' version magic, see ehdata.h
EXCEPTION_OUTPUT_DEBUG_STRING = 0x40010006, // OutputDebugString() call
EXCEPTION_THREAD_NAME = 0x406D1388; // Passing name of thread to the debugger
void OutputDebugPrintf (const char* format, ...);
//------------------------------------------------------------------------------------------------------------------------------
long WINAPI VectoredExceptionHandler (EXCEPTION_POINTERS* pointers)
{
const EXCEPTION_RECORD* exc = pointers->ExceptionRecord;
if (exc->ExceptionCode == EXCEPTION_OUTPUT_DEBUG_STRING ||
exc->ExceptionCode == EXCEPTION_THREAD_NAME)
return EXCEPTION_CONTINUE_SEARCH;
OutputDebugPrintf ("\n%s(): Start\n\n", __FUNCTION__);
OutputDebugPrintf ("exc->ExceptionCode = 0x%X\n", exc->ExceptionCode);
OutputDebugPrintf ("exc->ExceptionAddress = 0x%p\n", exc->ExceptionAddress);
if (exc->ExceptionInformation[0] == EXCEPTION_CPP_MICROSOFT_EH_MAGIC_NUMBER1 &&
exc->NumberParameters >= 3)
{
OutputDebugPrintf ("exc->NumberParameters = %u\n", exc->NumberParameters);
OutputDebugPrintf ("exc->ExceptionInformation[0] = 0x%p (sig)\n", (void*) exc->ExceptionInformation[0]);
OutputDebugPrintf ("exc->ExceptionInformation[1] = 0x%p (object)\n", (void*) exc->ExceptionInformation[1]);
OutputDebugPrintf ("exc->ExceptionInformation[2] = 0x%p (throwInfo)\n", (void*) exc->ExceptionInformation[2]);
if (exc->NumberParameters >= 4)
OutputDebugPrintf ("exc->ExceptionInformation[3] = 0x%p (module)\n", (void*) exc->ExceptionInformation[3]);
OutputDebugPrintf ("\n");
HMODULE module = (exc->NumberParameters >= 4)? (HMODULE) exc->ExceptionInformation[3] : NULL;
#define RVA_TO_VA_(type, addr) ( (type) ((uintptr_t) module + (uintptr_t) (addr)) )
const _ThrowInfo* _throwInfo = (const _ThrowInfo*) exc->ExceptionInformation[2];
const CORRECT::ThrowInfo* throwInfo = (const CORRECT::ThrowInfo*) exc->ExceptionInformation[2];
#define DUMP_(var, struc, field) OutputDebugPrintf ("%-32s = 0x%08X [ofs: %2u, size: %u, type: %s]\n", \
#var "->" #field, (var)->field, \
offsetof (struc, field), sizeof ((var)->field), \
typeid ((var)->field) .name());
if (_throwInfo)
{
OutputDebugPrintf ("Built-in: _ThrowInfo, size %u\n", sizeof (_ThrowInfo));
DUMP_ (_throwInfo, _ThrowInfo, attributes);
DUMP_ (_throwInfo, _ThrowInfo, pmfnUnwind);
DUMP_ (_throwInfo, _ThrowInfo, pForwardCompat);
DUMP_ (_throwInfo, _ThrowInfo, pCatchableTypeArray);
OutputDebugPrintf ("\n");
}
else
OutputDebugPrintf ("_throwInfo is NULL\n");
if (throwInfo)
{
OutputDebugPrintf ("Custom: CORRECT::ThrowInfo, size %u\n", sizeof (CORRECT::ThrowInfo));
DUMP_ ( throwInfo, CORRECT::ThrowInfo, attributes);
DUMP_ ( throwInfo, CORRECT::ThrowInfo, pmfnUnwind);
DUMP_ ( throwInfo, CORRECT::ThrowInfo, pForwardCompat);
DUMP_ ( throwInfo, CORRECT::ThrowInfo, pCatchableTypeArray);
OutputDebugPrintf ("\n");
}
else
OutputDebugPrintf ("throwInfo is NULL\n");
if (throwInfo)
{
const _CatchableTypeArray* cArray = RVA_TO_VA_(const _CatchableTypeArray*, throwInfo->pCatchableTypeArray);
OutputDebugPrintf ("throwInfo->pCatchableTypeArray = 0x%p\n", (void*)(ptrdiff_t) throwInfo->pCatchableTypeArray);
OutputDebugPrintf ("cArray = 0x%p\n\n", (void*) cArray);
const _CatchableType* _cType = RVA_TO_VA_(const _CatchableType*, cArray->arrayOfCatchableTypes[0]);
const CORRECT::CatchableType* cType = RVA_TO_VA_(const CORRECT::CatchableType*, cArray->arrayOfCatchableTypes[0]);
OutputDebugPrintf ("Built-in: _CatchableType, size %u\n", sizeof (_CatchableType));
DUMP_ (_cType, _CatchableType, properties);
DUMP_ (_cType, _CatchableType, pType);
DUMP_ (_cType, _CatchableType, thisDisplacement.mdisp);
DUMP_ (_cType, _CatchableType, thisDisplacement.pdisp);
DUMP_ (_cType, _CatchableType, thisDisplacement.vdisp);
DUMP_ (_cType, _CatchableType, sizeOrOffset);
DUMP_ (_cType, _CatchableType, copyFunction);
OutputDebugPrintf ("\n");
OutputDebugPrintf ("Custom: CORRECT::CatchableType, size %u\n", sizeof (CORRECT::CatchableType));
DUMP_ ( cType, CORRECT::CatchableType, properties);
DUMP_ ( cType, CORRECT::CatchableType, pType);
DUMP_ ( cType, CORRECT::CatchableType, thisDisplacement.mdisp);
DUMP_ ( cType, CORRECT::CatchableType, thisDisplacement.pdisp);
DUMP_ ( cType, CORRECT::CatchableType, thisDisplacement.vdisp);
DUMP_ ( cType, CORRECT::CatchableType, sizeOrOffset);
DUMP_ ( cType, CORRECT::CatchableType, copyFunction);
OutputDebugPrintf ("\n");
OutputDebugPrintf ("cArray->arrayOfCatchableTypes[0] = 0x%p\n", (void*) cArray->arrayOfCatchableTypes[0]);
OutputDebugPrintf ("cType = 0x%p\n\n", (void*) cType);
const std::type_info* type = RVA_TO_VA_(const std::type_info*, cType->pType);
OutputDebugPrintf ("cType->pType = 0x%p\n", (void*)(ptrdiff_t) cType->pType);
OutputDebugPrintf ("type = 0x%p\n\n", (void*) type);
OutputDebugPrintf ("type->name() = \"%s\"\n", type->name());
OutputDebugPrintf ("cType->sizeOrOffset = %u\n\n", (unsigned) cType->sizeOrOffset);
}
#undef DUMP_
#undef RVA_TO_VA_
}
OutputDebugPrintf ("%s(): End\n", __FUNCTION__);
return EXCEPTION_CONTINUE_SEARCH;
}
//------------------------------------------------------------------------------------------------------------------------------
void OutputDebugPrintf (const char* format, ...)
{
static char buf [1024] = "";
va_list arg; va_start (arg, format);
_vsnprintf_s (buf, sizeof (buf) - 1, _TRUNCATE, format, arg);
va_end (arg);
OutputDebugString (buf);
printf ("%s", buf);
}
//------------------------------------------------------------------------------------------------------------------------------
int main()
{
OutputDebugPrintf ("\nCompiled with MSVC %d, %d-bit\n", _MSC_VER, 8 * sizeof (void*));
OutputDebugPrintf ("\n%s(): Start\n", __FUNCTION__);
AddVectoredExceptionHandler (1, VectoredExceptionHandler);
struct meow_exception { int code; meow_exception() : code (3) {} };
try
{
throw meow_exception();
}
catch (const meow_exception& e)
{
OutputDebugPrintf ("\n%s(): catch (meow_exception { %d })\n", __FUNCTION__, e.code);
}
catch (...)
{
OutputDebugPrintf ("\n%s(): catch (...)\n", __FUNCTION__);
}
OutputDebugPrintf ("\n%s(): End\n", __FUNCTION__);
return 0;
}
关于Windows 64 位 VectoredExceptionHandler 中的 C++ RTTI,MS Visual Studio 2015,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39113168/
为什么我可以在控制台 window.window.window.window 中执行此操作并无限追加 .window 并返回 DOM 窗口? 最佳答案 因为 window 对象有一个指向它自身的 wi
Windows管理员用户和系统用户之间有什么权限区别吗? 有些时候,我必须将 cmd 窗口提升到系统权限才能删除一些文件。这可能是因为系统用户锁定了文件,或者系统用户可能具有更高的访问权限,我希望找出
按照目前的情况,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引发辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visit the
Windows 服务和 Windows 进程之间的区别是什么? 最佳答案 服务是真正的 Windows 进程,没有区别。服务的唯一特殊之处在于它由操作系统启动并在单独的 session 中运行。一个独
我有一个 Windows 网络 (peer-2-peer) 以及 Active Directory,我需要记录向服务器发送任何类型打印的用户的名称。我想编写一个程序来记录他们的用户名和/或他们各自的
当我让一个 Windows 服务尝试安装另一个 Windows 服务时遇到问题。 具体来说,我有一个 TeamCity 代理在 Windows 2008 AWS 实例上为我运行测试。这些测试是用 Ja
我创建了一个应用程序来接收广播的 Windows 消息,效果很好。当我把它变成一个服务、安装它并启动服务时,该服务没有收到消息。 最佳答案 服务可能必须被授予访问桌面的权限。从服务属性、“登录”选项卡
我正在使用 Delphi 2010 编写应用程序。我希望在 Windows 启动时启动我的应用程序。我需要它在最新版本的 Windows XP、7.0 和最新的服务器中工作。 将其存储在以下关键工作下
我想开发一个适用于所有三个版本的 Windows XP、Vista 和 7 的应用程序。该应用程序允许人们选择要打开的文件,并允许他们在某些操作后保存文件。三个版本的 Windows 中的每一个都有不
对于\Windows\中的文件类型与\Windows\System32 中的文件类型是否有标准约定? 我正在开发一个 SDK,其中包含各种 DLL、帮助程序 exe 和 Windows 服务 exe。
要求是,必须在 WINDOWS7 机器上配置自动登录,但是这个自动登录应该等待(即延迟)直到另一个 Windows 服务发出继续自动登录的信号。 我使用了自定义凭据提供程序,它在其中等待另一个 Win
很抱歉,这不是一个大问题,而是更多的帮助人们解决这些特定问题的方法。我正在解决的问题要求使用串行I/O,但主要在Windows CE 6.0下运行。但是,最近有人问我是否也可以在Windows下运行该
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 这个问题似乎不是关于 a specific programming problem, a softwar
不幸的是 SC 命令在 W2000 上还不可用,所以我不能使用它。 我正在尝试检查服务是否在 W2000 服务器上运行,如果它没有运行,脚本应该能够启动该服务。 如何在 Windows 2000 上执
如何在登录到 Windows 之前启动 Windows 窗体应用程序?是否可以在登录到 Windows 之前启动 Windows 窗体应用程序?如果不是,我是否有机会在登录前启动 Windows 服务
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 这个问题似乎不是关于 a specific programming problem, a softwar
我想在 XML 文件中区分 Windows XP 和 Windows 7。我想我会在 XML 中为它使用一个环境变量。 但是我找不到在 Windows 中定义的任何系统环境变量来提供此信息。 我看到了
有谁知道我可以在注册表中的哪个位置检查机器上是否安装了这些应用程序: Windows 通讯录 Windows 联系人 最佳答案 来自 Microsoft:我知道它说的是 win 95,但 reg 是一
我正在尝试从我的 Windows 服务器调用放置在远程 Windows 服务器上的批处理文件。我在远程服务器上安装了 freeSSHd。我尝试使用 putty/plink 但没有结果。 我使用的命令语
( 大家好。我是 Windows 编程的新手,所以如果已经有人问过我,我提前道歉,我只是不知道要搜索什么,但这个问题一直让我发疯,我知道有人可能真的很容易回答这个问题。) 我的公司有一个在 Windo
我是一名优秀的程序员,十分优秀!