include/mscvpdb.h: Use flexible array members for the rest of structures.
[wine.git] / dlls / windowscodecs / bmpencode.c
blob28da4a5e6d2219b7f94a23e533cdd44adb1285ef
1 /*
2 * Copyright 2009 Vincent Povirk for CodeWeavers
3 * Copyright 2016 Dmitry Timoshkov
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.
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.
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
20 #include <stdarg.h>
22 #define COBJMACROS
24 #include "windef.h"
25 #include "winbase.h"
26 #include "winreg.h"
27 #include "wingdi.h"
28 #include "objbase.h"
30 #include "wincodecs_private.h"
32 #include "wine/debug.h"
34 WINE_DEFAULT_DEBUG_CHANNEL(wincodecs);
36 struct bmp_pixelformat {
37 const WICPixelFormatGUID *guid;
38 UINT bpp;
39 UINT colors; /* palette size */
40 DWORD compression;
41 DWORD redmask;
42 DWORD greenmask;
43 DWORD bluemask;
44 DWORD alphamask;
47 static const struct bmp_pixelformat formats[] = {
48 {&GUID_WICPixelFormat24bppBGR, 24, 0, BI_RGB},
49 {&GUID_WICPixelFormatBlackWhite, 1, 2, BI_RGB},
50 {&GUID_WICPixelFormat1bppIndexed, 1, 2, BI_RGB},
51 {&GUID_WICPixelFormat2bppIndexed, 2, 4, BI_RGB},
52 {&GUID_WICPixelFormat4bppIndexed, 4, 16, BI_RGB},
53 {&GUID_WICPixelFormat8bppIndexed, 8, 256, BI_RGB},
54 {&GUID_WICPixelFormat16bppBGR555, 16, 0, BI_RGB},
55 {&GUID_WICPixelFormat16bppBGR565, 16, 0, BI_BITFIELDS, 0xf800, 0x7e0, 0x1f, 0},
56 {&GUID_WICPixelFormat32bppBGR, 32, 0, BI_RGB},
57 {&GUID_WICPixelFormat32bppBGRA, 32, 0, BI_BITFIELDS, 0xff0000, 0xff00, 0xff, 0xff000000},
58 {NULL}
61 typedef struct BmpFrameEncode {
62 IWICBitmapFrameEncode IWICBitmapFrameEncode_iface;
63 LONG ref;
64 IStream *stream;
65 BOOL initialized;
66 UINT width, height;
67 BYTE *bits;
68 const struct bmp_pixelformat *format;
69 double xres, yres;
70 UINT lineswritten;
71 UINT stride;
72 WICColor palette[256];
73 UINT colors;
74 BOOL committed;
75 } BmpFrameEncode;
77 static inline BmpFrameEncode *impl_from_IWICBitmapFrameEncode(IWICBitmapFrameEncode *iface)
79 return CONTAINING_RECORD(iface, BmpFrameEncode, IWICBitmapFrameEncode_iface);
82 static HRESULT WINAPI BmpFrameEncode_QueryInterface(IWICBitmapFrameEncode *iface, REFIID iid,
83 void **ppv)
85 BmpFrameEncode *This = impl_from_IWICBitmapFrameEncode(iface);
86 TRACE("(%p,%s,%p)\n", iface, debugstr_guid(iid), ppv);
88 if (!ppv) return E_INVALIDARG;
90 if (IsEqualIID(&IID_IUnknown, iid) ||
91 IsEqualIID(&IID_IWICBitmapFrameEncode, iid))
93 *ppv = &This->IWICBitmapFrameEncode_iface;
95 else
97 *ppv = NULL;
98 return E_NOINTERFACE;
101 IUnknown_AddRef((IUnknown*)*ppv);
102 return S_OK;
105 static ULONG WINAPI BmpFrameEncode_AddRef(IWICBitmapFrameEncode *iface)
107 BmpFrameEncode *This = impl_from_IWICBitmapFrameEncode(iface);
108 ULONG ref = InterlockedIncrement(&This->ref);
110 TRACE("(%p) refcount=%lu\n", iface, ref);
112 return ref;
115 static ULONG WINAPI BmpFrameEncode_Release(IWICBitmapFrameEncode *iface)
117 BmpFrameEncode *This = impl_from_IWICBitmapFrameEncode(iface);
118 ULONG ref = InterlockedDecrement(&This->ref);
120 TRACE("(%p) refcount=%lu\n", iface, ref);
122 if (ref == 0)
124 if (This->stream) IStream_Release(This->stream);
125 free(This->bits);
126 free(This);
129 return ref;
132 static HRESULT WINAPI BmpFrameEncode_Initialize(IWICBitmapFrameEncode *iface,
133 IPropertyBag2 *pIEncoderOptions)
135 BmpFrameEncode *This = impl_from_IWICBitmapFrameEncode(iface);
136 TRACE("(%p,%p)\n", iface, pIEncoderOptions);
138 if (This->initialized) return WINCODEC_ERR_WRONGSTATE;
140 if (pIEncoderOptions)
141 WARN("ignoring encoder options.\n");
143 This->initialized = TRUE;
145 return S_OK;
148 static HRESULT WINAPI BmpFrameEncode_SetSize(IWICBitmapFrameEncode *iface,
149 UINT uiWidth, UINT uiHeight)
151 BmpFrameEncode *This = impl_from_IWICBitmapFrameEncode(iface);
152 TRACE("(%p,%u,%u)\n", iface, uiWidth, uiHeight);
154 if (!This->initialized || This->bits) return WINCODEC_ERR_WRONGSTATE;
156 This->width = uiWidth;
157 This->height = uiHeight;
159 return S_OK;
162 static HRESULT WINAPI BmpFrameEncode_SetResolution(IWICBitmapFrameEncode *iface,
163 double dpiX, double dpiY)
165 BmpFrameEncode *This = impl_from_IWICBitmapFrameEncode(iface);
166 TRACE("(%p,%0.2f,%0.2f)\n", iface, dpiX, dpiY);
168 if (!This->initialized || This->bits) return WINCODEC_ERR_WRONGSTATE;
170 This->xres = dpiX;
171 This->yres = dpiY;
173 return S_OK;
176 static HRESULT WINAPI BmpFrameEncode_SetPixelFormat(IWICBitmapFrameEncode *iface,
177 WICPixelFormatGUID *pPixelFormat)
179 BmpFrameEncode *This = impl_from_IWICBitmapFrameEncode(iface);
180 int i;
181 TRACE("(%p,%s)\n", iface, debugstr_guid(pPixelFormat));
183 if (!This->initialized || This->bits) return WINCODEC_ERR_WRONGSTATE;
185 if (IsEqualGUID(pPixelFormat, &GUID_WICPixelFormatBlackWhite))
186 *pPixelFormat = GUID_WICPixelFormat1bppIndexed;
187 else if (IsEqualGUID(pPixelFormat, &GUID_WICPixelFormat2bppIndexed))
188 *pPixelFormat = GUID_WICPixelFormat4bppIndexed;
190 for (i=0; formats[i].guid; i++)
192 if (IsEqualGUID(formats[i].guid, pPixelFormat))
193 break;
196 if (!formats[i].guid) i = 0;
198 This->format = &formats[i];
199 memcpy(pPixelFormat, This->format->guid, sizeof(GUID));
201 return S_OK;
204 static HRESULT WINAPI BmpFrameEncode_SetColorContexts(IWICBitmapFrameEncode *iface,
205 UINT cCount, IWICColorContext **ppIColorContext)
207 FIXME("(%p,%u,%p): stub\n", iface, cCount, ppIColorContext);
208 return E_NOTIMPL;
211 static HRESULT WINAPI BmpFrameEncode_SetPalette(IWICBitmapFrameEncode *iface,
212 IWICPalette *palette)
214 BmpFrameEncode *This = impl_from_IWICBitmapFrameEncode(iface);
215 HRESULT hr;
217 TRACE("(%p,%p)\n", iface, palette);
219 if (!palette) return E_INVALIDARG;
221 if (!This->initialized)
222 return WINCODEC_ERR_NOTINITIALIZED;
224 hr = IWICPalette_GetColors(palette, 256, This->palette, &This->colors);
225 if (hr == S_OK)
227 UINT i;
228 for (i = 0; i < This->colors; i++)
229 This->palette[i] |= 0xff000000; /* BMP palette has no alpha */
231 return hr;
234 static HRESULT WINAPI BmpFrameEncode_SetThumbnail(IWICBitmapFrameEncode *iface,
235 IWICBitmapSource *pIThumbnail)
237 FIXME("(%p,%p): stub\n", iface, pIThumbnail);
238 return WINCODEC_ERR_UNSUPPORTEDOPERATION;
241 static HRESULT BmpFrameEncode_AllocateBits(BmpFrameEncode *This)
243 if (!This->bits)
245 if (!This->initialized || !This->width || !This->height || !This->format)
246 return WINCODEC_ERR_WRONGSTATE;
248 This->stride = (((This->width * This->format->bpp)+31)/32)*4;
249 This->bits = calloc(This->stride, This->height);
250 if (!This->bits) return E_OUTOFMEMORY;
253 return S_OK;
256 static HRESULT WINAPI BmpFrameEncode_WritePixels(IWICBitmapFrameEncode *iface,
257 UINT lineCount, UINT cbStride, UINT cbBufferSize, BYTE *pbPixels)
259 BmpFrameEncode *This = impl_from_IWICBitmapFrameEncode(iface);
260 UINT dstbuffersize, bytesperrow, row;
261 BYTE *dst, *src;
262 HRESULT hr;
264 TRACE("(%p,%u,%u,%u,%p)\n", iface, lineCount, cbStride, cbBufferSize, pbPixels);
266 if (!This->initialized || !This->width || !This->height || !This->format)
267 return WINCODEC_ERR_WRONGSTATE;
269 hr = BmpFrameEncode_AllocateBits(This);
270 if (FAILED(hr)) return hr;
272 bytesperrow = ((This->format->bpp * This->width) + 7) / 8;
274 if (This->stride < bytesperrow)
275 return E_INVALIDARG;
277 dstbuffersize = This->stride * (This->height - This->lineswritten);
278 if ((This->stride * (lineCount - 1)) + bytesperrow > dstbuffersize)
279 return E_INVALIDARG;
281 src = pbPixels;
282 dst = This->bits + This->stride * (This->height - This->lineswritten - 1);
283 for (row = 0; row < lineCount; row++)
285 memcpy(dst, src, bytesperrow);
286 src += cbStride;
287 dst -= This->stride;
290 This->lineswritten += lineCount;
292 return S_OK;
295 static HRESULT WINAPI BmpFrameEncode_WriteSource(IWICBitmapFrameEncode *iface,
296 IWICBitmapSource *pIBitmapSource, WICRect *prc)
298 BmpFrameEncode *This = impl_from_IWICBitmapFrameEncode(iface);
299 HRESULT hr;
300 TRACE("(%p,%p,%s)\n", iface, pIBitmapSource, debug_wic_rect(prc));
302 if (!This->initialized)
303 return WINCODEC_ERR_WRONGSTATE;
305 hr = configure_write_source(iface, pIBitmapSource, prc,
306 This->format ? This->format->guid : NULL, This->width, This->height,
307 This->xres, This->yres);
309 if (SUCCEEDED(hr))
311 hr = write_source(iface, pIBitmapSource, prc,
312 This->format->guid, This->format->bpp, !This->colors && This->format->colors,
313 This->width, This->height);
316 return hr;
319 static HRESULT WINAPI BmpFrameEncode_Commit(IWICBitmapFrameEncode *iface)
321 BmpFrameEncode *This = impl_from_IWICBitmapFrameEncode(iface);
322 BITMAPFILEHEADER bfh;
323 BITMAPV5HEADER bih;
324 UINT info_size;
325 LARGE_INTEGER pos;
326 ULONG byteswritten;
327 HRESULT hr;
329 TRACE("(%p)\n", iface);
331 if (!This->bits || This->committed || This->height != This->lineswritten)
332 return WINCODEC_ERR_WRONGSTATE;
334 bfh.bfType = 0x4d42; /* "BM" */
335 bfh.bfReserved1 = 0;
336 bfh.bfReserved2 = 0;
338 bih.bV5Size = info_size = sizeof(BITMAPINFOHEADER);
339 bih.bV5Width = This->width;
340 bih.bV5Height = This->height;
341 bih.bV5Planes = 1;
342 bih.bV5BitCount = This->format->bpp;
343 bih.bV5Compression = This->format->compression;
344 bih.bV5SizeImage = This->stride*This->height;
345 bih.bV5XPelsPerMeter = (This->xres+0.0127) / 0.0254;
346 bih.bV5YPelsPerMeter = (This->yres+0.0127) / 0.0254;
347 bih.bV5ClrUsed = (This->format->bpp <= 8) ? This->colors : 0;
348 bih.bV5ClrImportant = bih.bV5ClrUsed;
350 if (This->format->compression == BI_BITFIELDS)
352 if (This->format->alphamask)
353 bih.bV5Size = info_size = sizeof(BITMAPV4HEADER);
354 else
355 info_size = sizeof(BITMAPINFOHEADER)+12;
356 bih.bV5RedMask = This->format->redmask;
357 bih.bV5GreenMask = This->format->greenmask;
358 bih.bV5BlueMask = This->format->bluemask;
359 bih.bV5AlphaMask = This->format->alphamask;
360 bih.bV5CSType = LCS_DEVICE_RGB;
363 bfh.bfSize = sizeof(BITMAPFILEHEADER) + info_size + bih.bV5SizeImage;
364 bfh.bfOffBits = sizeof(BITMAPFILEHEADER) + info_size;
365 bfh.bfOffBits += bih.bV5ClrUsed * sizeof(WICColor);
367 pos.QuadPart = 0;
368 hr = IStream_Seek(This->stream, pos, STREAM_SEEK_SET, NULL);
369 if (FAILED(hr)) return hr;
371 hr = IStream_Write(This->stream, &bfh, sizeof(BITMAPFILEHEADER), &byteswritten);
372 if (FAILED(hr)) return hr;
373 if (byteswritten != sizeof(BITMAPFILEHEADER)) return E_FAIL;
375 hr = IStream_Write(This->stream, &bih, info_size, &byteswritten);
376 if (FAILED(hr)) return hr;
377 if (byteswritten != info_size) return E_FAIL;
379 /* write the palette */
380 if (This->format->colors)
382 hr = IStream_Write(This->stream, This->palette, This->colors * sizeof(WICColor), &byteswritten);
383 if (FAILED(hr)) return hr;
384 if (byteswritten != This->colors * sizeof(WICColor)) return E_FAIL;
387 hr = IStream_Write(This->stream, This->bits, bih.bV5SizeImage, &byteswritten);
388 if (FAILED(hr)) return hr;
389 if (byteswritten != bih.bV5SizeImage) return E_FAIL;
391 This->committed = TRUE;
393 return S_OK;
396 static HRESULT WINAPI BmpFrameEncode_GetMetadataQueryWriter(IWICBitmapFrameEncode *iface,
397 IWICMetadataQueryWriter **query_writer)
399 BmpFrameEncode *encoder = impl_from_IWICBitmapFrameEncode(iface);
401 TRACE("iface %p, query_writer %p.\n", iface, query_writer);
403 if (!encoder->initialized)
404 return WINCODEC_ERR_NOTINITIALIZED;
406 return WINCODEC_ERR_UNSUPPORTEDOPERATION;
409 static const IWICBitmapFrameEncodeVtbl BmpFrameEncode_Vtbl = {
410 BmpFrameEncode_QueryInterface,
411 BmpFrameEncode_AddRef,
412 BmpFrameEncode_Release,
413 BmpFrameEncode_Initialize,
414 BmpFrameEncode_SetSize,
415 BmpFrameEncode_SetResolution,
416 BmpFrameEncode_SetPixelFormat,
417 BmpFrameEncode_SetColorContexts,
418 BmpFrameEncode_SetPalette,
419 BmpFrameEncode_SetThumbnail,
420 BmpFrameEncode_WritePixels,
421 BmpFrameEncode_WriteSource,
422 BmpFrameEncode_Commit,
423 BmpFrameEncode_GetMetadataQueryWriter
426 typedef struct BmpEncoder {
427 IWICBitmapEncoder IWICBitmapEncoder_iface;
428 LONG ref;
429 IStream *stream;
430 BmpFrameEncode *frame;
431 } BmpEncoder;
433 static inline BmpEncoder *impl_from_IWICBitmapEncoder(IWICBitmapEncoder *iface)
435 return CONTAINING_RECORD(iface, BmpEncoder, IWICBitmapEncoder_iface);
438 static HRESULT WINAPI BmpEncoder_QueryInterface(IWICBitmapEncoder *iface, REFIID iid,
439 void **ppv)
441 BmpEncoder *This = impl_from_IWICBitmapEncoder(iface);
442 TRACE("(%p,%s,%p)\n", iface, debugstr_guid(iid), ppv);
444 if (!ppv) return E_INVALIDARG;
446 if (IsEqualIID(&IID_IUnknown, iid) ||
447 IsEqualIID(&IID_IWICBitmapEncoder, iid))
449 *ppv = &This->IWICBitmapEncoder_iface;
451 else
453 *ppv = NULL;
454 return E_NOINTERFACE;
457 IUnknown_AddRef((IUnknown*)*ppv);
458 return S_OK;
461 static ULONG WINAPI BmpEncoder_AddRef(IWICBitmapEncoder *iface)
463 BmpEncoder *This = impl_from_IWICBitmapEncoder(iface);
464 ULONG ref = InterlockedIncrement(&This->ref);
466 TRACE("(%p) refcount=%lu\n", iface, ref);
468 return ref;
471 static ULONG WINAPI BmpEncoder_Release(IWICBitmapEncoder *iface)
473 BmpEncoder *This = impl_from_IWICBitmapEncoder(iface);
474 ULONG ref = InterlockedDecrement(&This->ref);
476 TRACE("(%p) refcount=%lu\n", iface, ref);
478 if (ref == 0)
480 if (This->stream) IStream_Release(This->stream);
481 if (This->frame) IWICBitmapFrameEncode_Release(&This->frame->IWICBitmapFrameEncode_iface);
482 free(This);
485 return ref;
488 static HRESULT WINAPI BmpEncoder_Initialize(IWICBitmapEncoder *iface,
489 IStream *pIStream, WICBitmapEncoderCacheOption cacheOption)
491 BmpEncoder *This = impl_from_IWICBitmapEncoder(iface);
493 TRACE("(%p,%p,%u)\n", iface, pIStream, cacheOption);
495 IStream_AddRef(pIStream);
496 This->stream = pIStream;
498 return S_OK;
501 static HRESULT WINAPI BmpEncoder_GetContainerFormat(IWICBitmapEncoder *iface,
502 GUID *pguidContainerFormat)
504 TRACE("(%p,%p)\n", iface, pguidContainerFormat);
506 if (!pguidContainerFormat)
507 return E_INVALIDARG;
509 memcpy(pguidContainerFormat, &GUID_ContainerFormatBmp, sizeof(GUID));
510 return S_OK;
513 static HRESULT WINAPI BmpEncoder_GetEncoderInfo(IWICBitmapEncoder *iface, IWICBitmapEncoderInfo **info)
515 IWICComponentInfo *comp_info;
516 HRESULT hr;
518 TRACE("%p,%p\n", iface, info);
520 if (!info) return E_INVALIDARG;
522 hr = CreateComponentInfo(&CLSID_WICBmpEncoder, &comp_info);
523 if (hr == S_OK)
525 hr = IWICComponentInfo_QueryInterface(comp_info, &IID_IWICBitmapEncoderInfo, (void **)info);
526 IWICComponentInfo_Release(comp_info);
528 return hr;
531 static HRESULT WINAPI BmpEncoder_SetColorContexts(IWICBitmapEncoder *iface,
532 UINT cCount, IWICColorContext **ppIColorContext)
534 FIXME("(%p,%u,%p): stub\n", iface, cCount, ppIColorContext);
535 return E_NOTIMPL;
538 static HRESULT WINAPI BmpEncoder_SetPalette(IWICBitmapEncoder *iface, IWICPalette *palette)
540 BmpEncoder *This = impl_from_IWICBitmapEncoder(iface);
542 TRACE("(%p,%p)\n", iface, palette);
543 return This->stream ? WINCODEC_ERR_UNSUPPORTEDOPERATION : WINCODEC_ERR_NOTINITIALIZED;
546 static HRESULT WINAPI BmpEncoder_SetThumbnail(IWICBitmapEncoder *iface, IWICBitmapSource *pIThumbnail)
548 TRACE("(%p,%p)\n", iface, pIThumbnail);
549 return WINCODEC_ERR_UNSUPPORTEDOPERATION;
552 static HRESULT WINAPI BmpEncoder_SetPreview(IWICBitmapEncoder *iface, IWICBitmapSource *pIPreview)
554 TRACE("(%p,%p)\n", iface, pIPreview);
555 return WINCODEC_ERR_UNSUPPORTEDOPERATION;
558 static HRESULT WINAPI BmpEncoder_CreateNewFrame(IWICBitmapEncoder *iface,
559 IWICBitmapFrameEncode **ppIFrameEncode, IPropertyBag2 **ppIEncoderOptions)
561 BmpEncoder *This = impl_from_IWICBitmapEncoder(iface);
562 BmpFrameEncode *encode;
563 HRESULT hr;
564 static const PROPBAG2 opts[1] =
566 { PROPBAG2_TYPE_DATA, VT_BOOL, 0, 0, (LPOLESTR)L"EnableV5Header32bppBGRA" },
569 TRACE("(%p,%p,%p)\n", iface, ppIFrameEncode, ppIEncoderOptions);
571 if (This->frame) return WINCODEC_ERR_UNSUPPORTEDOPERATION;
573 if (!This->stream) return WINCODEC_ERR_NOTINITIALIZED;
575 if (ppIEncoderOptions)
577 hr = CreatePropertyBag2(opts, ARRAY_SIZE(opts), ppIEncoderOptions);
578 if (FAILED(hr)) return hr;
581 encode = malloc(sizeof(BmpFrameEncode));
582 if (!encode)
584 IPropertyBag2_Release(*ppIEncoderOptions);
585 *ppIEncoderOptions = NULL;
586 return E_OUTOFMEMORY;
588 encode->IWICBitmapFrameEncode_iface.lpVtbl = &BmpFrameEncode_Vtbl;
589 encode->ref = 2;
590 IStream_AddRef(This->stream);
591 encode->stream = This->stream;
592 encode->initialized = FALSE;
593 encode->width = 0;
594 encode->height = 0;
595 encode->bits = NULL;
596 encode->format = NULL;
597 encode->xres = 0.0;
598 encode->yres = 0.0;
599 encode->lineswritten = 0;
600 encode->colors = 0;
601 encode->committed = FALSE;
603 *ppIFrameEncode = &encode->IWICBitmapFrameEncode_iface;
604 This->frame = encode;
606 return S_OK;
609 static HRESULT WINAPI BmpEncoder_Commit(IWICBitmapEncoder *iface)
611 BmpEncoder *This = impl_from_IWICBitmapEncoder(iface);
612 TRACE("(%p)\n", iface);
614 if (!This->frame || !This->frame->committed) return WINCODEC_ERR_WRONGSTATE;
616 return S_OK;
619 static HRESULT WINAPI BmpEncoder_GetMetadataQueryWriter(IWICBitmapEncoder *iface,
620 IWICMetadataQueryWriter **ppIMetadataQueryWriter)
622 FIXME("(%p,%p): stub\n", iface, ppIMetadataQueryWriter);
623 return E_NOTIMPL;
626 static const IWICBitmapEncoderVtbl BmpEncoder_Vtbl = {
627 BmpEncoder_QueryInterface,
628 BmpEncoder_AddRef,
629 BmpEncoder_Release,
630 BmpEncoder_Initialize,
631 BmpEncoder_GetContainerFormat,
632 BmpEncoder_GetEncoderInfo,
633 BmpEncoder_SetColorContexts,
634 BmpEncoder_SetPalette,
635 BmpEncoder_SetThumbnail,
636 BmpEncoder_SetPreview,
637 BmpEncoder_CreateNewFrame,
638 BmpEncoder_Commit,
639 BmpEncoder_GetMetadataQueryWriter
642 HRESULT BmpEncoder_CreateInstance(REFIID iid, void** ppv)
644 BmpEncoder *This;
645 HRESULT ret;
647 TRACE("(%s,%p)\n", debugstr_guid(iid), ppv);
649 *ppv = NULL;
651 This = malloc(sizeof(BmpEncoder));
652 if (!This) return E_OUTOFMEMORY;
654 This->IWICBitmapEncoder_iface.lpVtbl = &BmpEncoder_Vtbl;
655 This->ref = 1;
656 This->stream = NULL;
657 This->frame = NULL;
659 ret = IWICBitmapEncoder_QueryInterface(&This->IWICBitmapEncoder_iface, iid, ppv);
660 IWICBitmapEncoder_Release(&This->IWICBitmapEncoder_iface);
662 return ret;