2 * Strmbase DLL functions
4 * Copyright (C) 2005 Rolf Kalbermatter
5 * Copyright (C) 2010 Aric Stewart, CodeWeavers
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
27 #define NONAMELESSSTRUCT
28 #define NONAMELESSUNION
39 #include "wine/unicode.h"
40 #include "wine/debug.h"
41 #include "wine/strmbase.h"
43 WINE_DEFAULT_DEBUG_CHANNEL(strmbase
);
45 extern const int g_cTemplates
;
46 extern const FactoryTemplate g_Templates
[];
48 static HINSTANCE g_hInst
= NULL
;
49 static LONG server_locks
= 0;
52 * defines and constants
55 static const WCHAR clsid_keyname
[] = {'C','L','S','I','D',0 };
56 static const WCHAR ips32_keyname
[] = {'I','n','P','r','o','c','S','e','r','v','e','r','3','2',0};
57 static const WCHAR tmodel_keyname
[] = {'T','h','r','e','a','d','i','n','g','M','o','d','e','l',0};
58 static const WCHAR tmodel_both
[] = {'B','o','t','h',0};
61 * SetupRegisterClass()
63 static HRESULT
SetupRegisterClass(HKEY clsid
, LPCWSTR szCLSID
,
64 LPCWSTR szDescription
,
67 LPCWSTR szThreadingModel
)
69 HKEY hkey
, hsubkey
= NULL
;
70 LONG ret
= RegCreateKeyW(clsid
, szCLSID
, &hkey
);
71 if (ERROR_SUCCESS
!= ret
)
72 return HRESULT_FROM_WIN32(ret
);
74 /* set description string */
75 ret
= RegSetValueW(hkey
, NULL
, REG_SZ
, szDescription
,
76 sizeof(WCHAR
) * (lstrlenW(szDescription
) + 1));
77 if (ERROR_SUCCESS
!= ret
)
80 /* create CLSID\\{"CLSID"}\\"ServerType" key, using key to CLSID\\{"CLSID"}
81 passed back by last call to RegCreateKeyW(). */
82 ret
= RegCreateKeyW(hkey
, szServerType
, &hsubkey
);
83 if (ERROR_SUCCESS
!= ret
)
87 ret
= RegSetValueW(hsubkey
, NULL
, REG_SZ
, szFileName
,
88 sizeof(WCHAR
) * (lstrlenW(szFileName
) + 1));
89 if (ERROR_SUCCESS
!= ret
)
92 /* set threading model */
93 ret
= RegSetValueExW(hsubkey
, tmodel_keyname
, 0L, REG_SZ
,
94 (const BYTE
*)szThreadingModel
,
95 sizeof(WCHAR
) * (lstrlenW(szThreadingModel
) + 1));
100 return HRESULT_FROM_WIN32(ret
);
104 * RegisterAllClasses()
106 static HRESULT
SetupRegisterAllClasses(const FactoryTemplate
* pList
, int num
,
107 LPCWSTR szFileName
, BOOL bRegister
)
109 HRESULT hr
= NOERROR
;
111 OLECHAR szCLSID
[CHARS_IN_GUID
];
112 LONG i
, ret
= RegCreateKeyW(HKEY_CLASSES_ROOT
, clsid_keyname
, &hkey
);
113 if (ERROR_SUCCESS
!= ret
)
114 return HRESULT_FROM_WIN32(ret
);
116 for (i
= 0; i
< num
; i
++, pList
++)
118 /* (un)register CLSID and InprocServer32 */
119 hr
= StringFromGUID2(pList
->m_ClsID
, szCLSID
, CHARS_IN_GUID
);
123 hr
= SetupRegisterClass(hkey
, szCLSID
,
124 pList
->m_Name
, szFileName
,
125 ips32_keyname
, tmodel_both
);
127 hr
= RegDeleteTreeW(hkey
, szCLSID
);
134 HRESULT WINAPI
AMovieSetupRegisterFilter2(const AMOVIESETUP_FILTER
*pFilter
, IFilterMapper2
*pIFM2
, BOOL bRegister
)
144 rf2
.dwMerit
= pFilter
->merit
;
145 rf2
.u
.s1
.cPins
= pFilter
->pins
;
146 rf2
.u
.s1
.rgPins
= pFilter
->pPin
;
148 return IFilterMapper2_RegisterFilter(pIFM2
, pFilter
->clsid
, pFilter
->name
, NULL
, &CLSID_LegacyAmFilterCategory
, NULL
, &rf2
);
152 return IFilterMapper2_UnregisterFilter(pIFM2
, &CLSID_LegacyAmFilterCategory
, NULL
, pFilter
->clsid
);
155 HRESULT WINAPI
AMovieDllRegisterServer2(BOOL bRegister
)
159 IFilterMapper2
*pIFM2
= NULL
;
160 WCHAR szFileName
[MAX_PATH
];
162 if (!GetModuleFileNameW(g_hInst
, szFileName
, MAX_PATH
))
164 ERR("Failed to get module file name for registration\n");
169 hr
= SetupRegisterAllClasses(g_Templates
, g_cTemplates
, szFileName
, TRUE
);
173 TRACE("Getting IFilterMapper2\r\n");
174 hr
= CoCreateInstance(&CLSID_FilterMapper2
, NULL
, CLSCTX_INPROC_SERVER
,
175 &IID_IFilterMapper2
, (void **)&pIFM2
);
177 for (i
= 0; SUCCEEDED(hr
) && i
< g_cTemplates
; i
++)
178 hr
= AMovieSetupRegisterFilter2(g_Templates
[i
].m_pAMovieSetup_Filter
, pIFM2
, bRegister
);
180 /* release interface */
182 IFilterMapper2_Release(pIFM2
);
185 CoFreeUnusedLibraries();
188 /* if unregistering, unregister all OLE servers */
189 if (SUCCEEDED(hr
) && !bRegister
)
190 hr
= SetupRegisterAllClasses(g_Templates
, g_cTemplates
, szFileName
, FALSE
);
195 /****************************************************************************
196 * SetupInitializeServers
198 * This function is table driven using the static members of the
199 * CFactoryTemplate class defined in the Dll.
201 * It calls the initialize function for any class in CFactoryTemplate with
204 ****************************************************************************/
205 static void SetupInitializeServers(const FactoryTemplate
* pList
, int num
,
210 for (i
= 0; i
< num
; i
++, pList
++)
212 if (pList
->m_lpfnInit
)
213 pList
->m_lpfnInit(bLoading
, pList
->m_ClsID
);
217 BOOL WINAPI
STRMBASE_DllMain(HINSTANCE hInstDLL
, DWORD fdwReason
, LPVOID lpv
)
221 case DLL_PROCESS_ATTACH
:
223 DisableThreadLibraryCalls(hInstDLL
);
224 SetupInitializeServers(g_Templates
, g_cTemplates
, TRUE
);
226 case DLL_PROCESS_DETACH
:
227 SetupInitializeServers(g_Templates
, g_cTemplates
, FALSE
);
233 /******************************************************************************
237 IClassFactory IClassFactory_iface
;
239 LPFNNewCOMObject pfnCreateInstance
;
242 static inline IClassFactoryImpl
*impl_from_IClassFactory(IClassFactory
*iface
)
244 return CONTAINING_RECORD(iface
, IClassFactoryImpl
, IClassFactory_iface
);
247 static HRESULT WINAPI
DSCF_QueryInterface(IClassFactory
*iface
, REFIID riid
, void **ppobj
)
249 if (IsEqualGUID(riid
, &IID_IUnknown
) ||
250 IsEqualGUID(riid
, &IID_IClassFactory
))
252 IClassFactory_AddRef(iface
);
258 WARN("(%p)->(%s,%p), not found\n", iface
, debugstr_guid(riid
), ppobj
);
259 return E_NOINTERFACE
;
262 static ULONG WINAPI
DSCF_AddRef(IClassFactory
*iface
)
264 IClassFactoryImpl
*This
= impl_from_IClassFactory(iface
);
265 return InterlockedIncrement(&This
->ref
);
268 static ULONG WINAPI
DSCF_Release(IClassFactory
*iface
)
270 IClassFactoryImpl
*This
= impl_from_IClassFactory(iface
);
271 ULONG ref
= InterlockedDecrement(&This
->ref
);
274 HeapFree(GetProcessHeap(), 0, This
);
279 static HRESULT WINAPI
DSCF_CreateInstance(IClassFactory
*iface
, IUnknown
*pOuter
,
280 REFIID riid
, void **ppobj
)
282 IClassFactoryImpl
*This
= impl_from_IClassFactory(iface
);
283 HRESULT hres
= ERROR_SUCCESS
;
286 TRACE("(%p)->(%p,%s,%p)\n", This
, pOuter
, debugstr_guid(riid
), ppobj
);
291 /* Enforce the normal OLE rules regarding interfaces and delegation */
292 if (pOuter
&& !IsEqualGUID(riid
, &IID_IUnknown
))
293 return E_NOINTERFACE
;
296 punk
= This
->pfnCreateInstance(pOuter
, &hres
);
299 /* No object created, update error if it isn't done already and return */
301 hres
= E_OUTOFMEMORY
;
307 hres
= IUnknown_QueryInterface(punk
, riid
, ppobj
);
309 /* Releasing the object. If everything was successful, QueryInterface
310 should have incremented the refcount once more, otherwise this will
312 IUnknown_Release(punk
);
316 static HRESULT WINAPI
DSCF_LockServer(IClassFactory
*iface
, BOOL dolock
)
318 IClassFactoryImpl
*This
= impl_from_IClassFactory(iface
);
319 TRACE("(%p)->(%d)\n",This
, dolock
);
322 InterlockedIncrement(&server_locks
);
324 InterlockedDecrement(&server_locks
);
328 static const IClassFactoryVtbl DSCF_Vtbl
=
337 /***********************************************************************
340 HRESULT WINAPI
STRMBASE_DllGetClassObject(REFCLSID rclsid
, REFIID riid
, LPVOID
*ppv
)
342 const FactoryTemplate
*pList
= g_Templates
;
343 IClassFactoryImpl
*factory
;
346 TRACE("(%s,%s,%p)\n", debugstr_guid(rclsid
), debugstr_guid(riid
), ppv
);
353 if (!IsEqualGUID(&IID_IClassFactory
, riid
) &&
354 !IsEqualGUID(&IID_IUnknown
, riid
))
355 return E_NOINTERFACE
;
357 for (i
= 0; i
< g_cTemplates
; i
++, pList
++)
359 if (IsEqualGUID(pList
->m_ClsID
, rclsid
))
363 if (i
== g_cTemplates
)
365 char dllname
[MAX_PATH
];
366 if (!GetModuleFileNameA(g_hInst
, dllname
, sizeof(dllname
)))
367 strcpy(dllname
, "???");
368 ERR("%s: no class found in %s.\n", debugstr_guid(rclsid
), dllname
);
369 return CLASS_E_CLASSNOTAVAILABLE
;
371 else if (!pList
->m_lpfnNew
)
373 FIXME("%s: class not implemented yet.\n", debugstr_guid(rclsid
));
374 return CLASS_E_CLASSNOTAVAILABLE
;
377 factory
= HeapAlloc(GetProcessHeap(), 0, sizeof(IClassFactoryImpl
));
379 return E_OUTOFMEMORY
;
381 factory
->IClassFactory_iface
.lpVtbl
= &DSCF_Vtbl
;
384 factory
->pfnCreateInstance
= pList
->m_lpfnNew
;
386 *ppv
= &factory
->IClassFactory_iface
;
390 /***********************************************************************
393 HRESULT WINAPI
STRMBASE_DllCanUnloadNow(void)
397 if (server_locks
== 0)