1 /*
2 * comctl32 MRU unit tests
3 *
4 * Copyright (C) 2004 Jon Griffiths <jon_p_griffiths@yahoo.com>
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 #include <stdarg.h>
21
22 #include "windef.h"
23 #include "winbase.h"
24 #include "wingdi.h"
25 #include "winuser.h"
26 #include "winnls.h"
27 #include "winreg.h"
28 #include "commctrl.h"
29 #include "shlwapi.h"
30
31 #include "wine/heap.h"
32 #include "wine/test.h"
33
34 /* Keys for testing MRU functions */
35 #define REG_TEST_BASEKEYA "Software\\Wine"
36 #define REG_TEST_BASESUBKEYA "Test"
37 #define REG_TEST_KEYA REG_TEST_BASEKEYA "\\" REG_TEST_BASESUBKEYA
38 #define REG_TEST_SUBKEYA "MRUTest"
39 #define REG_TEST_FULLKEY REG_TEST_KEYA "\\" REG_TEST_SUBKEYA
40
41 /* Undocumented MRU functions */
42 typedef struct tagMRUINFOA
43 {
44 DWORD cbSize;
45 UINT uMax;
46 UINT fFlags;
47 HKEY hKey;
48 LPCSTR lpszSubKey;
49 int (CALLBACK *lpfnCompare)(LPCSTR, LPCSTR);
50 } MRUINFOA;
51
52 typedef struct tagMRUINFOW
53 {
54 DWORD cbSize;
55 UINT uMax;
56 UINT fFlags;
57 HKEY hKey;
58 LPCWSTR lpszSubKey;
59 int (CALLBACK *lpfnCompare)(LPCWSTR, LPCWSTR);
60 } MRUINFOW;
61
62 #define MRU_STRING 0 /* this one's invented */
63 #define MRU_BINARY 1
64 #define MRU_CACHEWRITE 2
65
66 #define LIST_SIZE 3 /* Max entries for each mru */
67
68 static HMODULE hComctl32;
69 static HANDLE (WINAPI *pCreateMRUListA)(MRUINFOA*);
70 static void (WINAPI *pFreeMRUList)(HANDLE);
71 static INT (WINAPI *pAddMRUStringA)(HANDLE,LPCSTR);
72 static INT (WINAPI *pEnumMRUListA)(HANDLE,INT,LPVOID,DWORD);
73 static INT (WINAPI *pEnumMRUListW)(HANDLE,INT,LPVOID,DWORD);
74 static HANDLE (WINAPI *pCreateMRUListLazyA)(MRUINFOA*, DWORD, DWORD, DWORD);
75 static HANDLE (WINAPI *pCreateMRUListLazyW)(MRUINFOW*, DWORD, DWORD, DWORD);
76 static INT (WINAPI *pFindMRUData)(HANDLE, LPCVOID, DWORD, LPINT);
77 static INT (WINAPI *pAddMRUData)(HANDLE, LPCVOID, DWORD);
78 static HANDLE (WINAPI *pCreateMRUListW)(MRUINFOW*);
79
init_functions(void)80 static void init_functions(void)
81 {
82 hComctl32 = LoadLibraryA("comctl32.dll");
83
84 #define X2(f, ord) p##f = (void*)GetProcAddress(hComctl32, (const char *)ord);
85 X2(CreateMRUListA, 151);
86 X2(FreeMRUList, 152);
87 X2(AddMRUStringA, 153);
88 X2(EnumMRUListA, 154);
89 X2(CreateMRUListLazyA, 157);
90 X2(AddMRUData, 167);
91 X2(FindMRUData, 169);
92 X2(CreateMRUListW, 400);
93 X2(EnumMRUListW, 403);
94 X2(CreateMRUListLazyW, 404);
95 #undef X2
96 }
97
98 /* Based on RegDeleteTreeW from dlls/advapi32/registry.c */
mru_RegDeleteTreeA(HKEY hKey,LPCSTR lpszSubKey)99 static LSTATUS mru_RegDeleteTreeA(HKEY hKey, LPCSTR lpszSubKey)
100 {
101 LONG ret;
102 DWORD dwMaxSubkeyLen, dwMaxValueLen;
103 DWORD dwMaxLen, dwSize;
104 CHAR szNameBuf[MAX_PATH], *lpszName = szNameBuf;
105 HKEY hSubKey = hKey;
106
107 if(lpszSubKey)
108 {
109 ret = RegOpenKeyExA(hKey, lpszSubKey, 0, KEY_READ, &hSubKey);
110 if (ret) return ret;
111 }
112
113 /* Get highest length for keys, values */
114 ret = RegQueryInfoKeyA(hSubKey, NULL, NULL, NULL, NULL,
115 &dwMaxSubkeyLen, NULL, NULL, &dwMaxValueLen, NULL, NULL, NULL);
116 if (ret) goto cleanup;
117
118 dwMaxSubkeyLen++;
119 dwMaxValueLen++;
120 dwMaxLen = max(dwMaxSubkeyLen, dwMaxValueLen);
121 if (dwMaxLen > ARRAY_SIZE(szNameBuf))
122 {
123 /* Name too big: alloc a buffer for it */
124 if (!(lpszName = heap_alloc(dwMaxLen * sizeof(CHAR))))
125 {
126 ret = ERROR_NOT_ENOUGH_MEMORY;
127 goto cleanup;
128 }
129 }
130
131
132 /* Recursively delete all the subkeys */
133 while (TRUE)
134 {
135 dwSize = dwMaxLen;
136 if (RegEnumKeyExA(hSubKey, 0, lpszName, &dwSize, NULL,
137 NULL, NULL, NULL)) break;
138
139 ret = mru_RegDeleteTreeA(hSubKey, lpszName);
140 if (ret) goto cleanup;
141 }
142
143 if (lpszSubKey)
144 ret = RegDeleteKeyA(hKey, lpszSubKey);
145 else
146 while (TRUE)
147 {
148 dwSize = dwMaxLen;
149 if (RegEnumValueA(hKey, 0, lpszName, &dwSize,
150 NULL, NULL, NULL, NULL)) break;
151
152 ret = RegDeleteValueA(hKey, lpszName);
153 if (ret) goto cleanup;
154 }
155
156 cleanup:
157 /* Free buffer if allocated */
158 if (lpszName != szNameBuf)
159 heap_free(lpszName);
160 if(lpszSubKey)
161 RegCloseKey(hSubKey);
162 return ret;
163 }
164
create_reg_entries(void)165 static BOOL create_reg_entries(void)
166 {
167 HKEY hKey = NULL;
168
169 ok(!RegCreateKeyA(HKEY_CURRENT_USER, REG_TEST_FULLKEY, &hKey),
170 "Couldn't create test key \"%s\"\n", REG_TEST_KEYA);
171 if (!hKey) return FALSE;
172 RegCloseKey(hKey);
173 return TRUE;
174 }
175
delete_reg_entries(void)176 static void delete_reg_entries(void)
177 {
178 HKEY hKey;
179
180 if (RegOpenKeyExA(HKEY_CURRENT_USER, REG_TEST_BASEKEYA, 0, KEY_ALL_ACCESS,
181 &hKey))
182 return;
183 mru_RegDeleteTreeA(hKey, REG_TEST_BASESUBKEYA);
184 RegCloseKey(hKey);
185 }
186
check_reg_entries(const char * mrulist,const char ** items)187 static void check_reg_entries(const char *mrulist, const char**items)
188 {
189 char buff[128];
190 HKEY hKey = NULL;
191 DWORD type, size, ret;
192 unsigned int i;
193
194 ok(!RegOpenKeyA(HKEY_CURRENT_USER, REG_TEST_FULLKEY, &hKey),
195 "Couldn't open test key \"%s\"\n", REG_TEST_FULLKEY);
196 if (!hKey) return;
197
198 type = REG_SZ;
199 size = sizeof(buff);
200 buff[0] = '\0';
201 ret = RegQueryValueExA(hKey, "MRUList", NULL, &type, (LPBYTE)buff, &size);
202
203 ok(!ret && buff[0], "Checking MRU: got %d from RegQueryValueExW\n", ret);
204 if(ret || !buff[0]) return;
205
206 ok(strcmp(buff, mrulist) == 0, "Checking MRU: Expected list %s, got %s\n",
207 mrulist, buff);
208 if(strcmp(buff, mrulist)) return;
209
210 for (i = 0; i < strlen(mrulist); i++)
211 {
212 char name[2];
213 name[0] = mrulist[i];
214 name[1] = '\0';
215 type = REG_SZ;
216 size = sizeof(buff);
217 buff[0] = '\0';
218 ret = RegQueryValueExA(hKey, name, NULL, &type, (LPBYTE)buff, &size);
219 ok(!ret && buff[0],
220 "Checking MRU item %d ('%c'): got %d from RegQueryValueExW\n",
221 i, mrulist[i], ret);
222 if(ret || !buff[0]) return;
223 ok(!strcmp(buff, items[mrulist[i]-'a']),
224 "Checking MRU item %d ('%c'): expected \"%s\", got \"%s\"\n",
225 i, mrulist[i], buff, items[mrulist[i] - 'a']);
226 }
227 }
228
cmp_mru_strA(LPCSTR data1,LPCSTR data2)229 static int CALLBACK cmp_mru_strA(LPCSTR data1, LPCSTR data2)
230 {
231 return lstrcmpiA(data1, data2);
232 }
233
test_MRUListA(void)234 static void test_MRUListA(void)
235 {
236 const char *checks[LIST_SIZE+1];
237 MRUINFOA infoA;
238 HANDLE hMRU;
239 HKEY hKey;
240 INT iRet;
241
242 if (!pCreateMRUListA || !pFreeMRUList || !pAddMRUStringA || !pEnumMRUListA)
243 {
244 win_skip("MRU entry points not found\n");
245 return;
246 }
247
248 if (0)
249 {
250 /* Create (NULL) - crashes native */
251 hMRU = pCreateMRUListA(NULL);
252 }
253
254 /* size too small */
255 infoA.cbSize = sizeof(infoA) - 2;
256 infoA.uMax = LIST_SIZE;
257 infoA.fFlags = MRU_STRING;
258 infoA.hKey = NULL;
259 infoA.lpszSubKey = REG_TEST_SUBKEYA;
260 infoA.lpfnCompare = cmp_mru_strA;
261
262 SetLastError(0);
263 hMRU = pCreateMRUListA(&infoA);
264 ok (!hMRU && !GetLastError(),
265 "CreateMRUListA(too small) expected NULL,0 got %p,%d\n",
266 hMRU, GetLastError());
267
268 /* size too big */
269 infoA.cbSize = sizeof(infoA) + 2;
270 infoA.uMax = LIST_SIZE;
271 infoA.fFlags = MRU_STRING;
272 infoA.hKey = NULL;
273 infoA.lpszSubKey = REG_TEST_SUBKEYA;
274 infoA.lpfnCompare = cmp_mru_strA;
275
276 SetLastError(0);
277 hMRU = pCreateMRUListA(&infoA);
278 ok (!hMRU && !GetLastError(),
279 "CreateMRUListA(too big) expected NULL,0 got %p,%d\n",
280 hMRU, GetLastError());
281
282 /* NULL hKey */
283 infoA.cbSize = sizeof(infoA);
284 infoA.uMax = LIST_SIZE;
285 infoA.fFlags = MRU_STRING;
286 infoA.hKey = NULL;
287 infoA.lpszSubKey = REG_TEST_SUBKEYA;
288 infoA.lpfnCompare = cmp_mru_strA;
289
290 SetLastError(0);
291 hMRU = pCreateMRUListA(&infoA);
292 ok (!hMRU && !GetLastError(),
293 "CreateMRUListA(NULL key) expected NULL,0 got %p,%d\n",
294 hMRU, GetLastError());
295
296 /* NULL subkey name */
297 infoA.cbSize = sizeof(infoA);
298 infoA.uMax = LIST_SIZE;
299 infoA.fFlags = MRU_STRING;
300 infoA.hKey = NULL;
301 infoA.lpszSubKey = NULL;
302 infoA.lpfnCompare = cmp_mru_strA;
303
304 SetLastError(0);
305 hMRU = pCreateMRUListA(&infoA);
306 ok (!hMRU && !GetLastError(),
307 "CreateMRUListA(NULL name) expected NULL,0 got %p,%d\n",
308 hMRU, GetLastError());
309
310 /* Create a string MRU */
311 ok(!RegCreateKeyA(HKEY_CURRENT_USER, REG_TEST_KEYA, &hKey),
312 "Couldn't create test key \"%s\"\n", REG_TEST_KEYA);
313 if (!hKey) return;
314
315 infoA.cbSize = sizeof(infoA);
316 infoA.uMax = LIST_SIZE;
317 infoA.fFlags = MRU_STRING;
318 infoA.hKey = hKey;
319 infoA.lpszSubKey = REG_TEST_SUBKEYA;
320 infoA.lpfnCompare = cmp_mru_strA;
321
322 hMRU = pCreateMRUListA(&infoA);
323 ok(hMRU && !GetLastError(),
324 "CreateMRUListA(string) expected non-NULL,0 got %p,%d\n",
325 hMRU, GetLastError());
326
327 if (hMRU)
328 {
329 char buffer[255];
330 checks[0] = "Test 1";
331 checks[1] = "Test 2";
332 checks[2] = "Test 3";
333 checks[3] = "Test 4";
334
335 /* Add (NULL list) */
336 SetLastError(0);
337 iRet = pAddMRUStringA(NULL, checks[0]);
338 ok(iRet == -1 && !GetLastError(),
339 "AddMRUStringA(NULL list) expected -1,0 got %d,%d\n",
340 iRet, GetLastError());
341
342 /* Add (NULL string) */
343 if (0)
344 {
345 /* Some native versions crash when passed NULL or fail to SetLastError() */
346 SetLastError(0);
347 iRet = pAddMRUStringA(hMRU, NULL);
348 ok(iRet == 0 && GetLastError() == ERROR_INVALID_PARAMETER,
349 "AddMRUStringA(NULL str) expected 0,ERROR_INVALID_PARAMETER got %d,%d\n",
350 iRet, GetLastError());
351 }
352
353 /* Add 3 strings. Check the registry is correct after each add */
354 SetLastError(0);
355 iRet = pAddMRUStringA(hMRU, checks[0]);
356 ok(iRet == 0 && !GetLastError(),
357 "AddMRUStringA(1) expected 0,0 got %d,%d\n",
358 iRet, GetLastError());
359 check_reg_entries("a", checks);
360
361 SetLastError(0);
362 iRet = pAddMRUStringA(hMRU, checks[1]);
363 ok(iRet == 1 && !GetLastError(),
364 "AddMRUStringA(2) expected 1,0 got %d,%d\n",
365 iRet, GetLastError());
366 check_reg_entries("ba", checks);
367
368 SetLastError(0);
369 iRet = pAddMRUStringA(hMRU, checks[2]);
370 ok(iRet == 2 && !GetLastError(),
371 "AddMRUStringA(2) expected 2,0 got %d,%d\n",
372 iRet, GetLastError());
373 check_reg_entries("cba", checks);
374
375 /* Add a duplicate of the 2nd string - it should move to the front,
376 * but keep the same index in the registry.
377 */
378 SetLastError(0);
379 iRet = pAddMRUStringA(hMRU, checks[1]);
380 ok(iRet == 1 && !GetLastError(),
381 "AddMRUStringA(re-add 1) expected 1,0 got %d,%d\n",
382 iRet, GetLastError());
383 check_reg_entries("bca", checks);
384
385 /* Add a new string - replaces the oldest string + moves to the front */
386 SetLastError(0);
387 iRet = pAddMRUStringA(hMRU, checks[3]);
388 ok(iRet == 0 && !GetLastError(),
389 "AddMRUStringA(add new) expected 0,0 got %d,%d\n",
390 iRet, GetLastError());
391 checks[0] = checks[3];
392 check_reg_entries("abc", checks);
393
394 /* NULL buffer = get list size */
395 iRet = pEnumMRUListA(hMRU, 0, NULL, 0);
396 ok(iRet == 3 || iRet == -1 /* Vista */, "EnumMRUList expected %d or -1, got %d\n", LIST_SIZE, iRet);
397
398 /* negative item pos = get list size */
399 iRet = pEnumMRUListA(hMRU, -1, NULL, 0);
400 ok(iRet == 3 || iRet == -1 /* Vista */, "EnumMRUList expected %d or -1, got %d\n", LIST_SIZE, iRet);
401
402 /* negative item pos = get list size */
403 iRet = pEnumMRUListA(hMRU, -5, NULL, 0);
404 ok(iRet == 3 || iRet == -1 /* Vista */, "EnumMRUList expected %d or -1, got %d\n", LIST_SIZE, iRet);
405
406 /* negative item pos = get list size */
407 iRet = pEnumMRUListA(hMRU, -1, buffer, 255);
408 ok(iRet == 3, "EnumMRUList expected %d, got %d\n", LIST_SIZE, iRet);
409
410 /* negative item pos = get list size */
411 iRet = pEnumMRUListA(hMRU, -5, buffer, 255);
412 ok(iRet == 3, "EnumMRUList expected %d, got %d\n", LIST_SIZE, iRet);
413
414 /* check entry 0 */
415 buffer[0] = 0;
416 iRet = pEnumMRUListA(hMRU, 0, buffer, 255);
417 ok(iRet == lstrlenA(checks[3]), "EnumMRUList expected %d, got %d\n", lstrlenA(checks[3]), iRet);
418 ok(strcmp(buffer, checks[3]) == 0, "EnumMRUList expected %s, got %s\n", checks[3], buffer);
419
420 /* check entry 0 with a too small buffer */
421 buffer[0] = 0; /* overwritten with 'T' */
422 buffer[1] = 'A'; /* overwritten with 0 */
423 buffer[2] = 'A'; /* unchanged */
424 buffer[3] = 0; /* unchanged */
425 iRet = pEnumMRUListA(hMRU, 0, buffer, 2);
426 ok(iRet == lstrlenA(checks[3]), "EnumMRUList expected %d, got %d\n", lstrlenA(checks[3]), iRet);
427 ok(strcmp(buffer, "T") == 0, "EnumMRUList expected %s, got %s\n", "T", buffer);
428 /* make sure space after buffer has old values */
429 ok(buffer[2] == 'A', "EnumMRUList expected %02x, got %02x\n", 'A', buffer[2]);
430
431 /* check entry 1 */
432 buffer[0] = 0;
433 iRet = pEnumMRUListA(hMRU, 1, buffer, 255);
434 ok(iRet == lstrlenA(checks[1]), "EnumMRUList expected %d, got %d\n", lstrlenA(checks[1]), iRet);
435 ok(strcmp(buffer, checks[1]) == 0, "EnumMRUList expected %s, got %s\n", checks[1], buffer);
436
437 /* check entry 2 */
438 buffer[0] = 0;
439 iRet = pEnumMRUListA(hMRU, 2, buffer, 255);
440 ok(iRet == lstrlenA(checks[2]), "EnumMRUList expected %d, got %d\n", lstrlenA(checks[2]), iRet);
441 ok(strcmp(buffer, checks[2]) == 0, "EnumMRUList expected %s, got %s\n", checks[2], buffer);
442
443 /* check out of bounds entry 3 */
444 strcpy(buffer, "dummy");
445 iRet = pEnumMRUListA(hMRU, 3, buffer, 255);
446 ok(iRet == -1, "EnumMRUList expected %d, got %d\n", -1, iRet);
447 ok(strcmp(buffer, "dummy") == 0, "EnumMRUList expected unchanged buffer %s, got %s\n", "dummy", buffer);
448
449 /* Finished with this MRU */
450 pFreeMRUList(hMRU);
451 }
452
453 /* FreeMRUList(NULL) crashes on Win98 OSR0 */
454 }
455
456 typedef struct {
457 MRUINFOA mruA;
458 BOOL ret;
459 } create_lazya_t;
460
461 static const create_lazya_t create_lazyA[] = {
462 {{ sizeof(MRUINFOA) + 1, 0, 0, HKEY_CURRENT_USER, NULL, NULL }, FALSE },
463 {{ sizeof(MRUINFOA) - 1, 0, 0, HKEY_CURRENT_USER, NULL, NULL }, FALSE },
464 {{ sizeof(MRUINFOA) + 1, 0, 0, HKEY_CURRENT_USER, "WineTest", NULL }, TRUE },
465 {{ sizeof(MRUINFOA) - 1, 0, 0, HKEY_CURRENT_USER, "WineTest", NULL }, TRUE },
466 {{ sizeof(MRUINFOA), 0, 0, HKEY_CURRENT_USER, "WineTest", NULL }, TRUE },
467 {{ sizeof(MRUINFOA), 0, 0, HKEY_CURRENT_USER, NULL, NULL }, FALSE },
468 {{ sizeof(MRUINFOA), 0, 0, NULL, "WineTest", NULL }, FALSE },
469 {{ 0, 0, 0, NULL, "WineTest", NULL }, FALSE },
470 {{ 0, 0, 0, HKEY_CURRENT_USER, "WineTest", NULL }, TRUE }
471 };
472
test_CreateMRUListLazyA(void)473 static void test_CreateMRUListLazyA(void)
474 {
475 int i;
476
477 if (!pCreateMRUListLazyA || !pFreeMRUList)
478 {
479 win_skip("CreateMRUListLazyA or FreeMRUList entry points not found\n");
480 return;
481 }
482
483 for (i = 0; i < ARRAY_SIZE(create_lazyA); i++)
484 {
485 const create_lazya_t *ptr = &create_lazyA[i];
486 HANDLE hMRU;
487
488 hMRU = pCreateMRUListLazyA((MRUINFOA*)&ptr->mruA, 0, 0, 0);
489 if (ptr->ret)
490 {
491 ok(hMRU != NULL, "%d: got %p\n", i, hMRU);
492 pFreeMRUList(hMRU);
493 }
494 else
495 ok(hMRU == NULL, "%d: got %p\n", i, hMRU);
496 }
497 }
498
test_EnumMRUList(void)499 static void test_EnumMRUList(void)
500 {
501 if (!pEnumMRUListA || !pEnumMRUListW)
502 {
503 win_skip("EnumMRUListA/EnumMRUListW entry point not found\n");
504 return;
505 }
506
507 /* NULL handle */
508 if (0)
509 {
510 /* crashes on NT4, passed on Win2k, XP, 2k3, Vista, 2k8 */
511 pEnumMRUListA(NULL, 0, NULL, 0);
512 pEnumMRUListW(NULL, 0, NULL, 0);
513 }
514 }
515
test_FindMRUData(void)516 static void test_FindMRUData(void)
517 {
518 INT iRet;
519
520 if (!pFindMRUData)
521 {
522 win_skip("FindMRUData entry point not found\n");
523 return;
524 }
525
526 /* NULL handle */
527 iRet = pFindMRUData(NULL, NULL, 0, NULL);
528 ok(iRet == -1, "FindMRUData expected -1, got %d\n", iRet);
529 }
530
test_AddMRUData(void)531 static void test_AddMRUData(void)
532 {
533 INT iRet;
534
535 if (!pAddMRUData)
536 {
537 win_skip("AddMRUData entry point not found\n");
538 return;
539 }
540
541 /* NULL handle */
542 iRet = pFindMRUData(NULL, NULL, 0, NULL);
543 ok(iRet == -1, "AddMRUData expected -1, got %d\n", iRet);
544 }
545
test_CreateMRUListW(void)546 static void test_CreateMRUListW(void)
547 {
548 static const WCHAR mrutestW[] = {'M','R','U','T','e','s','t',0};
549 MRUINFOW infoW;
550 void *named;
551 HKEY hKey;
552 HANDLE hMru;
553
554 if (!pCreateMRUListW)
555 {
556 win_skip("CreateMRUListW entry point not found\n");
557 return;
558 }
559
560 /* exported by name too on recent versions */
561 named = GetProcAddress(hComctl32, "CreateMRUListW");
562 if (named)
563 ok(named == pCreateMRUListW, "got %p, expected %p\n", named, pCreateMRUListW);
564
565 ok(!RegCreateKeyA(HKEY_CURRENT_USER, REG_TEST_KEYA, &hKey),
566 "Couldn't create test key \"%s\"\n", REG_TEST_KEYA);
567
568 infoW.cbSize = sizeof(infoW);
569 infoW.uMax = 1;
570 infoW.fFlags = 0;
571 infoW.lpszSubKey = mrutestW;
572 infoW.hKey = hKey;
573 infoW.lpfnCompare = NULL;
574
575 hMru = pCreateMRUListW(&infoW);
576 ok(hMru != NULL, "got %p\n", hMru);
577 pFreeMRUList(hMru);
578
579 /* smaller size */
580 infoW.cbSize = sizeof(infoW) - 1;
581 infoW.uMax = 1;
582 infoW.fFlags = 0;
583 infoW.lpszSubKey = mrutestW;
584 infoW.hKey = hKey;
585 infoW.lpfnCompare = NULL;
586
587 hMru = pCreateMRUListW(&infoW);
588 ok(hMru != NULL, "got %p\n", hMru);
589 pFreeMRUList(hMru);
590
591 /* increased size */
592 infoW.cbSize = sizeof(infoW) + 1;
593 infoW.uMax = 1;
594 infoW.fFlags = 0;
595 infoW.lpszSubKey = mrutestW;
596 infoW.hKey = hKey;
597 infoW.lpfnCompare = NULL;
598
599 hMru = pCreateMRUListW(&infoW);
600 ok(hMru != NULL, "got %p\n", hMru);
601 pFreeMRUList(hMru);
602
603 /* zero size */
604 infoW.cbSize = 0;
605 infoW.uMax = 1;
606 infoW.fFlags = 0;
607 infoW.lpszSubKey = mrutestW;
608 infoW.hKey = hKey;
609 infoW.lpfnCompare = NULL;
610
611 hMru = pCreateMRUListW(&infoW);
612 ok(hMru != NULL, "got %p\n", hMru);
613 pFreeMRUList(hMru);
614
615 /* NULL hKey */
616 infoW.cbSize = sizeof(infoW);
617 infoW.uMax = 1;
618 infoW.fFlags = 0;
619 infoW.lpszSubKey = mrutestW;
620 infoW.hKey = NULL;
621 infoW.lpfnCompare = NULL;
622
623 hMru = pCreateMRUListW(&infoW);
624 ok(hMru == NULL, "got %p\n", hMru);
625
626 RegCloseKey(hKey);
627 }
628
test_CreateMRUListLazyW(void)629 static void test_CreateMRUListLazyW(void)
630 {
631 static const WCHAR mrutestW[] = {'M','R','U','T','e','s','t',0};
632 MRUINFOW infoW;
633 void *named;
634 HKEY hKey;
635 HANDLE hMru;
636
637 if (!pCreateMRUListLazyW)
638 {
639 win_skip("CreateMRUListLazyW entry point not found\n");
640 return;
641 }
642
643 /* check that it's not exported by name */
644 named = GetProcAddress(hComctl32, "CreateMRUListLazyW");
645 ok(named == NULL, "got %p\n", named);
646
647 ok(!RegCreateKeyA(HKEY_CURRENT_USER, REG_TEST_KEYA, &hKey),
648 "Couldn't create test key \"%s\"\n", REG_TEST_KEYA);
649
650 infoW.cbSize = sizeof(infoW);
651 infoW.uMax = 1;
652 infoW.fFlags = 0;
653 infoW.lpszSubKey = mrutestW;
654 infoW.hKey = hKey;
655 infoW.lpfnCompare = NULL;
656
657 hMru = pCreateMRUListLazyW(&infoW, 0, 0, 0);
658 ok(hMru != NULL, "got %p\n", hMru);
659 pFreeMRUList(hMru);
660
661 /* smaller size */
662 infoW.cbSize = sizeof(infoW) - 1;
663 infoW.uMax = 1;
664 infoW.fFlags = 0;
665 infoW.lpszSubKey = mrutestW;
666 infoW.hKey = hKey;
667 infoW.lpfnCompare = NULL;
668
669 hMru = pCreateMRUListLazyW(&infoW, 0, 0, 0);
670 ok(hMru != NULL, "got %p\n", hMru);
671 pFreeMRUList(hMru);
672
673 /* increased size */
674 infoW.cbSize = sizeof(infoW) + 1;
675 infoW.uMax = 1;
676 infoW.fFlags = 0;
677 infoW.lpszSubKey = mrutestW;
678 infoW.hKey = hKey;
679 infoW.lpfnCompare = NULL;
680
681 hMru = pCreateMRUListLazyW(&infoW, 0, 0, 0);
682 ok(hMru != NULL, "got %p\n", hMru);
683 pFreeMRUList(hMru);
684
685 /* zero size */
686 infoW.cbSize = 0;
687 infoW.uMax = 1;
688 infoW.fFlags = 0;
689 infoW.lpszSubKey = mrutestW;
690 infoW.hKey = hKey;
691 infoW.lpfnCompare = NULL;
692
693 hMru = pCreateMRUListLazyW(&infoW, 0, 0, 0);
694 ok(hMru != NULL, "got %p\n", hMru);
695 pFreeMRUList(hMru);
696
697 /* NULL hKey */
698 infoW.cbSize = sizeof(infoW);
699 infoW.uMax = 1;
700 infoW.fFlags = 0;
701 infoW.lpszSubKey = mrutestW;
702 infoW.hKey = NULL;
703 infoW.lpfnCompare = NULL;
704
705 hMru = pCreateMRUListLazyW(&infoW, 0, 0, 0);
706 ok(hMru == NULL, "got %p\n", hMru);
707
708 RegCloseKey(hKey);
709 }
710
START_TEST(mru)711 START_TEST(mru)
712 {
713 delete_reg_entries();
714 if (!create_reg_entries())
715 return;
716
717 init_functions();
718
719 test_MRUListA();
720 test_CreateMRUListLazyA();
721 test_CreateMRUListLazyW();
722 test_EnumMRUList();
723 test_FindMRUData();
724 test_AddMRUData();
725 test_CreateMRUListW();
726
727 delete_reg_entries();
728 }
729