ole32: Add a test for anti monikers. Fix AntiMonikerImpl_Hash and AntiMonikerROTDataI...
[wine] / dlls / ole32 / tests / moniker.c
1 /*
2  * Moniker Tests
3  *
4  * Copyright 2004 Robert Shearman
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., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
19  */
20
21 #define _WIN32_DCOM
22 #define COBJMACROS
23
24 #include <stdarg.h>
25
26 #include "windef.h"
27 #include "winbase.h"
28 #include "objbase.h"
29 #include "comcat.h"
30
31 #include "wine/test.h"
32
33 #define ok_ole_success(hr, func) ok(hr == S_OK, #func " failed with error 0x%08lx\n", hr)
34 #define COUNTOF(x) (sizeof(x) / sizeof(x[0]))
35
36 static const WCHAR wszFileName1[] = {'c',':','\\','w','i','n','d','o','w','s','\\','t','e','s','t','1','.','d','o','c',0};
37 static const WCHAR wszFileName2[] = {'c',':','\\','w','i','n','d','o','w','s','\\','t','e','s','t','2','.','d','o','c',0};
38
39 static int count_moniker_matches(IBindCtx * pbc, IEnumMoniker * spEM)
40 {
41     IMoniker * spMoniker;
42     int monCnt=0, matchCnt=0;
43
44     while ((IEnumMoniker_Next(spEM, 1, &spMoniker, NULL)==S_OK))
45     {
46         HRESULT hr;
47         WCHAR * szDisplayn;
48         monCnt++;
49         hr=IMoniker_GetDisplayName(spMoniker, pbc, NULL, &szDisplayn);
50         if (SUCCEEDED(hr))
51         {
52             if (!lstrcmpW(szDisplayn, wszFileName1) || !lstrcmpW(szDisplayn, wszFileName2))
53                 matchCnt++;
54             CoTaskMemFree(szDisplayn);
55         }
56     }
57     trace("Total number of monikers is %i\n", monCnt);
58     return matchCnt;
59 }
60
61 static void test_MkParseDisplayName(void)
62 {
63     IBindCtx * pbc = NULL;
64     HRESULT hr;
65     IMoniker * pmk  = NULL;
66     IMoniker * pmk1 = NULL;
67     IMoniker * pmk2 = NULL;
68     ULONG eaten;
69     int matchCnt;
70     IUnknown * object = NULL;
71
72     IUnknown *lpEM1;
73
74     IEnumMoniker *spEM1  = NULL;
75     IEnumMoniker *spEM2  = NULL;
76     IEnumMoniker *spEM3  = NULL;
77
78     DWORD pdwReg1=0;
79     DWORD grflags=0;
80     DWORD pdwReg2=0;
81     IRunningObjectTable * pprot=NULL;
82
83     /* CLSID of My Computer */
84     static const WCHAR wszDisplayName[] = {'c','l','s','i','d',':',
85         '2','0','D','0','4','F','E','0','-','3','A','E','A','-','1','0','6','9','-','A','2','D','8','-','0','8','0','0','2','B','3','0','3','0','9','D',':',0};
86
87     hr = CreateBindCtx(0, &pbc);
88     ok_ole_success(hr, CreateBindCtx);
89
90     hr = MkParseDisplayName(pbc, wszDisplayName, &eaten, &pmk);
91     todo_wine { ok_ole_success(hr, MkParseDisplayName); }
92
93     if (object)
94     {
95         hr = IMoniker_BindToObject(pmk, pbc, NULL, &IID_IUnknown, (LPVOID*)&object);
96         ok_ole_success(hr, IMoniker_BindToObject);
97
98         IUnknown_Release(object);
99     }
100     IBindCtx_Release(pbc);
101
102     /* Test the EnumMoniker interface */
103     hr = CreateBindCtx(0, &pbc);
104     ok_ole_success(hr, CreateBindCtx);
105
106     hr = CreateFileMoniker(wszFileName1, &pmk1);
107     ok(hr==0, "CreateFileMoniker for file hr=%08lx\n", hr);
108     hr = CreateFileMoniker(wszFileName2, &pmk2);
109     ok(hr==0, "CreateFileMoniker for file hr=%08lx\n", hr);
110     hr = IBindCtx_GetRunningObjectTable(pbc, &pprot);
111     ok(hr==0, "IBindCtx_GetRunningObjectTable hr=%08lx\n", hr);
112
113     /* Check EnumMoniker before registering */
114     hr = IRunningObjectTable_EnumRunning(pprot, &spEM1);
115     ok(hr==0, "IRunningObjectTable_EnumRunning hr=%08lx\n", hr);
116     hr = IEnumMoniker_QueryInterface(spEM1, &IID_IUnknown, (void*) &lpEM1);
117     /* Register a couple of Monikers and check is ok */
118     ok(hr==0, "IEnumMoniker_QueryInterface hr %08lx %p\n", hr, lpEM1);
119     hr = MK_E_NOOBJECT;
120     
121     matchCnt = count_moniker_matches(pbc, spEM1);
122     trace("Number of matches is %i\n", matchCnt);
123
124     grflags= grflags | ROTFLAGS_REGISTRATIONKEEPSALIVE;
125     hr = IRunningObjectTable_Register(pprot, grflags, lpEM1, pmk1, &pdwReg1);
126     ok(hr==0, "IRunningObjectTable_Register hr=%08lx %p %08lx %p %p %ld\n",
127         hr, pprot, grflags, lpEM1, pmk1, pdwReg1);
128
129     trace("IROT::Register\n");
130     grflags=0;
131     grflags= grflags | ROTFLAGS_REGISTRATIONKEEPSALIVE;
132     hr = IRunningObjectTable_Register(pprot, grflags, lpEM1, pmk2, &pdwReg2);
133     ok(hr==0, "IRunningObjectTable_Register hr=%08lx %p %08lx %p %p %ld\n", hr,
134        pprot, grflags, lpEM1, pmk2, pdwReg2);
135
136     hr = IRunningObjectTable_EnumRunning(pprot, &spEM2);
137     ok(hr==0, "IRunningObjectTable_EnumRunning hr=%08lx\n", hr);
138
139     matchCnt = count_moniker_matches(pbc, spEM2);
140     ok(matchCnt==2, "Number of matches should be equal to 2 not %i\n", matchCnt);
141
142     trace("IEnumMoniker::Clone\n");
143     IEnumMoniker_Clone(spEM2, &spEM3);
144
145     matchCnt = count_moniker_matches(pbc, spEM3);
146     ok(matchCnt==0, "Number of matches should be equal to 0 not %i\n", matchCnt);
147     trace("IEnumMoniker::Reset\n");
148     IEnumMoniker_Reset(spEM3);
149
150     matchCnt = count_moniker_matches(pbc, spEM3);
151     ok(matchCnt==2, "Number of matches should be equal to 2 not %i\n", matchCnt);
152
153     IRunningObjectTable_Revoke(pprot,pdwReg1);
154     IRunningObjectTable_Revoke(pprot,pdwReg2);
155     IEnumMoniker_Release(spEM1);
156     IEnumMoniker_Release(spEM1);
157     IEnumMoniker_Release(spEM2);
158     IEnumMoniker_Release(spEM3);
159     IMoniker_Release(pmk1);
160     IMoniker_Release(pmk2);
161     IRunningObjectTable_Release(pprot);
162
163     IBindCtx_Release(pbc);
164 }
165
166 static const LARGE_INTEGER llZero;
167
168 static const BYTE expected_class_moniker_marshal_data[] =
169 {
170     0x4d,0x45,0x4f,0x57,0x04,0x00,0x00,0x00,
171     0x0f,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
172     0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x46,
173     0x1a,0x03,0x00,0x00,0x00,0x00,0x00,0x00,
174     0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x46,
175     0x00,0x00,0x00,0x00,0x14,0x00,0x00,0x00,
176     0x05,0xe0,0x02,0x00,0x00,0x00,0x00,0x00,
177     0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x46,
178     0x00,0x00,0x00,0x00,
179 };
180
181 static const BYTE expected_class_moniker_saved_data[] =
182 {
183      0x05,0xe0,0x02,0x00,0x00,0x00,0x00,0x00,
184      0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x46,
185      0x00,0x00,0x00,0x00,
186 };
187
188 static const BYTE expected_class_moniker_comparison_data[] =
189 {
190      0x1a,0x03,0x00,0x00,0x00,0x00,0x00,0x00,
191      0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x46,
192      0x05,0xe0,0x02,0x00,0x00,0x00,0x00,0x00,
193      0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x46,
194 };
195
196 static const BYTE expected_item_moniker_comparison_data[] =
197 {
198      0x04,0x03,0x00,0x00,0x00,0x00,0x00,0x00,
199      0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x46,
200      0x21,0x00,0x54,0x00,0x45,0x00,0x53,0x00,
201      0x54,0x00,0x00,0x00,
202 };
203
204 static const BYTE expected_item_moniker_saved_data[] =
205 {
206      0x02,0x00,0x00,0x00,0x21,0x00,0x05,0x00,
207      0x00,0x00,0x54,0x65,0x73,0x74,0x00,
208 };
209
210 static const BYTE expected_item_moniker_marshal_data[] =
211 {
212      0x4d,0x45,0x4f,0x57,0x04,0x00,0x00,0x00,
213      0x0f,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
214      0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x46,
215      0x04,0x03,0x00,0x00,0x00,0x00,0x00,0x00,
216      0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x46,
217      0x00,0x00,0x00,0x00,0x36,0x00,0x00,0x00,
218      0x02,0x00,0x00,0x00,0x21,0x00,0x05,0x00,
219      0x00,0x00,0x54,0x65,0x73,0x74,0x00,
220 };
221
222 static const BYTE expected_anti_moniker_marshal_data[] =
223 {
224     0x4d,0x45,0x4f,0x57,0x04,0x00,0x00,0x00,
225     0x0f,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
226     0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x46,
227     0x05,0x03,0x00,0x00,0x00,0x00,0x00,0x00,
228     0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x46,
229     0x00,0x00,0x00,0x00,0x14,0x00,0x00,0x00,
230     0x01,0x00,0x00,0x00,
231 };
232
233 static const BYTE expected_anti_moniker_saved_data[] =
234 {
235     0x01,0x00,0x00,0x00,
236 };
237
238 static const BYTE expected_anti_moniker_comparison_data[] =
239 {
240     0x05,0x03,0x00,0x00,0x00,0x00,0x00,0x00,
241     0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x46,
242     0x01,0x00,0x00,0x00,
243 };
244
245 static void test_moniker(
246     const char *testname, IMoniker *moniker,
247     const BYTE *expected_moniker_marshal_data, size_t sizeof_expected_moniker_marshal_data,
248     const BYTE *expected_moniker_saved_data, size_t sizeof_expected_moniker_saved_data,
249     const BYTE *expected_moniker_comparison_data, size_t sizeof_expected_moniker_comparison_data)
250 {
251     IStream * stream;
252     IROTData * rotdata;
253     HRESULT hr;
254     HGLOBAL hglobal;
255     LPBYTE moniker_data;
256     DWORD moniker_size;
257     DWORD i;
258     BOOL same = TRUE;
259     BYTE buffer[128];
260
261     /* IROTData::GetComparisonData test */
262
263     hr = IMoniker_QueryInterface(moniker, &IID_IROTData, (void **)&rotdata);
264     ok_ole_success(hr, IMoniker_QueryInterface(IID_IROTData));
265
266     hr = IROTData_GetComparisonData(rotdata, buffer, sizeof(buffer), &moniker_size);
267     ok_ole_success(hr, IROTData_GetComparisonData);
268
269     if (hr != S_OK) moniker_size = 0;
270
271     /* first check we have the right amount of data */
272     ok(moniker_size == sizeof_expected_moniker_comparison_data,
273         "%s: Size of comparison data differs (expected %d, actual %ld)\n",
274         testname, sizeof_expected_moniker_comparison_data, moniker_size);
275
276     /* then do a byte-by-byte comparison */
277     for (i = 0; i < min(moniker_size, sizeof_expected_moniker_comparison_data); i++)
278     {
279         if (expected_moniker_comparison_data[i] != buffer[i])
280         {
281             same = FALSE;
282             break;
283         }
284     }
285
286     ok(same, "%s: Comparison data differs\n", testname);
287     if (!same)
288     {
289         for (i = 0; i < moniker_size; i++)
290         {
291             if (i % 8 == 0) trace("     ");
292             trace("0x%02x,", buffer[i]);
293             if (i % 8 == 7) trace("\n");
294         }
295         trace("\n");
296     }
297
298     IROTData_Release(rotdata);
299   
300     hr = CreateStreamOnHGlobal(NULL, TRUE, &stream);
301   
302     /* Saving */
303
304     hr = IMoniker_Save(moniker, stream, TRUE);
305     ok_ole_success(hr, IMoniker_Save);
306
307     hr = GetHGlobalFromStream(stream, &hglobal);
308     ok_ole_success(hr, GetHGlobalFromStream);
309
310     moniker_size = GlobalSize(hglobal);
311
312     moniker_data = GlobalLock(hglobal);
313
314     /* first check we have the right amount of data */
315     ok(moniker_size == sizeof_expected_moniker_saved_data,
316         "%s: Size of saved data differs (expected %d, actual %ld)\n",
317         testname, sizeof_expected_moniker_saved_data, moniker_size);
318
319     /* then do a byte-by-byte comparison */
320     for (i = 0; i < min(moniker_size, sizeof_expected_moniker_saved_data); i++)
321     {
322         if (expected_moniker_saved_data[i] != moniker_data[i])
323         {
324             same = FALSE;
325             break;
326         }
327     }
328
329     ok(same, "%s: Saved data differs\n", testname);
330     if (!same)
331     {
332         for (i = 0; i < moniker_size; i++)
333         {
334             if (i % 8 == 0) trace("     ");
335             trace("0x%02x,", moniker_data[i]);
336             if (i % 8 == 7) trace("\n");
337         }
338         trace("\n");
339     }
340
341     GlobalUnlock(hglobal);
342
343     IStream_Release(stream);
344
345     /* Marshaling tests */
346
347     hr = CreateStreamOnHGlobal(NULL, TRUE, &stream);
348     ok_ole_success(hr, CreateStreamOnHGlobal);
349
350     hr = CoMarshalInterface(stream, &IID_IMoniker, (IUnknown *)moniker, MSHCTX_INPROC, NULL, MSHLFLAGS_NORMAL);
351     ok_ole_success(hr, CoMarshalInterface);
352
353     hr = GetHGlobalFromStream(stream, &hglobal);
354     ok_ole_success(hr, GetHGlobalFromStream);
355
356     moniker_size = GlobalSize(hglobal);
357
358     moniker_data = GlobalLock(hglobal);
359
360     /* first check we have the right amount of data */
361     ok(moniker_size == sizeof_expected_moniker_marshal_data,
362         "%s: Size of marshaled data differs (expected %d, actual %ld)\n",
363         testname, sizeof_expected_moniker_marshal_data, moniker_size);
364
365     /* then do a byte-by-byte comparison */
366     for (i = 0; i < min(moniker_size, sizeof_expected_moniker_marshal_data); i++)
367     {
368         if (expected_moniker_marshal_data[i] != moniker_data[i])
369         {
370             same = FALSE;
371             break;
372         }
373     }
374
375     ok(same, "%s: Marshaled data differs\n", testname);
376     if (!same)
377     {
378         for (i = 0; i < moniker_size; i++)
379         {
380             if (i % 8 == 0) trace("     ");
381             trace("0x%02x,", moniker_data[i]);
382             if (i % 8 == 7) trace("\n");
383         }
384         trace("\n");
385     }
386
387     GlobalUnlock(hglobal);
388
389     IStream_Seek(stream, llZero, STREAM_SEEK_SET, NULL);
390     hr = CoReleaseMarshalData(stream);
391     ok_ole_success(hr, CoReleaseMarshalData);
392
393     IStream_Release(stream);
394 }
395
396 static void test_class_moniker(void)
397 {
398     HRESULT hr;
399     IMoniker *moniker;
400     DWORD moniker_type;
401     DWORD hash;
402
403     hr = CreateClassMoniker(&CLSID_StdComponentCategoriesMgr, &moniker);
404     todo_wine ok_ole_success(hr, CreateClassMoniker);
405     if (!moniker) return;
406
407     test_moniker("class moniker", moniker, 
408         expected_class_moniker_marshal_data, sizeof(expected_class_moniker_marshal_data),
409         expected_class_moniker_saved_data, sizeof(expected_class_moniker_saved_data),
410         expected_class_moniker_comparison_data, sizeof(expected_class_moniker_comparison_data));
411
412     /* Hashing */
413
414     hr = IMoniker_Hash(moniker, &hash);
415     ok_ole_success(hr, IMoniker_Hash);
416
417     ok(hash == CLSID_StdComponentCategoriesMgr.Data1,
418         "Hash value != Data1 field of clsid, instead was 0x%08lx\n",
419         hash);
420
421     /* IsSystemMoniker test */
422
423     hr = IMoniker_IsSystemMoniker(moniker, &moniker_type);
424     ok_ole_success(hr, IMoniker_IsSystemMoniker);
425
426     ok(moniker_type == MKSYS_CLASSMONIKER,
427         "dwMkSys != MKSYS_CLASSMONIKER, instead was 0x%08lx",
428         moniker_type);
429
430     IMoniker_Release(moniker);
431 }
432
433 static void test_file_moniker(WCHAR* path)
434 {
435     IStream *stream;
436     IMoniker *moniker1 = NULL, *moniker2 = NULL;
437     HRESULT hr;
438
439     hr = CreateFileMoniker(path, &moniker1);
440     ok_ole_success(hr, CreateFileMoniker); 
441
442     hr = CreateStreamOnHGlobal(NULL, TRUE, &stream);
443
444     /* Marshal */
445     hr = CoMarshalInterface(stream, &IID_IMoniker, (IUnknown *)moniker1, MSHCTX_INPROC, NULL, MSHLFLAGS_NORMAL);
446     ok_ole_success(hr, CoMarshalInterface);
447     
448     /* Rewind */
449     hr = IStream_Seek(stream, llZero, STREAM_SEEK_SET, NULL);
450     ok_ole_success(hr, IStream_Seek);
451
452     /* Unmarshal */
453     hr = CoUnmarshalInterface(stream, &IID_IMoniker, (void**)&moniker2);
454     ok_ole_success(hr, CoUnmarshalInterface);
455
456     hr = IMoniker_IsEqual(moniker1, moniker2);
457     ok_ole_success(hr, IsEqual);
458
459     IStream_Release(stream);
460     if (moniker1) 
461         IMoniker_Release(moniker1);
462     if (moniker2) 
463         IMoniker_Release(moniker2);
464 }
465
466 static void test_file_monikers(void)
467 {
468     static WCHAR wszFile[][30] = {
469         {'\\', 'w','i','n','d','o','w','s','\\','s','y','s','t','e','m','\\','t','e','s','t','1','.','d','o','c',0},
470         {'\\', 'a','b','c','d','e','f','g','\\','h','i','j','k','l','\\','m','n','o','p','q','r','s','t','u','.','m','n','o',0},
471         /* These map to themselves in Windows-1252 & 932 (Shift-JIS) */
472         {0xc0, 0xc1, 0xc2, 0xc3, 0xc4, 0xc5, 0xc6, 0xc7, 0xc8, 0xc9, 0xca, 0xcb, 0xcc, 0xcd, 0xce, 0xcf, 0xd0, 0},
473         /* U+2020 = DAGGER     = 0x86 (1252) = 0x813f (932)
474          * U+20AC = EURO SIGN  = 0x80 (1252) =  undef (932)
475          * U+0100 .. = Latin extended-A
476          */ 
477         {0x20ac, 0x2020, 0x100, 0x101, 0x102, 0x103, 0x104, 0x105, 0x106, 0x107, 0x108, 0x109, 0x10a, 0x10b, 0x10c,  0},
478         };
479
480     int i; 
481
482     trace("ACP is %u\n", GetACP());
483
484     for (i = 0; i < COUNTOF(wszFile); ++i)
485     {
486         int j ;
487         for (j = lstrlenW(wszFile[i]); j > 0; --j)
488         {
489             wszFile[i][j] = 0;
490             test_file_moniker(wszFile[i]);
491         }
492     }
493 }
494
495 static void test_item_moniker(void)
496 {
497     HRESULT hr;
498     IMoniker *moniker;
499     DWORD moniker_type;
500     DWORD hash;
501     static const WCHAR wszDelimeter[] = {'!',0};
502     static const WCHAR wszObjectName[] = {'T','e','s','t',0};
503
504     hr = CreateItemMoniker(wszDelimeter, wszObjectName, &moniker);
505     ok_ole_success(hr, CreateItemMoniker);
506
507     test_moniker("item moniker", moniker, 
508         expected_item_moniker_marshal_data, sizeof(expected_item_moniker_marshal_data),
509         expected_item_moniker_saved_data, sizeof(expected_item_moniker_saved_data),
510         expected_item_moniker_comparison_data, sizeof(expected_item_moniker_comparison_data));
511
512     /* Hashing */
513
514     hr = IMoniker_Hash(moniker, &hash);
515     ok_ole_success(hr, IMoniker_Hash);
516
517     todo_wine
518     ok(hash == 0x73c,
519         "Hash value != 0x73c, instead was 0x%08lx\n",
520         hash);
521
522     /* IsSystemMoniker test */
523
524     hr = IMoniker_IsSystemMoniker(moniker, &moniker_type);
525     ok_ole_success(hr, IMoniker_IsSystemMoniker);
526
527     ok(moniker_type == MKSYS_ITEMMONIKER,
528         "dwMkSys != MKSYS_ITEMMONIKER, instead was 0x%08lx",
529         moniker_type);
530
531     IMoniker_Release(moniker);
532 }
533
534 static void test_anti_moniker(void)
535 {
536     HRESULT hr;
537     IMoniker *moniker;
538     IMoniker *inverse;
539     DWORD moniker_type;
540     DWORD hash;
541
542     hr = CreateAntiMoniker(&moniker);
543     ok_ole_success(hr, CreateAntiMoniker);
544     if (!moniker) return;
545
546     test_moniker("anti moniker", moniker, 
547         expected_anti_moniker_marshal_data, sizeof(expected_anti_moniker_marshal_data),
548         expected_anti_moniker_saved_data, sizeof(expected_anti_moniker_saved_data),
549         expected_anti_moniker_comparison_data, sizeof(expected_anti_moniker_comparison_data));
550
551     /* Hashing */
552     hr = IMoniker_Hash(moniker, &hash);
553     ok_ole_success(hr, IMoniker_Hash);
554     ok(hash == 0x80000001,
555         "Hash value != 0x80000001, instead was 0x%08lx\n",
556         hash);
557
558     /* IsSystemMoniker test */
559     hr = IMoniker_IsSystemMoniker(moniker, &moniker_type);
560     ok_ole_success(hr, IMoniker_IsSystemMoniker);
561     ok(moniker_type == MKSYS_ANTIMONIKER,
562         "dwMkSys != MKSYS_ANTIMONIKER, instead was 0x%08lx",
563         moniker_type);
564
565     hr = IMoniker_Inverse(moniker, &inverse);
566     ok(hr == MK_E_NOINVERSE, "IMoniker_Inverse should have returned MK_E_NOINVERSE instead of 0x%08lx\n", hr);
567     ok(inverse == NULL, "inverse should have been set to NULL instead of %p\n", inverse);
568
569     IMoniker_Release(moniker);
570 }
571
572 START_TEST(moniker)
573 {
574     CoInitializeEx(NULL, COINIT_APARTMENTTHREADED);
575
576     test_MkParseDisplayName();
577     test_class_moniker();
578     test_file_monikers();
579     test_item_moniker();
580     test_anti_moniker();
581
582     /* FIXME: test moniker creation funcs and parsing other moniker formats */
583
584     CoUninitialize();
585 }