kernel32/tests: Comment out an UpdateResource call that causes failures.
[wine] / dlls / setupapi / tests / query.c
1 /*
2  * Unit tests for setupapi.dll query functions
3  *
4  * Copyright (C) 2006 James Hawkins
5  *
6  * This library is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU Lesser General Public
8  * License as published by the Free Software Foundation; either
9  * version 2.1 of the License, or (at your option) any later version.
10  *
11  * This library is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * Lesser General Public License for more details.
15  *
16  * You should have received a copy of the GNU Lesser General Public
17  * License along with this library; if not, write to the Free Software
18  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
19  */
20
21 #include <stdio.h>
22 #include <windows.h>
23 #include <setupapi.h>
24 #include "wine/test.h"
25
26 CHAR CURR_DIR[MAX_PATH];
27 CHAR WIN_DIR[MAX_PATH];
28
29 static void get_directories(void)
30 {
31     int len;
32
33     GetCurrentDirectoryA(MAX_PATH, CURR_DIR);
34     len = lstrlenA(CURR_DIR);
35
36     if(len && (CURR_DIR[len-1] == '\\'))
37         CURR_DIR[len-1] = 0;
38
39     GetWindowsDirectoryA(WIN_DIR, MAX_PATH);
40     len = lstrlenA(WIN_DIR);
41
42     if (len && (WIN_DIR[len-1] == '\\'))
43         WIN_DIR[len-1] = 0;
44 }
45
46 static void append_str(char **str, const char *data)
47 {
48     sprintf(*str, data);
49     *str += strlen(*str);
50 }
51
52 static void create_inf_file(LPSTR filename)
53 {
54     char data[1024];
55     char *ptr = data;
56     DWORD dwNumberOfBytesWritten;
57     HANDLE hf = CreateFile(filename, GENERIC_WRITE, 0, NULL,
58                            CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
59
60     append_str(&ptr, "[Version]\n");
61     append_str(&ptr, "Signature=\"$Chicago$\"\n");
62     append_str(&ptr, "AdvancedINF=2.5\n");
63     append_str(&ptr, "[SourceDisksNames]\n");
64     append_str(&ptr, "2 = %%SrcDiskName%%, LANCOM\\LANtools\\lanconf.cab\n");
65     append_str(&ptr, "[SourceDisksFiles]\n");
66     append_str(&ptr, "lanconf.exe = 2\n");
67     append_str(&ptr, "[DestinationDirs]\n");
68     append_str(&ptr, "DefaultDestDir = 24, %%DefaultDest%%\n");
69     append_str(&ptr, "[Strings]\n");
70     append_str(&ptr, "LangDir = english\n");
71     append_str(&ptr, "DefaultDest = LANCOM\n");
72     append_str(&ptr, "SrcDiskName = \"LANCOM Software CD\"\n");
73
74     WriteFile(hf, data, ptr - data, &dwNumberOfBytesWritten, NULL);
75     CloseHandle(hf);
76 }
77
78 static void create_inf_file2(LPSTR filename)
79 {
80     char data[1024];
81     char *ptr = data;
82     DWORD dwNumberOfBytesWritten;
83     HANDLE hf = CreateFile(filename, GENERIC_WRITE, 0, NULL,
84                            CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
85
86     append_str(&ptr, "[SourceFileInfo]\n");
87     append_str(&ptr, "sp1qfe\\bitsinst.exe=250B3702C7CCD7C2F9E4DAA1555C933E,000600060A28062C,27136,SP1QFE\n");
88     append_str(&ptr, "sp1qfe\\bitsprx2.dll=4EBEA67F4BB4EB402E725CA7CA2857AE,000600060A280621,7680,SP1QFE\n");
89     append_str(&ptr, "sp1qfe\\bitsprx3.dll=C788A1D9330DA011EF25E95D3BC7BDE5,000600060A280621,7168,SP1QFE\n");
90     append_str(&ptr, "sp1qfe\\qmgr.dll=696AC82FB290A03F205901442E0E9589,000600060A280621,361984,SP1QFE\n");
91     append_str(&ptr, "sp1qfe\\qmgrprxy.dll=8B5848144829E1BC985EA4C3D8CA7E3F,000600060A280621,17408,SP1QFE\n");
92     append_str(&ptr, "sp1qfe\\winhttp.dll=3EC6F518114606CA59D4160322077437,000500010A280615,331776,SP1QFE\n");
93     append_str(&ptr, "sp1qfe\\xpob2res.dll=DB83156B9F496F20D1EA70E4ABEC0166,000500010A280622,158720,SP1QFE\n");
94
95     WriteFile(hf, data, ptr - data, &dwNumberOfBytesWritten, NULL);
96     CloseHandle(hf);
97 }
98
99 static BOOL check_info_filename(PSP_INF_INFORMATION info, LPSTR test)
100 {
101     LPSTR filename;
102     DWORD size;
103     BOOL ret = FALSE;
104
105     if (!SetupQueryInfFileInformationA(info, 0, NULL, 0, &size))
106         return FALSE;
107
108     filename = HeapAlloc(GetProcessHeap(), 0, size);
109     if (!filename)
110         return FALSE;
111
112     SetupQueryInfFileInformationA(info, 0, filename, size, &size);
113
114     if (!lstrcmpiA(test, filename))
115         ret = TRUE;
116
117     HeapFree(GetProcessHeap(), 0, filename);
118     return ret;
119 }
120
121 static PSP_INF_INFORMATION alloc_inf_info(LPCSTR filename, DWORD search, PDWORD size)
122 {
123     PSP_INF_INFORMATION info;
124     BOOL ret;
125
126     ret = SetupGetInfInformationA(filename, search, NULL, 0, size);
127     if (!ret)
128         return NULL;
129
130     info = HeapAlloc(GetProcessHeap(), 0, *size);
131     return info;
132 }
133
134 static void test_SetupGetInfInformation(void)
135 {
136     PSP_INF_INFORMATION info;
137     CHAR inf_filename[MAX_PATH];
138     CHAR inf_one[MAX_PATH], inf_two[MAX_PATH];
139     LPSTR revfile;
140     DWORD size;
141     HINF hinf;
142     BOOL ret;
143
144     lstrcpyA(inf_filename, CURR_DIR);
145     lstrcatA(inf_filename, "\\");
146     lstrcatA(inf_filename, "test.inf");
147
148     /* try an invalid inf handle */
149     size = 0xdeadbeef;
150     SetLastError(0xbeefcafe);
151     ret = SetupGetInfInformationA(NULL, INFINFO_INF_SPEC_IS_HINF, NULL, 0, &size);
152     ok(ret == FALSE, "Expected SetupGetInfInformation to fail\n");
153     ok(GetLastError() == ERROR_INVALID_HANDLE ||
154        broken(GetLastError() == ERROR_BAD_PATHNAME) || /* win95 */
155        broken(GetLastError() == ERROR_FILE_NOT_FOUND) || /* win98 */
156        broken(GetLastError() == ERROR_PATH_NOT_FOUND) || /* NT4 */
157        broken(GetLastError() == ERROR_INVALID_NAME) || /* win2k */
158        broken(GetLastError() == ERROR_GENERAL_SYNTAX), /* another win2k */
159        "Expected ERROR_INVALID_HANDLE, got %d\n", GetLastError());
160     ok(size == 0xdeadbeef, "Expected size to remain unchanged\n");
161
162     /* try an invalid inf filename */
163     /* do not use NULL as absolute inf filename on win9x (crash) */
164     if ((GetLastError() != ERROR_BAD_PATHNAME) &&   /* win95 */
165         (GetLastError() != ERROR_FILE_NOT_FOUND))  /* win98 */
166     {
167         size = 0xdeadbeef;
168         SetLastError(0xbeefcafe);
169         ret = SetupGetInfInformationA(NULL, INFINFO_INF_NAME_IS_ABSOLUTE, NULL, 0, &size);
170         ok(ret == FALSE, "Expected SetupGetInfInformation to fail\n");
171         ok(GetLastError() == ERROR_INVALID_PARAMETER,
172            "Expected ERROR_INVALID_PARAMETER, got %d\n", GetLastError());
173         ok(size == 0xdeadbeef, "Expected size to remain unchanged\n");
174     }
175
176     create_inf_file(inf_filename);
177
178     /* try an invalid search flag */
179     size = 0xdeadbeef;
180     SetLastError(0xbeefcafe);
181     ret = SetupGetInfInformationA(inf_filename, -1, NULL, 0, &size);
182     ok(ret == FALSE, "Expected SetupGetInfInformation to fail\n");
183     ok(GetLastError() == ERROR_INVALID_PARAMETER,
184        "Expected ERROR_INVALID_PARAMETER, got %d\n", GetLastError());
185     ok(size == 0xdeadbeef, "Expected size to remain unchanged\n");
186
187     /* try a nonexistent inf file */
188     size = 0xdeadbeef;
189     SetLastError(0xbeefcafe);
190     ret = SetupGetInfInformationA("idontexist", INFINFO_INF_NAME_IS_ABSOLUTE, NULL, 0, &size);
191     ok(ret == FALSE, "Expected SetupGetInfInformation to fail\n");
192     ok(GetLastError() == ERROR_FILE_NOT_FOUND,
193        "Expected ERROR_FILE_NOT_FOUND, got %d\n", GetLastError());
194     ok(size == 0xdeadbeef, "Expected size to remain unchanged\n");
195
196     /* successfully open the inf file */
197     size = 0xdeadbeef;
198     ret = SetupGetInfInformationA(inf_filename, INFINFO_INF_NAME_IS_ABSOLUTE, NULL, 0, &size);
199     ok(ret == TRUE, "Expected SetupGetInfInformation to succeed\n");
200     ok(size != 0xdeadbeef, "Expected a valid size on return\n");
201
202     /* set ReturnBuffer to NULL and ReturnBufferSize to non-zero value 'size' */
203     SetLastError(0xbeefcafe);
204     ret = SetupGetInfInformationA(inf_filename, INFINFO_INF_NAME_IS_ABSOLUTE, NULL, size, &size);
205     ok(ret == FALSE, "Expected SetupGetInfInformation to fail\n");
206     ok(GetLastError() == ERROR_INVALID_PARAMETER,
207        "Expected ERROR_INVALID_PARAMETER, got %d\n", GetLastError());
208
209     /* set ReturnBuffer to NULL and ReturnBufferSize to non-zero value 'size-1' */
210     ret = SetupGetInfInformationA(inf_filename, INFINFO_INF_NAME_IS_ABSOLUTE, NULL, size-1, &size);
211     ok(ret == TRUE, "Expected SetupGetInfInformation to succeed\n");
212
213     /* some tests for behaviour with a NULL RequiredSize pointer */
214     ret = SetupGetInfInformationA(inf_filename, INFINFO_INF_NAME_IS_ABSOLUTE, NULL, 0, NULL);
215     ok(ret == TRUE, "Expected SetupGetInfInformation to succeed\n");
216     ret = SetupGetInfInformationA(inf_filename, INFINFO_INF_NAME_IS_ABSOLUTE, NULL, size - 1, NULL);
217     ok(ret == TRUE, "Expected SetupGetInfInformation to succeed\n");
218     ret = SetupGetInfInformationA(inf_filename, INFINFO_INF_NAME_IS_ABSOLUTE, NULL, size, NULL);
219     ok(ret == FALSE, "Expected SetupGetInfInformation to fail\n");
220
221     info = HeapAlloc(GetProcessHeap(), 0, size);
222
223     /* try valid ReturnBuffer but too small size */
224     SetLastError(0xbeefcafe);
225     ret = SetupGetInfInformationA(inf_filename, INFINFO_INF_NAME_IS_ABSOLUTE, info, size - 1, &size);
226     ok(ret == FALSE, "Expected SetupGetInfInformation to fail\n");
227     ok(GetLastError() == ERROR_INSUFFICIENT_BUFFER,
228        "Expected ERROR_INSUFFICIENT_BUFFER, got %d\n", GetLastError());
229
230     /* successfully get the inf information */
231     ret = SetupGetInfInformationA(inf_filename, INFINFO_INF_NAME_IS_ABSOLUTE, info, size, &size);
232     ok(ret == TRUE, "Expected SetupGetInfInformation to succeed\n");
233     ok(check_info_filename(info, inf_filename), "Expected returned filename to be equal\n");
234
235     HeapFree(GetProcessHeap(), 0, info);
236
237     /* try the INFINFO_INF_SPEC_IS_HINF search flag */
238     hinf = SetupOpenInfFileA(inf_filename, NULL, INF_STYLE_WIN4, NULL);
239     info = alloc_inf_info(hinf, INFINFO_INF_SPEC_IS_HINF, &size);
240     ret = SetupGetInfInformationA(hinf, INFINFO_INF_SPEC_IS_HINF, info, size, &size);
241     ok(ret == TRUE, "Expected SetupGetInfInformation to succeed\n");
242     ok(check_info_filename(info, inf_filename), "Expected returned filename to be equal\n");
243     SetupCloseInfFile(hinf);
244
245     lstrcpyA(inf_two, WIN_DIR);
246     lstrcatA(inf_two, "\\system32\\");
247     lstrcatA(inf_two, "test.inf");
248     create_inf_file(inf_two);
249
250     HeapFree(GetProcessHeap(), 0, info);
251     info = alloc_inf_info("test.inf", INFINFO_DEFAULT_SEARCH, &size);
252
253     /* check if system32 is searched for inf */
254     ret = SetupGetInfInformationA("test.inf", INFINFO_DEFAULT_SEARCH, info, size, &size);
255     if (!ret && GetLastError() == ERROR_FILE_NOT_FOUND)
256         revfile = inf_one; /* Vista */
257     else
258         revfile = inf_two;
259
260     lstrcpyA(inf_one, WIN_DIR);
261     lstrcatA(inf_one, "\\inf\\");
262     lstrcatA(inf_one, "test.inf");
263     create_inf_file(inf_one);
264
265     HeapFree(GetProcessHeap(), 0, info);
266     info = alloc_inf_info("test.inf", INFINFO_DEFAULT_SEARCH, &size);
267
268     /* test the INFINFO_DEFAULT_SEARCH search flag */
269     ret = SetupGetInfInformationA("test.inf", INFINFO_DEFAULT_SEARCH, info, size, &size);
270     ok(ret == TRUE, "Expected SetupGetInfInformation to succeed: %d\n", GetLastError());
271     ok(check_info_filename(info, inf_one), "Expected returned filename to be equal\n");
272
273     HeapFree(GetProcessHeap(), 0, info);
274     info = alloc_inf_info("test.inf", INFINFO_REVERSE_DEFAULT_SEARCH, &size);
275
276     /* test the INFINFO_REVERSE_DEFAULT_SEARCH search flag */
277     ret = SetupGetInfInformationA("test.inf", INFINFO_REVERSE_DEFAULT_SEARCH, info, size, &size);
278     ok(ret == TRUE, "Expected SetupGetInfInformation to succeed\n");
279     ok(check_info_filename(info, revfile), "Expected returned filename to be equal\n");
280
281     HeapFree(GetProcessHeap(), 0, info);
282
283     DeleteFileA(inf_filename);
284     DeleteFileA(inf_one);
285     DeleteFileA(inf_two);
286 }
287
288 static void test_SetupGetSourceFileLocation(void)
289 {
290     char buffer[MAX_PATH] = "not empty", inf_filename[MAX_PATH];
291     UINT source_id;
292     DWORD required, error;
293     HINF hinf;
294     BOOL ret;
295
296     lstrcpyA(inf_filename, CURR_DIR);
297     lstrcatA(inf_filename, "\\");
298     lstrcatA(inf_filename, "test.inf");
299
300     create_inf_file(inf_filename);
301
302     hinf = SetupOpenInfFileA(inf_filename, NULL, INF_STYLE_WIN4, NULL);
303     ok(hinf != INVALID_HANDLE_VALUE, "could not open inf file\n");
304
305     required = 0;
306     source_id = 0;
307
308     ret = SetupGetSourceFileLocationA(hinf, NULL, "lanconf.exe", &source_id, buffer, sizeof(buffer), &required);
309     ok(ret, "SetupGetSourceFileLocation failed\n");
310
311     ok(required == 1, "unexpected required size: %d\n", required);
312     ok(source_id == 2, "unexpected source id: %d\n", source_id);
313     ok(!lstrcmpA("", buffer), "unexpected result string: %s\n", buffer);
314
315     SetupCloseInfFile(hinf);
316     DeleteFileA(inf_filename);
317
318     create_inf_file2(inf_filename);
319
320     SetLastError(0xdeadbeef);
321     hinf = SetupOpenInfFileA(inf_filename, NULL, INF_STYLE_WIN4, NULL);
322     error = GetLastError();
323     ok(hinf == INVALID_HANDLE_VALUE, "could open inf file\n");
324     ok(error == ERROR_WRONG_INF_STYLE, "got wrong error: %d\n", error);
325
326     hinf = SetupOpenInfFileA(inf_filename, NULL, INF_STYLE_OLDNT, NULL);
327     ok(hinf != INVALID_HANDLE_VALUE, "could not open inf file\n");
328
329     ret = SetupGetSourceFileLocationA(hinf, NULL, "", &source_id, buffer, sizeof(buffer), &required);
330     ok(!ret, "SetupGetSourceFileLocation succeeded\n");
331
332     SetupCloseInfFile(hinf);
333     DeleteFileA(inf_filename);
334 }
335
336 static void test_SetupGetSourceInfo(void)
337 {
338     char buffer[MAX_PATH], inf_filename[MAX_PATH];
339     DWORD required;
340     HINF hinf;
341     BOOL ret;
342
343     lstrcpyA(inf_filename, CURR_DIR);
344     lstrcatA(inf_filename, "\\");
345     lstrcatA(inf_filename, "test.inf");
346
347     create_inf_file(inf_filename);
348
349     hinf = SetupOpenInfFileA(inf_filename, NULL, INF_STYLE_WIN4, NULL);
350     ok(hinf != INVALID_HANDLE_VALUE, "could not open inf file\n");
351
352     required = 0;
353
354     ret = SetupGetSourceInfoA(hinf, 2, SRCINFO_PATH, buffer, sizeof(buffer), &required);
355     ok(ret, "SetupGetSourceInfoA failed\n");
356
357     ok(required == 1, "unexpected required size: %d\n", required);
358     ok(!lstrcmpA("", buffer), "unexpected result string: %s\n", buffer);
359
360     required = 0;
361     buffer[0] = 0;
362
363     ret = SetupGetSourceInfoA(hinf, 2, SRCINFO_TAGFILE, buffer, sizeof(buffer), &required);
364     ok(ret, "SetupGetSourceInfoA failed\n");
365
366     ok(required == 28, "unexpected required size: %d\n", required);
367     ok(!lstrcmpA("LANCOM\\LANtools\\lanconf.cab", buffer), "unexpected result string: %s\n", buffer);
368
369     required = 0;
370     buffer[0] = 0;
371
372     ret = SetupGetSourceInfoA(hinf, 2, SRCINFO_DESCRIPTION, buffer, sizeof(buffer), &required);
373     ok(ret, "SetupGetSourceInfoA failed\n");
374
375     ok(required == 19, "unexpected required size: %d\n", required);
376     ok(!lstrcmpA("LANCOM Software CD", buffer), "unexpected result string: %s\n", buffer);
377
378     SetupCloseInfFile(hinf);
379     DeleteFileA(inf_filename);
380 }
381
382 static void test_SetupGetTargetPath(void)
383 {
384     char buffer[MAX_PATH], inf_filename[MAX_PATH];
385     DWORD required;
386     HINF hinf;
387     INFCONTEXT ctx;
388     BOOL ret;
389
390     lstrcpyA(inf_filename, CURR_DIR);
391     lstrcatA(inf_filename, "\\");
392     lstrcatA(inf_filename, "test.inf");
393
394     create_inf_file(inf_filename);
395
396     hinf = SetupOpenInfFileA(inf_filename, NULL, INF_STYLE_WIN4, NULL);
397     ok(hinf != INVALID_HANDLE_VALUE, "could not open inf file\n");
398
399     ctx.Inf = hinf;
400     ctx.CurrentInf = hinf;
401     ctx.Section = 7;
402     ctx.Line = 0;
403
404     required = 0;
405
406     ret = SetupGetTargetPathA(hinf, &ctx, NULL, buffer, sizeof(buffer), &required);
407     ok(ret, "SetupGetTargetPathA failed\n");
408
409     ok(required == 10, "unexpected required size: %d\n", required);
410     ok(!lstrcmpiA("C:\\LANCOM", buffer), "unexpected result string: %s\n", buffer);
411
412     SetupCloseInfFile(hinf);
413     DeleteFileA(inf_filename);
414 }
415
416 START_TEST(query)
417 {
418     get_directories();
419
420     test_SetupGetInfInformation();
421     test_SetupGetSourceFileLocation();
422     test_SetupGetSourceInfo();
423     test_SetupGetTargetPath();
424 }