2 * Qcap implementation, dllentry points
4 * Copyright (C) 2003 Dominik Strasser
5 * Copyright (C) 2005 Rolf Kalbermatter
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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
28 #define NONAMELESSSTRUCT
29 #define NONAMELESSUNION
39 #include "qcap_main.h"
41 #include "wine/unicode.h"
42 #include "wine/debug.h"
44 WINE_DEFAULT_DEBUG_CHANNEL(qcap
);
46 static LONG objects_ref
= 0;
47 static LONG server_locks
= 0;
48 static HINSTANCE ghInst
= NULL
;
50 static const WCHAR wAudioCaptFilter
[] =
51 {'A','u','d','i','o',' ','C','a','p','t','u','r','e',' ','F','i','l','t','e','r',0};
52 static const WCHAR wAVICompressor
[] =
53 {'A','V','I',' ','C','o','m','p','r','e','s','s','o','r',0};
54 static const WCHAR wVFWCaptFilter
[] =
55 {'V','F','W',' ','C','a','p','t','u','r','e',' ','F','i','l','t','e','r',0};
56 static const WCHAR wVFWCaptFilterProp
[] =
57 {'V','F','W',' ','C','a','p','t','u','r','e',' ','F','i','l','t','e','r',' ',
58 'P','r','o','p','e','r','t','y',' ','P','a','g','e',0};
59 static const WCHAR wAVIMux
[] =
60 {'A','V','I',' ','m','u','x',0};
61 static const WCHAR wAVIMuxPropPage
[] =
62 {'A','V','I',' ','m','u','x',' ','P','r','o','p','e','r','t','y',' ','P','a','g','e',0};
63 static const WCHAR wAVIMuxPropPage1
[] =
64 {'A','V','I',' ','m','u','x',' ','P','r','o','p','e','r','t','y',' ','P','a','g','e','1',0};
65 static const WCHAR wFileWriter
[] =
66 {'F','i','l','e',' ','W','r','i','t','e','r',0};
67 static const WCHAR wCaptGraphBuilder
[] =
68 {'C','a','p','t','u','r','e',' ','G','r','a','p','h',' ','B','u','i','l','d','e','r',0};
69 static const WCHAR wCaptGraphBuilder2
[] =
70 {'C','a','p','t','u','r','e',' ','G','r','a','p','h',' ','B','u','i','l','d','e','r','2',0};
71 static const WCHAR wInfPinTeeFilter
[] =
72 {'I','n','f','i','n','i','t','e',' ','P','i','n',' ','T','e','e',' ','F','i',
74 static const WCHAR wSmartTeeFilter
[] =
75 {'S','m','a','r','t',' ','T','e','e',' ','F','i','l','t','e','r',0};
76 static const WCHAR wAudioInMixerProp
[] =
77 {'A','u','d','i','o','I','n','p','u','t','M','i','x','e','r',' ','P','r','o',
78 'p','e','r','t','y',' ','P','a','g','e',0};
80 static CFactoryTemplate
const g_cTemplates
[] = {
84 &CLSID_AudioCaptureFilter,
85 QCAP_createAudioCaptureFilter,
90 QCAP_createAVICompressor,
95 QCAP_createVFWCaptureFilter
,
99 &CLSID_VFWCaptureFilterPropertyPage,
100 QCAP_createVFWCaptureFilterPropertyPage,
109 &CLSID_AVImuxPropertyPage,
110 QCAP_createAVImuxPropertyPage,
114 &CLSID_AVImuxPropertyPage1,
115 QCAP_createAVImuxPropertyPage1,
120 QCAP_createFileWriter,
124 &CLSID_CaptureGraphBuilder
,
125 QCAP_createCaptureGraphBuilder2
,
129 &CLSID_CaptureGraphBuilder2
,
130 QCAP_createCaptureGraphBuilder2
,
134 &CLSID_InfinitePinTeeFilter,
135 QCAP_createInfinitePinTeeFilter,
139 &CLSID_SmartTeeFilter,
140 QCAP_createSmartTeeFilter,
144 &CLSID_AudioInputMixerPropertyPage,
145 QCAP_createAudioInputMixerPropertyPage,
150 static int g_numTemplates
= sizeof(g_cTemplates
) / sizeof(g_cTemplates
[0]);
152 /***********************************************************************
153 * Dll EntryPoint (QCAP.@)
155 BOOL WINAPI
DllMain(HINSTANCE hInstDLL
, DWORD fdwReason
, LPVOID lpv
)
159 case DLL_PROCESS_ATTACH
:
160 DisableThreadLibraryCalls(hInstDLL
);
162 SetupInitializeServers(g_cTemplates
, g_numTemplates
, TRUE
);
164 case DLL_PROCESS_DETACH
:
165 SetupInitializeServers(g_cTemplates
, g_numTemplates
, FALSE
);
171 /***********************************************************************
172 * DllRegisterServer (QCAP.@)
174 HRESULT WINAPI
DllRegisterServer(void)
178 return SetupRegisterServers(g_cTemplates
, g_numTemplates
, ghInst
, TRUE
);
181 /***********************************************************************
182 * DllUnregisterServer (QCAP.@)
184 HRESULT WINAPI
DllUnregisterServer(void)
188 return SetupRegisterServers(g_cTemplates
, g_numTemplates
, ghInst
, FALSE
);
191 /***********************************************************************
192 * DllCanUnloadNow (QCAP.@)
194 HRESULT WINAPI
DllCanUnloadNow(void)
198 if (objects_ref
== 0 && server_locks
== 0)
203 /******************************************************************************
207 IClassFactory ITF_IClassFactory
;
210 LPFNNewCOMObject pfnCreateInstance
;
213 static HRESULT WINAPI
214 DSCF_QueryInterface(LPCLASSFACTORY iface
,REFIID riid
,LPVOID
*ppobj
)
216 IClassFactoryImpl
*This
= (IClassFactoryImpl
*)iface
;
218 if (IsEqualGUID(riid
, &IID_IUnknown
) ||
219 IsEqualGUID(riid
, &IID_IClassFactory
))
221 IClassFactory_AddRef(iface
);
226 WARN("(%p)->(%s,%p), not found\n", This
, debugstr_guid(riid
), ppobj
);
227 return E_NOINTERFACE
;
230 static ULONG WINAPI
DSCF_AddRef(LPCLASSFACTORY iface
)
232 IClassFactoryImpl
*This
= (IClassFactoryImpl
*)iface
;
233 return InterlockedIncrement(&This
->ref
);
236 static ULONG WINAPI
DSCF_Release(LPCLASSFACTORY iface
)
238 IClassFactoryImpl
*This
= (IClassFactoryImpl
*)iface
;
240 ULONG ref
= InterlockedDecrement(&This
->ref
);
243 HeapFree(GetProcessHeap(), 0, This
);
248 static HRESULT WINAPI
DSCF_CreateInstance(LPCLASSFACTORY iface
, LPUNKNOWN pOuter
,
249 REFIID riid
, LPVOID
*ppobj
)
251 IClassFactoryImpl
*This
= (IClassFactoryImpl
*)iface
;
252 HRESULT hres
= ERROR_SUCCESS
;
255 TRACE("(%p)->(%p,%s,%p)\n", This
, pOuter
, debugstr_guid(riid
), ppobj
);
260 /* Enforce the normal OLE rules regarding interfaces and delegation */
261 if (pOuter
&& !IsEqualGUID(riid
, &IID_IUnknown
))
262 return E_NOINTERFACE
;
265 punk
= This
->pfnCreateInstance(pOuter
, &hres
);
268 /* No object created, update error if it isn't done already and return */
270 hres
= E_OUTOFMEMORY
;
276 hres
= IUnknown_QueryInterface(punk
, riid
, ppobj
);
278 /* Releasing the object. If everything was successful, QueryInterface
279 should have incremented the refcount once more, otherwise this will
281 IUnknown_Release(punk
);
285 static HRESULT WINAPI
DSCF_LockServer(LPCLASSFACTORY iface
, BOOL dolock
)
287 IClassFactoryImpl
*This
= (IClassFactoryImpl
*)iface
;
288 TRACE("(%p)->(%d)\n",This
, dolock
);
291 InterlockedIncrement(&server_locks
);
293 InterlockedDecrement(&server_locks
);
297 static const IClassFactoryVtbl DSCF_Vtbl
=
306 /***********************************************************************
307 * DllGetClassObject (QCAP.@)
309 HRESULT WINAPI
DllGetClassObject(REFCLSID rclsid
, REFIID riid
, LPVOID
*ppv
)
311 const CFactoryTemplate
*pList
= g_cTemplates
;
312 IClassFactoryImpl
*factory
;
315 TRACE("(%s,%s,%p)\n", debugstr_guid(rclsid
), debugstr_guid(riid
), ppv
);
322 if (!IsEqualGUID(&IID_IClassFactory
, riid
) &&
323 !IsEqualGUID(&IID_IUnknown
, riid
))
324 return E_NOINTERFACE
;
326 for (i
= 0; i
< g_numTemplates
; i
++, pList
++)
328 if (IsEqualGUID(pList
->m_ClsID
, rclsid
))
332 if (i
== g_numTemplates
)
334 FIXME("%s: no class found.\n", debugstr_guid(rclsid
));
335 return CLASS_E_CLASSNOTAVAILABLE
;
338 factory
= HeapAlloc(GetProcessHeap(), 0, sizeof(IClassFactoryImpl
));
340 return E_OUTOFMEMORY
;
342 factory
->ITF_IClassFactory
.lpVtbl
= &DSCF_Vtbl
;
345 factory
->pfnCreateInstance
= pList
->m_lpfnNew
;
347 *ppv
= &(factory
->ITF_IClassFactory
);
351 DWORD
ObjectRefCount(BOOL increment
)
354 return InterlockedIncrement(&objects_ref
);
355 return InterlockedDecrement(&objects_ref
);