2 * Copyright 2010 Damjan Jovanovic
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.
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.
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
20 #include "wine/port.h"
35 #include "wincodecs_private.h"
37 #include "wine/debug.h"
38 #include "wine/library.h"
40 WINE_DEFAULT_DEBUG_CHANNEL(wincodecs);
44 static void *libicns_handle;
45 #define MAKE_FUNCPTR(f) static typeof(f) * p##f
46 MAKE_FUNCPTR(icns_create_family);
47 MAKE_FUNCPTR(icns_export_family_data);
48 MAKE_FUNCPTR(icns_free_image);
49 MAKE_FUNCPTR(icns_get_mask_type_for_icon_type);
50 MAKE_FUNCPTR(icns_get_type_from_image_info);
51 MAKE_FUNCPTR(icns_init_image_for_type);
52 MAKE_FUNCPTR(icns_new_element_from_image);
53 MAKE_FUNCPTR(icns_set_element_in_family);
56 static void *load_libicns(void)
58 if((libicns_handle = wine_dlopen(SONAME_LIBICNS, RTLD_NOW, NULL, 0)) != NULL) {
60 #define LOAD_FUNCPTR(f) \
61 if((p##f = wine_dlsym(libicns_handle, #f, NULL, 0)) == NULL) { \
62 libicns_handle = NULL; \
65 LOAD_FUNCPTR(icns_create_family);
66 LOAD_FUNCPTR(icns_export_family_data);
67 LOAD_FUNCPTR(icns_free_image);
68 LOAD_FUNCPTR(icns_get_mask_type_for_icon_type);
69 LOAD_FUNCPTR(icns_get_type_from_image_info);
70 LOAD_FUNCPTR(icns_init_image_for_type);
71 LOAD_FUNCPTR(icns_new_element_from_image);
72 LOAD_FUNCPTR(icns_set_element_in_family);
75 return libicns_handle;
78 typedef struct IcnsEncoder {
79 IWICBitmapEncoder IWICBitmapEncoder_iface;
82 icns_family_t *icns_family;
83 BOOL any_frame_committed;
84 int outstanding_commits;
86 CRITICAL_SECTION lock;
89 static inline IcnsEncoder *impl_from_IWICBitmapEncoder(IWICBitmapEncoder *iface)
91 return CONTAINING_RECORD(iface, IcnsEncoder, IWICBitmapEncoder_iface);
94 typedef struct IcnsFrameEncode {
95 IWICBitmapFrameEncode IWICBitmapFrameEncode_iface;
101 icns_type_t icns_type;
102 icns_image_t icns_image;
107 static inline IcnsFrameEncode *impl_from_IWICBitmapFrameEncode(IWICBitmapFrameEncode *iface)
109 return CONTAINING_RECORD(iface, IcnsFrameEncode, IWICBitmapFrameEncode_iface);
112 static HRESULT WINAPI IcnsFrameEncode_QueryInterface(IWICBitmapFrameEncode *iface, REFIID iid,
115 IcnsFrameEncode *This = impl_from_IWICBitmapFrameEncode(iface);
116 TRACE("(%p,%s,%p)\n", iface, debugstr_guid(iid), ppv);
118 if (!ppv) return E_INVALIDARG;
120 if (IsEqualIID(&IID_IUnknown, iid) ||
121 IsEqualIID(&IID_IWICBitmapFrameEncode, iid))
123 *ppv = &This->IWICBitmapFrameEncode_iface;
128 return E_NOINTERFACE;
131 IUnknown_AddRef((IUnknown*)*ppv);
135 static ULONG WINAPI IcnsFrameEncode_AddRef(IWICBitmapFrameEncode *iface)
137 IcnsFrameEncode *This = impl_from_IWICBitmapFrameEncode(iface);
138 ULONG ref = InterlockedIncrement(&This->ref);
140 TRACE("(%p) refcount=%u\n", iface, ref);
145 static ULONG WINAPI IcnsFrameEncode_Release(IWICBitmapFrameEncode *iface)
147 IcnsFrameEncode *This = impl_from_IWICBitmapFrameEncode(iface);
148 ULONG ref = InterlockedDecrement(&This->ref);
150 TRACE("(%p) refcount=%u\n", iface, ref);
154 if (!This->committed)
156 EnterCriticalSection(&This->encoder->lock);
157 This->encoder->outstanding_commits--;
158 LeaveCriticalSection(&This->encoder->lock);
160 if (This->icns_image.imageData != NULL)
161 picns_free_image(&This->icns_image);
163 IUnknown_Release((IUnknown*)This->encoder);
164 HeapFree(GetProcessHeap(), 0, This);
170 static HRESULT WINAPI IcnsFrameEncode_Initialize(IWICBitmapFrameEncode *iface,
171 IPropertyBag2 *pIEncoderOptions)
173 IcnsFrameEncode *This = impl_from_IWICBitmapFrameEncode(iface);
176 TRACE("(%p,%p)\n", iface, pIEncoderOptions);
178 EnterCriticalSection(&This->encoder->lock);
180 if (This->initialized)
182 hr = WINCODEC_ERR_WRONGSTATE;
185 This->initialized = TRUE;
188 LeaveCriticalSection(&This->encoder->lock);
192 static HRESULT WINAPI IcnsFrameEncode_SetSize(IWICBitmapFrameEncode *iface,
193 UINT uiWidth, UINT uiHeight)
195 IcnsFrameEncode *This = impl_from_IWICBitmapFrameEncode(iface);
198 TRACE("(%p,%u,%u)\n", iface, uiWidth, uiHeight);
200 EnterCriticalSection(&This->encoder->lock);
202 if (!This->initialized || This->icns_image.imageData)
204 hr = WINCODEC_ERR_WRONGSTATE;
208 This->width = uiWidth;
209 This->height = uiHeight;
212 LeaveCriticalSection(&This->encoder->lock);
216 static HRESULT WINAPI IcnsFrameEncode_SetResolution(IWICBitmapFrameEncode *iface,
217 double dpiX, double dpiY)
219 IcnsFrameEncode *This = impl_from_IWICBitmapFrameEncode(iface);
222 TRACE("(%p,%0.2f,%0.2f)\n", iface, dpiX, dpiY);
224 EnterCriticalSection(&This->encoder->lock);
226 if (!This->initialized || This->icns_image.imageData)
228 hr = WINCODEC_ERR_WRONGSTATE;
233 LeaveCriticalSection(&This->encoder->lock);
237 static HRESULT WINAPI IcnsFrameEncode_SetPixelFormat(IWICBitmapFrameEncode *iface,
238 WICPixelFormatGUID *pPixelFormat)
240 IcnsFrameEncode *This = impl_from_IWICBitmapFrameEncode(iface);
243 TRACE("(%p,%s)\n", iface, debugstr_guid(pPixelFormat));
245 EnterCriticalSection(&This->encoder->lock);
247 if (!This->initialized || This->icns_image.imageData)
249 hr = WINCODEC_ERR_WRONGSTATE;
253 memcpy(pPixelFormat, &GUID_WICPixelFormat32bppBGRA, sizeof(GUID));
256 LeaveCriticalSection(&This->encoder->lock);
260 static HRESULT WINAPI IcnsFrameEncode_SetColorContexts(IWICBitmapFrameEncode *iface,
261 UINT cCount, IWICColorContext **ppIColorContext)
263 FIXME("(%p,%u,%p): stub\n", iface, cCount, ppIColorContext);
267 static HRESULT WINAPI IcnsFrameEncode_SetPalette(IWICBitmapFrameEncode *iface,
268 IWICPalette *pIPalette)
270 FIXME("(%p,%p): stub\n", iface, pIPalette);
271 return WINCODEC_ERR_UNSUPPORTEDOPERATION;
274 static HRESULT WINAPI IcnsFrameEncode_SetThumbnail(IWICBitmapFrameEncode *iface,
275 IWICBitmapSource *pIThumbnail)
277 FIXME("(%p,%p): stub\n", iface, pIThumbnail);
278 return WINCODEC_ERR_UNSUPPORTEDOPERATION;
281 static HRESULT WINAPI IcnsFrameEncode_WritePixels(IWICBitmapFrameEncode *iface,
282 UINT lineCount, UINT cbStride, UINT cbBufferSize, BYTE *pbPixels)
284 IcnsFrameEncode *This = impl_from_IWICBitmapFrameEncode(iface);
289 TRACE("(%p,%u,%u,%u,%p)\n", iface, lineCount, cbStride, cbBufferSize, pbPixels);
291 EnterCriticalSection(&This->encoder->lock);
293 if (!This->initialized || !This->width || !This->height)
295 hr = WINCODEC_ERR_WRONGSTATE;
298 if (lineCount == 0 || lineCount + This->lines_written > This->height)
304 if (!This->icns_image.imageData)
306 icns_icon_info_t icns_info;
307 icns_info.isImage = 1;
308 icns_info.iconWidth = This->width;
309 icns_info.iconHeight = This->height;
310 icns_info.iconBitDepth = 32;
311 icns_info.iconChannels = 4;
312 icns_info.iconPixelDepth = icns_info.iconBitDepth / icns_info.iconChannels;
313 This->icns_type = picns_get_type_from_image_info(icns_info);
314 if (This->icns_type == ICNS_NULL_TYPE)
316 WARN("cannot generate ICNS icon from %dx%d image\n", This->width, This->height);
320 ret = picns_init_image_for_type(This->icns_type, &This->icns_image);
321 if (ret != ICNS_STATUS_OK)
323 WARN("error %d in icns_init_image_for_type\n", ret);
329 for (i = 0; i < lineCount; i++)
331 BYTE *src_row, *dst_row;
333 src_row = pbPixels + cbStride * i;
334 dst_row = This->icns_image.imageData + (This->lines_written + i)*(This->width*4);
335 /* swap bgr -> rgb */
336 for (j = 0; j < This->width*4; j += 4)
338 dst_row[j] = src_row[j+2];
339 dst_row[j+1] = src_row[j+1];
340 dst_row[j+2] = src_row[j];
341 dst_row[j+3] = src_row[j+3];
344 This->lines_written += lineCount;
347 LeaveCriticalSection(&This->encoder->lock);
351 static HRESULT WINAPI IcnsFrameEncode_WriteSource(IWICBitmapFrameEncode *iface,
352 IWICBitmapSource *pIBitmapSource, WICRect *prc)
354 IcnsFrameEncode *This = impl_from_IWICBitmapFrameEncode(iface);
357 WICPixelFormatGUID guid;
359 BYTE *pixeldata = NULL;
361 TRACE("(%p,%p,%p)\n", iface, pIBitmapSource, prc);
363 if (!This->initialized || !This->width || !This->height)
365 hr = WINCODEC_ERR_WRONGSTATE;
369 hr = IWICBitmapSource_GetPixelFormat(pIBitmapSource, &guid);
372 if (!IsEqualGUID(&guid, &GUID_WICPixelFormat32bppBGRA))
374 FIXME("format %s unsupported, could use WICConvertBitmapSource to convert\n", debugstr_guid(&guid));
382 hr = IWICBitmapSource_GetSize(pIBitmapSource, &width, &height);
392 if (prc->Width != This->width)
398 stride = (32 * This->width + 7)/8;
399 pixeldata = HeapAlloc(GetProcessHeap(), 0, stride * prc->Height);
406 hr = IWICBitmapSource_CopyPixels(pIBitmapSource, prc, stride,
407 stride*prc->Height, pixeldata);
410 hr = IWICBitmapFrameEncode_WritePixels(iface, prc->Height, stride,
411 stride*prc->Height, pixeldata);
415 HeapFree(GetProcessHeap(), 0, pixeldata);
419 static HRESULT WINAPI IcnsFrameEncode_Commit(IWICBitmapFrameEncode *iface)
421 IcnsFrameEncode *This = impl_from_IWICBitmapFrameEncode(iface);
422 icns_element_t *icns_element = NULL;
424 icns_element_t *mask_element = NULL;
429 TRACE("(%p): stub\n", iface);
431 memset(&mask, 0, sizeof(mask));
433 EnterCriticalSection(&This->encoder->lock);
435 if (!This->icns_image.imageData || This->lines_written != This->height || This->committed)
437 hr = WINCODEC_ERR_WRONGSTATE;
441 ret = picns_new_element_from_image(&This->icns_image, This->icns_type, &icns_element);
442 if (ret != ICNS_STATUS_OK && icns_element != NULL)
444 WARN("icns_new_element_from_image failed with error %d\n", ret);
449 if (This->icns_type != ICNS_512x512_32BIT_ARGB_DATA && This->icns_type != ICNS_256x256_32BIT_ARGB_DATA)
451 /* we need to write the mask too */
452 ret = picns_init_image_for_type(picns_get_mask_type_for_icon_type(This->icns_type), &mask);
453 if (ret != ICNS_STATUS_OK)
455 WARN("icns_init_image_from_type failed to make mask, error %d\n", ret);
459 for (i = 0; i < mask.imageHeight; i++)
462 for (j = 0; j < mask.imageWidth; j++)
463 mask.imageData[i*mask.imageWidth + j] = This->icns_image.imageData[i*mask.imageWidth*4 + j*4 + 3];
465 ret = picns_new_element_from_image(&mask, picns_get_mask_type_for_icon_type(This->icns_type), &mask_element);
466 if (ret != ICNS_STATUS_OK)
468 WARN("icns_new_element_from image failed to make element from mask, error %d\n", ret);
474 ret = picns_set_element_in_family(&This->encoder->icns_family, icns_element);
475 if (ret != ICNS_STATUS_OK)
477 WARN("icns_set_element_in_family failed for image with error %d\n", ret);
484 ret = picns_set_element_in_family(&This->encoder->icns_family, mask_element);
485 if (ret != ICNS_STATUS_OK)
487 WARN("icns_set_element_in_family failed for mask with error %d\n", ret);
493 This->committed = TRUE;
494 This->encoder->any_frame_committed = TRUE;
495 This->encoder->outstanding_commits--;
498 LeaveCriticalSection(&This->encoder->lock);
499 picns_free_image(&mask);
505 static HRESULT WINAPI IcnsFrameEncode_GetMetadataQueryWriter(IWICBitmapFrameEncode *iface,
506 IWICMetadataQueryWriter **ppIMetadataQueryWriter)
508 FIXME("(%p, %p): stub\n", iface, ppIMetadataQueryWriter);
512 static const IWICBitmapFrameEncodeVtbl IcnsEncoder_FrameVtbl = {
513 IcnsFrameEncode_QueryInterface,
514 IcnsFrameEncode_AddRef,
515 IcnsFrameEncode_Release,
516 IcnsFrameEncode_Initialize,
517 IcnsFrameEncode_SetSize,
518 IcnsFrameEncode_SetResolution,
519 IcnsFrameEncode_SetPixelFormat,
520 IcnsFrameEncode_SetColorContexts,
521 IcnsFrameEncode_SetPalette,
522 IcnsFrameEncode_SetThumbnail,
523 IcnsFrameEncode_WritePixels,
524 IcnsFrameEncode_WriteSource,
525 IcnsFrameEncode_Commit,
526 IcnsFrameEncode_GetMetadataQueryWriter
529 static HRESULT WINAPI IcnsEncoder_QueryInterface(IWICBitmapEncoder *iface, REFIID iid,
532 IcnsEncoder *This = impl_from_IWICBitmapEncoder(iface);
533 TRACE("(%p,%s,%p)\n", iface, debugstr_guid(iid), ppv);
535 if (!ppv) return E_INVALIDARG;
537 if (IsEqualIID(&IID_IUnknown, iid) ||
538 IsEqualIID(&IID_IWICBitmapEncoder, iid))
545 return E_NOINTERFACE;
548 IUnknown_AddRef((IUnknown*)*ppv);
552 static ULONG WINAPI IcnsEncoder_AddRef(IWICBitmapEncoder *iface)
554 IcnsEncoder *This = impl_from_IWICBitmapEncoder(iface);
555 ULONG ref = InterlockedIncrement(&This->ref);
557 TRACE("(%p) refcount=%u\n", iface, ref);
562 static ULONG WINAPI IcnsEncoder_Release(IWICBitmapEncoder *iface)
564 IcnsEncoder *This = impl_from_IWICBitmapEncoder(iface);
565 ULONG ref = InterlockedDecrement(&This->ref);
567 TRACE("(%p) refcount=%u\n", iface, ref);
571 This->lock.DebugInfo->Spare[0] = 0;
572 DeleteCriticalSection(&This->lock);
573 if (This->icns_family)
574 free(This->icns_family);
576 IStream_Release(This->stream);
577 HeapFree(GetProcessHeap(), 0, This);
583 static HRESULT WINAPI IcnsEncoder_Initialize(IWICBitmapEncoder *iface,
584 IStream *pIStream, WICBitmapEncoderCacheOption cacheOption)
586 IcnsEncoder *This = impl_from_IWICBitmapEncoder(iface);
590 TRACE("(%p,%p,%u)\n", iface, pIStream, cacheOption);
592 EnterCriticalSection(&This->lock);
594 if (This->icns_family)
596 hr = WINCODEC_ERR_WRONGSTATE;
599 ret = picns_create_family(&This->icns_family);
600 if (ret != ICNS_STATUS_OK)
602 WARN("error %d creating icns family\n", ret);
606 IStream_AddRef(pIStream);
607 This->stream = pIStream;
610 LeaveCriticalSection(&This->lock);
615 static HRESULT WINAPI IcnsEncoder_GetContainerFormat(IWICBitmapEncoder *iface,
616 GUID *pguidContainerFormat)
618 FIXME("(%p,%s): stub\n", iface, debugstr_guid(pguidContainerFormat));
622 static HRESULT WINAPI IcnsEncoder_GetEncoderInfo(IWICBitmapEncoder *iface,
623 IWICBitmapEncoderInfo **ppIEncoderInfo)
625 FIXME("(%p,%p): stub\n", iface, ppIEncoderInfo);
629 static HRESULT WINAPI IcnsEncoder_SetColorContexts(IWICBitmapEncoder *iface,
630 UINT cCount, IWICColorContext **ppIColorContext)
632 FIXME("(%p,%u,%p): stub\n", iface, cCount, ppIColorContext);
636 static HRESULT WINAPI IcnsEncoder_SetPalette(IWICBitmapEncoder *iface, IWICPalette *pIPalette)
638 TRACE("(%p,%p)\n", iface, pIPalette);
639 return WINCODEC_ERR_UNSUPPORTEDOPERATION;
642 static HRESULT WINAPI IcnsEncoder_SetThumbnail(IWICBitmapEncoder *iface, IWICBitmapSource *pIThumbnail)
644 TRACE("(%p,%p)\n", iface, pIThumbnail);
645 return WINCODEC_ERR_UNSUPPORTEDOPERATION;
648 static HRESULT WINAPI IcnsEncoder_SetPreview(IWICBitmapEncoder *iface, IWICBitmapSource *pIPreview)
650 TRACE("(%p,%p)\n", iface, pIPreview);
651 return WINCODEC_ERR_UNSUPPORTEDOPERATION;
654 static HRESULT WINAPI IcnsEncoder_CreateNewFrame(IWICBitmapEncoder *iface,
655 IWICBitmapFrameEncode **ppIFrameEncode, IPropertyBag2 **ppIEncoderOptions)
657 IcnsEncoder *This = impl_from_IWICBitmapEncoder(iface);
659 IcnsFrameEncode *frameEncode = NULL;
661 TRACE("(%p,%p,%p)\n", iface, ppIFrameEncode, ppIEncoderOptions);
663 EnterCriticalSection(&This->lock);
665 if (!This->icns_family)
667 hr = WINCODEC_ERR_NOTINITIALIZED;
671 hr = CreatePropertyBag2(ppIEncoderOptions);
675 frameEncode = HeapAlloc(GetProcessHeap(), 0, sizeof(IcnsFrameEncode));
676 if (frameEncode == NULL)
681 frameEncode->IWICBitmapFrameEncode_iface.lpVtbl = &IcnsEncoder_FrameVtbl;
682 frameEncode->encoder = This;
683 frameEncode->ref = 1;
684 frameEncode->initialized = FALSE;
685 frameEncode->width = 0;
686 frameEncode->height = 0;
687 memset(&frameEncode->icns_image, 0, sizeof(icns_image_t));
688 frameEncode->lines_written = 0;
689 frameEncode->committed = FALSE;
690 *ppIFrameEncode = &frameEncode->IWICBitmapFrameEncode_iface;
691 This->outstanding_commits++;
692 IUnknown_AddRef((IUnknown*)This);
695 LeaveCriticalSection(&This->lock);
700 static HRESULT WINAPI IcnsEncoder_Commit(IWICBitmapEncoder *iface)
702 IcnsEncoder *This = impl_from_IWICBitmapEncoder(iface);
703 icns_byte_t *buffer = NULL;
704 icns_size_t buffer_size;
709 TRACE("(%p)\n", iface);
711 EnterCriticalSection(&This->lock);
713 if (!This->any_frame_committed || This->outstanding_commits > 0 || This->committed)
715 hr = WINCODEC_ERR_WRONGSTATE;
719 ret = picns_export_family_data(This->icns_family, &buffer_size, &buffer);
720 if (ret != ICNS_STATUS_OK)
722 WARN("icns_export_family_data failed with error %d\n", ret);
726 hr = IStream_Write(This->stream, buffer, buffer_size, &byteswritten);
727 if (FAILED(hr) || byteswritten != buffer_size)
729 WARN("writing file failed, hr = 0x%08X\n", hr);
734 This->committed = TRUE;
737 LeaveCriticalSection(&This->lock);
742 static HRESULT WINAPI IcnsEncoder_GetMetadataQueryWriter(IWICBitmapEncoder *iface,
743 IWICMetadataQueryWriter **ppIMetadataQueryWriter)
745 FIXME("(%p,%p): stub\n", iface, ppIMetadataQueryWriter);
749 static const IWICBitmapEncoderVtbl IcnsEncoder_Vtbl = {
750 IcnsEncoder_QueryInterface,
753 IcnsEncoder_Initialize,
754 IcnsEncoder_GetContainerFormat,
755 IcnsEncoder_GetEncoderInfo,
756 IcnsEncoder_SetColorContexts,
757 IcnsEncoder_SetPalette,
758 IcnsEncoder_SetThumbnail,
759 IcnsEncoder_SetPreview,
760 IcnsEncoder_CreateNewFrame,
762 IcnsEncoder_GetMetadataQueryWriter
765 HRESULT IcnsEncoder_CreateInstance(IUnknown *pUnkOuter, REFIID iid, void** ppv)
770 TRACE("(%p,%s,%p)\n", pUnkOuter, debugstr_guid(iid), ppv);
774 if (pUnkOuter) return CLASS_E_NOAGGREGATION;
776 if (!libicns_handle && !load_libicns())
778 ERR("Failed writing ICNS because unable to find %s\n",SONAME_LIBICNS);
782 This = HeapAlloc(GetProcessHeap(), 0, sizeof(IcnsEncoder));
783 if (!This) return E_OUTOFMEMORY;
785 This->IWICBitmapEncoder_iface.lpVtbl = &IcnsEncoder_Vtbl;
788 This->icns_family = NULL;
789 This->any_frame_committed = FALSE;
790 This->outstanding_commits = 0;
791 This->committed = FALSE;
792 InitializeCriticalSection(&This->lock);
793 This->lock.DebugInfo->Spare[0] = (DWORD_PTR)(__FILE__ ": IcnsEncoder.lock");
795 ret = IUnknown_QueryInterface((IUnknown*)This, iid, ppv);
796 IUnknown_Release((IUnknown*)This);
801 #else /* !defined(SONAME_LIBICNS) */
803 HRESULT IcnsEncoder_CreateInstance(IUnknown *pUnkOuter, REFIID iid, void** ppv)
805 ERR("Trying to save ICNS picture, but ICNS support is not compiled in.\n");