From de34bd81dbdd16f99dc16609916a9e15c9ed02e8 Mon Sep 17 00:00:00 2001 From: gibbed Date: Tue, 20 Nov 2018 14:24:13 -0600 Subject: [PATCH 1/9] [Kernel] Set default license mask when available in XamContentCreateEx. --- src/xenia/kernel/xam/xam_content.cc | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/src/xenia/kernel/xam/xam_content.cc b/src/xenia/kernel/xam/xam_content.cc index a1675fea8..6276d551c 100644 --- a/src/xenia/kernel/xam/xam_content.cc +++ b/src/xenia/kernel/xam/xam_content.cc @@ -180,8 +180,6 @@ dword_result_t XamContentCreateEx(dword_t user_index, lpstring_t root_name, lpdword_t license_mask_ptr, dword_t cache_size, qword_t content_size, lpvoid_t overlapped_ptr) { - assert_null(license_mask_ptr); - X_RESULT result = X_ERROR_INVALID_PARAMETER; auto content_data = XCONTENT_DATA((uint8_t*)content_data_ptr); @@ -256,6 +254,10 @@ dword_result_t XamContentCreateEx(dword_t user_index, lpstring_t root_name, result = content_manager->OpenContent(root_name.value(), content_data); } + if (license_mask_ptr && XSUCCEEDED(result)) { + *license_mask_ptr = 0; + } + if (overlapped_ptr) { kernel_state()->CompleteOverlappedImmediateEx(overlapped_ptr, result, 0, disposition); @@ -421,8 +423,7 @@ dword_result_t XamContentDelete(dword_t user_index, lpvoid_t content_data_ptr, DECLARE_XAM_EXPORT1(XamContentDelete, kContent, kImplemented); void RegisterContentExports(xe::cpu::ExportResolver* export_resolver, - KernelState* kernel_state) { -} + KernelState* kernel_state) {} } // namespace xam } // namespace kernel From 7e78a79a2d0be122f5aa66789a74746d0921a18f Mon Sep 17 00:00:00 2001 From: gibbed Date: Tue, 20 Nov 2018 14:25:18 -0600 Subject: [PATCH 2/9] [Kernel] Add a note to previous commit. --- src/xenia/kernel/xam/xam_content.cc | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/xenia/kernel/xam/xam_content.cc b/src/xenia/kernel/xam/xam_content.cc index 6276d551c..01766d693 100644 --- a/src/xenia/kernel/xam/xam_content.cc +++ b/src/xenia/kernel/xam/xam_content.cc @@ -255,7 +255,7 @@ dword_result_t XamContentCreateEx(dword_t user_index, lpstring_t root_name, } if (license_mask_ptr && XSUCCEEDED(result)) { - *license_mask_ptr = 0; + *license_mask_ptr = 0; // Stub! } if (overlapped_ptr) { From 22477424948545a939f96b6582682254bfd28c56 Mon Sep 17 00:00:00 2001 From: gibbed Date: Tue, 20 Nov 2018 14:48:32 -0600 Subject: [PATCH 3/9] [Kernel] Add command-line flag to toggle XEX patching. --- src/xenia/kernel/user_module.cc | 36 ++++++++++++++++++--------------- 1 file changed, 20 insertions(+), 16 deletions(-) diff --git a/src/xenia/kernel/user_module.cc b/src/xenia/kernel/user_module.cc index fb0e48179..a79091d3c 100644 --- a/src/xenia/kernel/user_module.cc +++ b/src/xenia/kernel/user_module.cc @@ -20,6 +20,8 @@ #include "xenia/kernel/xfile.h" #include "xenia/kernel/xthread.h" +DEFINE_bool(xex_apply_patches, true, "Apply XEX patches."); + namespace xe { namespace kernel { @@ -100,27 +102,29 @@ X_STATUS UserModule::LoadFromFile(std::string path) { return result; } - // Search for xexp patch file - auto patch_entry = kernel_state()->file_system()->ResolvePath(path_ + "p"); + if (FLAGS_xex_apply_patches) { + // Search for xexp patch file + auto patch_entry = kernel_state()->file_system()->ResolvePath(path_ + "p"); - if (patch_entry) { - auto patch_path = patch_entry->absolute_path(); + if (patch_entry) { + auto patch_path = patch_entry->absolute_path(); - XELOGI("Loading XEX patch from %s", patch_path.c_str()); + XELOGI("Loading XEX patch from %s", patch_path.c_str()); - auto patch_module = object_ref(new UserModule(kernel_state_)); - result = patch_module->LoadFromFile(patch_path); - if (!result) { - result = patch_module->xex_module()->ApplyPatch(xex_module()); - if (result) { - XELOGE("Failed to apply XEX patch, code: %d", result); + auto patch_module = object_ref(new UserModule(kernel_state_)); + result = patch_module->LoadFromFile(patch_path); + if (!result) { + result = patch_module->xex_module()->ApplyPatch(xex_module()); + if (result) { + XELOGE("Failed to apply XEX patch, code: %d", result); + } + } else { + XELOGE("Failed to load XEX patch, code: %d", result); } - } else { - XELOGE("Failed to load XEX patch, code: %d", result); - } - if (result) { - return X_STATUS_UNSUCCESSFUL; + if (result) { + return X_STATUS_UNSUCCESSFUL; + } } } From d7fc74d788406a6f98d930ef8ceec11cbf38da73 Mon Sep 17 00:00:00 2001 From: gibbed Date: Tue, 20 Nov 2018 14:53:00 -0600 Subject: [PATCH 4/9] [Kernel] Move xam_ui to new shim convention. --- src/xenia/kernel/xam/xam_ui.cc | 68 +++++++++++++--------------------- 1 file changed, 26 insertions(+), 42 deletions(-) diff --git a/src/xenia/kernel/xam/xam_ui.cc b/src/xenia/kernel/xam/xam_ui.cc index debbb03f0..03019f9a4 100644 --- a/src/xenia/kernel/xam/xam_ui.cc +++ b/src/xenia/kernel/xam/xam_ui.cc @@ -23,11 +23,8 @@ namespace xam { std::atomic xam_dialogs_shown_ = {0}; -SHIM_CALL XamIsUIActive_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - XELOGD("XamIsUIActive()"); - SHIM_SET_RETURN_32(xam_dialogs_shown_ > 0 ? 1 : 0); -} +dword_result_t XamIsUIActive() { return xam_dialogs_shown_ > 0 ? 1 : 0; } +DECLARE_XAM_EXPORT2(XamIsUIActive, kUI, kImplemented, kHighFrequency); class MessageBoxDialog : public xe::ui::ImGuiDialog { public: @@ -87,30 +84,26 @@ class MessageBoxDialog : public xe::ui::ImGuiDialog { }; // http://www.se7ensins.com/forums/threads/working-xshowmessageboxui.844116/?jdfwkey=sb0vm -SHIM_CALL XamShowMessageBoxUI_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - uint32_t user_index = SHIM_GET_ARG_32(0); - uint32_t title_ptr = SHIM_GET_ARG_32(1); - uint32_t text_ptr = SHIM_GET_ARG_32(2); - uint32_t button_count = SHIM_GET_ARG_32(3); - uint32_t button_ptrs = SHIM_GET_ARG_32(4); - uint32_t active_button = SHIM_GET_ARG_32(5); - uint32_t flags = SHIM_GET_ARG_32(6); - uint32_t result_ptr = SHIM_GET_ARG_32(7); - uint32_t overlapped_ptr = SHIM_GET_ARG_32(8); - +dword_result_t XamShowMessageBoxUI(dword_t user_index, lpwstring_t title_ptr, + lpwstring_t text_ptr, dword_t button_count, + lpdword_t button_ptrs, dword_t active_button, + dword_t flags, lpdword_t result_ptr, + pointer_t overlapped) { std::wstring title; if (title_ptr) { - title = xe::load_and_swap(SHIM_MEM_ADDR(title_ptr)); + title = title_ptr.value(); } else { title = L""; // TODO(gibbed): default title based on flags? } - auto text = xe::load_and_swap(SHIM_MEM_ADDR(text_ptr)); + auto text = text_ptr.value(); + std::vector buttons; std::wstring all_buttons; for (uint32_t j = 0; j < button_count; ++j) { - uint32_t button_ptr = SHIM_MEM_32(button_ptrs + j * 4); - auto button = xe::load_and_swap(SHIM_MEM_ADDR(button_ptr)); + auto test = button_ptrs[1]; + uint32_t button_ptr = button_ptrs[j]; + auto button = xe::load_and_swap( + kernel_state()->memory()->TranslateVirtual(button_ptr)); all_buttons.append(button); if (j + 1 < button_count) { all_buttons.append(L" | "); @@ -118,19 +111,12 @@ SHIM_CALL XamShowMessageBoxUI_shim(PPCContext* ppc_context, buttons.push_back(button); } - XELOGD( - "XamShowMessageBoxUI(%d, %.8X(%S), %.8X(%S), %d, %.8X(%S), %d, %X, %.8X, " - "%.8X)", - user_index, title_ptr, title.c_str(), text_ptr, text.c_str(), - button_count, button_ptrs, all_buttons.c_str(), active_button, flags, - result_ptr, overlapped_ptr); - uint32_t chosen_button; if (FLAGS_headless) { // Auto-pick the focused button. chosen_button = active_button; } else { - auto display_window = kernel_state->emulator()->display_window(); + auto display_window = kernel_state()->emulator()->display_window(); xe::threading::Fence fence; display_window->loop()->PostSynchronous([&]() { // TODO(benvanik): setup icon states. @@ -156,11 +142,16 @@ SHIM_CALL XamShowMessageBoxUI_shim(PPCContext* ppc_context, fence.Wait(); --xam_dialogs_shown_; } - SHIM_SET_MEM_32(result_ptr, chosen_button); + *result_ptr = chosen_button; - kernel_state->CompleteOverlappedImmediate(overlapped_ptr, X_ERROR_SUCCESS); - SHIM_SET_RETURN_32(X_ERROR_IO_PENDING); + if (overlapped) { + kernel_state()->CompleteOverlappedImmediate(overlapped, X_ERROR_SUCCESS); + return X_ERROR_IO_PENDING; + } else { + return X_ERROR_SUCCESS; + } } +DECLARE_XAM_EXPORT1(XamShowMessageBoxUI, kUI, kImplemented); class KeyboardInputDialog : public xe::ui::ImGuiDialog { public: @@ -318,19 +309,14 @@ dword_result_t XamShowDeviceSelectorUI(dword_t user_index, dword_t content_type, } DECLARE_XAM_EXPORT1(XamShowDeviceSelectorUI, kUI, kImplemented); -SHIM_CALL XamShowDirtyDiscErrorUI_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - uint32_t user_index = SHIM_GET_ARG_32(0); - - XELOGD("XamShowDirtyDiscErrorUI(%d)", user_index); - +void XamShowDirtyDiscErrorUI(dword_t user_index) { if (FLAGS_headless) { assert_always(); exit(1); return; } - auto display_window = kernel_state->emulator()->display_window(); + auto display_window = kernel_state()->emulator()->display_window(); xe::threading::Fence fence; display_window->loop()->PostSynchronous([&]() { xe::ui::ImGuiDialog::ShowMessageBox( @@ -347,12 +333,10 @@ SHIM_CALL XamShowDirtyDiscErrorUI_shim(PPCContext* ppc_context, // TODO(benvanik): cleaner exit. exit(1); } +DECLARE_XAM_EXPORT1(XamShowDirtyDiscErrorUI, kUI, kImplemented); void RegisterUIExports(xe::cpu::ExportResolver* export_resolver, KernelState* kernel_state) { - SHIM_SET_MAPPING("xam.xex", XamIsUIActive, state); - SHIM_SET_MAPPING("xam.xex", XamShowMessageBoxUI, state); - SHIM_SET_MAPPING("xam.xex", XamShowDirtyDiscErrorUI, state); } } // namespace xam From 5b776b99da2e79c24f8cff5fd0252fbeeb966cd1 Mon Sep 17 00:00:00 2001 From: gibbed Date: Tue, 20 Nov 2018 15:49:06 -0600 Subject: [PATCH 5/9] [Kernel] Move xboxkrnl_threading to new shim convention. --- .../kernel/xboxkrnl/xboxkrnl_threading.cc | 601 ++++++------------ 1 file changed, 209 insertions(+), 392 deletions(-) diff --git a/src/xenia/kernel/xboxkrnl/xboxkrnl_threading.cc b/src/xenia/kernel/xboxkrnl/xboxkrnl_threading.cc index 5a5caddd7..4f9a883e0 100644 --- a/src/xenia/kernel/xboxkrnl/xboxkrnl_threading.cc +++ b/src/xenia/kernel/xboxkrnl/xboxkrnl_threading.cc @@ -93,20 +93,11 @@ object_ref LookupNamedObject(KernelState* kernel_state, return nullptr; } -SHIM_CALL ExCreateThread_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - uint32_t handle_ptr = SHIM_GET_ARG_32(0); - uint32_t stack_size = SHIM_GET_ARG_32(1); - uint32_t thread_id_ptr = SHIM_GET_ARG_32(2); - uint32_t xapi_thread_startup = SHIM_GET_ARG_32(3); - uint32_t start_address = SHIM_GET_ARG_32(4); - uint32_t start_context = SHIM_GET_ARG_32(5); - uint32_t creation_flags = SHIM_GET_ARG_32(6); - - XELOGD("ExCreateThread(%.8X, %d, %.8X, %.8X, %.8X, %.8X, %.8X)", handle_ptr, - stack_size, thread_id_ptr, xapi_thread_startup, start_address, - start_context, creation_flags); - +dword_result_t ExCreateThread(lpdword_t handle_ptr, dword_t stack_size, + lpdword_t thread_id_ptr, + dword_t xapi_thread_startup, + lpvoid_t start_address, lpvoid_t start_context, + dword_t creation_flags) { // http://jafile.com/uploads/scoop/main.cpp.txt // DWORD // LPHANDLE Handle, @@ -118,103 +109,86 @@ SHIM_CALL ExCreateThread_shim(PPCContext* ppc_context, // DWORD CreationFlags // 0x80? // Inherit default stack size - if (stack_size == 0) { - stack_size = kernel_state->GetExecutableModule()->stack_size(); + uint32_t actual_stack_size = stack_size; + + if (actual_stack_size == 0) { + actual_stack_size = kernel_state()->GetExecutableModule()->stack_size(); } // Stack must be aligned to 16kb pages - stack_size = std::max((uint32_t)0x4000, ((stack_size + 0xFFF) & 0xFFFFF000)); + actual_stack_size = + std::max((uint32_t)0x4000, ((actual_stack_size + 0xFFF) & 0xFFFFF000)); auto thread = object_ref( - new XThread(kernel_state, stack_size, xapi_thread_startup, start_address, - start_context, creation_flags, true)); + new XThread(kernel_state(), actual_stack_size, xapi_thread_startup, + start_address.guest_address(), start_context.guest_address(), + creation_flags, true)); X_STATUS result = thread->Create(); if (XFAILED(result)) { // Failed! XELOGE("Thread creation failed: %.8X", result); - SHIM_SET_RETURN_32(result); - return; + return result; } if (XSUCCEEDED(result)) { if (handle_ptr) { if (creation_flags & 0x80) { - SHIM_SET_MEM_32(handle_ptr, thread->guest_object()); + *handle_ptr = thread->guest_object(); } else { - SHIM_SET_MEM_32(handle_ptr, thread->handle()); + *handle_ptr = thread->handle(); } } if (thread_id_ptr) { - SHIM_SET_MEM_32(thread_id_ptr, thread->thread_id()); + *thread_id_ptr = thread->thread_id(); } } - SHIM_SET_RETURN_32(result); + return result; } +DECLARE_XBOXKRNL_EXPORT1(ExCreateThread, kThreading, kImplemented); -SHIM_CALL ExTerminateThread_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - uint32_t exit_code = SHIM_GET_ARG_32(0); - - XELOGD("ExTerminateThread(%d)", exit_code); - +dword_result_t ExTerminateThread(dword_t exit_code) { XThread* thread = XThread::GetCurrentThread(); // NOTE: this kills us right now. We won't return from it. - X_STATUS result = thread->Exit(exit_code); - SHIM_SET_RETURN_32(result); + return thread->Exit(exit_code); } +DECLARE_XBOXKRNL_EXPORT1(ExTerminateThread, kThreading, kImplemented); -SHIM_CALL NtResumeThread_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - uint32_t handle = SHIM_GET_ARG_32(0); - uint32_t suspend_count_ptr = SHIM_GET_ARG_32(1); - - XELOGD("NtResumeThread(%.8X, %.8X)", handle, suspend_count_ptr); - +dword_result_t NtResumeThread(dword_t handle, lpdword_t suspend_count_ptr) { X_RESULT result = X_STATUS_INVALID_HANDLE; uint32_t suspend_count = 0; - auto thread = kernel_state->object_table()->LookupObject(handle); + auto thread = kernel_state()->object_table()->LookupObject(handle); if (thread) { result = thread->Resume(&suspend_count); } if (suspend_count_ptr) { - SHIM_SET_MEM_32(suspend_count_ptr, suspend_count); + *suspend_count_ptr = suspend_count; } - SHIM_SET_RETURN_32(result); + return result; } +DECLARE_XBOXKRNL_EXPORT1(NtResumeThread, kThreading, kImplemented); -SHIM_CALL KeResumeThread_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - uint32_t thread_ptr = SHIM_GET_ARG_32(0); - - XELOGD("KeResumeThread(%.8X)", thread_ptr); - +dword_result_t KeResumeThread(lpvoid_t thread_ptr) { X_STATUS result = X_STATUS_SUCCESS; - auto thread = XObject::GetNativeObject(kernel_state, - SHIM_MEM_ADDR(thread_ptr)); + auto thread = XObject::GetNativeObject(kernel_state(), thread_ptr); if (thread) { result = thread->Resume(); } else { result = X_STATUS_INVALID_HANDLE; } - SHIM_SET_RETURN_32(result); + return result; } +DECLARE_XBOXKRNL_EXPORT1(KeResumeThread, kThreading, kImplemented); -SHIM_CALL NtSuspendThread_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - uint32_t handle = SHIM_GET_ARG_32(0); - uint32_t suspend_count_ptr = SHIM_GET_ARG_32(1); - - XELOGD("NtSuspendThread(%.8X, %.8X)", handle, suspend_count_ptr); - +dword_result_t NtSuspendThread(dword_t handle, lpdword_t suspend_count_ptr) { X_RESULT result = X_STATUS_SUCCESS; uint32_t suspend_count = 0; - auto thread = kernel_state->object_table()->LookupObject(handle); + auto thread = kernel_state()->object_table()->LookupObject(handle); if (thread) { result = thread->Suspend(&suspend_count); } else { @@ -222,11 +196,12 @@ SHIM_CALL NtSuspendThread_shim(PPCContext* ppc_context, } if (suspend_count_ptr) { - SHIM_SET_MEM_32(suspend_count_ptr, suspend_count); + *suspend_count_ptr = suspend_count; } - SHIM_SET_RETURN_32(result); + return result; } +DECLARE_XBOXKRNL_EXPORT1(NtSuspendThread, kThreading, kImplemented); void KeSetCurrentStackPointers(lpvoid_t stack_ptr, pointer_t cur_thread, @@ -245,104 +220,72 @@ void KeSetCurrentStackPointers(lpvoid_t stack_ptr, } DECLARE_XBOXKRNL_EXPORT1(KeSetCurrentStackPointers, kThreading, kImplemented); -SHIM_CALL KeSetAffinityThread_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - uint32_t thread_ptr = SHIM_GET_ARG_32(0); - uint32_t affinity = SHIM_GET_ARG_32(1); - - XELOGD("KeSetAffinityThread(%.8X, %.8X)", thread_ptr, affinity); - - auto thread = XObject::GetNativeObject(kernel_state, - SHIM_MEM_ADDR(thread_ptr)); +dword_result_t KeSetAffinityThread(lpvoid_t thread_ptr, dword_t affinity) { + auto thread = XObject::GetNativeObject(kernel_state(), thread_ptr); if (thread) { thread->SetAffinity(affinity); } - SHIM_SET_RETURN_32(affinity); + return (uint32_t)affinity; } +DECLARE_XBOXKRNL_EXPORT1(KeSetAffinityThread, kThreading, kImplemented); -SHIM_CALL KeQueryBasePriorityThread_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - uint32_t thread_ptr = SHIM_GET_ARG_32(0); - - XELOGD("KeQueryBasePriorityThread(%.8X)", thread_ptr); - +dword_result_t KeQueryBasePriorityThread(lpvoid_t thread_ptr) { int32_t priority = 0; - auto thread = XObject::GetNativeObject(kernel_state, - SHIM_MEM_ADDR(thread_ptr)); + auto thread = XObject::GetNativeObject(kernel_state(), thread_ptr); if (thread) { priority = thread->QueryPriority(); } - SHIM_SET_RETURN_32(priority); + return priority; } +DECLARE_XBOXKRNL_EXPORT1(KeQueryBasePriorityThread, kThreading, kImplemented); -SHIM_CALL KeSetBasePriorityThread_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - uint32_t thread_ptr = SHIM_GET_ARG_32(0); - uint32_t increment = SHIM_GET_ARG_32(1); - - XELOGD("KeSetBasePriorityThread(%.8X, %.8X)", thread_ptr, increment); - +dword_result_t KeSetBasePriorityThread(lpvoid_t thread_ptr, dword_t increment) { int32_t prev_priority = 0; - auto thread = XObject::GetNativeObject(kernel_state, - SHIM_MEM_ADDR(thread_ptr)); + auto thread = XObject::GetNativeObject(kernel_state(), thread_ptr); if (thread) { prev_priority = thread->QueryPriority(); thread->SetPriority(increment); } - SHIM_SET_RETURN_32(prev_priority); + return prev_priority; } +DECLARE_XBOXKRNL_EXPORT1(KeSetBasePriorityThread, kThreading, kImplemented); -SHIM_CALL KeSetDisableBoostThread_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - uint32_t thread_ptr = SHIM_GET_ARG_32(0); - uint32_t disabled = SHIM_GET_ARG_32(1); - - XELOGD("KeSetDisableBoostThread(%.8X, %.8X)", thread_ptr, disabled); - - auto thread = XObject::GetNativeObject(kernel_state, - SHIM_MEM_ADDR(thread_ptr)); +dword_result_t KeSetDisableBoostThread(lpvoid_t thread_ptr, dword_t disabled) { + auto thread = XObject::GetNativeObject(kernel_state(), thread_ptr); if (thread) { // Uhm? } - SHIM_SET_RETURN_32(0); + return 0; } +DECLARE_XBOXKRNL_EXPORT1(KeSetDisableBoostThread, kThreading, kImplemented); -SHIM_CALL KeGetCurrentProcessType_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - // XELOGD( - // "KeGetCurrentProcessType()"); - - // DWORD - - SHIM_SET_RETURN_32(kernel_state->process_type()); +dword_result_t KeGetCurrentProcessType() { + return kernel_state()->process_type(); } +DECLARE_XBOXKRNL_EXPORT2(KeGetCurrentProcessType, kThreading, kImplemented, + kHighFrequency); -SHIM_CALL KeSetCurrentProcessType_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - uint32_t type = SHIM_GET_ARG_32(0); +void KeSetCurrentProcessType(dword_t type) { // One of X_PROCTYPE_? - XELOGD("KeSetCurrentProcessType(%d)", type); - assert_true(type <= 2); - kernel_state->set_process_type(type); + kernel_state()->set_process_type(type); } +DECLARE_XBOXKRNL_EXPORT1(KeSetCurrentProcessType, kThreading, kImplemented); -SHIM_CALL KeQueryPerformanceFrequency_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - // XELOGD( - // "KeQueryPerformanceFrequency()"); - +dword_result_t KeQueryPerformanceFrequency() { uint64_t result = Clock::guest_tick_frequency(); - SHIM_SET_RETURN_32(result); + return static_cast(result); } +DECLARE_XBOXKRNL_EXPORT2(KeQueryPerformanceFrequency, kThreading, kImplemented, + kHighFrequency); dword_result_t KeDelayExecutionThread(dword_t processor_mode, dword_t alertable, lpqword_t interval_ptr) { @@ -354,25 +297,21 @@ dword_result_t KeDelayExecutionThread(dword_t processor_mode, dword_t alertable, DECLARE_XBOXKRNL_EXPORT3(KeDelayExecutionThread, kThreading, kImplemented, kBlocking, kHighFrequency); -SHIM_CALL NtYieldExecution_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - // XELOGD("NtYieldExecution()"); +dword_result_t NtYieldExecution() { auto thread = XThread::GetCurrentThread(); thread->Delay(0, 0, 0); - SHIM_SET_RETURN_32(0); + return 0; } +DECLARE_XBOXKRNL_EXPORT2(NtYieldExecution, kThreading, kImplemented, + kHighFrequency); -SHIM_CALL KeQuerySystemTime_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - uint32_t time_ptr = SHIM_GET_ARG_32(0); - - XELOGD("KeQuerySystemTime(%.8X)", time_ptr); - +void KeQuerySystemTime(lpqword_t time_ptr) { uint64_t time = Clock::QueryGuestSystemTime(); if (time_ptr) { - SHIM_SET_MEM_64(time_ptr, time); + *time_ptr = time; } } +DECLARE_XBOXKRNL_EXPORT1(KeQuerySystemTime, kThreading, kImplemented); // http://msdn.microsoft.com/en-us/library/ms686801 dword_result_t KeTlsAlloc() { @@ -586,34 +525,26 @@ dword_result_t KeReleaseSemaphore(pointer_t semaphore_ptr, } DECLARE_XBOXKRNL_EXPORT1(KeReleaseSemaphore, kThreading, kImplemented); -SHIM_CALL NtCreateSemaphore_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - uint32_t handle_ptr = SHIM_GET_ARG_32(0); - uint32_t obj_attributes_ptr = SHIM_GET_ARG_32(1); - int32_t count = SHIM_GET_ARG_32(2); - int32_t limit = SHIM_GET_ARG_32(3); - - XELOGD("NtCreateSemaphore(%.8X, %.8X, %d, %d)", handle_ptr, - obj_attributes_ptr, count, limit); - +dword_result_t NtCreateSemaphore(lpdword_t handle_ptr, + lpvoid_t obj_attributes_ptr, dword_t count, + dword_t limit) { // Check for an existing timer with the same name. auto existing_object = - LookupNamedObject(kernel_state, obj_attributes_ptr); + LookupNamedObject(kernel_state(), obj_attributes_ptr); if (existing_object) { if (existing_object->type() == XObject::kTypeSemaphore) { if (handle_ptr) { existing_object->RetainHandle(); - SHIM_SET_MEM_32(handle_ptr, existing_object->handle()); + *handle_ptr = existing_object->handle(); } - SHIM_SET_RETURN_32(X_STATUS_SUCCESS); + return X_STATUS_SUCCESS; } else { - SHIM_SET_RETURN_32(X_STATUS_INVALID_HANDLE); + return X_STATUS_INVALID_HANDLE; } - return; } - auto sem = object_ref(new XSemaphore(kernel_state)); - sem->Initialize(count, limit); + auto sem = object_ref(new XSemaphore(kernel_state())); + sem->Initialize((int32_t)count, (int32_t)limit); // obj_attributes may have a name inside of it, if != NULL. if (obj_attributes_ptr) { @@ -621,36 +552,32 @@ SHIM_CALL NtCreateSemaphore_shim(PPCContext* ppc_context, } if (handle_ptr) { - SHIM_SET_MEM_32(handle_ptr, sem->handle()); + *handle_ptr = sem->handle(); } - SHIM_SET_RETURN_32(X_STATUS_SUCCESS); + return X_STATUS_SUCCESS; } +DECLARE_XBOXKRNL_EXPORT1(NtCreateSemaphore, kThreading, kImplemented); -SHIM_CALL NtReleaseSemaphore_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - uint32_t sem_handle = SHIM_GET_ARG_32(0); - int32_t release_count = SHIM_GET_ARG_32(1); - int32_t previous_count_ptr = SHIM_GET_ARG_32(2); - - XELOGD("NtReleaseSemaphore(%.8X, %d, %.8X)", sem_handle, release_count, - previous_count_ptr); - +dword_result_t NtReleaseSemaphore(dword_t sem_handle, dword_t release_count, + lpdword_t previous_count_ptr) { X_STATUS result = X_STATUS_SUCCESS; int32_t previous_count = 0; - auto sem = kernel_state->object_table()->LookupObject(sem_handle); + auto sem = + kernel_state()->object_table()->LookupObject(sem_handle); if (sem) { - previous_count = sem->ReleaseSemaphore(release_count); + previous_count = sem->ReleaseSemaphore((int32_t)release_count); } else { result = X_STATUS_INVALID_HANDLE; } if (previous_count_ptr) { - SHIM_SET_MEM_32(previous_count_ptr, previous_count); + *previous_count_ptr = (uint32_t)previous_count; } - SHIM_SET_RETURN_32(result); + return result; } +DECLARE_XBOXKRNL_EXPORT1(NtReleaseSemaphore, kThreading, kImplemented); dword_result_t NtCreateMutant(lpdword_t handle_out, pointer_t obj_attributes, @@ -686,10 +613,7 @@ dword_result_t NtCreateMutant(lpdword_t handle_out, } DECLARE_XBOXKRNL_EXPORT1(NtCreateMutant, kThreading, kImplemented); -SHIM_CALL NtReleaseMutant_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - uint32_t mutant_handle = SHIM_GET_ARG_32(0); - int32_t unknown = SHIM_GET_ARG_32(1); +dword_result_t NtReleaseMutant(dword_t mutant_handle, dword_t unknown) { // This doesn't seem to be supported. // int32_t previous_count_ptr = SHIM_GET_ARG_32(2); @@ -706,44 +630,37 @@ SHIM_CALL NtReleaseMutant_shim(PPCContext* ppc_context, X_STATUS result = X_STATUS_SUCCESS; auto mutant = - kernel_state->object_table()->LookupObject(mutant_handle); + kernel_state()->object_table()->LookupObject(mutant_handle); if (mutant) { result = mutant->ReleaseMutant(priority_increment, abandon, wait); } else { result = X_STATUS_INVALID_HANDLE; } - SHIM_SET_RETURN_32(result); + return result; } +DECLARE_XBOXKRNL_EXPORT1(NtReleaseMutant, kThreading, kImplemented); -SHIM_CALL NtCreateTimer_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - uint32_t handle_ptr = SHIM_GET_ARG_32(0); - uint32_t obj_attributes_ptr = SHIM_GET_ARG_32(1); - uint32_t timer_type = SHIM_GET_ARG_32(2); - +dword_result_t NtCreateTimer(lpdword_t handle_ptr, lpvoid_t obj_attributes_ptr, + dword_t timer_type) { // timer_type = NotificationTimer (0) or SynchronizationTimer (1) - XELOGD("NtCreateTimer(%.8X, %.8X, %.1X)", handle_ptr, obj_attributes_ptr, - timer_type); - // Check for an existing timer with the same name. auto existing_object = - LookupNamedObject(kernel_state, obj_attributes_ptr); + LookupNamedObject(kernel_state(), obj_attributes_ptr); if (existing_object) { if (existing_object->type() == XObject::kTypeTimer) { if (handle_ptr) { existing_object->RetainHandle(); - SHIM_SET_MEM_32(handle_ptr, existing_object->handle()); + *handle_ptr = existing_object->handle(); } - SHIM_SET_RETURN_32(X_STATUS_SUCCESS); + return X_STATUS_SUCCESS; } else { - SHIM_SET_RETURN_32(X_STATUS_INVALID_HANDLE); + return X_STATUS_INVALID_HANDLE; } - return; } - auto timer = object_ref(new XTimer(kernel_state)); + auto timer = object_ref(new XTimer(kernel_state())); timer->Initialize(timer_type); // obj_attributes may have a name inside of it, if != NULL. @@ -752,66 +669,57 @@ SHIM_CALL NtCreateTimer_shim(PPCContext* ppc_context, } if (handle_ptr) { - SHIM_SET_MEM_32(handle_ptr, timer->handle()); + *handle_ptr = timer->handle(); } - SHIM_SET_RETURN_32(X_STATUS_SUCCESS); + return X_STATUS_SUCCESS; } +DECLARE_XBOXKRNL_EXPORT1(NtCreateTimer, kThreading, kImplemented); -SHIM_CALL NtSetTimerEx_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - uint32_t timer_handle = SHIM_GET_ARG_32(0); - uint32_t due_time_ptr = SHIM_GET_ARG_32(1); - uint32_t routine = SHIM_GET_ARG_32(2); // PTIMERAPCROUTINE - uint32_t unk_one = SHIM_GET_ARG_32(3); - uint32_t routine_arg = SHIM_GET_ARG_32(4); - uint32_t resume = SHIM_GET_ARG_32(5); - uint32_t period_ms = SHIM_GET_ARG_32(6); - uint32_t unk_zero = SHIM_GET_ARG_32(7); - +dword_result_t NtSetTimerEx(dword_t timer_handle, lpqword_t due_time_ptr, + lpvoid_t routine_ptr /*PTIMERAPCROUTINE*/, + dword_t unk_one, lpvoid_t routine_arg, + dword_t resume, dword_t period_ms, + dword_t unk_zero) { assert_true(unk_one == 1); assert_true(unk_zero == 0); - uint64_t due_time = SHIM_MEM_64(due_time_ptr); - - XELOGD("NtSetTimerEx(%.8X, %.8X(%lld), %.8X, %.8X, %.8X, %.1X, %d, %.8X)", - timer_handle, due_time_ptr, due_time, routine, unk_one, routine_arg, - resume, period_ms, unk_zero); + uint64_t due_time = *due_time_ptr; X_STATUS result = X_STATUS_SUCCESS; - auto timer = kernel_state->object_table()->LookupObject(timer_handle); + auto timer = + kernel_state()->object_table()->LookupObject(timer_handle); if (timer) { - result = timer->SetTimer(due_time, period_ms, routine, routine_arg, - resume ? true : false); + result = + timer->SetTimer(due_time, period_ms, routine_ptr.guest_address(), + routine_arg.guest_address(), resume ? true : false); } else { result = X_STATUS_INVALID_HANDLE; } - SHIM_SET_RETURN_32(result); + return result; } +DECLARE_XBOXKRNL_EXPORT1(NtSetTimerEx, kThreading, kImplemented); -SHIM_CALL NtCancelTimer_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - uint32_t timer_handle = SHIM_GET_ARG_32(0); - uint32_t current_state_ptr = SHIM_GET_ARG_32(1); - - XELOGD("NtCancelTimer(%.8X, %.8X)", timer_handle, current_state_ptr); - +dword_result_t NtCancelTimer(dword_t timer_handle, + lpdword_t current_state_ptr) { X_STATUS result = X_STATUS_SUCCESS; - auto timer = kernel_state->object_table()->LookupObject(timer_handle); + auto timer = + kernel_state()->object_table()->LookupObject(timer_handle); if (timer) { result = timer->Cancel(); } else { result = X_STATUS_INVALID_HANDLE; } if (current_state_ptr) { - SHIM_SET_MEM_32(current_state_ptr, 0); + *current_state_ptr = 0; } - SHIM_SET_RETURN_32(result); + return result; } +DECLARE_XBOXKRNL_EXPORT1(NtCancelTimer, kThreading, kImplemented); dword_result_t KeWaitForSingleObject(lpvoid_t object_ptr, dword_t wait_reason, dword_t processor_mode, dword_t alertable, @@ -991,96 +899,63 @@ void KeReleaseSpinLockFromRaisedIrql(lpdword_t lock_ptr) { DECLARE_XBOXKRNL_EXPORT2(KeReleaseSpinLockFromRaisedIrql, kThreading, kImplemented, kHighFrequency); -SHIM_CALL KeEnterCriticalRegion_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - // XELOGD( - // "KeEnterCriticalRegion()"); - XThread::EnterCriticalRegion(); -} +void KeEnterCriticalRegion() { XThread::EnterCriticalRegion(); } +DECLARE_XBOXKRNL_EXPORT2(KeEnterCriticalRegion, kThreading, kImplemented, + kHighFrequency); -SHIM_CALL KeLeaveCriticalRegion_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - // XELOGD( - // "KeLeaveCriticalRegion()"); +void KeLeaveCriticalRegion() { XThread::LeaveCriticalRegion(); XThread::GetCurrentThread()->CheckApcs(); } +DECLARE_XBOXKRNL_EXPORT2(KeLeaveCriticalRegion, kThreading, kImplemented, + kHighFrequency); -SHIM_CALL KeRaiseIrqlToDpcLevel_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - // XELOGD( - // "KeRaiseIrqlToDpcLevel()"); - auto old_value = kernel_state->processor()->RaiseIrql(cpu::Irql::DPC); - SHIM_SET_RETURN_32(old_value); +dword_result_t KeRaiseIrqlToDpcLevel() { + auto old_value = kernel_state()->processor()->RaiseIrql(cpu::Irql::DPC); + return (uint32_t)old_value; } +DECLARE_XBOXKRNL_EXPORT2(KeRaiseIrqlToDpcLevel, kThreading, kImplemented, + kHighFrequency); -SHIM_CALL KfLowerIrql_shim(PPCContext* ppc_context, KernelState* kernel_state) { - uint32_t old_value = SHIM_GET_ARG_32(0); - // XELOGD( - // "KfLowerIrql(%d)", - // old_value); - kernel_state->processor()->LowerIrql(static_cast(old_value)); +void KfLowerIrql(dword_t old_value) { + kernel_state()->processor()->LowerIrql( + static_cast((uint32_t)old_value)); XThread::GetCurrentThread()->CheckApcs(); } +DECLARE_XBOXKRNL_EXPORT2(KfLowerIrql, kThreading, kImplemented, kHighFrequency); -SHIM_CALL NtQueueApcThread_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - uint32_t thread_handle = SHIM_GET_ARG_32(0); - uint32_t apc_routine = SHIM_GET_ARG_32(1); - uint32_t arg1 = SHIM_GET_ARG_32(2); - uint32_t arg2 = SHIM_GET_ARG_32(3); - uint32_t arg3 = SHIM_GET_ARG_32(4); // ? - XELOGD("NtQueueApcThread(%.8X, %.8X, %.8X, %.8X, %.8X)", thread_handle, - apc_routine, arg1, arg2, arg3); - +void NtQueueApcThread(dword_t thread_handle, lpvoid_t apc_routine, + lpvoid_t arg1, lpvoid_t arg2, lpvoid_t arg3) { // Alloc APC object (from somewhere) and insert. assert_always("not implemented"); } +// DECLARE_XBOXKRNL_EXPORT1(NtQueueApcThread, kThreading, kStub); -SHIM_CALL KeInitializeApc_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - uint32_t apc_ptr = SHIM_GET_ARG_32(0); - uint32_t thread = SHIM_GET_ARG_32(1); - uint32_t kernel_routine = SHIM_GET_ARG_32(2); - uint32_t rundown_routine = SHIM_GET_ARG_32(3); - uint32_t normal_routine = SHIM_GET_ARG_32(4); - uint32_t processor_mode = SHIM_GET_ARG_32(5); - uint32_t normal_context = SHIM_GET_ARG_32(6); - - XELOGD("KeInitializeApc(%.8X, %.8X, %.8X, %.8X, %.8X, %.8X, %.8X)", apc_ptr, - thread, kernel_routine, rundown_routine, normal_routine, - processor_mode, normal_context); - - auto apc = SHIM_STRUCT(XAPC, apc_ptr); +void KeInitializeApc(pointer_t apc, lpvoid_t thread_ptr, + lpvoid_t kernel_routine, lpvoid_t rundown_routine, + lpvoid_t normal_routine, dword_t processor_mode, + lpvoid_t normal_context) { apc->Initialize(); apc->processor_mode = processor_mode; - apc->thread_ptr = thread; - apc->kernel_routine = kernel_routine; - apc->rundown_routine = rundown_routine; - apc->normal_routine = normal_routine; - apc->normal_context = normal_routine ? normal_context : 0; + apc->thread_ptr = thread_ptr.guest_address(); + apc->kernel_routine = kernel_routine.guest_address(); + apc->rundown_routine = rundown_routine.guest_address(); + apc->normal_routine = normal_routine.guest_address(); + apc->normal_context = + normal_routine.guest_address() ? normal_context.guest_address() : 0; } +DECLARE_XBOXKRNL_EXPORT1(KeInitializeApc, kThreading, kImplemented); -SHIM_CALL KeInsertQueueApc_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - uint32_t apc_ptr = SHIM_GET_ARG_32(0); - uint32_t arg1 = SHIM_GET_ARG_32(1); - uint32_t arg2 = SHIM_GET_ARG_32(2); - uint32_t priority_increment = SHIM_GET_ARG_32(3); - - XELOGD("KeInsertQueueApc(%.8X, %.8X, %.8X, %.8X)", apc_ptr, arg1, arg2, - priority_increment); - - auto apc = SHIM_STRUCT(XAPC, apc_ptr); - +dword_result_t KeInsertQueueApc(pointer_t apc, lpvoid_t arg1, + lpvoid_t arg2, dword_t priority_increment) { auto thread = XObject::GetNativeObject( - kernel_state, SHIM_MEM_ADDR(apc->thread_ptr)); + kernel_state(), + kernel_state()->memory()->TranslateVirtual(apc->thread_ptr)); if (!thread) { - SHIM_SET_RETURN_32(0); - return; + return 0; } // Lock thread. @@ -1089,53 +964,45 @@ SHIM_CALL KeInsertQueueApc_shim(PPCContext* ppc_context, // Fail if already inserted. if (apc->enqueued) { thread->UnlockApc(false); - SHIM_SET_RETURN_32(0); - return; + return 0; } // Prep APC. - apc->arg1 = arg1; - apc->arg2 = arg2; + apc->arg1 = arg1.guest_address(); + apc->arg2 = arg2.guest_address(); apc->enqueued = 1; auto apc_list = thread->apc_list(); - uint32_t list_entry_ptr = apc_ptr + 8; + uint32_t list_entry_ptr = apc.guest_address() + 8; apc_list->Insert(list_entry_ptr); // Unlock thread. thread->UnlockApc(true); - SHIM_SET_RETURN_32(1); + return 1; } +DECLARE_XBOXKRNL_EXPORT1(KeInsertQueueApc, kThreading, kImplemented); -SHIM_CALL KeRemoveQueueApc_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - uint32_t apc_ptr = SHIM_GET_ARG_32(0); - - XELOGD("KeRemoveQueueApc(%.8X)", apc_ptr); - +dword_result_t KeRemoveQueueApc(pointer_t apc) { bool result = false; - auto apc = SHIM_STRUCT(XAPC, apc_ptr); - auto thread = XObject::GetNativeObject( - kernel_state, SHIM_MEM_ADDR(apc->thread_ptr)); + kernel_state(), + kernel_state()->memory()->TranslateVirtual(apc->thread_ptr)); if (!thread) { - SHIM_SET_RETURN_32(0); - return; + return 0; } thread->LockApc(); if (!apc->enqueued) { thread->UnlockApc(false); - SHIM_SET_RETURN_32(0); - return; + return 0; } auto apc_list = thread->apc_list(); - uint32_t list_entry_ptr = apc_ptr + 8; + uint32_t list_entry_ptr = apc.guest_address() + 8; if (apc_list->IsQueued(list_entry_ptr)) { apc_list->Remove(list_entry_ptr); result = true; @@ -1143,90 +1010,81 @@ SHIM_CALL KeRemoveQueueApc_shim(PPCContext* ppc_context, thread->UnlockApc(true); - SHIM_SET_RETURN_32(result ? 1 : 0); + return result ? 1 : 0; } +DECLARE_XBOXKRNL_EXPORT1(KeRemoveQueueApc, kThreading, kImplemented); -SHIM_CALL KiApcNormalRoutineNop_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - uint32_t unk0 = SHIM_GET_ARG_32(0); // output? - uint32_t unk1 = SHIM_GET_ARG_32(1); // 0x13 - - XELOGD("KiApcNormalRoutineNop(%.8X, %.8X)", unk0, unk1); - - SHIM_SET_RETURN_32(0); +dword_result_t KiApcNormalRoutineNop(dword_t unk0 /* output? */, + dword_t unk1 /* 0x13 */) { + return 0; } +DECLARE_XBOXKRNL_EXPORT1(KiApcNormalRoutineNop, kThreading, kStub); -SHIM_CALL KeInitializeDpc_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - uint32_t dpc_ptr = SHIM_GET_ARG_32(0); - uint32_t routine = SHIM_GET_ARG_32(1); - uint32_t context = SHIM_GET_ARG_32(2); - - XELOGD("KeInitializeDpc(%.8X, %.8X, %.8X)", dpc_ptr, routine, context); +typedef struct { + xe::be unknown; + xe::be flink; + xe::be blink; + xe::be routine; + xe::be context; + xe::be arg1; + xe::be arg2; +} XDPC; +void KeInitializeDpc(pointer_t dpc, lpvoid_t routine, lpvoid_t context) { // KDPC (maybe) 0x18 bytes? uint32_t type = 19; // DpcObject uint32_t importance = 0; uint32_t number = 0; // ? - SHIM_SET_MEM_32(dpc_ptr + 0, (type << 24) | (importance << 16) | (number)); - SHIM_SET_MEM_32(dpc_ptr + 4, 0); // flink - SHIM_SET_MEM_32(dpc_ptr + 8, 0); // blink - SHIM_SET_MEM_32(dpc_ptr + 12, routine); - SHIM_SET_MEM_32(dpc_ptr + 16, context); - SHIM_SET_MEM_32(dpc_ptr + 20, 0); // arg1 - SHIM_SET_MEM_32(dpc_ptr + 24, 0); // arg2 + dpc->unknown = (type << 24) | (importance << 16) | (number); + dpc->flink = 0; + dpc->blink = 0; + dpc->routine = routine.guest_address(); + dpc->context = context.guest_address(); + dpc->arg1 = 0; + dpc->arg2 = 0; } +DECLARE_XBOXKRNL_EXPORT2(KeInitializeDpc, kThreading, kImplemented, kSketchy); -SHIM_CALL KeInsertQueueDpc_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - uint32_t dpc_ptr = SHIM_GET_ARG_32(0); - uint32_t arg1 = SHIM_GET_ARG_32(1); - uint32_t arg2 = SHIM_GET_ARG_32(2); - +dword_result_t KeInsertQueueDpc(pointer_t dpc, dword_t arg1, + dword_t arg2) { assert_always("DPC does not dispatch yet; going to hang!"); - XELOGD("KeInsertQueueDpc(%.8X, %.8X, %.8X)", dpc_ptr, arg1, arg2); - - uint32_t list_entry_ptr = dpc_ptr + 4; + uint32_t list_entry_ptr = dpc.guest_address() + 4; // Lock dispatcher. auto global_lock = xe::global_critical_region::AcquireDirect(); - auto dpc_list = kernel_state->dpc_list(); + auto dpc_list = kernel_state()->dpc_list(); // If already in a queue, abort. if (dpc_list->IsQueued(list_entry_ptr)) { - SHIM_SET_RETURN_32(0); - return; + return 0; } // Prep DPC. - SHIM_SET_MEM_32(dpc_ptr + 20, arg1); - SHIM_SET_MEM_32(dpc_ptr + 24, arg2); + dpc->arg1 = (uint32_t)arg1; + dpc->arg2 = (uint32_t)arg2; dpc_list->Insert(list_entry_ptr); - SHIM_SET_RETURN_32(1); + return 1; } +DECLARE_XBOXKRNL_EXPORT2(KeInsertQueueDpc, kThreading, kStub, kSketchy); -SHIM_CALL KeRemoveQueueDpc_shim(PPCContext* ppc_context, - KernelState* kernel_state) { - uint32_t dpc_ptr = SHIM_GET_ARG_32(0); - - XELOGD("KeRemoveQueueDpc(%.8X)", dpc_ptr); - +dword_result_t KeRemoveQueueDpc(pointer_t dpc) { bool result = false; - uint32_t list_entry_ptr = dpc_ptr + 4; + uint32_t list_entry_ptr = dpc.guest_address() + 4; auto global_lock = xe::global_critical_region::AcquireDirect(); - auto dpc_list = kernel_state->dpc_list(); + auto dpc_list = kernel_state()->dpc_list(); if (dpc_list->IsQueued(list_entry_ptr)) { dpc_list->Remove(list_entry_ptr); result = true; } - SHIM_SET_RETURN_32(result ? 1 : 0); + return result ? 1 : 0; } +DECLARE_XBOXKRNL_EXPORT1(KeRemoveQueueDpc, kThreading, kImplemented); // https://github.com/Cxbx-Reloaded/Cxbx-Reloaded/blob/51e4dfcaacfdbd1a9692272931a436371492f72d/import/OpenXDK/include/xboxkrnl/xboxkrnl.h#L1372 struct X_ERWLOCK { @@ -1324,47 +1182,6 @@ DECLARE_XBOXKRNL_EXPORT1(InterlockedFlushSList, kThreading, kImplemented); void RegisterThreadingExports(xe::cpu::ExportResolver* export_resolver, KernelState* kernel_state) { - SHIM_SET_MAPPING("xboxkrnl.exe", ExCreateThread, state); - SHIM_SET_MAPPING("xboxkrnl.exe", ExTerminateThread, state); - SHIM_SET_MAPPING("xboxkrnl.exe", NtResumeThread, state); - SHIM_SET_MAPPING("xboxkrnl.exe", KeResumeThread, state); - SHIM_SET_MAPPING("xboxkrnl.exe", NtSuspendThread, state); - SHIM_SET_MAPPING("xboxkrnl.exe", KeSetAffinityThread, state); - SHIM_SET_MAPPING("xboxkrnl.exe", KeQueryBasePriorityThread, state); - SHIM_SET_MAPPING("xboxkrnl.exe", KeSetBasePriorityThread, state); - SHIM_SET_MAPPING("xboxkrnl.exe", KeSetDisableBoostThread, state); - - SHIM_SET_MAPPING("xboxkrnl.exe", KeGetCurrentProcessType, state); - SHIM_SET_MAPPING("xboxkrnl.exe", KeSetCurrentProcessType, state); - - SHIM_SET_MAPPING("xboxkrnl.exe", KeQueryPerformanceFrequency, state); - SHIM_SET_MAPPING("xboxkrnl.exe", NtYieldExecution, state); - SHIM_SET_MAPPING("xboxkrnl.exe", KeQuerySystemTime, state); - - SHIM_SET_MAPPING("xboxkrnl.exe", NtCreateSemaphore, state); - SHIM_SET_MAPPING("xboxkrnl.exe", NtReleaseSemaphore, state); - - SHIM_SET_MAPPING("xboxkrnl.exe", NtReleaseMutant, state); - - SHIM_SET_MAPPING("xboxkrnl.exe", NtCreateTimer, state); - SHIM_SET_MAPPING("xboxkrnl.exe", NtSetTimerEx, state); - SHIM_SET_MAPPING("xboxkrnl.exe", NtCancelTimer, state); - - SHIM_SET_MAPPING("xboxkrnl.exe", KeEnterCriticalRegion, state); - SHIM_SET_MAPPING("xboxkrnl.exe", KeLeaveCriticalRegion, state); - - SHIM_SET_MAPPING("xboxkrnl.exe", KeRaiseIrqlToDpcLevel, state); - SHIM_SET_MAPPING("xboxkrnl.exe", KfLowerIrql, state); - - // SHIM_SET_MAPPING("xboxkrnl.exe", NtQueueApcThread, state); - SHIM_SET_MAPPING("xboxkrnl.exe", KeInitializeApc, state); - SHIM_SET_MAPPING("xboxkrnl.exe", KeInsertQueueApc, state); - SHIM_SET_MAPPING("xboxkrnl.exe", KeRemoveQueueApc, state); - SHIM_SET_MAPPING("xboxkrnl.exe", KiApcNormalRoutineNop, state); - - SHIM_SET_MAPPING("xboxkrnl.exe", KeInitializeDpc, state); - SHIM_SET_MAPPING("xboxkrnl.exe", KeInsertQueueDpc, state); - SHIM_SET_MAPPING("xboxkrnl.exe", KeRemoveQueueDpc, state); } } // namespace xboxkrnl From 4f2326e5d7d7c8e471806740327ec17aaf38e59d Mon Sep 17 00:00:00 2001 From: gibbed Date: Tue, 20 Nov 2018 15:52:15 -0600 Subject: [PATCH 6/9] [Kernel] Oops. --- src/xenia/kernel/xam/xam_ui.cc | 1 - 1 file changed, 1 deletion(-) diff --git a/src/xenia/kernel/xam/xam_ui.cc b/src/xenia/kernel/xam/xam_ui.cc index 03019f9a4..bf1deb834 100644 --- a/src/xenia/kernel/xam/xam_ui.cc +++ b/src/xenia/kernel/xam/xam_ui.cc @@ -100,7 +100,6 @@ dword_result_t XamShowMessageBoxUI(dword_t user_index, lpwstring_t title_ptr, std::vector buttons; std::wstring all_buttons; for (uint32_t j = 0; j < button_count; ++j) { - auto test = button_ptrs[1]; uint32_t button_ptr = button_ptrs[j]; auto button = xe::load_and_swap( kernel_state()->memory()->TranslateVirtual(button_ptr)); From 7ae0e2b8e5d3280d20ea6d0eb87e312e92fa6d34 Mon Sep 17 00:00:00 2001 From: gibbed Date: Tue, 20 Nov 2018 16:05:45 -0600 Subject: [PATCH 7/9] [Kernel] Fix Travis whining. --- src/xenia/kernel/xam/xam_ui.cc | 3 +-- src/xenia/kernel/xboxkrnl/xboxkrnl_threading.cc | 3 +-- 2 files changed, 2 insertions(+), 4 deletions(-) diff --git a/src/xenia/kernel/xam/xam_ui.cc b/src/xenia/kernel/xam/xam_ui.cc index bf1deb834..28daeb0ce 100644 --- a/src/xenia/kernel/xam/xam_ui.cc +++ b/src/xenia/kernel/xam/xam_ui.cc @@ -335,8 +335,7 @@ void XamShowDirtyDiscErrorUI(dword_t user_index) { DECLARE_XAM_EXPORT1(XamShowDirtyDiscErrorUI, kUI, kImplemented); void RegisterUIExports(xe::cpu::ExportResolver* export_resolver, - KernelState* kernel_state) { -} + KernelState* kernel_state) {} } // namespace xam } // namespace kernel diff --git a/src/xenia/kernel/xboxkrnl/xboxkrnl_threading.cc b/src/xenia/kernel/xboxkrnl/xboxkrnl_threading.cc index 4f9a883e0..98f4d6363 100644 --- a/src/xenia/kernel/xboxkrnl/xboxkrnl_threading.cc +++ b/src/xenia/kernel/xboxkrnl/xboxkrnl_threading.cc @@ -1181,8 +1181,7 @@ pointer_result_t InterlockedFlushSList(pointer_t plist_ptr) { DECLARE_XBOXKRNL_EXPORT1(InterlockedFlushSList, kThreading, kImplemented); void RegisterThreadingExports(xe::cpu::ExportResolver* export_resolver, - KernelState* kernel_state) { -} + KernelState* kernel_state) {} } // namespace xboxkrnl } // namespace kernel From 0206e90398050bb2bc8d0172661cf72198ddc477 Mon Sep 17 00:00:00 2001 From: gibbed Date: Tue, 20 Nov 2018 16:25:07 -0600 Subject: [PATCH 8/9] [Kernel] Simplify RtlNtStatusToDosError logging. --- src/xenia/kernel/xboxkrnl/xboxkrnl_error.cc | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/src/xenia/kernel/xboxkrnl/xboxkrnl_error.cc b/src/xenia/kernel/xboxkrnl/xboxkrnl_error.cc index 8e533398c..676dcfea0 100644 --- a/src/xenia/kernel/xboxkrnl/xboxkrnl_error.cc +++ b/src/xenia/kernel/xboxkrnl/xboxkrnl_error.cc @@ -997,7 +997,7 @@ dword_result_t RtlNtStatusToDosError(dword_t source_status) { if (!result) { break; } - XELOGI("RtlNtStatusToDosError => %X", result); + XELOGI("RtlNtStatusToDosError %X => %X", status, result); return result; } ++error_table; @@ -1010,7 +1010,8 @@ dword_result_t RtlNtStatusToDosError(dword_t source_status) { XELOGE("RtlNtStatusToDosError lookup NOT IMPLEMENTED"); return 317; // ERROR_MR_MID_NOT_FOUND } -DECLARE_XBOXKRNL_EXPORT2(RtlNtStatusToDosError, kNone, kImportant, kLogResult); +DECLARE_XBOXKRNL_EXPORT3(RtlNtStatusToDosError, kNone, kImportant, + kHighFrequency, kLogResult); void RegisterErrorExports(xe::cpu::ExportResolver* export_resolver, KernelState* kernel_state) {} From 22e7500f6379570d0dd64f8409d19ae2d21e4bf8 Mon Sep 17 00:00:00 2001 From: gibbed Date: Tue, 20 Nov 2018 23:21:38 -0600 Subject: [PATCH 9/9] [CPU] Handle NaN in fctidxx, fctiwxx. --- src/xenia/cpu/ppc/ppc_emit_fpu.cc | 24 ++++++++++++++++++++++-- 1 file changed, 22 insertions(+), 2 deletions(-) diff --git a/src/xenia/cpu/ppc/ppc_emit_fpu.cc b/src/xenia/cpu/ppc/ppc_emit_fpu.cc index 24f2cd732..1aadfa82f 100644 --- a/src/xenia/cpu/ppc/ppc_emit_fpu.cc +++ b/src/xenia/cpu/ppc/ppc_emit_fpu.cc @@ -242,10 +242,20 @@ int InstrEmit_fcfidx(PPCHIRBuilder& f, const InstrData& i) { int InstrEmit_fctidxx_(PPCHIRBuilder& f, const InstrData& i, RoundMode round_mode) { - Value* v = f.Convert(f.LoadFPR(i.X.RB), INT64_TYPE, round_mode); + auto end = f.NewLabel(); + auto isnan = f.NewLabel(); + Value* v; + f.BranchTrue(f.IsNan(f.LoadFPR(i.X.RB)), isnan); + v = f.Convert(f.LoadFPR(i.X.RB), INT64_TYPE, round_mode); v = f.Cast(v, FLOAT64_TYPE); f.StoreFPR(i.X.RT, v); f.UpdateFPSCR(v, i.X.Rc); + f.Branch(end); + f.MarkLabel(isnan); + v = f.Cast(f.LoadConstantUint64(0x8000000000000000u), FLOAT64_TYPE); + f.StoreFPR(i.X.RT, v); + f.UpdateFPSCR(v, i.X.Rc); + f.MarkLabel(end); return 0; } @@ -260,10 +270,20 @@ int InstrEmit_fctidzx(PPCHIRBuilder& f, const InstrData& i) { int InstrEmit_fctiwxx_(PPCHIRBuilder& f, const InstrData& i, RoundMode round_mode) { - Value* v = f.Convert(f.LoadFPR(i.X.RB), INT32_TYPE, round_mode); + auto end = f.NewLabel(); + auto isnan = f.NewLabel(); + Value* v; + f.BranchTrue(f.IsNan(f.LoadFPR(i.X.RB)), isnan); + v = f.Convert(f.LoadFPR(i.X.RB), INT32_TYPE, round_mode); v = f.Cast(f.SignExtend(v, INT64_TYPE), FLOAT64_TYPE); f.StoreFPR(i.X.RT, v); f.UpdateFPSCR(v, i.X.Rc); + f.Branch(end); + f.MarkLabel(isnan); + v = f.Cast(f.LoadConstantUint32(0x80000000u), FLOAT64_TYPE); + f.StoreFPR(i.X.RT, v); + f.UpdateFPSCR(v, i.X.Rc); + f.MarkLabel(end); return 0; }