Commit af7b97e7 authored by Eric Pouech's avatar Eric Pouech Committed by Alexandre Julliard

dbghelp: Reimplement EnumerateLoadedModules().

- correctly taking into accoung SYMOPT_INCLUDE_32BIT_MODULES option - converting, for 32bit modules requested from a 64bit module, the system32 paths into syswow64 Signed-off-by: 's avatarEric Pouech <eric.pouech@gmail.com>
parent 79bae0ed
MODULE = dbghelp.dll MODULE = dbghelp.dll
IMPORTLIB = dbghelp IMPORTLIB = dbghelp
IMPORTS = $(ZLIB_PE_LIBS) IMPORTS = kernelbase $(ZLIB_PE_LIBS)
EXTRAINCL = $(ZLIB_PE_CFLAGS) EXTRAINCL = $(ZLIB_PE_CFLAGS)
EXTRADEFS = -D_IMAGEHLP_SOURCE_ EXTRADEFS = -D_IMAGEHLP_SOURCE_
DELAYIMPORTS = version DELAYIMPORTS = version
......
...@@ -1262,53 +1262,118 @@ BOOL WINAPI EnumerateLoadedModules(HANDLE hProcess, ...@@ -1262,53 +1262,118 @@ BOOL WINAPI EnumerateLoadedModules(HANDLE hProcess,
return EnumerateLoadedModulesW64(hProcess, enum_load_modW64_32, &x); return EnumerateLoadedModulesW64(hProcess, enum_load_modW64_32, &x);
} }
static unsigned int load_and_grow_modules(HANDLE process, HMODULE** hmods, unsigned start, unsigned* alloc, DWORD filter)
{
DWORD needed;
BOOL ret;
while ((ret = EnumProcessModulesEx(process, *hmods + start, (*alloc - start) * sizeof(HMODULE),
&needed, filter)) &&
needed > (*alloc - start) * sizeof(HMODULE))
{
HMODULE* new = HeapReAlloc(GetProcessHeap(), 0, *hmods, (*alloc) * 2 * sizeof(HMODULE));
if (!new) return 0;
*hmods = new;
*alloc *= 2;
}
return ret ? needed / sizeof(HMODULE) : 0;
}
/****************************************************************** /******************************************************************
* EnumerateLoadedModulesW64 (DBGHELP.@) * EnumerateLoadedModulesW64 (DBGHELP.@)
* *
*/ */
BOOL WINAPI EnumerateLoadedModulesW64(HANDLE hProcess, BOOL WINAPI EnumerateLoadedModulesW64(HANDLE process,
PENUMLOADED_MODULES_CALLBACKW64 EnumLoadedModulesCallback, PENUMLOADED_MODULES_CALLBACKW64 enum_cb,
PVOID UserContext) PVOID user)
{ {
HMODULE* hMods; HMODULE* hmods;
WCHAR imagenameW[MAX_PATH]; unsigned alloc = 256, count, count32, i;
DWORD i, sz; USHORT pcs_machine, native_machine;
MODULEINFO mi; BOOL with_32bit_modules;
BOOL wow64; WCHAR imagenameW[MAX_PATH];
DWORD filter = LIST_MODULES_DEFAULT; MODULEINFO mi;
WCHAR* sysdir = NULL;
hMods = HeapAlloc(GetProcessHeap(), 0, 256 * sizeof(hMods[0])); WCHAR* wowdir = NULL;
if (!hMods) return FALSE; size_t sysdir_len = 0, wowdir_len = 0;
if (sizeof(void*) > sizeof(int) && /* process might not be a handle to a live process */
IsWow64Process(hProcess, &wow64) && if (!IsWow64Process2(process, &pcs_machine, &native_machine)) return FALSE;
wow64) with_32bit_modules = sizeof(void*) > sizeof(int) &&
filter = LIST_MODULES_32BIT; pcs_machine != IMAGE_FILE_MACHINE_UNKNOWN &&
(dbghelp_options & SYMOPT_INCLUDE_32BIT_MODULES);
if (!EnumProcessModulesEx(hProcess, hMods, 256 * sizeof(hMods[0]), &sz, filter))
{ if (!(hmods = HeapAlloc(GetProcessHeap(), 0, alloc * sizeof(hmods[0]))))
/* hProcess should also be a valid process handle !! */
HeapFree(GetProcessHeap(), 0, hMods);
return FALSE; return FALSE;
}
if (sz > 256 * sizeof(hMods[0])) /* Note:
* - we report modules returned from kernelbase.EnumProcessModulesEx
* - appending 32bit modules when possible and requested
*
* When considering 32bit modules in a wow64 child process, required from
* a 64bit process:
* - native returns from kernelbase.EnumProcessModulesEx
* redirected paths (that is in system32 directory), while
* dbghelp.EnumerateLoadedModulesWine returns the effective path
* (eg. syswow64 for x86_64).
* - (Except for the main module, if gotten from syswow64, where kernelbase
* will return the effective path)
* - Wine kernelbase (and ntdll) incorrectly return these modules from
* syswow64 (except for ntdll which is returned from system32).
* => for these modules, always perform a system32 => syswow64 path
* conversion (it'll work even if ntdll/kernelbase is fixed).
*/
if ((count = load_and_grow_modules(process, &hmods, 0, &alloc, LIST_MODULES_DEFAULT)) && with_32bit_modules)
{ {
hMods = HeapReAlloc(GetProcessHeap(), 0, hMods, sz); /* append 32bit modules when required */
if (!hMods || !EnumProcessModulesEx(hProcess, hMods, sz, &sz, filter)) if ((count32 = load_and_grow_modules(process, &hmods, count, &alloc, LIST_MODULES_32BIT)))
return FALSE; {
sysdir_len = GetSystemDirectoryW(NULL, 0);
wowdir_len = GetSystemWow64Directory2W(NULL, 0, pcs_machine);
if (!sysdir_len || !wowdir_len ||
!(sysdir = HeapAlloc(GetProcessHeap(), 0, (sysdir_len + 1 + wowdir_len + 1) * sizeof(WCHAR))))
{
HeapFree(GetProcessHeap(), 0, hmods);
return FALSE;
}
wowdir = sysdir + sysdir_len + 1;
if (GetSystemDirectoryW(sysdir, sysdir_len) >= sysdir_len)
FIXME("shouldn't happen\n");
if (GetSystemWow64Directory2W(wowdir, wowdir_len, pcs_machine) >= wowdir_len)
FIXME("shouldn't happen\n");
wcscat(sysdir, L"\\");
wcscat(wowdir, L"\\");
}
} }
sz /= sizeof(HMODULE); else count32 = 0;
for (i = 0; i < sz; i++)
for (i = 0; i < count + count32; i++)
{ {
if (!GetModuleInformation(hProcess, hMods[i], &mi, sizeof(mi)) || if (GetModuleInformation(process, hmods[i], &mi, sizeof(mi)) &&
!GetModuleFileNameExW(hProcess, hMods[i], imagenameW, ARRAY_SIZE(imagenameW))) GetModuleFileNameExW(process, hmods[i], imagenameW, ARRAY_SIZE(imagenameW)))
continue; {
EnumLoadedModulesCallback(imagenameW, (DWORD_PTR)mi.lpBaseOfDll, mi.SizeOfImage, /* rewrite path in system32 into syswow64 for 32bit modules */
UserContext); if (i >= count)
{
size_t len = wcslen(imagenameW);
if (!wcsnicmp(imagenameW, sysdir, sysdir_len) &&
(len - sysdir_len + wowdir_len) + 1 <= ARRAY_SIZE(imagenameW))
{
memmove(&imagenameW[wowdir_len], &imagenameW[sysdir_len], (len - sysdir_len) * sizeof(WCHAR));
memcpy(imagenameW, wowdir, wowdir_len * sizeof(WCHAR));
}
}
if (!enum_cb(imagenameW, (DWORD_PTR)mi.lpBaseOfDll, mi.SizeOfImage, user))
break;
}
} }
HeapFree(GetProcessHeap(), 0, hMods);
return sz != 0 && i == sz; HeapFree(GetProcessHeap(), 0, hmods);
HeapFree(GetProcessHeap(), 0, sysdir);
return count != 0;
} }
static void dbghelp_str_WtoA(const WCHAR *src, char *dst, int dst_len) static void dbghelp_str_WtoA(const WCHAR *src, char *dst, int dst_len)
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment