msxml3/tests: Add test for parsing from IStream.
[wine] / dlls / devenum / mediacatenum.c
1 /*
2  *      IEnumMoniker implementation for DEVENUM.dll
3  *
4  * Copyright (C) 2002 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., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
19  *
20  * NOTES ON THIS FILE:
21  * - Implements IEnumMoniker interface which enumerates through moniker
22  *   objects created from HKEY_CLASSES/CLSID/{DEVICE_CLSID}/Instance
23  */
24
25 #include "devenum_private.h"
26 #include "oleauto.h"
27 #include "ocidl.h"
28
29 #include "wine/debug.h"
30
31 WINE_DEFAULT_DEBUG_CHANNEL(devenum);
32
33 static ULONG WINAPI DEVENUM_IEnumMoniker_AddRef(LPENUMMONIKER iface);
34 static ULONG WINAPI DEVENUM_IMediaCatMoniker_AddRef(LPMONIKER iface);
35 static ULONG WINAPI DEVENUM_IPropertyBag_AddRef(LPPROPERTYBAG iface);
36
37 typedef struct
38 {
39     const IPropertyBagVtbl *lpVtbl;
40     LONG ref;
41     HKEY hkey;
42 } RegPropBagImpl;
43
44
45 static HRESULT WINAPI DEVENUM_IPropertyBag_QueryInterface(
46     LPPROPERTYBAG iface,
47     REFIID riid,
48     LPVOID *ppvObj)
49 {
50     RegPropBagImpl *This = (RegPropBagImpl *)iface;
51     TRACE("\n\tIID:\t%s\n",debugstr_guid(riid));
52
53     if (This == NULL || ppvObj == NULL) return E_POINTER;
54
55     if (IsEqualGUID(riid, &IID_IUnknown) ||
56         IsEqualGUID(riid, &IID_IPropertyBag))
57     {
58         *ppvObj = (LPVOID)iface;
59         DEVENUM_IPropertyBag_AddRef(iface);
60         return S_OK;
61     }
62
63     FIXME("- no interface\n\tIID:\t%s\n", debugstr_guid(riid));
64     return E_NOINTERFACE;
65 }
66
67 /**********************************************************************
68  * DEVENUM_IPropertyBag_AddRef (also IUnknown)
69  */
70 static ULONG WINAPI DEVENUM_IPropertyBag_AddRef(LPPROPERTYBAG iface)
71 {
72     RegPropBagImpl *This = (RegPropBagImpl *)iface;
73     TRACE("\n");
74
75     return InterlockedIncrement(&This->ref);
76 }
77
78 /**********************************************************************
79  * DEVENUM_IPropertyBag_Release (also IUnknown)
80  */
81 static ULONG WINAPI DEVENUM_IPropertyBag_Release(LPPROPERTYBAG iface)
82 {
83     RegPropBagImpl *This = (RegPropBagImpl *)iface;
84     ULONG ref;
85
86     TRACE("\n");
87
88     ref = InterlockedDecrement(&This->ref);
89     if (ref == 0) {
90         RegCloseKey(This->hkey);
91         CoTaskMemFree(This);
92         DEVENUM_UnlockModule();
93     }
94     return ref;
95 }
96
97 static HRESULT WINAPI DEVENUM_IPropertyBag_Read(
98     LPPROPERTYBAG iface,
99     LPCOLESTR pszPropName,
100     VARIANT* pVar,
101     IErrorLog* pErrorLog)
102 {
103     LPVOID pData = NULL;
104     DWORD received;
105     DWORD type = 0;
106     RegPropBagImpl *This = (RegPropBagImpl *)iface;
107     HRESULT res = S_OK;
108     LONG reswin32;
109
110     TRACE("(%p)->(%s, %p, %p)\n", This, debugstr_w(pszPropName), pVar, pErrorLog);
111
112     if (!pszPropName || !pVar)
113         return E_POINTER;
114
115     reswin32 = RegQueryValueExW(This->hkey, pszPropName, NULL, NULL, NULL, &received);
116     res = HRESULT_FROM_WIN32(reswin32);
117
118     if (SUCCEEDED(res))
119     {
120         pData = HeapAlloc(GetProcessHeap(), 0, received);
121
122         /* work around a GCC bug that occurs here unless we use the reswin32 variable as well */
123         reswin32 = RegQueryValueExW(This->hkey, pszPropName, NULL, &type, (LPBYTE)pData, &received);
124         res = HRESULT_FROM_WIN32(reswin32);
125     }
126
127     if (SUCCEEDED(res))
128     {
129         res = E_INVALIDARG; /* assume we cannot coerce into right type */
130
131         TRACE("Read %d bytes (%s)\n", received, type == REG_SZ ? debugstr_w((LPWSTR)pData) : "binary data");
132
133         switch (type)
134         {
135         case REG_SZ:
136             switch (V_VT(pVar))
137             {
138             case VT_LPWSTR:
139                 V_UNION(pVar, bstrVal) = CoTaskMemAlloc(received);
140                 memcpy(V_UNION(pVar, bstrVal), (LPWSTR)pData, received);
141                 res = S_OK;
142                 break;
143             case VT_EMPTY:
144                 V_VT(pVar) = VT_BSTR;
145             /* fall through */
146             case VT_BSTR:
147                 V_UNION(pVar, bstrVal) = SysAllocStringLen((LPWSTR)pData, received/sizeof(WCHAR) - 1);
148                 res = S_OK;
149                 break;
150             }
151             break;
152         case REG_DWORD:
153             TRACE("REG_DWORD: %x\n", *(DWORD *)pData);
154             switch (V_VT(pVar))
155             {
156             case VT_EMPTY:
157                 V_VT(pVar) = VT_I4;
158                 /* fall through */
159             case VT_I4:
160             case VT_UI4:
161                 V_UNION(pVar, ulVal) = *(DWORD *)pData;
162                 res = S_OK;
163                 break;
164             }
165             break;
166         case REG_BINARY:
167             {
168                 SAFEARRAYBOUND bound;
169                 void * pArrayElements;
170                 bound.lLbound = 0;
171                 bound.cElements = received;
172                 TRACE("REG_BINARY: len = %d\n", received);
173                 switch (V_VT(pVar))
174                 {
175                 case VT_EMPTY:
176                     V_VT(pVar) = VT_ARRAY | VT_UI1;
177                     /* fall through */
178                 case VT_ARRAY | VT_UI1:
179                     if (!(V_UNION(pVar, parray) = SafeArrayCreate(VT_UI1, 1, &bound)))
180                         res = E_OUTOFMEMORY;
181                     else
182                         res = S_OK;
183                     break;
184                 }
185
186                 if (res == E_INVALIDARG)
187                     break;
188
189                 res = SafeArrayAccessData(V_UNION(pVar, parray), &pArrayElements);
190                 if (FAILED(res))
191                     break;
192
193                 CopyMemory(pArrayElements, pData, received);
194                 res = SafeArrayUnaccessData(V_UNION(pVar, parray));
195                 break;
196             }
197         }
198         if (res == E_INVALIDARG)
199             FIXME("Variant type %x not supported for regtype %x\n", V_VT(pVar), type);
200     }
201
202     HeapFree(GetProcessHeap(), 0, pData);
203
204     TRACE("<- %x\n", res);
205     return res;
206 }
207
208 static HRESULT WINAPI DEVENUM_IPropertyBag_Write(
209     LPPROPERTYBAG iface,
210     LPCOLESTR pszPropName,
211     VARIANT* pVar)
212 {
213     RegPropBagImpl *This = (RegPropBagImpl *)iface;
214     LPVOID lpData = NULL;
215     DWORD cbData = 0;
216     DWORD dwType = 0;
217     HRESULT res = S_OK;
218
219     TRACE("(%p)->(%s, %p)\n", This, debugstr_w(pszPropName), pVar);
220
221     switch (V_VT(pVar))
222     {
223     case VT_BSTR:
224         TRACE("writing %s\n", debugstr_w(V_UNION(pVar, bstrVal)));
225         lpData = (LPVOID)V_UNION(pVar, bstrVal);
226         dwType = REG_SZ;
227         cbData = (lstrlenW(V_UNION(pVar, bstrVal)) + 1) * sizeof(WCHAR);
228         break;
229     case VT_I4:
230     case VT_UI4:
231         TRACE("writing %u\n", V_UNION(pVar, ulVal));
232         lpData = (LPVOID)&V_UNION(pVar, ulVal);
233         dwType = REG_DWORD;
234         cbData = sizeof(DWORD);
235         break;
236     case VT_ARRAY | VT_UI1:
237     {
238         LONG lUbound = 0;
239         LONG lLbound = 0;
240         dwType = REG_BINARY;
241         res = SafeArrayGetLBound(V_UNION(pVar, parray), 1, &lLbound);
242         res = SafeArrayGetUBound(V_UNION(pVar, parray), 1, &lUbound);
243         cbData = (lUbound - lLbound + 1) /* * sizeof(BYTE)*/;
244         TRACE("cbData: %d\n", cbData);
245         res = SafeArrayAccessData(V_UNION(pVar, parray), &lpData);
246         break;
247     }
248     default:
249         FIXME("Variant type %d not handled\n", V_VT(pVar));
250         return E_FAIL;
251     }
252
253     if (RegSetValueExW(This->hkey,
254                        pszPropName, 0,
255                        dwType, lpData, cbData) != ERROR_SUCCESS)
256         res = E_FAIL;
257
258     if (V_VT(pVar) & VT_ARRAY)
259         res = SafeArrayUnaccessData(V_UNION(pVar, parray));
260
261     return res;
262 }
263
264 static const IPropertyBagVtbl IPropertyBag_Vtbl =
265 {
266     DEVENUM_IPropertyBag_QueryInterface,
267     DEVENUM_IPropertyBag_AddRef,
268     DEVENUM_IPropertyBag_Release,
269     DEVENUM_IPropertyBag_Read,
270     DEVENUM_IPropertyBag_Write
271 };
272
273 static HRESULT DEVENUM_IPropertyBag_Construct(HANDLE hkey, IPropertyBag **ppBag)
274 {
275     RegPropBagImpl * rpb = CoTaskMemAlloc(sizeof(RegPropBagImpl));
276     if (!rpb)
277         return E_OUTOFMEMORY;
278     rpb->lpVtbl = &IPropertyBag_Vtbl;
279     rpb->ref = 1;
280     rpb->hkey = hkey;
281     *ppBag = (IPropertyBag*)rpb;
282     DEVENUM_LockModule();
283     return S_OK;
284 }
285
286
287 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_QueryInterface(
288     LPMONIKER iface,
289     REFIID riid,
290     LPVOID *ppvObj)
291 {
292     MediaCatMoniker *This = (MediaCatMoniker *)iface;
293     TRACE("\n\tIID:\t%s\n",debugstr_guid(riid));
294
295     if (This == NULL || ppvObj == NULL) return E_POINTER;
296
297     *ppvObj = NULL;
298
299     if (IsEqualGUID(riid, &IID_IUnknown) ||
300         IsEqualGUID(riid, &IID_IPersist) ||
301         IsEqualGUID(riid, &IID_IPersistStream) ||
302         IsEqualGUID(riid, &IID_IMoniker))
303     {
304         *ppvObj = (LPVOID)iface;
305         DEVENUM_IMediaCatMoniker_AddRef(iface);
306         return S_OK;
307     }
308
309     FIXME("- no interface\n\tIID:\t%s\n", debugstr_guid(riid));
310     return E_NOINTERFACE;
311 }
312
313 /**********************************************************************
314  * DEVENUM_IMediaCatMoniker_AddRef (also IUnknown)
315  */
316 static ULONG WINAPI DEVENUM_IMediaCatMoniker_AddRef(LPMONIKER iface)
317 {
318     MediaCatMoniker *This = (MediaCatMoniker *)iface;
319     TRACE("\n");
320
321     return InterlockedIncrement(&This->ref);
322 }
323
324 /**********************************************************************
325  * DEVENUM_IMediaCatMoniker_Release (also IUnknown)
326  */
327 static ULONG WINAPI DEVENUM_IMediaCatMoniker_Release(LPMONIKER iface)
328 {
329     MediaCatMoniker *This = (MediaCatMoniker *)iface;
330     ULONG ref;
331     TRACE("\n");
332
333     ref = InterlockedDecrement(&This->ref);
334     if (ref == 0) {
335         RegCloseKey(This->hkey);
336         CoTaskMemFree(This);
337         DEVENUM_UnlockModule();
338     }
339     return ref;
340 }
341
342 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_GetClassID(
343     LPMONIKER iface,
344     CLSID* pClassID)
345 {
346     MediaCatMoniker *This = (MediaCatMoniker *)iface;
347     FIXME("(%p)->(%p)\n", This, pClassID);
348
349     if (pClassID == NULL)
350         return E_POINTER;
351
352     return E_NOTIMPL;
353 }
354
355 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_IsDirty(LPMONIKER iface)
356 {
357     FIXME("()\n");
358
359     return S_FALSE;
360 }
361
362 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_Load(LPMONIKER iface, IStream* pStm)
363 {
364     FIXME("(%p)\n", pStm);
365
366     return E_NOTIMPL;
367 }
368
369 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_Save(LPMONIKER iface, IStream* pStm, BOOL fClearDirty)
370 {
371     FIXME("(%p, %s)\n", pStm, fClearDirty ? "true" : "false");
372
373     return STG_E_CANTSAVE;
374 }
375
376 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_GetSizeMax(
377     LPMONIKER iface,
378     ULARGE_INTEGER* pcbSize)
379 {
380     FIXME("(%p)\n", pcbSize);
381
382     ZeroMemory(pcbSize, sizeof(*pcbSize));
383
384     return S_OK;
385 }
386
387 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_BindToObject(
388     LPMONIKER iface,
389     IBindCtx* pbc,
390     IMoniker* pmkToLeft,
391     REFIID riidResult,
392     void** ppvResult)
393 {
394     IUnknown * pObj = NULL;
395     IPropertyBag * pProp = NULL;
396     CLSID clsID;
397     VARIANT var;
398     HRESULT res = E_FAIL;
399
400     MediaCatMoniker *This = (MediaCatMoniker *)iface;
401
402     VariantInit(&var);
403
404     TRACE("(%p)->(%p, %p, %s, %p)\n", This, pbc, pmkToLeft, debugstr_guid(riidResult), ppvResult);
405
406     *ppvResult = NULL;
407
408     if(pmkToLeft==NULL)
409     {
410             /* first activation of this class */
411             LPVOID pvptr;
412             res=IMoniker_BindToStorage(iface, NULL, NULL, &IID_IPropertyBag, &pvptr);
413             pProp = (IPropertyBag*)pvptr;
414             if (SUCCEEDED(res))
415             {
416                 V_VT(&var) = VT_LPWSTR;
417                 res = IPropertyBag_Read(pProp, clsid_keyname, &var, NULL);
418             }
419             if (SUCCEEDED(res))
420             {
421                 res = CLSIDFromString(V_UNION(&var,bstrVal), &clsID);
422                 CoTaskMemFree(V_UNION(&var, bstrVal));
423             }
424             if (SUCCEEDED(res))
425             {
426                 res=CoCreateInstance(&clsID,NULL,CLSCTX_ALL,&IID_IUnknown,&pvptr);
427                 pObj = (IUnknown*)pvptr;
428             }
429     }
430
431     if (pObj!=NULL)
432     {
433         /* get the requested interface from the loaded class */
434         res = S_OK;
435         if (pProp) {
436            HRESULT res2;
437            LPVOID ppv = NULL;
438            res2 = IUnknown_QueryInterface(pObj, &IID_IPersistPropertyBag, &ppv);
439            if (SUCCEEDED(res2)) {
440               res = IPersistPropertyBag_Load((IPersistPropertyBag *) ppv, pProp, NULL);
441               IPersistPropertyBag_Release((IPersistPropertyBag *) ppv);
442            }
443         }
444         if (SUCCEEDED(res))
445            res= IUnknown_QueryInterface(pObj,riidResult,ppvResult);
446         IUnknown_Release(pObj);
447     }
448
449     if (pProp)
450     {
451         IPropertyBag_Release(pProp);
452     }
453
454     TRACE("<- 0x%x\n", res);
455
456     return res;
457 }
458
459 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_BindToStorage(
460     LPMONIKER iface,
461     IBindCtx* pbc,
462     IMoniker* pmkToLeft,
463     REFIID riid,
464     void** ppvObj)
465 {
466     MediaCatMoniker *This = (MediaCatMoniker *)iface;
467     TRACE("(%p)->(%p, %p, %s, %p)\n", This, pbc, pmkToLeft, debugstr_guid(riid), ppvObj);
468
469     *ppvObj = NULL;
470
471     if (pbc || pmkToLeft)
472         return MK_E_NOSTORAGE;
473
474     if (IsEqualGUID(riid, &IID_IPropertyBag))
475     {
476         HANDLE hkey;
477         DuplicateHandle(GetCurrentProcess(), This->hkey, GetCurrentProcess(), &hkey, 0, 0, DUPLICATE_SAME_ACCESS);
478         return DEVENUM_IPropertyBag_Construct(hkey, (IPropertyBag**)ppvObj);
479     }
480
481     return MK_E_NOSTORAGE;
482 }
483
484 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_Reduce(
485     LPMONIKER iface,
486     IBindCtx* pbc,
487     DWORD dwReduceHowFar,
488     IMoniker** ppmkToLeft,
489     IMoniker** ppmkReduced)
490 {
491     TRACE("(%p, %d, %p, %p)\n", pbc, dwReduceHowFar, ppmkToLeft, ppmkReduced);
492
493     if (ppmkToLeft)
494         *ppmkToLeft = NULL;
495     *ppmkReduced = iface;
496
497     return MK_S_REDUCED_TO_SELF;
498 }
499
500 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_ComposeWith(
501     LPMONIKER iface,
502     IMoniker* pmkRight,
503     BOOL fOnlyIfNotGeneric,
504     IMoniker** ppmkComposite)
505 {
506     FIXME("(%p, %s, %p): stub\n", pmkRight, fOnlyIfNotGeneric ? "true" : "false", ppmkComposite);
507
508     /* FIXME: use CreateGenericComposite? */
509     *ppmkComposite = NULL;
510
511     return E_NOTIMPL;
512 }
513
514 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_Enum(
515     LPMONIKER iface,
516     BOOL fForward,
517     IEnumMoniker** ppenumMoniker)
518 {
519     FIXME("(%s, %p): stub\n", fForward ? "true" : "false", ppenumMoniker);
520
521     *ppenumMoniker = NULL;
522
523     return S_OK;
524 }
525
526 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_IsEqual(
527     LPMONIKER iface,
528     IMoniker* pmkOtherMoniker)
529 {
530     FIXME("(%p)\n", pmkOtherMoniker);
531
532     return E_NOTIMPL;
533 }
534
535 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_Hash(
536     LPMONIKER iface,
537     DWORD* pdwHash)
538 {
539     TRACE("(%p)\n", pdwHash);
540
541     *pdwHash = 0;
542
543     return S_OK;
544 }
545
546 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_IsRunning(
547     LPMONIKER iface,
548     IBindCtx* pbc,
549     IMoniker* pmkToLeft,
550     IMoniker* pmkNewlyRunning)
551 {
552     FIXME("(%p, %p, %p)\n", pbc, pmkToLeft, pmkNewlyRunning);
553
554     return S_FALSE;
555 }
556
557 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_GetTimeOfLastChange(
558     LPMONIKER iface,
559     IBindCtx* pbc,
560     IMoniker* pmkToLeft,
561     FILETIME* pFileTime)
562 {
563     TRACE("(%p, %p, %p)\n", pbc, pmkToLeft, pFileTime);
564
565     pFileTime->dwLowDateTime = 0xFFFFFFFF;
566     pFileTime->dwHighDateTime = 0x7FFFFFFF;
567
568     return MK_E_UNAVAILABLE;
569 }
570
571 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_Inverse(
572     LPMONIKER iface,
573     IMoniker** ppmk)
574 {
575     TRACE("(%p)\n", ppmk);
576
577     *ppmk = NULL;
578
579     return MK_E_NOINVERSE;
580 }
581
582 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_CommonPrefixWith(
583     LPMONIKER iface,
584     IMoniker* pmkOtherMoniker,
585     IMoniker** ppmkPrefix)
586 {
587     TRACE("(%p, %p)\n", pmkOtherMoniker, ppmkPrefix);
588
589     *ppmkPrefix = NULL;
590
591     return MK_E_NOPREFIX;
592 }
593
594 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_RelativePathTo(
595     LPMONIKER iface,
596     IMoniker* pmkOther,
597     IMoniker** ppmkRelPath)
598 {
599     TRACE("(%p, %p)\n", pmkOther, ppmkRelPath);
600
601     *ppmkRelPath = pmkOther;
602
603     return MK_S_HIM;
604 }
605
606 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_GetDisplayName(
607     LPMONIKER iface,
608     IBindCtx* pbc,
609     IMoniker* pmkToLeft,
610     LPOLESTR* ppszDisplayName)
611 {
612     MediaCatMoniker *This = (MediaCatMoniker *)iface;
613     WCHAR wszBuffer[MAX_PATH];
614     static const WCHAR wszFriendlyName[] = {'F','r','i','e','n','d','l','y','N','a','m','e',0};
615     LONG received = sizeof(wszFriendlyName);
616
617     TRACE("(%p, %p, %p)\n", pbc, pmkToLeft, ppszDisplayName);
618
619     *ppszDisplayName = NULL;
620
621     /* FIXME: should this be the weird stuff we have to parse in IParseDisplayName? */
622     if (RegQueryValueW(This->hkey, wszFriendlyName, wszBuffer, &received) == ERROR_SUCCESS)
623     {
624         *ppszDisplayName = CoTaskMemAlloc(received);
625         strcpyW(*ppszDisplayName, wszBuffer);
626         return S_OK;
627     }
628
629     return E_FAIL;
630 }
631
632 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_ParseDisplayName(
633     LPMONIKER iface,
634     IBindCtx* pbc,
635     IMoniker* pmkToLeft,
636     LPOLESTR pszDisplayName,
637     ULONG* pchEaten,
638     IMoniker** ppmkOut)
639 {
640     FIXME("(%p, %p, %s, %p, %p)\n", pbc, pmkToLeft, debugstr_w(pszDisplayName), pchEaten, ppmkOut);
641
642     *pchEaten = 0;
643     *ppmkOut = NULL;
644
645     return MK_E_SYNTAX;
646 }
647
648 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_IsSystemMoniker(
649     LPMONIKER iface,
650     DWORD* pdwMksys)
651 {
652     TRACE("(%p)\n", pdwMksys);
653
654     return S_FALSE;
655 }
656
657 static const IMonikerVtbl IMoniker_Vtbl =
658 {
659     DEVENUM_IMediaCatMoniker_QueryInterface,
660     DEVENUM_IMediaCatMoniker_AddRef,
661     DEVENUM_IMediaCatMoniker_Release,
662     DEVENUM_IMediaCatMoniker_GetClassID,
663     DEVENUM_IMediaCatMoniker_IsDirty,
664     DEVENUM_IMediaCatMoniker_Load,
665     DEVENUM_IMediaCatMoniker_Save,
666     DEVENUM_IMediaCatMoniker_GetSizeMax,
667     DEVENUM_IMediaCatMoniker_BindToObject,
668     DEVENUM_IMediaCatMoniker_BindToStorage,
669     DEVENUM_IMediaCatMoniker_Reduce,
670     DEVENUM_IMediaCatMoniker_ComposeWith,
671     DEVENUM_IMediaCatMoniker_Enum,
672     DEVENUM_IMediaCatMoniker_IsEqual,
673     DEVENUM_IMediaCatMoniker_Hash,
674     DEVENUM_IMediaCatMoniker_IsRunning,
675     DEVENUM_IMediaCatMoniker_GetTimeOfLastChange,
676     DEVENUM_IMediaCatMoniker_Inverse,
677     DEVENUM_IMediaCatMoniker_CommonPrefixWith,
678     DEVENUM_IMediaCatMoniker_RelativePathTo,
679     DEVENUM_IMediaCatMoniker_GetDisplayName,
680     DEVENUM_IMediaCatMoniker_ParseDisplayName,
681     DEVENUM_IMediaCatMoniker_IsSystemMoniker
682 };
683
684 MediaCatMoniker * DEVENUM_IMediaCatMoniker_Construct(void)
685 {
686     MediaCatMoniker * pMoniker = NULL;
687     pMoniker = CoTaskMemAlloc(sizeof(MediaCatMoniker));
688     if (!pMoniker)
689         return NULL;
690
691     pMoniker->lpVtbl = &IMoniker_Vtbl;
692     pMoniker->ref = 0;
693     pMoniker->hkey = NULL;
694
695     DEVENUM_IMediaCatMoniker_AddRef((LPMONIKER)pMoniker);
696
697     DEVENUM_LockModule();
698
699     return pMoniker;
700 }
701
702 /**********************************************************************
703  * DEVENUM_IEnumMoniker_QueryInterface (also IUnknown)
704  */
705 static HRESULT WINAPI DEVENUM_IEnumMoniker_QueryInterface(
706     LPENUMMONIKER iface,
707     REFIID riid,
708     LPVOID *ppvObj)
709 {
710     EnumMonikerImpl *This = (EnumMonikerImpl *)iface;
711     TRACE("\n\tIID:\t%s\n",debugstr_guid(riid));
712
713     if (This == NULL || ppvObj == NULL) return E_POINTER;
714
715     if (IsEqualGUID(riid, &IID_IUnknown) ||
716         IsEqualGUID(riid, &IID_IEnumMoniker))
717     {
718         *ppvObj = (LPVOID)iface;
719         DEVENUM_IEnumMoniker_AddRef(iface);
720         return S_OK;
721     }
722
723     FIXME("- no interface\n\tIID:\t%s\n", debugstr_guid(riid));
724     return E_NOINTERFACE;
725 }
726
727 /**********************************************************************
728  * DEVENUM_IEnumMoniker_AddRef (also IUnknown)
729  */
730 static ULONG WINAPI DEVENUM_IEnumMoniker_AddRef(LPENUMMONIKER iface)
731 {
732     EnumMonikerImpl *This = (EnumMonikerImpl *)iface;
733     ULONG ref = InterlockedIncrement(&This->ref);
734
735     TRACE("(%p)->() AddRef from %d\n", iface, ref - 1);
736
737     return ref;
738 }
739
740 /**********************************************************************
741  * DEVENUM_IEnumMoniker_Release (also IUnknown)
742  */
743 static ULONG WINAPI DEVENUM_IEnumMoniker_Release(LPENUMMONIKER iface)
744 {
745     EnumMonikerImpl *This = (EnumMonikerImpl *)iface;
746     ULONG ref = InterlockedDecrement(&This->ref);
747
748     TRACE("(%p)->() Release from %d\n", iface, ref + 1);
749
750     if (!ref)
751     {
752         RegCloseKey(This->hkey);
753         CoTaskMemFree(This);
754         DEVENUM_UnlockModule();
755         return 0;
756     }
757     return ref;
758 }
759
760 static HRESULT WINAPI DEVENUM_IEnumMoniker_Next(LPENUMMONIKER iface, ULONG celt, IMoniker ** rgelt, ULONG * pceltFetched)
761 {
762     WCHAR buffer[MAX_PATH + 1];
763     LONG res;
764     ULONG fetched = 0;
765     MediaCatMoniker * pMoniker;
766     EnumMonikerImpl *This = (EnumMonikerImpl *)iface;
767
768     TRACE("(%p)->(%d, %p, %p)\n", iface, celt, rgelt, pceltFetched);
769
770     while (fetched < celt)
771     {
772         res = RegEnumKeyW(This->hkey, This->index, buffer, sizeof(buffer) / sizeof(WCHAR));
773         if (res != ERROR_SUCCESS)
774         {
775             break;
776         }
777         pMoniker = DEVENUM_IMediaCatMoniker_Construct();
778         if (!pMoniker)
779             return E_OUTOFMEMORY;
780
781         if (RegOpenKeyW(This->hkey, buffer, &pMoniker->hkey) != ERROR_SUCCESS)
782         {
783             DEVENUM_IMediaCatMoniker_Release((LPMONIKER)pMoniker);
784             break;
785         }
786         rgelt[fetched] = (LPMONIKER)pMoniker;
787         fetched++;
788     }
789
790     This->index += fetched;
791
792     TRACE("-- fetched %d\n", fetched);
793
794     if (pceltFetched)
795         *pceltFetched = fetched;
796
797     if (fetched != celt)
798         return S_FALSE;
799     else
800         return S_OK;
801 }
802
803 static HRESULT WINAPI DEVENUM_IEnumMoniker_Skip(LPENUMMONIKER iface, ULONG celt)
804 {
805     EnumMonikerImpl *This = (EnumMonikerImpl *)iface;
806     DWORD subKeys;
807
808     TRACE("(%p)->(%d)\n", iface, celt);
809
810     /* Before incrementing, check if there are any more values to run thru.
811        Some programs use the Skip() function to get the amount of devices */
812     if(RegQueryInfoKeyW(This->hkey, NULL, NULL, NULL, &subKeys, NULL, NULL, NULL, NULL, NULL, NULL, NULL) != ERROR_SUCCESS)
813     {
814         return S_FALSE;
815     }
816     if((This->index + celt) >= subKeys)
817     {
818         return S_FALSE;
819     }
820
821     This->index += celt;
822
823     return S_OK;
824 }
825
826 static HRESULT WINAPI DEVENUM_IEnumMoniker_Reset(LPENUMMONIKER iface)
827 {
828     EnumMonikerImpl *This = (EnumMonikerImpl *)iface;
829
830     TRACE("(%p)->()\n", iface);
831
832     This->index = 0;
833
834     return S_OK;
835 }
836
837 static HRESULT WINAPI DEVENUM_IEnumMoniker_Clone(LPENUMMONIKER iface, IEnumMoniker ** ppenum)
838 {
839     FIXME("(%p)->(%p): stub\n", iface, ppenum);
840
841     return E_NOTIMPL;
842 }
843
844 /**********************************************************************
845  * IEnumMoniker_Vtbl
846  */
847 static const IEnumMonikerVtbl IEnumMoniker_Vtbl =
848 {
849     DEVENUM_IEnumMoniker_QueryInterface,
850     DEVENUM_IEnumMoniker_AddRef,
851     DEVENUM_IEnumMoniker_Release,
852     DEVENUM_IEnumMoniker_Next,
853     DEVENUM_IEnumMoniker_Skip,
854     DEVENUM_IEnumMoniker_Reset,
855     DEVENUM_IEnumMoniker_Clone
856 };
857
858 HRESULT DEVENUM_IEnumMoniker_Construct(HKEY hkey, IEnumMoniker ** ppEnumMoniker)
859 {
860     EnumMonikerImpl * pEnumMoniker = CoTaskMemAlloc(sizeof(EnumMonikerImpl));
861     if (!pEnumMoniker)
862         return E_OUTOFMEMORY;
863
864     pEnumMoniker->lpVtbl = &IEnumMoniker_Vtbl;
865     pEnumMoniker->ref = 1;
866     pEnumMoniker->index = 0;
867     pEnumMoniker->hkey = hkey;
868
869     *ppEnumMoniker = (IEnumMoniker *)pEnumMoniker;
870
871     DEVENUM_LockModule();
872
873     return S_OK;
874 }