2 * Unit tests for DPA functions
4 * Copyright 2003 Uwe Bonnes
5 * Copyright 2005 Felix Nawothnig
7 * This library is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU Lesser General Public
9 * License as published by the Free Software Foundation; either
10 * version 2.1 of the License, or (at your option) any later version.
12 * This library is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * Lesser General Public License for more details.
17 * You should have received a copy of the GNU Lesser General Public
18 * License along with this library; if not, write to the Free Software
19 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
32 #include "wine/test.h"
34 #define expect(expected, got) ok(got == expected, "Expected %d, got %d\n", expected, got)
36 typedef struct _STREAMDATA
41 } STREAMDATA
, *PSTREAMDATA
;
43 static HDPA (WINAPI
*pDPA_Clone
)(const HDPA
,const HDPA
);
44 static HDPA (WINAPI
*pDPA_Create
)(INT
);
45 static HDPA (WINAPI
*pDPA_CreateEx
)(INT
,HANDLE
);
46 static PVOID (WINAPI
*pDPA_DeleteAllPtrs
)(const HDPA
);
47 static PVOID (WINAPI
*pDPA_DeletePtr
)(const HDPA
,INT
);
48 static BOOL (WINAPI
*pDPA_Destroy
)(const HDPA
);
49 static VOID (WINAPI
*pDPA_DestroyCallback
)(HDPA
,PFNDPAENUMCALLBACK
,PVOID
);
50 static VOID (WINAPI
*pDPA_EnumCallback
)(HDPA
,PFNDPAENUMCALLBACK
,PVOID
);
51 static INT (WINAPI
*pDPA_GetPtr
)(const HDPA
,INT
);
52 static INT (WINAPI
*pDPA_GetPtrIndex
)(const HDPA
,PVOID
);
53 static BOOL (WINAPI
*pDPA_Grow
)(HDPA
,INT
);
54 static INT (WINAPI
*pDPA_InsertPtr
)(const HDPA
,INT
,PVOID
);
55 static HRESULT (WINAPI
*pDPA_LoadStream
)(HDPA
*,PFNDPASTREAM
,IStream
*,LPVOID
);
56 static BOOL (WINAPI
*pDPA_Merge
)(const HDPA
,const HDPA
,DWORD
,PFNDPACOMPARE
,PFNDPAMERGE
,LPARAM
);
57 static HRESULT (WINAPI
*pDPA_SaveStream
)(HDPA
,PFNDPASTREAM
,IStream
*,LPVOID
);
58 static INT (WINAPI
*pDPA_Search
)(HDPA
,PVOID
,INT
,PFNDPACOMPARE
,LPARAM
,UINT
);
59 static BOOL (WINAPI
*pDPA_SetPtr
)(const HDPA
,INT
,PVOID
);
60 static BOOL (WINAPI
*pDPA_Sort
)(const HDPA
,PFNDPACOMPARE
,LPARAM
);
62 #define COMCTL32_GET_PROC(func, ord) \
63 ((p ## func = (PVOID)GetProcAddress(hcomctl32,(LPCSTR)ord)) ? 1 \
64 : (trace( #func " not exported\n"), 0))
66 static BOOL
InitFunctionPtrs(HMODULE hcomctl32
)
69 if(COMCTL32_GET_PROC(DPA_Clone
, 331) &&
70 COMCTL32_GET_PROC(DPA_Create
, 328) &&
71 COMCTL32_GET_PROC(DPA_CreateEx
, 340) &&
72 COMCTL32_GET_PROC(DPA_DeleteAllPtrs
, 337) &&
73 COMCTL32_GET_PROC(DPA_DeletePtr
, 336) &&
74 COMCTL32_GET_PROC(DPA_Destroy
, 329) &&
75 COMCTL32_GET_PROC(DPA_GetPtr
, 332) &&
76 COMCTL32_GET_PROC(DPA_GetPtrIndex
, 333) &&
77 COMCTL32_GET_PROC(DPA_Grow
, 330) &&
78 COMCTL32_GET_PROC(DPA_InsertPtr
, 334) &&
79 COMCTL32_GET_PROC(DPA_Search
, 339) &&
80 COMCTL32_GET_PROC(DPA_SetPtr
, 335) &&
81 COMCTL32_GET_PROC(DPA_Sort
, 338))
84 COMCTL32_GET_PROC(DPA_DestroyCallback
, 386) &&
85 COMCTL32_GET_PROC(DPA_EnumCallback
, 385) &&
86 COMCTL32_GET_PROC(DPA_LoadStream
, 9) &&
87 COMCTL32_GET_PROC(DPA_Merge
, 11) &&
88 COMCTL32_GET_PROC(DPA_SaveStream
, 10);
97 static INT CALLBACK
CB_CmpLT(PVOID p1
, PVOID p2
, LPARAM lp
)
99 ok(lp
== 0xdeadbeef, "lp=%ld\n", lp
);
100 return p1
< p2
? -1 : p1
> p2
? 1 : 0;
103 static INT CALLBACK
CB_CmpGT(PVOID p1
, PVOID p2
, LPARAM lp
)
105 ok(lp
== 0xdeadbeef, "lp=%ld\n", lp
);
106 return p1
> p2
? -1 : p1
< p2
? 1 : 0;
109 /* merge callback messages counter
113 static INT nMessages
[4];
115 static PVOID CALLBACK
CB_MergeInsertSrc(UINT op
, PVOID p1
, PVOID p2
, LPARAM lp
)
118 ok(lp
== 0xdeadbeef, "lp=%ld\n", lp
);
122 static PVOID CALLBACK
CB_MergeDeleteOddSrc(UINT op
, PVOID p1
, PVOID p2
, LPARAM lp
)
125 ok(lp
== 0xdeadbeef, "lp=%ld\n", lp
);
126 return ((PCHAR
)p2
)+1;
131 static INT CALLBACK
CB_EnumFirstThree(PVOID pItem
, PVOID lp
)
135 i
= pDPA_GetPtrIndex(lp
, pItem
);
136 ok(i
== nEnum
, "i=%d nEnum=%d\n", i
, nEnum
);
138 pDPA_SetPtr(lp
, i
, (PVOID
)7);
139 return pItem
!= (PVOID
)3;
142 static HRESULT CALLBACK
CB_Save(DPASTREAMINFO
*pInfo
, IStream
*pStm
, LPVOID lp
)
146 ok(lp
== (LPVOID
)0xdeadbeef, "lp=%p\n", lp
);
147 hRes
= IStream_Write(pStm
, &pInfo
->iPos
, sizeof(INT
), NULL
);
148 ok(hRes
== S_OK
, "hRes=0x%x\n", hRes
);
149 hRes
= IStream_Write(pStm
, &pInfo
->pvItem
, sizeof(PVOID
), NULL
);
150 ok(hRes
== S_OK
, "hRes=0x%x\n", hRes
);
154 static HRESULT CALLBACK
CB_Load(DPASTREAMINFO
*pInfo
, IStream
*pStm
, LPVOID lp
)
159 iOldPos
= pInfo
->iPos
;
160 ok(lp
== (LPVOID
)0xdeadbeef, "lp=%p\n", lp
);
161 hRes
= IStream_Read(pStm
, &pInfo
->iPos
, sizeof(INT
), NULL
);
162 ok(hRes
== S_OK
, "hRes=0x%x\n", hRes
);
163 ok(pInfo
->iPos
== iOldPos
, "iPos=%d iOldPos=%d\n", pInfo
->iPos
, iOldPos
);
164 hRes
= IStream_Read(pStm
, &pInfo
->pvItem
, sizeof(PVOID
), NULL
);
165 ok(hRes
== S_OK
, "hRes=0x%x\n", hRes
);
169 static BOOL
CheckDPA(HDPA dpa
, DWORD dwIn
, PDWORD pdwOut
)
176 ULONG_PTR ulItem
= (ULONG_PTR
)pDPA_GetPtr(dpa
, i
++);
178 dwOut
= dwOut
<< 4 | (ulItem
& 0xf);
185 pDPA_DeleteAllPtrs(dpa
);
189 pDPA_InsertPtr(dpa
, 0, (PVOID
)(ULONG_PTR
)(dwIn
& 0xf));
200 static void test_dpa(void)
204 HDPA dpa
, dpa2
, dpa3
;
211 hHeap
= HeapCreate(0, 1, 2);
212 ok(hHeap
!= NULL
, "error=%d\n", GetLastError());
213 dpa3
= pDPA_CreateEx(0, hHeap
);
214 ok(dpa3
!= NULL
, "\n");
215 ret
= pDPA_Grow(dpa3
, si
.dwPageSize
+ 1);
216 todo_wine
ok(!ret
&& GetLastError() == ERROR_NOT_ENOUGH_MEMORY
,
217 "ret=%d error=%d\n", ret
, GetLastError());
219 dpa
= pDPA_Create(0);
220 ok(dpa
!= NULL
, "\n");
222 /* Set item with out of bound index */
223 ok(pDPA_SetPtr(dpa
, 1, (PVOID
)6), "\n");
224 /* Fill the created gap */
225 ok(pDPA_SetPtr(dpa
, 0, (PVOID
)5), "\n");
226 rc
=CheckDPA(dpa
, 0x56, &dw
);
227 ok(rc
, "dw=0x%x\n", dw
);
230 ret
= pDPA_InsertPtr(dpa
, 1, (PVOID
)1);
231 ok(ret
== 1, "ret=%d\n", ret
);
232 /* Append item using correct index */
233 ret
= pDPA_InsertPtr(dpa
, 3, (PVOID
)3);
234 ok(ret
== 3, "ret=%d\n", ret
);
235 /* Append item using out of bound index */
236 ret
= pDPA_InsertPtr(dpa
, 5, (PVOID
)2);
237 ok(ret
== 4, "ret=%d\n", ret
);
238 /* Append item using DPA_APPEND */
239 ret
= pDPA_InsertPtr(dpa
, DPA_APPEND
, (PVOID
)4);
240 ok(ret
== 5, "ret=%d\n", ret
);
242 rc
=CheckDPA(dpa
, 0x516324, &dw
);
243 ok(rc
, "dw=0x%x\n", dw
);
245 for(i
= 1; i
<= 6; i
++)
248 k
= pDPA_GetPtrIndex(dpa
, (PVOID
)(INT_PTR
)i
);
249 /* Linear searches should work on unsorted DPAs */
250 j
= pDPA_Search(dpa
, (PVOID
)(INT_PTR
)i
, 0, CB_CmpLT
, 0xdeadbeef, 0);
251 ok(j
== k
, "j=%d k=%d\n", j
, k
);
255 ok(pDPA_Sort(dpa
, CB_CmpGT
, 0xdeadbeef), "\n");
256 rc
=CheckDPA(dpa
, 0x654321, &dw
);
257 ok(rc
, "dw=0x%x\n", dw
);
259 /* Clone into a new DPA */
260 dpa2
= pDPA_Clone(dpa
, NULL
);
261 ok(dpa2
!= NULL
, "\n");
262 /* The old data should have been preserved */
263 rc
=CheckDPA(dpa2
, 0x654321, &dw2
);
264 ok(rc
, "dw=0x%x\n", dw2
);
265 ok(pDPA_Sort(dpa
, CB_CmpLT
, 0xdeadbeef), "\n");
267 /* Test if the DPA itself was really copied */
268 rc
=CheckDPA(dpa
, 0x123456, &dw
);
269 ok(rc
, "dw=0x%x\n", dw
);
270 rc
=CheckDPA(dpa2
, 0x654321, &dw2
);
271 ok(rc
, "dw2=0x%x\n", dw2
);
273 /* Clone into an old DPA */
274 p
= NULL
; SetLastError(ERROR_SUCCESS
);
275 p
= pDPA_Clone(dpa
, dpa3
);
276 ok(p
== dpa3
, "p=%p\n", p
);
277 rc
=CheckDPA(dpa3
, 0x123456, &dw3
);
278 ok(rc
, "dw3=0x%x\n", dw3
);
280 for(i
= 1; i
<= 6; i
++)
284 /* The array is in order so ptr == index+1 */
285 j
= pDPA_GetPtrIndex(dpa
, (PVOID
)(INT_PTR
)i
);
286 ok(j
+1 == i
, "j=%d i=%d\n", j
, i
);
287 j
= pDPA_Search(dpa
, (PVOID
)(INT_PTR
)i
, 0, CB_CmpLT
, 0xdeadbeef, DPAS_SORTED
);
288 ok(j
+1 == i
, "j=%d i=%d\n", j
, i
);
290 /* Linear searches respect iStart ... */
291 j
= pDPA_Search(dpa
, (PVOID
)(INT_PTR
)i
, i
+1, CB_CmpLT
, 0xdeadbeef, 0);
292 ok(j
== DPA_ERR
, "j=%d\n", j
);
293 /* ... but for a binary search it's ignored */
294 j
= pDPA_Search(dpa
, (PVOID
)(INT_PTR
)i
, i
+1, CB_CmpLT
, 0xdeadbeef, DPAS_SORTED
);
295 ok(j
+1 == i
, "j=%d i=%d\n", j
, i
);
298 /* Try to get the index of a nonexistent item */
299 i
= pDPA_GetPtrIndex(dpa
, (PVOID
)7);
300 ok(i
== DPA_ERR
, "i=%d\n", i
);
302 /* Try to delete out of bound indexes */
303 p
= pDPA_DeletePtr(dpa
, -1);
304 ok(p
== NULL
, "p=%p\n", p
);
305 p
= pDPA_DeletePtr(dpa
, 6);
306 ok(p
== NULL
, "p=%p\n", p
);
308 /* Delete the third item */
309 p
= pDPA_DeletePtr(dpa
, 2);
310 ok(p
== (PVOID
)3, "p=%p\n", p
);
311 rc
=CheckDPA(dpa
, 0x12456, &dw
);
312 ok(rc
, "dw=0x%x\n", dw
);
314 /* Check where to re-insert the deleted item */
315 i
= pDPA_Search(dpa
, (PVOID
)3, 0,
316 CB_CmpLT
, 0xdeadbeef, DPAS_SORTED
|DPAS_INSERTAFTER
);
317 ok(i
== 2, "i=%d\n", i
);
318 /* DPAS_INSERTBEFORE works just like DPAS_INSERTAFTER */
319 i
= pDPA_Search(dpa
, (PVOID
)3, 0,
320 CB_CmpLT
, 0xdeadbeef, DPAS_SORTED
|DPAS_INSERTBEFORE
);
321 ok(i
== 2, "i=%d\n", i
);
322 /* without DPAS_INSERTBEFORE/AFTER */
323 i
= pDPA_Search(dpa
, (PVOID
)3, 0,
324 CB_CmpLT
, 0xdeadbeef, DPAS_SORTED
);
325 ok(i
== -1, "i=%d\n", i
);
327 /* Re-insert the item */
328 ret
= pDPA_InsertPtr(dpa
, 2, (PVOID
)3);
329 ok(ret
== 2, "ret=%d i=%d\n", ret
, 2);
330 rc
=CheckDPA(dpa
, 0x123456, &dw
);
331 ok(rc
, "dw=0x%x\n", dw
);
333 /* When doing a binary search while claiming reverse order all indexes
335 for(i
= 0; i
< 6; i
++)
337 INT j
= pDPA_Search(dpa
, (PVOID
)(INT_PTR
)i
, 0, CB_CmpGT
, 0xdeadbeef,
338 DPAS_SORTED
|DPAS_INSERTBEFORE
);
339 ok(j
!= i
, "i=%d\n", i
);
342 /* Setting item with huge index should work */
343 ok(pDPA_SetPtr(dpa2
, 0x12345, (PVOID
)0xdeadbeef), "\n");
344 ret
= pDPA_GetPtrIndex(dpa2
, (PVOID
)0xdeadbeef);
345 ok(ret
== 0x12345, "ret=%d\n", ret
);
347 pDPA_DeleteAllPtrs(dpa2
);
348 rc
=CheckDPA(dpa2
, 0, &dw2
);
349 ok(rc
, "dw2=0x%x\n", dw2
);
354 static void test_DPA_Merge(void)
356 HDPA dpa
, dpa2
, dpa3
;
363 win_skip("DPA_Merge() not available\n");
367 dpa
= pDPA_Create(0);
368 dpa2
= pDPA_Create(0);
369 dpa3
= pDPA_Create(0);
371 ret
= pDPA_InsertPtr(dpa
, 0, (PVOID
)1);
372 ok(ret
== 0, "ret=%d\n", ret
);
373 ret
= pDPA_InsertPtr(dpa
, 1, (PVOID
)3);
374 ok(ret
== 1, "ret=%d\n", ret
);
375 ret
= pDPA_InsertPtr(dpa
, 2, (PVOID
)5);
376 ok(ret
== 2, "ret=%d\n", ret
);
378 rc
= CheckDPA(dpa
, 0x135, &dw
);
379 ok(rc
, "dw=0x%x\n", dw
);
381 for (i
= 0; i
< 6; i
++)
383 ret
= pDPA_InsertPtr(dpa2
, i
, (PVOID
)(6-i
));
384 ok(ret
== i
, "ret=%d\n", ret
);
385 ret
= pDPA_InsertPtr(dpa3
, i
, (PVOID
)(i
+1));
386 ok(ret
== i
, "ret=%d\n", ret
);
389 rc
= CheckDPA(dpa2
, 0x654321, &dw
);
390 ok(rc
, "dw=0x%x\n", dw
);
391 rc
= CheckDPA(dpa3
, 0x123456, &dw
);
392 ok(rc
, "dw=0x%x\n", dw
);
394 /* Delete all odd entries from dpa2 */
395 memset(nMessages
, 0, sizeof(nMessages
));
396 pDPA_Merge(dpa2
, dpa
, DPAM_INTERSECT
,
397 CB_CmpLT
, CB_MergeDeleteOddSrc
, 0xdeadbeef);
398 rc
= CheckDPA(dpa2
, 0x246, &dw
);
399 ok(rc
, "dw=0x%x\n", dw
);
401 expect(3, nMessages
[DPAMM_MERGE
]);
402 expect(3, nMessages
[DPAMM_DELETE
]);
403 expect(0, nMessages
[DPAMM_INSERT
]);
405 for (i
= 0; i
< 6; i
++)
407 ret
= pDPA_InsertPtr(dpa2
, i
, (PVOID
)(6-i
));
408 ok(ret
== i
, "ret=%d\n", ret
);
411 /* DPAM_INTERSECT - returning source while merging */
412 memset(nMessages
, 0, sizeof(nMessages
));
413 pDPA_Merge(dpa2
, dpa
, DPAM_INTERSECT
,
414 CB_CmpLT
, CB_MergeInsertSrc
, 0xdeadbeef);
415 rc
= CheckDPA(dpa2
, 0x135, &dw
);
416 ok(rc
, "dw=0x%x\n", dw
);
418 expect(3, nMessages
[DPAMM_MERGE
]);
419 expect(6, nMessages
[DPAMM_DELETE
]);
420 expect(0, nMessages
[DPAMM_INSERT
]);
423 pDPA_DeleteAllPtrs(dpa
);
424 pDPA_InsertPtr(dpa
, 0, (PVOID
)1);
425 pDPA_InsertPtr(dpa
, 1, (PVOID
)3);
426 pDPA_InsertPtr(dpa
, 2, (PVOID
)5);
427 pDPA_DeleteAllPtrs(dpa2
);
428 pDPA_InsertPtr(dpa2
, 0, (PVOID
)2);
429 pDPA_InsertPtr(dpa2
, 1, (PVOID
)4);
430 pDPA_InsertPtr(dpa2
, 2, (PVOID
)6);
432 memset(nMessages
, 0, sizeof(nMessages
));
433 pDPA_Merge(dpa2
, dpa
, DPAM_UNION
,
434 CB_CmpLT
, CB_MergeInsertSrc
, 0xdeadbeef);
435 rc
= CheckDPA(dpa2
, 0x123456, &dw
);
436 ok(rc
, "dw=0x%x\n", dw
);
438 expect(0, nMessages
[DPAMM_MERGE
]);
439 expect(0, nMessages
[DPAMM_DELETE
]);
440 expect(3, nMessages
[DPAMM_INSERT
]);
442 /* Merge dpa3 into dpa2 and dpa */
443 memset(nMessages
, 0, sizeof(nMessages
));
444 pDPA_Merge(dpa
, dpa3
, DPAM_UNION
|DPAM_SORTED
,
445 CB_CmpLT
, CB_MergeInsertSrc
, 0xdeadbeef);
446 expect(3, nMessages
[DPAMM_MERGE
]);
447 expect(0, nMessages
[DPAMM_DELETE
]);
448 expect(3, nMessages
[DPAMM_INSERT
]);
451 pDPA_DeleteAllPtrs(dpa2
);
452 pDPA_InsertPtr(dpa2
, 0, (PVOID
)2);
453 pDPA_InsertPtr(dpa2
, 1, (PVOID
)4);
454 pDPA_InsertPtr(dpa2
, 2, (PVOID
)6);
456 memset(nMessages
, 0, sizeof(nMessages
));
457 pDPA_Merge(dpa2
, dpa3
, DPAM_UNION
|DPAM_SORTED
,
458 CB_CmpLT
, CB_MergeInsertSrc
, 0xdeadbeef);
459 expect(3, nMessages
[DPAMM_MERGE
]);
460 expect(0, nMessages
[DPAMM_DELETE
]);
461 expect(3, nMessages
[DPAMM_INSERT
]);
463 rc
= CheckDPA(dpa
, 0x123456, &dw
);
464 ok(rc
, "dw=0x%x\n", dw
);
465 rc
= CheckDPA(dpa2
, 0x123456, &dw
);
467 broken(!rc
), /* win98 */
469 rc
= CheckDPA(dpa3
, 0x123456, &dw
);
470 ok(rc
, "dw=0x%x\n", dw
);
477 static void test_DPA_EnumCallback(void)
484 if(!pDPA_EnumCallback
)
486 win_skip("DPA_EnumCallback() not available\n");
490 dpa
= pDPA_Create(0);
492 for (i
= 0; i
< 6; i
++)
494 ret
= pDPA_InsertPtr(dpa
, i
, (PVOID
)(i
+1));
495 ok(ret
== i
, "ret=%d\n", ret
);
498 rc
= CheckDPA(dpa
, 0x123456, &dw
);
499 ok(rc
, "dw=0x%x\n", dw
);
502 /* test callback sets first 3 items to 7 */
503 pDPA_EnumCallback(dpa
, CB_EnumFirstThree
, dpa
);
504 rc
= CheckDPA(dpa
, 0x777456, &dw
);
505 ok(rc
, "dw=0x%x\n", dw
);
506 ok(nEnum
== 3, "nEnum=%d\n", nEnum
);
511 static void test_DPA_DestroyCallback(void)
516 if(!pDPA_DestroyCallback
)
518 win_skip("DPA_DestroyCallback() not available\n");
522 dpa
= pDPA_Create(0);
524 for (i
= 0; i
< 3; i
++)
526 ret
= pDPA_InsertPtr(dpa
, i
, (PVOID
)(i
+1));
527 ok(ret
== i
, "ret=%d\n", ret
);
531 pDPA_DestroyCallback(dpa
, CB_EnumFirstThree
, dpa
);
532 ok(nEnum
== 3, "nEnum=%d\n", nEnum
);
535 static void test_DPA_LoadStream(void)
537 static const WCHAR szStg
[] = { 'S','t','g',0 };
538 IStorage
* pStg
= NULL
;
539 IStream
* pStm
= NULL
;
548 hRes
= CoInitialize(NULL
);
551 ok(0, "hResult: %d\n", hRes
);
555 dwMode
= STGM_DIRECT
|STGM_CREATE
|STGM_READWRITE
|STGM_SHARE_EXCLUSIVE
;
556 hRes
= StgCreateDocfile(NULL
, dwMode
|STGM_DELETEONRELEASE
, 0, &pStg
);
559 hRes
= IStorage_CreateStream(pStg
, szStg
, dwMode
, 0, 0, &pStm
);
562 /* write less than header size */
564 hRes
= IStream_Seek(pStm
, li
, STREAM_SEEK_SET
, NULL
);
567 memset(&header
, 0, sizeof(header
));
569 uli
.QuadPart
= sizeof(header
)-1;
570 hRes
= IStream_SetSize(pStm
, uli
);
572 hRes
= IStream_Write(pStm
, &header
, sizeof(header
)-1, &written
);
574 written
-= sizeof(header
)-1;
578 hRes
= IStream_Seek(pStm
, li
, STREAM_SEEK_SET
, NULL
);
581 hRes
= pDPA_LoadStream(&dpa
, CB_Load
, pStm
, NULL
);
582 expect(E_FAIL
, hRes
);
584 /* check stream position after header read failed */
587 hRes
= IStream_Seek(pStm
, li
, STREAM_SEEK_CUR
, &uli
);
589 ok(uli
.QuadPart
== 0, "Expected to position reset\n");
591 /* write valid header for empty DPA */
592 header
.dwSize
= sizeof(header
);
598 hRes
= IStream_Seek(pStm
, li
, STREAM_SEEK_SET
, NULL
);
601 uli
.QuadPart
= sizeof(header
);
602 hRes
= IStream_SetSize(pStm
, uli
);
605 hRes
= IStream_Write(pStm
, &header
, sizeof(header
), &written
);
607 written
-= sizeof(header
);
611 hRes
= IStream_Seek(pStm
, li
, STREAM_SEEK_SET
, NULL
);
614 hRes
= pDPA_LoadStream(&dpa
, CB_Load
, pStm
, NULL
);
617 /* try with altered dwData2 field */
618 header
.dwSize
= sizeof(header
);
623 hRes
= IStream_Seek(pStm
, li
, STREAM_SEEK_SET
, NULL
);
625 hRes
= IStream_Write(pStm
, &header
, sizeof(header
), &written
);
627 written
-= sizeof(header
);
631 hRes
= IStream_Seek(pStm
, li
, STREAM_SEEK_SET
, NULL
);
634 hRes
= pDPA_LoadStream(&dpa
, CB_Load
, pStm
, (void*)0xdeadbeef);
635 expect(E_FAIL
, hRes
);
637 ret
= IStream_Release(pStm
);
638 ok(!ret
, "ret=%d\n", ret
);
640 ret
= IStorage_Release(pStg
);
641 ok(!ret
, "ret=%d\n", ret
);
646 static void test_dpa_stream(void)
653 static const WCHAR szStg
[] = { 'S','t','g',0 };
654 IStorage
* pStg
= NULL
;
655 IStream
* pStm
= NULL
;
656 LARGE_INTEGER liZero
;
661 win_skip("DPA_SaveStream() not available. Skipping stream tests.\n");
665 hRes
= CoInitialize(NULL
);
668 ok(0, "hResult: %d\n", hRes
);
672 dpa
= pDPA_Create(0);
674 for (i
= 0; i
< 6; i
++)
676 ret
= pDPA_InsertPtr(dpa
, i
, (PVOID
)(i
+1));
677 ok(ret
== i
, "ret=%d\n", ret
);
680 dwMode
= STGM_DIRECT
|STGM_CREATE
|STGM_READWRITE
|STGM_SHARE_EXCLUSIVE
;
681 hRes
= StgCreateDocfile(NULL
, dwMode
|STGM_DELETEONRELEASE
, 0, &pStg
);
682 ok(hRes
== S_OK
, "hRes=0x%x\n", hRes
);
684 hRes
= IStorage_CreateStream(pStg
, szStg
, dwMode
, 0, 0, &pStm
);
685 ok(hRes
== S_OK
, "hRes=0x%x\n", hRes
);
687 hRes
= pDPA_SaveStream(dpa
, CB_Save
, pStm
, (void*)0xdeadbeef);
688 todo_wine
ok(hRes
== S_OK
, "hRes=0x%x\n", hRes
);
692 hRes
= IStream_Seek(pStm
, liZero
, STREAM_SEEK_SET
, NULL
);
693 ok(hRes
== S_OK
, "hRes=0x%x\n", hRes
);
694 hRes
= pDPA_LoadStream(&dpa
, CB_Load
, pStm
, (void*)0xdeadbeef);
697 ok(hRes
== S_OK
, "hRes=0x%x\n", hRes
);
698 rc
= CheckDPA(dpa
, 0x123456, &dw
);
699 ok(rc
, "dw=0x%x\n", dw
);
703 ret
= IStream_Release(pStm
);
704 ok(!ret
, "ret=%d\n", ret
);
706 ret
= IStorage_Release(pStg
);
707 ok(!ret
, "ret=%d\n", ret
);
716 hcomctl32
= GetModuleHandleA("comctl32.dll");
718 if(!InitFunctionPtrs(hcomctl32
))
720 win_skip("Needed functions are not available\n");
726 test_DPA_EnumCallback();
727 test_DPA_DestroyCallback();
728 test_DPA_LoadStream();