/* * Copyright (C) 2003 Michael G�nnewig * Copyright (C) 2003 CodeWeavers Inc. (Ulrich Czekalla) * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA */ #include <stdarg.h> #include "windef.h" #include "winbase.h" #include "winuser.h" #include "winerror.h" #include "winreg.h" #include "objbase.h" #include "wine/unicode.h" #include "wine/debug.h" #include "initguid.h" #include "dmo.h" WINE_DEFAULT_DEBUG_CHANNEL(msdmo); static const WCHAR szDMORootKey[] = { 'D','i','r','e','c','t','S','h','o','w','\\', 'M','e','d','i','a','O','b','j','e','c','t','s',0 }; static const WCHAR szDMOInputType[] = { 'I','n','p','u','t','T','y','p','e','s',0 }; static const WCHAR szDMOOutputType[] = { 'O','u','t','p','u','t','T','y','p','e','s',0 }; static const WCHAR szDMOKeyed[] = { 'K','e','y','e','d',0 }; static const WCHAR szDMOCategories[] = { 'C','a','t','e','g','o','r','i','e','s',0 }; static const WCHAR szGUIDFmt[] = { '%','0','8','X','-','%','0','4','X','-','%','0','4','X','-','%','0', '2','X','%','0','2','X','-','%','0','2','X','%','0','2','X','%','0','2', 'X','%','0','2','X','%','0','2','X','%','0','2','X',0 }; static const WCHAR szCat3Fmt[] = { '%','s','\\','%','s','\\','%','s',0 }; static const WCHAR szCat2Fmt[] = { '%','s','\\','%','s',0 }; typedef struct { const IEnumDMOVtbl *lpVtbl; LONG ref; DWORD index; const GUID* guidCategory; DWORD dwFlags; DWORD cInTypes; DMO_PARTIAL_MEDIATYPE *pInTypes; DWORD cOutTypes; DMO_PARTIAL_MEDIATYPE *pOutTypes; HKEY hkey; } IEnumDMOImpl; static const IEnumDMOVtbl edmovt; static LPWSTR GUIDToString(LPWSTR lpwstr, REFGUID lpcguid) { wsprintfW(lpwstr, szGUIDFmt, lpcguid->Data1, lpcguid->Data2, lpcguid->Data3, lpcguid->Data4[0], lpcguid->Data4[1], lpcguid->Data4[2], lpcguid->Data4[3], lpcguid->Data4[4], lpcguid->Data4[5], lpcguid->Data4[6], lpcguid->Data4[7]); return lpwstr; } static BOOL IsMediaTypeEqual(DMO_PARTIAL_MEDIATYPE* mt1, DMO_PARTIAL_MEDIATYPE* mt2) { return (IsEqualCLSID(&mt1->type, &mt2->type) || IsEqualCLSID(&mt2->type, &GUID_NULL) || IsEqualCLSID(&mt1->type, &GUID_NULL)) && (IsEqualCLSID(&mt1->subtype, &mt2->subtype) || IsEqualCLSID(&mt2->subtype, &GUID_NULL) || IsEqualCLSID(&mt1->subtype, &GUID_NULL)); } /*************************************************************** * DMORegister * * Register a DirectX Media Object. */ HRESULT WINAPI DMORegister( LPCWSTR szName, REFCLSID clsidDMO, REFGUID guidCategory, DWORD dwFlags, DWORD cInTypes, const DMO_PARTIAL_MEDIATYPE *pInTypes, DWORD cOutTypes, const DMO_PARTIAL_MEDIATYPE *pOutTypes ) { WCHAR szguid[64]; HRESULT hres; HKEY hrkey = 0; HKEY hkey = 0; HKEY hckey = 0; HKEY hclskey = 0; TRACE("%s\n", debugstr_w(szName)); hres = RegOpenKeyExW(HKEY_CLASSES_ROOT, szDMORootKey, 0, KEY_WRITE, &hrkey); if (ERROR_SUCCESS != hres) goto lend; /* Create clsidDMO key under MediaObjects */ hres = RegCreateKeyExW(hrkey, GUIDToString(szguid, clsidDMO), 0, NULL, REG_OPTION_NON_VOLATILE, KEY_WRITE, NULL, &hkey, NULL); if (ERROR_SUCCESS != hres) goto lend; /* Set default Name value */ hres = RegSetValueExW(hkey, NULL, 0, REG_SZ, (const BYTE*) szName, (strlenW(szName) + 1)) * sizeof(WCHAR); /* Set InputTypes */ hres = RegSetValueExW(hkey, szDMOInputType, 0, REG_BINARY, (const BYTE*) pInTypes, cInTypes * sizeof(DMO_PARTIAL_MEDIATYPE)); /* Set OutputTypes */ hres = RegSetValueExW(hkey, szDMOOutputType, 0, REG_BINARY, (const BYTE*) pOutTypes, cOutTypes * sizeof(DMO_PARTIAL_MEDIATYPE)); if (dwFlags & DMO_REGISTERF_IS_KEYED) { /* Create Keyed key */ hres = RegCreateKeyExW(hkey, szDMOKeyed, 0, NULL, REG_OPTION_NON_VOLATILE, KEY_WRITE, NULL, &hckey, NULL); if (ERROR_SUCCESS != hres) goto lend; RegCloseKey(hckey); } /* Register the category */ hres = RegOpenKeyExW(hrkey, szDMOCategories, 0, KEY_WRITE, &hckey); if (ERROR_SUCCESS != hres) goto lend; RegCloseKey(hkey); hres = RegOpenKeyExW(hckey, GUIDToString(szguid, guidCategory), 0, KEY_WRITE, &hkey); if (ERROR_SUCCESS != hres) goto lend; hres = RegCreateKeyExW(hkey, GUIDToString(szguid, clsidDMO), 0, NULL, REG_OPTION_NON_VOLATILE, KEY_WRITE, NULL, &hclskey, NULL); if (ERROR_SUCCESS != hres) goto lend; lend: if (hkey) RegCloseKey(hkey); if (hckey) RegCloseKey(hckey); if (hclskey) RegCloseKey(hclskey); if (hrkey) RegCloseKey(hrkey); TRACE(" hresult=0x%08x\n", hres); return hres; } /*************************************************************** * DMOUnregister * * Unregister a DirectX Media Object. */ HRESULT WINAPI DMOUnregister(REFCLSID clsidDMO, REFGUID guidCategory) { HRESULT hres; WCHAR szguid[64]; HKEY hrkey = 0; HKEY hckey = 0; GUIDToString(szguid, clsidDMO); TRACE("%s %p\n", debugstr_w(szguid), guidCategory); hres = RegOpenKeyExW(HKEY_CLASSES_ROOT, szDMORootKey, 0, KEY_WRITE, &hrkey); if (ERROR_SUCCESS != hres) goto lend; hres = RegDeleteKeyW(hrkey, szguid); if (ERROR_SUCCESS != hres) goto lend; hres = RegOpenKeyExW(hrkey, szDMOCategories, 0, KEY_WRITE, &hckey); if (ERROR_SUCCESS != hres) goto lend; hres = RegDeleteKeyW(hckey, szguid); if (ERROR_SUCCESS != hres) goto lend; lend: if (hckey) RegCloseKey(hckey); if (hrkey) RegCloseKey(hrkey); return hres; } /*************************************************************** * DMOGetName * * Get DMP Name from the registry */ HRESULT WINAPI DMOGetName(REFCLSID clsidDMO, WCHAR* szName) { WCHAR szguid[64]; HRESULT hres; HKEY hrkey = 0; HKEY hkey = 0; DWORD count; TRACE("%s\n", debugstr_guid(clsidDMO)); hres = RegOpenKeyExW(HKEY_CLASSES_ROOT, szDMORootKey, 0, KEY_READ, &hrkey); if (ERROR_SUCCESS != hres) goto lend; hres = RegOpenKeyExW(hrkey, GUIDToString(szguid, clsidDMO), 0, KEY_READ, &hkey); if (ERROR_SUCCESS != hres) goto lend; count = 80 * sizeof(WCHAR); /* 80 by API definition */ hres = RegQueryValueExW(hkey, NULL, NULL, NULL, (LPBYTE) szName, &count); TRACE(" szName=%s\n", debugstr_w(szName)); lend: if (hkey) RegCloseKey(hrkey); if (hkey) RegCloseKey(hkey); return hres; } /************************************************************************** * IEnumDMO_Destructor */ static BOOL IEnumDMO_Destructor(IEnumDMO* iface) { IEnumDMOImpl *This = (IEnumDMOImpl *)iface; TRACE("%p\n", This); if (This->hkey) RegCloseKey(This->hkey); HeapFree(GetProcessHeap(), 0, This->pInTypes); HeapFree(GetProcessHeap(), 0, This->pOutTypes); return TRUE; } /************************************************************************** * IEnumDMO_Constructor */ IEnumDMO * IEnumDMO_Constructor( REFGUID guidCategory, DWORD dwFlags, DWORD cInTypes, const DMO_PARTIAL_MEDIATYPE *pInTypes, DWORD cOutTypes, const DMO_PARTIAL_MEDIATYPE *pOutTypes) { UINT size; IEnumDMOImpl* lpedmo; BOOL ret = FALSE; lpedmo = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(IEnumDMOImpl)); if (lpedmo) { lpedmo->ref = 1; lpedmo->lpVtbl = &edmovt; lpedmo->index = -1; lpedmo->guidCategory = guidCategory; lpedmo->dwFlags = dwFlags; size = cInTypes * sizeof(DMO_PARTIAL_MEDIATYPE); lpedmo->pInTypes = HeapAlloc(GetProcessHeap(), 0, size); if (!lpedmo->pInTypes) goto lerr; memcpy(lpedmo->pInTypes, pInTypes, size); lpedmo->cInTypes = cInTypes; size = cOutTypes * sizeof(DMO_PARTIAL_MEDIATYPE); lpedmo->pOutTypes = HeapAlloc(GetProcessHeap(), 0, size); if (!lpedmo->pOutTypes) goto lerr; memcpy(lpedmo->pOutTypes, pOutTypes, size); lpedmo->cOutTypes = cOutTypes; /* If not filtering by category enum from media objects root */ if (IsEqualGUID(guidCategory, &GUID_NULL)) { if (ERROR_SUCCESS == RegOpenKeyExW(HKEY_CLASSES_ROOT, szDMORootKey, 0, KEY_READ, &lpedmo->hkey)) ret = TRUE; } else { WCHAR szguid[64]; WCHAR szKey[MAX_PATH]; wsprintfW(szKey, szCat3Fmt, szDMORootKey, szDMOCategories, GUIDToString(szguid, guidCategory)); if (ERROR_SUCCESS == RegOpenKeyExW(HKEY_CLASSES_ROOT, szKey, 0, KEY_READ, &lpedmo->hkey)) ret = TRUE; } lerr: if(!ret) { IEnumDMO_Destructor((IEnumDMO*)lpedmo); HeapFree(GetProcessHeap(),0,lpedmo); lpedmo = NULL; } } TRACE("returning %p\n", lpedmo); return (IEnumDMO*)lpedmo; } /****************************************************************************** * IEnumDMO_fnAddRef */ static ULONG WINAPI IEnumDMO_fnAddRef(IEnumDMO * iface) { IEnumDMOImpl *This = (IEnumDMOImpl *)iface; return InterlockedIncrement(&This->ref); } /************************************************************************** * EnumDMO_QueryInterface */ static HRESULT WINAPI IEnumDMO_fnQueryInterface( IEnumDMO* iface, REFIID riid, LPVOID *ppvObj) { IEnumDMOImpl *This = (IEnumDMOImpl *)iface; *ppvObj = NULL; if(IsEqualIID(riid, &IID_IUnknown)) *ppvObj = This; else if(IsEqualIID(riid, &IID_IEnumDMO)) *ppvObj = (IEnumDMO*)This; if(*ppvObj) { IEnumDMO_fnAddRef((IEnumDMO*)*ppvObj); return S_OK; } return E_NOINTERFACE; } /****************************************************************************** * IEnumDMO_fnRelease */ static ULONG WINAPI IEnumDMO_fnRelease(IEnumDMO * iface) { IEnumDMOImpl *This = (IEnumDMOImpl *)iface; ULONG refCount = InterlockedDecrement(&This->ref); if (!refCount) { IEnumDMO_Destructor((IEnumDMO*)This); HeapFree(GetProcessHeap(),0,This); } return refCount; } /****************************************************************************** * IEnumDMO_fnNext */ static HRESULT WINAPI IEnumDMO_fnNext( IEnumDMO * iface, DWORD cItemsToFetch, CLSID * pCLSID, WCHAR ** Names, DWORD * pcItemsFetched) { FILETIME ft; HKEY hkey; WCHAR szNextKey[MAX_PATH]; WCHAR szKey[MAX_PATH]; WCHAR szValue[MAX_PATH]; DWORD len; UINT count = 0; HRESULT hres = S_OK; IEnumDMOImpl *This = (IEnumDMOImpl *)iface; TRACE("%d\n", cItemsToFetch); if (!pCLSID || !Names || !pcItemsFetched) return E_POINTER; while (count < cItemsToFetch) { This->index++; hres = RegEnumKeyExW(This->hkey, This->index, szNextKey, &len, NULL, NULL, NULL, &ft); if (hres != ERROR_SUCCESS) break; TRACE("found %s\n", debugstr_w(szNextKey)); if (This->dwFlags & DMO_REGISTERF_IS_KEYED) { wsprintfW(szKey, szCat3Fmt, szDMORootKey, szNextKey, szDMOKeyed); hres = RegOpenKeyExW(HKEY_CLASSES_ROOT, szKey, 0, KEY_READ, &hkey); if (ERROR_SUCCESS != hres) continue; RegCloseKey(hkey); } wsprintfW(szKey, szCat2Fmt, szDMORootKey, szNextKey); hres = RegOpenKeyExW(HKEY_CLASSES_ROOT, szKey, 0, KEY_READ, &hkey); if (This->pInTypes) { UINT i, j; DWORD cInTypes; DMO_PARTIAL_MEDIATYPE* pInTypes; len = MAX_PATH * sizeof(WCHAR); hres = RegQueryValueExW(hkey, szDMOInputType, NULL, NULL, (LPBYTE) szValue, &len); if (ERROR_SUCCESS != hres) { RegCloseKey(hkey); continue; } cInTypes = len / sizeof(DMO_PARTIAL_MEDIATYPE); pInTypes = (DMO_PARTIAL_MEDIATYPE*) szValue; for (i = 0; i < This->cInTypes; i++) { for (j = 0; j < cInTypes; j++) { if (IsMediaTypeEqual(&pInTypes[j], &This->pInTypes[i])) break; } if (j >= cInTypes) break; } if (i < This->cInTypes) { RegCloseKey(hkey); continue; } } if (This->pOutTypes) { UINT i, j; DWORD cOutTypes; DMO_PARTIAL_MEDIATYPE* pOutTypes; len = MAX_PATH * sizeof(WCHAR); hres = RegQueryValueExW(hkey, szDMOOutputType, NULL, NULL, (LPBYTE) szValue, &len); if (ERROR_SUCCESS != hres) { RegCloseKey(hkey); continue; } cOutTypes = len / sizeof(DMO_PARTIAL_MEDIATYPE); pOutTypes = (DMO_PARTIAL_MEDIATYPE*) szValue; for (i = 0; i < This->cOutTypes; i++) { for (j = 0; j < cOutTypes; j++) { if (IsMediaTypeEqual(&pOutTypes[j], &This->pOutTypes[i])) break; } if (j >= cOutTypes) break; } if (i < This->cOutTypes) { RegCloseKey(hkey); continue; } } /* Media object wasn't filtered so add it to return list */ Names[count] = NULL; len = MAX_PATH * sizeof(WCHAR); hres = RegQueryValueExW(hkey, NULL, NULL, NULL, (LPBYTE) szValue, &len); if (ERROR_SUCCESS == hres) { Names[count] = HeapAlloc(GetProcessHeap(), 0, strlenW(szValue) + 1); if (Names[count]) strcmpW(Names[count], szValue); } CLSIDFromString(szNextKey, &pCLSID[count]); TRACE("found match %s %s\n", debugstr_w(szValue), debugstr_w(szNextKey)); RegCloseKey(hkey); count++; } *pcItemsFetched = count; if (*pcItemsFetched < cItemsToFetch) hres = S_FALSE; return hres; } /****************************************************************************** * IEnumDMO_fnSkip */ static HRESULT WINAPI IEnumDMO_fnSkip(IEnumDMO * iface, DWORD cItemsToSkip) { IEnumDMOImpl *This = (IEnumDMOImpl *)iface; This->index += cItemsToSkip; return S_OK; } /****************************************************************************** * IEnumDMO_fnReset */ static HRESULT WINAPI IEnumDMO_fnReset(IEnumDMO * iface) { IEnumDMOImpl *This = (IEnumDMOImpl *)iface; This->index = -1; return S_OK; } /****************************************************************************** * IEnumDMO_fnClone */ static HRESULT WINAPI IEnumDMO_fnClone(IEnumDMO * iface, IEnumDMO **ppEnum) { IEnumDMOImpl *This = (IEnumDMOImpl *)iface; FIXME("(%p)->() to (%p)->() E_NOTIMPL\n", This, ppEnum); return E_NOTIMPL; } /*************************************************************** * DMOEnum * * Enumerate DirectX Media Objects in the registry. */ HRESULT WINAPI DMOEnum( REFGUID guidCategory, DWORD dwFlags, DWORD cInTypes, const DMO_PARTIAL_MEDIATYPE *pInTypes, DWORD cOutTypes, const DMO_PARTIAL_MEDIATYPE *pOutTypes, IEnumDMO **ppEnum) { HRESULT hres = E_FAIL; TRACE("guidCategory=%p dwFlags=0x%08x cInTypes=%d cOutTypes=%d\n", guidCategory, dwFlags, cInTypes, cOutTypes); *ppEnum = IEnumDMO_Constructor(guidCategory, dwFlags, cInTypes, pInTypes, cOutTypes, pOutTypes); if (*ppEnum) hres = S_OK; return hres; } static const IEnumDMOVtbl edmovt = { IEnumDMO_fnQueryInterface, IEnumDMO_fnAddRef, IEnumDMO_fnRelease, IEnumDMO_fnNext, IEnumDMO_fnSkip, IEnumDMO_fnReset, IEnumDMO_fnClone, }; HRESULT WINAPI DMOGetTypes(REFCLSID a, unsigned long b, unsigned long* c, DMO_PARTIAL_MEDIATYPE* d, unsigned long e, unsigned long* f, DMO_PARTIAL_MEDIATYPE* g) { FIXME("(%p,%lu,%p,%p,%lu,%p,%p),stub!\n",a,b,c,d,e,f,g); return E_NOTIMPL; }