1 /*
2 * Copyright 2009 Tony Wasserka
3 *
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Lesser General Public
6 * License as published by the Free Software Foundation; either
7 * version 2.1 of the License, or (at your option) any later version.
8 *
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Lesser General Public License for more details.
13 *
14 * You should have received a copy of the GNU Lesser General Public
15 * License along with this library; if not, write to the Free Software
16 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
17 */
18
19 #include "wine/debug.h"
20
21 #define COBJMACROS
22 #include "windef.h"
23 #include "winbase.h"
24 #include "winreg.h"
25 #include "objbase.h"
26 #include "shlwapi.h"
27 #include "wincodecs_private.h"
28
29 WINE_DEFAULT_DEBUG_CHANNEL(wincodecs);
30
31 /******************************************
32 * StreamOnMemory implementation
33 *
34 * Used by IWICStream_InitializeFromMemory
35 *
36 */
37 typedef struct StreamOnMemory {
38 IStream IStream_iface;
39 LONG ref;
40
41 BYTE *pbMemory;
42 DWORD dwMemsize;
43 DWORD dwCurPos;
44
45 CRITICAL_SECTION lock; /* must be held when pbMemory or dwCurPos is accessed */
46 } StreamOnMemory;
47
StreamOnMemory_from_IStream(IStream * iface)48 static inline StreamOnMemory *StreamOnMemory_from_IStream(IStream *iface)
49 {
50 return CONTAINING_RECORD(iface, StreamOnMemory, IStream_iface);
51 }
52
StreamOnMemory_QueryInterface(IStream * iface,REFIID iid,void ** ppv)53 static HRESULT WINAPI StreamOnMemory_QueryInterface(IStream *iface,
54 REFIID iid, void **ppv)
55 {
56 TRACE("(%p,%s,%p)\n", iface, debugstr_guid(iid), ppv);
57
58 if (!ppv) return E_INVALIDARG;
59
60 if (IsEqualIID(&IID_IUnknown, iid) || IsEqualIID(&IID_IStream, iid) ||
61 IsEqualIID(&IID_ISequentialStream, iid))
62 {
63 *ppv = iface;
64 IUnknown_AddRef((IUnknown*)*ppv);
65 return S_OK;
66 }
67 else
68 {
69 *ppv = NULL;
70 return E_NOINTERFACE;
71 }
72 }
73
StreamOnMemory_AddRef(IStream * iface)74 static ULONG WINAPI StreamOnMemory_AddRef(IStream *iface)
75 {
76 StreamOnMemory *This = StreamOnMemory_from_IStream(iface);
77 ULONG ref = InterlockedIncrement(&This->ref);
78
79 TRACE("(%p) refcount=%u\n", iface, ref);
80
81 return ref;
82 }
83
StreamOnMemory_Release(IStream * iface)84 static ULONG WINAPI StreamOnMemory_Release(IStream *iface)
85 {
86 StreamOnMemory *This = StreamOnMemory_from_IStream(iface);
87 ULONG ref = InterlockedDecrement(&This->ref);
88
89 TRACE("(%p) refcount=%u\n", iface, ref);
90
91 if (ref == 0) {
92 This->lock.DebugInfo->Spare[0] = 0;
93 DeleteCriticalSection(&This->lock);
94 HeapFree(GetProcessHeap(), 0, This);
95 }
96 return ref;
97 }
98
StreamOnMemory_Read(IStream * iface,void * pv,ULONG cb,ULONG * pcbRead)99 static HRESULT WINAPI StreamOnMemory_Read(IStream *iface,
100 void *pv, ULONG cb, ULONG *pcbRead)
101 {
102 StreamOnMemory *This = StreamOnMemory_from_IStream(iface);
103 ULONG uBytesRead;
104
105 TRACE("(%p, %p, %u, %p)\n", This, pv, cb, pcbRead);
106
107 if (!pv) return E_INVALIDARG;
108
109 EnterCriticalSection(&This->lock);
110 uBytesRead = min(cb, This->dwMemsize - This->dwCurPos);
111 memmove(pv, This->pbMemory + This->dwCurPos, uBytesRead);
112 This->dwCurPos += uBytesRead;
113 LeaveCriticalSection(&This->lock);
114
115 if (pcbRead) *pcbRead = uBytesRead;
116
117 return S_OK;
118 }
119
StreamOnMemory_Write(IStream * iface,void const * pv,ULONG cb,ULONG * pcbWritten)120 static HRESULT WINAPI StreamOnMemory_Write(IStream *iface,
121 void const *pv, ULONG cb, ULONG *pcbWritten)
122 {
123 StreamOnMemory *This = StreamOnMemory_from_IStream(iface);
124 HRESULT hr;
125
126 TRACE("(%p, %p, %u, %p)\n", This, pv, cb, pcbWritten);
127
128 if (!pv) return E_INVALIDARG;
129
130 EnterCriticalSection(&This->lock);
131 if (cb > This->dwMemsize - This->dwCurPos) {
132 hr = STG_E_MEDIUMFULL;
133 }
134 else {
135 memmove(This->pbMemory + This->dwCurPos, pv, cb);
136 This->dwCurPos += cb;
137 hr = S_OK;
138 if (pcbWritten) *pcbWritten = cb;
139 }
140 LeaveCriticalSection(&This->lock);
141
142 return hr;
143 }
144
StreamOnMemory_Seek(IStream * iface,LARGE_INTEGER dlibMove,DWORD dwOrigin,ULARGE_INTEGER * plibNewPosition)145 static HRESULT WINAPI StreamOnMemory_Seek(IStream *iface,
146 LARGE_INTEGER dlibMove, DWORD dwOrigin, ULARGE_INTEGER *plibNewPosition)
147 {
148 StreamOnMemory *This = StreamOnMemory_from_IStream(iface);
149 LARGE_INTEGER NewPosition;
150 HRESULT hr=S_OK;
151
152 TRACE("(%p, %s, %d, %p)\n", This, wine_dbgstr_longlong(dlibMove.QuadPart), dwOrigin, plibNewPosition);
153
154 EnterCriticalSection(&This->lock);
155 if (dwOrigin == STREAM_SEEK_SET) NewPosition.QuadPart = dlibMove.QuadPart;
156 else if (dwOrigin == STREAM_SEEK_CUR) NewPosition.QuadPart = This->dwCurPos + dlibMove.QuadPart;
157 else if (dwOrigin == STREAM_SEEK_END) NewPosition.QuadPart = This->dwMemsize + dlibMove.QuadPart;
158 else hr = E_INVALIDARG;
159
160 if (SUCCEEDED(hr)) {
161 if (NewPosition.u.HighPart) hr = HRESULT_FROM_WIN32(ERROR_ARITHMETIC_OVERFLOW);
162 else if (NewPosition.QuadPart > This->dwMemsize) hr = E_INVALIDARG;
163 else if (NewPosition.QuadPart < 0) hr = E_INVALIDARG;
164 }
165
166 if (SUCCEEDED(hr)) {
167 This->dwCurPos = NewPosition.u.LowPart;
168
169 if(plibNewPosition) plibNewPosition->QuadPart = This->dwCurPos;
170 }
171 LeaveCriticalSection(&This->lock);
172
173 return hr;
174 }
175
176 /* SetSize isn't implemented in the native windowscodecs DLL either */
StreamOnMemory_SetSize(IStream * iface,ULARGE_INTEGER libNewSize)177 static HRESULT WINAPI StreamOnMemory_SetSize(IStream *iface,
178 ULARGE_INTEGER libNewSize)
179 {
180 TRACE("(%p, %s)\n", iface, wine_dbgstr_longlong(libNewSize.QuadPart));
181 return E_NOTIMPL;
182 }
183
184 /* CopyTo isn't implemented in the native windowscodecs DLL either */
StreamOnMemory_CopyTo(IStream * iface,IStream * pstm,ULARGE_INTEGER cb,ULARGE_INTEGER * pcbRead,ULARGE_INTEGER * pcbWritten)185 static HRESULT WINAPI StreamOnMemory_CopyTo(IStream *iface,
186 IStream *pstm, ULARGE_INTEGER cb, ULARGE_INTEGER *pcbRead, ULARGE_INTEGER *pcbWritten)
187 {
188 TRACE("(%p, %p, %s, %p, %p)\n", iface, pstm, wine_dbgstr_longlong(cb.QuadPart), pcbRead, pcbWritten);
189 return E_NOTIMPL;
190 }
191
192 /* Commit isn't implemented in the native windowscodecs DLL either */
StreamOnMemory_Commit(IStream * iface,DWORD grfCommitFlags)193 static HRESULT WINAPI StreamOnMemory_Commit(IStream *iface,
194 DWORD grfCommitFlags)
195 {
196 TRACE("(%p, %#x)\n", iface, grfCommitFlags);
197 return E_NOTIMPL;
198 }
199
200 /* Revert isn't implemented in the native windowscodecs DLL either */
StreamOnMemory_Revert(IStream * iface)201 static HRESULT WINAPI StreamOnMemory_Revert(IStream *iface)
202 {
203 TRACE("(%p)\n", iface);
204 return E_NOTIMPL;
205 }
206
207 /* LockRegion isn't implemented in the native windowscodecs DLL either */
StreamOnMemory_LockRegion(IStream * iface,ULARGE_INTEGER libOffset,ULARGE_INTEGER cb,DWORD dwLockType)208 static HRESULT WINAPI StreamOnMemory_LockRegion(IStream *iface,
209 ULARGE_INTEGER libOffset, ULARGE_INTEGER cb, DWORD dwLockType)
210 {
211 TRACE("(%p, %s, %s, %d)\n", iface, wine_dbgstr_longlong(libOffset.QuadPart),
212 wine_dbgstr_longlong(cb.QuadPart), dwLockType);
213 return E_NOTIMPL;
214 }
215
216 /* UnlockRegion isn't implemented in the native windowscodecs DLL either */
StreamOnMemory_UnlockRegion(IStream * iface,ULARGE_INTEGER libOffset,ULARGE_INTEGER cb,DWORD dwLockType)217 static HRESULT WINAPI StreamOnMemory_UnlockRegion(IStream *iface,
218 ULARGE_INTEGER libOffset, ULARGE_INTEGER cb, DWORD dwLockType)
219 {
220 TRACE("(%p, %s, %s, %d)\n", iface, wine_dbgstr_longlong(libOffset.QuadPart),
221 wine_dbgstr_longlong(cb.QuadPart), dwLockType);
222 return E_NOTIMPL;
223 }
224
StreamOnMemory_Stat(IStream * iface,STATSTG * pstatstg,DWORD grfStatFlag)225 static HRESULT WINAPI StreamOnMemory_Stat(IStream *iface,
226 STATSTG *pstatstg, DWORD grfStatFlag)
227 {
228 StreamOnMemory *This = StreamOnMemory_from_IStream(iface);
229 TRACE("(%p, %p, %#x)\n", This, pstatstg, grfStatFlag);
230
231 if (!pstatstg) return E_INVALIDARG;
232
233 ZeroMemory(pstatstg, sizeof(STATSTG));
234 pstatstg->type = STGTY_STREAM;
235 pstatstg->cbSize.QuadPart = This->dwMemsize;
236
237 return S_OK;
238 }
239
240 /* Clone isn't implemented in the native windowscodecs DLL either */
StreamOnMemory_Clone(IStream * iface,IStream ** ppstm)241 static HRESULT WINAPI StreamOnMemory_Clone(IStream *iface,
242 IStream **ppstm)
243 {
244 TRACE("(%p, %p)\n", iface, ppstm);
245 return E_NOTIMPL;
246 }
247
248
249 static const IStreamVtbl StreamOnMemory_Vtbl =
250 {
251 /*** IUnknown methods ***/
252 StreamOnMemory_QueryInterface,
253 StreamOnMemory_AddRef,
254 StreamOnMemory_Release,
255 /*** ISequentialStream methods ***/
256 StreamOnMemory_Read,
257 StreamOnMemory_Write,
258 /*** IStream methods ***/
259 StreamOnMemory_Seek,
260 StreamOnMemory_SetSize,
261 StreamOnMemory_CopyTo,
262 StreamOnMemory_Commit,
263 StreamOnMemory_Revert,
264 StreamOnMemory_LockRegion,
265 StreamOnMemory_UnlockRegion,
266 StreamOnMemory_Stat,
267 StreamOnMemory_Clone,
268 };
269
270 /******************************************
271 * StreamOnFileHandle implementation (internal)
272 *
273 */
274 typedef struct StreamOnFileHandle {
275 IStream IStream_iface;
276 LONG ref;
277
278 HANDLE map;
279 void *mem;
280 IWICStream *stream;
281 } StreamOnFileHandle;
282
StreamOnFileHandle_from_IStream(IStream * iface)283 static inline StreamOnFileHandle *StreamOnFileHandle_from_IStream(IStream *iface)
284 {
285 return CONTAINING_RECORD(iface, StreamOnFileHandle, IStream_iface);
286 }
287
StreamOnFileHandle_QueryInterface(IStream * iface,REFIID iid,void ** ppv)288 static HRESULT WINAPI StreamOnFileHandle_QueryInterface(IStream *iface,
289 REFIID iid, void **ppv)
290 {
291 TRACE("(%p,%s,%p)\n", iface, debugstr_guid(iid), ppv);
292
293 if (!ppv) return E_INVALIDARG;
294
295 if (IsEqualIID(&IID_IUnknown, iid) || IsEqualIID(&IID_IStream, iid) ||
296 IsEqualIID(&IID_ISequentialStream, iid))
297 {
298 *ppv = iface;
299 IUnknown_AddRef((IUnknown*)*ppv);
300 return S_OK;
301 }
302 else
303 {
304 *ppv = NULL;
305 return E_NOINTERFACE;
306 }
307 }
308
StreamOnFileHandle_AddRef(IStream * iface)309 static ULONG WINAPI StreamOnFileHandle_AddRef(IStream *iface)
310 {
311 StreamOnFileHandle *This = StreamOnFileHandle_from_IStream(iface);
312 ULONG ref = InterlockedIncrement(&This->ref);
313
314 TRACE("(%p) refcount=%u\n", iface, ref);
315
316 return ref;
317 }
318
StreamOnFileHandle_Release(IStream * iface)319 static ULONG WINAPI StreamOnFileHandle_Release(IStream *iface)
320 {
321 StreamOnFileHandle *This = StreamOnFileHandle_from_IStream(iface);
322 ULONG ref = InterlockedDecrement(&This->ref);
323
324 TRACE("(%p) refcount=%u\n", iface, ref);
325
326 if (ref == 0) {
327 IWICStream_Release(This->stream);
328 UnmapViewOfFile(This->mem);
329 CloseHandle(This->map);
330 HeapFree(GetProcessHeap(), 0, This);
331 }
332 return ref;
333 }
334
StreamOnFileHandle_Read(IStream * iface,void * pv,ULONG cb,ULONG * pcbRead)335 static HRESULT WINAPI StreamOnFileHandle_Read(IStream *iface,
336 void *pv, ULONG cb, ULONG *pcbRead)
337 {
338 StreamOnFileHandle *This = StreamOnFileHandle_from_IStream(iface);
339 TRACE("(%p, %p, %u, %p)\n", This, pv, cb, pcbRead);
340
341 return IWICStream_Read(This->stream, pv, cb, pcbRead);
342 }
343
StreamOnFileHandle_Write(IStream * iface,void const * pv,ULONG cb,ULONG * pcbWritten)344 static HRESULT WINAPI StreamOnFileHandle_Write(IStream *iface,
345 void const *pv, ULONG cb, ULONG *pcbWritten)
346 {
347 ERR("(%p, %p, %u, %p)\n", iface, pv, cb, pcbWritten);
348 return HRESULT_FROM_WIN32(ERROR_ACCESS_DENIED);
349 }
350
StreamOnFileHandle_Seek(IStream * iface,LARGE_INTEGER dlibMove,DWORD dwOrigin,ULARGE_INTEGER * plibNewPosition)351 static HRESULT WINAPI StreamOnFileHandle_Seek(IStream *iface,
352 LARGE_INTEGER dlibMove, DWORD dwOrigin, ULARGE_INTEGER *plibNewPosition)
353 {
354 StreamOnFileHandle *This = StreamOnFileHandle_from_IStream(iface);
355 TRACE("(%p, %s, %d, %p)\n", This, wine_dbgstr_longlong(dlibMove.QuadPart), dwOrigin, plibNewPosition);
356
357 return IWICStream_Seek(This->stream, dlibMove, dwOrigin, plibNewPosition);
358 }
359
StreamOnFileHandle_SetSize(IStream * iface,ULARGE_INTEGER libNewSize)360 static HRESULT WINAPI StreamOnFileHandle_SetSize(IStream *iface,
361 ULARGE_INTEGER libNewSize)
362 {
363 TRACE("(%p, %s)\n", iface, wine_dbgstr_longlong(libNewSize.QuadPart));
364 return E_NOTIMPL;
365 }
366
StreamOnFileHandle_CopyTo(IStream * iface,IStream * pstm,ULARGE_INTEGER cb,ULARGE_INTEGER * pcbRead,ULARGE_INTEGER * pcbWritten)367 static HRESULT WINAPI StreamOnFileHandle_CopyTo(IStream *iface,
368 IStream *pstm, ULARGE_INTEGER cb, ULARGE_INTEGER *pcbRead, ULARGE_INTEGER *pcbWritten)
369 {
370 TRACE("(%p, %p, %s, %p, %p)\n", iface, pstm, wine_dbgstr_longlong(cb.QuadPart), pcbRead, pcbWritten);
371 return E_NOTIMPL;
372 }
373
StreamOnFileHandle_Commit(IStream * iface,DWORD grfCommitFlags)374 static HRESULT WINAPI StreamOnFileHandle_Commit(IStream *iface,
375 DWORD grfCommitFlags)
376 {
377 TRACE("(%p, %#x)\n", iface, grfCommitFlags);
378 return E_NOTIMPL;
379 }
380
StreamOnFileHandle_Revert(IStream * iface)381 static HRESULT WINAPI StreamOnFileHandle_Revert(IStream *iface)
382 {
383 TRACE("(%p)\n", iface);
384 return E_NOTIMPL;
385 }
386
StreamOnFileHandle_LockRegion(IStream * iface,ULARGE_INTEGER libOffset,ULARGE_INTEGER cb,DWORD dwLockType)387 static HRESULT WINAPI StreamOnFileHandle_LockRegion(IStream *iface,
388 ULARGE_INTEGER libOffset, ULARGE_INTEGER cb, DWORD dwLockType)
389 {
390 TRACE("(%p, %s, %s, %d)\n", iface, wine_dbgstr_longlong(libOffset.QuadPart),
391 wine_dbgstr_longlong(cb.QuadPart), dwLockType);
392 return E_NOTIMPL;
393 }
394
StreamOnFileHandle_UnlockRegion(IStream * iface,ULARGE_INTEGER libOffset,ULARGE_INTEGER cb,DWORD dwLockType)395 static HRESULT WINAPI StreamOnFileHandle_UnlockRegion(IStream *iface,
396 ULARGE_INTEGER libOffset, ULARGE_INTEGER cb, DWORD dwLockType)
397 {
398 TRACE("(%p, %s, %s, %d)\n", iface, wine_dbgstr_longlong(libOffset.QuadPart),
399 wine_dbgstr_longlong(cb.QuadPart), dwLockType);
400 return E_NOTIMPL;
401 }
402
StreamOnFileHandle_Stat(IStream * iface,STATSTG * pstatstg,DWORD grfStatFlag)403 static HRESULT WINAPI StreamOnFileHandle_Stat(IStream *iface,
404 STATSTG *pstatstg, DWORD grfStatFlag)
405 {
406 StreamOnFileHandle *This = StreamOnFileHandle_from_IStream(iface);
407 TRACE("(%p, %p, %#x)\n", This, pstatstg, grfStatFlag);
408
409 return IWICStream_Stat(This->stream, pstatstg, grfStatFlag);
410 }
411
StreamOnFileHandle_Clone(IStream * iface,IStream ** ppstm)412 static HRESULT WINAPI StreamOnFileHandle_Clone(IStream *iface,
413 IStream **ppstm)
414 {
415 TRACE("(%p, %p)\n", iface, ppstm);
416 return E_NOTIMPL;
417 }
418
419 static const IStreamVtbl StreamOnFileHandle_Vtbl =
420 {
421 /*** IUnknown methods ***/
422 StreamOnFileHandle_QueryInterface,
423 StreamOnFileHandle_AddRef,
424 StreamOnFileHandle_Release,
425 /*** ISequentialStream methods ***/
426 StreamOnFileHandle_Read,
427 StreamOnFileHandle_Write,
428 /*** IStream methods ***/
429 StreamOnFileHandle_Seek,
430 StreamOnFileHandle_SetSize,
431 StreamOnFileHandle_CopyTo,
432 StreamOnFileHandle_Commit,
433 StreamOnFileHandle_Revert,
434 StreamOnFileHandle_LockRegion,
435 StreamOnFileHandle_UnlockRegion,
436 StreamOnFileHandle_Stat,
437 StreamOnFileHandle_Clone,
438 };
439
440 /******************************************
441 * StreamOnStreamRange implementation
442 *
443 * Used by IWICStream_InitializeFromIStreamRegion
444 *
445 */
446 typedef struct StreamOnStreamRange {
447 IStream IStream_iface;
448 LONG ref;
449
450 IStream *stream;
451 ULARGE_INTEGER pos;
452 ULARGE_INTEGER offset;
453 ULARGE_INTEGER max_size;
454
455 CRITICAL_SECTION lock;
456 } StreamOnStreamRange;
457
StreamOnStreamRange_from_IStream(IStream * iface)458 static inline StreamOnStreamRange *StreamOnStreamRange_from_IStream(IStream *iface)
459 {
460 return CONTAINING_RECORD(iface, StreamOnStreamRange, IStream_iface);
461 }
462
StreamOnStreamRange_QueryInterface(IStream * iface,REFIID iid,void ** ppv)463 static HRESULT WINAPI StreamOnStreamRange_QueryInterface(IStream *iface,
464 REFIID iid, void **ppv)
465 {
466 TRACE("(%p,%s,%p)\n", iface, debugstr_guid(iid), ppv);
467
468 if (!ppv) return E_INVALIDARG;
469
470 if (IsEqualIID(&IID_IUnknown, iid) || IsEqualIID(&IID_IStream, iid) ||
471 IsEqualIID(&IID_ISequentialStream, iid))
472 {
473 *ppv = iface;
474 IUnknown_AddRef((IUnknown*)*ppv);
475 return S_OK;
476 }
477 else
478 {
479 *ppv = NULL;
480 return E_NOINTERFACE;
481 }
482 }
483
StreamOnStreamRange_AddRef(IStream * iface)484 static ULONG WINAPI StreamOnStreamRange_AddRef(IStream *iface)
485 {
486 StreamOnStreamRange *This = StreamOnStreamRange_from_IStream(iface);
487 ULONG ref = InterlockedIncrement(&This->ref);
488
489 TRACE("(%p) refcount=%u\n", iface, ref);
490
491 return ref;
492 }
493
StreamOnStreamRange_Release(IStream * iface)494 static ULONG WINAPI StreamOnStreamRange_Release(IStream *iface)
495 {
496 StreamOnStreamRange *This = StreamOnStreamRange_from_IStream(iface);
497 ULONG ref = InterlockedDecrement(&This->ref);
498
499 TRACE("(%p) refcount=%u\n", iface, ref);
500
501 if (ref == 0) {
502 This->lock.DebugInfo->Spare[0] = 0;
503 DeleteCriticalSection(&This->lock);
504 IStream_Release(This->stream);
505 HeapFree(GetProcessHeap(), 0, This);
506 }
507 return ref;
508 }
509
StreamOnStreamRange_Read(IStream * iface,void * pv,ULONG cb,ULONG * pcbRead)510 static HRESULT WINAPI StreamOnStreamRange_Read(IStream *iface,
511 void *pv, ULONG cb, ULONG *pcbRead)
512 {
513 StreamOnStreamRange *This = StreamOnStreamRange_from_IStream(iface);
514 ULONG uBytesRead=0;
515 HRESULT hr;
516 ULARGE_INTEGER OldPosition;
517 LARGE_INTEGER SetPosition;
518
519 TRACE("(%p, %p, %u, %p)\n", This, pv, cb, pcbRead);
520
521 if (!pv) return E_INVALIDARG;
522
523 EnterCriticalSection(&This->lock);
524 SetPosition.QuadPart = 0;
525 hr = IStream_Seek(This->stream, SetPosition, STREAM_SEEK_CUR, &OldPosition);
526 if (SUCCEEDED(hr))
527 {
528 SetPosition.QuadPart = This->pos.QuadPart + This->offset.QuadPart;
529 hr = IStream_Seek(This->stream, SetPosition, STREAM_SEEK_SET, NULL);
530 }
531 if (SUCCEEDED(hr))
532 {
533 if (This->pos.QuadPart + cb > This->max_size.QuadPart)
534 {
535 /* This would read past the end of the stream. */
536 if (This->pos.QuadPart > This->max_size.QuadPart)
537 cb = 0;
538 else
539 cb = This->max_size.QuadPart - This->pos.QuadPart;
540 }
541 hr = IStream_Read(This->stream, pv, cb, &uBytesRead);
542 SetPosition.QuadPart = OldPosition.QuadPart;
543 IStream_Seek(This->stream, SetPosition, STREAM_SEEK_SET, NULL);
544 }
545 if (SUCCEEDED(hr))
546 This->pos.QuadPart += uBytesRead;
547 LeaveCriticalSection(&This->lock);
548
549 if (SUCCEEDED(hr) && pcbRead) *pcbRead = uBytesRead;
550
551 return hr;
552 }
553
StreamOnStreamRange_Write(IStream * iface,void const * pv,ULONG cb,ULONG * pcbWritten)554 static HRESULT WINAPI StreamOnStreamRange_Write(IStream *iface,
555 void const *pv, ULONG cb, ULONG *pcbWritten)
556 {
557 StreamOnStreamRange *This = StreamOnStreamRange_from_IStream(iface);
558 HRESULT hr;
559 ULARGE_INTEGER OldPosition;
560 LARGE_INTEGER SetPosition;
561 ULONG uBytesWritten=0;
562 TRACE("(%p, %p, %u, %p)\n", This, pv, cb, pcbWritten);
563
564 if (!pv) return E_INVALIDARG;
565
566 EnterCriticalSection(&This->lock);
567 SetPosition.QuadPart = 0;
568 hr = IStream_Seek(This->stream, SetPosition, STREAM_SEEK_CUR, &OldPosition);
569 if (SUCCEEDED(hr))
570 {
571 SetPosition.QuadPart = This->pos.QuadPart + This->offset.QuadPart;
572 hr = IStream_Seek(This->stream, SetPosition, STREAM_SEEK_SET, NULL);
573 }
574 if (SUCCEEDED(hr))
575 {
576 if (This->pos.QuadPart + cb > This->max_size.QuadPart)
577 {
578 /* This would read past the end of the stream. */
579 if (This->pos.QuadPart > This->max_size.QuadPart)
580 cb = 0;
581 else
582 cb = This->max_size.QuadPart - This->pos.QuadPart;
583 }
584 hr = IStream_Write(This->stream, pv, cb, &uBytesWritten);
585 SetPosition.QuadPart = OldPosition.QuadPart;
586 IStream_Seek(This->stream, SetPosition, STREAM_SEEK_SET, NULL);
587 }
588 if (SUCCEEDED(hr))
589 This->pos.QuadPart += uBytesWritten;
590 LeaveCriticalSection(&This->lock);
591
592 if (SUCCEEDED(hr) && pcbWritten) *pcbWritten = uBytesWritten;
593
594 return hr;
595 }
596
StreamOnStreamRange_Seek(IStream * iface,LARGE_INTEGER dlibMove,DWORD dwOrigin,ULARGE_INTEGER * plibNewPosition)597 static HRESULT WINAPI StreamOnStreamRange_Seek(IStream *iface,
598 LARGE_INTEGER dlibMove, DWORD dwOrigin, ULARGE_INTEGER *plibNewPosition)
599 {
600 StreamOnStreamRange *This = StreamOnStreamRange_from_IStream(iface);
601 ULARGE_INTEGER NewPosition, actual_size;
602 HRESULT hr=S_OK;
603 STATSTG statstg;
604 TRACE("(%p, %s, %d, %p)\n", This, wine_dbgstr_longlong(dlibMove.QuadPart), dwOrigin, plibNewPosition);
605
606 EnterCriticalSection(&This->lock);
607 actual_size = This->max_size;
608 if (dwOrigin == STREAM_SEEK_SET)
609 NewPosition.QuadPart = dlibMove.QuadPart;
610 else if (dwOrigin == STREAM_SEEK_CUR)
611 NewPosition.QuadPart = This->pos.QuadPart + dlibMove.QuadPart;
612 else if (dwOrigin == STREAM_SEEK_END)
613 {
614 hr = IStream_Stat(This->stream, &statstg, STATFLAG_NONAME);
615 if (SUCCEEDED(hr))
616 {
617 if (This->max_size.QuadPart + This->offset.QuadPart > statstg.cbSize.QuadPart)
618 actual_size.QuadPart = statstg.cbSize.QuadPart - This->offset.QuadPart;
619 NewPosition.QuadPart = dlibMove.QuadPart + actual_size.QuadPart;
620 }
621 }
622 else hr = E_INVALIDARG;
623
624 if (SUCCEEDED(hr) && (NewPosition.u.HighPart != 0 || NewPosition.QuadPart > actual_size.QuadPart))
625 hr = WINCODEC_ERR_VALUEOUTOFRANGE;
626
627 if (SUCCEEDED(hr)) {
628 This->pos.QuadPart = NewPosition.QuadPart;
629
630 if(plibNewPosition) plibNewPosition->QuadPart = This->pos.QuadPart;
631 }
632 LeaveCriticalSection(&This->lock);
633
634 return hr;
635 }
636
637 /* SetSize isn't implemented in the native windowscodecs DLL either */
StreamOnStreamRange_SetSize(IStream * iface,ULARGE_INTEGER libNewSize)638 static HRESULT WINAPI StreamOnStreamRange_SetSize(IStream *iface,
639 ULARGE_INTEGER libNewSize)
640 {
641 TRACE("(%p, %s)\n", iface, wine_dbgstr_longlong(libNewSize.QuadPart));
642 return E_NOTIMPL;
643 }
644
645 /* CopyTo isn't implemented in the native windowscodecs DLL either */
StreamOnStreamRange_CopyTo(IStream * iface,IStream * pstm,ULARGE_INTEGER cb,ULARGE_INTEGER * pcbRead,ULARGE_INTEGER * pcbWritten)646 static HRESULT WINAPI StreamOnStreamRange_CopyTo(IStream *iface,
647 IStream *pstm, ULARGE_INTEGER cb, ULARGE_INTEGER *pcbRead, ULARGE_INTEGER *pcbWritten)
648 {
649 TRACE("(%p, %p, %s, %p, %p)\n", iface, pstm, wine_dbgstr_longlong(cb.QuadPart),
650 pcbRead, pcbWritten);
651 return E_NOTIMPL;
652 }
653
654 /* Commit isn't implemented in the native windowscodecs DLL either */
StreamOnStreamRange_Commit(IStream * iface,DWORD grfCommitFlags)655 static HRESULT WINAPI StreamOnStreamRange_Commit(IStream *iface,
656 DWORD grfCommitFlags)
657 {
658 TRACE("(%p, %#x)\n", iface, grfCommitFlags);
659 return E_NOTIMPL;
660 }
661
662 /* Revert isn't implemented in the native windowscodecs DLL either */
StreamOnStreamRange_Revert(IStream * iface)663 static HRESULT WINAPI StreamOnStreamRange_Revert(IStream *iface)
664 {
665 TRACE("(%p)\n", iface);
666 return E_NOTIMPL;
667 }
668
669 /* LockRegion isn't implemented in the native windowscodecs DLL either */
StreamOnStreamRange_LockRegion(IStream * iface,ULARGE_INTEGER libOffset,ULARGE_INTEGER cb,DWORD dwLockType)670 static HRESULT WINAPI StreamOnStreamRange_LockRegion(IStream *iface,
671 ULARGE_INTEGER libOffset, ULARGE_INTEGER cb, DWORD dwLockType)
672 {
673 TRACE("(%p, %s, %s, %d)\n", iface, wine_dbgstr_longlong(libOffset.QuadPart),
674 wine_dbgstr_longlong(cb.QuadPart), dwLockType);
675 return E_NOTIMPL;
676 }
677
678 /* UnlockRegion isn't implemented in the native windowscodecs DLL either */
StreamOnStreamRange_UnlockRegion(IStream * iface,ULARGE_INTEGER libOffset,ULARGE_INTEGER cb,DWORD dwLockType)679 static HRESULT WINAPI StreamOnStreamRange_UnlockRegion(IStream *iface,
680 ULARGE_INTEGER libOffset, ULARGE_INTEGER cb, DWORD dwLockType)
681 {
682 TRACE("(%p, %s, %s, %d)\n", iface, wine_dbgstr_longlong(libOffset.QuadPart),
683 wine_dbgstr_longlong(cb.QuadPart), dwLockType);
684 return E_NOTIMPL;
685 }
686
StreamOnStreamRange_Stat(IStream * iface,STATSTG * pstatstg,DWORD grfStatFlag)687 static HRESULT WINAPI StreamOnStreamRange_Stat(IStream *iface,
688 STATSTG *pstatstg, DWORD grfStatFlag)
689 {
690 StreamOnStreamRange *This = StreamOnStreamRange_from_IStream(iface);
691 HRESULT hr;
692 TRACE("(%p, %p, %#x)\n", This, pstatstg, grfStatFlag);
693
694 if (!pstatstg) return E_INVALIDARG;
695
696 EnterCriticalSection(&This->lock);
697 hr = IStream_Stat(This->stream, pstatstg, grfStatFlag);
698 if (SUCCEEDED(hr))
699 {
700 pstatstg->cbSize.QuadPart -= This->offset.QuadPart;
701 if (This->max_size.QuadPart < pstatstg->cbSize.QuadPart)
702 pstatstg->cbSize.QuadPart = This->max_size.QuadPart;
703 }
704
705 LeaveCriticalSection(&This->lock);
706
707 return hr;
708 }
709
710 /* Clone isn't implemented in the native windowscodecs DLL either */
StreamOnStreamRange_Clone(IStream * iface,IStream ** ppstm)711 static HRESULT WINAPI StreamOnStreamRange_Clone(IStream *iface,
712 IStream **ppstm)
713 {
714 TRACE("(%p, %p)\n", iface, ppstm);
715 return E_NOTIMPL;
716 }
717
718 static const IStreamVtbl StreamOnStreamRange_Vtbl =
719 {
720 /*** IUnknown methods ***/
721 StreamOnStreamRange_QueryInterface,
722 StreamOnStreamRange_AddRef,
723 StreamOnStreamRange_Release,
724 /*** ISequentialStream methods ***/
725 StreamOnStreamRange_Read,
726 StreamOnStreamRange_Write,
727 /*** IStream methods ***/
728 StreamOnStreamRange_Seek,
729 StreamOnStreamRange_SetSize,
730 StreamOnStreamRange_CopyTo,
731 StreamOnStreamRange_Commit,
732 StreamOnStreamRange_Revert,
733 StreamOnStreamRange_LockRegion,
734 StreamOnStreamRange_UnlockRegion,
735 StreamOnStreamRange_Stat,
736 StreamOnStreamRange_Clone,
737 };
738
739
740 /******************************************
741 * IWICStream implementation
742 *
743 */
744 typedef struct IWICStreamImpl
745 {
746 IWICStream IWICStream_iface;
747 LONG ref;
748
749 IStream *pStream;
750 } IWICStreamImpl;
751
impl_from_IWICStream(IWICStream * iface)752 static inline IWICStreamImpl *impl_from_IWICStream(IWICStream *iface)
753 {
754 return CONTAINING_RECORD(iface, IWICStreamImpl, IWICStream_iface);
755 }
756
IWICStreamImpl_QueryInterface(IWICStream * iface,REFIID iid,void ** ppv)757 static HRESULT WINAPI IWICStreamImpl_QueryInterface(IWICStream *iface,
758 REFIID iid, void **ppv)
759 {
760 IWICStreamImpl *This = impl_from_IWICStream(iface);
761 TRACE("(%p,%s,%p)\n", iface, debugstr_guid(iid), ppv);
762
763 if (!ppv) return E_INVALIDARG;
764
765 if (IsEqualIID(&IID_IUnknown, iid) || IsEqualIID(&IID_IStream, iid) ||
766 IsEqualIID(&IID_ISequentialStream, iid) || IsEqualIID(&IID_IWICStream, iid))
767 {
768 *ppv = &This->IWICStream_iface;
769 IUnknown_AddRef((IUnknown*)*ppv);
770 return S_OK;
771 }
772 else
773 {
774 *ppv = NULL;
775 return E_NOINTERFACE;
776 }
777 }
778
IWICStreamImpl_AddRef(IWICStream * iface)779 static ULONG WINAPI IWICStreamImpl_AddRef(IWICStream *iface)
780 {
781 IWICStreamImpl *This = impl_from_IWICStream(iface);
782 ULONG ref = InterlockedIncrement(&This->ref);
783
784 TRACE("(%p) refcount=%u\n", iface, ref);
785
786 return ref;
787 }
788
IWICStreamImpl_Release(IWICStream * iface)789 static ULONG WINAPI IWICStreamImpl_Release(IWICStream *iface)
790 {
791 IWICStreamImpl *This = impl_from_IWICStream(iface);
792 ULONG ref = InterlockedDecrement(&This->ref);
793
794 TRACE("(%p) refcount=%u\n", iface, ref);
795
796 if (ref == 0) {
797 if (This->pStream) IStream_Release(This->pStream);
798 HeapFree(GetProcessHeap(), 0, This);
799 }
800 return ref;
801 }
802
IWICStreamImpl_Read(IWICStream * iface,void * pv,ULONG cb,ULONG * pcbRead)803 static HRESULT WINAPI IWICStreamImpl_Read(IWICStream *iface,
804 void *pv, ULONG cb, ULONG *pcbRead)
805 {
806 IWICStreamImpl *This = impl_from_IWICStream(iface);
807 TRACE("(%p, %p, %u, %p)\n", This, pv, cb, pcbRead);
808
809 if (!This->pStream) return WINCODEC_ERR_NOTINITIALIZED;
810 return IStream_Read(This->pStream, pv, cb, pcbRead);
811 }
812
IWICStreamImpl_Write(IWICStream * iface,void const * pv,ULONG cb,ULONG * pcbWritten)813 static HRESULT WINAPI IWICStreamImpl_Write(IWICStream *iface,
814 void const *pv, ULONG cb, ULONG *pcbWritten)
815 {
816 IWICStreamImpl *This = impl_from_IWICStream(iface);
817 TRACE("(%p, %p, %u, %p)\n", This, pv, cb, pcbWritten);
818
819 if (!This->pStream) return WINCODEC_ERR_NOTINITIALIZED;
820 return IStream_Write(This->pStream, pv, cb, pcbWritten);
821 }
822
IWICStreamImpl_Seek(IWICStream * iface,LARGE_INTEGER dlibMove,DWORD dwOrigin,ULARGE_INTEGER * plibNewPosition)823 static HRESULT WINAPI IWICStreamImpl_Seek(IWICStream *iface,
824 LARGE_INTEGER dlibMove, DWORD dwOrigin, ULARGE_INTEGER *plibNewPosition)
825 {
826 IWICStreamImpl *This = impl_from_IWICStream(iface);
827 TRACE("(%p, %s, %d, %p)\n", This, wine_dbgstr_longlong(dlibMove.QuadPart),
828 dwOrigin, plibNewPosition);
829
830 if (!This->pStream) return WINCODEC_ERR_NOTINITIALIZED;
831 return IStream_Seek(This->pStream, dlibMove, dwOrigin, plibNewPosition);
832 }
833
IWICStreamImpl_SetSize(IWICStream * iface,ULARGE_INTEGER libNewSize)834 static HRESULT WINAPI IWICStreamImpl_SetSize(IWICStream *iface,
835 ULARGE_INTEGER libNewSize)
836 {
837 IWICStreamImpl *This = impl_from_IWICStream(iface);
838 TRACE("(%p, %s)\n", This, wine_dbgstr_longlong(libNewSize.QuadPart));
839
840 if (!This->pStream) return WINCODEC_ERR_NOTINITIALIZED;
841 return IStream_SetSize(This->pStream, libNewSize);
842 }
843
IWICStreamImpl_CopyTo(IWICStream * iface,IStream * pstm,ULARGE_INTEGER cb,ULARGE_INTEGER * pcbRead,ULARGE_INTEGER * pcbWritten)844 static HRESULT WINAPI IWICStreamImpl_CopyTo(IWICStream *iface,
845 IStream *pstm, ULARGE_INTEGER cb, ULARGE_INTEGER *pcbRead, ULARGE_INTEGER *pcbWritten)
846 {
847 IWICStreamImpl *This = impl_from_IWICStream(iface);
848 TRACE("(%p, %p, %s, %p, %p)\n", This, pstm, wine_dbgstr_longlong(cb.QuadPart), pcbRead, pcbWritten);
849
850 if (!This->pStream) return WINCODEC_ERR_NOTINITIALIZED;
851 return IStream_CopyTo(This->pStream, pstm, cb, pcbRead, pcbWritten);
852 }
853
IWICStreamImpl_Commit(IWICStream * iface,DWORD grfCommitFlags)854 static HRESULT WINAPI IWICStreamImpl_Commit(IWICStream *iface,
855 DWORD grfCommitFlags)
856 {
857 IWICStreamImpl *This = impl_from_IWICStream(iface);
858 TRACE("(%p, %#x)\n", This, grfCommitFlags);
859
860 if (!This->pStream) return WINCODEC_ERR_NOTINITIALIZED;
861 return IStream_Commit(This->pStream, grfCommitFlags);
862 }
863
IWICStreamImpl_Revert(IWICStream * iface)864 static HRESULT WINAPI IWICStreamImpl_Revert(IWICStream *iface)
865 {
866 IWICStreamImpl *This = impl_from_IWICStream(iface);
867 TRACE("(%p)\n", This);
868
869 if (!This->pStream) return WINCODEC_ERR_NOTINITIALIZED;
870 return IStream_Revert(This->pStream);
871 }
872
IWICStreamImpl_LockRegion(IWICStream * iface,ULARGE_INTEGER libOffset,ULARGE_INTEGER cb,DWORD dwLockType)873 static HRESULT WINAPI IWICStreamImpl_LockRegion(IWICStream *iface,
874 ULARGE_INTEGER libOffset, ULARGE_INTEGER cb, DWORD dwLockType)
875 {
876 IWICStreamImpl *This = impl_from_IWICStream(iface);
877 TRACE("(%p, %s, %s, %d)\n", This, wine_dbgstr_longlong(libOffset.QuadPart),
878 wine_dbgstr_longlong(cb.QuadPart), dwLockType);
879
880 if (!This->pStream) return WINCODEC_ERR_NOTINITIALIZED;
881 return IStream_LockRegion(This->pStream, libOffset, cb, dwLockType);
882 }
883
IWICStreamImpl_UnlockRegion(IWICStream * iface,ULARGE_INTEGER libOffset,ULARGE_INTEGER cb,DWORD dwLockType)884 static HRESULT WINAPI IWICStreamImpl_UnlockRegion(IWICStream *iface,
885 ULARGE_INTEGER libOffset, ULARGE_INTEGER cb, DWORD dwLockType)
886 {
887 IWICStreamImpl *This = impl_from_IWICStream(iface);
888 TRACE("(%p, %s, %s, %d)\n", This, wine_dbgstr_longlong(libOffset.QuadPart),
889 wine_dbgstr_longlong(cb.QuadPart), dwLockType);
890
891 if (!This->pStream) return WINCODEC_ERR_NOTINITIALIZED;
892 return IStream_UnlockRegion(This->pStream, libOffset, cb, dwLockType);
893 }
894
IWICStreamImpl_Stat(IWICStream * iface,STATSTG * pstatstg,DWORD grfStatFlag)895 static HRESULT WINAPI IWICStreamImpl_Stat(IWICStream *iface,
896 STATSTG *pstatstg, DWORD grfStatFlag)
897 {
898 IWICStreamImpl *This = impl_from_IWICStream(iface);
899 TRACE("(%p, %p, %#x)\n", This, pstatstg, grfStatFlag);
900
901 if (!This->pStream) return WINCODEC_ERR_NOTINITIALIZED;
902 return IStream_Stat(This->pStream, pstatstg, grfStatFlag);
903 }
904
IWICStreamImpl_Clone(IWICStream * iface,IStream ** ppstm)905 static HRESULT WINAPI IWICStreamImpl_Clone(IWICStream *iface,
906 IStream **ppstm)
907 {
908 IWICStreamImpl *This = impl_from_IWICStream(iface);
909 TRACE("(%p, %p)\n", This, ppstm);
910
911 if (!This->pStream) return WINCODEC_ERR_NOTINITIALIZED;
912 return IStream_Clone(This->pStream, ppstm);
913 }
914
IWICStreamImpl_InitializeFromIStream(IWICStream * iface,IStream * stream)915 static HRESULT WINAPI IWICStreamImpl_InitializeFromIStream(IWICStream *iface, IStream *stream)
916 {
917 IWICStreamImpl *This = impl_from_IWICStream(iface);
918 HRESULT hr = S_OK;
919
920 TRACE("(%p, %p)\n", iface, stream);
921
922 if (!stream) return E_INVALIDARG;
923 if (This->pStream) return WINCODEC_ERR_WRONGSTATE;
924
925 IStream_AddRef(stream);
926
927 if (InterlockedCompareExchangePointer((void **)&This->pStream, stream, NULL))
928 {
929 /* Some other thread set the stream first. */
930 IStream_Release(stream);
931 hr = WINCODEC_ERR_WRONGSTATE;
932 }
933
934 return hr;
935 }
936
IWICStreamImpl_InitializeFromFilename(IWICStream * iface,LPCWSTR wzFileName,DWORD dwDesiredAccess)937 static HRESULT WINAPI IWICStreamImpl_InitializeFromFilename(IWICStream *iface,
938 LPCWSTR wzFileName, DWORD dwDesiredAccess)
939 {
940 IWICStreamImpl *This = impl_from_IWICStream(iface);
941 HRESULT hr;
942 DWORD dwMode;
943 IStream *stream;
944
945 TRACE("(%p, %s, %u)\n", iface, debugstr_w(wzFileName), dwDesiredAccess);
946
947 if (This->pStream) return WINCODEC_ERR_WRONGSTATE;
948
949 if(dwDesiredAccess & GENERIC_WRITE)
950 dwMode = STGM_SHARE_DENY_WRITE | STGM_WRITE | STGM_CREATE;
951 else if(dwDesiredAccess & GENERIC_READ)
952 dwMode = STGM_SHARE_DENY_WRITE | STGM_READ | STGM_FAILIFTHERE;
953 else
954 return E_INVALIDARG;
955
956 hr = SHCreateStreamOnFileW(wzFileName, dwMode, &stream);
957
958 if (SUCCEEDED(hr))
959 {
960 if (InterlockedCompareExchangePointer((void**)&This->pStream, stream, NULL))
961 {
962 /* Some other thread set the stream first. */
963 IStream_Release(stream);
964 hr = WINCODEC_ERR_WRONGSTATE;
965 }
966 }
967
968 return hr;
969 }
970
971 /******************************************
972 * IWICStream_InitializeFromMemory
973 *
974 * Initializes the internal IStream object to retrieve its data from a memory chunk.
975 *
976 * PARAMS
977 * pbBuffer [I] pointer to the memory chunk
978 * cbBufferSize [I] number of bytes to use from the memory chunk
979 *
980 * RETURNS
981 * SUCCESS: S_OK
982 * FAILURE: E_INVALIDARG, if pbBuffer is NULL
983 * E_OUTOFMEMORY, if we run out of memory
984 * WINCODEC_ERR_WRONGSTATE, if the IStream object has already been initialized before
985 *
986 */
IWICStreamImpl_InitializeFromMemory(IWICStream * iface,BYTE * pbBuffer,DWORD cbBufferSize)987 static HRESULT WINAPI IWICStreamImpl_InitializeFromMemory(IWICStream *iface,
988 BYTE *pbBuffer, DWORD cbBufferSize)
989 {
990 IWICStreamImpl *This = impl_from_IWICStream(iface);
991 StreamOnMemory *pObject;
992 TRACE("(%p, %p, %u)\n", iface, pbBuffer, cbBufferSize);
993
994 if (!pbBuffer) return E_INVALIDARG;
995 if (This->pStream) return WINCODEC_ERR_WRONGSTATE;
996
997 pObject = HeapAlloc(GetProcessHeap(), 0, sizeof(StreamOnMemory));
998 if (!pObject) return E_OUTOFMEMORY;
999
1000 pObject->IStream_iface.lpVtbl = &StreamOnMemory_Vtbl;
1001 pObject->ref = 1;
1002 pObject->pbMemory = pbBuffer;
1003 pObject->dwMemsize = cbBufferSize;
1004 pObject->dwCurPos = 0;
1005 InitializeCriticalSection(&pObject->lock);
1006 pObject->lock.DebugInfo->Spare[0] = (DWORD_PTR)(__FILE__ ": StreamOnMemory.lock");
1007
1008 if (InterlockedCompareExchangePointer((void**)&This->pStream, pObject, NULL))
1009 {
1010 /* Some other thread set the stream first. */
1011 IStream_Release(&pObject->IStream_iface);
1012 return WINCODEC_ERR_WRONGSTATE;
1013 }
1014
1015 return S_OK;
1016 }
1017
map_file(HANDLE file,HANDLE * map,void ** mem,LARGE_INTEGER * size)1018 static HRESULT map_file(HANDLE file, HANDLE *map, void **mem, LARGE_INTEGER *size)
1019 {
1020 *map = NULL;
1021 if (!GetFileSizeEx(file, size)) return HRESULT_FROM_WIN32(GetLastError());
1022 if (size->u.HighPart)
1023 {
1024 WARN("file too large\n");
1025 return E_FAIL;
1026 }
1027 if (!(*map = CreateFileMappingW(file, NULL, PAGE_READONLY, 0, size->u.LowPart, NULL)))
1028 {
1029 return HRESULT_FROM_WIN32(GetLastError());
1030 }
1031 if (!(*mem = MapViewOfFile(*map, FILE_MAP_READ, 0, 0, size->u.LowPart)))
1032 {
1033 CloseHandle(*map);
1034 return HRESULT_FROM_WIN32(GetLastError());
1035 }
1036 return S_OK;
1037 }
1038
stream_initialize_from_filehandle(IWICStream * iface,HANDLE file)1039 HRESULT stream_initialize_from_filehandle(IWICStream *iface, HANDLE file)
1040 {
1041 IWICStreamImpl *This = impl_from_IWICStream(iface);
1042 StreamOnFileHandle *pObject;
1043 IWICStream *stream = NULL;
1044 HANDLE map;
1045 void *mem;
1046 LARGE_INTEGER size;
1047 HRESULT hr;
1048 TRACE("(%p,%p)\n", iface, file);
1049
1050 if (This->pStream) return WINCODEC_ERR_WRONGSTATE;
1051
1052 hr = map_file(file, &map, &mem, &size);
1053 if (FAILED(hr)) return hr;
1054
1055 hr = StreamImpl_Create(&stream);
1056 if (FAILED(hr)) goto error;
1057
1058 hr = IWICStreamImpl_InitializeFromMemory(stream, mem, size.u.LowPart);
1059 if (FAILED(hr)) goto error;
1060
1061 pObject = HeapAlloc(GetProcessHeap(), 0, sizeof(StreamOnFileHandle));
1062 if (!pObject)
1063 {
1064 hr = E_OUTOFMEMORY;
1065 goto error;
1066 }
1067 pObject->IStream_iface.lpVtbl = &StreamOnFileHandle_Vtbl;
1068 pObject->ref = 1;
1069 pObject->map = map;
1070 pObject->mem = mem;
1071 pObject->stream = stream;
1072
1073 if (InterlockedCompareExchangePointer((void**)&This->pStream, pObject, NULL))
1074 {
1075 /* Some other thread set the stream first. */
1076 IStream_Release(&pObject->IStream_iface);
1077 return WINCODEC_ERR_WRONGSTATE;
1078 }
1079 return S_OK;
1080
1081 error:
1082 if (stream) IWICStream_Release(stream);
1083 UnmapViewOfFile(mem);
1084 CloseHandle(map);
1085 return hr;
1086 }
1087
IWICStreamImpl_InitializeFromIStreamRegion(IWICStream * iface,IStream * pIStream,ULARGE_INTEGER ulOffset,ULARGE_INTEGER ulMaxSize)1088 static HRESULT WINAPI IWICStreamImpl_InitializeFromIStreamRegion(IWICStream *iface,
1089 IStream *pIStream, ULARGE_INTEGER ulOffset, ULARGE_INTEGER ulMaxSize)
1090 {
1091 IWICStreamImpl *This = impl_from_IWICStream(iface);
1092 StreamOnStreamRange *pObject;
1093
1094 TRACE("(%p,%p,%s,%s)\n", iface, pIStream, wine_dbgstr_longlong(ulOffset.QuadPart),
1095 wine_dbgstr_longlong(ulMaxSize.QuadPart));
1096
1097 if (!pIStream) return E_INVALIDARG;
1098 if (This->pStream) return WINCODEC_ERR_WRONGSTATE;
1099
1100 pObject = HeapAlloc(GetProcessHeap(), 0, sizeof(StreamOnStreamRange));
1101 if (!pObject) return E_OUTOFMEMORY;
1102
1103 pObject->IStream_iface.lpVtbl = &StreamOnStreamRange_Vtbl;
1104 pObject->ref = 1;
1105 IStream_AddRef(pIStream);
1106 pObject->stream = pIStream;
1107 pObject->pos.QuadPart = 0;
1108 pObject->offset = ulOffset;
1109 pObject->max_size = ulMaxSize;
1110 InitializeCriticalSection(&pObject->lock);
1111 pObject->lock.DebugInfo->Spare[0] = (DWORD_PTR)(__FILE__ ": StreamOnStreamRange.lock");
1112
1113 if (InterlockedCompareExchangePointer((void**)&This->pStream, pObject, NULL))
1114 {
1115 /* Some other thread set the stream first. */
1116 IStream_Release(&pObject->IStream_iface);
1117 return WINCODEC_ERR_WRONGSTATE;
1118 }
1119
1120 return S_OK;
1121 }
1122
1123
1124 static const IWICStreamVtbl WICStream_Vtbl =
1125 {
1126 /*** IUnknown methods ***/
1127 IWICStreamImpl_QueryInterface,
1128 IWICStreamImpl_AddRef,
1129 IWICStreamImpl_Release,
1130 /*** ISequentialStream methods ***/
1131 IWICStreamImpl_Read,
1132 IWICStreamImpl_Write,
1133 /*** IStream methods ***/
1134 IWICStreamImpl_Seek,
1135 IWICStreamImpl_SetSize,
1136 IWICStreamImpl_CopyTo,
1137 IWICStreamImpl_Commit,
1138 IWICStreamImpl_Revert,
1139 IWICStreamImpl_LockRegion,
1140 IWICStreamImpl_UnlockRegion,
1141 IWICStreamImpl_Stat,
1142 IWICStreamImpl_Clone,
1143 /*** IWICStream methods ***/
1144 IWICStreamImpl_InitializeFromIStream,
1145 IWICStreamImpl_InitializeFromFilename,
1146 IWICStreamImpl_InitializeFromMemory,
1147 IWICStreamImpl_InitializeFromIStreamRegion,
1148 };
1149
StreamImpl_Create(IWICStream ** stream)1150 HRESULT StreamImpl_Create(IWICStream **stream)
1151 {
1152 IWICStreamImpl *pObject;
1153
1154 if( !stream ) return E_INVALIDARG;
1155
1156 pObject = HeapAlloc(GetProcessHeap(), 0, sizeof(IWICStreamImpl));
1157 if( !pObject ) {
1158 *stream = NULL;
1159 return E_OUTOFMEMORY;
1160 }
1161
1162 pObject->IWICStream_iface.lpVtbl = &WICStream_Vtbl;
1163 pObject->ref = 1;
1164 pObject->pStream = NULL;
1165
1166 *stream = &pObject->IWICStream_iface;
1167
1168 return S_OK;
1169 }
1170