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