[Kernel] Research on C++ exceptions.
This commit is contained in:
parent
64403d5dc6
commit
70b07712b2
|
@ -31,49 +31,101 @@ typedef struct {
|
||||||
} X_THREADNAME_INFO;
|
} X_THREADNAME_INFO;
|
||||||
static_assert_size(X_THREADNAME_INFO, 0x10);
|
static_assert_size(X_THREADNAME_INFO, 0x10);
|
||||||
|
|
||||||
void RtlRaiseException(pointer_t<X_EXCEPTION_RECORD> record) {
|
void HandleSetThreadName(pointer_t<X_EXCEPTION_RECORD> record) {
|
||||||
if (record->exception_code == 0x406D1388) {
|
// SetThreadName. FFS.
|
||||||
// SetThreadName. FFS.
|
// https://msdn.microsoft.com/en-us/library/xcb2z8hs.aspx
|
||||||
// https://msdn.microsoft.com/en-us/library/xcb2z8hs.aspx
|
|
||||||
|
|
||||||
// TODO(benvanik): check record->number_parameters to make sure it's a
|
// TODO(benvanik): check record->number_parameters to make sure it's a
|
||||||
// correct size.
|
// correct size.
|
||||||
auto thread_info =
|
auto thread_info =
|
||||||
reinterpret_cast<X_THREADNAME_INFO*>(&record->exception_information[0]);
|
reinterpret_cast<X_THREADNAME_INFO*>(&record->exception_information[0]);
|
||||||
|
|
||||||
assert_true(thread_info->type == 0x1000);
|
assert_true(thread_info->type == 0x1000);
|
||||||
|
|
||||||
if (!thread_info->name_ptr) {
|
if (!thread_info->name_ptr) {
|
||||||
XELOGD("SetThreadName called with null name_ptr");
|
XELOGD("SetThreadName called with null name_ptr");
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
auto name =
|
|
||||||
kernel_memory()->TranslateVirtual<const char*>(thread_info->name_ptr);
|
|
||||||
|
|
||||||
object_ref<XThread> thread;
|
|
||||||
if (thread_info->thread_id == -1) {
|
|
||||||
// Current thread.
|
|
||||||
thread = retain_object(XThread::GetCurrentThread());
|
|
||||||
} else {
|
|
||||||
// Lookup thread by ID.
|
|
||||||
thread = kernel_state()->GetThreadByID(thread_info->thread_id);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (thread) {
|
|
||||||
XELOGD("SetThreadName(%d, %s)", thread->thread_id(), name);
|
|
||||||
thread->set_name(name);
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO(benvanik): unwinding required here?
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (record->exception_code == 0xE06D7363) {
|
auto name =
|
||||||
// C++ exception.
|
kernel_memory()->TranslateVirtual<const char*>(thread_info->name_ptr);
|
||||||
// https://blogs.msdn.com/b/oldnewthing/archive/2010/07/30/10044061.aspx
|
|
||||||
xe::debugging::Break();
|
object_ref<XThread> thread;
|
||||||
return;
|
if (thread_info->thread_id == -1) {
|
||||||
|
// Current thread.
|
||||||
|
thread = retain_object(XThread::GetCurrentThread());
|
||||||
|
} else {
|
||||||
|
// Lookup thread by ID.
|
||||||
|
thread = kernel_state()->GetThreadByID(thread_info->thread_id);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (thread) {
|
||||||
|
XELOGD("SetThreadName(%d, %s)", thread->thread_id(), name);
|
||||||
|
thread->set_name(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO(benvanik): unwinding required here?
|
||||||
|
}
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
xe::be<int32_t> mdisp;
|
||||||
|
xe::be<int32_t> pdisp;
|
||||||
|
xe::be<int32_t> vdisp;
|
||||||
|
} x_PMD;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
xe::be<uint32_t> properties;
|
||||||
|
xe::be<uint32_t> type_ptr;
|
||||||
|
x_PMD this_displacement;
|
||||||
|
xe::be<int32_t> size_or_offset;
|
||||||
|
xe::be<uint32_t> copy_function_ptr;
|
||||||
|
} x_s__CatchableType;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
xe::be<int32_t> number_catchable_types;
|
||||||
|
xe::be<uint32_t> catchable_type_ptrs[1];
|
||||||
|
} x_s__CatchableTypeArray;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
xe::be<uint32_t> attributes;
|
||||||
|
xe::be<uint32_t> unwind_ptr;
|
||||||
|
xe::be<uint32_t> forward_compat_ptr;
|
||||||
|
xe::be<uint32_t> catchable_type_array_ptr;
|
||||||
|
} x_s__ThrowInfo;
|
||||||
|
|
||||||
|
void HandleCppException(pointer_t<X_EXCEPTION_RECORD> record) {
|
||||||
|
// C++ exception.
|
||||||
|
// https://blogs.msdn.com/b/oldnewthing/archive/2010/07/30/10044061.aspx
|
||||||
|
// http://www.drdobbs.com/visual-c-exception-handling-instrumentat/184416600
|
||||||
|
// http://www.openrce.org/articles/full_view/21
|
||||||
|
|
||||||
|
assert_true(record->number_parameters == 3);
|
||||||
|
assert_true(record->exception_information[0] == 0x19930520);
|
||||||
|
|
||||||
|
auto thrown_ptr = record->exception_information[1];
|
||||||
|
auto thrown = kernel_memory()->TranslateVirtual(thrown_ptr);
|
||||||
|
auto vftable_ptr = *reinterpret_cast<xe::be<uint32_t>*>(thrown);
|
||||||
|
|
||||||
|
auto throw_info_ptr = record->exception_information[2];
|
||||||
|
auto throw_info =
|
||||||
|
kernel_memory()->TranslateVirtual<x_s__ThrowInfo*>(throw_info_ptr);
|
||||||
|
auto catchable_types =
|
||||||
|
kernel_memory()->TranslateVirtual<x_s__CatchableTypeArray*>(
|
||||||
|
throw_info->catchable_type_array_ptr);
|
||||||
|
|
||||||
|
xe::debugging::Break();
|
||||||
|
}
|
||||||
|
|
||||||
|
void RtlRaiseException(pointer_t<X_EXCEPTION_RECORD> record) {
|
||||||
|
switch (record->exception_code) {
|
||||||
|
case 0x406D1388: {
|
||||||
|
HandleSetThreadName(record);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
case 0xE06D7363: {
|
||||||
|
HandleCppException(record);
|
||||||
|
return;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO(benvanik): unwinding.
|
// TODO(benvanik): unwinding.
|
||||||
|
|
Loading…
Reference in New Issue