| /* |
| * Copyright (C) 2004 Stefan Leichter |
| * Copyright (C) 2017 Akihiro Sagawa |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Lesser General Public |
| * License as published by the Free Software Foundation; either |
| * version 2.1 of the License, or (at your option) any later version. |
| * |
| * This library is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * Lesser General Public License for more details. |
| * |
| * You should have received a copy of the GNU Lesser General Public |
| * License along with this library; if not, write to the Free Software |
| * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA |
| */ |
| |
| #include <stdarg.h> |
| #include <stdio.h> |
| #include <assert.h> |
| |
| #include "windef.h" |
| #include "winbase.h" |
| #include "winerror.h" |
| #include "winnls.h" |
| #include "winuser.h" |
| #include "winver.h" |
| #include "verrsrc.h" |
| #include "wine/test.h" |
| |
| #define MY_LAST_ERROR ((DWORD)-1) |
| #define EXPECT_BAD_PATH__NOT_FOUND \ |
| ok( (ERROR_PATH_NOT_FOUND == GetLastError()) || \ |
| (ERROR_RESOURCE_DATA_NOT_FOUND == GetLastError()) || \ |
| (ERROR_FILE_NOT_FOUND == GetLastError()) || \ |
| (ERROR_BAD_PATHNAME == GetLastError()) || \ |
| (ERROR_SUCCESS == GetLastError()), \ |
| "Last error wrong! ERROR_RESOURCE_DATA_NOT_FOUND/ERROR_BAD_PATHNAME (98)/" \ |
| "ERROR_PATH_NOT_FOUND (NT4)/ERROR_FILE_NOT_FOUND (2k3) " \ |
| "ERROR_SUCCESS (2k) expected, got %u\n", GetLastError()); |
| #define EXPECT_INVALID__NOT_FOUND \ |
| ok( (ERROR_PATH_NOT_FOUND == GetLastError()) || \ |
| (ERROR_RESOURCE_DATA_NOT_FOUND == GetLastError()) || \ |
| (ERROR_FILE_NOT_FOUND == GetLastError()) || \ |
| (ERROR_INVALID_PARAMETER == GetLastError()) || \ |
| (ERROR_SUCCESS == GetLastError()), \ |
| "Last error wrong! ERROR_RESOURCE_DATA_NOT_FOUND/ERROR_INVALID_PARAMETER (98)/" \ |
| "ERROR_PATH_NOT_FOUND (NT4)/ERROR_FILE_NOT_FOUND (2k3) " \ |
| "ERROR_SUCCESS (2k) expected, got %u\n", GetLastError()); |
| |
| static void create_file(const CHAR *name) |
| { |
| HANDLE file; |
| DWORD written; |
| |
| file = CreateFileA(name, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, 0, NULL); |
| ok(file != INVALID_HANDLE_VALUE, "Failure to open file %s\n", name); |
| WriteFile(file, name, strlen(name), &written, NULL); |
| WriteFile(file, "\n", strlen("\n"), &written, NULL); |
| CloseHandle(file); |
| } |
| |
| static void test_info_size(void) |
| { DWORD hdl, retval; |
| char mypath[MAX_PATH] = ""; |
| |
| SetLastError(MY_LAST_ERROR); |
| retval = GetFileVersionInfoSizeA( NULL, NULL); |
| ok( !retval, |
| "GetFileVersionInfoSizeA result wrong! 0L expected, got 0x%08x\n", |
| retval); |
| EXPECT_INVALID__NOT_FOUND; |
| |
| hdl = 0x55555555; |
| SetLastError(MY_LAST_ERROR); |
| retval = GetFileVersionInfoSizeA( NULL, &hdl); |
| ok( !retval, |
| "GetFileVersionInfoSizeA result wrong! 0L expected, got 0x%08x\n", |
| retval); |
| EXPECT_INVALID__NOT_FOUND; |
| ok( hdl == 0L, |
| "Handle wrong! 0L expected, got 0x%08x\n", hdl); |
| |
| SetLastError(MY_LAST_ERROR); |
| retval = GetFileVersionInfoSizeA( "", NULL); |
| ok( !retval, |
| "GetFileVersionInfoSizeA result wrong! 0L expected, got 0x%08x\n", |
| retval); |
| EXPECT_BAD_PATH__NOT_FOUND; |
| |
| hdl = 0x55555555; |
| SetLastError(MY_LAST_ERROR); |
| retval = GetFileVersionInfoSizeA( "", &hdl); |
| ok( !retval, |
| "GetFileVersionInfoSizeA result wrong! 0L expected, got 0x%08x\n", |
| retval); |
| EXPECT_BAD_PATH__NOT_FOUND; |
| ok( hdl == 0L, |
| "Handle wrong! 0L expected, got 0x%08x\n", hdl); |
| |
| SetLastError(MY_LAST_ERROR); |
| retval = GetFileVersionInfoSizeA( "kernel32.dll", NULL); |
| ok( retval, |
| "GetFileVersionInfoSizeA result wrong! <> 0L expected, got 0x%08x\n", |
| retval); |
| ok((NO_ERROR == GetLastError()) || (MY_LAST_ERROR == GetLastError()), |
| "Last error wrong! NO_ERROR/0x%08x (NT4) expected, got %u\n", |
| MY_LAST_ERROR, GetLastError()); |
| |
| hdl = 0x55555555; |
| SetLastError(MY_LAST_ERROR); |
| retval = GetFileVersionInfoSizeA( "kernel32.dll", &hdl); |
| ok( retval, |
| "GetFileVersionInfoSizeA result wrong! <> 0L expected, got 0x%08x\n", |
| retval); |
| ok((NO_ERROR == GetLastError()) || (MY_LAST_ERROR == GetLastError()), |
| "Last error wrong! NO_ERROR/0x%08x (NT4) expected, got %u\n", |
| MY_LAST_ERROR, GetLastError()); |
| ok( hdl == 0L, |
| "Handle wrong! 0L expected, got 0x%08x\n", hdl); |
| |
| SetLastError(MY_LAST_ERROR); |
| retval = GetFileVersionInfoSizeA( "notexist.dll", NULL); |
| ok( !retval, |
| "GetFileVersionInfoSizeA result wrong! 0L expected, got 0x%08x\n", |
| retval); |
| ok( (ERROR_FILE_NOT_FOUND == GetLastError()) || |
| (ERROR_RESOURCE_DATA_NOT_FOUND == GetLastError()) || |
| (MY_LAST_ERROR == GetLastError()) || |
| (ERROR_SUCCESS == GetLastError()), /* win2k */ |
| "Last error wrong! ERROR_FILE_NOT_FOUND/ERROR_RESOURCE_DATA_NOT_FOUND " |
| "(XP)/0x%08x (NT4) expected, got %u\n", MY_LAST_ERROR, GetLastError()); |
| |
| /* test a currently loaded executable */ |
| if(GetModuleFileNameA(NULL, mypath, MAX_PATH)) { |
| hdl = 0x55555555; |
| SetLastError(MY_LAST_ERROR); |
| retval = GetFileVersionInfoSizeA( mypath, &hdl); |
| ok( retval, |
| "GetFileVersionInfoSizeA result wrong! <> 0L expected, got 0x%08x\n", |
| retval); |
| ok((NO_ERROR == GetLastError()) || (MY_LAST_ERROR == GetLastError()), |
| "Last error wrong! NO_ERROR/0x%08x (NT4) expected, got %u\n", |
| MY_LAST_ERROR, GetLastError()); |
| ok( hdl == 0L, |
| "Handle wrong! 0L expected, got 0x%08x\n", hdl); |
| } |
| else |
| trace("skipping GetModuleFileNameA(NULL,..) failed\n"); |
| |
| /* test a not loaded executable */ |
| if(GetSystemDirectoryA(mypath, MAX_PATH)) { |
| lstrcatA(mypath, "\\regsvr32.exe"); |
| |
| if(INVALID_FILE_ATTRIBUTES == GetFileAttributesA(mypath)) |
| trace("GetFileAttributesA(%s) failed\n", mypath); |
| else { |
| hdl = 0x55555555; |
| SetLastError(MY_LAST_ERROR); |
| retval = GetFileVersionInfoSizeA( mypath, &hdl); |
| ok( retval, |
| "GetFileVersionInfoSizeA result wrong! <> 0L expected, got 0x%08x\n", |
| retval); |
| ok((NO_ERROR == GetLastError()) || (MY_LAST_ERROR == GetLastError()), |
| "Last error wrong! NO_ERROR/0x%08x (NT4) expected, got %u\n", |
| MY_LAST_ERROR, GetLastError()); |
| ok( hdl == 0L, |
| "Handle wrong! 0L expected, got 0x%08x\n", hdl); |
| } |
| } |
| else |
| trace("skipping GetSystemDirectoryA(mypath,..) failed\n"); |
| |
| create_file("test.txt"); |
| |
| /* no version info */ |
| SetLastError(0xdeadbeef); |
| hdl = 0xcafe; |
| retval = GetFileVersionInfoSizeA("test.txt", &hdl); |
| ok(retval == 0, "Expected 0, got %d\n", retval); |
| ok(hdl == 0, "Expected 0, got %d\n", hdl); |
| ok(GetLastError() == ERROR_RESOURCE_DATA_NOT_FOUND || |
| GetLastError() == ERROR_SUCCESS, /* win2k */ |
| "Expected ERROR_RESOURCE_DATA_NOT_FOUND, got %d\n", GetLastError()); |
| |
| DeleteFileA("test.txt"); |
| } |
| |
| static void VersionDwordLong2String(DWORDLONG Version, LPSTR lpszVerString) |
| { |
| WORD a, b, c, d; |
| |
| a = (WORD)(Version >> 48); |
| b = (WORD)((Version >> 32) & 0xffff); |
| c = (WORD)((Version >> 16) & 0xffff); |
| d = (WORD)(Version & 0xffff); |
| |
| sprintf(lpszVerString, "%d.%d.%d.%d", a, b, c, d); |
| } |
| |
| static void test_info(void) |
| { |
| DWORD hdl, retval; |
| PVOID pVersionInfo = NULL; |
| BOOL boolret; |
| VS_FIXEDFILEINFO *pFixedVersionInfo; |
| UINT uiLength; |
| char VersionString[MAX_PATH]; |
| static const char backslash[] = "\\"; |
| DWORDLONG dwlVersion; |
| |
| hdl = 0x55555555; |
| SetLastError(MY_LAST_ERROR); |
| retval = GetFileVersionInfoSizeA( "kernel32.dll", &hdl); |
| ok( retval, |
| "GetFileVersionInfoSizeA result wrong! <> 0L expected, got 0x%08x\n", |
| retval); |
| ok((NO_ERROR == GetLastError()) || (MY_LAST_ERROR == GetLastError()), |
| "Last error wrong! NO_ERROR/0x%08x (NT4) expected, got %u\n", |
| MY_LAST_ERROR, GetLastError()); |
| ok( hdl == 0L, |
| "Handle wrong! 0L expected, got 0x%08x\n", hdl); |
| |
| if ( retval == 0 || hdl != 0) |
| return; |
| |
| pVersionInfo = HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, retval ); |
| ok(pVersionInfo != 0, "HeapAlloc failed\n" ); |
| if (pVersionInfo == 0) |
| return; |
| |
| if (0) |
| { |
| /* this test crashes on WinNT4 |
| */ |
| boolret = GetFileVersionInfoA( "kernel32.dll", 0, retval, 0); |
| ok (!boolret, "GetFileVersionInfoA should have failed: GetLastError = %u\n", GetLastError()); |
| ok ((GetLastError() == ERROR_INVALID_DATA) || (GetLastError() == ERROR_BAD_PATHNAME) || |
| (GetLastError() == NO_ERROR), |
| "Last error wrong! ERROR_INVALID_DATA/ERROR_BAD_PATHNAME (ME)/" |
| "NO_ERROR (95) expected, got %u\n", |
| GetLastError()); |
| } |
| |
| boolret = GetFileVersionInfoA( "kernel32.dll", 0, retval, pVersionInfo ); |
| ok (boolret, "GetFileVersionInfoA failed: GetLastError = %u\n", GetLastError()); |
| if (!boolret) |
| goto cleanup; |
| |
| boolret = VerQueryValueA( pVersionInfo, NULL, (LPVOID *)&pFixedVersionInfo, &uiLength ); |
| ok (boolret || GetLastError() == NO_ERROR /* Win98 */, |
| "VerQueryValueA failed: GetLastError = %u\n", GetLastError()); |
| |
| boolret = VerQueryValueA( pVersionInfo, "", (LPVOID *)&pFixedVersionInfo, &uiLength ); |
| ok (boolret, "VerQueryValueA failed: GetLastError = %u\n", GetLastError()); |
| |
| boolret = VerQueryValueA( pVersionInfo, backslash, (LPVOID *)&pFixedVersionInfo, &uiLength ); |
| ok (boolret, "VerQueryValueA failed: GetLastError = %u\n", GetLastError()); |
| if (!boolret) |
| goto cleanup; |
| |
| dwlVersion = (((DWORDLONG)pFixedVersionInfo->dwFileVersionMS) << 32) + |
| pFixedVersionInfo->dwFileVersionLS; |
| |
| VersionDwordLong2String(dwlVersion, VersionString); |
| |
| trace("kernel32.dll version: %s\n", VersionString); |
| |
| if (0) |
| { |
| /* this test crashes on WinNT4 |
| */ |
| boolret = VerQueryValueA( pVersionInfo, backslash, (LPVOID *)&pFixedVersionInfo, 0); |
| ok (boolret, "VerQueryValue failed: GetLastError = %u\n", GetLastError()); |
| } |
| |
| cleanup: |
| HeapFree( GetProcessHeap(), 0, pVersionInfo); |
| } |
| |
| static void test_32bit_win(void) |
| { |
| DWORD hdlA, retvalA; |
| DWORD hdlW, retvalW = 0; |
| BOOL retA,retW; |
| PVOID pVersionInfoA = NULL; |
| PVOID pVersionInfoW = NULL; |
| char *pBufA; |
| WCHAR *pBufW; |
| UINT uiLengthA, uiLengthW; |
| char mypathA[MAX_PATH]; |
| WCHAR mypathW[MAX_PATH]; |
| char rootA[] = "\\"; |
| WCHAR rootW[] = { '\\', 0 }; |
| WCHAR emptyW[] = { 0 }; |
| char varfileinfoA[] = "\\VarFileInfo\\Translation"; |
| WCHAR varfileinfoW[] = { '\\','V','a','r','F','i','l','e','I','n','f','o', |
| '\\','T','r','a','n','s','l','a','t','i','o','n', 0 }; |
| char WineVarFileInfoA[] = { 0x09, 0x04, 0xE4, 0x04 }; |
| char FileDescriptionA[] = "\\StringFileInfo\\040904E4\\FileDescription"; |
| WCHAR FileDescriptionW[] = { '\\','S','t','r','i','n','g','F','i','l','e','I','n','f','o', |
| '\\','0','4','0','9','0','4','E','4', |
| '\\','F','i','l','e','D','e','s','c','r','i','p','t','i','o','n', 0 }; |
| char WineFileDescriptionA[] = "FileDescription"; |
| WCHAR WineFileDescriptionW[] = { 'F','i','l','e','D','e','s','c','r','i','p','t','i','o','n', 0 }; |
| BOOL is_unicode_enabled = TRUE; |
| |
| /* A copy from dlls/version/info.c */ |
| typedef struct |
| { |
| WORD wLength; |
| WORD wValueLength; |
| WORD wType; |
| WCHAR szKey[1]; |
| #if 0 /* variable length structure */ |
| /* DWORD aligned */ |
| BYTE Value[]; |
| /* DWORD aligned */ |
| VS_VERSION_INFO_STRUCT32 Children[]; |
| #endif |
| } VS_VERSION_INFO_STRUCT32; |
| |
| /* If we call GetFileVersionInfoA on a system that supports Unicode, NT/W2K/XP/W2K3 (by default) and Wine, |
| * the versioninfo will contain Unicode strings. |
| * Part of the test is to call both the A and W versions, which should have the same Version Information |
| * for some requests, on systems that support both calls. |
| */ |
| |
| /* First get the versioninfo via the W versions */ |
| SetLastError(0xdeadbeef); |
| GetModuleFileNameW(NULL, mypathW, MAX_PATH); |
| if (GetLastError() == ERROR_CALL_NOT_IMPLEMENTED) |
| { |
| win_skip("GetModuleFileNameW not existing on this platform, skipping comparison between A- and W-calls\n"); |
| is_unicode_enabled = FALSE; |
| } |
| |
| if (is_unicode_enabled) |
| { |
| retvalW = GetFileVersionInfoSizeW( mypathW, &hdlW); |
| pVersionInfoW = HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, retvalW ); |
| retW = GetFileVersionInfoW( mypathW, 0, retvalW, pVersionInfoW ); |
| ok(retW, "GetFileVersionInfo failed: GetLastError = %u\n", GetLastError()); |
| } |
| |
| GetModuleFileNameA(NULL, mypathA, MAX_PATH); |
| retvalA = GetFileVersionInfoSizeA( mypathA, &hdlA); |
| pVersionInfoA = HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, retvalA ); |
| retA = GetFileVersionInfoA( mypathA, 0, retvalA, pVersionInfoA ); |
| ok(retA, "GetFileVersionInfo failed: GetLastError = %u\n", GetLastError()); |
| |
| if (is_unicode_enabled) |
| { |
| ok( retvalA == retvalW, "The size of the struct should be the same for both A/W calls, it is (%d) vs. (%d)\n", |
| retvalA, retvalW); |
| ok( !memcmp(pVersionInfoA, pVersionInfoW, retvalA), "Both structs should be the same, they aren't\n"); |
| } |
| |
| /* The structs on Windows are bigger than just the struct for the basic information. The total struct |
| * contains also an empty part, which is used for converted strings. The converted strings are a result |
| * of calling VerQueryValueA on a 32bit resource and calling VerQueryValueW on a 16bit resource. |
| * The first WORD of the structure (wLength) shows the size of the base struct. The total struct size depends |
| * on the Windows version: |
| * |
| * 16bits resource (numbers are from a sample app): |
| * |
| * Windows Version Retrieved with A/W wLength StructSize |
| * ==================================================================================== |
| * Win98 A 0x01B4 (436) 436 |
| * NT4 A/W 0x01B4 (436) 2048 ??? |
| * W2K/XP/W2K3 A/W 0x01B4 (436) 1536 which is (436 - sizeof(VS_FIXEDFILEINFO)) * 4 |
| * |
| * 32bits resource (numbers are from this test executable version_crosstest.exe): |
| * Windows Version Retrieved with A/W wLength StructSize |
| * ============================================================= |
| * Win98 A 0x01E0 (480) 848 (structure data doesn't seem correct) |
| * NT4 A/W 0x0350 (848) 1272 (848 * 1.5) |
| * W2K/XP/W2K3 A/W 0x0350 (848) 1700 which is (848 * 2) + 4 |
| * |
| * Wine will follow the implementation (eventually) of W2K/XP/W2K3 |
| */ |
| |
| /* Now some tests for the above (only if we are unicode enabled) */ |
| |
| if (is_unicode_enabled) |
| { |
| VS_VERSION_INFO_STRUCT32 *vvis = pVersionInfoW; |
| ok ( retvalW == ((vvis->wLength * 2) + 4) || retvalW == (vvis->wLength * 1.5), |
| "Structure is not of the correct size\n"); |
| } |
| |
| /* Although the 32bit resource structures contain Unicode strings, VerQueryValueA will always return normal strings, |
| * VerQueryValueW will always return Unicode ones. (That means everything returned for StringFileInfo requests). |
| */ |
| |
| /* Get the VS_FIXEDFILEINFO information, this must be the same for both A- and W-Calls */ |
| |
| retA = VerQueryValueA( pVersionInfoA, rootA, (LPVOID *)&pBufA, &uiLengthA ); |
| ok (retA, "VerQueryValueA failed: GetLastError = %u\n", GetLastError()); |
| ok ( uiLengthA == sizeof(VS_FIXEDFILEINFO), "Size (%d) doesn't match the size of the VS_FIXEDFILEINFO struct\n", uiLengthA); |
| |
| if (is_unicode_enabled) |
| { |
| if(0) |
| { /* This causes Vista and w2k8 to crash */ |
| retW = VerQueryValueW( pVersionInfoW, NULL, (LPVOID *)&pBufW, &uiLengthW ); |
| ok (retW, "VerQueryValueW failed: GetLastError = %u\n", GetLastError()); |
| } |
| |
| retW = VerQueryValueW( pVersionInfoW, emptyW, (LPVOID *)&pBufW, &uiLengthW ); |
| ok (retW, "VerQueryValueW failed: GetLastError = %u\n", GetLastError()); |
| |
| retW = VerQueryValueW( pVersionInfoW, rootW, (LPVOID *)&pBufW, &uiLengthW ); |
| ok (retW, "VerQueryValueW failed: GetLastError = %u\n", GetLastError()); |
| ok ( uiLengthW == sizeof(VS_FIXEDFILEINFO), "Size (%d) doesn't match the size of the VS_FIXEDFILEINFO struct\n", uiLengthW ); |
| |
| ok( uiLengthA == uiLengthW, "The size of VS_FIXEDFILEINFO should be the same for both A/W calls, it is (%d) vs. (%d)\n", |
| uiLengthA, uiLengthW); |
| ok( !memcmp(pBufA, pBufW, uiLengthA), "Both values should be the same, they aren't\n"); |
| } |
| |
| /* Get some VarFileInfo information, this must be the same for both A- and W-Calls */ |
| |
| retA = VerQueryValueA( pVersionInfoA, varfileinfoA, (LPVOID *)&pBufA, &uiLengthA ); |
| ok (retA, "VerQueryValueA failed: GetLastError = %u\n", GetLastError()); |
| ok( !memcmp(pBufA, WineVarFileInfoA, uiLengthA), "The VarFileInfo should have matched 0904e404 (non case sensitive)\n"); |
| |
| if (is_unicode_enabled) |
| { |
| retW = VerQueryValueW( pVersionInfoW, varfileinfoW, (LPVOID *)&pBufW, &uiLengthW ); |
| ok (retW, "VerQueryValueW failed: GetLastError = %u\n", GetLastError()); |
| ok( uiLengthA == uiLengthW, "The size of the VarFileInfo information should be the same for both A/W calls, it is (%d) vs. (%d)\n", |
| uiLengthA, uiLengthW); |
| ok( !memcmp(pBufA, pBufW, uiLengthA), "Both values should be the same, they aren't\n"); |
| } |
| |
| /* Get some StringFileInfo information, this will be ANSI for A-Calls and Unicode for W-Calls */ |
| |
| retA = VerQueryValueA( pVersionInfoA, FileDescriptionA, (LPVOID *)&pBufA, &uiLengthA ); |
| ok (retA, "VerQueryValueA failed: GetLastError = %u\n", GetLastError()); |
| ok( !lstrcmpA(WineFileDescriptionA, pBufA), "expected '%s' got '%s'\n", |
| WineFileDescriptionA, pBufA); |
| |
| /* Test a second time */ |
| retA = VerQueryValueA( pVersionInfoA, FileDescriptionA, (LPVOID *)&pBufA, &uiLengthA ); |
| ok (retA, "VerQueryValueA failed: GetLastError = %u\n", GetLastError()); |
| ok( !lstrcmpA(WineFileDescriptionA, pBufA), "expected '%s' got '%s'\n", |
| WineFileDescriptionA, pBufA); |
| |
| if (is_unicode_enabled) |
| { |
| retW = VerQueryValueW( pVersionInfoW, FileDescriptionW, (LPVOID *)&pBufW, &uiLengthW ); |
| ok (retW, "VerQueryValueW failed: GetLastError = %u\n", GetLastError()); |
| ok( !lstrcmpW(WineFileDescriptionW, pBufW), "FileDescription should have been '%s'\n", WineFileDescriptionA); |
| } |
| |
| HeapFree( GetProcessHeap(), 0, pVersionInfoA); |
| if (is_unicode_enabled) |
| HeapFree( GetProcessHeap(), 0, pVersionInfoW); |
| } |
| |
| static void test_VerQueryValueA(void) |
| { |
| static const char * const value_name[] = { |
| "Product", "CompanyName", "FileDescription", "Internal", |
| "ProductVersion", "InternalName", "File", "LegalCopyright", |
| "FileVersion", "Legal", "OriginalFilename", "ProductName", |
| "Company", "Original" }; |
| char *ver, *p; |
| UINT len, ret, translation, i; |
| char buf[MAX_PATH]; |
| |
| ret = GetModuleFileNameA(NULL, buf, sizeof(buf)); |
| assert(ret); |
| |
| SetLastError(0xdeadbeef); |
| len = GetFileVersionInfoSizeA(buf, NULL); |
| ok(len, "GetFileVersionInfoSizeA(%s) error %u\n", buf, GetLastError()); |
| |
| ver = HeapAlloc(GetProcessHeap(), 0, len); |
| assert(ver); |
| |
| SetLastError(0xdeadbeef); |
| ret = GetFileVersionInfoA(buf, 0, len, ver); |
| ok(ret, "GetFileVersionInfoA error %u\n", GetLastError()); |
| |
| p = (char *)0xdeadbeef; |
| len = 0xdeadbeef; |
| SetLastError(0xdeadbeef); |
| ret = VerQueryValueA(ver, "\\VarFileInfo\\Translation", (LPVOID*)&p, &len); |
| ok(ret, "VerQueryValue error %u\n", GetLastError()); |
| ok(len == 4, "VerQueryValue returned %u, expected 4\n", len); |
| |
| translation = *(UINT *)p; |
| translation = MAKELONG(HIWORD(translation), LOWORD(translation)); |
| |
| p = (char *)0xdeadbeef; |
| len = 0xdeadbeef; |
| SetLastError(0xdeadbeef); |
| ret = VerQueryValueA(ver, "String", (LPVOID*)&p, &len); |
| ok(!ret, "VerQueryValue should fail\n"); |
| ok(GetLastError() == ERROR_RESOURCE_TYPE_NOT_FOUND || |
| GetLastError() == 0xdeadbeef /* NT4, W2K */, |
| "VerQueryValue returned %u\n", GetLastError()); |
| ok(p == (char *)0xdeadbeef, "expected 0xdeadbeef got %p\n", p); |
| ok(len == 0, "expected 0 got %x\n", len); |
| |
| p = (char *)0xdeadbeef; |
| len = 0xdeadbeef; |
| SetLastError(0xdeadbeef); |
| ret = VerQueryValueA(ver, "StringFileInfo", (LPVOID*)&p, &len); |
| ok(ret, "VerQueryValue error %u\n", GetLastError()); |
| ok(len == 0, "VerQueryValue returned %u, expected 0\n", len); |
| ok(p != (char *)0xdeadbeef, "not expected 0xdeadbeef\n"); |
| |
| p = (char *)0xdeadbeef; |
| len = 0xdeadbeef; |
| SetLastError(0xdeadbeef); |
| ret = VerQueryValueA(ver, "\\StringFileInfo", (LPVOID*)&p, &len); |
| ok(ret, "VerQueryValue error %u\n", GetLastError()); |
| ok(len == 0, "VerQueryValue returned %u, expected 0\n", len); |
| ok(p != (char *)0xdeadbeef, "not expected 0xdeadbeef\n"); |
| |
| p = (char *)0xdeadbeef; |
| len = 0xdeadbeef; |
| SetLastError(0xdeadbeef); |
| ret = VerQueryValueA(ver, "\\\\StringFileInfo", (LPVOID*)&p, &len); |
| ok(ret, "VerQueryValue error %u\n", GetLastError()); |
| ok(len == 0, "VerQueryValue returned %u, expected 0\n", len); |
| ok(p != (char *)0xdeadbeef, "not expected 0xdeadbeef\n"); |
| |
| p = (char *)0xdeadbeef; |
| len = 0xdeadbeef; |
| SetLastError(0xdeadbeef); |
| ret = VerQueryValueA(ver, "\\StringFileInfo\\\\", (LPVOID*)&p, &len); |
| ok(ret, "VerQueryValue error %u\n", GetLastError()); |
| ok(len == 0, "VerQueryValue returned %u, expected 0\n", len); |
| ok(p != (char *)0xdeadbeef, "not expected 0xdeadbeef\n"); |
| |
| sprintf(buf, "\\StringFileInfo\\%08x", translation); |
| p = (char *)0xdeadbeef; |
| len = 0xdeadbeef; |
| SetLastError(0xdeadbeef); |
| ret = VerQueryValueA(ver, buf, (LPVOID*)&p, &len); |
| ok(ret, "VerQueryValue error %u\n", GetLastError()); |
| ok(len == 0, "VerQueryValue returned %u, expected 0\n", len); |
| ok(p != (char *)0xdeadbeef, "not expected 0xdeadbeef\n"); |
| |
| for (i = 0; i < sizeof(value_name)/sizeof(value_name[0]); i++) |
| { |
| sprintf(buf, "\\StringFileInfo\\%08x\\%s", translation, value_name[i]); |
| p = (char *)0xdeadbeef; |
| len = 0xdeadbeef; |
| SetLastError(0xdeadbeef); |
| ret = VerQueryValueA(ver, buf, (LPVOID*)&p, &len); |
| ok(ret, "VerQueryValueA(%s) error %u\n", buf, GetLastError()); |
| ok(len == strlen(value_name[i]) + 1, "VerQueryValue returned %u\n", len); |
| ok(!strcmp(value_name[i], p), "expected \"%s\", got \"%s\"\n", |
| value_name[i], p); |
| |
| /* test partial value names */ |
| len = lstrlenA(buf); |
| buf[len - 2] = 0; |
| p = (char *)0xdeadbeef; |
| len = 0xdeadbeef; |
| SetLastError(0xdeadbeef); |
| ret = VerQueryValueA(ver, buf, (LPVOID*)&p, &len); |
| ok(!ret, "VerQueryValueA(%s) succeeded\n", buf); |
| ok(GetLastError() == ERROR_RESOURCE_TYPE_NOT_FOUND || |
| GetLastError() == 0xdeadbeef /* NT4, W2K */, |
| "VerQueryValue returned %u\n", GetLastError()); |
| ok(p == (char *)0xdeadbeef, "expected 0xdeadbeef got %p\n", p); |
| ok(len == 0, "expected 0 or 0xbeef, got %x\n", len); |
| } |
| |
| HeapFree(GetProcessHeap(), 0, ver); |
| } |
| |
| static void test_extra_block(void) |
| { |
| WORD extra_block[] = { |
| 72, 0, 0, 'W', 'i', 'n', 'e', 'T', 'e', 's', 't', '\0', |
| 24, 4, 0, 'B', 'i', 'n', 'a', 'r', 'y', '\0', 0xbeef, 0xdead, |
| 24, 4, 1, 'T', 'e', 'x', 't', '\0', 'B', '-', ')', '\0', |
| }; |
| char buf[MAX_PATH]; |
| UINT len, ret; |
| ULONG w; |
| char *ver, *p; |
| WORD *length; |
| |
| ret = GetModuleFileNameA(NULL, buf, sizeof(buf)); |
| ok(ret, "GetModuleFileNameA failed\n"); |
| |
| len = GetFileVersionInfoSizeA(buf, NULL); |
| ok(len, "GetFileVersionInfoSizeA(%s) error %u\n", buf, GetLastError()); |
| |
| ver = HeapAlloc(GetProcessHeap(), 0, len + sizeof(extra_block) * 2); |
| ok(ver != NULL, "Can't allocate memory\n"); |
| |
| ret = GetFileVersionInfoA(buf, 0, len, ver); |
| ok(ret, "GetFileVersionInfoA error %u\n", GetLastError()); |
| |
| /* forge the string table, as windres dislike an extra block */ |
| length = (WORD *)ver; /* see VS_VERSION_INFO_STRUCT32 for details */ |
| memcpy(ver + *length, extra_block, sizeof(extra_block)); |
| *length += sizeof(extra_block); |
| |
| p = (char *)0xdeadbeef; |
| len = 0xdeadbeef; |
| w = 0xdeadbeef; |
| ret = VerQueryValueA(ver, "WineTest\\Binary", (LPVOID*)&p, &len); |
| ok(ret, "VerQueryValue error %u\n", GetLastError()); |
| ok(len == 4, "VerQueryValue returned %u, expected 4\n", len); |
| ok(p != (char *)0xdeadbeef, "not expected 0xdeadbeef\n"); |
| ok(memcmp(p, &w, sizeof(w)) == 0, "got 0x%08x, expected 0x%08x\n", *(PULONG)p, w); |
| |
| p = (char *)0xdeadbeef; |
| len = 0xdeadbeef; |
| ret = VerQueryValueA(ver, "WineTest\\Text", (LPVOID*)&p, &len); |
| ok(ret, "VerQueryValue error %u\n", GetLastError()); |
| ok(len == 4, "VerQueryValue returned %u, expected 4\n", len); |
| ok(p != (char *)0xdeadbeef, "not expected 0xdeadbeef\n"); |
| ok(strcmp(p, "B-)") == 0, "got '%s', expected '%s'\n", p, "B-)"); |
| |
| HeapFree(GetProcessHeap(), 0, ver); |
| } |
| |
| static void test_GetFileVersionInfoEx(void) |
| { |
| char *ver, *p; |
| BOOL ret; |
| UINT size, translation, i; |
| HMODULE mod; |
| BOOL (WINAPI *pGetFileVersionInfoExW)(DWORD, LPCWSTR, DWORD, DWORD, LPVOID); |
| DWORD (WINAPI *pGetFileVersionInfoSizeExW)(DWORD, LPCWSTR, LPDWORD); |
| const LANGID lang = GetUserDefaultUILanguage(); |
| const LANGID english = MAKELANGID(LANG_ENGLISH, SUBLANG_DEFAULT); |
| const WORD unicode = 1200; /* = UNICODE */ |
| const WCHAR kernel32W[] = {'k','e','r','n','e','l','3','2','.','d','l','l',0}; |
| const DWORD test_flags[] = { |
| 0, FILE_VER_GET_LOCALISED, FILE_VER_GET_NEUTRAL, |
| FILE_VER_GET_LOCALISED | FILE_VER_GET_NEUTRAL, |
| 0xdeadbeef, /* invalid value (ignored) */ |
| }; |
| char desc[MAX_PATH]; |
| |
| mod = GetModuleHandleA("kernel32.dll"); |
| assert(mod); |
| |
| if (!FindResourceExA(mod, (LPCSTR)RT_VERSION, (LPCSTR)VS_VERSION_INFO, lang) && |
| !FindResourceExA(mod, (LPCSTR)RT_VERSION, (LPCSTR)VS_VERSION_INFO, |
| MAKELANGID(PRIMARYLANGID(lang),SUBLANG_NEUTRAL))) |
| { |
| skip("Translation is not available\n"); |
| return; |
| } |
| |
| size = GetFileVersionInfoSizeW(kernel32W, NULL); |
| ok(size, "GetFileVersionInfoSize(kernel32) error %u\n", GetLastError()); |
| |
| ver = HeapAlloc(GetProcessHeap(), 0, size); |
| assert(ver); |
| |
| ret = GetFileVersionInfoW(kernel32W, 0, size, ver); |
| ok(ret, "GetFileVersionInfo error %u\n", GetLastError()); |
| |
| ret = VerQueryValueA(ver, "\\VarFileInfo\\Translation", (void **)&p, &size); |
| translation = *(UINT *)p; |
| ok(ret, "VerQueryValue error %u\n", GetLastError()); |
| ok(size == 4, "VerQueryValue returned %u, expected 4\n", size); |
| |
| /* test default version resource */ |
| ok(LOWORD(translation) == lang, "got %u, expected lang is %u\n", |
| LOWORD(translation), lang); |
| ok(HIWORD(translation) == unicode, "got %u, expected codepage is %u\n", |
| HIWORD(translation), unicode); |
| |
| HeapFree(GetProcessHeap(), 0, ver); |
| |
| mod = GetModuleHandleA("version.dll"); |
| assert(mod); |
| |
| /* prefer W-version as A-version is not available on Windows 7 */ |
| pGetFileVersionInfoExW = (void *)GetProcAddress(mod, "GetFileVersionInfoExW"); |
| pGetFileVersionInfoSizeExW = (void *)GetProcAddress(mod, "GetFileVersionInfoSizeExW"); |
| if (!pGetFileVersionInfoExW && !pGetFileVersionInfoSizeExW) |
| { |
| win_skip("GetFileVersionInfoEx family is not available\n"); |
| return; |
| } |
| |
| for (i = 0; i < sizeof(test_flags)/sizeof(test_flags[0]); i++) |
| { |
| size = pGetFileVersionInfoSizeExW(test_flags[i], kernel32W, NULL); |
| ok(size, "[%u] GetFileVersionInfoSizeEx(kernel32) error %u\n", i, GetLastError()); |
| |
| ver = HeapAlloc(GetProcessHeap(), 0, size); |
| assert(ver); |
| |
| ret = pGetFileVersionInfoExW(test_flags[i], kernel32W, 0, size, ver); |
| ok(ret, "[%u] GetFileVersionInfoEx error %u\n", i, GetLastError()); |
| |
| ret = VerQueryValueA(ver, "\\VarFileInfo\\Translation", (void **)&p, &size); |
| ok(ret, "[%u] VerQueryValue error %u\n", i, GetLastError()); |
| ok(size == 4, "[%u] VerQueryValue returned %u, expected 4\n", i, size); |
| translation = *(UINT *)p; |
| |
| /* test MUI version resource */ |
| if (test_flags[i] & FILE_VER_GET_LOCALISED) |
| ok(LOWORD(translation) == lang, "[%u] got %u, expected lang is %u\n", |
| i, LOWORD(translation), lang); |
| else |
| ok(LOWORD(translation) == english, "[%u] got %u, expected lang is %u\n", |
| i, LOWORD(translation), english); |
| ok(HIWORD(translation) == unicode, "[%u] got %u, expected codepage is %u\n", |
| i, HIWORD(translation), unicode); |
| |
| /* test string info using translation info */ |
| size = 0; |
| sprintf(desc, "\\StringFileInfo\\%04x%04x\\FileDescription", |
| LOWORD(translation), HIWORD(translation)); |
| ret = VerQueryValueA(ver, desc, (void **)&p, &size); |
| ok(ret, "[%u] VerQueryValue error %u\n", i, GetLastError()); |
| ok(size == strlen(p) + 1, "[%u] VerQueryValue returned %u\n", i, size); |
| |
| HeapFree(GetProcessHeap(), 0, ver); |
| } |
| |
| return; |
| } |
| |
| START_TEST(info) |
| { |
| test_info_size(); |
| test_info(); |
| test_32bit_win(); |
| test_VerQueryValueA(); |
| test_extra_block(); |
| test_GetFileVersionInfoEx(); |
| } |