| 
									
										
										
										
											2014-10-19 07:43:41 +02:00
										 |  |  | From 1de55db4e9e0412f144ae9e675c4f419b0d2e3da Mon Sep 17 00:00:00 2001 | 
					
						
							| 
									
										
										
										
											2014-10-19 00:45:51 +02:00
										 |  |  | From: Bernhard Reiter <ockham@raz.or.at> | 
					
						
							|  |  |  | Date: Wed, 9 Apr 2014 00:52:31 +0200 | 
					
						
							|  |  |  | Subject: imagehlp: Implement parts of BindImageEx to make freezing Python | 
					
						
							|  |  |  |  scripts work. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Fixes http://bugs.winehq.org/show_bug.cgi?id=3591 | 
					
						
							|  |  |  | ---
 | 
					
						
							|  |  |  |  dlls/imagehlp/modify.c      | 111 +++++++++++++++++++++++++++++++++++++++++--- | 
					
						
							|  |  |  |  dlls/imagehlp/tests/image.c |  18 +++---- | 
					
						
							|  |  |  |  2 files changed, 114 insertions(+), 15 deletions(-) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | diff --git a/dlls/imagehlp/modify.c b/dlls/imagehlp/modify.c
 | 
					
						
							| 
									
										
										
										
											2014-10-19 07:43:41 +02:00
										 |  |  | index debccc0..4ac3248 100644
 | 
					
						
							| 
									
										
										
										
											2014-10-19 00:45:51 +02:00
										 |  |  | --- a/dlls/imagehlp/modify.c
 | 
					
						
							|  |  |  | +++ b/dlls/imagehlp/modify.c
 | 
					
						
							|  |  |  | @@ -31,9 +31,11 @@ WINE_DEFAULT_DEBUG_CHANNEL(imagehlp);
 | 
					
						
							|  |  |  |   | 
					
						
							|  |  |  |  static WORD CalcCheckSum(DWORD StartValue, LPVOID BaseAddress, DWORD WordCount); | 
					
						
							|  |  |  |   | 
					
						
							|  |  |  | -
 | 
					
						
							|  |  |  |  /*********************************************************************** | 
					
						
							|  |  |  |   *		BindImage (IMAGEHLP.@) | 
					
						
							|  |  |  | + *
 | 
					
						
							|  |  |  | + * NOTES
 | 
					
						
							|  |  |  | + *   See BindImageEx
 | 
					
						
							|  |  |  |   */ | 
					
						
							|  |  |  |  BOOL WINAPI BindImage( | 
					
						
							|  |  |  |    PCSTR ImageName, PCSTR DllPath, PCSTR SymbolPath) | 
					
						
							|  |  |  | @@ -43,16 +45,113 @@ BOOL WINAPI BindImage(
 | 
					
						
							|  |  |  |   | 
					
						
							|  |  |  |  /*********************************************************************** | 
					
						
							|  |  |  |   *		BindImageEx (IMAGEHLP.@) | 
					
						
							|  |  |  | + *
 | 
					
						
							|  |  |  | + * Compute the virtual address of each function imported by a PE image
 | 
					
						
							|  |  |  | + *
 | 
					
						
							|  |  |  | + * PARAMS
 | 
					
						
							|  |  |  | + *
 | 
					
						
							|  |  |  | + *   Flags         [in] Bind options
 | 
					
						
							|  |  |  | + *   ImageName     [in] File name of the image to be bound
 | 
					
						
							|  |  |  | + *   DllPath       [in] Root of the fallback search path in case the ImageName file cannot be opened
 | 
					
						
							|  |  |  | + *   SymbolPath    [in] Symbol file root search path
 | 
					
						
							|  |  |  | + *   StatusRoutine [in] Pointer to a status routine which will be called during the binding process
 | 
					
						
							|  |  |  | + *
 | 
					
						
							|  |  |  | + * RETURNS
 | 
					
						
							|  |  |  | + *   Success: TRUE
 | 
					
						
							|  |  |  | + *   Failure: FALSE
 | 
					
						
							|  |  |  | + *
 | 
					
						
							|  |  |  | + * NOTES
 | 
					
						
							|  |  |  | + *  Binding is not implemented yet, so far this function only enumerates
 | 
					
						
							|  |  |  | + *  all imported dlls/functions and returns TRUE.
 | 
					
						
							|  |  |  |   */ | 
					
						
							|  |  |  |  BOOL WINAPI BindImageEx( | 
					
						
							|  |  |  |    DWORD Flags, PCSTR ImageName, PCSTR DllPath, PCSTR SymbolPath, | 
					
						
							|  |  |  |    PIMAGEHLP_STATUS_ROUTINE StatusRoutine) | 
					
						
							|  |  |  |  { | 
					
						
							|  |  |  | -  FIXME("(%d, %s, %s, %s, %p): stub\n",
 | 
					
						
							|  |  |  | -    Flags, debugstr_a(ImageName), debugstr_a(DllPath),
 | 
					
						
							|  |  |  | -    debugstr_a(SymbolPath), StatusRoutine
 | 
					
						
							|  |  |  | -  );
 | 
					
						
							|  |  |  | -  return TRUE;
 | 
					
						
							|  |  |  | +    LOADED_IMAGE loaded_image;
 | 
					
						
							|  |  |  | +    const IMAGE_IMPORT_DESCRIPTOR *import_desc;
 | 
					
						
							|  |  |  | +    ULONG size;
 | 
					
						
							|  |  |  | +
 | 
					
						
							|  |  |  | +    FIXME("(%d, %s, %s, %s, %p): semi-stub\n",
 | 
					
						
							|  |  |  | +        Flags, debugstr_a(ImageName), debugstr_a(DllPath),
 | 
					
						
							|  |  |  | +        debugstr_a(SymbolPath), StatusRoutine
 | 
					
						
							|  |  |  | +    );
 | 
					
						
							|  |  |  | +
 | 
					
						
							|  |  |  | +    if (!(MapAndLoad(ImageName, DllPath, &loaded_image, TRUE, TRUE))) return FALSE;
 | 
					
						
							|  |  |  | +
 | 
					
						
							|  |  |  | +    if (!(import_desc = RtlImageDirectoryEntryToData((HMODULE)loaded_image.MappedAddress, FALSE,
 | 
					
						
							|  |  |  | +                                                     IMAGE_DIRECTORY_ENTRY_IMPORT, &size)))
 | 
					
						
							|  |  |  | +    {
 | 
					
						
							|  |  |  | +        UnMapAndLoad(&loaded_image);
 | 
					
						
							|  |  |  | +        return TRUE; /* No imported modules means nothing to bind, so we're done. */
 | 
					
						
							|  |  |  | +    }
 | 
					
						
							|  |  |  | +
 | 
					
						
							|  |  |  | +    /* FIXME: Does native imagehlp support both 32-bit and 64-bit PE executables? */
 | 
					
						
							|  |  |  | +#ifdef _WIN64
 | 
					
						
							|  |  |  | +    if (loaded_image.FileHeader->OptionalHeader.Magic != IMAGE_NT_OPTIONAL_HDR64_MAGIC)
 | 
					
						
							|  |  |  | +#else
 | 
					
						
							|  |  |  | +    if (loaded_image.FileHeader->OptionalHeader.Magic != IMAGE_NT_OPTIONAL_HDR32_MAGIC)
 | 
					
						
							|  |  |  | +#endif
 | 
					
						
							|  |  |  | +    {
 | 
					
						
							|  |  |  | +        FIXME("Wrong architecture in PE header, unable to enumerate imports\n");
 | 
					
						
							|  |  |  | +        UnMapAndLoad(&loaded_image);
 | 
					
						
							|  |  |  | +        return TRUE;
 | 
					
						
							|  |  |  | +    }
 | 
					
						
							|  |  |  | +
 | 
					
						
							|  |  |  | +    for (; import_desc->Name && import_desc->FirstThunk; ++import_desc)
 | 
					
						
							|  |  |  | +    {
 | 
					
						
							|  |  |  | +        IMAGE_THUNK_DATA *thunk;
 | 
					
						
							|  |  |  | +        char dll_fullname[MAX_PATH];
 | 
					
						
							|  |  |  | +        const char *dll_name;
 | 
					
						
							|  |  |  | +
 | 
					
						
							|  |  |  | +        if (!(dll_name = ImageRvaToVa(loaded_image.FileHeader, loaded_image.MappedAddress,
 | 
					
						
							|  |  |  | +                                      import_desc->Name, 0)))
 | 
					
						
							|  |  |  | +        {
 | 
					
						
							|  |  |  | +            UnMapAndLoad(&loaded_image);
 | 
					
						
							|  |  |  | +            SetLastError(ERROR_INVALID_ACCESS); /* FIXME */
 | 
					
						
							|  |  |  | +            return FALSE;
 | 
					
						
							|  |  |  | +        }
 | 
					
						
							|  |  |  | +
 | 
					
						
							|  |  |  | +        if (StatusRoutine)
 | 
					
						
							|  |  |  | +            StatusRoutine(BindImportModule, ImageName, dll_name, 0, 0);
 | 
					
						
							|  |  |  | +
 | 
					
						
							|  |  |  | +        if (!SearchPathA(DllPath, dll_name, 0, sizeof(dll_fullname), dll_fullname, 0))
 | 
					
						
							|  |  |  | +        {
 | 
					
						
							|  |  |  | +            UnMapAndLoad(&loaded_image);
 | 
					
						
							|  |  |  | +            SetLastError(ERROR_FILE_NOT_FOUND);
 | 
					
						
							|  |  |  | +            return FALSE;
 | 
					
						
							|  |  |  | +        }
 | 
					
						
							|  |  |  | +
 | 
					
						
							|  |  |  | +        if (!(thunk = ImageRvaToVa(loaded_image.FileHeader, loaded_image.MappedAddress,
 | 
					
						
							|  |  |  | +                                   import_desc->OriginalFirstThunk ? import_desc->OriginalFirstThunk :
 | 
					
						
							|  |  |  | +                                   import_desc->FirstThunk, 0)))
 | 
					
						
							|  |  |  | +        {
 | 
					
						
							| 
									
										
										
										
											2014-10-19 07:43:41 +02:00
										 |  |  | +            ERR("Can't grab thunk data of %s, going to next imported DLL\n", dll_name);
 | 
					
						
							| 
									
										
										
										
											2014-10-19 00:45:51 +02:00
										 |  |  | +            continue;
 | 
					
						
							|  |  |  | +        }
 | 
					
						
							|  |  |  | +
 | 
					
						
							|  |  |  | +        for (; thunk->u1.Ordinal; ++thunk)
 | 
					
						
							|  |  |  | +        {
 | 
					
						
							|  |  |  | +            /* Ignoring ordinal imports for now */
 | 
					
						
							|  |  |  | +            if(!IMAGE_SNAP_BY_ORDINAL(thunk->u1.Ordinal))
 | 
					
						
							|  |  |  | +            {
 | 
					
						
							|  |  |  | +                IMAGE_IMPORT_BY_NAME *iibn;
 | 
					
						
							|  |  |  | +
 | 
					
						
							|  |  |  | +                if (!(iibn = ImageRvaToVa(loaded_image.FileHeader, loaded_image.MappedAddress,
 | 
					
						
							|  |  |  | +                                          thunk->u1.AddressOfData, 0)))
 | 
					
						
							|  |  |  | +                {
 | 
					
						
							|  |  |  | +                    ERR("Can't grab import by name info, skipping to next ordinal\n");
 | 
					
						
							|  |  |  | +                    continue;
 | 
					
						
							|  |  |  | +                }
 | 
					
						
							|  |  |  | +
 | 
					
						
							|  |  |  | +                if (StatusRoutine)
 | 
					
						
							|  |  |  | +                    StatusRoutine(BindImportProcedure, ImageName, dll_fullname, 0, (ULONG_PTR)iibn->Name);
 | 
					
						
							|  |  |  | +            }
 | 
					
						
							|  |  |  | +        }
 | 
					
						
							|  |  |  | +    }
 | 
					
						
							|  |  |  | +
 | 
					
						
							|  |  |  | +    UnMapAndLoad(&loaded_image);
 | 
					
						
							|  |  |  | +    return TRUE;
 | 
					
						
							|  |  |  |  } | 
					
						
							|  |  |  |   | 
					
						
							|  |  |  |   | 
					
						
							|  |  |  | diff --git a/dlls/imagehlp/tests/image.c b/dlls/imagehlp/tests/image.c
 | 
					
						
							|  |  |  | index 48443f5..4a19c16 100644
 | 
					
						
							|  |  |  | --- a/dlls/imagehlp/tests/image.c
 | 
					
						
							|  |  |  | +++ b/dlls/imagehlp/tests/image.c
 | 
					
						
							|  |  |  | @@ -390,10 +390,10 @@ static void test_bind_image_ex(void)
 | 
					
						
							|  |  |  |      SetLastError(0xdeadbeef); | 
					
						
							|  |  |  |      ret = pBindImageEx(BIND_NO_BOUND_IMPORTS | BIND_NO_UPDATE | BIND_ALL_IMAGES, "nonexistent.dll", 0, 0, | 
					
						
							|  |  |  |                         testing_status_routine); | 
					
						
							|  |  |  | -    todo_wine ok(!ret && ((GetLastError() == ERROR_FILE_NOT_FOUND) ||
 | 
					
						
							|  |  |  | -                 (GetLastError() == ERROR_INVALID_PARAMETER)),
 | 
					
						
							|  |  |  | -                 "expected ERROR_FILE_NOT_FOUND or ERROR_INVALID_PARAMETER, got %d\n",
 | 
					
						
							|  |  |  | -                 GetLastError());
 | 
					
						
							|  |  |  | +    ok(!ret && ((GetLastError() == ERROR_FILE_NOT_FOUND) ||
 | 
					
						
							|  |  |  | +       (GetLastError() == ERROR_INVALID_PARAMETER)),
 | 
					
						
							|  |  |  | +       "expected ERROR_FILE_NOT_FOUND or ERROR_INVALID_PARAMETER, got %d\n",
 | 
					
						
							|  |  |  | +       GetLastError());
 | 
					
						
							|  |  |  |   | 
					
						
							|  |  |  |      file = create_temp_file(temp_file); | 
					
						
							|  |  |  |      if (file == INVALID_HANDLE_VALUE) | 
					
						
							|  |  |  | @@ -415,14 +415,14 @@ static void test_bind_image_ex(void)
 | 
					
						
							|  |  |  |                         testing_status_routine); | 
					
						
							|  |  |  |      ok(ret, "BindImageEx failed: %d\n", GetLastError()); | 
					
						
							|  |  |  |   | 
					
						
							|  |  |  | -    todo_wine ok(status_routine_called[BindImportModule] == 1,
 | 
					
						
							|  |  |  | -                 "StatusRoutine was called %d times\n", status_routine_called[BindImportModule]);
 | 
					
						
							|  |  |  | +    ok(status_routine_called[BindImportModule] == 1,
 | 
					
						
							|  |  |  | +       "StatusRoutine was called %d times\n", status_routine_called[BindImportModule]);
 | 
					
						
							|  |  |  |   | 
					
						
							|  |  |  | -    todo_wine ok((status_routine_called[BindImportProcedure] == 1)
 | 
					
						
							|  |  |  | +    ok((status_routine_called[BindImportProcedure] == 1)
 | 
					
						
							|  |  |  |  #if defined(_WIN64) | 
					
						
							|  |  |  | -                 || broken(status_routine_called[BindImportProcedure] == 0) /* < Win8 */
 | 
					
						
							|  |  |  | +       || broken(status_routine_called[BindImportProcedure] == 0) /* < Win8 */
 | 
					
						
							|  |  |  |  #endif | 
					
						
							|  |  |  | -                 , "StatusRoutine was called %d times\n", status_routine_called[BindImportProcedure]);
 | 
					
						
							|  |  |  | +       , "StatusRoutine was called %d times\n", status_routine_called[BindImportProcedure]);
 | 
					
						
							|  |  |  |   | 
					
						
							|  |  |  |      DeleteFileA(temp_file); | 
					
						
							|  |  |  |  } | 
					
						
							|  |  |  | -- 
 | 
					
						
							|  |  |  | 2.1.2 | 
					
						
							|  |  |  | 
 |