1 /* Unit test suite for SHLWAPI Compact List and IStream ordinal functions
3 * Copyright 2002 Jon Griffiths
5 * This library is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU Lesser General Public
7 * License as published by the Free Software Foundation; either
8 * version 2.1 of the License, or (at your option) any later version.
10 * This library is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * Lesser General Public License for more details.
15 * You should have received a copy of the GNU Lesser General Public
16 * License along with this library; if not, write to the Free Software
17 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
22 #include "wine/test.h"
27 typedef struct tagSHLWAPI_CLIST
31 } SHLWAPI_CLIST
, *LPSHLWAPI_CLIST
;
33 typedef const SHLWAPI_CLIST
* LPCSHLWAPI_CLIST
;
36 static const SHLWAPI_CLIST SHLWAPI_CLIST_items
[] =
52 /* Dummy IStream object for testing calls */
68 LPCSHLWAPI_CLIST item
;
73 HRESULT WINAPI
QueryInterface(_IDummyStream
*This
,REFIID riid
, LPVOID
*ppvObj
)
78 static ULONG WINAPI
AddRef(_IDummyStream
*This
)
80 return InterlockedIncrement(&This
->ref
);
83 static ULONG WINAPI
Release(_IDummyStream
*This
)
85 return InterlockedDecrement(&This
->ref
);
88 static HRESULT WINAPI
Read(_IDummyStream
* This
, LPVOID lpMem
, ULONG ulSize
,
94 if (This
->failreadcall
)
96 return STG_E_ACCESSDENIED
;
98 else if (This
->failreadsize
)
100 *lpRead
= ulSize
+ 8;
103 else if (This
->readreturnlarge
)
105 *((ULONG
*)lpMem
) = 0xffff01;
107 This
->readreturnlarge
= FALSE
;
110 if (ulSize
== sizeof(ULONG
))
112 /* Read size of item */
113 *((ULONG
*)lpMem
) = This
->item
->ulSize
? This
->item
->ulSize
+ sizeof(SHLWAPI_CLIST
) : 0;
119 char* buff
= (char*)lpMem
;
122 if (!This
->item
->ulSize
)
124 This
->readbeyondend
= TRUE
;
126 return E_FAIL
; /* Should never happen */
128 *((ULONG
*)lpMem
) = This
->item
->ulId
;
131 for (i
= 0; i
< This
->item
->ulSize
; i
++)
139 static HRESULT WINAPI
Write(_IDummyStream
* This
, LPVOID lpMem
, ULONG ulSize
,
145 if (This
->failwritecall
)
147 return STG_E_ACCESSDENIED
;
149 else if (This
->failwritesize
)
158 static HRESULT WINAPI
Seek(_IDummyStream
* This
, LARGE_INTEGER dlibMove
,
159 DWORD dwOrigin
, ULARGE_INTEGER
* plibNewPosition
)
162 This
->pos
.QuadPart
= dlibMove
.QuadPart
;
164 plibNewPosition
->QuadPart
= dlibMove
.QuadPart
;
168 static HRESULT WINAPI
Stat(_IDummyStream
* This
, STATSTG
* pstatstg
,
172 if (This
->failstatcall
)
175 pstatstg
->cbSize
.QuadPart
= This
->pos
.QuadPart
;
180 static void* iclvt
[] =
192 NULL
, /* LockRegion */
193 NULL
, /* UnlockRegion */
198 /* Function ptrs for ordinal calls */
199 static HMODULE SHLWAPI_hshlwapi
= 0;
201 static VOID (WINAPI
*pSHLWAPI_19
)(LPSHLWAPI_CLIST
);
202 static HRESULT (WINAPI
*pSHLWAPI_20
)(LPSHLWAPI_CLIST
*,LPCSHLWAPI_CLIST
);
203 static BOOL (WINAPI
*pSHLWAPI_21
)(LPSHLWAPI_CLIST
*,ULONG
);
204 static LPSHLWAPI_CLIST (WINAPI
*pSHLWAPI_22
)(LPSHLWAPI_CLIST
,ULONG
);
205 static HRESULT (WINAPI
*pSHLWAPI_17
)(_IDummyStream
*,LPSHLWAPI_CLIST
);
206 static HRESULT (WINAPI
*pSHLWAPI_18
)(_IDummyStream
*,LPSHLWAPI_CLIST
*);
208 static BOOL (WINAPI
*pSHLWAPI_166
)(_IDummyStream
*);
209 static HRESULT (WINAPI
*pSHLWAPI_184
)(_IDummyStream
*,LPVOID
,ULONG
);
210 static HRESULT (WINAPI
*pSHLWAPI_212
)(_IDummyStream
*,LPCVOID
,ULONG
);
211 static HRESULT (WINAPI
*pSHLWAPI_213
)(_IDummyStream
*);
212 static HRESULT (WINAPI
*pSHLWAPI_214
)(_IDummyStream
*,ULARGE_INTEGER
*);
215 static void InitFunctionPtrs(void)
217 SHLWAPI_hshlwapi
= LoadLibraryA("shlwapi.dll");
218 ok(SHLWAPI_hshlwapi
!= 0, "LoadLibrary failed\n");
219 if (SHLWAPI_hshlwapi
)
221 pSHLWAPI_17
= (void *)GetProcAddress( SHLWAPI_hshlwapi
, (LPSTR
)17);
222 ok(pSHLWAPI_17
!= 0, "No Ordinal 17\n");
223 pSHLWAPI_18
= (void *)GetProcAddress( SHLWAPI_hshlwapi
, (LPSTR
)18);
224 ok(pSHLWAPI_18
!= 0, "No Ordinal 18\n");
225 pSHLWAPI_19
= (void *)GetProcAddress( SHLWAPI_hshlwapi
, (LPSTR
)19);
226 ok(pSHLWAPI_19
!= 0, "No Ordinal 19\n");
227 pSHLWAPI_20
= (void *)GetProcAddress( SHLWAPI_hshlwapi
, (LPSTR
)20);
228 ok(pSHLWAPI_20
!= 0, "No Ordinal 20\n");
229 pSHLWAPI_21
= (void *)GetProcAddress( SHLWAPI_hshlwapi
, (LPSTR
)21);
230 ok(pSHLWAPI_21
!= 0, "No Ordinal 21\n");
231 pSHLWAPI_22
= (void *)GetProcAddress( SHLWAPI_hshlwapi
, (LPSTR
)22);
232 ok(pSHLWAPI_22
!= 0, "No Ordinal 22\n");
233 pSHLWAPI_166
= (void *)GetProcAddress( SHLWAPI_hshlwapi
, (LPSTR
)166);
234 ok(pSHLWAPI_166
!= 0, "No Ordinal 166\n");
235 pSHLWAPI_184
= (void *)GetProcAddress( SHLWAPI_hshlwapi
, (LPSTR
)184);
236 ok(pSHLWAPI_184
!= 0, "No Ordinal 184\n");
237 pSHLWAPI_212
= (void *)GetProcAddress( SHLWAPI_hshlwapi
, (LPSTR
)212);
238 ok(pSHLWAPI_212
!= 0, "No Ordinal 212\n");
239 pSHLWAPI_213
= (void *)GetProcAddress( SHLWAPI_hshlwapi
, (LPSTR
)213);
240 ok(pSHLWAPI_213
!= 0, "No Ordinal 213\n");
241 pSHLWAPI_214
= (void *)GetProcAddress( SHLWAPI_hshlwapi
, (LPSTR
)214);
242 ok(pSHLWAPI_214
!= 0, "No Ordinal 214\n");
246 static void InitDummyStream(_IDummyStream
* iface
)
248 iface
->lpVtbl
= (void*)iclvt
;
250 iface
->readcalls
= 0;
251 iface
->failreadcall
= FALSE
;
252 iface
->failreadsize
= FALSE
;
253 iface
->readbeyondend
= FALSE
;
254 iface
->readreturnlarge
= FALSE
;
255 iface
->writecalls
= 0;
256 iface
->failwritecall
= FALSE
;
257 iface
->failwritesize
= FALSE
;
258 iface
->seekcalls
= 0;
259 iface
->statcalls
= 0;
260 iface
->failstatcall
= FALSE
;
261 iface
->item
= SHLWAPI_CLIST_items
;
262 iface
->pos
.QuadPart
= 0;
266 static void test_CList(void)
268 _IDummyStream streamobj
;
269 LPSHLWAPI_CLIST list
= NULL
;
270 LPCSHLWAPI_CLIST item
= SHLWAPI_CLIST_items
;
272 LPSHLWAPI_CLIST inserted
;
276 if (!pSHLWAPI_17
|| !pSHLWAPI_18
|| !pSHLWAPI_19
|| !pSHLWAPI_20
||
277 !pSHLWAPI_21
|| !pSHLWAPI_22
)
280 /* Populate a list and test the items are added correctly */
283 /* Create item and fill with data */
284 inserted
= (LPSHLWAPI_CLIST
)buff
;
285 inserted
->ulSize
= item
->ulSize
+ sizeof(SHLWAPI_CLIST
);
286 inserted
->ulId
= item
->ulId
;
287 for (i
= 0; i
< item
->ulSize
; i
++)
288 buff
[sizeof(SHLWAPI_CLIST
)+i
] = i
*2;
291 hRet
= pSHLWAPI_20(&list
, inserted
);
292 ok(hRet
> S_OK
, "failed list add\n");
296 ok(list
&& list
->ulSize
, "item not added\n");
299 inserted
= pSHLWAPI_22(list
, item
->ulId
);
300 ok(inserted
!= NULL
, "lost after adding\n");
302 ok(!inserted
|| inserted
->ulId
!= ~0U, "find returned a container\n");
305 if (inserted
&& inserted
->ulSize
& 0x3)
308 ok(inserted
[-1].ulId
== ~0U, "invalid size is not countained\n");
309 ok(inserted
[-1].ulSize
> inserted
->ulSize
+sizeof(SHLWAPI_CLIST
),
310 "container too small\n");
314 ok(inserted
->ulSize
==item
->ulSize
+sizeof(SHLWAPI_CLIST
),
315 "id %d wrong size %d\n", inserted
->ulId
, inserted
->ulSize
);
320 LPBYTE bufftest
= (LPBYTE
)inserted
;
322 for (i
= 0; i
< inserted
->ulSize
- sizeof(SHLWAPI_CLIST
); i
++)
323 if (bufftest
[sizeof(SHLWAPI_CLIST
)+i
] != i
*2)
326 ok(bDataOK
== TRUE
, "data corrupted on insert\n");
328 ok(!inserted
|| inserted
->ulId
==item
->ulId
, "find got wrong item\n");
334 InitDummyStream(&streamobj
);
336 hRet
= pSHLWAPI_17(&streamobj
, list
);
337 ok(hRet
== S_OK
, "write failed\n");
340 /* 1 call for each element, + 1 for OK (use our null element for this) */
341 ok(streamobj
.writecalls
== sizeof(SHLWAPI_CLIST_items
)/sizeof(SHLWAPI_CLIST
),
342 "wrong call count\n");
343 ok(streamobj
.readcalls
== 0,"called Read() in write\n");
344 ok(streamobj
.seekcalls
== 0,"called Seek() in write\n");
347 /* Failure cases for writing */
348 InitDummyStream(&streamobj
);
349 streamobj
.failwritecall
= TRUE
;
350 hRet
= pSHLWAPI_17(&streamobj
, list
);
351 ok(hRet
== STG_E_ACCESSDENIED
, "changed object failure return\n");
352 ok(streamobj
.writecalls
== 1, "called object after failure\n");
353 ok(streamobj
.readcalls
== 0,"called Read() after failure\n");
354 ok(streamobj
.seekcalls
== 0,"called Seek() after failure\n");
356 InitDummyStream(&streamobj
);
357 streamobj
.failwritesize
= TRUE
;
358 hRet
= pSHLWAPI_17(&streamobj
, list
);
359 ok(hRet
== STG_E_MEDIUMFULL
, "changed size failure return\n");
360 ok(streamobj
.writecalls
== 1, "called object after size failure\n");
361 ok(streamobj
.readcalls
== 0,"called Read() after failure\n");
362 ok(streamobj
.seekcalls
== 0,"called Seek() after failure\n");
364 /* Invalid inputs for adding */
365 inserted
= (LPSHLWAPI_CLIST
)buff
;
366 inserted
->ulSize
= sizeof(SHLWAPI_CLIST
) -1;
368 hRet
= pSHLWAPI_20(&list
, inserted
);
369 /* The call succeeds but the item is not inserted, except on some early
370 * versions which return failure. Wine behaves like later versions.
374 ok(hRet
== S_OK
, "failed bad element size\n");
376 inserted
= pSHLWAPI_22(list
, 33);
377 ok(inserted
== NULL
, "inserted bad element size\n");
379 inserted
= (LPSHLWAPI_CLIST
)buff
;
380 inserted
->ulSize
= 44;
381 inserted
->ulId
= ~0U;
382 hRet
= pSHLWAPI_20(&list
, inserted
);
383 /* See comment above, some early versions fail this call */
386 ok(hRet
== S_OK
, "failed adding a container\n");
388 item
= SHLWAPI_CLIST_items
;
390 /* Look for nonexistent item in populated list */
391 inserted
= pSHLWAPI_22(list
, 99999999);
392 ok(inserted
== NULL
, "found a nonexistent item\n");
397 BOOL bRet
= pSHLWAPI_21(&list
, item
->ulId
);
398 ok(bRet
== TRUE
, "couldn't find item to delete\n");
402 /* Look for nonexistent item in empty list */
403 inserted
= pSHLWAPI_22(list
, 99999999);
404 ok(inserted
== NULL
, "found an item in empty list\n");
406 /* Create a list by reading in data */
407 InitDummyStream(&streamobj
);
409 hRet
= pSHLWAPI_18(&streamobj
, &list
);
410 ok(hRet
== S_OK
, "failed create from Read()\n");
413 ok(streamobj
.readbeyondend
== FALSE
, "read beyond end\n");
414 /* 2 calls per item, but only 1 for the terminator */
415 ok(streamobj
.readcalls
== sizeof(SHLWAPI_CLIST_items
)/sizeof(SHLWAPI_CLIST
)*2-1,
416 "wrong call count\n");
417 ok(streamobj
.writecalls
== 0, "called Write() from create\n");
418 ok(streamobj
.seekcalls
== 0,"called Seek() from create\n");
420 item
= SHLWAPI_CLIST_items
;
422 /* Check the items were added correctly */
425 inserted
= pSHLWAPI_22(list
, item
->ulId
);
426 ok(inserted
!= NULL
, "lost after adding\n");
428 ok(!inserted
|| inserted
->ulId
!= ~0U, "find returned a container\n");
431 if (inserted
&& inserted
->ulSize
& 0x3)
434 ok(inserted
[-1].ulId
== ~0U, "invalid size is not countained\n");
435 ok(inserted
[-1].ulSize
> inserted
->ulSize
+sizeof(SHLWAPI_CLIST
),
436 "container too small\n");
440 ok(inserted
->ulSize
==item
->ulSize
+sizeof(SHLWAPI_CLIST
),
441 "id %d wrong size %d\n", inserted
->ulId
, inserted
->ulSize
);
443 ok(!inserted
|| inserted
->ulId
==item
->ulId
, "find got wrong item\n");
447 LPBYTE bufftest
= (LPBYTE
)inserted
;
449 for (i
= 0; i
< inserted
->ulSize
- sizeof(SHLWAPI_CLIST
); i
++)
450 if (bufftest
[sizeof(SHLWAPI_CLIST
)+i
] != i
*2)
453 ok(bDataOK
== TRUE
, "data corrupted on insert\n");
459 /* Failure cases for reading */
460 InitDummyStream(&streamobj
);
461 streamobj
.failreadcall
= TRUE
;
462 hRet
= pSHLWAPI_18(&streamobj
, &list
);
463 ok(hRet
== STG_E_ACCESSDENIED
, "changed object failure return\n");
464 ok(streamobj
.readbeyondend
== FALSE
, "read beyond end\n");
465 ok(streamobj
.readcalls
== 1, "called object after read failure\n");
466 ok(streamobj
.writecalls
== 0,"called Write() after read failure\n");
467 ok(streamobj
.seekcalls
== 0,"called Seek() after read failure\n");
469 /* Read returns large object */
470 InitDummyStream(&streamobj
);
471 streamobj
.readreturnlarge
= TRUE
;
472 hRet
= pSHLWAPI_18(&streamobj
, &list
);
473 ok(hRet
== S_OK
, "failed create from Read() with large item\n");
474 ok(streamobj
.readbeyondend
== FALSE
, "read beyond end\n");
475 ok(streamobj
.readcalls
== 1,"wrong call count\n");
476 ok(streamobj
.writecalls
== 0,"called Write() after read failure\n");
477 ok(streamobj
.seekcalls
== 2,"wrong Seek() call count (%d)\n", streamobj
.seekcalls
);
482 static BOOL
test_SHLWAPI_166(void)
484 _IDummyStream streamobj
;
490 InitDummyStream(&streamobj
);
491 bRet
= pSHLWAPI_166(&streamobj
);
494 return FALSE
; /* This version doesn't support stream ops on clists */
496 ok(streamobj
.readcalls
== 0, "called Read()\n");
497 ok(streamobj
.writecalls
== 0, "called Write()\n");
498 ok(streamobj
.seekcalls
== 0, "called Seek()\n");
499 ok(streamobj
.statcalls
== 1, "wrong call count\n");
501 streamobj
.statcalls
= 0;
502 streamobj
.pos
.QuadPart
= 50001;
504 bRet
= pSHLWAPI_166(&streamobj
);
506 ok(bRet
== FALSE
, "failed after seek adjusted\n");
507 ok(streamobj
.readcalls
== 0, "called Read()\n");
508 ok(streamobj
.writecalls
== 0, "called Write()\n");
509 ok(streamobj
.seekcalls
== 0, "called Seek()\n");
510 ok(streamobj
.statcalls
== 1, "wrong call count\n");
513 InitDummyStream(&streamobj
);
514 streamobj
.pos
.QuadPart
= 50001;
515 streamobj
.failstatcall
= TRUE
; /* 1: Stat() Bad, Read() OK */
516 bRet
= pSHLWAPI_166(&streamobj
);
517 ok(bRet
== FALSE
, "should be FALSE after read is OK\n");
518 ok(streamobj
.readcalls
== 1, "wrong call count\n");
519 ok(streamobj
.writecalls
== 0, "called Write()\n");
520 ok(streamobj
.seekcalls
== 1, "wrong call count\n");
521 ok(streamobj
.statcalls
== 1, "wrong call count\n");
522 ok(streamobj
.pos
.QuadPart
== 0, "Didn't seek to start\n");
524 InitDummyStream(&streamobj
);
525 streamobj
.pos
.QuadPart
= 50001;
526 streamobj
.failstatcall
= TRUE
;
527 streamobj
.failreadcall
= TRUE
; /* 2: Stat() Bad, Read() Bad Also */
528 bRet
= pSHLWAPI_166(&streamobj
);
529 ok(bRet
== TRUE
, "Should be true after read fails\n");
530 ok(streamobj
.readcalls
== 1, "wrong call count\n");
531 ok(streamobj
.writecalls
== 0, "called Write()\n");
532 ok(streamobj
.seekcalls
== 0, "Called Seek()\n");
533 ok(streamobj
.statcalls
== 1, "wrong call count\n");
534 ok(streamobj
.pos
.QuadPart
== 50001, "called Seek() after read failed\n");
538 static void test_SHLWAPI_184(void)
540 _IDummyStream streamobj
;
547 InitDummyStream(&streamobj
);
548 hRet
= pSHLWAPI_184(&streamobj
, buff
, sizeof(buff
));
550 ok(hRet
== S_OK
, "failed Read()\n");
551 ok(streamobj
.readcalls
== 1, "wrong call count\n");
552 ok(streamobj
.writecalls
== 0, "called Write()\n");
553 ok(streamobj
.seekcalls
== 0, "called Seek()\n");
556 static void test_SHLWAPI_212(void)
558 _IDummyStream streamobj
;
565 InitDummyStream(&streamobj
);
566 hRet
= pSHLWAPI_212(&streamobj
, buff
, sizeof(buff
));
568 ok(hRet
== S_OK
, "failed Write()\n");
569 ok(streamobj
.readcalls
== 0, "called Read()\n");
570 ok(streamobj
.writecalls
== 1, "wrong call count\n");
571 ok(streamobj
.seekcalls
== 0, "called Seek()\n");
574 static void test_SHLWAPI_213(void)
576 _IDummyStream streamobj
;
581 if (!pSHLWAPI_213
|| !pSHLWAPI_214
)
584 InitDummyStream(&streamobj
);
586 Seek(&streamobj
, ll
, 0, NULL
); /* Seek to 5000l */
588 streamobj
.seekcalls
= 0;
589 pSHLWAPI_213(&streamobj
); /* Should rewind */
590 ok(streamobj
.statcalls
== 0, "called Stat()\n");
591 ok(streamobj
.readcalls
== 0, "called Read()\n");
592 ok(streamobj
.writecalls
== 0, "called Write()\n");
593 ok(streamobj
.seekcalls
== 1, "wrong call count\n");
596 hRet
= pSHLWAPI_214(&streamobj
, &ul
);
597 ok(hRet
== S_OK
, "failed Stat()\n");
598 ok(ul
.QuadPart
== 0, "213 didn't rewind stream\n");
601 static void test_SHLWAPI_214(void)
603 _IDummyStream streamobj
;
611 InitDummyStream(&streamobj
);
613 Seek(&streamobj
, ll
, 0, NULL
);
615 streamobj
.seekcalls
= 0;
616 hRet
= pSHLWAPI_214(&streamobj
, &ul
);
618 ok(hRet
== S_OK
, "failed Stat()\n");
619 ok(streamobj
.statcalls
== 1, "wrong call count\n");
620 ok(streamobj
.readcalls
== 0, "called Read()\n");
621 ok(streamobj
.writecalls
== 0, "called Write()\n");
622 ok(streamobj
.seekcalls
== 0, "called Seek()\n");
623 ok(ul
.QuadPart
== 5000l, "Stat gave wrong size\n");
632 /* Test streaming if this version supports it */
633 if (test_SHLWAPI_166())
641 if (SHLWAPI_hshlwapi
)
642 FreeLibrary(SHLWAPI_hshlwapi
);