wine.inf: We should not override existing associations.
[wine/hacks.git] / dlls / version / tests / info.c
blobb857b1d074862f8e548a200108c13d02084a6b85
1 /*
2 * Copyright (C) 2004 Stefan Leichter
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Lesser General Public
6 * License as published by the Free Software Foundation; either
7 * version 2.1 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Lesser General Public License for more details.
14 * You should have received a copy of the GNU Lesser General Public
15 * License along with this library; if not, write to the Free Software
16 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
19 #include <stdarg.h>
20 #include <stdio.h>
22 #include "wine/test.h"
23 #include "windef.h"
24 #include "winbase.h"
25 #include "winerror.h"
26 #include "winver.h"
28 #define MY_LAST_ERROR ((DWORD)-1)
29 #define EXPECT_BAD_PATH__NOT_FOUND \
30 ok( (ERROR_PATH_NOT_FOUND == GetLastError()) || \
31 (ERROR_RESOURCE_DATA_NOT_FOUND == GetLastError()) || \
32 (ERROR_FILE_NOT_FOUND == GetLastError()) || \
33 (ERROR_BAD_PATHNAME == GetLastError()), \
34 "Last error wrong! ERROR_RESOURCE_DATA_NOT_FOUND/ERROR_BAD_PATHNAME (98)/" \
35 "ERROR_PATH_NOT_FOUND (NT4)/ERROR_FILE_NOT_FOUND (2k3)" \
36 "expected, got 0x%08lx\n", GetLastError());
37 #define EXPECT_INVALID__NOT_FOUND \
38 ok( (ERROR_PATH_NOT_FOUND == GetLastError()) || \
39 (ERROR_RESOURCE_DATA_NOT_FOUND == GetLastError()) || \
40 (ERROR_FILE_NOT_FOUND == GetLastError()) || \
41 (ERROR_INVALID_PARAMETER == GetLastError()), \
42 "Last error wrong! ERROR_RESOURCE_DATA_NOT_FOUND/ERROR_INVALID_PARAMETER (98)/" \
43 "ERROR_PATH_NOT_FOUND (NT4)/ERROR_FILE_NOT_FOUND (2k3)" \
44 "expected, got 0x%08lx\n", GetLastError());
46 static void test_info_size(void)
47 { DWORD hdl, retval;
48 char mypath[MAX_PATH] = "";
50 SetLastError(MY_LAST_ERROR);
51 retval = GetFileVersionInfoSizeA( NULL, NULL);
52 ok( !retval,
53 "GetFileVersionInfoSizeA result wrong! 0L expected, got 0x%08lx\n",
54 retval);
55 EXPECT_INVALID__NOT_FOUND;
57 hdl = 0x55555555;
58 SetLastError(MY_LAST_ERROR);
59 retval = GetFileVersionInfoSizeA( NULL, &hdl);
60 ok( !retval,
61 "GetFileVersionInfoSizeA result wrong! 0L expected, got 0x%08lx\n",
62 retval);
63 EXPECT_INVALID__NOT_FOUND;
64 ok( hdl == 0L,
65 "Handle wrong! 0L expected, got 0x%08lx\n", hdl);
67 SetLastError(MY_LAST_ERROR);
68 retval = GetFileVersionInfoSizeA( "", NULL);
69 ok( !retval,
70 "GetFileVersionInfoSizeA result wrong! 0L expected, got 0x%08lx\n",
71 retval);
72 EXPECT_BAD_PATH__NOT_FOUND;
74 hdl = 0x55555555;
75 SetLastError(MY_LAST_ERROR);
76 retval = GetFileVersionInfoSizeA( "", &hdl);
77 ok( !retval,
78 "GetFileVersionInfoSizeA result wrong! 0L expected, got 0x%08lx\n",
79 retval);
80 EXPECT_BAD_PATH__NOT_FOUND;
81 ok( hdl == 0L,
82 "Handle wrong! 0L expected, got 0x%08lx\n", hdl);
84 SetLastError(MY_LAST_ERROR);
85 retval = GetFileVersionInfoSizeA( "kernel32.dll", NULL);
86 ok( retval,
87 "GetFileVersionInfoSizeA result wrong! <> 0L expected, got 0x%08lx\n",
88 retval);
89 ok((NO_ERROR == GetLastError()) || (MY_LAST_ERROR == GetLastError()),
90 "Last error wrong! NO_ERROR/0x%08lx (NT4) expected, got 0x%08lx\n",
91 MY_LAST_ERROR, GetLastError());
93 hdl = 0x55555555;
94 SetLastError(MY_LAST_ERROR);
95 retval = GetFileVersionInfoSizeA( "kernel32.dll", &hdl);
96 ok( retval,
97 "GetFileVersionInfoSizeA result wrong! <> 0L expected, got 0x%08lx\n",
98 retval);
99 ok((NO_ERROR == GetLastError()) || (MY_LAST_ERROR == GetLastError()),
100 "Last error wrong! NO_ERROR/0x%08lx (NT4) expected, got 0x%08lx\n",
101 MY_LAST_ERROR, GetLastError());
102 ok( hdl == 0L,
103 "Handle wrong! 0L expected, got 0x%08lx\n", hdl);
105 SetLastError(MY_LAST_ERROR);
106 retval = GetFileVersionInfoSizeA( "notexist.dll", NULL);
107 ok( !retval,
108 "GetFileVersionInfoSizeA result wrong! 0L expected, got 0x%08lx\n",
109 retval);
110 ok( (ERROR_FILE_NOT_FOUND == GetLastError()) ||
111 (ERROR_RESOURCE_DATA_NOT_FOUND == GetLastError()) ||
112 (MY_LAST_ERROR == GetLastError()),
113 "Last error wrong! ERROR_FILE_NOT_FOUND/ERROR_RESOURCE_DATA_NOT_FOUND "
114 "(XP)/0x%08lx (NT4) expected, got 0x%08lx\n", MY_LAST_ERROR, GetLastError());
116 /* test a currently loaded executable */
117 if(GetModuleFileNameA(NULL, mypath, MAX_PATH)) {
118 hdl = 0x55555555;
119 SetLastError(MY_LAST_ERROR);
120 retval = GetFileVersionInfoSizeA( mypath, &hdl);
121 ok( retval,
122 "GetFileVersionInfoSizeA result wrong! <> 0L expected, got 0x%08lx\n",
123 retval);
124 ok((NO_ERROR == GetLastError()) || (MY_LAST_ERROR == GetLastError()),
125 "Last error wrong! NO_ERROR/0x%08lx (NT4) expected, got 0x%08lx\n",
126 MY_LAST_ERROR, GetLastError());
127 ok( hdl == 0L,
128 "Handle wrong! 0L expected, got 0x%08lx\n", hdl);
130 else
131 trace("skipping GetModuleFileNameA(NULL,..) failed\n");
133 /* test a not loaded executable */
134 if(GetSystemDirectoryA(mypath, MAX_PATH)) {
135 lstrcatA(mypath, "\\regsvr32.exe");
137 if(INVALID_FILE_ATTRIBUTES == GetFileAttributesA(mypath))
138 trace("GetFileAttributesA(%s) failed\n", mypath);
139 else {
140 hdl = 0x55555555;
141 SetLastError(MY_LAST_ERROR);
142 retval = GetFileVersionInfoSizeA( mypath, &hdl);
143 ok( retval,
144 "GetFileVersionInfoSizeA result wrong! <> 0L expected, got 0x%08lx\n",
145 retval);
146 ok((NO_ERROR == GetLastError()) || (MY_LAST_ERROR == GetLastError()),
147 "Last error wrong! NO_ERROR/0x%08lx (NT4) expected, got 0x%08lx\n",
148 MY_LAST_ERROR, GetLastError());
149 ok( hdl == 0L,
150 "Handle wrong! 0L expected, got 0x%08lx\n", hdl);
153 else
154 trace("skipping GetModuleFileNameA(NULL,..) failed\n");
157 static void VersionDwordLong2String(DWORDLONG Version, LPSTR lpszVerString)
159 WORD a, b, c, d;
161 a = (WORD)(Version >> 48);
162 b = (WORD)((Version >> 32) & 0xffff);
163 c = (WORD)((Version >> 16) & 0xffff);
164 d = (WORD)(Version & 0xffff);
166 sprintf(lpszVerString, "%d.%d.%d.%d", a, b, c, d);
168 return;
171 static void test_info(void)
173 DWORD hdl, retval;
174 PVOID pVersionInfo = NULL;
175 BOOL boolret;
176 VS_FIXEDFILEINFO *pFixedVersionInfo;
177 UINT uiLength;
178 char VersionString[MAX_PATH];
179 DWORDLONG dwlVersion;
181 hdl = 0x55555555;
182 SetLastError(MY_LAST_ERROR);
183 retval = GetFileVersionInfoSizeA( "kernel32.dll", &hdl);
184 ok( retval,
185 "GetFileVersionInfoSizeA result wrong! <> 0L expected, got 0x%08lx\n",
186 retval);
187 ok((NO_ERROR == GetLastError()) || (MY_LAST_ERROR == GetLastError()),
188 "Last error wrong! NO_ERROR/0x%08lx (NT4) expected, got 0x%08lx\n",
189 MY_LAST_ERROR, GetLastError());
190 ok( hdl == 0L,
191 "Handle wrong! 0L expected, got 0x%08lx\n", hdl);
193 if ( retval == 0 || hdl != 0)
194 return;
196 pVersionInfo = HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, retval );
197 ok(pVersionInfo != 0, "HeapAlloc failed\n" );
198 if (pVersionInfo == 0)
199 return;
201 #if 0
202 /* this test crashes on WinNT4
204 boolret = GetFileVersionInfoA( "kernel32.dll", 0, retval, 0);
205 ok (!boolret, "GetFileVersionInfoA should have failed: GetLastError = 0x%08lx\n", GetLastError());
206 ok ((GetLastError() == ERROR_INVALID_DATA) || (GetLastError() == ERROR_BAD_PATHNAME) ||
207 (GetLastError() == NO_ERROR),
208 "Last error wrong! ERROR_INVALID_DATA/ERROR_BAD_PATHNAME (ME)/"
209 "NO_ERROR (95) expected, got 0x%08lx\n",
210 GetLastError());
211 #endif
213 boolret = GetFileVersionInfoA( "kernel32.dll", 0, retval, pVersionInfo );
214 ok (boolret, "GetFileVersionInfoA failed: GetLastError = 0x%08lx\n", GetLastError());
215 if (!boolret)
216 return;
218 boolret = VerQueryValueA( pVersionInfo, "\\", (LPVOID *)&pFixedVersionInfo, &uiLength );
219 ok (boolret, "VerQueryValueA failed: GetLastError = 0x%08lx\n", GetLastError());
220 if (!boolret)
221 return;
223 dwlVersion = (((DWORDLONG)pFixedVersionInfo->dwFileVersionMS) << 32) +
224 pFixedVersionInfo->dwFileVersionLS;
226 VersionDwordLong2String(dwlVersion, VersionString);
228 trace("kernel32.dll version: %s\n", VersionString);
230 #if 0
231 /* this test crashes on WinNT4
233 boolret = VerQueryValueA( pVersionInfo, "\\", (LPVOID *)&pFixedVersionInfo, 0);
234 ok (boolret, "VerQueryValue failed: GetLastError = 0x%08lx\n", GetLastError());
235 #endif
238 static void test_32bit_win(void)
240 DWORD hdlA, retvalA;
241 DWORD hdlW, retvalW = 0;
242 BOOL retA,retW;
243 PVOID pVersionInfoA = NULL;
244 PVOID pVersionInfoW = NULL;
245 char *pBufA;
246 WCHAR *pBufW;
247 UINT uiLengthA, uiLengthW;
248 char mypathA[MAX_PATH];
249 WCHAR mypathW[MAX_PATH];
250 char rootA[] = "\\";
251 WCHAR rootW[] = { '\\', 0 };
252 char varfileinfoA[] = "\\VarFileInfo\\Translation";
253 WCHAR varfileinfoW[] = { '\\','V','a','r','F','i','l','e','I','n','f','o',
254 '\\','T','r','a','n','s','l','a','t','i','o','n', 0 };
255 char WineVarFileInfoA[] = { 0x09, 0x04, 0xE4, 0x04 };
256 char FileDescriptionA[] = "\\StringFileInfo\\040904E4\\FileDescription";
257 WCHAR FileDescriptionW[] = { '\\','S','t','r','i','n','g','F','i','l','e','I','n','f','o',
258 '\\','0','4','0','9','0','4','E','4',
259 '\\','F','i','l','e','D','e','s','c','r','i','p','t','i','o','n', 0 };
260 char WineFileDescriptionA[] = "Wine version test";
261 WCHAR WineFileDescriptionW[] = { 'W','i','n','e',' ','v','e','r','s','i','o','n',' ','t','e','s','t', 0 };
262 BOOL is_unicode_enabled = TRUE;
264 /* A copy from dlls/version/info.c */
265 typedef struct
267 WORD wLength;
268 WORD wValueLength;
269 WORD wType;
270 WCHAR szKey[1];
271 #if 0 /* variable length structure */
272 /* DWORD aligned */
273 BYTE Value[];
274 /* DWORD aligned */
275 VS_VERSION_INFO_STRUCT32 Children[];
276 #endif
277 } VS_VERSION_INFO_STRUCT32;
279 /* If we call GetFileVersionInfoA on a system that supports Unicode, NT/W2K/XP/W2K3 (by default) and Wine,
280 * the versioninfo will contain Unicode strings.
281 * Part of the test is to call both the A and W versions, which should have the same Version Information
282 * for some requests, on systems that support both calls.
285 /* First get the versioninfo via the W versions */
286 GetModuleFileNameW(NULL, mypathW, MAX_PATH);
287 if (GetLastError() == ERROR_CALL_NOT_IMPLEMENTED)
289 trace("GetModuleFileNameW not existing on this platform, skipping comparison between A- and W-calls\n");
290 is_unicode_enabled = FALSE;
293 if (is_unicode_enabled)
295 retvalW = GetFileVersionInfoSizeW( mypathW, &hdlW);
296 pVersionInfoW = HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, retvalW );
297 retW = GetFileVersionInfoW( mypathW, 0, retvalW, pVersionInfoW );
300 GetModuleFileNameA(NULL, mypathA, MAX_PATH);
301 retvalA = GetFileVersionInfoSizeA( mypathA, &hdlA);
302 pVersionInfoA = HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, retvalA );
303 retA = GetFileVersionInfoA( mypathA, 0, retvalA, pVersionInfoA );
305 if (is_unicode_enabled)
307 ok( retvalA == retvalW, "The size of the struct should be the same for both A/W calls, it is (%ld) vs. (%ld)\n",
308 retvalA, retvalW);
309 ok( !memcmp(pVersionInfoA, pVersionInfoW, retvalA), "Both structs should be the same, they aren't\n");
312 /* The structs on Windows are bigger then just the struct for the basic information. The total struct
313 * contains also an empty part, which is used for converted strings. The converted strings are a result
314 * of calling VerQueryValueA on a 32bit resource and calling VerQueryValueW on a 16bit resource.
315 * The first WORD of the structure (wLength) shows the size of the base struct. The total struct size depends
316 * on the Windows version:
318 * 16bits resource (numbers are from a sample app):
320 * Windows Version Retrieved with A/W wLength StructSize
321 * ====================================================================================
322 * Win98 A 0x01B4 (436) 436
323 * NT4 A/W 0x01B4 (436) 2048 ???
324 * W2K/XP/W2K3 A/W 0x01B4 (436) 1536 which is (436 - sizeof(VS_FIXEDFILEINFO)) * 4
326 * 32bits resource (numbers are from this test executable version_crosstest.exe):
327 * Windows Version Retrieved with A/W wLength StructSize
328 * =============================================================
329 * Win98 A 0x01E0 (480) 848 (structure data doesn't seem correct)
330 * NT4 A/W 0x0350 (848) 1272 (848 * 1.5)
331 * W2K/XP/W2K3 A/W 0x0350 (848) 1700 which is (848 * 2) + 4
333 * Wine will follow the implementation (eventually) of W2K/XP/W2K3
336 /* Now some tests for the above (only if we are unicode enabled) */
338 if (is_unicode_enabled)
340 VS_VERSION_INFO_STRUCT32 *vvis = (VS_VERSION_INFO_STRUCT32 *)pVersionInfoW;
341 ok ( retvalW == ((vvis->wLength * 2) + 4) || retvalW == (vvis->wLength * 1.5),
342 "Structure is not of the correct size\n");
345 /* Although the 32bit resource structures contain Unicode strings, VerQueryValueA will always return normal strings,
346 * VerQueryValueW will always return Unicode ones. (That means everything returned for StringFileInfo requests).
349 /* Get the VS_FIXEDFILEINFO information, this must be the same for both A- and W-Calls */
351 retA = VerQueryValueA( pVersionInfoA, rootA, (LPVOID *)&pBufA, &uiLengthA );
352 ok (retA, "VerQueryValueA failed: GetLastError = 0x%08lx\n", GetLastError());
353 ok ( uiLengthA == sizeof(VS_FIXEDFILEINFO), "Size (%d) doesn't match the size of the VS_FIXEDFILEINFO struct\n", uiLengthA);
355 if (is_unicode_enabled)
357 retW = VerQueryValueW( pVersionInfoW, rootW, (LPVOID *)&pBufW, &uiLengthW );
358 ok (retW, "VerQueryValueW failed: GetLastError = 0x%08lx\n", GetLastError());
359 ok ( uiLengthA == sizeof(VS_FIXEDFILEINFO), "Size (%d) doesn't match the size of the VS_FIXEDFILEINFO struct\n", uiLengthA);
361 ok( uiLengthA == uiLengthW, "The size of VS_FIXEDFILEINFO should be the same for both A/W calls, it is (%d) vs. (%d)\n",
362 uiLengthA, uiLengthW);
363 ok( !memcmp(pBufA, pBufW, uiLengthA), "Both values should be the same, they aren't\n");
366 /* Get some VarFileInfo information, this must be the same for both A- and W-Calls */
368 retA = VerQueryValueA( pVersionInfoA, varfileinfoA, (LPVOID *)&pBufA, &uiLengthA );
369 ok (retA, "VerQueryValueA failed: GetLastError = 0x%08lx\n", GetLastError());
370 ok( !memcmp(pBufA, WineVarFileInfoA, uiLengthA), "The VarFileInfo should have matched 0904e404 (non case sensitive)\n");
372 if (is_unicode_enabled)
374 retW = VerQueryValueW( pVersionInfoW, varfileinfoW, (LPVOID *)&pBufW, &uiLengthW );
375 ok (retW, "VerQueryValueW failed: GetLastError = 0x%08lx\n", GetLastError());
376 ok( uiLengthA == uiLengthW, "The size of the VarFileInfo information should be the same for both A/W calls, it is (%d) vs. (%d)\n",
377 uiLengthA, uiLengthW);
378 ok( !memcmp(pBufA, pBufW, uiLengthA), "Both values should be the same, they aren't\n");
381 /* Get some StringFileInfo information, this will be ANSI for A-Calls and Unicode for W-Calls */
383 retA = VerQueryValueA( pVersionInfoA, FileDescriptionA, (LPVOID *)&pBufA, &uiLengthA );
384 ok (retA, "VerQueryValueA failed: GetLastError = 0x%08lx\n", GetLastError());
385 ok( !lstrcmpA(WineFileDescriptionA, pBufA), "FileDescription should have been 'Wine version test'\n");
387 /* Test a second time */
388 retA = VerQueryValueA( pVersionInfoA, FileDescriptionA, (LPVOID *)&pBufA, &uiLengthA );
389 ok (retA, "VerQueryValueA failed: GetLastError = 0x%08lx\n", GetLastError());
390 ok( !lstrcmpA(WineFileDescriptionA, pBufA), "FileDescription should have been 'Wine version test'\n");
392 if (is_unicode_enabled)
394 retW = VerQueryValueW( pVersionInfoW, FileDescriptionW, (LPVOID *)&pBufW, &uiLengthW );
395 ok (retW, "VerQueryValueW failed: GetLastError = 0x%08lx\n", GetLastError());
396 ok( !lstrcmpW(WineFileDescriptionW, pBufW), "FileDescription should have been 'Wine version test' (unicode)\n");
399 HeapFree( GetProcessHeap(), 0, pVersionInfoA);
400 if (is_unicode_enabled)
401 HeapFree( GetProcessHeap(), 0, pVersionInfoW);
404 START_TEST(info)
406 test_info_size();
407 test_info();
409 /* Test several AW-calls on a 32 bit windows executable */
410 trace("Testing 32 bit windows application\n");
411 test_32bit_win();