/* * Copyright 2016 Andrew Eikum for CodeWeavers * Copyright 2017 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 #include #define COBJMACROS #define NONAMELESSUNION #include "windef.h" #include "winbase.h" #include "objbase.h" #include "propvarutil.h" #include "wincodecs_private.h" #include "wine/debug.h" WINE_DEFAULT_DEBUG_CHANNEL(wincodecs); static const WCHAR *map_shortname_to_schema(const GUID *format, const WCHAR *name); typedef struct { IWICMetadataQueryReader IWICMetadataQueryReader_iface; LONG ref; IWICMetadataBlockReader *block; WCHAR *root; } QueryReader; static inline QueryReader *impl_from_IWICMetadataQueryReader(IWICMetadataQueryReader *iface) { return CONTAINING_RECORD(iface, QueryReader, IWICMetadataQueryReader_iface); } static HRESULT WINAPI mqr_QueryInterface(IWICMetadataQueryReader *iface, REFIID riid, void **ppvObject) { QueryReader *This = impl_from_IWICMetadataQueryReader(iface); TRACE("(%p,%s,%p)\n", This, debugstr_guid(riid), ppvObject); if (IsEqualGUID(riid, &IID_IUnknown) || IsEqualGUID(riid, &IID_IWICMetadataQueryReader)) *ppvObject = &This->IWICMetadataQueryReader_iface; else *ppvObject = NULL; if (*ppvObject) { IUnknown_AddRef((IUnknown*)*ppvObject); return S_OK; } return E_NOINTERFACE; } static ULONG WINAPI mqr_AddRef(IWICMetadataQueryReader *iface) { QueryReader *This = impl_from_IWICMetadataQueryReader(iface); ULONG ref = InterlockedIncrement(&This->ref); TRACE("(%p) refcount=%u\n", This, ref); return ref; } static ULONG WINAPI mqr_Release(IWICMetadataQueryReader *iface) { QueryReader *This = impl_from_IWICMetadataQueryReader(iface); ULONG ref = InterlockedDecrement(&This->ref); TRACE("(%p) refcount=%u\n", This, ref); if (!ref) { IWICMetadataBlockReader_Release(This->block); HeapFree(GetProcessHeap(), 0, This->root); HeapFree(GetProcessHeap(), 0, This); } return ref; } static HRESULT WINAPI mqr_GetContainerFormat(IWICMetadataQueryReader *iface, GUID *format) { QueryReader *This = impl_from_IWICMetadataQueryReader(iface); TRACE("(%p,%p)\n", This, format); return IWICMetadataBlockReader_GetContainerFormat(This->block, format); } static HRESULT WINAPI mqr_GetLocation(IWICMetadataQueryReader *iface, UINT len, WCHAR *location, UINT *ret_len) { QueryReader *This = impl_from_IWICMetadataQueryReader(iface); const WCHAR *root; UINT actual_len; TRACE("(%p,%u,%p,%p)\n", This, len, location, ret_len); if (!ret_len) return E_INVALIDARG; root = This->root ? This->root : L"/"; actual_len = lstrlenW(root) + 1; if (location) { if (len < actual_len) return WINCODEC_ERR_INSUFFICIENTBUFFER; memcpy(location, root, actual_len * sizeof(WCHAR)); } *ret_len = actual_len; return S_OK; } struct string_t { const WCHAR *str; int len; }; static const struct { int len; WCHAR str[10]; VARTYPE vt; } str2vt[] = { { 4, {'c','h','a','r'}, VT_I1 }, { 5, {'u','c','h','a','r'}, VT_UI1 }, { 5, {'s','h','o','r','t'}, VT_I2 }, { 6, {'u','s','h','o','r','t'}, VT_UI2 }, { 4, {'l','o','n','g'}, VT_I4 }, { 5, {'u','l','o','n','g'}, VT_UI4 }, { 3, {'i','n','t'}, VT_I4 }, { 4, {'u','i','n','t'}, VT_UI4 }, { 8, {'l','o','n','g','l','o','n','g'}, VT_I8 }, { 9, {'u','l','o','n','g','l','o','n','g'}, VT_UI8 }, { 5, {'f','l','o','a','t'}, VT_R4 }, { 6, {'d','o','u','b','l','e'}, VT_R8 }, { 3, {'s','t','r'}, VT_LPSTR }, { 4, {'w','s','t','r'}, VT_LPWSTR }, { 4, {'g','u','i','d'}, VT_CLSID }, { 4, {'b','o','o','l'}, VT_BOOL } }; static VARTYPE map_type(struct string_t *str) { UINT i; for (i = 0; i < ARRAY_SIZE(str2vt); i++) { if (str2vt[i].len == str->len) { if (CompareStringW(LOCALE_SYSTEM_DEFAULT, NORM_IGNORECASE, str->str, str->len, str2vt[i].str, str2vt[i].len) == CSTR_EQUAL) return str2vt[i].vt; } } WARN("type %s is not recognized\n", wine_dbgstr_wn(str->str, str->len)); return VT_ILLEGAL; } static HRESULT get_token(struct string_t *elem, PROPVARIANT *id, PROPVARIANT *schema, int *idx) { const WCHAR *start, *end, *p; WCHAR *bstr; struct string_t next_elem; HRESULT hr; TRACE("%s, len %d\n", wine_dbgstr_wn(elem->str, elem->len), elem->len); PropVariantInit(id); PropVariantInit(schema); if (!elem->len) return S_OK; start = elem->str; if (*start == '[') { WCHAR *idx_end; if (start[1] < '0' || start[1] > '9') return DISP_E_TYPEMISMATCH; *idx = wcstol(start + 1, &idx_end, 10); if (idx_end > elem->str + elem->len) return WINCODEC_ERR_INVALIDQUERYREQUEST; if (*idx_end != ']') return WINCODEC_ERR_INVALIDQUERYREQUEST; if (*idx < 0) return WINCODEC_ERR_INVALIDQUERYREQUEST; end = idx_end + 1; next_elem.str = end; next_elem.len = elem->len - (end - start); hr = get_token(&next_elem, id, schema, idx); if (hr != S_OK) { TRACE("get_token error %#x\n", hr); return hr; } elem->len = (end - start) + next_elem.len; TRACE("indexed %s [%d]\n", wine_dbgstr_wn(elem->str, elem->len), *idx); return S_OK; } else if (*start == '{') { VARTYPE vt; PROPVARIANT next_token; end = wmemchr(start + 1, '=', elem->len - 1); if (!end) return WINCODEC_ERR_INVALIDQUERYREQUEST; if (end > elem->str + elem->len) return WINCODEC_ERR_INVALIDQUERYREQUEST; next_elem.str = start + 1; next_elem.len = end - start - 1; vt = map_type(&next_elem); TRACE("type %s => %d\n", wine_dbgstr_wn(next_elem.str, next_elem.len), vt); if (vt == VT_ILLEGAL) return WINCODEC_ERR_WRONGSTATE; next_token.vt = VT_BSTR; next_token.u.bstrVal = SysAllocStringLen(NULL, elem->len - (end - start) + 1); if (!next_token.u.bstrVal) return E_OUTOFMEMORY; bstr = next_token.u.bstrVal; end++; while (*end && *end != '}' && end - start < elem->len) { if (*end == '\\') end++; *bstr++ = *end++; } if (*end != '}') { PropVariantClear(&next_token); return WINCODEC_ERR_INVALIDQUERYREQUEST; } *bstr = 0; TRACE("schema/id %s\n", wine_dbgstr_w(next_token.u.bstrVal)); if (vt == VT_CLSID) { id->vt = VT_CLSID; id->u.puuid = CoTaskMemAlloc(sizeof(GUID)); if (!id->u.puuid) { PropVariantClear(&next_token); return E_OUTOFMEMORY; } hr = UuidFromStringW(next_token.u.bstrVal, id->u.puuid); } else hr = PropVariantChangeType(id, &next_token, 0, vt); PropVariantClear(&next_token); if (hr != S_OK) { PropVariantClear(id); PropVariantClear(schema); return hr; } end++; if (*end == ':') { PROPVARIANT next_id, next_schema; int next_idx = 0; next_elem.str = end + 1; next_elem.len = elem->len - (end - start + 1); hr = get_token(&next_elem, &next_id, &next_schema, &next_idx); if (hr != S_OK) { TRACE("get_token error %#x\n", hr); return hr; } elem->len = (end - start + 1) + next_elem.len; TRACE("id %s [%d]\n", wine_dbgstr_wn(elem->str, elem->len), *idx); if (next_schema.vt != VT_EMPTY) { PropVariantClear(&next_id); PropVariantClear(&next_schema); return WINCODEC_ERR_WRONGSTATE; } *schema = *id; *id = next_id; return S_OK; } elem->len = end - start; return S_OK; } end = wmemchr(start, '/', elem->len); if (!end) end = start + elem->len; p = wmemchr(start, ':', end - start); if (p) { next_elem.str = p + 1; next_elem.len = end - p - 1; elem->len = p - start; } else elem->len = end - start; id->vt = VT_BSTR; id->u.bstrVal = SysAllocStringLen(NULL, elem->len + 1); if (!id->u.bstrVal) return E_OUTOFMEMORY; bstr = id->u.bstrVal; p = elem->str; while (p - elem->str < elem->len) { if (*p == '\\') p++; *bstr++ = *p++; } *bstr = 0; TRACE("%s [%d]\n", wine_dbgstr_variant((VARIANT *)id), *idx); if (*p == ':') { PROPVARIANT next_id, next_schema; int next_idx = 0; hr = get_token(&next_elem, &next_id, &next_schema, &next_idx); if (hr != S_OK) { TRACE("get_token error %#x\n", hr); PropVariantClear(id); PropVariantClear(schema); return hr; } elem->len += next_elem.len + 1; TRACE("id %s [%d]\n", wine_dbgstr_wn(elem->str, elem->len), *idx); if (next_schema.vt != VT_EMPTY) { PropVariantClear(&next_id); PropVariantClear(&next_schema); PropVariantClear(id); PropVariantClear(schema); return WINCODEC_ERR_WRONGSTATE; } *schema = *id; *id = next_id; } return S_OK; } static HRESULT find_reader_from_block(IWICMetadataBlockReader *block_reader, UINT index, GUID *guid, IWICMetadataReader **reader) { HRESULT hr; GUID format; IWICMetadataReader *new_reader; UINT count, i, matched_index; *reader = NULL; hr = IWICMetadataBlockReader_GetCount(block_reader, &count); if (hr != S_OK) return hr; matched_index = 0; for (i = 0; i < count; i++) { hr = IWICMetadataBlockReader_GetReaderByIndex(block_reader, i, &new_reader); if (hr != S_OK) return hr; hr = IWICMetadataReader_GetMetadataFormat(new_reader, &format); if (hr == S_OK) { if (IsEqualGUID(&format, guid)) { if (matched_index == index) { *reader = new_reader; return S_OK; } matched_index++; } } IWICMetadataReader_Release(new_reader); if (hr != S_OK) return hr; } return WINCODEC_ERR_PROPERTYNOTFOUND; } static HRESULT get_next_reader(IWICMetadataReader *reader, UINT index, GUID *guid, IWICMetadataReader **new_reader) { HRESULT hr; PROPVARIANT schema, id, value; *new_reader = NULL; PropVariantInit(&schema); PropVariantInit(&id); PropVariantInit(&value); if (index) { schema.vt = VT_UI2; schema.u.uiVal = index; } id.vt = VT_CLSID; id.u.puuid = guid; hr = IWICMetadataReader_GetValue(reader, &schema, &id, &value); if (hr != S_OK) return hr; if (value.vt == VT_UNKNOWN) hr = IUnknown_QueryInterface(value.u.punkVal, &IID_IWICMetadataReader, (void **)new_reader); else hr = WINCODEC_ERR_UNEXPECTEDMETADATATYPE; PropVariantClear(&value); return hr; } static HRESULT WINAPI mqr_GetMetadataByName(IWICMetadataQueryReader *iface, LPCWSTR query, PROPVARIANT *value) { QueryReader *This = impl_from_IWICMetadataQueryReader(iface); struct string_t elem; WCHAR *full_query; const WCHAR *p; int index, len; PROPVARIANT tk_id, tk_schema, new_value; GUID guid; IWICMetadataReader *reader; HRESULT hr = S_OK; TRACE("(%p,%s,%p)\n", This, wine_dbgstr_w(query), value); len = lstrlenW(query) + 1; if (This->root) len += lstrlenW(This->root); full_query = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); full_query[0] = 0; if (This->root) lstrcpyW(full_query, This->root); lstrcatW(full_query, query); PropVariantInit(&tk_id); PropVariantInit(&tk_schema); PropVariantInit(&new_value); reader = NULL; p = full_query; while (*p) { if (*p != '/') { WARN("query should start with '/'\n"); hr = WINCODEC_ERR_PROPERTYNOTSUPPORTED; break; } p++; index = 0; elem.str = p; elem.len = lstrlenW(p); hr = get_token(&elem, &tk_id, &tk_schema, &index); if (hr != S_OK) { WARN("get_token error %#x\n", hr); break; } TRACE("parsed %d characters: %s, index %d\n", elem.len, wine_dbgstr_wn(elem.str, elem.len), index); TRACE("id %s, schema %s\n", wine_dbgstr_variant((VARIANT *)&tk_id), wine_dbgstr_variant((VARIANT *)&tk_schema)); if (!elem.len) break; if (tk_id.vt == VT_CLSID || (tk_id.vt == VT_BSTR && WICMapShortNameToGuid(tk_id.u.bstrVal, &guid) == S_OK)) { WCHAR *root; if (tk_schema.vt != VT_EMPTY) { FIXME("unsupported schema vt %u\n", tk_schema.vt); PropVariantClear(&tk_schema); } if (tk_id.vt == VT_CLSID) guid = *tk_id.u.puuid; if (reader) { IWICMetadataReader *new_reader; hr = get_next_reader(reader, index, &guid, &new_reader); IWICMetadataReader_Release(reader); reader = new_reader; } else hr = find_reader_from_block(This->block, index, &guid, &reader); if (hr != S_OK) break; root = SysAllocStringLen(NULL, elem.str + elem.len - full_query + 2); if (!root) { hr = E_OUTOFMEMORY; break; } lstrcpynW(root, full_query, p - full_query + elem.len + 1); PropVariantClear(&new_value); new_value.vt = VT_UNKNOWN; hr = MetadataQueryReader_CreateInstance(This->block, root, (IWICMetadataQueryReader **)&new_value.u.punkVal); SysFreeString(root); if (hr != S_OK) break; } else { PROPVARIANT schema, id; if (!reader) { hr = WINCODEC_ERR_INVALIDQUERYREQUEST; break; } if (tk_schema.vt == VT_BSTR) { hr = IWICMetadataReader_GetMetadataFormat(reader, &guid); if (hr != S_OK) break; schema.vt = VT_LPWSTR; schema.u.pwszVal = (LPWSTR)map_shortname_to_schema(&guid, tk_schema.u.bstrVal); if (!schema.u.pwszVal) schema.u.pwszVal = tk_schema.u.bstrVal; } else schema = tk_schema; if (tk_id.vt == VT_BSTR) { id.vt = VT_LPWSTR; id.u.pwszVal = tk_id.u.bstrVal; } else id = tk_id; PropVariantClear(&new_value); hr = IWICMetadataReader_GetValue(reader, &schema, &id, &new_value); if (hr != S_OK) break; } p += elem.len; PropVariantClear(&tk_id); PropVariantClear(&tk_schema); } if (reader) IWICMetadataReader_Release(reader); PropVariantClear(&tk_id); PropVariantClear(&tk_schema); if (hr == S_OK && value) *value = new_value; else PropVariantClear(&new_value); HeapFree(GetProcessHeap(), 0, full_query); return hr; } static HRESULT WINAPI mqr_GetEnumerator(IWICMetadataQueryReader *iface, IEnumString **ppIEnumString) { QueryReader *This = impl_from_IWICMetadataQueryReader(iface); FIXME("(%p,%p)\n", This, ppIEnumString); return E_NOTIMPL; } static IWICMetadataQueryReaderVtbl mqr_vtbl = { mqr_QueryInterface, mqr_AddRef, mqr_Release, mqr_GetContainerFormat, mqr_GetLocation, mqr_GetMetadataByName, mqr_GetEnumerator }; HRESULT MetadataQueryReader_CreateInstance(IWICMetadataBlockReader *mbr, const WCHAR *root, IWICMetadataQueryReader **out) { QueryReader *obj; obj = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(*obj)); if (!obj) return E_OUTOFMEMORY; obj->IWICMetadataQueryReader_iface.lpVtbl = &mqr_vtbl; obj->ref = 1; IWICMetadataBlockReader_AddRef(mbr); obj->block = mbr; obj->root = root ? heap_strdupW(root) : NULL; *out = &obj->IWICMetadataQueryReader_iface; return S_OK; } static const struct { const GUID *guid; const WCHAR *name; } guid2name[] = { { &GUID_ContainerFormatBmp, L"bmp" }, { &GUID_ContainerFormatPng, L"png" }, { &GUID_ContainerFormatIco, L"ico" }, { &GUID_ContainerFormatJpeg, L"jpg" }, { &GUID_ContainerFormatTiff, L"tiff" }, { &GUID_ContainerFormatGif, L"gif" }, { &GUID_ContainerFormatWmp, L"wmphoto" }, { &GUID_MetadataFormatUnknown, L"unknown" }, { &GUID_MetadataFormatIfd, L"ifd" }, { &GUID_MetadataFormatSubIfd, L"sub" }, { &GUID_MetadataFormatExif, L"exif" }, { &GUID_MetadataFormatGps, L"gps" }, { &GUID_MetadataFormatInterop, L"interop" }, { &GUID_MetadataFormatApp0, L"app0" }, { &GUID_MetadataFormatApp1, L"app1" }, { &GUID_MetadataFormatApp13, L"app13" }, { &GUID_MetadataFormatIPTC, L"iptc" }, { &GUID_MetadataFormatIRB, L"irb" }, { &GUID_MetadataFormat8BIMIPTC, L"8bimiptc" }, { &GUID_MetadataFormat8BIMResolutionInfo, L"8bimResInfo" }, { &GUID_MetadataFormat8BIMIPTCDigest, L"8bimiptcdigest" }, { &GUID_MetadataFormatXMP, L"xmp" }, { &GUID_MetadataFormatThumbnail, L"thumb" }, { &GUID_MetadataFormatChunktEXt, L"tEXt" }, { &GUID_MetadataFormatXMPStruct, L"xmpstruct" }, { &GUID_MetadataFormatXMPBag, L"xmpbag" }, { &GUID_MetadataFormatXMPSeq, L"xmpseq" }, { &GUID_MetadataFormatXMPAlt, L"xmpalt" }, { &GUID_MetadataFormatLSD, L"logscrdesc" }, { &GUID_MetadataFormatIMD, L"imgdesc" }, { &GUID_MetadataFormatGCE, L"grctlext" }, { &GUID_MetadataFormatAPE, L"appext" }, { &GUID_MetadataFormatJpegChrominance, L"chrominance" }, { &GUID_MetadataFormatJpegLuminance, L"luminance" }, { &GUID_MetadataFormatJpegComment, L"com" }, { &GUID_MetadataFormatGifComment, L"commentext" }, { &GUID_MetadataFormatChunkgAMA, L"gAMA" }, { &GUID_MetadataFormatChunkbKGD, L"bKGD" }, { &GUID_MetadataFormatChunkiTXt, L"iTXt" }, { &GUID_MetadataFormatChunkcHRM, L"cHRM" }, { &GUID_MetadataFormatChunkhIST, L"hIST" }, { &GUID_MetadataFormatChunkiCCP, L"iCCP" }, { &GUID_MetadataFormatChunksRGB, L"sRGB" }, { &GUID_MetadataFormatChunktIME, L"tIME" } }; HRESULT WINAPI WICMapGuidToShortName(REFGUID guid, UINT len, WCHAR *name, UINT *ret_len) { UINT i; TRACE("%s,%u,%p,%p\n", wine_dbgstr_guid(guid), len, name, ret_len); if (!guid) return E_INVALIDARG; for (i = 0; i < ARRAY_SIZE(guid2name); i++) { if (IsEqualGUID(guid, guid2name[i].guid)) { if (name) { if (!len) return E_INVALIDARG; len = min(len - 1, lstrlenW(guid2name[i].name)); memcpy(name, guid2name[i].name, len * sizeof(WCHAR)); name[len] = 0; if (len < lstrlenW(guid2name[i].name)) return HRESULT_FROM_WIN32(ERROR_INSUFFICIENT_BUFFER); } if (ret_len) *ret_len = lstrlenW(guid2name[i].name) + 1; return S_OK; } } return WINCODEC_ERR_PROPERTYNOTFOUND; } HRESULT WINAPI WICMapShortNameToGuid(PCWSTR name, GUID *guid) { UINT i; TRACE("%s,%p\n", debugstr_w(name), guid); if (!name || !guid) return E_INVALIDARG; for (i = 0; i < ARRAY_SIZE(guid2name); i++) { if (!lstrcmpiW(name, guid2name[i].name)) { *guid = *guid2name[i].guid; return S_OK; } } return WINCODEC_ERR_PROPERTYNOTFOUND; } static const struct { const WCHAR *name; const WCHAR *schema; } name2schema[] = { { L"rdf", L"http://www.w3.org/1999/02/22-rdf-syntax-ns#" }, { L"dc", L"http://purl.org/dc/elements/1.1/" }, { L"xmp", L"http://ns.adobe.com/xap/1.0/" }, { L"xmpidq", L"http://ns.adobe.com/xmp/Identifier/qual/1.0/" }, { L"xmpRights", L"http://ns.adobe.com/xap/1.0/rights/" }, { L"xmpMM", L"http://ns.adobe.com/xap/1.0/mm/" }, { L"xmpBJ", L"http://ns.adobe.com/xap/1.0/bj/" }, { L"xmpTPg", L"http://ns.adobe.com/xap/1.0/t/pg/" }, { L"pdf", L"http://ns.adobe.com/pdf/1.3/" }, { L"photoshop", L"http://ns.adobe.com/photoshop/1.0/" }, { L"tiff", L"http://ns.adobe.com/tiff/1.0/" }, { L"exif", L"http://ns.adobe.com/exif/1.0/" }, { L"stDim", L"http://ns.adobe.com/xap/1.0/sType/Dimensions#" }, { L"xapGImg", L"http://ns.adobe.com/xap/1.0/g/img/" }, { L"stEvt", L"http://ns.adobe.com/xap/1.0/sType/ResourceEvent#" }, { L"stRef", L"http://ns.adobe.com/xap/1.0/sType/ResourceRef#" }, { L"stVer", L"http://ns.adobe.com/xap/1.0/sType/Version#" }, { L"stJob", L"http://ns.adobe.com/xap/1.0/sType/Job#" }, { L"aux", L"http://ns.adobe.com/exif/1.0/aux/" }, { L"crs", L"http://ns.adobe.com/camera-raw-settings/1.0/" }, { L"xmpDM", L"http://ns.adobe.com/xmp/1.0/DynamicMedia/" }, { L"Iptc4xmpCore", L"http://iptc.org/std/Iptc4xmpCore/1.0/xmlns/" }, { L"MicrosoftPhoto", L"http://ns.microsoft.com/photo/1.0/" }, { L"MP", L"http://ns.microsoft.com/photo/1.2/" }, { L"MPRI", L"http://ns.microsoft.com/photo/1.2/t/RegionInfo#" }, { L"MPReg", L"http://ns.microsoft.com/photo/1.2/t/Region#" } }; static const WCHAR *map_shortname_to_schema(const GUID *format, const WCHAR *name) { UINT i; /* It appears that the only metadata formats * that support schemas are xmp and xmpstruct. */ if (!IsEqualGUID(format, &GUID_MetadataFormatXMP) && !IsEqualGUID(format, &GUID_MetadataFormatXMPStruct)) return NULL; for (i = 0; i < ARRAY_SIZE(name2schema); i++) { if (!wcscmp(name2schema[i].name, name)) return name2schema[i].schema; } return NULL; } HRESULT WINAPI WICMapSchemaToName(REFGUID format, LPWSTR schema, UINT len, WCHAR *name, UINT *ret_len) { UINT i; TRACE("%s,%s,%u,%p,%p\n", wine_dbgstr_guid(format), debugstr_w(schema), len, name, ret_len); if (!format || !schema || !ret_len) return E_INVALIDARG; /* It appears that the only metadata formats * that support schemas are xmp and xmpstruct. */ if (!IsEqualGUID(format, &GUID_MetadataFormatXMP) && !IsEqualGUID(format, &GUID_MetadataFormatXMPStruct)) return WINCODEC_ERR_PROPERTYNOTFOUND; for (i = 0; i < ARRAY_SIZE(name2schema); i++) { if (!wcscmp(name2schema[i].schema, schema)) { if (name) { if (!len) return E_INVALIDARG; len = min(len - 1, lstrlenW(name2schema[i].name)); memcpy(name, name2schema[i].name, len * sizeof(WCHAR)); name[len] = 0; if (len < lstrlenW(name2schema[i].name)) return HRESULT_FROM_WIN32(ERROR_INSUFFICIENT_BUFFER); } *ret_len = lstrlenW(name2schema[i].name) + 1; return S_OK; } } return WINCODEC_ERR_PROPERTYNOTFOUND; }