1 /*
2 * Copyright 2009 Vincent Povirk for CodeWeavers
3 * Copyright 2013 Ludger Sprenker
4 *
5 * This library is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU Lesser General Public
7 * License as published by the Free Software Foundation; either
8 * version 2.1 of the License, or (at your option) any later version.
9 *
10 * This library is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * Lesser General Public License for more details.
14 *
15 * You should have received a copy of the GNU Lesser General Public
16 * License along with this library; if not, write to the Free Software
17 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
18 */
19
20 #include "config.h"
21
22 #include <stdarg.h>
23
24 #define COBJMACROS
25
26 #include "windef.h"
27 #include "winbase.h"
28 #include "objbase.h"
29 #include "wine/unicode.h"
30
31 #include "wincodecs_private.h"
32
33 #include "wine/debug.h"
34
35 WINE_DEFAULT_DEBUG_CHANNEL(wincodecs);
36
37 typedef struct PropertyBag {
38 IPropertyBag2 IPropertyBag2_iface;
39 LONG ref;
40 UINT prop_count;
41 PROPBAG2 *properties;
42 VARIANT *values;
43 } PropertyBag;
44
impl_from_IPropertyBag2(IPropertyBag2 * iface)45 static inline PropertyBag *impl_from_IPropertyBag2(IPropertyBag2 *iface)
46 {
47 return CONTAINING_RECORD(iface, PropertyBag, IPropertyBag2_iface);
48 }
49
PropertyBag_QueryInterface(IPropertyBag2 * iface,REFIID iid,void ** ppv)50 static HRESULT WINAPI PropertyBag_QueryInterface(IPropertyBag2 *iface, REFIID iid,
51 void **ppv)
52 {
53 PropertyBag *This = impl_from_IPropertyBag2(iface);
54 TRACE("(%p,%s,%p)\n", iface, debugstr_guid(iid), ppv);
55
56 if (!ppv) return E_INVALIDARG;
57
58 if (IsEqualIID(&IID_IUnknown, iid) ||
59 IsEqualIID(&IID_IPropertyBag2, iid))
60 {
61 *ppv = &This->IPropertyBag2_iface;
62 }
63 else
64 {
65 *ppv = NULL;
66 return E_NOINTERFACE;
67 }
68
69 IUnknown_AddRef((IUnknown*)*ppv);
70 return S_OK;
71 }
72
PropertyBag_AddRef(IPropertyBag2 * iface)73 static ULONG WINAPI PropertyBag_AddRef(IPropertyBag2 *iface)
74 {
75 PropertyBag *This = impl_from_IPropertyBag2(iface);
76 ULONG ref = InterlockedIncrement(&This->ref);
77
78 TRACE("(%p) refcount=%u\n", iface, ref);
79
80 return ref;
81 }
82
PropertyBag_Release(IPropertyBag2 * iface)83 static ULONG WINAPI PropertyBag_Release(IPropertyBag2 *iface)
84 {
85 PropertyBag *This = impl_from_IPropertyBag2(iface);
86 ULONG ref = InterlockedDecrement(&This->ref);
87
88 TRACE("(%p) refcount=%u\n", iface, ref);
89
90 if (ref == 0)
91 {
92 ULONG i;
93 if (This->properties && This->values)
94 {
95 for (i=0; i < This->prop_count; i++)
96 {
97 CoTaskMemFree(This->properties[i].pstrName);
98 VariantClear( This->values+i );
99 }
100 }
101
102 HeapFree(GetProcessHeap(), 0, This->properties);
103 HeapFree(GetProcessHeap(), 0, This->values);
104 HeapFree(GetProcessHeap(), 0, This);
105 }
106
107 return ref;
108 }
109
find_item(PropertyBag * This,LPCOLESTR name)110 static LONG find_item(PropertyBag *This, LPCOLESTR name)
111 {
112 LONG i;
113 if (!This->properties)
114 return -1;
115 if (!name)
116 return -1;
117
118 for (i=0; i < This->prop_count; i++)
119 {
120 if (strcmpW(name, This->properties[i].pstrName) == 0)
121 return i;
122 }
123
124 return -1;
125 }
126
PropertyBag_Read(IPropertyBag2 * iface,ULONG cProperties,PROPBAG2 * pPropBag,IErrorLog * pErrLog,VARIANT * pvarValue,HRESULT * phrError)127 static HRESULT WINAPI PropertyBag_Read(IPropertyBag2 *iface, ULONG cProperties,
128 PROPBAG2 *pPropBag, IErrorLog *pErrLog, VARIANT *pvarValue, HRESULT *phrError)
129 {
130 HRESULT res = S_OK;
131 ULONG i;
132 PropertyBag *This = impl_from_IPropertyBag2(iface);
133
134 TRACE("(%p,%u,%p,%p,%p,%p)\n", iface, cProperties, pPropBag, pErrLog, pvarValue, phrError);
135
136 for (i=0; i < cProperties; i++)
137 {
138 LONG idx;
139 if (pPropBag[i].dwHint && pPropBag[i].dwHint <= This->prop_count)
140 idx = pPropBag[i].dwHint-1;
141 else
142 idx = find_item(This, pPropBag[i].pstrName);
143
144 if (idx > -1)
145 {
146 VariantInit(pvarValue+i);
147 res = VariantCopy(pvarValue+i, This->values+idx);
148 if (FAILED(res))
149 break;
150 phrError[i] = res;
151 }
152 else
153 {
154 res = E_FAIL;
155 break;
156 }
157 }
158
159 return res;
160 }
161
PropertyBag_Write(IPropertyBag2 * iface,ULONG cProperties,PROPBAG2 * pPropBag,VARIANT * pvarValue)162 static HRESULT WINAPI PropertyBag_Write(IPropertyBag2 *iface, ULONG cProperties,
163 PROPBAG2 *pPropBag, VARIANT *pvarValue)
164 {
165 HRESULT res = S_OK;
166 ULONG i;
167 PropertyBag *This = impl_from_IPropertyBag2(iface);
168
169 TRACE("(%p,%u,%p,%p)\n", iface, cProperties, pPropBag, pvarValue);
170
171 for (i=0; i < cProperties; i++)
172 {
173 LONG idx;
174 if (pPropBag[i].dwHint && pPropBag[i].dwHint <= This->prop_count)
175 idx = pPropBag[i].dwHint-1;
176 else
177 idx = find_item(This, pPropBag[i].pstrName);
178
179 if (idx > -1)
180 {
181 if (This->properties[idx].vt != V_VT(pvarValue+i))
182 return WINCODEC_ERR_PROPERTYUNEXPECTEDTYPE;
183 res = VariantCopy(This->values+idx, pvarValue+i);
184 if (FAILED(res))
185 return E_FAIL;
186 }
187 else
188 {
189 if (pPropBag[i].pstrName)
190 FIXME("Application tried to set the unknown option %s.\n",
191 debugstr_w(pPropBag[i].pstrName));
192
193 /* FIXME: Function is not atomar on error, but MSDN does not say anything about it
194 * (no reset of items between 0 and i-1) */
195 return E_FAIL;
196 }
197 }
198
199 return res;
200 }
201
PropertyBag_CountProperties(IPropertyBag2 * iface,ULONG * pcProperties)202 static HRESULT WINAPI PropertyBag_CountProperties(IPropertyBag2 *iface, ULONG *pcProperties)
203 {
204 PropertyBag *This = impl_from_IPropertyBag2(iface);
205
206 TRACE("(%p,%p)\n", iface, pcProperties);
207
208 if (!pcProperties)
209 return E_INVALIDARG;
210
211 *pcProperties = This->prop_count;
212
213 return S_OK;
214 }
215
copy_propbag2(PROPBAG2 * dest,const PROPBAG2 * src)216 static HRESULT copy_propbag2(PROPBAG2 *dest, const PROPBAG2 *src)
217 {
218 dest->cfType = src->cfType;
219 dest->clsid = src->clsid;
220 dest->dwHint = src->dwHint;
221 dest->dwType = src->dwType;
222 dest->vt = src->vt;
223 dest->pstrName = CoTaskMemAlloc((strlenW(src->pstrName)+1) * sizeof(WCHAR));
224 if(!dest->pstrName)
225 return E_OUTOFMEMORY;
226
227 strcpyW(dest->pstrName, src->pstrName);
228
229 return S_OK;
230 }
231
PropertyBag_GetPropertyInfo(IPropertyBag2 * iface,ULONG iProperty,ULONG cProperties,PROPBAG2 * pPropBag,ULONG * pcProperties)232 static HRESULT WINAPI PropertyBag_GetPropertyInfo(IPropertyBag2 *iface, ULONG iProperty,
233 ULONG cProperties, PROPBAG2 *pPropBag, ULONG *pcProperties)
234 {
235 HRESULT res = S_OK;
236 ULONG i;
237 PropertyBag *This = impl_from_IPropertyBag2(iface);
238
239 TRACE("(%p,%u,%u,%p,%p)\n", iface, iProperty, cProperties, pPropBag, pcProperties);
240
241 if (iProperty >= This->prop_count && iProperty > 0)
242 return WINCODEC_ERR_VALUEOUTOFRANGE;
243 if (iProperty+cProperties > This->prop_count )
244 return WINCODEC_ERR_VALUEOUTOFRANGE;
245
246 *pcProperties = min(cProperties, This->prop_count-iProperty);
247
248 for (i=0; i < *pcProperties; i++)
249 {
250 res = copy_propbag2(pPropBag+i, This->properties+iProperty+i);
251 if (FAILED(res))
252 {
253 do {
254 CoTaskMemFree( pPropBag[--i].pstrName );
255 } while (i);
256 break;
257 }
258 }
259
260 return res;
261 }
262
PropertyBag_LoadObject(IPropertyBag2 * iface,LPCOLESTR pstrName,DWORD dwHint,IUnknown * pUnkObject,IErrorLog * pErrLog)263 static HRESULT WINAPI PropertyBag_LoadObject(IPropertyBag2 *iface, LPCOLESTR pstrName,
264 DWORD dwHint, IUnknown *pUnkObject, IErrorLog *pErrLog)
265 {
266 FIXME("(%p,%s,%u,%p,%p): stub\n", iface, debugstr_w(pstrName), dwHint, pUnkObject, pErrLog);
267 return E_NOTIMPL;
268 }
269
270 static const IPropertyBag2Vtbl PropertyBag_Vtbl = {
271 PropertyBag_QueryInterface,
272 PropertyBag_AddRef,
273 PropertyBag_Release,
274 PropertyBag_Read,
275 PropertyBag_Write,
276 PropertyBag_CountProperties,
277 PropertyBag_GetPropertyInfo,
278 PropertyBag_LoadObject
279 };
280
CreatePropertyBag2(const PROPBAG2 * options,UINT count,IPropertyBag2 ** ppPropertyBag2)281 HRESULT CreatePropertyBag2(const PROPBAG2 *options, UINT count,
282 IPropertyBag2 **ppPropertyBag2)
283 {
284 UINT i;
285 HRESULT res = S_OK;
286 PropertyBag *This;
287
288 This = HeapAlloc(GetProcessHeap(), 0, sizeof(PropertyBag));
289 if (!This) return E_OUTOFMEMORY;
290
291 This->IPropertyBag2_iface.lpVtbl = &PropertyBag_Vtbl;
292 This->ref = 1;
293 This->prop_count = count;
294
295 if (count == 0)
296 {
297 This->properties = NULL;
298 This->values = NULL;
299 }
300 else
301 {
302 This->properties = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(PROPBAG2)*count);
303 This->values = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(VARIANT)*count);
304
305 if (!This->properties || !This->values)
306 res = E_OUTOFMEMORY;
307 else
308 for (i=0; i < count; i++)
309 {
310 res = copy_propbag2(This->properties+i, options+i);
311 if (FAILED(res))
312 break;
313 This->properties[i].dwHint = i+1; /* 0 means unset, so we start with 1 */
314 }
315 }
316
317 if (FAILED(res))
318 {
319 PropertyBag_Release(&This->IPropertyBag2_iface);
320 *ppPropertyBag2 = NULL;
321 }
322 else
323 *ppPropertyBag2 = &This->IPropertyBag2_iface;
324
325 return res;
326 }
327