/* * Copyright 2009 Vincent Povirk for CodeWeavers * Copyright 2012 Dmitry Timoshkov * * 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 "config.h" #include <stdarg.h> #define COBJMACROS #include "windef.h" #include "winbase.h" #include "winreg.h" #include "objbase.h" #include "wincodecs_private.h" #include "wine/debug.h" #include "wine/unicode.h" #include "wine/list.h" #include "wine/rbtree.h" #include "wine/heap.h" WINE_DEFAULT_DEBUG_CHANNEL(wincodecs); static const WCHAR mimetypes_valuename[] = {'M','i','m','e','T','y','p','e','s',0}; static const WCHAR author_valuename[] = {'A','u','t','h','o','r',0}; static const WCHAR friendlyname_valuename[] = {'F','r','i','e','n','d','l','y','N','a','m','e',0}; static const WCHAR pixelformats_keyname[] = {'P','i','x','e','l','F','o','r','m','a','t','s',0}; static const WCHAR formats_keyname[] = {'F','o','r','m','a','t','s',0}; static const WCHAR containerformat_valuename[] = {'C','o','n','t','a','i','n','e','r','F','o','r','m','a','t',0}; static const WCHAR metadataformat_valuename[] = {'M','e','t','a','d','a','t','a','F','o','r','m','a','t',0}; static const WCHAR vendor_valuename[] = {'V','e','n','d','o','r',0}; static const WCHAR version_valuename[] = {'V','e','r','s','i','o','n',0}; static const WCHAR specversion_valuename[] = {'S','p','e','c','V','e','r','s','i','o','n',0}; static const WCHAR bitsperpixel_valuename[] = {'B','i','t','L','e','n','g','t','h',0}; static const WCHAR channelcount_valuename[] = {'C','h','a','n','n','e','l','C','o','u','n','t',0}; static const WCHAR channelmasks_keyname[] = {'C','h','a','n','n','e','l','M','a','s','k','s',0}; static const WCHAR numericrepresentation_valuename[] = {'N','u','m','e','r','i','c','R','e','p','r','e','s','e','n','t','a','t','i','o','n',0}; static const WCHAR supportstransparency_valuename[] = {'S','u','p','p','o','r','t','s','T','r','a','n','s','p','a','r','e','n','c','y',0}; static const WCHAR requiresfullstream_valuename[] = {'R','e','q','u','i','r','e','s','F','u','l','l','S','t','r','e','a','m',0}; static const WCHAR supportspadding_valuename[] = {'S','u','p','p','o','r','t','s','P','a','d','d','i','n','g',0}; static const WCHAR fileextensions_valuename[] = {'F','i','l','e','E','x','t','e','n','s','i','o','n','s',0}; static const WCHAR containers_keyname[] = {'C','o','n','t','a','i','n','e','r','s',0}; typedef struct { IWICComponentInfo IWICComponentInfo_iface; LONG ref; CLSID clsid; struct wine_rb_entry entry; } ComponentInfo; static HRESULT ComponentInfo_GetStringValue(HKEY classkey, LPCWSTR value, UINT buffer_size, WCHAR *buffer, UINT *actual_size) { LONG ret; DWORD cbdata=buffer_size * sizeof(WCHAR); if (!actual_size) return E_INVALIDARG; ret = RegGetValueW(classkey, NULL, value, RRF_RT_REG_SZ|RRF_NOEXPAND, NULL, buffer, &cbdata); if (ret == ERROR_FILE_NOT_FOUND) { *actual_size = 0; return S_OK; } if (ret == 0 || ret == ERROR_MORE_DATA) *actual_size = cbdata/sizeof(WCHAR); if (!buffer && buffer_size != 0) /* Yes, native returns the correct size in this case. */ return E_INVALIDARG; if (ret == ERROR_MORE_DATA) return WINCODEC_ERR_INSUFFICIENTBUFFER; return HRESULT_FROM_WIN32(ret); } static HRESULT ComponentInfo_GetGUIDValue(HKEY classkey, LPCWSTR value, GUID *result) { LONG ret; WCHAR guid_string[39]; DWORD cbdata = sizeof(guid_string); HRESULT hr; if (!result) return E_INVALIDARG; ret = RegGetValueW(classkey, NULL, value, RRF_RT_REG_SZ|RRF_NOEXPAND, NULL, guid_string, &cbdata); if (ret != ERROR_SUCCESS) return HRESULT_FROM_WIN32(ret); if (cbdata < sizeof(guid_string)) { ERR("incomplete GUID value\n"); return E_FAIL; } hr = CLSIDFromString(guid_string, result); return hr; } static HRESULT ComponentInfo_GetDWORDValue(HKEY classkey, LPCWSTR value, DWORD *result) { LONG ret; DWORD cbdata = sizeof(DWORD); if (!result) return E_INVALIDARG; ret = RegGetValueW(classkey, NULL, value, RRF_RT_DWORD, NULL, result, &cbdata); if (ret == ERROR_FILE_NOT_FOUND) { *result = 0; return S_OK; } return HRESULT_FROM_WIN32(ret); } static HRESULT ComponentInfo_GetGuidList(HKEY classkey, LPCWSTR subkeyname, UINT buffersize, GUID *buffer, UINT *actual_size) { LONG ret; HKEY subkey; UINT items_returned; WCHAR guid_string[39]; DWORD guid_string_size; HRESULT hr=S_OK; if (!actual_size) return E_INVALIDARG; ret = RegOpenKeyExW(classkey, subkeyname, 0, KEY_READ, &subkey); if (ret == ERROR_FILE_NOT_FOUND) { *actual_size = 0; return S_OK; } else if (ret != ERROR_SUCCESS) return HRESULT_FROM_WIN32(ret); if (buffer) { items_returned = 0; guid_string_size = 39; while (items_returned < buffersize) { ret = RegEnumKeyExW(subkey, items_returned, guid_string, &guid_string_size, NULL, NULL, NULL, NULL); if (ret != ERROR_SUCCESS) { hr = HRESULT_FROM_WIN32(ret); break; } if (guid_string_size != 38) { hr = E_FAIL; break; } hr = CLSIDFromString(guid_string, &buffer[items_returned]); if (FAILED(hr)) break; items_returned++; guid_string_size = 39; } if (ret == ERROR_NO_MORE_ITEMS) hr = S_OK; *actual_size = items_returned; } else { ret = RegQueryInfoKeyW(subkey, NULL, NULL, NULL, actual_size, NULL, NULL, NULL, NULL, NULL, NULL, NULL); if (ret != ERROR_SUCCESS) hr = HRESULT_FROM_WIN32(ret); } RegCloseKey(subkey); return hr; } typedef struct { ComponentInfo base; HKEY classkey; WICBitmapPattern *patterns; UINT pattern_count; UINT patterns_size; } BitmapDecoderInfo; static inline BitmapDecoderInfo *impl_from_IWICBitmapDecoderInfo(IWICBitmapDecoderInfo *iface) { return CONTAINING_RECORD(iface, BitmapDecoderInfo, base.IWICComponentInfo_iface); } static HRESULT WINAPI BitmapDecoderInfo_QueryInterface(IWICBitmapDecoderInfo *iface, REFIID iid, void **ppv) { BitmapDecoderInfo *This = impl_from_IWICBitmapDecoderInfo(iface); TRACE("(%p,%s,%p)\n", iface, debugstr_guid(iid), ppv); if (!ppv) return E_INVALIDARG; if (IsEqualIID(&IID_IUnknown, iid) || IsEqualIID(&IID_IWICComponentInfo, iid) || IsEqualIID(&IID_IWICBitmapCodecInfo, iid) || IsEqualIID(&IID_IWICBitmapDecoderInfo ,iid)) { *ppv = &This->base.IWICComponentInfo_iface; } else { *ppv = NULL; return E_NOINTERFACE; } IUnknown_AddRef((IUnknown*)*ppv); return S_OK; } static ULONG WINAPI BitmapDecoderInfo_AddRef(IWICBitmapDecoderInfo *iface) { BitmapDecoderInfo *This = impl_from_IWICBitmapDecoderInfo(iface); ULONG ref = InterlockedIncrement(&This->base.ref); TRACE("(%p) refcount=%u\n", iface, ref); return ref; } static ULONG WINAPI BitmapDecoderInfo_Release(IWICBitmapDecoderInfo *iface) { BitmapDecoderInfo *This = impl_from_IWICBitmapDecoderInfo(iface); ULONG ref = InterlockedDecrement(&This->base.ref); TRACE("(%p) refcount=%u\n", iface, ref); if (ref == 0) { RegCloseKey(This->classkey); heap_free(This->patterns); HeapFree(GetProcessHeap(), 0, This); } return ref; } static HRESULT WINAPI BitmapDecoderInfo_GetComponentType(IWICBitmapDecoderInfo *iface, WICComponentType *pType) { TRACE("(%p,%p)\n", iface, pType); if (!pType) return E_INVALIDARG; *pType = WICDecoder; return S_OK; } static HRESULT WINAPI BitmapDecoderInfo_GetCLSID(IWICBitmapDecoderInfo *iface, CLSID *pclsid) { BitmapDecoderInfo *This = impl_from_IWICBitmapDecoderInfo(iface); TRACE("(%p,%p)\n", iface, pclsid); if (!pclsid) return E_INVALIDARG; *pclsid = This->base.clsid; return S_OK; } static HRESULT WINAPI BitmapDecoderInfo_GetSigningStatus(IWICBitmapDecoderInfo *iface, DWORD *pStatus) { FIXME("(%p,%p): stub\n", iface, pStatus); return E_NOTIMPL; } static HRESULT WINAPI BitmapDecoderInfo_GetAuthor(IWICBitmapDecoderInfo *iface, UINT cchAuthor, WCHAR *wzAuthor, UINT *pcchActual) { BitmapDecoderInfo *This = impl_from_IWICBitmapDecoderInfo(iface); TRACE("(%p,%u,%p,%p)\n", iface, cchAuthor, wzAuthor, pcchActual); return ComponentInfo_GetStringValue(This->classkey, author_valuename, cchAuthor, wzAuthor, pcchActual); } static HRESULT WINAPI BitmapDecoderInfo_GetVendorGUID(IWICBitmapDecoderInfo *iface, GUID *pguidVendor) { BitmapDecoderInfo *This = impl_from_IWICBitmapDecoderInfo(iface); TRACE("(%p,%p)\n", iface, pguidVendor); return ComponentInfo_GetGUIDValue(This->classkey, vendor_valuename, pguidVendor); } static HRESULT WINAPI BitmapDecoderInfo_GetVersion(IWICBitmapDecoderInfo *iface, UINT cchVersion, WCHAR *wzVersion, UINT *pcchActual) { BitmapDecoderInfo *This = impl_from_IWICBitmapDecoderInfo(iface); TRACE("(%p,%u,%p,%p)\n", iface, cchVersion, wzVersion, pcchActual); return ComponentInfo_GetStringValue(This->classkey, version_valuename, cchVersion, wzVersion, pcchActual); } static HRESULT WINAPI BitmapDecoderInfo_GetSpecVersion(IWICBitmapDecoderInfo *iface, UINT cchSpecVersion, WCHAR *wzSpecVersion, UINT *pcchActual) { BitmapDecoderInfo *This = impl_from_IWICBitmapDecoderInfo(iface); TRACE("(%p,%u,%p,%p)\n", iface, cchSpecVersion, wzSpecVersion, pcchActual); return ComponentInfo_GetStringValue(This->classkey, specversion_valuename, cchSpecVersion, wzSpecVersion, pcchActual); } static HRESULT WINAPI BitmapDecoderInfo_GetFriendlyName(IWICBitmapDecoderInfo *iface, UINT cchFriendlyName, WCHAR *wzFriendlyName, UINT *pcchActual) { BitmapDecoderInfo *This = impl_from_IWICBitmapDecoderInfo(iface); TRACE("(%p,%u,%p,%p)\n", iface, cchFriendlyName, wzFriendlyName, pcchActual); return ComponentInfo_GetStringValue(This->classkey, friendlyname_valuename, cchFriendlyName, wzFriendlyName, pcchActual); } static HRESULT WINAPI BitmapDecoderInfo_GetContainerFormat(IWICBitmapDecoderInfo *iface, GUID *pguidContainerFormat) { BitmapDecoderInfo *This = impl_from_IWICBitmapDecoderInfo(iface); TRACE("(%p,%p)\n", iface, pguidContainerFormat); return ComponentInfo_GetGUIDValue(This->classkey, containerformat_valuename, pguidContainerFormat); } static HRESULT WINAPI BitmapDecoderInfo_GetPixelFormats(IWICBitmapDecoderInfo *iface, UINT cFormats, GUID *pguidPixelFormats, UINT *pcActual) { BitmapDecoderInfo *This = impl_from_IWICBitmapDecoderInfo(iface); TRACE("(%p,%u,%p,%p)\n", iface, cFormats, pguidPixelFormats, pcActual); return ComponentInfo_GetGuidList(This->classkey, formats_keyname, cFormats, pguidPixelFormats, pcActual); } static HRESULT WINAPI BitmapDecoderInfo_GetColorManagementVersion(IWICBitmapDecoderInfo *iface, UINT cchColorManagementVersion, WCHAR *wzColorManagementVersion, UINT *pcchActual) { FIXME("(%p,%u,%p,%p): stub\n", iface, cchColorManagementVersion, wzColorManagementVersion, pcchActual); return E_NOTIMPL; } static HRESULT WINAPI BitmapDecoderInfo_GetDeviceManufacturer(IWICBitmapDecoderInfo *iface, UINT cchDeviceManufacturer, WCHAR *wzDeviceManufacturer, UINT *pcchActual) { FIXME("(%p,%u,%p,%p): stub\n", iface, cchDeviceManufacturer, wzDeviceManufacturer, pcchActual); return E_NOTIMPL; } static HRESULT WINAPI BitmapDecoderInfo_GetDeviceModels(IWICBitmapDecoderInfo *iface, UINT cchDeviceModels, WCHAR *wzDeviceModels, UINT *pcchActual) { FIXME("(%p,%u,%p,%p): stub\n", iface, cchDeviceModels, wzDeviceModels, pcchActual); return E_NOTIMPL; } static HRESULT WINAPI BitmapDecoderInfo_GetMimeTypes(IWICBitmapDecoderInfo *iface, UINT cchMimeTypes, WCHAR *wzMimeTypes, UINT *pcchActual) { BitmapDecoderInfo *This = impl_from_IWICBitmapDecoderInfo(iface); TRACE("(%p,%u,%p,%p)\n", iface, cchMimeTypes, wzMimeTypes, pcchActual); return ComponentInfo_GetStringValue(This->classkey, mimetypes_valuename, cchMimeTypes, wzMimeTypes, pcchActual); } static HRESULT WINAPI BitmapDecoderInfo_GetFileExtensions(IWICBitmapDecoderInfo *iface, UINT cchFileExtensions, WCHAR *wzFileExtensions, UINT *pcchActual) { BitmapDecoderInfo *This = impl_from_IWICBitmapDecoderInfo(iface); TRACE("(%p,%u,%p,%p)\n", iface, cchFileExtensions, wzFileExtensions, pcchActual); return ComponentInfo_GetStringValue(This->classkey, fileextensions_valuename, cchFileExtensions, wzFileExtensions, pcchActual); } static HRESULT WINAPI BitmapDecoderInfo_DoesSupportAnimation(IWICBitmapDecoderInfo *iface, BOOL *pfSupportAnimation) { FIXME("(%p,%p): stub\n", iface, pfSupportAnimation); return E_NOTIMPL; } static HRESULT WINAPI BitmapDecoderInfo_DoesSupportChromaKey(IWICBitmapDecoderInfo *iface, BOOL *pfSupportChromaKey) { FIXME("(%p,%p): stub\n", iface, pfSupportChromaKey); return E_NOTIMPL; } static HRESULT WINAPI BitmapDecoderInfo_DoesSupportLossless(IWICBitmapDecoderInfo *iface, BOOL *pfSupportLossless) { FIXME("(%p,%p): stub\n", iface, pfSupportLossless); return E_NOTIMPL; } static HRESULT WINAPI BitmapDecoderInfo_DoesSupportMultiframe(IWICBitmapDecoderInfo *iface, BOOL *pfSupportMultiframe) { FIXME("(%p,%p): stub\n", iface, pfSupportMultiframe); return E_NOTIMPL; } static HRESULT WINAPI BitmapDecoderInfo_MatchesMimeType(IWICBitmapDecoderInfo *iface, LPCWSTR wzMimeType, BOOL *pfMatches) { FIXME("(%p,%s,%p): stub\n", iface, debugstr_w(wzMimeType), pfMatches); return E_NOTIMPL; } static HRESULT WINAPI BitmapDecoderInfo_GetPatterns(IWICBitmapDecoderInfo *iface, UINT cbSizePatterns, WICBitmapPattern *pPatterns, UINT *pcPatterns, UINT *pcbPatternsActual) { BitmapDecoderInfo *This = impl_from_IWICBitmapDecoderInfo(iface); TRACE("(%p,%i,%p,%p,%p)\n", iface, cbSizePatterns, pPatterns, pcPatterns, pcbPatternsActual); if (!pcPatterns || !pcbPatternsActual) return E_INVALIDARG; *pcPatterns = This->pattern_count; *pcbPatternsActual = This->patterns_size; if (pPatterns) { if (This->patterns_size && cbSizePatterns < This->patterns_size) return WINCODEC_ERR_INSUFFICIENTBUFFER; memcpy(pPatterns, This->patterns, This->patterns_size); } return S_OK; } static HRESULT WINAPI BitmapDecoderInfo_MatchesPattern(IWICBitmapDecoderInfo *iface, IStream *pIStream, BOOL *pfMatches) { BitmapDecoderInfo *This = impl_from_IWICBitmapDecoderInfo(iface); HRESULT hr; UINT i; ULONG pos; BYTE *data=NULL; ULONG datasize=0; ULONG bytesread; LARGE_INTEGER seekpos; TRACE("(%p,%p,%p)\n", iface, pIStream, pfMatches); for (i=0; i < This->pattern_count; i++) { if (datasize < This->patterns[i].Length) { HeapFree(GetProcessHeap(), 0, data); datasize = This->patterns[i].Length; data = HeapAlloc(GetProcessHeap(), 0, This->patterns[i].Length); if (!data) { hr = E_OUTOFMEMORY; break; } } if (This->patterns[i].EndOfStream) seekpos.QuadPart = -This->patterns[i].Position.QuadPart; else seekpos.QuadPart = This->patterns[i].Position.QuadPart; hr = IStream_Seek(pIStream, seekpos, This->patterns[i].EndOfStream ? STREAM_SEEK_END : STREAM_SEEK_SET, NULL); if (hr == STG_E_INVALIDFUNCTION) continue; /* before start of stream */ if (FAILED(hr)) break; hr = IStream_Read(pIStream, data, This->patterns[i].Length, &bytesread); if (hr == S_FALSE || (hr == S_OK && bytesread != This->patterns[i].Length)) /* past end of stream */ continue; if (FAILED(hr)) break; for (pos=0; pos < This->patterns[i].Length; pos++) { if ((data[pos] & This->patterns[i].Mask[pos]) != This->patterns[i].Pattern[pos]) break; } if (pos == This->patterns[i].Length) /* matches pattern */ { hr = S_OK; *pfMatches = TRUE; break; } } if (i == This->pattern_count) /* does not match any pattern */ { hr = S_OK; *pfMatches = FALSE; } HeapFree(GetProcessHeap(), 0, data); return hr; } static HRESULT WINAPI BitmapDecoderInfo_CreateInstance(IWICBitmapDecoderInfo *iface, IWICBitmapDecoder **ppIBitmapDecoder) { BitmapDecoderInfo *This = impl_from_IWICBitmapDecoderInfo(iface); TRACE("(%p,%p)\n", iface, ppIBitmapDecoder); return create_instance(&This->base.clsid, &IID_IWICBitmapDecoder, (void**)ppIBitmapDecoder); } static const IWICBitmapDecoderInfoVtbl BitmapDecoderInfo_Vtbl = { BitmapDecoderInfo_QueryInterface, BitmapDecoderInfo_AddRef, BitmapDecoderInfo_Release, BitmapDecoderInfo_GetComponentType, BitmapDecoderInfo_GetCLSID, BitmapDecoderInfo_GetSigningStatus, BitmapDecoderInfo_GetAuthor, BitmapDecoderInfo_GetVendorGUID, BitmapDecoderInfo_GetVersion, BitmapDecoderInfo_GetSpecVersion, BitmapDecoderInfo_GetFriendlyName, BitmapDecoderInfo_GetContainerFormat, BitmapDecoderInfo_GetPixelFormats, BitmapDecoderInfo_GetColorManagementVersion, BitmapDecoderInfo_GetDeviceManufacturer, BitmapDecoderInfo_GetDeviceModels, BitmapDecoderInfo_GetMimeTypes, BitmapDecoderInfo_GetFileExtensions, BitmapDecoderInfo_DoesSupportAnimation, BitmapDecoderInfo_DoesSupportChromaKey, BitmapDecoderInfo_DoesSupportLossless, BitmapDecoderInfo_DoesSupportMultiframe, BitmapDecoderInfo_MatchesMimeType, BitmapDecoderInfo_GetPatterns, BitmapDecoderInfo_MatchesPattern, BitmapDecoderInfo_CreateInstance }; static void read_bitmap_patterns(BitmapDecoderInfo *info) { UINT pattern_count=0, patterns_size=0; WCHAR subkeyname[11]; LONG res; HKEY patternskey, patternkey; static const WCHAR uintformatW[] = {'%','u',0}; static const WCHAR patternsW[] = {'P','a','t','t','e','r','n','s',0}; static const WCHAR positionW[] = {'P','o','s','i','t','i','o','n',0}; static const WCHAR lengthW[] = {'L','e','n','g','t','h',0}; static const WCHAR patternW[] = {'P','a','t','t','e','r','n',0}; static const WCHAR maskW[] = {'M','a','s','k',0}; static const WCHAR endofstreamW[] = {'E','n','d','O','f','S','t','r','e','a','m',0}; UINT i; WICBitmapPattern *patterns; BYTE *patterns_ptr; DWORD length, valuesize; res = RegOpenKeyExW(info->classkey, patternsW, 0, KEY_READ, &patternskey); if (res != ERROR_SUCCESS) return; res = RegQueryInfoKeyW(patternskey, NULL, NULL, NULL, &pattern_count, NULL, NULL, NULL, NULL, NULL, NULL, NULL); if (res != ERROR_SUCCESS) { RegCloseKey(patternskey); return; } patterns_size = pattern_count * sizeof(WICBitmapPattern); patterns = heap_alloc(patterns_size); if (!patterns) { RegCloseKey(patternskey); return; } for (i=0; res == ERROR_SUCCESS && i < pattern_count; i++) { snprintfW(subkeyname, 11, uintformatW, i); res = RegOpenKeyExW(patternskey, subkeyname, 0, KEY_READ, &patternkey); if (res != ERROR_SUCCESS) break; valuesize = sizeof(ULONG); res = RegGetValueW(patternkey, NULL, lengthW, RRF_RT_DWORD, NULL, &length, &valuesize); if (res == ERROR_SUCCESS) { patterns_size += length*2; patterns[i].Length = length; valuesize = sizeof(BOOL); res = RegGetValueW(patternkey, NULL, endofstreamW, RRF_RT_DWORD, NULL, &patterns[i].EndOfStream, &valuesize); if (res) patterns[i].EndOfStream = 0; patterns[i].Position.QuadPart = 0; valuesize = sizeof(ULARGE_INTEGER); res = RegGetValueW(patternkey, NULL, positionW, RRF_RT_DWORD|RRF_RT_QWORD, NULL, &patterns[i].Position, &valuesize); } RegCloseKey(patternkey); } if (res != ERROR_SUCCESS || !(patterns_ptr = heap_realloc(patterns, patterns_size))) { heap_free(patterns); RegCloseKey(patternskey); return; } patterns = (WICBitmapPattern*)patterns_ptr; patterns_ptr += pattern_count * sizeof(*patterns); for (i=0; res == ERROR_SUCCESS && i < pattern_count; i++) { snprintfW(subkeyname, 11, uintformatW, i); res = RegOpenKeyExW(patternskey, subkeyname, 0, KEY_READ, &patternkey); if (res != ERROR_SUCCESS) break; length = patterns[i].Length; patterns[i].Pattern = patterns_ptr; valuesize = length; res = RegGetValueW(patternkey, NULL, patternW, RRF_RT_REG_BINARY, NULL, patterns[i].Pattern, &valuesize); patterns_ptr += length; if (res == ERROR_SUCCESS) { patterns[i].Mask = patterns_ptr; valuesize = length; res = RegGetValueW(patternkey, NULL, maskW, RRF_RT_REG_BINARY, NULL, patterns[i].Mask, &valuesize); patterns_ptr += length; } RegCloseKey(patternkey); } RegCloseKey(patternskey); if (res != ERROR_SUCCESS) { heap_free(patterns); return; } info->pattern_count = pattern_count; info->patterns_size = patterns_size; info->patterns = patterns; } static HRESULT BitmapDecoderInfo_Constructor(HKEY classkey, REFCLSID clsid, ComponentInfo **ret) { BitmapDecoderInfo *This; This = heap_alloc_zero(sizeof(BitmapDecoderInfo)); if (!This) { RegCloseKey(classkey); return E_OUTOFMEMORY; } This->base.IWICComponentInfo_iface.lpVtbl = (const IWICComponentInfoVtbl*)&BitmapDecoderInfo_Vtbl; This->base.ref = 1; This->classkey = classkey; This->base.clsid = *clsid; read_bitmap_patterns(This); *ret = &This->base; return S_OK; } typedef struct { ComponentInfo base; HKEY classkey; } BitmapEncoderInfo; static inline BitmapEncoderInfo *impl_from_IWICBitmapEncoderInfo(IWICBitmapEncoderInfo *iface) { return CONTAINING_RECORD(iface, BitmapEncoderInfo, base.IWICComponentInfo_iface); } static HRESULT WINAPI BitmapEncoderInfo_QueryInterface(IWICBitmapEncoderInfo *iface, REFIID iid, void **ppv) { BitmapEncoderInfo *This = impl_from_IWICBitmapEncoderInfo(iface); TRACE("(%p,%s,%p)\n", iface, debugstr_guid(iid), ppv); if (!ppv) return E_INVALIDARG; if (IsEqualIID(&IID_IUnknown, iid) || IsEqualIID(&IID_IWICComponentInfo, iid) || IsEqualIID(&IID_IWICBitmapCodecInfo, iid) || IsEqualIID(&IID_IWICBitmapEncoderInfo ,iid)) { *ppv = &This->base.IWICComponentInfo_iface; } else { *ppv = NULL; return E_NOINTERFACE; } IUnknown_AddRef((IUnknown*)*ppv); return S_OK; } static ULONG WINAPI BitmapEncoderInfo_AddRef(IWICBitmapEncoderInfo *iface) { BitmapEncoderInfo *This = impl_from_IWICBitmapEncoderInfo(iface); ULONG ref = InterlockedIncrement(&This->base.ref); TRACE("(%p) refcount=%u\n", iface, ref); return ref; } static ULONG WINAPI BitmapEncoderInfo_Release(IWICBitmapEncoderInfo *iface) { BitmapEncoderInfo *This = impl_from_IWICBitmapEncoderInfo(iface); ULONG ref = InterlockedDecrement(&This->base.ref); TRACE("(%p) refcount=%u\n", iface, ref); if (ref == 0) { RegCloseKey(This->classkey); HeapFree(GetProcessHeap(), 0, This); } return ref; } static HRESULT WINAPI BitmapEncoderInfo_GetComponentType(IWICBitmapEncoderInfo *iface, WICComponentType *pType) { TRACE("(%p,%p)\n", iface, pType); if (!pType) return E_INVALIDARG; *pType = WICEncoder; return S_OK; } static HRESULT WINAPI BitmapEncoderInfo_GetCLSID(IWICBitmapEncoderInfo *iface, CLSID *pclsid) { BitmapEncoderInfo *This = impl_from_IWICBitmapEncoderInfo(iface); TRACE("(%p,%p)\n", iface, pclsid); if (!pclsid) return E_INVALIDARG; *pclsid = This->base.clsid; return S_OK; } static HRESULT WINAPI BitmapEncoderInfo_GetSigningStatus(IWICBitmapEncoderInfo *iface, DWORD *pStatus) { FIXME("(%p,%p): stub\n", iface, pStatus); return E_NOTIMPL; } static HRESULT WINAPI BitmapEncoderInfo_GetAuthor(IWICBitmapEncoderInfo *iface, UINT cchAuthor, WCHAR *wzAuthor, UINT *pcchActual) { BitmapEncoderInfo *This = impl_from_IWICBitmapEncoderInfo(iface); TRACE("(%p,%u,%p,%p)\n", iface, cchAuthor, wzAuthor, pcchActual); return ComponentInfo_GetStringValue(This->classkey, author_valuename, cchAuthor, wzAuthor, pcchActual); } static HRESULT WINAPI BitmapEncoderInfo_GetVendorGUID(IWICBitmapEncoderInfo *iface, GUID *pguidVendor) { BitmapEncoderInfo *This = impl_from_IWICBitmapEncoderInfo(iface); TRACE("(%p,%p)\n", iface, pguidVendor); return ComponentInfo_GetGUIDValue(This->classkey, vendor_valuename, pguidVendor); } static HRESULT WINAPI BitmapEncoderInfo_GetVersion(IWICBitmapEncoderInfo *iface, UINT cchVersion, WCHAR *wzVersion, UINT *pcchActual) { BitmapEncoderInfo *This = impl_from_IWICBitmapEncoderInfo(iface); TRACE("(%p,%u,%p,%p)\n", iface, cchVersion, wzVersion, pcchActual); return ComponentInfo_GetStringValue(This->classkey, version_valuename, cchVersion, wzVersion, pcchActual); } static HRESULT WINAPI BitmapEncoderInfo_GetSpecVersion(IWICBitmapEncoderInfo *iface, UINT cchSpecVersion, WCHAR *wzSpecVersion, UINT *pcchActual) { BitmapEncoderInfo *This = impl_from_IWICBitmapEncoderInfo(iface); TRACE("(%p,%u,%p,%p)\n", iface, cchSpecVersion, wzSpecVersion, pcchActual); return ComponentInfo_GetStringValue(This->classkey, specversion_valuename, cchSpecVersion, wzSpecVersion, pcchActual); } static HRESULT WINAPI BitmapEncoderInfo_GetFriendlyName(IWICBitmapEncoderInfo *iface, UINT cchFriendlyName, WCHAR *wzFriendlyName, UINT *pcchActual) { BitmapEncoderInfo *This = impl_from_IWICBitmapEncoderInfo(iface); TRACE("(%p,%u,%p,%p)\n", iface, cchFriendlyName, wzFriendlyName, pcchActual); return ComponentInfo_GetStringValue(This->classkey, friendlyname_valuename, cchFriendlyName, wzFriendlyName, pcchActual); } static HRESULT WINAPI BitmapEncoderInfo_GetContainerFormat(IWICBitmapEncoderInfo *iface, GUID *pguidContainerFormat) { BitmapEncoderInfo *This = impl_from_IWICBitmapEncoderInfo(iface); TRACE("(%p,%p)\n", iface, pguidContainerFormat); return ComponentInfo_GetGUIDValue(This->classkey, containerformat_valuename, pguidContainerFormat); } static HRESULT WINAPI BitmapEncoderInfo_GetPixelFormats(IWICBitmapEncoderInfo *iface, UINT cFormats, GUID *pguidPixelFormats, UINT *pcActual) { BitmapEncoderInfo *This = impl_from_IWICBitmapEncoderInfo(iface); TRACE("(%p,%u,%p,%p)\n", iface, cFormats, pguidPixelFormats, pcActual); return ComponentInfo_GetGuidList(This->classkey, formats_keyname, cFormats, pguidPixelFormats, pcActual); } static HRESULT WINAPI BitmapEncoderInfo_GetColorManagementVersion(IWICBitmapEncoderInfo *iface, UINT cchColorManagementVersion, WCHAR *wzColorManagementVersion, UINT *pcchActual) { FIXME("(%p,%u,%p,%p): stub\n", iface, cchColorManagementVersion, wzColorManagementVersion, pcchActual); return E_NOTIMPL; } static HRESULT WINAPI BitmapEncoderInfo_GetDeviceManufacturer(IWICBitmapEncoderInfo *iface, UINT cchDeviceManufacturer, WCHAR *wzDeviceManufacturer, UINT *pcchActual) { FIXME("(%p,%u,%p,%p): stub\n", iface, cchDeviceManufacturer, wzDeviceManufacturer, pcchActual); return E_NOTIMPL; } static HRESULT WINAPI BitmapEncoderInfo_GetDeviceModels(IWICBitmapEncoderInfo *iface, UINT cchDeviceModels, WCHAR *wzDeviceModels, UINT *pcchActual) { FIXME("(%p,%u,%p,%p): stub\n", iface, cchDeviceModels, wzDeviceModels, pcchActual); return E_NOTIMPL; } static HRESULT WINAPI BitmapEncoderInfo_GetMimeTypes(IWICBitmapEncoderInfo *iface, UINT cchMimeTypes, WCHAR *wzMimeTypes, UINT *pcchActual) { BitmapEncoderInfo *This = impl_from_IWICBitmapEncoderInfo(iface); TRACE("(%p,%u,%p,%p)\n", iface, cchMimeTypes, wzMimeTypes, pcchActual); return ComponentInfo_GetStringValue(This->classkey, mimetypes_valuename, cchMimeTypes, wzMimeTypes, pcchActual); } static HRESULT WINAPI BitmapEncoderInfo_GetFileExtensions(IWICBitmapEncoderInfo *iface, UINT cchFileExtensions, WCHAR *wzFileExtensions, UINT *pcchActual) { BitmapEncoderInfo *This = impl_from_IWICBitmapEncoderInfo(iface); TRACE("(%p,%u,%p,%p)\n", iface, cchFileExtensions, wzFileExtensions, pcchActual); return ComponentInfo_GetStringValue(This->classkey, fileextensions_valuename, cchFileExtensions, wzFileExtensions, pcchActual); } static HRESULT WINAPI BitmapEncoderInfo_DoesSupportAnimation(IWICBitmapEncoderInfo *iface, BOOL *pfSupportAnimation) { FIXME("(%p,%p): stub\n", iface, pfSupportAnimation); return E_NOTIMPL; } static HRESULT WINAPI BitmapEncoderInfo_DoesSupportChromaKey(IWICBitmapEncoderInfo *iface, BOOL *pfSupportChromaKey) { FIXME("(%p,%p): stub\n", iface, pfSupportChromaKey); return E_NOTIMPL; } static HRESULT WINAPI BitmapEncoderInfo_DoesSupportLossless(IWICBitmapEncoderInfo *iface, BOOL *pfSupportLossless) { FIXME("(%p,%p): stub\n", iface, pfSupportLossless); return E_NOTIMPL; } static HRESULT WINAPI BitmapEncoderInfo_DoesSupportMultiframe(IWICBitmapEncoderInfo *iface, BOOL *pfSupportMultiframe) { FIXME("(%p,%p): stub\n", iface, pfSupportMultiframe); return E_NOTIMPL; } static HRESULT WINAPI BitmapEncoderInfo_MatchesMimeType(IWICBitmapEncoderInfo *iface, LPCWSTR wzMimeType, BOOL *pfMatches) { FIXME("(%p,%s,%p): stub\n", iface, debugstr_w(wzMimeType), pfMatches); return E_NOTIMPL; } static HRESULT WINAPI BitmapEncoderInfo_CreateInstance(IWICBitmapEncoderInfo *iface, IWICBitmapEncoder **ppIBitmapEncoder) { BitmapEncoderInfo *This = impl_from_IWICBitmapEncoderInfo(iface); TRACE("(%p,%p)\n", iface, ppIBitmapEncoder); return create_instance(&This->base.clsid, &IID_IWICBitmapEncoder, (void**)ppIBitmapEncoder); } static const IWICBitmapEncoderInfoVtbl BitmapEncoderInfo_Vtbl = { BitmapEncoderInfo_QueryInterface, BitmapEncoderInfo_AddRef, BitmapEncoderInfo_Release, BitmapEncoderInfo_GetComponentType, BitmapEncoderInfo_GetCLSID, BitmapEncoderInfo_GetSigningStatus, BitmapEncoderInfo_GetAuthor, BitmapEncoderInfo_GetVendorGUID, BitmapEncoderInfo_GetVersion, BitmapEncoderInfo_GetSpecVersion, BitmapEncoderInfo_GetFriendlyName, BitmapEncoderInfo_GetContainerFormat, BitmapEncoderInfo_GetPixelFormats, BitmapEncoderInfo_GetColorManagementVersion, BitmapEncoderInfo_GetDeviceManufacturer, BitmapEncoderInfo_GetDeviceModels, BitmapEncoderInfo_GetMimeTypes, BitmapEncoderInfo_GetFileExtensions, BitmapEncoderInfo_DoesSupportAnimation, BitmapEncoderInfo_DoesSupportChromaKey, BitmapEncoderInfo_DoesSupportLossless, BitmapEncoderInfo_DoesSupportMultiframe, BitmapEncoderInfo_MatchesMimeType, BitmapEncoderInfo_CreateInstance }; static HRESULT BitmapEncoderInfo_Constructor(HKEY classkey, REFCLSID clsid, ComponentInfo **ret) { BitmapEncoderInfo *This; This = HeapAlloc(GetProcessHeap(), 0, sizeof(BitmapEncoderInfo)); if (!This) { RegCloseKey(classkey); return E_OUTOFMEMORY; } This->base.IWICComponentInfo_iface.lpVtbl = (const IWICComponentInfoVtbl*)&BitmapEncoderInfo_Vtbl; This->base.ref = 1; This->classkey = classkey; This->base.clsid = *clsid; *ret = &This->base; return S_OK; } typedef struct { ComponentInfo base; HKEY classkey; } FormatConverterInfo; static inline FormatConverterInfo *impl_from_IWICFormatConverterInfo(IWICFormatConverterInfo *iface) { return CONTAINING_RECORD(iface, FormatConverterInfo, base.IWICComponentInfo_iface); } static HRESULT WINAPI FormatConverterInfo_QueryInterface(IWICFormatConverterInfo *iface, REFIID iid, void **ppv) { FormatConverterInfo *This = impl_from_IWICFormatConverterInfo(iface); TRACE("(%p,%s,%p)\n", iface, debugstr_guid(iid), ppv); if (!ppv) return E_INVALIDARG; if (IsEqualIID(&IID_IUnknown, iid) || IsEqualIID(&IID_IWICComponentInfo, iid) || IsEqualIID(&IID_IWICFormatConverterInfo ,iid)) { *ppv = &This->base.IWICComponentInfo_iface; } else { *ppv = NULL; return E_NOINTERFACE; } IUnknown_AddRef((IUnknown*)*ppv); return S_OK; } static ULONG WINAPI FormatConverterInfo_AddRef(IWICFormatConverterInfo *iface) { FormatConverterInfo *This = impl_from_IWICFormatConverterInfo(iface); ULONG ref = InterlockedIncrement(&This->base.ref); TRACE("(%p) refcount=%u\n", iface, ref); return ref; } static ULONG WINAPI FormatConverterInfo_Release(IWICFormatConverterInfo *iface) { FormatConverterInfo *This = impl_from_IWICFormatConverterInfo(iface); ULONG ref = InterlockedDecrement(&This->base.ref); TRACE("(%p) refcount=%u\n", iface, ref); if (ref == 0) { RegCloseKey(This->classkey); HeapFree(GetProcessHeap(), 0, This); } return ref; } static HRESULT WINAPI FormatConverterInfo_GetComponentType(IWICFormatConverterInfo *iface, WICComponentType *pType) { TRACE("(%p,%p)\n", iface, pType); if (!pType) return E_INVALIDARG; *pType = WICPixelFormatConverter; return S_OK; } static HRESULT WINAPI FormatConverterInfo_GetCLSID(IWICFormatConverterInfo *iface, CLSID *pclsid) { FormatConverterInfo *This = impl_from_IWICFormatConverterInfo(iface); TRACE("(%p,%p)\n", iface, pclsid); if (!pclsid) return E_INVALIDARG; *pclsid = This->base.clsid; return S_OK; } static HRESULT WINAPI FormatConverterInfo_GetSigningStatus(IWICFormatConverterInfo *iface, DWORD *pStatus) { FIXME("(%p,%p): stub\n", iface, pStatus); return E_NOTIMPL; } static HRESULT WINAPI FormatConverterInfo_GetAuthor(IWICFormatConverterInfo *iface, UINT cchAuthor, WCHAR *wzAuthor, UINT *pcchActual) { FormatConverterInfo *This = impl_from_IWICFormatConverterInfo(iface); TRACE("(%p,%u,%p,%p)\n", iface, cchAuthor, wzAuthor, pcchActual); return ComponentInfo_GetStringValue(This->classkey, author_valuename, cchAuthor, wzAuthor, pcchActual); } static HRESULT WINAPI FormatConverterInfo_GetVendorGUID(IWICFormatConverterInfo *iface, GUID *pguidVendor) { FormatConverterInfo *This = impl_from_IWICFormatConverterInfo(iface); TRACE("(%p,%p)\n", iface, pguidVendor); return ComponentInfo_GetGUIDValue(This->classkey, vendor_valuename, pguidVendor); } static HRESULT WINAPI FormatConverterInfo_GetVersion(IWICFormatConverterInfo *iface, UINT cchVersion, WCHAR *wzVersion, UINT *pcchActual) { FormatConverterInfo *This = impl_from_IWICFormatConverterInfo(iface); TRACE("(%p,%u,%p,%p)\n", iface, cchVersion, wzVersion, pcchActual); return ComponentInfo_GetStringValue(This->classkey, version_valuename, cchVersion, wzVersion, pcchActual); } static HRESULT WINAPI FormatConverterInfo_GetSpecVersion(IWICFormatConverterInfo *iface, UINT cchSpecVersion, WCHAR *wzSpecVersion, UINT *pcchActual) { FormatConverterInfo *This = impl_from_IWICFormatConverterInfo(iface); TRACE("(%p,%u,%p,%p)\n", iface, cchSpecVersion, wzSpecVersion, pcchActual); return ComponentInfo_GetStringValue(This->classkey, specversion_valuename, cchSpecVersion, wzSpecVersion, pcchActual); } static HRESULT WINAPI FormatConverterInfo_GetFriendlyName(IWICFormatConverterInfo *iface, UINT cchFriendlyName, WCHAR *wzFriendlyName, UINT *pcchActual) { FormatConverterInfo *This = impl_from_IWICFormatConverterInfo(iface); TRACE("(%p,%u,%p,%p)\n", iface, cchFriendlyName, wzFriendlyName, pcchActual); return ComponentInfo_GetStringValue(This->classkey, friendlyname_valuename, cchFriendlyName, wzFriendlyName, pcchActual); } static HRESULT WINAPI FormatConverterInfo_GetPixelFormats(IWICFormatConverterInfo *iface, UINT cFormats, GUID *pguidPixelFormats, UINT *pcActual) { FIXME("(%p,%u,%p,%p): stub\n", iface, cFormats, pguidPixelFormats, pcActual); return E_NOTIMPL; } static HRESULT WINAPI FormatConverterInfo_CreateInstance(IWICFormatConverterInfo *iface, IWICFormatConverter **ppIFormatConverter) { FormatConverterInfo *This = impl_from_IWICFormatConverterInfo(iface); TRACE("(%p,%p)\n", iface, ppIFormatConverter); return create_instance(&This->base.clsid, &IID_IWICFormatConverter, (void**)ppIFormatConverter); } static BOOL ConverterSupportsFormat(IWICFormatConverterInfo *iface, const WCHAR *formatguid) { LONG res; FormatConverterInfo *This = impl_from_IWICFormatConverterInfo(iface); HKEY formats_key, guid_key; /* Avoid testing using IWICFormatConverter_GetPixelFormats because that would be O(n). A registry test should do better. */ res = RegOpenKeyExW(This->classkey, pixelformats_keyname, 0, KEY_READ, &formats_key); if (res != ERROR_SUCCESS) return FALSE; res = RegOpenKeyExW(formats_key, formatguid, 0, KEY_READ, &guid_key); if (res == ERROR_SUCCESS) RegCloseKey(guid_key); RegCloseKey(formats_key); return (res == ERROR_SUCCESS); } static const IWICFormatConverterInfoVtbl FormatConverterInfo_Vtbl = { FormatConverterInfo_QueryInterface, FormatConverterInfo_AddRef, FormatConverterInfo_Release, FormatConverterInfo_GetComponentType, FormatConverterInfo_GetCLSID, FormatConverterInfo_GetSigningStatus, FormatConverterInfo_GetAuthor, FormatConverterInfo_GetVendorGUID, FormatConverterInfo_GetVersion, FormatConverterInfo_GetSpecVersion, FormatConverterInfo_GetFriendlyName, FormatConverterInfo_GetPixelFormats, FormatConverterInfo_CreateInstance }; static HRESULT FormatConverterInfo_Constructor(HKEY classkey, REFCLSID clsid, ComponentInfo **ret) { FormatConverterInfo *This; This = HeapAlloc(GetProcessHeap(), 0, sizeof(FormatConverterInfo)); if (!This) { RegCloseKey(classkey); return E_OUTOFMEMORY; } This->base.IWICComponentInfo_iface.lpVtbl = (const IWICComponentInfoVtbl*)&FormatConverterInfo_Vtbl; This->base.ref = 1; This->classkey = classkey; This->base.clsid = *clsid; *ret = &This->base; return S_OK; } typedef struct { ComponentInfo base; HKEY classkey; } PixelFormatInfo; static inline PixelFormatInfo *impl_from_IWICPixelFormatInfo2(IWICPixelFormatInfo2 *iface) { return CONTAINING_RECORD(iface, PixelFormatInfo, base.IWICComponentInfo_iface); } static HRESULT WINAPI PixelFormatInfo_QueryInterface(IWICPixelFormatInfo2 *iface, REFIID iid, void **ppv) { PixelFormatInfo *This = impl_from_IWICPixelFormatInfo2(iface); TRACE("(%p,%s,%p)\n", iface, debugstr_guid(iid), ppv); if (!ppv) return E_INVALIDARG; if (IsEqualIID(&IID_IUnknown, iid) || IsEqualIID(&IID_IWICComponentInfo, iid) || IsEqualIID(&IID_IWICPixelFormatInfo, iid) || IsEqualIID(&IID_IWICPixelFormatInfo2 ,iid)) { *ppv = &This->base.IWICComponentInfo_iface; } else { *ppv = NULL; return E_NOINTERFACE; } IUnknown_AddRef((IUnknown*)*ppv); return S_OK; } static ULONG WINAPI PixelFormatInfo_AddRef(IWICPixelFormatInfo2 *iface) { PixelFormatInfo *This = impl_from_IWICPixelFormatInfo2(iface); ULONG ref = InterlockedIncrement(&This->base.ref); TRACE("(%p) refcount=%u\n", iface, ref); return ref; } static ULONG WINAPI PixelFormatInfo_Release(IWICPixelFormatInfo2 *iface) { PixelFormatInfo *This = impl_from_IWICPixelFormatInfo2(iface); ULONG ref = InterlockedDecrement(&This->base.ref); TRACE("(%p) refcount=%u\n", iface, ref); if (ref == 0) { RegCloseKey(This->classkey); HeapFree(GetProcessHeap(), 0, This); } return ref; } static HRESULT WINAPI PixelFormatInfo_GetComponentType(IWICPixelFormatInfo2 *iface, WICComponentType *pType) { TRACE("(%p,%p)\n", iface, pType); if (!pType) return E_INVALIDARG; *pType = WICPixelFormat; return S_OK; } static HRESULT WINAPI PixelFormatInfo_GetCLSID(IWICPixelFormatInfo2 *iface, CLSID *pclsid) { PixelFormatInfo *This = impl_from_IWICPixelFormatInfo2(iface); TRACE("(%p,%p)\n", iface, pclsid); if (!pclsid) return E_INVALIDARG; *pclsid = This->base.clsid; return S_OK; } static HRESULT WINAPI PixelFormatInfo_GetSigningStatus(IWICPixelFormatInfo2 *iface, DWORD *pStatus) { TRACE("(%p,%p)\n", iface, pStatus); if (!pStatus) return E_INVALIDARG; /* Pixel formats don't require code, so they are considered signed. */ *pStatus = WICComponentSigned; return S_OK; } static HRESULT WINAPI PixelFormatInfo_GetAuthor(IWICPixelFormatInfo2 *iface, UINT cchAuthor, WCHAR *wzAuthor, UINT *pcchActual) { PixelFormatInfo *This = impl_from_IWICPixelFormatInfo2(iface); TRACE("(%p,%u,%p,%p)\n", iface, cchAuthor, wzAuthor, pcchActual); return ComponentInfo_GetStringValue(This->classkey, author_valuename, cchAuthor, wzAuthor, pcchActual); } static HRESULT WINAPI PixelFormatInfo_GetVendorGUID(IWICPixelFormatInfo2 *iface, GUID *pguidVendor) { PixelFormatInfo *This = impl_from_IWICPixelFormatInfo2(iface); TRACE("(%p,%p)\n", iface, pguidVendor); return ComponentInfo_GetGUIDValue(This->classkey, vendor_valuename, pguidVendor); } static HRESULT WINAPI PixelFormatInfo_GetVersion(IWICPixelFormatInfo2 *iface, UINT cchVersion, WCHAR *wzVersion, UINT *pcchActual) { PixelFormatInfo *This = impl_from_IWICPixelFormatInfo2(iface); TRACE("(%p,%u,%p,%p)\n", iface, cchVersion, wzVersion, pcchActual); return ComponentInfo_GetStringValue(This->classkey, version_valuename, cchVersion, wzVersion, pcchActual); } static HRESULT WINAPI PixelFormatInfo_GetSpecVersion(IWICPixelFormatInfo2 *iface, UINT cchSpecVersion, WCHAR *wzSpecVersion, UINT *pcchActual) { PixelFormatInfo *This = impl_from_IWICPixelFormatInfo2(iface); TRACE("(%p,%u,%p,%p)\n", iface, cchSpecVersion, wzSpecVersion, pcchActual); return ComponentInfo_GetStringValue(This->classkey, specversion_valuename, cchSpecVersion, wzSpecVersion, pcchActual); } static HRESULT WINAPI PixelFormatInfo_GetFriendlyName(IWICPixelFormatInfo2 *iface, UINT cchFriendlyName, WCHAR *wzFriendlyName, UINT *pcchActual) { PixelFormatInfo *This = impl_from_IWICPixelFormatInfo2(iface); TRACE("(%p,%u,%p,%p)\n", iface, cchFriendlyName, wzFriendlyName, pcchActual); return ComponentInfo_GetStringValue(This->classkey, friendlyname_valuename, cchFriendlyName, wzFriendlyName, pcchActual); } static HRESULT WINAPI PixelFormatInfo_GetFormatGUID(IWICPixelFormatInfo2 *iface, GUID *pFormat) { PixelFormatInfo *This = impl_from_IWICPixelFormatInfo2(iface); TRACE("(%p,%p)\n", iface, pFormat); if (!pFormat) return E_INVALIDARG; *pFormat = This->base.clsid; return S_OK; } static HRESULT WINAPI PixelFormatInfo_GetColorContext(IWICPixelFormatInfo2 *iface, IWICColorContext **ppIColorContext) { FIXME("(%p,%p): stub\n", iface, ppIColorContext); return E_NOTIMPL; } static HRESULT WINAPI PixelFormatInfo_GetBitsPerPixel(IWICPixelFormatInfo2 *iface, UINT *puiBitsPerPixel) { PixelFormatInfo *This = impl_from_IWICPixelFormatInfo2(iface); TRACE("(%p,%p)\n", iface, puiBitsPerPixel); return ComponentInfo_GetDWORDValue(This->classkey, bitsperpixel_valuename, puiBitsPerPixel); } static HRESULT WINAPI PixelFormatInfo_GetChannelCount(IWICPixelFormatInfo2 *iface, UINT *puiChannelCount) { PixelFormatInfo *This = impl_from_IWICPixelFormatInfo2(iface); TRACE("(%p,%p)\n", iface, puiChannelCount); return ComponentInfo_GetDWORDValue(This->classkey, channelcount_valuename, puiChannelCount); } static HRESULT WINAPI PixelFormatInfo_GetChannelMask(IWICPixelFormatInfo2 *iface, UINT uiChannelIndex, UINT cbMaskBuffer, BYTE *pbMaskBuffer, UINT *pcbActual) { static const WCHAR uintformatW[] = {'%','u',0}; PixelFormatInfo *This = impl_from_IWICPixelFormatInfo2(iface); UINT channel_count; HRESULT hr; LONG ret; WCHAR valuename[11]; DWORD cbData; TRACE("(%p,%u,%u,%p,%p)\n", iface, uiChannelIndex, cbMaskBuffer, pbMaskBuffer, pcbActual); if (!pcbActual) return E_INVALIDARG; hr = PixelFormatInfo_GetChannelCount(iface, &channel_count); if (SUCCEEDED(hr) && uiChannelIndex >= channel_count) hr = E_INVALIDARG; if (SUCCEEDED(hr)) { snprintfW(valuename, 11, uintformatW, uiChannelIndex); cbData = cbMaskBuffer; ret = RegGetValueW(This->classkey, channelmasks_keyname, valuename, RRF_RT_REG_BINARY, NULL, pbMaskBuffer, &cbData); if (ret == ERROR_SUCCESS || ret == ERROR_MORE_DATA) *pcbActual = cbData; if (ret == ERROR_MORE_DATA) hr = E_INVALIDARG; else hr = HRESULT_FROM_WIN32(ret); } return hr; } static HRESULT WINAPI PixelFormatInfo_SupportsTransparency(IWICPixelFormatInfo2 *iface, BOOL *pfSupportsTransparency) { PixelFormatInfo *This = impl_from_IWICPixelFormatInfo2(iface); TRACE("(%p,%p)\n", iface, pfSupportsTransparency); return ComponentInfo_GetDWORDValue(This->classkey, supportstransparency_valuename, (DWORD*)pfSupportsTransparency); } static HRESULT WINAPI PixelFormatInfo_GetNumericRepresentation(IWICPixelFormatInfo2 *iface, WICPixelFormatNumericRepresentation *pNumericRepresentation) { PixelFormatInfo *This = impl_from_IWICPixelFormatInfo2(iface); TRACE("(%p,%p)\n", iface, pNumericRepresentation); return ComponentInfo_GetDWORDValue(This->classkey, numericrepresentation_valuename, pNumericRepresentation); } static const IWICPixelFormatInfo2Vtbl PixelFormatInfo_Vtbl = { PixelFormatInfo_QueryInterface, PixelFormatInfo_AddRef, PixelFormatInfo_Release, PixelFormatInfo_GetComponentType, PixelFormatInfo_GetCLSID, PixelFormatInfo_GetSigningStatus, PixelFormatInfo_GetAuthor, PixelFormatInfo_GetVendorGUID, PixelFormatInfo_GetVersion, PixelFormatInfo_GetSpecVersion, PixelFormatInfo_GetFriendlyName, PixelFormatInfo_GetFormatGUID, PixelFormatInfo_GetColorContext, PixelFormatInfo_GetBitsPerPixel, PixelFormatInfo_GetChannelCount, PixelFormatInfo_GetChannelMask, PixelFormatInfo_SupportsTransparency, PixelFormatInfo_GetNumericRepresentation }; static HRESULT PixelFormatInfo_Constructor(HKEY classkey, REFCLSID clsid, ComponentInfo **ret) { PixelFormatInfo *This; This = HeapAlloc(GetProcessHeap(), 0, sizeof(PixelFormatInfo)); if (!This) { RegCloseKey(classkey); return E_OUTOFMEMORY; } This->base.IWICComponentInfo_iface.lpVtbl = (const IWICComponentInfoVtbl*)&PixelFormatInfo_Vtbl; This->base.ref = 1; This->classkey = classkey; This->base.clsid = *clsid; *ret = &This->base; return S_OK; } struct metadata_container { WICMetadataPattern *patterns; UINT pattern_count; UINT patterns_size; }; typedef struct { ComponentInfo base; HKEY classkey; GUID *container_formats; struct metadata_container *containers; UINT container_count; } MetadataReaderInfo; static struct metadata_container *get_metadata_container(MetadataReaderInfo *info, const GUID *guid) { unsigned i; for (i = 0; i < info->container_count; i++) if (IsEqualGUID(info->container_formats + i, guid)) return info->containers + i; return NULL; } static inline MetadataReaderInfo *impl_from_IWICMetadataReaderInfo(IWICMetadataReaderInfo *iface) { return CONTAINING_RECORD(iface, MetadataReaderInfo, base.IWICComponentInfo_iface); } static HRESULT WINAPI MetadataReaderInfo_QueryInterface(IWICMetadataReaderInfo *iface, REFIID riid, void **ppv) { MetadataReaderInfo *This = impl_from_IWICMetadataReaderInfo(iface); TRACE("(%p,%s,%p)\n", iface, debugstr_guid(riid), ppv); if (!ppv) return E_INVALIDARG; if (IsEqualIID(&IID_IUnknown, riid) || IsEqualIID(&IID_IWICComponentInfo, riid) || IsEqualIID(&IID_IWICMetadataHandlerInfo, riid) || IsEqualIID(&IID_IWICMetadataReaderInfo, riid)) { *ppv = &This->base.IWICComponentInfo_iface; } else { *ppv = NULL; return E_NOINTERFACE; } IUnknown_AddRef((IUnknown *)*ppv); return S_OK; } static ULONG WINAPI MetadataReaderInfo_AddRef(IWICMetadataReaderInfo *iface) { MetadataReaderInfo *This = impl_from_IWICMetadataReaderInfo(iface); ULONG ref = InterlockedIncrement(&This->base.ref); TRACE("(%p) refcount=%u\n", iface, ref); return ref; } static ULONG WINAPI MetadataReaderInfo_Release(IWICMetadataReaderInfo *iface) { MetadataReaderInfo *This = impl_from_IWICMetadataReaderInfo(iface); ULONG ref = InterlockedDecrement(&This->base.ref); TRACE("(%p) refcount=%u\n", iface, ref); if (!ref) { unsigned i; RegCloseKey(This->classkey); for (i = 0; i < This->container_count; i++) heap_free(This->containers[i].patterns); heap_free(This->containers); heap_free(This->container_formats); HeapFree(GetProcessHeap(), 0, This); } return ref; } static HRESULT WINAPI MetadataReaderInfo_GetComponentType(IWICMetadataReaderInfo *iface, WICComponentType *type) { TRACE("(%p,%p)\n", iface, type); if (!type) return E_INVALIDARG; *type = WICMetadataReader; return S_OK; } static HRESULT WINAPI MetadataReaderInfo_GetCLSID(IWICMetadataReaderInfo *iface, CLSID *clsid) { MetadataReaderInfo *This = impl_from_IWICMetadataReaderInfo(iface); TRACE("(%p,%p)\n", iface, clsid); if (!clsid) return E_INVALIDARG; *clsid = This->base.clsid; return S_OK; } static HRESULT WINAPI MetadataReaderInfo_GetSigningStatus(IWICMetadataReaderInfo *iface, DWORD *status) { FIXME("(%p,%p): stub\n", iface, status); return E_NOTIMPL; } static HRESULT WINAPI MetadataReaderInfo_GetAuthor(IWICMetadataReaderInfo *iface, UINT length, WCHAR *author, UINT *actual_length) { MetadataReaderInfo *This = impl_from_IWICMetadataReaderInfo(iface); TRACE("(%p,%u,%p,%p)\n", iface, length, author, actual_length); return ComponentInfo_GetStringValue(This->classkey, author_valuename, length, author, actual_length); } static HRESULT WINAPI MetadataReaderInfo_GetVendorGUID(IWICMetadataReaderInfo *iface, GUID *vendor) { MetadataReaderInfo *This = impl_from_IWICMetadataReaderInfo(iface); TRACE("(%p,%p)\n", iface, vendor); return ComponentInfo_GetGUIDValue(This->classkey, vendor_valuename, vendor); } static HRESULT WINAPI MetadataReaderInfo_GetVersion(IWICMetadataReaderInfo *iface, UINT length, WCHAR *version, UINT *actual_length) { MetadataReaderInfo *This = impl_from_IWICMetadataReaderInfo(iface); TRACE("(%p,%u,%p,%p)\n", iface, length, version, actual_length); return ComponentInfo_GetStringValue(This->classkey, version_valuename, length, version, actual_length); } static HRESULT WINAPI MetadataReaderInfo_GetSpecVersion(IWICMetadataReaderInfo *iface, UINT length, WCHAR *version, UINT *actual_length) { MetadataReaderInfo *This = impl_from_IWICMetadataReaderInfo(iface); TRACE("(%p,%u,%p,%p)\n", iface, length, version, actual_length); return ComponentInfo_GetStringValue(This->classkey, specversion_valuename, length, version, actual_length); } static HRESULT WINAPI MetadataReaderInfo_GetFriendlyName(IWICMetadataReaderInfo *iface, UINT length, WCHAR *name, UINT *actual_length) { MetadataReaderInfo *This = impl_from_IWICMetadataReaderInfo(iface); TRACE("(%p,%u,%p,%p)\n", iface, length, name, actual_length); return ComponentInfo_GetStringValue(This->classkey, friendlyname_valuename, length, name, actual_length); } static HRESULT WINAPI MetadataReaderInfo_GetMetadataFormat(IWICMetadataReaderInfo *iface, GUID *format) { MetadataReaderInfo *This = impl_from_IWICMetadataReaderInfo(iface); TRACE("(%p,%p)\n", iface, format); return ComponentInfo_GetGUIDValue(This->classkey, metadataformat_valuename, format); } static HRESULT WINAPI MetadataReaderInfo_GetContainerFormats(IWICMetadataReaderInfo *iface, UINT length, GUID *formats, UINT *actual_length) { MetadataReaderInfo *This = impl_from_IWICMetadataReaderInfo(iface); TRACE("(%p,%u,%p,%p)\n", iface, length, formats, actual_length); if (!actual_length) return E_INVALIDARG; *actual_length = This->container_count; if (formats) { if (This->container_count && length < This->container_count) return WINCODEC_ERR_INSUFFICIENTBUFFER; memcpy(formats, This->container_formats, This->container_count); } return S_OK; } static HRESULT WINAPI MetadataReaderInfo_GetDeviceManufacturer(IWICMetadataReaderInfo *iface, UINT length, WCHAR *manufacturer, UINT *actual_length) { FIXME("(%p,%u,%p,%p): stub\n", iface, length, manufacturer, actual_length); return E_NOTIMPL; } static HRESULT WINAPI MetadataReaderInfo_GetDeviceModels(IWICMetadataReaderInfo *iface, UINT length, WCHAR *models, UINT *actual_length) { FIXME("(%p,%u,%p,%p): stub\n", iface, length, models, actual_length); return E_NOTIMPL; } static HRESULT WINAPI MetadataReaderInfo_DoesRequireFullStream(IWICMetadataReaderInfo *iface, BOOL *param) { MetadataReaderInfo *This = impl_from_IWICMetadataReaderInfo(iface); TRACE("(%p,%p)\n", iface, param); return ComponentInfo_GetDWORDValue(This->classkey, requiresfullstream_valuename, (DWORD *)param); } static HRESULT WINAPI MetadataReaderInfo_DoesSupportPadding(IWICMetadataReaderInfo *iface, BOOL *param) { MetadataReaderInfo *This = impl_from_IWICMetadataReaderInfo(iface); TRACE("(%p,%p)\n", iface, param); return ComponentInfo_GetDWORDValue(This->classkey, supportspadding_valuename, (DWORD *)param); } static HRESULT WINAPI MetadataReaderInfo_DoesRequireFixedSize(IWICMetadataReaderInfo *iface, BOOL *param) { FIXME("(%p,%p): stub\n", iface, param); return E_NOTIMPL; } static HRESULT WINAPI MetadataReaderInfo_GetPatterns(IWICMetadataReaderInfo *iface, REFGUID container_guid, UINT length, WICMetadataPattern *patterns, UINT *count, UINT *actual_length) { MetadataReaderInfo *This = impl_from_IWICMetadataReaderInfo(iface); struct metadata_container *container; TRACE("(%p,%s,%u,%p,%p,%p)\n", iface, debugstr_guid(container_guid), length, patterns, count, actual_length); if (!actual_length || !container_guid) return E_INVALIDARG; if (!(container = get_metadata_container(This, container_guid))) return WINCODEC_ERR_COMPONENTNOTFOUND; *count = container->pattern_count; *actual_length = container->patterns_size; if (patterns) { if (container->patterns_size && length < container->patterns_size) return WINCODEC_ERR_INSUFFICIENTBUFFER; memcpy(patterns, container->patterns, container->patterns_size); } return S_OK; } static HRESULT WINAPI MetadataReaderInfo_MatchesPattern(IWICMetadataReaderInfo *iface, REFGUID container_guid, IStream *stream, BOOL *matches) { MetadataReaderInfo *This = impl_from_IWICMetadataReaderInfo(iface); struct metadata_container *container; HRESULT hr; ULONG datasize=0; BYTE *data=NULL; ULONG bytesread; UINT i; LARGE_INTEGER seekpos; ULONG pos; TRACE("(%p,%s,%p,%p)\n", iface, debugstr_guid(container_guid), stream, matches); if (!(container = get_metadata_container(This, container_guid))) return WINCODEC_ERR_COMPONENTNOTFOUND; for (i=0; i < container->pattern_count; i++) { if (datasize < container->patterns[i].Length) { HeapFree(GetProcessHeap(), 0, data); datasize = container->patterns[i].Length; data = HeapAlloc(GetProcessHeap(), 0, container->patterns[i].Length); if (!data) { hr = E_OUTOFMEMORY; break; } } seekpos.QuadPart = container->patterns[i].Position.QuadPart; hr = IStream_Seek(stream, seekpos, STREAM_SEEK_SET, NULL); if (FAILED(hr)) break; hr = IStream_Read(stream, data, container->patterns[i].Length, &bytesread); if (hr == S_FALSE || (hr == S_OK && bytesread != container->patterns[i].Length)) /* past end of stream */ continue; if (FAILED(hr)) break; for (pos=0; pos < container->patterns[i].Length; pos++) { if ((data[pos] & container->patterns[i].Mask[pos]) != container->patterns[i].Pattern[pos]) break; } if (pos == container->patterns[i].Length) /* matches pattern */ { hr = S_OK; *matches = TRUE; break; } } if (i == container->pattern_count) /* does not match any pattern */ { hr = S_OK; *matches = FALSE; } HeapFree(GetProcessHeap(), 0, data); return hr; } static HRESULT WINAPI MetadataReaderInfo_CreateInstance(IWICMetadataReaderInfo *iface, IWICMetadataReader **reader) { MetadataReaderInfo *This = impl_from_IWICMetadataReaderInfo(iface); TRACE("(%p,%p)\n", iface, reader); return create_instance(&This->base.clsid, &IID_IWICMetadataReader, (void **)reader); } static const IWICMetadataReaderInfoVtbl MetadataReaderInfo_Vtbl = { MetadataReaderInfo_QueryInterface, MetadataReaderInfo_AddRef, MetadataReaderInfo_Release, MetadataReaderInfo_GetComponentType, MetadataReaderInfo_GetCLSID, MetadataReaderInfo_GetSigningStatus, MetadataReaderInfo_GetAuthor, MetadataReaderInfo_GetVendorGUID, MetadataReaderInfo_GetVersion, MetadataReaderInfo_GetSpecVersion, MetadataReaderInfo_GetFriendlyName, MetadataReaderInfo_GetMetadataFormat, MetadataReaderInfo_GetContainerFormats, MetadataReaderInfo_GetDeviceManufacturer, MetadataReaderInfo_GetDeviceModels, MetadataReaderInfo_DoesRequireFullStream, MetadataReaderInfo_DoesSupportPadding, MetadataReaderInfo_DoesRequireFixedSize, MetadataReaderInfo_GetPatterns, MetadataReaderInfo_MatchesPattern, MetadataReaderInfo_CreateInstance }; static void read_metadata_patterns(MetadataReaderInfo *info, GUID *container_guid, struct metadata_container *container) { UINT pattern_count=0, patterns_size=0; WCHAR subkeyname[11], guidkeyname[39]; LONG res; HKEY containers_key, guid_key, patternkey; static const WCHAR uintformatW[] = {'%','u',0}; static const WCHAR patternW[] = {'P','a','t','t','e','r','n',0}; static const WCHAR positionW[] = {'P','o','s','i','t','i','o','n',0}; static const WCHAR maskW[] = {'M','a','s','k',0}; static const WCHAR dataoffsetW[] = {'D','a','t','a','O','f','f','s','e','t',0}; UINT i; WICMetadataPattern *patterns; BYTE *patterns_ptr; DWORD length, valuesize; res = RegOpenKeyExW(info->classkey, containers_keyname, 0, KEY_READ, &containers_key); if (res != ERROR_SUCCESS) return; StringFromGUID2(container_guid, guidkeyname, 39); res = RegOpenKeyExW(containers_key, guidkeyname, 0, KEY_READ, &guid_key); RegCloseKey(containers_key); if (res != ERROR_SUCCESS) return; res = RegQueryInfoKeyW(guid_key, NULL, NULL, NULL, &pattern_count, NULL, NULL, NULL, NULL, NULL, NULL, NULL); if (res != ERROR_SUCCESS) { RegCloseKey(guid_key); return; } patterns_size = pattern_count * sizeof(WICMetadataPattern); patterns = heap_alloc(patterns_size); if (!patterns) { RegCloseKey(guid_key); return; } for (i=0; res == ERROR_SUCCESS && i < pattern_count; i++) { snprintfW(subkeyname, 11, uintformatW, i); res = RegOpenKeyExW(guid_key, subkeyname, 0, KEY_READ, &patternkey); if (res != ERROR_SUCCESS) break; res = RegGetValueW(patternkey, NULL, patternW, RRF_RT_REG_BINARY, NULL, NULL, &length); if (res == ERROR_SUCCESS) { patterns_size += length*2; patterns[i].Length = length; valuesize = sizeof(DWORD64); res = RegGetValueW(patternkey, NULL, dataoffsetW, RRF_RT_DWORD|RRF_RT_QWORD, NULL, &patterns[i].DataOffset, &valuesize); if (res) patterns[i].DataOffset.QuadPart = 0; patterns[i].Position.QuadPart = 0; valuesize = sizeof(DWORD64); res = RegGetValueW(patternkey, NULL, positionW, RRF_RT_DWORD|RRF_RT_QWORD, NULL, &patterns[i].Position, &valuesize); } RegCloseKey(patternkey); } if (res != ERROR_SUCCESS || !(patterns_ptr = heap_realloc(patterns, patterns_size))) { heap_free(patterns); RegCloseKey(guid_key); return; } patterns = (WICMetadataPattern*)patterns_ptr; patterns_ptr += pattern_count * sizeof(*patterns); for (i=0; res == ERROR_SUCCESS && i < pattern_count; i++) { snprintfW(subkeyname, 11, uintformatW, i); res = RegOpenKeyExW(guid_key, subkeyname, 0, KEY_READ, &patternkey); if (res != ERROR_SUCCESS) break; length = patterns[i].Length; patterns[i].Pattern = patterns_ptr; valuesize = length; res = RegGetValueW(patternkey, NULL, patternW, RRF_RT_REG_BINARY, NULL, patterns[i].Pattern, &valuesize); patterns_ptr += length; if (res == ERROR_SUCCESS) { patterns[i].Mask = patterns_ptr; valuesize = length; res = RegGetValueW(patternkey, NULL, maskW, RRF_RT_REG_BINARY, NULL, patterns[i].Mask, &valuesize); patterns_ptr += length; } RegCloseKey(patternkey); } RegCloseKey(guid_key); if (res != ERROR_SUCCESS) { heap_free(patterns); return; } container->pattern_count = pattern_count; container->patterns_size = patterns_size; container->patterns = patterns; } static BOOL read_metadata_info(MetadataReaderInfo *info) { UINT format_count; GUID *formats; HRESULT hr; hr = ComponentInfo_GetGuidList(info->classkey, containers_keyname, 0, NULL, &format_count); if (FAILED(hr)) return TRUE; formats = heap_calloc(format_count, sizeof(*formats)); if (!formats) return FALSE; hr = ComponentInfo_GetGuidList(info->classkey, containers_keyname, format_count, formats, &format_count); if (FAILED(hr)) { heap_free(formats); return FALSE; } info->container_formats = formats; info->container_count = format_count; if (format_count) { unsigned i; info->containers = heap_calloc(format_count, sizeof(*info->containers)); if (!info->containers) return FALSE; for (i = 0; i < format_count; i++) read_metadata_patterns(info, info->container_formats + i, info->containers + i); } return TRUE; } static HRESULT MetadataReaderInfo_Constructor(HKEY classkey, REFCLSID clsid, ComponentInfo **info) { MetadataReaderInfo *This; This = heap_alloc_zero(sizeof(*This)); if (!This) { RegCloseKey(classkey); return E_OUTOFMEMORY; } This->base.IWICComponentInfo_iface.lpVtbl = (const IWICComponentInfoVtbl*)&MetadataReaderInfo_Vtbl; This->base.ref = 1; This->classkey = classkey; This->base.clsid = *clsid; if (!read_metadata_info(This)) { IWICComponentInfo_Release(&This->base.IWICComponentInfo_iface); return WINCODEC_ERR_COMPONENTNOTFOUND; } *info = &This->base; return S_OK; } static const WCHAR clsid_keyname[] = {'C','L','S','I','D',0}; static const WCHAR instance_keyname[] = {'I','n','s','t','a','n','c','e',0}; struct category { WICComponentType type; const GUID *catid; HRESULT (*constructor)(HKEY,REFCLSID,ComponentInfo**); }; static const struct category categories[] = { {WICDecoder, &CATID_WICBitmapDecoders, BitmapDecoderInfo_Constructor}, {WICEncoder, &CATID_WICBitmapEncoders, BitmapEncoderInfo_Constructor}, {WICPixelFormatConverter, &CATID_WICFormatConverters, FormatConverterInfo_Constructor}, {WICPixelFormat, &CATID_WICPixelFormats, PixelFormatInfo_Constructor}, {WICMetadataReader, &CATID_WICMetadataReader, MetadataReaderInfo_Constructor}, {0} }; static int ComponentInfo_Compare(const void *key, const struct wine_rb_entry *entry) { ComponentInfo *info = WINE_RB_ENTRY_VALUE(entry, ComponentInfo, entry); return memcmp(key, &info->clsid, sizeof(info->clsid)); } static struct wine_rb_tree component_info_cache = { ComponentInfo_Compare }; static CRITICAL_SECTION component_info_cache_cs; static CRITICAL_SECTION_DEBUG component_info_cache_cs_dbg = { 0, 0, &component_info_cache_cs, { &component_info_cache_cs_dbg.ProcessLocksList, &component_info_cache_cs_dbg.ProcessLocksList }, 0, 0, { (DWORD_PTR)(__FILE__ ": component_info_cache") } }; static CRITICAL_SECTION component_info_cache_cs = { &component_info_cache_cs_dbg, -1, 0, 0, 0, 0 }; HRESULT CreateComponentInfo(REFCLSID clsid, IWICComponentInfo **ppIInfo) { struct wine_rb_entry *cache_entry; ComponentInfo *info; HKEY clsidkey; HKEY classkey; HKEY catidkey; HKEY instancekey; WCHAR guidstring[39]; LONG res; const struct category *category; BOOL found = FALSE; HRESULT hr; EnterCriticalSection(&component_info_cache_cs); cache_entry = wine_rb_get(&component_info_cache, clsid); if(cache_entry) { info = WINE_RB_ENTRY_VALUE(cache_entry, ComponentInfo, entry); IWICComponentInfo_AddRef(*ppIInfo = &info->IWICComponentInfo_iface); LeaveCriticalSection(&component_info_cache_cs); return S_OK; } res = RegOpenKeyExW(HKEY_CLASSES_ROOT, clsid_keyname, 0, KEY_READ, &clsidkey); if (res != ERROR_SUCCESS) { LeaveCriticalSection(&component_info_cache_cs); return HRESULT_FROM_WIN32(res); } for (category=categories; category->type; category++) { StringFromGUID2(category->catid, guidstring, 39); res = RegOpenKeyExW(clsidkey, guidstring, 0, KEY_READ, &catidkey); if (res == ERROR_SUCCESS) { res = RegOpenKeyExW(catidkey, instance_keyname, 0, KEY_READ, &instancekey); if (res == ERROR_SUCCESS) { StringFromGUID2(clsid, guidstring, 39); res = RegOpenKeyExW(instancekey, guidstring, 0, KEY_READ, &classkey); if (res == ERROR_SUCCESS) { RegCloseKey(classkey); found = TRUE; } RegCloseKey(instancekey); } RegCloseKey(catidkey); } if (found) break; } if (found) { res = RegOpenKeyExW(clsidkey, guidstring, 0, KEY_READ, &classkey); if (res == ERROR_SUCCESS) hr = category->constructor(classkey, clsid, &info); else hr = HRESULT_FROM_WIN32(res); } else { FIXME("%s is not supported\n", wine_dbgstr_guid(clsid)); hr = E_FAIL; } RegCloseKey(clsidkey); if (SUCCEEDED(hr)) { wine_rb_put(&component_info_cache, clsid, &info->entry); IWICComponentInfo_AddRef(*ppIInfo = &info->IWICComponentInfo_iface); } LeaveCriticalSection(&component_info_cache_cs); return hr; } void ReleaseComponentInfos(void) { ComponentInfo *info, *next_info; WINE_RB_FOR_EACH_ENTRY_DESTRUCTOR(info, next_info, &component_info_cache, ComponentInfo, entry) IWICComponentInfo_Release(&info->IWICComponentInfo_iface); } HRESULT get_decoder_info(REFCLSID clsid, IWICBitmapDecoderInfo **info) { IWICComponentInfo *compinfo; HRESULT hr; hr = CreateComponentInfo(clsid, &compinfo); if (FAILED(hr)) return hr; hr = IWICComponentInfo_QueryInterface(compinfo, &IID_IWICBitmapDecoderInfo, (void **)info); IWICComponentInfo_Release(compinfo); return hr; } typedef struct { IEnumUnknown IEnumUnknown_iface; LONG ref; struct list objects; struct list *cursor; CRITICAL_SECTION lock; /* Must be held when reading or writing cursor */ } ComponentEnum; static inline ComponentEnum *impl_from_IEnumUnknown(IEnumUnknown *iface) { return CONTAINING_RECORD(iface, ComponentEnum, IEnumUnknown_iface); } typedef struct { struct list entry; IUnknown *unk; } ComponentEnumItem; static const IEnumUnknownVtbl ComponentEnumVtbl; static HRESULT WINAPI ComponentEnum_QueryInterface(IEnumUnknown *iface, REFIID iid, void **ppv) { ComponentEnum *This = impl_from_IEnumUnknown(iface); TRACE("(%p,%s,%p)\n", iface, debugstr_guid(iid), ppv); if (!ppv) return E_INVALIDARG; if (IsEqualIID(&IID_IUnknown, iid) || IsEqualIID(&IID_IEnumUnknown, iid)) { *ppv = &This->IEnumUnknown_iface; } else { *ppv = NULL; return E_NOINTERFACE; } IUnknown_AddRef((IUnknown*)*ppv); return S_OK; } static ULONG WINAPI ComponentEnum_AddRef(IEnumUnknown *iface) { ComponentEnum *This = impl_from_IEnumUnknown(iface); ULONG ref = InterlockedIncrement(&This->ref); TRACE("(%p) refcount=%u\n", iface, ref); return ref; } static ULONG WINAPI ComponentEnum_Release(IEnumUnknown *iface) { ComponentEnum *This = impl_from_IEnumUnknown(iface); ULONG ref = InterlockedDecrement(&This->ref); ComponentEnumItem *cursor, *cursor2; TRACE("(%p) refcount=%u\n", iface, ref); if (ref == 0) { LIST_FOR_EACH_ENTRY_SAFE(cursor, cursor2, &This->objects, ComponentEnumItem, entry) { IUnknown_Release(cursor->unk); list_remove(&cursor->entry); HeapFree(GetProcessHeap(), 0, cursor); } This->lock.DebugInfo->Spare[0] = 0; DeleteCriticalSection(&This->lock); HeapFree(GetProcessHeap(), 0, This); } return ref; } static HRESULT WINAPI ComponentEnum_Next(IEnumUnknown *iface, ULONG celt, IUnknown **rgelt, ULONG *pceltFetched) { ComponentEnum *This = impl_from_IEnumUnknown(iface); ULONG num_fetched=0; ComponentEnumItem *item; HRESULT hr=S_OK; TRACE("(%p,%u,%p,%p)\n", iface, celt, rgelt, pceltFetched); EnterCriticalSection(&This->lock); while (num_fetched<celt) { if (!This->cursor) { hr = S_FALSE; break; } item = LIST_ENTRY(This->cursor, ComponentEnumItem, entry); IUnknown_AddRef(item->unk); rgelt[num_fetched] = item->unk; num_fetched++; This->cursor = list_next(&This->objects, This->cursor); } LeaveCriticalSection(&This->lock); if (pceltFetched) *pceltFetched = num_fetched; return hr; } static HRESULT WINAPI ComponentEnum_Skip(IEnumUnknown *iface, ULONG celt) { ComponentEnum *This = impl_from_IEnumUnknown(iface); ULONG i; HRESULT hr=S_OK; TRACE("(%p,%u)\n", iface, celt); EnterCriticalSection(&This->lock); for (i=0; i<celt; i++) { if (!This->cursor) { hr = S_FALSE; break; } This->cursor = list_next(&This->objects, This->cursor); } LeaveCriticalSection(&This->lock); return hr; } static HRESULT WINAPI ComponentEnum_Reset(IEnumUnknown *iface) { ComponentEnum *This = impl_from_IEnumUnknown(iface); TRACE("(%p)\n", iface); EnterCriticalSection(&This->lock); This->cursor = list_head(&This->objects); LeaveCriticalSection(&This->lock); return S_OK; } static HRESULT WINAPI ComponentEnum_Clone(IEnumUnknown *iface, IEnumUnknown **ppenum) { ComponentEnum *This = impl_from_IEnumUnknown(iface); ComponentEnum *new_enum; ComponentEnumItem *old_item, *new_item; HRESULT ret=S_OK; struct list *old_cursor; new_enum = HeapAlloc(GetProcessHeap(), 0, sizeof(ComponentEnum)); if (!new_enum) { *ppenum = NULL; return E_OUTOFMEMORY; } new_enum->IEnumUnknown_iface.lpVtbl = &ComponentEnumVtbl; new_enum->ref = 1; new_enum->cursor = NULL; list_init(&new_enum->objects); InitializeCriticalSection(&new_enum->lock); new_enum->lock.DebugInfo->Spare[0] = (DWORD_PTR)(__FILE__ ": ComponentEnum.lock"); EnterCriticalSection(&This->lock); old_cursor = This->cursor; LeaveCriticalSection(&This->lock); LIST_FOR_EACH_ENTRY(old_item, &This->objects, ComponentEnumItem, entry) { new_item = HeapAlloc(GetProcessHeap(), 0, sizeof(ComponentEnumItem)); if (!new_item) { ret = E_OUTOFMEMORY; break; } new_item->unk = old_item->unk; list_add_tail(&new_enum->objects, &new_item->entry); IUnknown_AddRef(new_item->unk); if (&old_item->entry == old_cursor) new_enum->cursor = &new_item->entry; } if (FAILED(ret)) { IEnumUnknown_Release(&new_enum->IEnumUnknown_iface); *ppenum = NULL; } else *ppenum = &new_enum->IEnumUnknown_iface; return ret; } static const IEnumUnknownVtbl ComponentEnumVtbl = { ComponentEnum_QueryInterface, ComponentEnum_AddRef, ComponentEnum_Release, ComponentEnum_Next, ComponentEnum_Skip, ComponentEnum_Reset, ComponentEnum_Clone }; HRESULT CreateComponentEnumerator(DWORD componentTypes, DWORD options, IEnumUnknown **ppIEnumUnknown) { ComponentEnum *This; ComponentEnumItem *item; const struct category *category; HKEY clsidkey, catidkey, instancekey; WCHAR guidstring[39]; LONG res; int i; HRESULT hr=S_OK; CLSID clsid; if (options) FIXME("ignoring flags %x\n", options); res = RegOpenKeyExW(HKEY_CLASSES_ROOT, clsid_keyname, 0, KEY_READ, &clsidkey); if (res != ERROR_SUCCESS) return HRESULT_FROM_WIN32(res); This = HeapAlloc(GetProcessHeap(), 0, sizeof(ComponentEnum)); if (!This) { RegCloseKey(clsidkey); return E_OUTOFMEMORY; } This->IEnumUnknown_iface.lpVtbl = &ComponentEnumVtbl; This->ref = 1; list_init(&This->objects); InitializeCriticalSection(&This->lock); This->lock.DebugInfo->Spare[0] = (DWORD_PTR)(__FILE__ ": ComponentEnum.lock"); for (category=categories; category->type && hr == S_OK; category++) { if ((category->type & componentTypes) == 0) continue; StringFromGUID2(category->catid, guidstring, 39); res = RegOpenKeyExW(clsidkey, guidstring, 0, KEY_READ, &catidkey); if (res == ERROR_SUCCESS) { res = RegOpenKeyExW(catidkey, instance_keyname, 0, KEY_READ, &instancekey); if (res == ERROR_SUCCESS) { i=0; for (;;i++) { DWORD guidstring_size = 39; res = RegEnumKeyExW(instancekey, i, guidstring, &guidstring_size, NULL, NULL, NULL, NULL); if (res != ERROR_SUCCESS) break; item = HeapAlloc(GetProcessHeap(), 0, sizeof(ComponentEnumItem)); if (!item) { hr = E_OUTOFMEMORY; break; } hr = CLSIDFromString(guidstring, &clsid); if (SUCCEEDED(hr)) { hr = CreateComponentInfo(&clsid, (IWICComponentInfo**)&item->unk); if (SUCCEEDED(hr)) list_add_tail(&This->objects, &item->entry); } if (FAILED(hr)) { HeapFree(GetProcessHeap(), 0, item); hr = S_OK; } } RegCloseKey(instancekey); } RegCloseKey(catidkey); } if (res != ERROR_SUCCESS && res != ERROR_NO_MORE_ITEMS) hr = HRESULT_FROM_WIN32(res); } RegCloseKey(clsidkey); if (SUCCEEDED(hr)) { IEnumUnknown_Reset(&This->IEnumUnknown_iface); *ppIEnumUnknown = &This->IEnumUnknown_iface; } else { *ppIEnumUnknown = NULL; IEnumUnknown_Release(&This->IEnumUnknown_iface); } return hr; } static BOOL is_1bpp_format(const WICPixelFormatGUID *format) { return IsEqualGUID(format, &GUID_WICPixelFormatBlackWhite) || IsEqualGUID(format, &GUID_WICPixelFormat1bppIndexed); } HRESULT WINAPI WICConvertBitmapSource(REFWICPixelFormatGUID dstFormat, IWICBitmapSource *pISrc, IWICBitmapSource **ppIDst) { HRESULT res; IEnumUnknown *enumconverters; IUnknown *unkconverterinfo; IWICFormatConverterInfo *converterinfo=NULL; IWICFormatConverter *converter=NULL; GUID srcFormat; WCHAR srcformatstr[39], dstformatstr[39]; BOOL canconvert; ULONG num_fetched; TRACE("%s,%p,%p\n", debugstr_guid(dstFormat), pISrc, ppIDst); res = IWICBitmapSource_GetPixelFormat(pISrc, &srcFormat); if (FAILED(res)) return res; if (IsEqualGUID(&srcFormat, dstFormat) || (is_1bpp_format(&srcFormat) && is_1bpp_format(dstFormat))) { IWICBitmapSource_AddRef(pISrc); *ppIDst = pISrc; return S_OK; } StringFromGUID2(&srcFormat, srcformatstr, 39); StringFromGUID2(dstFormat, dstformatstr, 39); res = CreateComponentEnumerator(WICPixelFormatConverter, 0, &enumconverters); if (FAILED(res)) return res; while (!converter) { res = IEnumUnknown_Next(enumconverters, 1, &unkconverterinfo, &num_fetched); if (res == S_OK) { res = IUnknown_QueryInterface(unkconverterinfo, &IID_IWICFormatConverterInfo, (void**)&converterinfo); if (SUCCEEDED(res)) { canconvert = ConverterSupportsFormat(converterinfo, srcformatstr); if (canconvert) canconvert = ConverterSupportsFormat(converterinfo, dstformatstr); if (canconvert) { res = IWICFormatConverterInfo_CreateInstance(converterinfo, &converter); if (SUCCEEDED(res)) res = IWICFormatConverter_CanConvert(converter, &srcFormat, dstFormat, &canconvert); if (SUCCEEDED(res) && canconvert) res = IWICFormatConverter_Initialize(converter, pISrc, dstFormat, WICBitmapDitherTypeNone, NULL, 0.0, WICBitmapPaletteTypeMedianCut); if (FAILED(res) || !canconvert) { if (converter) { IWICFormatConverter_Release(converter); converter = NULL; } } } IWICFormatConverterInfo_Release(converterinfo); } IUnknown_Release(unkconverterinfo); } else break; } IEnumUnknown_Release(enumconverters); if (converter) { res = IWICFormatConverter_QueryInterface(converter, &IID_IWICBitmapSource, (void **)ppIDst); IWICFormatConverter_Release(converter); return res; } else { FIXME("cannot convert %s to %s\n", debugstr_guid(&srcFormat), debugstr_guid(dstFormat)); *ppIDst = NULL; return WINCODEC_ERR_COMPONENTNOTFOUND; } }