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
54 #define MAX_KEY_LEN 260
56 static const WCHAR clsid_keyname
[] = {'C','L','S','I','D',0 };
57 static const WCHAR ips32_keyname
[] = {'I','n','P','r','o','c','S','e','r','v','e','r','3','2',0};
58 static const WCHAR tmodel_keyname
[] = {'T','h','r','e','a','d','i','n','g','M','o','d','e','l',0};
59 static const WCHAR tmodel_both
[] = {'B','o','t','h',0};
62 * SetupRegisterClass()
64 static HRESULT
SetupRegisterClass(HKEY clsid
, LPCWSTR szCLSID
,
65 LPCWSTR szDescription
,
68 LPCWSTR szThreadingModel
)
70 HKEY hkey
, hsubkey
= NULL
;
71 LONG ret
= RegCreateKeyW(clsid
, szCLSID
, &hkey
);
72 if (ERROR_SUCCESS
!= ret
)
73 return HRESULT_FROM_WIN32(ret
);
75 /* set description string */
76 ret
= RegSetValueW(hkey
, NULL
, REG_SZ
, szDescription
,
77 sizeof(WCHAR
) * (lstrlenW(szDescription
) + 1));
78 if (ERROR_SUCCESS
!= ret
)
81 /* create CLSID\\{"CLSID"}\\"ServerType" key, using key to CLSID\\{"CLSID"}
82 passed back by last call to RegCreateKeyW(). */
83 ret
= RegCreateKeyW(hkey
, szServerType
, &hsubkey
);
84 if (ERROR_SUCCESS
!= ret
)
88 ret
= RegSetValueW(hsubkey
, NULL
, REG_SZ
, szFileName
,
89 sizeof(WCHAR
) * (lstrlenW(szFileName
) + 1));
90 if (ERROR_SUCCESS
!= ret
)
93 /* set threading model */
94 ret
= RegSetValueExW(hsubkey
, tmodel_keyname
, 0L, REG_SZ
,
95 (const BYTE
*)szThreadingModel
,
96 sizeof(WCHAR
) * (lstrlenW(szThreadingModel
) + 1));
101 return HRESULT_FROM_WIN32(ret
);
105 * RegisterAllClasses()
107 static HRESULT
SetupRegisterAllClasses(const FactoryTemplate
* pList
, int num
,
108 LPCWSTR szFileName
, BOOL bRegister
)
110 HRESULT hr
= NOERROR
;
112 OLECHAR szCLSID
[CHARS_IN_GUID
];
113 LONG i
, ret
= RegCreateKeyW(HKEY_CLASSES_ROOT
, clsid_keyname
, &hkey
);
114 if (ERROR_SUCCESS
!= ret
)
115 return HRESULT_FROM_WIN32(ret
);
117 for (i
= 0; i
< num
; i
++, pList
++)
119 /* (un)register CLSID and InprocServer32 */
120 hr
= StringFromGUID2(pList
->m_ClsID
, szCLSID
, CHARS_IN_GUID
);
124 hr
= SetupRegisterClass(hkey
, szCLSID
,
125 pList
->m_Name
, szFileName
,
126 ips32_keyname
, tmodel_both
);
128 hr
= RegDeleteTreeW(hkey
, szCLSID
);
135 HRESULT WINAPI
AMovieSetupRegisterFilter2(const AMOVIESETUP_FILTER
*pFilter
, IFilterMapper2
*pIFM2
, BOOL bRegister
)
145 rf2
.dwMerit
= pFilter
->merit
;
146 rf2
.u
.s1
.cPins
= pFilter
->pins
;
147 rf2
.u
.s1
.rgPins
= pFilter
->pPin
;
149 return IFilterMapper2_RegisterFilter(pIFM2
, pFilter
->clsid
, pFilter
->name
, NULL
, &CLSID_LegacyAmFilterCategory
, NULL
, &rf2
);
153 return IFilterMapper2_UnregisterFilter(pIFM2
, &CLSID_LegacyAmFilterCategory
, NULL
, pFilter
->clsid
);
156 HRESULT WINAPI
AMovieDllRegisterServer2(BOOL bRegister
)
160 IFilterMapper2
*pIFM2
= NULL
;
161 WCHAR szFileName
[MAX_PATH
];
163 if (!GetModuleFileNameW(g_hInst
, szFileName
, MAX_PATH
))
165 ERR("Failed to get module file name for registration\n");
170 hr
= SetupRegisterAllClasses(g_Templates
, g_cTemplates
, szFileName
, TRUE
);
172 hr
= CoInitialize(NULL
);
174 TRACE("Getting IFilterMapper2\r\n");
175 hr
= CoCreateInstance(&CLSID_FilterMapper2
, NULL
, CLSCTX_INPROC_SERVER
,
176 &IID_IFilterMapper2
, (void **)&pIFM2
);
178 for (i
= 0; SUCCEEDED(hr
) && i
< g_cTemplates
; i
++)
179 hr
= AMovieSetupRegisterFilter2(g_Templates
[i
].m_pAMovieSetup_Filter
, pIFM2
, bRegister
);
181 /* release interface */
183 IFilterMapper2_Release(pIFM2
);
186 CoFreeUnusedLibraries();
189 /* if unregistering, unregister all OLE servers */
190 if (SUCCEEDED(hr
) && !bRegister
)
191 hr
= SetupRegisterAllClasses(g_Templates
, g_cTemplates
, szFileName
, FALSE
);
196 /****************************************************************************
197 * SetupInitializeServers
199 * This function is table driven using the static members of the
200 * CFactoryTemplate class defined in the Dll.
202 * It calls the initialize function for any class in CFactoryTemplate with
205 ****************************************************************************/
206 static void SetupInitializeServers(const FactoryTemplate
* pList
, int num
,
211 for (i
= 0; i
< num
; i
++, pList
++)
213 if (pList
->m_lpfnInit
)
214 pList
->m_lpfnInit(bLoading
, pList
->m_ClsID
);
218 BOOL WINAPI
STRMBASE_DllMain(HINSTANCE hInstDLL
, DWORD fdwReason
, LPVOID lpv
)
222 case DLL_PROCESS_ATTACH
:
224 DisableThreadLibraryCalls(hInstDLL
);
225 SetupInitializeServers(g_Templates
, g_cTemplates
, TRUE
);
227 case DLL_PROCESS_DETACH
:
228 SetupInitializeServers(g_Templates
, g_cTemplates
, FALSE
);
234 /******************************************************************************
238 IClassFactory IClassFactory_iface
;
240 LPFNNewCOMObject pfnCreateInstance
;
243 static inline IClassFactoryImpl
*impl_from_IClassFactory(IClassFactory
*iface
)
245 return CONTAINING_RECORD(iface
, IClassFactoryImpl
, IClassFactory_iface
);
248 static HRESULT WINAPI
DSCF_QueryInterface(IClassFactory
*iface
, REFIID riid
, void **ppobj
)
250 if (IsEqualGUID(riid
, &IID_IUnknown
) ||
251 IsEqualGUID(riid
, &IID_IClassFactory
))
253 IClassFactory_AddRef(iface
);
259 WARN("(%p)->(%s,%p), not found\n", iface
, debugstr_guid(riid
), ppobj
);
260 return E_NOINTERFACE
;
263 static ULONG WINAPI
DSCF_AddRef(IClassFactory
*iface
)
265 IClassFactoryImpl
*This
= impl_from_IClassFactory(iface
);
266 return InterlockedIncrement(&This
->ref
);
269 static ULONG WINAPI
DSCF_Release(IClassFactory
*iface
)
271 IClassFactoryImpl
*This
= impl_from_IClassFactory(iface
);
272 ULONG ref
= InterlockedDecrement(&This
->ref
);
275 HeapFree(GetProcessHeap(), 0, This
);
280 static HRESULT WINAPI
DSCF_CreateInstance(IClassFactory
*iface
, IUnknown
*pOuter
,
281 REFIID riid
, void **ppobj
)
283 IClassFactoryImpl
*This
= impl_from_IClassFactory(iface
);
284 HRESULT hres
= ERROR_SUCCESS
;
287 TRACE("(%p)->(%p,%s,%p)\n", This
, pOuter
, debugstr_guid(riid
), ppobj
);
292 /* Enforce the normal OLE rules regarding interfaces and delegation */
293 if (pOuter
&& !IsEqualGUID(riid
, &IID_IUnknown
))
294 return E_NOINTERFACE
;
297 punk
= This
->pfnCreateInstance(pOuter
, &hres
);
300 /* No object created, update error if it isn't done already and return */
302 hres
= E_OUTOFMEMORY
;
308 hres
= IUnknown_QueryInterface(punk
, riid
, ppobj
);
310 /* Releasing the object. If everything was successful, QueryInterface
311 should have incremented the refcount once more, otherwise this will
313 IUnknown_Release(punk
);
317 static HRESULT WINAPI
DSCF_LockServer(IClassFactory
*iface
, BOOL dolock
)
319 IClassFactoryImpl
*This
= impl_from_IClassFactory(iface
);
320 TRACE("(%p)->(%d)\n",This
, dolock
);
323 InterlockedIncrement(&server_locks
);
325 InterlockedDecrement(&server_locks
);
329 static const IClassFactoryVtbl DSCF_Vtbl
=
338 /***********************************************************************
341 HRESULT WINAPI
STRMBASE_DllGetClassObject(REFCLSID rclsid
, REFIID riid
, LPVOID
*ppv
)
343 const FactoryTemplate
*pList
= g_Templates
;
344 IClassFactoryImpl
*factory
;
347 TRACE("(%s,%s,%p)\n", debugstr_guid(rclsid
), debugstr_guid(riid
), ppv
);
354 if (!IsEqualGUID(&IID_IClassFactory
, riid
) &&
355 !IsEqualGUID(&IID_IUnknown
, riid
))
356 return E_NOINTERFACE
;
358 for (i
= 0; i
< g_cTemplates
; i
++, pList
++)
360 if (IsEqualGUID(pList
->m_ClsID
, rclsid
))
364 if (i
== g_cTemplates
)
366 char dllname
[MAX_PATH
];
367 if (!GetModuleFileNameA(g_hInst
, dllname
, sizeof(dllname
)))
368 strcpy(dllname
, "???");
369 ERR("%s: no class found in %s.\n", debugstr_guid(rclsid
), dllname
);
370 return CLASS_E_CLASSNOTAVAILABLE
;
372 else if (!pList
->m_lpfnNew
)
374 FIXME("%s: class not implemented yet.\n", debugstr_guid(rclsid
));
375 return CLASS_E_CLASSNOTAVAILABLE
;
378 factory
= HeapAlloc(GetProcessHeap(), 0, sizeof(IClassFactoryImpl
));
380 return E_OUTOFMEMORY
;
382 factory
->IClassFactory_iface
.lpVtbl
= &DSCF_Vtbl
;
385 factory
->pfnCreateInstance
= pList
->m_lpfnNew
;
387 *ppv
= &factory
->IClassFactory_iface
;
391 /***********************************************************************
394 HRESULT WINAPI
STRMBASE_DllCanUnloadNow(void)
398 if (server_locks
== 0)