mirror of
				https://github.com/dolphin-emu/dolphin.git
				synced 2025-10-24 20:09:47 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			178 lines
		
	
	
		
			5.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			178 lines
		
	
	
		
			5.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright 2008 Dolphin Emulator Project
 | |
| // SPDX-License-Identifier: GPL-2.0-or-later
 | |
| 
 | |
| #include "Common/LdrWatcher.h"
 | |
| 
 | |
| #include <Windows.h>
 | |
| #include <TlHelp32.h>
 | |
| #include <string>
 | |
| #include <winternl.h>
 | |
| 
 | |
| typedef struct _LDR_DLL_LOADED_NOTIFICATION_DATA
 | |
| {
 | |
|   ULONG Flags;                   // Reserved.
 | |
|   PCUNICODE_STRING FullDllName;  // The full path name of the DLL module.
 | |
|   PCUNICODE_STRING BaseDllName;  // The base file name of the DLL module.
 | |
|   PVOID DllBase;                 // A pointer to the base address for the DLL in memory.
 | |
|   ULONG SizeOfImage;             // The size of the DLL image, in bytes.
 | |
| } LDR_DLL_LOADED_NOTIFICATION_DATA, *PLDR_DLL_LOADED_NOTIFICATION_DATA;
 | |
| 
 | |
| typedef struct _LDR_DLL_UNLOADED_NOTIFICATION_DATA
 | |
| {
 | |
|   ULONG Flags;                   // Reserved.
 | |
|   PCUNICODE_STRING FullDllName;  // The full path name of the DLL module.
 | |
|   PCUNICODE_STRING BaseDllName;  // The base file name of the DLL module.
 | |
|   PVOID DllBase;                 // A pointer to the base address for the DLL in memory.
 | |
|   ULONG SizeOfImage;             // The size of the DLL image, in bytes.
 | |
| } LDR_DLL_UNLOADED_NOTIFICATION_DATA, *PLDR_DLL_UNLOADED_NOTIFICATION_DATA;
 | |
| 
 | |
| typedef union _LDR_DLL_NOTIFICATION_DATA
 | |
| {
 | |
|   LDR_DLL_LOADED_NOTIFICATION_DATA Loaded;
 | |
|   LDR_DLL_UNLOADED_NOTIFICATION_DATA Unloaded;
 | |
| } LDR_DLL_NOTIFICATION_DATA, *PLDR_DLL_NOTIFICATION_DATA;
 | |
| typedef const LDR_DLL_NOTIFICATION_DATA* PCLDR_DLL_NOTIFICATION_DATA;
 | |
| 
 | |
| #define LDR_DLL_NOTIFICATION_REASON_LOADED (1)
 | |
| #define LDR_DLL_NOTIFICATION_REASON_UNLOADED (2)
 | |
| 
 | |
| typedef VOID NTAPI LDR_DLL_NOTIFICATION_FUNCTION(_In_ ULONG NotificationReason,
 | |
|                                                  _In_ PCLDR_DLL_NOTIFICATION_DATA NotificationData,
 | |
|                                                  _In_opt_ PVOID Context);
 | |
| typedef LDR_DLL_NOTIFICATION_FUNCTION* PLDR_DLL_NOTIFICATION_FUNCTION;
 | |
| 
 | |
| typedef NTSTATUS(NTAPI* LdrRegisterDllNotification_t)(
 | |
|     _In_ ULONG Flags, _In_ PLDR_DLL_NOTIFICATION_FUNCTION NotificationFunction,
 | |
|     _In_opt_ PVOID Context, _Out_ PVOID* Cookie);
 | |
| 
 | |
| typedef NTSTATUS(NTAPI* LdrUnregisterDllNotification_t)(_In_ PVOID Cookie);
 | |
| 
 | |
| static void LdrObserverRun(const LdrObserver& observer, PCUNICODE_STRING module_name,
 | |
|                            uintptr_t base_address)
 | |
| {
 | |
|   for (auto& needle : observer.module_names)
 | |
|   {
 | |
|     // Like RtlCompareUnicodeString, but saves dynamically resolving it.
 | |
|     // NOTE: Does not compare null terminator.
 | |
|     auto compare_length = module_name->Length / sizeof(wchar_t);
 | |
|     if (!_wcsnicmp(needle.c_str(), module_name->Buffer, compare_length))
 | |
|       observer.action({needle, base_address});
 | |
|   }
 | |
| }
 | |
| 
 | |
| static VOID DllNotificationCallback(ULONG NotificationReason,
 | |
|                                     PCLDR_DLL_NOTIFICATION_DATA NotificationData, PVOID Context)
 | |
| {
 | |
|   if (NotificationReason != LDR_DLL_NOTIFICATION_REASON_LOADED)
 | |
|     return;
 | |
|   auto& data = NotificationData->Loaded;
 | |
|   auto observer = static_cast<const LdrObserver*>(Context);
 | |
|   LdrObserverRun(*observer, data.BaseDllName, reinterpret_cast<uintptr_t>(data.DllBase));
 | |
| }
 | |
| 
 | |
| // This only works on Vista+. On lower platforms, it will be a no-op.
 | |
| class LdrDllNotifier
 | |
| {
 | |
| public:
 | |
|   static LdrDllNotifier& GetInstance()
 | |
|   {
 | |
|     static LdrDllNotifier notifier;
 | |
|     return notifier;
 | |
|   }
 | |
|   void Install(LdrObserver* observer);
 | |
|   void Uninstall(LdrObserver* observer);
 | |
| 
 | |
| private:
 | |
|   LdrDllNotifier();
 | |
|   bool Init();
 | |
|   LdrRegisterDllNotification_t LdrRegisterDllNotification{};
 | |
|   LdrUnregisterDllNotification_t LdrUnregisterDllNotification{};
 | |
|   bool initialized{};
 | |
| };
 | |
| 
 | |
| LdrDllNotifier::LdrDllNotifier()
 | |
| {
 | |
|   initialized = Init();
 | |
| }
 | |
| 
 | |
| bool LdrDllNotifier::Init()
 | |
| {
 | |
|   auto ntdll = GetModuleHandleW(L"ntdll");
 | |
|   if (!ntdll)
 | |
|     return false;
 | |
|   LdrRegisterDllNotification = reinterpret_cast<decltype(LdrRegisterDllNotification)>(
 | |
|       GetProcAddress(ntdll, "LdrRegisterDllNotification"));
 | |
|   if (!LdrRegisterDllNotification)
 | |
|     return false;
 | |
|   LdrUnregisterDllNotification = reinterpret_cast<decltype(LdrUnregisterDllNotification)>(
 | |
|       GetProcAddress(ntdll, "LdrUnregisterDllNotification"));
 | |
|   if (!LdrUnregisterDllNotification)
 | |
|     return false;
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| void LdrDllNotifier::Install(LdrObserver* observer)
 | |
| {
 | |
|   if (!initialized)
 | |
|     return;
 | |
|   void* cookie{};
 | |
|   if (!NT_SUCCESS(LdrRegisterDllNotification(0, DllNotificationCallback,
 | |
|                                              static_cast<PVOID>(observer), &cookie)))
 | |
|     cookie = {};
 | |
|   observer->cookie = cookie;
 | |
|   return;
 | |
| }
 | |
| 
 | |
| void LdrDllNotifier::Uninstall(LdrObserver* observer)
 | |
| {
 | |
|   if (!initialized)
 | |
|     return;
 | |
|   LdrUnregisterDllNotification(observer->cookie);
 | |
|   observer->cookie = {};
 | |
|   return;
 | |
| }
 | |
| 
 | |
| LdrWatcher::~LdrWatcher()
 | |
| {
 | |
|   UninstallAll();
 | |
| }
 | |
| 
 | |
| // Needed for RtlInitUnicodeString
 | |
| #pragma comment(lib, "ntdll")
 | |
| 
 | |
| bool LdrWatcher::InjectCurrentModules(const LdrObserver& observer)
 | |
| {
 | |
|   // Use TlHelp32 instead of psapi functions to reduce dolphin's dependency on psapi
 | |
|   // (revisit this when Win7 support is dropped).
 | |
|   HANDLE snapshot = CreateToolhelp32Snapshot(TH32CS_SNAPMODULE, GetCurrentProcessId());
 | |
|   if (snapshot == INVALID_HANDLE_VALUE)
 | |
|     return false;
 | |
|   MODULEENTRY32 entry;
 | |
|   entry.dwSize = sizeof(entry);
 | |
|   for (BOOL rv = Module32First(snapshot, &entry); rv == TRUE; rv = Module32Next(snapshot, &entry))
 | |
|   {
 | |
|     UNICODE_STRING module_name;
 | |
|     RtlInitUnicodeString(&module_name, entry.szModule);
 | |
|     LdrObserverRun(observer, &module_name, reinterpret_cast<uintptr_t>(entry.modBaseAddr));
 | |
|   }
 | |
|   CloseHandle(snapshot);
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| void LdrWatcher::Install(const LdrObserver& observer)
 | |
| {
 | |
|   observers.emplace_back(observer);
 | |
|   auto& new_observer = observers.back();
 | |
|   // Register for notifications before looking at the list of current modules.
 | |
|   // This ensures none are missed, but there is a tiny chance some will be seen twice.
 | |
|   LdrDllNotifier::GetInstance().Install(&new_observer);
 | |
|   InjectCurrentModules(new_observer);
 | |
| }
 | |
| 
 | |
| void LdrWatcher::UninstallAll()
 | |
| {
 | |
|   for (auto& observer : observers)
 | |
|     LdrDllNotifier::GetInstance().Uninstall(&observer);
 | |
|   observers.clear();
 | |
| }
 | 
