#include #include "loader/component_loader.hpp" #include "arxan.hpp" #include "scheduler.hpp" #include "game/game.hpp" #include namespace arxan { namespace { utils::hook::detour nt_close_hook; utils::hook::detour nt_query_information_process_hook; NTSTATUS WINAPI nt_query_information_process_stub(const HANDLE handle, const PROCESSINFOCLASS info_class, const PVOID info, const ULONG info_length, const PULONG ret_length) { auto* orig = static_cast(nt_query_information_process_hook. get_original()); const auto status = orig(handle, info_class, info, info_length, ret_length); if (NT_SUCCESS(status)) { if (info_class == ProcessBasicInformation) { static DWORD explorer_pid = 0; if (!explorer_pid) { auto* const shell_window = GetShellWindow(); GetWindowThreadProcessId(shell_window, &explorer_pid); } static_cast(info)->Reserved3 = PVOID(DWORD64(explorer_pid)); } else if (info_class == 30) // ProcessDebugObjectHandle { *static_cast(info) = nullptr; return 0xC0000353; } else if (info_class == 7) // ProcessDebugPort { *static_cast(info) = nullptr; } else if (info_class == 31) { *static_cast(info) = 1; } //https://docs.microsoft.com/en-us/windows/win32/api/winternl/nf-winternl-ntqueryinformationprocess } return status; } NTSTATUS NTAPI nt_close_stub(const HANDLE handle) { char info[16]; if (NtQueryObject(handle, OBJECT_INFORMATION_CLASS(4), &info, 2, nullptr) >= 0 && size_t(handle) != 0x12345) { auto* orig = static_cast(nt_close_hook.get_original()); return orig(handle); } return STATUS_INVALID_HANDLE; } LONG WINAPI exception_filter(const LPEXCEPTION_POINTERS info) { if (info->ExceptionRecord->ExceptionCode == STATUS_INVALID_HANDLE) { return EXCEPTION_CONTINUE_EXECUTION; } return EXCEPTION_CONTINUE_SEARCH; } void hide_being_debugged() { auto* const peb = PPEB(__readgsqword(0x60)); peb->BeingDebugged = false; *reinterpret_cast(LPSTR(peb) + 0xBC) &= ~0x70; } void remove_hardware_breakpoints() { CONTEXT context; ZeroMemory(&context, sizeof(context)); context.ContextFlags = CONTEXT_DEBUG_REGISTERS; auto* const thread = GetCurrentThread(); GetThreadContext(thread, &context); context.Dr0 = 0; context.Dr1 = 0; context.Dr2 = 0; context.Dr3 = 0; context.Dr6 = 0; context.Dr7 = 0; SetThreadContext(thread, &context); } BOOL WINAPI set_thread_context_stub(const HANDLE thread, CONTEXT* context) { return SetThreadContext(thread, context); } } class component final : public component_interface { public: void* load_import(const std::string& library, const std::string& function) override { if (function == "SetThreadContext") { //return set_thread_context_stub; } return nullptr; } void post_load() override { hide_being_debugged(); scheduler::loop(hide_being_debugged, scheduler::pipeline::async); const utils::nt::library ntdll("ntdll.dll"); nt_close_hook.create(ntdll.get_proc("NtClose"), nt_close_stub); nt_query_information_process_hook.create(ntdll.get_proc("NtQueryInformationProcess"), nt_query_information_process_stub); // https://www.geoffchappell.com/studies/windows/win32/ntdll/api/index.htm AddVectoredExceptionHandler(1, exception_filter); } void post_unpack() override { // cba to implement sp, not sure if it's even needed if (game::environment::is_sp()) { return; } } }; } REGISTER_COMPONENT(arxan::component)