2 * DirectX DLL registration and unregistration
4 * Copyright (C) 2005 Rolf Kalbermatter
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.
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.
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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
26 #define NONAMELESSSTRUCT
27 #define NONAMELESSUNION
38 #include "wine/unicode.h"
39 #include "wine/debug.h"
41 WINE_DEFAULT_DEBUG_CHANNEL(qcap);
44 * defines and constants
46 #define MAX_KEY_LEN 260
48 static WCHAR const clsid_keyname[6] =
49 {'C','L','S','I','D',0 };
50 static WCHAR const ips32_keyname[15] =
51 {'I','n','P','r','o','c','S','e','r','v','e','r','3','2',0};
52 static WCHAR const tmodel_keyname[15] =
53 {'T','h','r','e','a','d','i','n','g','M','o','d','e','l',0};
54 static WCHAR const tmodel_both[] =
58 * Delete a key and all its subkeys
60 HRESULT DeleteEntireSubKey(HKEY hkey, LPWSTR strSubKey)
62 WCHAR buffer[MAX_KEY_LEN];
63 DWORD dw = MAX_KEY_LEN;
66 LONG ret = RegOpenKeyExW(hkey, strSubKey, 0, MAXIMUM_ALLOWED, &hk);
68 if (ERROR_SUCCESS == ret)
70 /* Keep on enumerating the first key and deleting that */
75 ret = RegEnumKeyExW(hk, 0, buffer, &dw, NULL, NULL, NULL, &ft);
77 if (ERROR_SUCCESS == ret)
78 DeleteEntireSubKey(hk, buffer);
83 RegDeleteKeyW(hkey, strSubKey);
89 * SetupRegisterClass()
91 static HRESULT SetupRegisterClass(HKEY clsid, LPCWSTR szCLSID,
92 LPCWSTR szDescription,
95 LPCWSTR szThreadingModel)
98 LONG ret = RegCreateKeyW(clsid, szCLSID, &hkey);
99 if (ERROR_SUCCESS != ret)
100 return HRESULT_FROM_WIN32(ret);
102 /* set description string */
103 ret = RegSetValueW(hkey, NULL, REG_SZ, szDescription,
104 sizeof(WCHAR) * (lstrlenW(szDescription) + 1));
105 if (ERROR_SUCCESS != ret)
108 /* create CLSID\\{"CLSID"}\\"ServerType" key, using key to CLSID\\{"CLSID"}
109 passed back by last call to RegCreateKeyW(). */
110 ret = RegCreateKeyW(hkey, szServerType, &hsubkey);
111 if (ERROR_SUCCESS != ret)
114 /* set server path */
115 ret = RegSetValueW(hsubkey, NULL, REG_SZ, szFileName,
116 sizeof(WCHAR) * (lstrlenW(szFileName) + 1));
117 if (ERROR_SUCCESS != ret)
120 /* set threading model */
121 ret = RegSetValueExW(hsubkey, tmodel_keyname, 0L, REG_SZ,
122 (const BYTE*)szThreadingModel,
123 sizeof(WCHAR) * (lstrlenW(szThreadingModel) + 1));
126 RegCloseKey(hsubkey);
128 return HRESULT_FROM_WIN32(ret);
132 * SetupRegisterFilter through IFilterMapper2
134 static HRESULT SetupRegisterFilter2(const AMOVIESETUP_FILTER * const pSetup,
135 IFilterMapper2 * pIFM2, BOOL bRegister)
142 /* unregister filter */
143 hr = IFilterMapper2_UnregisterFilter(pIFM2, 0, 0, pSetup->clsID);
149 rf2.dwMerit = pSetup->dwMerit;
150 rf2.u.s.cPins = pSetup->nPins;
151 rf2.u.s.rgPins = pSetup->lpPin;
153 /* register filter */
154 hr = IFilterMapper2_RegisterFilter(pIFM2, pSetup->clsID,
155 pSetup->strName, 0, 0, NULL, &rf2);
159 /* filter not found is ignored here,
160 but there is no #define for 0x80070002 */
161 if (HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND) == hr)
168 * SetupRegisterFilter through IFilterMapper
170 static HRESULT SetupRegisterFilter(const AMOVIESETUP_FILTER * const pSetup,
171 IFilterMapper * pIFM, BOOL bRegister)
178 /* unregister filter */
179 hr = IFilterMapper_UnregisterFilter(pIFM, *pSetup->clsID);
183 /* register filter */
184 hr = IFilterMapper_RegisterFilter(pIFM, *pSetup->clsID,
185 pSetup->strName, pSetup->dwMerit);
188 const AMOVIESETUP_PIN *lpPin = pSetup->lpPin;
189 const AMOVIESETUP_MEDIATYPE *lpType;
192 for (i = 0; i < pSetup->nPins; i++, lpPin++)
194 hr = IFilterMapper_RegisterPin(pIFM, *(pSetup->clsID),
200 *(lpPin->clsConnectsToFilter),
201 lpPin->strConnectsToPin);
205 lpType = lpPin->lpMediaType;
207 /* and each pin's media types */
208 for (j = 0; j < lpPin->nMediaTypes; j++, lpType++)
210 hr = IFilterMapper_RegisterPinType(pIFM, *(pSetup->clsID),
212 *(lpType->clsMajorType),
213 *(lpType->clsMinorType));
214 if (FAILED(hr)) break;
216 if (FAILED(hr)) break;
218 if (FAILED(hr)) break;
224 /* filter not registered is ignored here, there is no definition for 0x80070002 */
225 if (HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND) == hr)
232 * RegisterAllClasses()
234 static HRESULT SetupRegisterAllClasses(const CFactoryTemplate * pList, int num,
235 LPCWSTR szFileName, BOOL bRegister)
237 HRESULT hr = NOERROR;
239 OLECHAR szCLSID[CHARS_IN_GUID];
240 LONG i, ret = RegCreateKeyW(HKEY_CLASSES_ROOT, clsid_keyname, &hkey);
241 if (ERROR_SUCCESS != ret)
242 return HRESULT_FROM_WIN32(ret);
244 for (i = 0; i < num; i++, pList++)
246 /* (un)register CLSID and InprocServer32 */
247 hr = StringFromGUID2(pList->m_ClsID, szCLSID, CHARS_IN_GUID);
251 hr = SetupRegisterClass(hkey, szCLSID,
252 pList->m_Name, szFileName,
253 ips32_keyname, tmodel_both);
255 hr = DeleteEntireSubKey(hkey, szCLSID);
263 /****************************************************************************
264 * SetupRegisterServers
266 * This function is table driven using the static members of the
267 * CFactoryTemplate class defined in the Dll.
269 * It registers the Dll as the InprocServer32 for all the classes in
272 ****************************************************************************/
273 HRESULT SetupRegisterServers(const CFactoryTemplate * pList, int num,
274 HINSTANCE hinst, BOOL bRegister)
276 HRESULT hr = NOERROR;
277 WCHAR szFileName[MAX_PATH];
278 IFilterMapper2 *pIFM2 = NULL;
279 IFilterMapper *pIFM = NULL;
281 /* Win95 wouldn't support the Unicode version of this API!! */
282 if (!GetModuleFileNameW(hinst, szFileName, MAX_PATH))
283 return HRESULT_FROM_WIN32(GetLastError());
285 /* first register all server classes, just to make sure */
287 hr = SetupRegisterAllClasses(pList, num, szFileName, TRUE );
289 /* next, register/unregister all filters */
292 hr = CoInitialize(NULL);
294 TRACE("Getting IFilterMapper2\r\n");
295 hr = CoCreateInstance(&CLSID_FilterMapper2, NULL, CLSCTX_INPROC_SERVER,
296 &IID_IFilterMapper2, (void **)&pIFM2);
299 TRACE("- trying IFilterMapper instead\r\n");
301 hr = CoCreateInstance(&CLSID_FilterMapper, NULL, CLSCTX_INPROC_SERVER,
302 &IID_IFilterMapper, (void **)&pIFM);
309 /* scan through array of CFactoryTemplates registering all filters */
310 for (i = 0; i < num; i++, pList++)
312 if (NULL != pList->m_pAMovieSetup_Filter)
315 hr = SetupRegisterFilter2(pList->m_pAMovieSetup_Filter,
318 hr = SetupRegisterFilter(pList->m_pAMovieSetup_Filter,
322 /* check final error for this pass and break loop if we failed */
327 /* release interface */
329 IFilterMapper2_Release(pIFM2);
331 IFilterMapper_Release(pIFM);
335 CoFreeUnusedLibraries();
339 /* if unregistering, unregister all OLE servers */
340 if (SUCCEEDED(hr) && !bRegister)
341 hr = SetupRegisterAllClasses(pList, num, szFileName, FALSE);
345 /****************************************************************************
346 * SetupInitializeServers
348 * This function is table driven using the static members of the
349 * CFactoryTemplate class defined in the Dll.
351 * It calls the intialize function for any class in CFactoryTemplate with
354 ****************************************************************************/
355 void SetupInitializeServers(const CFactoryTemplate * pList, int num,
360 for (i = 0; i < num; i++, pList++)
362 if (pList->m_lpfnInit)
363 pList->m_lpfnInit(bLoading, pList->m_ClsID);