xref: /reactos/dll/win32/windowscodecs/gifformat.c (revision 5bfe6a53)
1 /*
2  * Copyright 2009 Vincent Povirk for CodeWeavers
3  * Copyright 2012,2016 Dmitry Timoshkov
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 #define NONAMELESSUNION
26 
27 #include "windef.h"
28 #include "winbase.h"
29 #include "objbase.h"
30 
31 #include "ungif.h"
32 
33 #include "wincodecs_private.h"
34 
35 #include "wine/debug.h"
36 
37 #ifdef __REACTOS__
38 #include <ole2.h>
39 #endif
40 
41 WINE_DEFAULT_DEBUG_CHANNEL(wincodecs);
42 
43 #include "pshpack1.h"
44 
45 struct logical_screen_descriptor
46 {
47     char signature[6];
48     USHORT width;
49     USHORT height;
50     BYTE packed;
51     /* global_color_table_flag : 1;
52      * color_resolution : 3;
53      * sort_flag : 1;
54      * global_color_table_size : 3;
55      */
56     BYTE background_color_index;
57     BYTE pixel_aspect_ratio;
58 };
59 
60 struct image_descriptor
61 {
62     USHORT left;
63     USHORT top;
64     USHORT width;
65     USHORT height;
66     BYTE packed;
67     /* local_color_table_flag : 1;
68      * interlace_flag : 1;
69      * sort_flag : 1;
70      * reserved : 2;
71      * local_color_table_size : 3;
72      */
73 };
74 
75 #include "poppack.h"
76 
77 static LPWSTR strdupAtoW(const char *src)
78 {
79     int len = MultiByteToWideChar(CP_ACP, 0, src, -1, NULL, 0);
80     LPWSTR dst = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR));
81     if (dst) MultiByteToWideChar(CP_ACP, 0, src, -1, dst, len);
82     return dst;
83 }
84 
85 static HRESULT load_LSD_metadata(IStream *stream, const GUID *vendor, DWORD options,
86                                  MetadataItem **items, DWORD *count)
87 {
88     struct logical_screen_descriptor lsd_data;
89     HRESULT hr;
90     ULONG bytesread, i;
91     MetadataItem *result;
92 
93     *items = NULL;
94     *count = 0;
95 
96     hr = IStream_Read(stream, &lsd_data, sizeof(lsd_data), &bytesread);
97     if (FAILED(hr) || bytesread != sizeof(lsd_data)) return S_OK;
98 
99     result = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(MetadataItem) * 9);
100     if (!result) return E_OUTOFMEMORY;
101 
102     for (i = 0; i < 9; i++)
103     {
104         PropVariantInit(&result[i].schema);
105         PropVariantInit(&result[i].id);
106         PropVariantInit(&result[i].value);
107     }
108 
109     result[0].id.vt = VT_LPWSTR;
110     result[0].id.u.pwszVal = strdupAtoW("Signature");
111     result[0].value.vt = VT_UI1|VT_VECTOR;
112     result[0].value.u.caub.cElems = sizeof(lsd_data.signature);
113     result[0].value.u.caub.pElems = HeapAlloc(GetProcessHeap(), 0, sizeof(lsd_data.signature));
114     memcpy(result[0].value.u.caub.pElems, lsd_data.signature, sizeof(lsd_data.signature));
115 
116     result[1].id.vt = VT_LPWSTR;
117     result[1].id.u.pwszVal = strdupAtoW("Width");
118     result[1].value.vt = VT_UI2;
119     result[1].value.u.uiVal = lsd_data.width;
120 
121     result[2].id.vt = VT_LPWSTR;
122     result[2].id.u.pwszVal = strdupAtoW("Height");
123     result[2].value.vt = VT_UI2;
124     result[2].value.u.uiVal = lsd_data.height;
125 
126     result[3].id.vt = VT_LPWSTR;
127     result[3].id.u.pwszVal = strdupAtoW("GlobalColorTableFlag");
128     result[3].value.vt = VT_BOOL;
129     result[3].value.u.boolVal = (lsd_data.packed >> 7) & 1;
130 
131     result[4].id.vt = VT_LPWSTR;
132     result[4].id.u.pwszVal = strdupAtoW("ColorResolution");
133     result[4].value.vt = VT_UI1;
134     result[4].value.u.bVal = (lsd_data.packed >> 4) & 7;
135 
136     result[5].id.vt = VT_LPWSTR;
137     result[5].id.u.pwszVal = strdupAtoW("SortFlag");
138     result[5].value.vt = VT_BOOL;
139     result[5].value.u.boolVal = (lsd_data.packed >> 3) & 1;
140 
141     result[6].id.vt = VT_LPWSTR;
142     result[6].id.u.pwszVal = strdupAtoW("GlobalColorTableSize");
143     result[6].value.vt = VT_UI1;
144     result[6].value.u.bVal = lsd_data.packed & 7;
145 
146     result[7].id.vt = VT_LPWSTR;
147     result[7].id.u.pwszVal = strdupAtoW("BackgroundColorIndex");
148     result[7].value.vt = VT_UI1;
149     result[7].value.u.bVal = lsd_data.background_color_index;
150 
151     result[8].id.vt = VT_LPWSTR;
152     result[8].id.u.pwszVal = strdupAtoW("PixelAspectRatio");
153     result[8].value.vt = VT_UI1;
154     result[8].value.u.bVal = lsd_data.pixel_aspect_ratio;
155 
156     *items = result;
157     *count = 9;
158 
159     return S_OK;
160 }
161 
162 static const MetadataHandlerVtbl LSDReader_Vtbl = {
163     0,
164     &CLSID_WICLSDMetadataReader,
165     load_LSD_metadata
166 };
167 
168 HRESULT LSDReader_CreateInstance(REFIID iid, void **ppv)
169 {
170     return MetadataReader_Create(&LSDReader_Vtbl, iid, ppv);
171 }
172 
173 static HRESULT load_IMD_metadata(IStream *stream, const GUID *vendor, DWORD options,
174                                  MetadataItem **items, DWORD *count)
175 {
176     struct image_descriptor imd_data;
177     HRESULT hr;
178     ULONG bytesread, i;
179     MetadataItem *result;
180 
181     *items = NULL;
182     *count = 0;
183 
184     hr = IStream_Read(stream, &imd_data, sizeof(imd_data), &bytesread);
185     if (FAILED(hr) || bytesread != sizeof(imd_data)) return S_OK;
186 
187     result = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(MetadataItem) * 8);
188     if (!result) return E_OUTOFMEMORY;
189 
190     for (i = 0; i < 8; i++)
191     {
192         PropVariantInit(&result[i].schema);
193         PropVariantInit(&result[i].id);
194         PropVariantInit(&result[i].value);
195     }
196 
197     result[0].id.vt = VT_LPWSTR;
198     result[0].id.u.pwszVal = strdupAtoW("Left");
199     result[0].value.vt = VT_UI2;
200     result[0].value.u.uiVal = imd_data.left;
201 
202     result[1].id.vt = VT_LPWSTR;
203     result[1].id.u.pwszVal = strdupAtoW("Top");
204     result[1].value.vt = VT_UI2;
205     result[1].value.u.uiVal = imd_data.top;
206 
207     result[2].id.vt = VT_LPWSTR;
208     result[2].id.u.pwszVal = strdupAtoW("Width");
209     result[2].value.vt = VT_UI2;
210     result[2].value.u.uiVal = imd_data.width;
211 
212     result[3].id.vt = VT_LPWSTR;
213     result[3].id.u.pwszVal = strdupAtoW("Height");
214     result[3].value.vt = VT_UI2;
215     result[3].value.u.uiVal = imd_data.height;
216 
217     result[4].id.vt = VT_LPWSTR;
218     result[4].id.u.pwszVal = strdupAtoW("LocalColorTableFlag");
219     result[4].value.vt = VT_BOOL;
220     result[4].value.u.boolVal = (imd_data.packed >> 7) & 1;
221 
222     result[5].id.vt = VT_LPWSTR;
223     result[5].id.u.pwszVal = strdupAtoW("InterlaceFlag");
224     result[5].value.vt = VT_BOOL;
225     result[5].value.u.boolVal = (imd_data.packed >> 6) & 1;
226 
227     result[6].id.vt = VT_LPWSTR;
228     result[6].id.u.pwszVal = strdupAtoW("SortFlag");
229     result[6].value.vt = VT_BOOL;
230     result[6].value.u.boolVal = (imd_data.packed >> 5) & 1;
231 
232     result[7].id.vt = VT_LPWSTR;
233     result[7].id.u.pwszVal = strdupAtoW("LocalColorTableSize");
234     result[7].value.vt = VT_UI1;
235     result[7].value.u.bVal = imd_data.packed & 7;
236 
237     *items = result;
238     *count = 8;
239 
240     return S_OK;
241 }
242 
243 static const MetadataHandlerVtbl IMDReader_Vtbl = {
244     0,
245     &CLSID_WICIMDMetadataReader,
246     load_IMD_metadata
247 };
248 
249 HRESULT IMDReader_CreateInstance(REFIID iid, void **ppv)
250 {
251     return MetadataReader_Create(&IMDReader_Vtbl, iid, ppv);
252 }
253 
254 static HRESULT load_GCE_metadata(IStream *stream, const GUID *vendor, DWORD options,
255                                  MetadataItem **items, DWORD *count)
256 {
257 #include "pshpack1.h"
258     struct graphic_control_extension
259     {
260         BYTE packed;
261         /* reservred: 3;
262          * disposal : 3;
263          * user_input_flag : 1;
264          * transparency_flag : 1;
265          */
266          USHORT delay;
267          BYTE transparent_color_index;
268     } gce_data;
269 #include "poppack.h"
270     HRESULT hr;
271     ULONG bytesread, i;
272     MetadataItem *result;
273 
274     *items = NULL;
275     *count = 0;
276 
277     hr = IStream_Read(stream, &gce_data, sizeof(gce_data), &bytesread);
278     if (FAILED(hr) || bytesread != sizeof(gce_data)) return S_OK;
279 
280     result = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(MetadataItem) * 5);
281     if (!result) return E_OUTOFMEMORY;
282 
283     for (i = 0; i < 5; i++)
284     {
285         PropVariantInit(&result[i].schema);
286         PropVariantInit(&result[i].id);
287         PropVariantInit(&result[i].value);
288     }
289 
290     result[0].id.vt = VT_LPWSTR;
291     result[0].id.u.pwszVal = strdupAtoW("Disposal");
292     result[0].value.vt = VT_UI1;
293     result[0].value.u.bVal = (gce_data.packed >> 2) & 7;
294 
295     result[1].id.vt = VT_LPWSTR;
296     result[1].id.u.pwszVal = strdupAtoW("UserInputFlag");
297     result[1].value.vt = VT_BOOL;
298     result[1].value.u.boolVal = (gce_data.packed >> 1) & 1;
299 
300     result[2].id.vt = VT_LPWSTR;
301     result[2].id.u.pwszVal = strdupAtoW("TransparencyFlag");
302     result[2].value.vt = VT_BOOL;
303     result[2].value.u.boolVal = gce_data.packed & 1;
304 
305     result[3].id.vt = VT_LPWSTR;
306     result[3].id.u.pwszVal = strdupAtoW("Delay");
307     result[3].value.vt = VT_UI2;
308     result[3].value.u.uiVal = gce_data.delay;
309 
310     result[4].id.vt = VT_LPWSTR;
311     result[4].id.u.pwszVal = strdupAtoW("TransparentColorIndex");
312     result[4].value.vt = VT_UI1;
313     result[4].value.u.bVal = gce_data.transparent_color_index;
314 
315     *items = result;
316     *count = 5;
317 
318     return S_OK;
319 }
320 
321 static const MetadataHandlerVtbl GCEReader_Vtbl = {
322     0,
323     &CLSID_WICGCEMetadataReader,
324     load_GCE_metadata
325 };
326 
327 HRESULT GCEReader_CreateInstance(REFIID iid, void **ppv)
328 {
329     return MetadataReader_Create(&GCEReader_Vtbl, iid, ppv);
330 }
331 
332 static HRESULT load_APE_metadata(IStream *stream, const GUID *vendor, DWORD options,
333                                  MetadataItem **items, DWORD *count)
334 {
335 #include "pshpack1.h"
336     struct application_extension
337     {
338         BYTE extension_introducer;
339         BYTE extension_label;
340         BYTE block_size;
341         BYTE application[11];
342     } ape_data;
343 #include "poppack.h"
344     HRESULT hr;
345     ULONG bytesread, data_size, i;
346     MetadataItem *result;
347     BYTE subblock_size;
348     BYTE *data;
349 
350     *items = NULL;
351     *count = 0;
352 
353     hr = IStream_Read(stream, &ape_data, sizeof(ape_data), &bytesread);
354     if (FAILED(hr) || bytesread != sizeof(ape_data)) return S_OK;
355     if (ape_data.extension_introducer != 0x21 ||
356         ape_data.extension_label != APPLICATION_EXT_FUNC_CODE ||
357         ape_data.block_size != 11)
358         return S_OK;
359 
360     data = NULL;
361     data_size = 0;
362 
363     for (;;)
364     {
365         hr = IStream_Read(stream, &subblock_size, sizeof(subblock_size), &bytesread);
366         if (FAILED(hr) || bytesread != sizeof(subblock_size))
367         {
368             HeapFree(GetProcessHeap(), 0, data);
369             return S_OK;
370         }
371         if (!subblock_size) break;
372 
373         if (!data)
374             data = HeapAlloc(GetProcessHeap(), 0, subblock_size + 1);
375         else
376         {
377             BYTE *new_data = HeapReAlloc(GetProcessHeap(), 0, data, data_size + subblock_size + 1);
378             if (!new_data)
379             {
380                 HeapFree(GetProcessHeap(), 0, data);
381                 return S_OK;
382             }
383             data = new_data;
384         }
385         data[data_size] = subblock_size;
386         hr = IStream_Read(stream, data + data_size + 1, subblock_size, &bytesread);
387         if (FAILED(hr) || bytesread != subblock_size)
388         {
389             HeapFree(GetProcessHeap(), 0, data);
390             return S_OK;
391         }
392         data_size += subblock_size + 1;
393     }
394 
395     result = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(MetadataItem) * 2);
396     if (!result)
397     {
398         HeapFree(GetProcessHeap(), 0, data);
399         return E_OUTOFMEMORY;
400     }
401 
402     for (i = 0; i < 2; i++)
403     {
404         PropVariantInit(&result[i].schema);
405         PropVariantInit(&result[i].id);
406         PropVariantInit(&result[i].value);
407     }
408 
409     result[0].id.vt = VT_LPWSTR;
410     result[0].id.u.pwszVal = strdupAtoW("Application");
411     result[0].value.vt = VT_UI1|VT_VECTOR;
412     result[0].value.u.caub.cElems = sizeof(ape_data.application);
413     result[0].value.u.caub.pElems = HeapAlloc(GetProcessHeap(), 0, sizeof(ape_data.application));
414     memcpy(result[0].value.u.caub.pElems, ape_data.application, sizeof(ape_data.application));
415 
416     result[1].id.vt = VT_LPWSTR;
417     result[1].id.u.pwszVal = strdupAtoW("Data");
418     result[1].value.vt = VT_UI1|VT_VECTOR;
419     result[1].value.u.caub.cElems = data_size;
420     result[1].value.u.caub.pElems = data;
421 
422     *items = result;
423     *count = 2;
424 
425     return S_OK;
426 }
427 
428 static const MetadataHandlerVtbl APEReader_Vtbl = {
429     0,
430     &CLSID_WICAPEMetadataReader,
431     load_APE_metadata
432 };
433 
434 HRESULT APEReader_CreateInstance(REFIID iid, void **ppv)
435 {
436     return MetadataReader_Create(&APEReader_Vtbl, iid, ppv);
437 }
438 
439 static HRESULT load_GifComment_metadata(IStream *stream, const GUID *vendor, DWORD options,
440                                         MetadataItem **items, DWORD *count)
441 {
442 #include "pshpack1.h"
443     struct gif_extension
444     {
445         BYTE extension_introducer;
446         BYTE extension_label;
447     } ext_data;
448 #include "poppack.h"
449     HRESULT hr;
450     ULONG bytesread, data_size;
451     MetadataItem *result;
452     BYTE subblock_size;
453     char *data;
454 
455     *items = NULL;
456     *count = 0;
457 
458     hr = IStream_Read(stream, &ext_data, sizeof(ext_data), &bytesread);
459     if (FAILED(hr) || bytesread != sizeof(ext_data)) return S_OK;
460     if (ext_data.extension_introducer != 0x21 ||
461         ext_data.extension_label != COMMENT_EXT_FUNC_CODE)
462         return S_OK;
463 
464     data = NULL;
465     data_size = 0;
466 
467     for (;;)
468     {
469         hr = IStream_Read(stream, &subblock_size, sizeof(subblock_size), &bytesread);
470         if (FAILED(hr) || bytesread != sizeof(subblock_size))
471         {
472             HeapFree(GetProcessHeap(), 0, data);
473             return S_OK;
474         }
475         if (!subblock_size) break;
476 
477         if (!data)
478             data = HeapAlloc(GetProcessHeap(), 0, subblock_size + 1);
479         else
480         {
481             char *new_data = HeapReAlloc(GetProcessHeap(), 0, data, data_size + subblock_size + 1);
482             if (!new_data)
483             {
484                 HeapFree(GetProcessHeap(), 0, data);
485                 return S_OK;
486             }
487             data = new_data;
488         }
489         hr = IStream_Read(stream, data + data_size, subblock_size, &bytesread);
490         if (FAILED(hr) || bytesread != subblock_size)
491         {
492             HeapFree(GetProcessHeap(), 0, data);
493             return S_OK;
494         }
495         data_size += subblock_size;
496     }
497 
498     data[data_size] = 0;
499 
500     result = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(MetadataItem));
501     if (!result)
502     {
503         HeapFree(GetProcessHeap(), 0, data);
504         return E_OUTOFMEMORY;
505     }
506 
507     PropVariantInit(&result->schema);
508     PropVariantInit(&result->id);
509     PropVariantInit(&result->value);
510 
511     result->id.vt = VT_LPWSTR;
512     result->id.u.pwszVal = strdupAtoW("TextEntry");
513     result->value.vt = VT_LPSTR;
514     result->value.u.pszVal = data;
515 
516     *items = result;
517     *count = 1;
518 
519     return S_OK;
520 }
521 
522 static const MetadataHandlerVtbl GifCommentReader_Vtbl = {
523     0,
524     &CLSID_WICGifCommentMetadataReader,
525     load_GifComment_metadata
526 };
527 
528 HRESULT GifCommentReader_CreateInstance(REFIID iid, void **ppv)
529 {
530     return MetadataReader_Create(&GifCommentReader_Vtbl, iid, ppv);
531 }
532 
533 static IStream *create_stream(const void *data, int data_size)
534 {
535     HRESULT hr;
536     IStream *stream;
537     HGLOBAL hdata;
538     void *locked_data;
539 
540     hdata = GlobalAlloc(GMEM_MOVEABLE, data_size);
541     if (!hdata) return NULL;
542 
543     locked_data = GlobalLock(hdata);
544     memcpy(locked_data, data, data_size);
545     GlobalUnlock(hdata);
546 
547     hr = CreateStreamOnHGlobal(hdata, TRUE, &stream);
548     return FAILED(hr) ? NULL : stream;
549 }
550 
551 static HRESULT create_metadata_reader(const void *data, int data_size,
552                                       class_constructor constructor,
553                                       IWICMetadataReader **reader)
554 {
555     HRESULT hr;
556     IWICMetadataReader *metadata_reader;
557     IWICPersistStream *persist;
558     IStream *stream;
559 
560     /* FIXME: Use IWICComponentFactory_CreateMetadataReader once it's implemented */
561 
562     hr = constructor(&IID_IWICMetadataReader, (void**)&metadata_reader);
563     if (FAILED(hr)) return hr;
564 
565     hr = IWICMetadataReader_QueryInterface(metadata_reader, &IID_IWICPersistStream, (void **)&persist);
566     if (FAILED(hr))
567     {
568         IWICMetadataReader_Release(metadata_reader);
569         return hr;
570     }
571 
572     stream = create_stream(data, data_size);
573     IWICPersistStream_LoadEx(persist, stream, NULL, WICPersistOptionDefault);
574     IStream_Release(stream);
575 
576     IWICPersistStream_Release(persist);
577 
578     *reader = metadata_reader;
579     return S_OK;
580 }
581 
582 typedef struct {
583     IWICBitmapDecoder IWICBitmapDecoder_iface;
584     IWICMetadataBlockReader IWICMetadataBlockReader_iface;
585     IStream *stream;
586     BYTE LSD_data[13]; /* Logical Screen Descriptor */
587     LONG ref;
588     BOOL initialized;
589     GifFileType *gif;
590     UINT current_frame;
591     CRITICAL_SECTION lock;
592 } GifDecoder;
593 
594 typedef struct {
595     IWICBitmapFrameDecode IWICBitmapFrameDecode_iface;
596     IWICMetadataBlockReader IWICMetadataBlockReader_iface;
597     LONG ref;
598     SavedImage *frame;
599     GifDecoder *parent;
600 } GifFrameDecode;
601 
602 static inline GifDecoder *impl_from_IWICBitmapDecoder(IWICBitmapDecoder *iface)
603 {
604     return CONTAINING_RECORD(iface, GifDecoder, IWICBitmapDecoder_iface);
605 }
606 
607 static inline GifDecoder *impl_from_IWICMetadataBlockReader(IWICMetadataBlockReader *iface)
608 {
609     return CONTAINING_RECORD(iface, GifDecoder, IWICMetadataBlockReader_iface);
610 }
611 
612 static inline GifFrameDecode *impl_from_IWICBitmapFrameDecode(IWICBitmapFrameDecode *iface)
613 {
614     return CONTAINING_RECORD(iface, GifFrameDecode, IWICBitmapFrameDecode_iface);
615 }
616 
617 static inline GifFrameDecode *frame_from_IWICMetadataBlockReader(IWICMetadataBlockReader *iface)
618 {
619     return CONTAINING_RECORD(iface, GifFrameDecode, IWICMetadataBlockReader_iface);
620 }
621 
622 static HRESULT WINAPI GifFrameDecode_QueryInterface(IWICBitmapFrameDecode *iface, REFIID iid,
623     void **ppv)
624 {
625     GifFrameDecode *This = impl_from_IWICBitmapFrameDecode(iface);
626     TRACE("(%p,%s,%p)\n", iface, debugstr_guid(iid), ppv);
627 
628     if (!ppv) return E_INVALIDARG;
629 
630     if (IsEqualIID(&IID_IUnknown, iid) ||
631         IsEqualIID(&IID_IWICBitmapSource, iid) ||
632         IsEqualIID(&IID_IWICBitmapFrameDecode, iid))
633     {
634         *ppv = &This->IWICBitmapFrameDecode_iface;
635     }
636     else if (IsEqualIID(&IID_IWICMetadataBlockReader, iid))
637     {
638         *ppv = &This->IWICMetadataBlockReader_iface;
639     }
640     else
641     {
642         *ppv = NULL;
643         return E_NOINTERFACE;
644     }
645 
646     IUnknown_AddRef((IUnknown*)*ppv);
647     return S_OK;
648 }
649 
650 static ULONG WINAPI GifFrameDecode_AddRef(IWICBitmapFrameDecode *iface)
651 {
652     GifFrameDecode *This = impl_from_IWICBitmapFrameDecode(iface);
653     ULONG ref = InterlockedIncrement(&This->ref);
654 
655     TRACE("(%p) refcount=%u\n", iface, ref);
656 
657     return ref;
658 }
659 
660 static ULONG WINAPI GifFrameDecode_Release(IWICBitmapFrameDecode *iface)
661 {
662     GifFrameDecode *This = impl_from_IWICBitmapFrameDecode(iface);
663     ULONG ref = InterlockedDecrement(&This->ref);
664 
665     TRACE("(%p) refcount=%u\n", iface, ref);
666 
667     if (ref == 0)
668     {
669         IWICBitmapDecoder_Release(&This->parent->IWICBitmapDecoder_iface);
670         HeapFree(GetProcessHeap(), 0, This);
671     }
672 
673     return ref;
674 }
675 
676 static HRESULT WINAPI GifFrameDecode_GetSize(IWICBitmapFrameDecode *iface,
677     UINT *puiWidth, UINT *puiHeight)
678 {
679     GifFrameDecode *This = impl_from_IWICBitmapFrameDecode(iface);
680     TRACE("(%p,%p,%p)\n", iface, puiWidth, puiHeight);
681 
682     *puiWidth = This->frame->ImageDesc.Width;
683     *puiHeight = This->frame->ImageDesc.Height;
684 
685     return S_OK;
686 }
687 
688 static HRESULT WINAPI GifFrameDecode_GetPixelFormat(IWICBitmapFrameDecode *iface,
689     WICPixelFormatGUID *pPixelFormat)
690 {
691     memcpy(pPixelFormat, &GUID_WICPixelFormat8bppIndexed, sizeof(GUID));
692 
693     return S_OK;
694 }
695 
696 static HRESULT WINAPI GifFrameDecode_GetResolution(IWICBitmapFrameDecode *iface,
697     double *pDpiX, double *pDpiY)
698 {
699     GifFrameDecode *This = impl_from_IWICBitmapFrameDecode(iface);
700     const GifWord aspect_word = This->parent->gif->SAspectRatio;
701     const double aspect = (aspect_word > 0) ? ((aspect_word + 15.0) / 64.0) : 1.0;
702     TRACE("(%p,%p,%p)\n", iface, pDpiX, pDpiY);
703 
704     *pDpiX = 96.0 / aspect;
705     *pDpiY = 96.0;
706 
707     return S_OK;
708 }
709 
710 static HRESULT WINAPI GifFrameDecode_CopyPalette(IWICBitmapFrameDecode *iface,
711     IWICPalette *pIPalette)
712 {
713     GifFrameDecode *This = impl_from_IWICBitmapFrameDecode(iface);
714     WICColor colors[256];
715     ColorMapObject *cm = This->frame->ImageDesc.ColorMap;
716     int i, trans;
717     ExtensionBlock *eb;
718     TRACE("(%p,%p)\n", iface, pIPalette);
719 
720     if (!cm) cm = This->parent->gif->SColorMap;
721 
722     if (cm->ColorCount > 256)
723     {
724         ERR("GIF contains %i colors???\n", cm->ColorCount);
725         return E_FAIL;
726     }
727 
728     for (i = 0; i < cm->ColorCount; i++) {
729         colors[i] = 0xff000000| /* alpha */
730                     cm->Colors[i].Red << 16|
731                     cm->Colors[i].Green << 8|
732                     cm->Colors[i].Blue;
733     }
734 
735     /* look for the transparent color extension */
736     for (i = 0; i < This->frame->Extensions.ExtensionBlockCount; ++i) {
737 	eb = This->frame->Extensions.ExtensionBlocks + i;
738 	if (eb->Function == GRAPHICS_EXT_FUNC_CODE && eb->ByteCount == 8) {
739 	    if (eb->Bytes[3] & 1) {
740 	        trans = (unsigned char)eb->Bytes[6];
741 	        colors[trans] &= 0xffffff; /* set alpha to 0 */
742 	        break;
743 	    }
744 	}
745     }
746 
747     return IWICPalette_InitializeCustom(pIPalette, colors, cm->ColorCount);
748 }
749 
750 static HRESULT copy_interlaced_pixels(const BYTE *srcbuffer,
751     UINT srcwidth, UINT srcheight, INT srcstride, const WICRect *rc,
752     UINT dststride, UINT dstbuffersize, BYTE *dstbuffer)
753 {
754     UINT row_offset; /* number of bytes into the source rows where the data starts */
755     const BYTE *src;
756     BYTE *dst;
757     UINT y;
758     WICRect rect;
759 
760     if (!rc)
761     {
762         rect.X = 0;
763         rect.Y = 0;
764         rect.Width = srcwidth;
765         rect.Height = srcheight;
766         rc = &rect;
767     }
768     else
769     {
770         if (rc->X < 0 || rc->Y < 0 || rc->X+rc->Width > srcwidth || rc->Y+rc->Height > srcheight)
771             return E_INVALIDARG;
772     }
773 
774     if (dststride < rc->Width)
775         return E_INVALIDARG;
776 
777     if ((dststride * rc->Height) > dstbuffersize)
778         return E_INVALIDARG;
779 
780     row_offset = rc->X;
781 
782     dst = dstbuffer;
783     for (y=rc->Y; y-rc->Y < rc->Height; y++)
784     {
785         if (y%8 == 0)
786             src = srcbuffer + srcstride * (y/8);
787         else if (y%4 == 0)
788             src = srcbuffer + srcstride * ((srcheight+7)/8 + y/8);
789         else if (y%2 == 0)
790             src = srcbuffer + srcstride * ((srcheight+3)/4 + y/4);
791         else /* y%2 == 1 */
792             src = srcbuffer + srcstride * ((srcheight+1)/2 + y/2);
793         src += row_offset;
794         memcpy(dst, src, rc->Width);
795         dst += dststride;
796     }
797     return S_OK;
798 }
799 
800 static HRESULT WINAPI GifFrameDecode_CopyPixels(IWICBitmapFrameDecode *iface,
801     const WICRect *prc, UINT cbStride, UINT cbBufferSize, BYTE *pbBuffer)
802 {
803     GifFrameDecode *This = impl_from_IWICBitmapFrameDecode(iface);
804     TRACE("(%p,%s,%u,%u,%p)\n", iface, debug_wic_rect(prc), cbStride, cbBufferSize, pbBuffer);
805 
806     if (This->frame->ImageDesc.Interlace)
807     {
808         return copy_interlaced_pixels(This->frame->RasterBits, This->frame->ImageDesc.Width,
809             This->frame->ImageDesc.Height, This->frame->ImageDesc.Width,
810             prc, cbStride, cbBufferSize, pbBuffer);
811     }
812     else
813     {
814         return copy_pixels(8, This->frame->RasterBits, This->frame->ImageDesc.Width,
815             This->frame->ImageDesc.Height, This->frame->ImageDesc.Width,
816             prc, cbStride, cbBufferSize, pbBuffer);
817     }
818 }
819 
820 static HRESULT WINAPI GifFrameDecode_GetMetadataQueryReader(IWICBitmapFrameDecode *iface,
821     IWICMetadataQueryReader **ppIMetadataQueryReader)
822 {
823     GifFrameDecode *This = impl_from_IWICBitmapFrameDecode(iface);
824 
825     TRACE("(%p,%p)\n", iface, ppIMetadataQueryReader);
826 
827     if (!ppIMetadataQueryReader)
828         return E_INVALIDARG;
829 
830     return MetadataQueryReader_CreateInstance(&This->IWICMetadataBlockReader_iface, NULL, ppIMetadataQueryReader);
831 }
832 
833 static HRESULT WINAPI GifFrameDecode_GetColorContexts(IWICBitmapFrameDecode *iface,
834     UINT cCount, IWICColorContext **ppIColorContexts, UINT *pcActualCount)
835 {
836     TRACE("(%p,%u,%p,%p)\n", iface, cCount, ppIColorContexts, pcActualCount);
837     return WINCODEC_ERR_UNSUPPORTEDOPERATION;
838 }
839 
840 static HRESULT WINAPI GifFrameDecode_GetThumbnail(IWICBitmapFrameDecode *iface,
841     IWICBitmapSource **ppIThumbnail)
842 {
843     TRACE("(%p,%p)\n", iface, ppIThumbnail);
844     return WINCODEC_ERR_CODECNOTHUMBNAIL;
845 }
846 
847 static const IWICBitmapFrameDecodeVtbl GifFrameDecode_Vtbl = {
848     GifFrameDecode_QueryInterface,
849     GifFrameDecode_AddRef,
850     GifFrameDecode_Release,
851     GifFrameDecode_GetSize,
852     GifFrameDecode_GetPixelFormat,
853     GifFrameDecode_GetResolution,
854     GifFrameDecode_CopyPalette,
855     GifFrameDecode_CopyPixels,
856     GifFrameDecode_GetMetadataQueryReader,
857     GifFrameDecode_GetColorContexts,
858     GifFrameDecode_GetThumbnail
859 };
860 
861 static HRESULT WINAPI GifFrameDecode_Block_QueryInterface(IWICMetadataBlockReader *iface,
862     REFIID iid, void **ppv)
863 {
864     GifFrameDecode *This = frame_from_IWICMetadataBlockReader(iface);
865     return IWICBitmapFrameDecode_QueryInterface(&This->IWICBitmapFrameDecode_iface, iid, ppv);
866 }
867 
868 static ULONG WINAPI GifFrameDecode_Block_AddRef(IWICMetadataBlockReader *iface)
869 {
870     GifFrameDecode *This = frame_from_IWICMetadataBlockReader(iface);
871     return IWICBitmapFrameDecode_AddRef(&This->IWICBitmapFrameDecode_iface);
872 }
873 
874 static ULONG WINAPI GifFrameDecode_Block_Release(IWICMetadataBlockReader *iface)
875 {
876     GifFrameDecode *This = frame_from_IWICMetadataBlockReader(iface);
877     return IWICBitmapFrameDecode_Release(&This->IWICBitmapFrameDecode_iface);
878 }
879 
880 static HRESULT WINAPI GifFrameDecode_Block_GetContainerFormat(IWICMetadataBlockReader *iface,
881     GUID *guid)
882 {
883     TRACE("(%p,%p)\n", iface, guid);
884 
885     if (!guid) return E_INVALIDARG;
886 
887     *guid = GUID_ContainerFormatGif;
888     return S_OK;
889 }
890 
891 static HRESULT WINAPI GifFrameDecode_Block_GetCount(IWICMetadataBlockReader *iface,
892     UINT *count)
893 {
894     GifFrameDecode *This = frame_from_IWICMetadataBlockReader(iface);
895 
896     TRACE("%p,%p\n", iface, count);
897 
898     if (!count) return E_INVALIDARG;
899 
900     *count = This->frame->Extensions.ExtensionBlockCount + 1;
901     return S_OK;
902 }
903 
904 static HRESULT create_IMD_metadata_reader(GifFrameDecode *This, IWICMetadataReader **reader)
905 {
906     HRESULT hr;
907     IWICMetadataReader *metadata_reader;
908     IWICPersistStream *persist;
909     IStream *stream;
910     struct image_descriptor IMD_data;
911 
912     /* FIXME: Use IWICComponentFactory_CreateMetadataReader once it's implemented */
913 
914     hr = IMDReader_CreateInstance(&IID_IWICMetadataReader, (void **)&metadata_reader);
915     if (FAILED(hr)) return hr;
916 
917     hr = IWICMetadataReader_QueryInterface(metadata_reader, &IID_IWICPersistStream, (void **)&persist);
918     if (FAILED(hr))
919     {
920         IWICMetadataReader_Release(metadata_reader);
921         return hr;
922     }
923 
924     /* recreate IMD structure from GIF decoder data */
925     IMD_data.left = This->frame->ImageDesc.Left;
926     IMD_data.top = This->frame->ImageDesc.Top;
927     IMD_data.width = This->frame->ImageDesc.Width;
928     IMD_data.height = This->frame->ImageDesc.Height;
929     IMD_data.packed = 0;
930     /* interlace_flag */
931     IMD_data.packed |= This->frame->ImageDesc.Interlace ? (1 << 6) : 0;
932     if (This->frame->ImageDesc.ColorMap)
933     {
934         /* local_color_table_flag */
935         IMD_data.packed |= 1 << 7;
936         /* local_color_table_size */
937         IMD_data.packed |= This->frame->ImageDesc.ColorMap->BitsPerPixel - 1;
938         /* sort_flag */
939         IMD_data.packed |= This->frame->ImageDesc.ColorMap->SortFlag ? 0x20 : 0;
940     }
941 
942     stream = create_stream(&IMD_data, sizeof(IMD_data));
943     IWICPersistStream_LoadEx(persist, stream, NULL, WICPersistOptionDefault);
944     IStream_Release(stream);
945 
946     IWICPersistStream_Release(persist);
947 
948     *reader = metadata_reader;
949     return S_OK;
950 }
951 
952 static HRESULT WINAPI GifFrameDecode_Block_GetReaderByIndex(IWICMetadataBlockReader *iface,
953     UINT index, IWICMetadataReader **reader)
954 {
955     GifFrameDecode *This = frame_from_IWICMetadataBlockReader(iface);
956     int i, gce_index = -1, gce_skipped = 0;
957 
958     TRACE("(%p,%u,%p)\n", iface, index, reader);
959 
960     if (!reader) return E_INVALIDARG;
961 
962     if (index == 0)
963         return create_IMD_metadata_reader(This, reader);
964 
965     if (index >= This->frame->Extensions.ExtensionBlockCount + 1)
966         return E_INVALIDARG;
967 
968     for (i = 0; i < This->frame->Extensions.ExtensionBlockCount; i++)
969     {
970         class_constructor constructor;
971         const void *data;
972         int data_size;
973 
974         if (index != i + 1 - gce_skipped) continue;
975 
976         if (This->frame->Extensions.ExtensionBlocks[i].Function == GRAPHICS_EXT_FUNC_CODE)
977         {
978             gce_index = i;
979             gce_skipped = 1;
980             continue;
981         }
982         else if (This->frame->Extensions.ExtensionBlocks[i].Function == COMMENT_EXT_FUNC_CODE)
983         {
984             constructor = GifCommentReader_CreateInstance;
985             data = This->frame->Extensions.ExtensionBlocks[i].Bytes;
986             data_size = This->frame->Extensions.ExtensionBlocks[i].ByteCount;
987         }
988         else
989         {
990             constructor = UnknownMetadataReader_CreateInstance;
991             data = This->frame->Extensions.ExtensionBlocks[i].Bytes;
992             data_size = This->frame->Extensions.ExtensionBlocks[i].ByteCount;
993         }
994         return create_metadata_reader(data, data_size, constructor, reader);
995     }
996 
997     if (gce_index == -1) return E_INVALIDARG;
998 
999     return create_metadata_reader(This->frame->Extensions.ExtensionBlocks[gce_index].Bytes + 3,
1000                                   This->frame->Extensions.ExtensionBlocks[gce_index].ByteCount - 4,
1001                                   GCEReader_CreateInstance, reader);
1002 }
1003 
1004 static HRESULT WINAPI GifFrameDecode_Block_GetEnumerator(IWICMetadataBlockReader *iface,
1005     IEnumUnknown **enumerator)
1006 {
1007     FIXME("(%p,%p): stub\n", iface, enumerator);
1008     return E_NOTIMPL;
1009 }
1010 
1011 static const IWICMetadataBlockReaderVtbl GifFrameDecode_BlockVtbl =
1012 {
1013     GifFrameDecode_Block_QueryInterface,
1014     GifFrameDecode_Block_AddRef,
1015     GifFrameDecode_Block_Release,
1016     GifFrameDecode_Block_GetContainerFormat,
1017     GifFrameDecode_Block_GetCount,
1018     GifFrameDecode_Block_GetReaderByIndex,
1019     GifFrameDecode_Block_GetEnumerator
1020 };
1021 
1022 static HRESULT WINAPI GifDecoder_QueryInterface(IWICBitmapDecoder *iface, REFIID iid,
1023     void **ppv)
1024 {
1025     GifDecoder *This = impl_from_IWICBitmapDecoder(iface);
1026     TRACE("(%p,%s,%p)\n", iface, debugstr_guid(iid), ppv);
1027 
1028     if (!ppv) return E_INVALIDARG;
1029 
1030     if (IsEqualIID(&IID_IUnknown, iid) ||
1031         IsEqualIID(&IID_IWICBitmapDecoder, iid))
1032     {
1033         *ppv = &This->IWICBitmapDecoder_iface;
1034     }
1035     else if (IsEqualIID(&IID_IWICMetadataBlockReader, iid))
1036     {
1037         *ppv = &This->IWICMetadataBlockReader_iface;
1038     }
1039     else
1040     {
1041         *ppv = NULL;
1042         return E_NOINTERFACE;
1043     }
1044 
1045     IUnknown_AddRef((IUnknown*)*ppv);
1046     return S_OK;
1047 }
1048 
1049 static ULONG WINAPI GifDecoder_AddRef(IWICBitmapDecoder *iface)
1050 {
1051     GifDecoder *This = impl_from_IWICBitmapDecoder(iface);
1052     ULONG ref = InterlockedIncrement(&This->ref);
1053 
1054     TRACE("(%p) refcount=%u\n", iface, ref);
1055 
1056     return ref;
1057 }
1058 
1059 static ULONG WINAPI GifDecoder_Release(IWICBitmapDecoder *iface)
1060 {
1061     GifDecoder *This = impl_from_IWICBitmapDecoder(iface);
1062     ULONG ref = InterlockedDecrement(&This->ref);
1063 
1064     TRACE("(%p) refcount=%u\n", iface, ref);
1065 
1066     if (ref == 0)
1067     {
1068         if (This->stream)
1069         {
1070             IStream_Release(This->stream);
1071             DGifCloseFile(This->gif);
1072         }
1073         This->lock.DebugInfo->Spare[0] = 0;
1074         DeleteCriticalSection(&This->lock);
1075         HeapFree(GetProcessHeap(), 0, This);
1076     }
1077 
1078     return ref;
1079 }
1080 
1081 static HRESULT WINAPI GifDecoder_QueryCapability(IWICBitmapDecoder *iface, IStream *stream,
1082     DWORD *capability)
1083 {
1084     HRESULT hr;
1085 
1086     TRACE("(%p,%p,%p)\n", iface, stream, capability);
1087 
1088     if (!stream || !capability) return E_INVALIDARG;
1089 
1090     hr = IWICBitmapDecoder_Initialize(iface, stream, WICDecodeMetadataCacheOnDemand);
1091     if (hr != S_OK) return hr;
1092 
1093     *capability = WICBitmapDecoderCapabilityCanDecodeAllImages |
1094                   WICBitmapDecoderCapabilityCanDecodeSomeImages |
1095                   WICBitmapDecoderCapabilityCanEnumerateMetadata;
1096     return S_OK;
1097 }
1098 
1099 static int _gif_inputfunc(GifFileType *gif, GifByteType *data, int len) {
1100     IStream *stream = gif->UserData;
1101     ULONG bytesread;
1102     HRESULT hr;
1103 
1104     if (!stream)
1105     {
1106         ERR("attempting to read file after initialization\n");
1107         return 0;
1108     }
1109 
1110     hr = IStream_Read(stream, data, len, &bytesread);
1111     if (FAILED(hr)) bytesread = 0;
1112     return bytesread;
1113 }
1114 
1115 static HRESULT WINAPI GifDecoder_Initialize(IWICBitmapDecoder *iface, IStream *pIStream,
1116     WICDecodeOptions cacheOptions)
1117 {
1118     GifDecoder *This = impl_from_IWICBitmapDecoder(iface);
1119     LARGE_INTEGER seek;
1120     int ret;
1121 
1122     TRACE("(%p,%p,%x)\n", iface, pIStream, cacheOptions);
1123 
1124     EnterCriticalSection(&This->lock);
1125 
1126     if (This->initialized || This->gif)
1127     {
1128         WARN("already initialized\n");
1129         LeaveCriticalSection(&This->lock);
1130         return WINCODEC_ERR_WRONGSTATE;
1131     }
1132 
1133     /* seek to start of stream */
1134     seek.QuadPart = 0;
1135     IStream_Seek(pIStream, seek, STREAM_SEEK_SET, NULL);
1136 
1137     /* read all data from the stream */
1138     This->gif = DGifOpen((void*)pIStream, _gif_inputfunc);
1139     if (!This->gif)
1140     {
1141         LeaveCriticalSection(&This->lock);
1142         return E_FAIL;
1143     }
1144 
1145     ret = DGifSlurp(This->gif);
1146     if (ret == GIF_ERROR)
1147     {
1148         LeaveCriticalSection(&This->lock);
1149         return E_FAIL;
1150     }
1151 
1152     /* make sure we don't use the stream after this method returns */
1153     This->gif->UserData = NULL;
1154 
1155     seek.QuadPart = 0;
1156     IStream_Seek(pIStream, seek, STREAM_SEEK_SET, NULL);
1157     IStream_Read(pIStream, This->LSD_data, sizeof(This->LSD_data), NULL);
1158 
1159     This->stream = pIStream;
1160     IStream_AddRef(This->stream);
1161 
1162     This->initialized = TRUE;
1163 
1164     LeaveCriticalSection(&This->lock);
1165 
1166     return S_OK;
1167 }
1168 
1169 static HRESULT WINAPI GifDecoder_GetContainerFormat(IWICBitmapDecoder *iface,
1170     GUID *pguidContainerFormat)
1171 {
1172     memcpy(pguidContainerFormat, &GUID_ContainerFormatGif, sizeof(GUID));
1173     return S_OK;
1174 }
1175 
1176 static HRESULT WINAPI GifDecoder_GetDecoderInfo(IWICBitmapDecoder *iface,
1177     IWICBitmapDecoderInfo **ppIDecoderInfo)
1178 {
1179     TRACE("(%p,%p)\n", iface, ppIDecoderInfo);
1180 
1181     return get_decoder_info(&CLSID_WICGifDecoder, ppIDecoderInfo);
1182 }
1183 
1184 static HRESULT WINAPI GifDecoder_CopyPalette(IWICBitmapDecoder *iface, IWICPalette *palette)
1185 {
1186     GifDecoder *This = impl_from_IWICBitmapDecoder(iface);
1187     WICColor colors[256];
1188     ColorMapObject *cm;
1189     int i, trans, count;
1190     ExtensionBlock *eb;
1191 
1192     TRACE("(%p,%p)\n", iface, palette);
1193 
1194     if (!This->gif)
1195         return WINCODEC_ERR_WRONGSTATE;
1196 
1197     cm = This->gif->SColorMap;
1198     if (cm)
1199     {
1200         if (cm->ColorCount > 256)
1201         {
1202             ERR("GIF contains invalid number of colors: %d\n", cm->ColorCount);
1203             return E_FAIL;
1204         }
1205 
1206         for (i = 0; i < cm->ColorCount; i++)
1207         {
1208             colors[i] = 0xff000000 | /* alpha */
1209                         cm->Colors[i].Red << 16 |
1210                         cm->Colors[i].Green << 8 |
1211                         cm->Colors[i].Blue;
1212         }
1213 
1214         count = cm->ColorCount;
1215     }
1216     else
1217     {
1218         colors[0] = 0xff000000;
1219         colors[1] = 0xffffffff;
1220 
1221         for (i = 2; i < 256; i++)
1222             colors[i] = 0xff000000;
1223 
1224         count = 256;
1225     }
1226 
1227     /* look for the transparent color extension */
1228     for (i = 0; i < This->gif->SavedImages[This->current_frame].Extensions.ExtensionBlockCount; i++)
1229     {
1230         eb = This->gif->SavedImages[This->current_frame].Extensions.ExtensionBlocks + i;
1231         if (eb->Function == GRAPHICS_EXT_FUNC_CODE && eb->ByteCount == 8)
1232         {
1233             if (eb->Bytes[3] & 1)
1234             {
1235                 trans = (unsigned char)eb->Bytes[6];
1236                 colors[trans] &= 0xffffff; /* set alpha to 0 */
1237                 break;
1238             }
1239         }
1240     }
1241 
1242     return IWICPalette_InitializeCustom(palette, colors, count);
1243 }
1244 
1245 static HRESULT WINAPI GifDecoder_GetMetadataQueryReader(IWICBitmapDecoder *iface,
1246     IWICMetadataQueryReader **ppIMetadataQueryReader)
1247 {
1248     GifDecoder *This = impl_from_IWICBitmapDecoder(iface);
1249 
1250     TRACE("(%p,%p)\n", iface, ppIMetadataQueryReader);
1251 
1252     if (!ppIMetadataQueryReader) return E_INVALIDARG;
1253 
1254     return MetadataQueryReader_CreateInstance(&This->IWICMetadataBlockReader_iface, NULL, ppIMetadataQueryReader);
1255 }
1256 
1257 static HRESULT WINAPI GifDecoder_GetPreview(IWICBitmapDecoder *iface,
1258     IWICBitmapSource **ppIBitmapSource)
1259 {
1260     TRACE("(%p,%p)\n", iface, ppIBitmapSource);
1261     return WINCODEC_ERR_UNSUPPORTEDOPERATION;
1262 }
1263 
1264 static HRESULT WINAPI GifDecoder_GetColorContexts(IWICBitmapDecoder *iface,
1265     UINT cCount, IWICColorContext **ppIColorContexts, UINT *pcActualCount)
1266 {
1267     TRACE("(%p,%u,%p,%p)\n", iface, cCount, ppIColorContexts, pcActualCount);
1268     return WINCODEC_ERR_UNSUPPORTEDOPERATION;
1269 }
1270 
1271 static HRESULT WINAPI GifDecoder_GetThumbnail(IWICBitmapDecoder *iface,
1272     IWICBitmapSource **ppIThumbnail)
1273 {
1274     TRACE("(%p,%p)\n", iface, ppIThumbnail);
1275     return WINCODEC_ERR_CODECNOTHUMBNAIL;
1276 }
1277 
1278 static HRESULT WINAPI GifDecoder_GetFrameCount(IWICBitmapDecoder *iface,
1279     UINT *pCount)
1280 {
1281     GifDecoder *This = impl_from_IWICBitmapDecoder(iface);
1282 
1283     if (!pCount) return E_INVALIDARG;
1284 
1285     EnterCriticalSection(&This->lock);
1286     *pCount = This->gif ? This->gif->ImageCount : 0;
1287     LeaveCriticalSection(&This->lock);
1288 
1289     TRACE("(%p) <-- %d\n", iface, *pCount);
1290 
1291     return S_OK;
1292 }
1293 
1294 static HRESULT WINAPI GifDecoder_GetFrame(IWICBitmapDecoder *iface,
1295     UINT index, IWICBitmapFrameDecode **ppIBitmapFrame)
1296 {
1297     GifDecoder *This = impl_from_IWICBitmapDecoder(iface);
1298     GifFrameDecode *result;
1299     TRACE("(%p,%u,%p)\n", iface, index, ppIBitmapFrame);
1300 
1301     if (!This->initialized) return WINCODEC_ERR_FRAMEMISSING;
1302 
1303     if (index >= This->gif->ImageCount) return E_INVALIDARG;
1304 
1305     result = HeapAlloc(GetProcessHeap(), 0, sizeof(GifFrameDecode));
1306     if (!result) return E_OUTOFMEMORY;
1307 
1308     result->IWICBitmapFrameDecode_iface.lpVtbl = &GifFrameDecode_Vtbl;
1309     result->IWICMetadataBlockReader_iface.lpVtbl = &GifFrameDecode_BlockVtbl;
1310     result->ref = 1;
1311     result->frame = &This->gif->SavedImages[index];
1312     IWICBitmapDecoder_AddRef(iface);
1313     result->parent = This;
1314     This->current_frame = index;
1315 
1316     *ppIBitmapFrame = &result->IWICBitmapFrameDecode_iface;
1317 
1318     return S_OK;
1319 }
1320 
1321 static const IWICBitmapDecoderVtbl GifDecoder_Vtbl = {
1322     GifDecoder_QueryInterface,
1323     GifDecoder_AddRef,
1324     GifDecoder_Release,
1325     GifDecoder_QueryCapability,
1326     GifDecoder_Initialize,
1327     GifDecoder_GetContainerFormat,
1328     GifDecoder_GetDecoderInfo,
1329     GifDecoder_CopyPalette,
1330     GifDecoder_GetMetadataQueryReader,
1331     GifDecoder_GetPreview,
1332     GifDecoder_GetColorContexts,
1333     GifDecoder_GetThumbnail,
1334     GifDecoder_GetFrameCount,
1335     GifDecoder_GetFrame
1336 };
1337 
1338 static HRESULT WINAPI GifDecoder_Block_QueryInterface(IWICMetadataBlockReader *iface,
1339     REFIID iid, void **ppv)
1340 {
1341     GifDecoder *This = impl_from_IWICMetadataBlockReader(iface);
1342     return IWICBitmapDecoder_QueryInterface(&This->IWICBitmapDecoder_iface, iid, ppv);
1343 }
1344 
1345 static ULONG WINAPI GifDecoder_Block_AddRef(IWICMetadataBlockReader *iface)
1346 {
1347     GifDecoder *This = impl_from_IWICMetadataBlockReader(iface);
1348     return IWICBitmapDecoder_AddRef(&This->IWICBitmapDecoder_iface);
1349 }
1350 
1351 static ULONG WINAPI GifDecoder_Block_Release(IWICMetadataBlockReader *iface)
1352 {
1353     GifDecoder *This = impl_from_IWICMetadataBlockReader(iface);
1354     return IWICBitmapDecoder_Release(&This->IWICBitmapDecoder_iface);
1355 }
1356 
1357 static HRESULT WINAPI GifDecoder_Block_GetContainerFormat(IWICMetadataBlockReader *iface,
1358     GUID *guid)
1359 {
1360     TRACE("(%p,%p)\n", iface, guid);
1361 
1362     if (!guid) return E_INVALIDARG;
1363 
1364     *guid = GUID_ContainerFormatGif;
1365     return S_OK;
1366 }
1367 
1368 static HRESULT WINAPI GifDecoder_Block_GetCount(IWICMetadataBlockReader *iface,
1369     UINT *count)
1370 {
1371     GifDecoder *This = impl_from_IWICMetadataBlockReader(iface);
1372 
1373     TRACE("%p,%p\n", iface, count);
1374 
1375     if (!count) return E_INVALIDARG;
1376 
1377     *count = This->gif->Extensions.ExtensionBlockCount + 1;
1378     return S_OK;
1379 }
1380 
1381 static HRESULT WINAPI GifDecoder_Block_GetReaderByIndex(IWICMetadataBlockReader *iface,
1382     UINT index, IWICMetadataReader **reader)
1383 {
1384     GifDecoder *This = impl_from_IWICMetadataBlockReader(iface);
1385     int i;
1386 
1387     TRACE("(%p,%u,%p)\n", iface, index, reader);
1388 
1389     if (!reader) return E_INVALIDARG;
1390 
1391     if (index == 0)
1392         return create_metadata_reader(This->LSD_data, sizeof(This->LSD_data),
1393                                       LSDReader_CreateInstance, reader);
1394 
1395     for (i = 0; i < This->gif->Extensions.ExtensionBlockCount; i++)
1396     {
1397         class_constructor constructor;
1398 
1399         if (index != i + 1) continue;
1400 
1401         if (This->gif->Extensions.ExtensionBlocks[i].Function == APPLICATION_EXT_FUNC_CODE)
1402             constructor = APEReader_CreateInstance;
1403         else if (This->gif->Extensions.ExtensionBlocks[i].Function == COMMENT_EXT_FUNC_CODE)
1404             constructor = GifCommentReader_CreateInstance;
1405         else
1406             constructor = UnknownMetadataReader_CreateInstance;
1407 
1408         return create_metadata_reader(This->gif->Extensions.ExtensionBlocks[i].Bytes,
1409                                       This->gif->Extensions.ExtensionBlocks[i].ByteCount,
1410                                       constructor, reader);
1411     }
1412 
1413     return E_INVALIDARG;
1414 }
1415 
1416 static HRESULT WINAPI GifDecoder_Block_GetEnumerator(IWICMetadataBlockReader *iface,
1417     IEnumUnknown **enumerator)
1418 {
1419     FIXME("(%p,%p): stub\n", iface, enumerator);
1420     return E_NOTIMPL;
1421 }
1422 
1423 static const IWICMetadataBlockReaderVtbl GifDecoder_BlockVtbl =
1424 {
1425     GifDecoder_Block_QueryInterface,
1426     GifDecoder_Block_AddRef,
1427     GifDecoder_Block_Release,
1428     GifDecoder_Block_GetContainerFormat,
1429     GifDecoder_Block_GetCount,
1430     GifDecoder_Block_GetReaderByIndex,
1431     GifDecoder_Block_GetEnumerator
1432 };
1433 
1434 HRESULT GifDecoder_CreateInstance(REFIID iid, void** ppv)
1435 {
1436     GifDecoder *This;
1437     HRESULT ret;
1438 
1439     TRACE("(%s,%p)\n", debugstr_guid(iid), ppv);
1440 
1441     *ppv = NULL;
1442 
1443     This = HeapAlloc(GetProcessHeap(), 0, sizeof(GifDecoder));
1444     if (!This) return E_OUTOFMEMORY;
1445 
1446     This->IWICBitmapDecoder_iface.lpVtbl = &GifDecoder_Vtbl;
1447     This->IWICMetadataBlockReader_iface.lpVtbl = &GifDecoder_BlockVtbl;
1448     This->stream = NULL;
1449     This->ref = 1;
1450     This->initialized = FALSE;
1451     This->gif = NULL;
1452     This->current_frame = 0;
1453     InitializeCriticalSection(&This->lock);
1454     This->lock.DebugInfo->Spare[0] = (DWORD_PTR)(__FILE__ ": GifDecoder.lock");
1455 
1456     ret = IWICBitmapDecoder_QueryInterface(&This->IWICBitmapDecoder_iface, iid, ppv);
1457     IWICBitmapDecoder_Release(&This->IWICBitmapDecoder_iface);
1458 
1459     return ret;
1460 }
1461 
1462 typedef struct GifEncoder
1463 {
1464     IWICBitmapEncoder IWICBitmapEncoder_iface;
1465     LONG ref;
1466     IStream *stream;
1467     CRITICAL_SECTION lock;
1468     BOOL initialized, info_written, committed;
1469     UINT n_frames;
1470     WICColor palette[256];
1471     UINT colors;
1472 } GifEncoder;
1473 
1474 static inline GifEncoder *impl_from_IWICBitmapEncoder(IWICBitmapEncoder *iface)
1475 {
1476     return CONTAINING_RECORD(iface, GifEncoder, IWICBitmapEncoder_iface);
1477 }
1478 
1479 typedef struct GifFrameEncode
1480 {
1481     IWICBitmapFrameEncode IWICBitmapFrameEncode_iface;
1482     LONG ref;
1483     GifEncoder *encoder;
1484     BOOL initialized, interlace, committed;
1485     UINT width, height, lines;
1486     double xres, yres;
1487     WICColor palette[256];
1488     UINT colors;
1489     BYTE *image_data;
1490 } GifFrameEncode;
1491 
1492 static inline GifFrameEncode *impl_from_IWICBitmapFrameEncode(IWICBitmapFrameEncode *iface)
1493 {
1494     return CONTAINING_RECORD(iface, GifFrameEncode, IWICBitmapFrameEncode_iface);
1495 }
1496 
1497 static HRESULT WINAPI GifFrameEncode_QueryInterface(IWICBitmapFrameEncode *iface, REFIID iid, void **ppv)
1498 {
1499     TRACE("%p,%s,%p\n", iface, debugstr_guid(iid), ppv);
1500 
1501     if (!ppv) return E_INVALIDARG;
1502 
1503     if (IsEqualIID(&IID_IUnknown, iid) ||
1504         IsEqualIID(&IID_IWICBitmapFrameEncode, iid))
1505     {
1506         IWICBitmapFrameEncode_AddRef(iface);
1507         *ppv = iface;
1508         return S_OK;
1509     }
1510 
1511     *ppv = NULL;
1512     return E_NOINTERFACE;
1513 }
1514 
1515 static ULONG WINAPI GifFrameEncode_AddRef(IWICBitmapFrameEncode *iface)
1516 {
1517     GifFrameEncode *This = impl_from_IWICBitmapFrameEncode(iface);
1518     ULONG ref = InterlockedIncrement(&This->ref);
1519 
1520     TRACE("%p -> %u\n", iface, ref);
1521     return ref;
1522 }
1523 
1524 static ULONG WINAPI GifFrameEncode_Release(IWICBitmapFrameEncode *iface)
1525 {
1526     GifFrameEncode *This = impl_from_IWICBitmapFrameEncode(iface);
1527     ULONG ref = InterlockedDecrement(&This->ref);
1528 
1529     TRACE("%p -> %u\n", iface, ref);
1530 
1531     if (!ref)
1532     {
1533         IWICBitmapEncoder_Release(&This->encoder->IWICBitmapEncoder_iface);
1534         HeapFree(GetProcessHeap(), 0, This->image_data);
1535         HeapFree(GetProcessHeap(), 0, This);
1536     }
1537 
1538     return ref;
1539 }
1540 
1541 static HRESULT WINAPI GifFrameEncode_Initialize(IWICBitmapFrameEncode *iface, IPropertyBag2 *options)
1542 {
1543     GifFrameEncode *This = impl_from_IWICBitmapFrameEncode(iface);
1544     HRESULT hr;
1545 
1546     TRACE("%p,%p\n", iface, options);
1547 
1548     EnterCriticalSection(&This->encoder->lock);
1549 
1550     if (!This->initialized)
1551     {
1552         This->initialized = TRUE;
1553         hr = S_OK;
1554     }
1555     else
1556         hr = WINCODEC_ERR_WRONGSTATE;
1557 
1558     LeaveCriticalSection(&This->encoder->lock);
1559 
1560     return hr;
1561 }
1562 
1563 static HRESULT WINAPI GifFrameEncode_SetSize(IWICBitmapFrameEncode *iface, UINT width, UINT height)
1564 {
1565     GifFrameEncode *This = impl_from_IWICBitmapFrameEncode(iface);
1566     HRESULT hr;
1567 
1568     TRACE("%p,%u,%u\n", iface, width, height);
1569 
1570     if (!width || !height) return E_INVALIDARG;
1571 
1572     EnterCriticalSection(&This->encoder->lock);
1573 
1574     if (This->initialized)
1575     {
1576         HeapFree(GetProcessHeap(), 0, This->image_data);
1577 
1578         This->image_data = HeapAlloc(GetProcessHeap(), 0, width * height);
1579         if (This->image_data)
1580         {
1581             This->width = width;
1582             This->height = height;
1583             hr = S_OK;
1584         }
1585         else
1586             hr = E_OUTOFMEMORY;
1587     }
1588     else
1589         hr = WINCODEC_ERR_WRONGSTATE;
1590 
1591     LeaveCriticalSection(&This->encoder->lock);
1592 
1593     return hr;
1594 }
1595 
1596 static HRESULT WINAPI GifFrameEncode_SetResolution(IWICBitmapFrameEncode *iface, double xres, double yres)
1597 {
1598     GifFrameEncode *This = impl_from_IWICBitmapFrameEncode(iface);
1599     HRESULT hr;
1600 
1601     TRACE("%p,%f,%f\n", iface, xres, yres);
1602 
1603     EnterCriticalSection(&This->encoder->lock);
1604 
1605     if (This->initialized)
1606     {
1607         This->xres = xres;
1608         This->yres = yres;
1609         hr = S_OK;
1610     }
1611     else
1612         hr = WINCODEC_ERR_WRONGSTATE;
1613 
1614     LeaveCriticalSection(&This->encoder->lock);
1615 
1616     return hr;
1617 }
1618 
1619 static HRESULT WINAPI GifFrameEncode_SetPixelFormat(IWICBitmapFrameEncode *iface, WICPixelFormatGUID *format)
1620 {
1621     GifFrameEncode *This = impl_from_IWICBitmapFrameEncode(iface);
1622     HRESULT hr;
1623 
1624     TRACE("%p,%s\n", iface, debugstr_guid(format));
1625 
1626     if (!format) return E_INVALIDARG;
1627 
1628     EnterCriticalSection(&This->encoder->lock);
1629 
1630     if (This->initialized)
1631     {
1632         *format = GUID_WICPixelFormat8bppIndexed;
1633         hr = S_OK;
1634     }
1635     else
1636         hr = WINCODEC_ERR_WRONGSTATE;
1637 
1638     LeaveCriticalSection(&This->encoder->lock);
1639 
1640     return hr;
1641 }
1642 
1643 static HRESULT WINAPI GifFrameEncode_SetColorContexts(IWICBitmapFrameEncode *iface, UINT count, IWICColorContext **context)
1644 {
1645     FIXME("%p,%u,%p: stub\n", iface, count, context);
1646     return E_NOTIMPL;
1647 }
1648 
1649 static HRESULT WINAPI GifFrameEncode_SetPalette(IWICBitmapFrameEncode *iface, IWICPalette *palette)
1650 {
1651     GifFrameEncode *This = impl_from_IWICBitmapFrameEncode(iface);
1652     HRESULT hr;
1653 
1654     TRACE("%p,%p\n", iface, palette);
1655 
1656     if (!palette) return E_INVALIDARG;
1657 
1658     EnterCriticalSection(&This->encoder->lock);
1659 
1660     if (This->initialized)
1661         hr = IWICPalette_GetColors(palette, 256, This->palette, &This->colors);
1662     else
1663         hr = WINCODEC_ERR_NOTINITIALIZED;
1664 
1665     LeaveCriticalSection(&This->encoder->lock);
1666     return hr;
1667 }
1668 
1669 static HRESULT WINAPI GifFrameEncode_SetThumbnail(IWICBitmapFrameEncode *iface, IWICBitmapSource *thumbnail)
1670 {
1671     FIXME("%p,%p: stub\n", iface, thumbnail);
1672     return E_NOTIMPL;
1673 }
1674 
1675 static HRESULT WINAPI GifFrameEncode_WritePixels(IWICBitmapFrameEncode *iface, UINT lines, UINT stride, UINT size, BYTE *pixels)
1676 {
1677     GifFrameEncode *This = impl_from_IWICBitmapFrameEncode(iface);
1678     HRESULT hr;
1679 
1680     TRACE("%p,%u,%u,%u,%p\n", iface, lines, stride, size, pixels);
1681 
1682     if (!pixels) return E_INVALIDARG;
1683 
1684     EnterCriticalSection(&This->encoder->lock);
1685 
1686     if (This->initialized && This->image_data)
1687     {
1688         if (This->lines + lines <= This->height)
1689         {
1690             UINT i;
1691             BYTE *src, *dst;
1692 
1693             src = pixels;
1694             dst = This->image_data + This->lines * This->width;
1695 
1696             for (i = 0; i < lines; i++)
1697             {
1698                 memcpy(dst, src, This->width);
1699                 src += stride;
1700                 dst += This->width;
1701             }
1702 
1703             This->lines += lines;
1704             hr = S_OK;
1705         }
1706         else
1707             hr = E_INVALIDARG;
1708     }
1709     else
1710         hr = WINCODEC_ERR_WRONGSTATE;
1711 
1712     LeaveCriticalSection(&This->encoder->lock);
1713     return hr;
1714 }
1715 
1716 static HRESULT WINAPI GifFrameEncode_WriteSource(IWICBitmapFrameEncode *iface, IWICBitmapSource *source, WICRect *rc)
1717 {
1718     GifFrameEncode *This = impl_from_IWICBitmapFrameEncode(iface);
1719     HRESULT hr;
1720 
1721     TRACE("%p,%p,%p\n", iface, source, rc);
1722 
1723     if (!source) return E_INVALIDARG;
1724 
1725     EnterCriticalSection(&This->encoder->lock);
1726 
1727     if (This->initialized)
1728     {
1729         const GUID *format = &GUID_WICPixelFormat8bppIndexed;
1730 
1731         hr = configure_write_source(iface, source, rc, format,
1732                                     This->width, This->height, This->xres, This->yres);
1733         if (hr == S_OK)
1734             hr = write_source(iface, source, rc, format, 8, This->width, This->height);
1735     }
1736     else
1737         hr = WINCODEC_ERR_WRONGSTATE;
1738 
1739     LeaveCriticalSection(&This->encoder->lock);
1740     return hr;
1741 }
1742 
1743 #define LZW_DICT_SIZE (1 << 12)
1744 
1745 struct lzw_dict
1746 {
1747     short prefix[LZW_DICT_SIZE];
1748     unsigned char suffix[LZW_DICT_SIZE];
1749 };
1750 
1751 struct lzw_state
1752 {
1753     struct lzw_dict dict;
1754     short init_code_bits, code_bits, next_code, clear_code, eof_code;
1755     unsigned bits_buf;
1756     int bits_count;
1757     int (*user_write_data)(void *user_ptr, void *data, int length);
1758     void *user_ptr;
1759 };
1760 
1761 struct input_stream
1762 {
1763     unsigned len;
1764     const BYTE *in;
1765 };
1766 
1767 struct output_stream
1768 {
1769     struct
1770     {
1771         unsigned char len;
1772         char data[255];
1773     } gif_block;
1774     IStream *out;
1775 };
1776 
1777 static int lzw_output_code(struct lzw_state *state, short code)
1778 {
1779     state->bits_buf |= code << state->bits_count;
1780     state->bits_count += state->code_bits;
1781 
1782     while (state->bits_count >= 8)
1783     {
1784         unsigned char byte = (unsigned char)state->bits_buf;
1785         if (state->user_write_data(state->user_ptr, &byte, 1) != 1)
1786             return 0;
1787         state->bits_buf >>= 8;
1788         state->bits_count -= 8;
1789     }
1790 
1791     return 1;
1792 }
1793 
1794 static inline int lzw_output_clear_code(struct lzw_state *state)
1795 {
1796     return lzw_output_code(state, state->clear_code);
1797 }
1798 
1799 static inline int lzw_output_eof_code(struct lzw_state *state)
1800 {
1801     return lzw_output_code(state, state->eof_code);
1802 }
1803 
1804 static int lzw_flush_bits(struct lzw_state *state)
1805 {
1806     unsigned char byte;
1807 
1808     while (state->bits_count >= 8)
1809     {
1810         byte = (unsigned char)state->bits_buf;
1811         if (state->user_write_data(state->user_ptr, &byte, 1) != 1)
1812             return 0;
1813         state->bits_buf >>= 8;
1814         state->bits_count -= 8;
1815     }
1816 
1817     if (state->bits_count)
1818     {
1819         static const char mask[8] = { 0x00,0x01,0x03,0x07,0x0f,0x1f,0x3f,0x7f };
1820 
1821         byte = (unsigned char)state->bits_buf & mask[state->bits_count];
1822         if (state->user_write_data(state->user_ptr, &byte, 1) != 1)
1823             return 0;
1824     }
1825 
1826     state->bits_buf = 0;
1827     state->bits_count = 0;
1828 
1829     return 1;
1830 }
1831 
1832 static void lzw_dict_reset(struct lzw_state *state)
1833 {
1834     int i;
1835 
1836     state->code_bits = state->init_code_bits + 1;
1837     state->next_code = (1 << state->init_code_bits) + 2;
1838 
1839     for(i = 0; i < LZW_DICT_SIZE; i++)
1840     {
1841         state->dict.prefix[i] = 1 << 12; /* impossible LZW code value */
1842         state->dict.suffix[i] = 0;
1843     }
1844 }
1845 
1846 static void lzw_state_init(struct lzw_state *state, short init_code_bits, void *user_write_data, void *user_ptr)
1847 {
1848     state->init_code_bits = init_code_bits;
1849     state->clear_code = 1 << init_code_bits;
1850     state->eof_code = state->clear_code + 1;
1851     state->bits_buf = 0;
1852     state->bits_count = 0;
1853     state->user_write_data = user_write_data;
1854     state->user_ptr = user_ptr;
1855 
1856     lzw_dict_reset(state);
1857 }
1858 
1859 static int lzw_dict_add(struct lzw_state *state, short prefix, unsigned char suffix)
1860 {
1861     if (state->next_code < LZW_DICT_SIZE)
1862     {
1863         state->dict.prefix[state->next_code] = prefix;
1864         state->dict.suffix[state->next_code] = suffix;
1865 
1866         if ((state->next_code & (state->next_code - 1)) == 0)
1867             state->code_bits++;
1868 
1869         state->next_code++;
1870         return state->next_code;
1871     }
1872 
1873     return -1;
1874 }
1875 
1876 static short lzw_dict_lookup(const struct lzw_state *state, short prefix, unsigned char suffix)
1877 {
1878     short i;
1879 
1880     for (i = 0; i < state->next_code; i++)
1881     {
1882         if (state->dict.prefix[i] == prefix && state->dict.suffix[i] == suffix)
1883             return i;
1884     }
1885 
1886     return -1;
1887 }
1888 
1889 static inline int write_byte(struct output_stream *out, char byte)
1890 {
1891     if (out->gif_block.len == 255)
1892     {
1893         if (IStream_Write(out->out, &out->gif_block, sizeof(out->gif_block), NULL) != S_OK)
1894             return 0;
1895 
1896         out->gif_block.len = 0;
1897     }
1898 
1899     out->gif_block.data[out->gif_block.len++] = byte;
1900 
1901     return 1;
1902 }
1903 
1904 static int write_data(void *user_ptr, void *user_data, int length)
1905 {
1906     unsigned char *data = user_data;
1907     struct output_stream *out = user_ptr;
1908     int len = length;
1909 
1910     while (len-- > 0)
1911     {
1912         if (!write_byte(out, *data++)) return 0;
1913     }
1914 
1915     return length;
1916 }
1917 
1918 static int flush_output_data(void *user_ptr)
1919 {
1920     struct output_stream *out = user_ptr;
1921 
1922     if (out->gif_block.len)
1923     {
1924         if (IStream_Write(out->out, &out->gif_block, out->gif_block.len + sizeof(out->gif_block.len), NULL) != S_OK)
1925             return 0;
1926     }
1927 
1928     /* write GIF block terminator */
1929     out->gif_block.len = 0;
1930     return IStream_Write(out->out, &out->gif_block, sizeof(out->gif_block.len), NULL) == S_OK;
1931 }
1932 
1933 static inline int read_byte(struct input_stream *in, unsigned char *byte)
1934 {
1935     if (in->len)
1936     {
1937         in->len--;
1938         *byte = *in->in++;
1939         return 1;
1940     }
1941 
1942     return 0;
1943 }
1944 
1945 static HRESULT gif_compress(IStream *out_stream, const BYTE *in_data, ULONG in_size)
1946 {
1947     struct input_stream in;
1948     struct output_stream out;
1949     struct lzw_state state;
1950     short init_code_bits, prefix, code;
1951     unsigned char suffix;
1952 
1953     in.in = in_data;
1954     in.len = in_size;
1955 
1956     out.gif_block.len = 0;
1957     out.out = out_stream;
1958 
1959     init_code_bits = suffix = 8;
1960     if (IStream_Write(out.out, &suffix, sizeof(suffix), NULL) != S_OK)
1961         return E_FAIL;
1962 
1963     lzw_state_init(&state, init_code_bits, write_data, &out);
1964 
1965     if (!lzw_output_clear_code(&state))
1966         return E_FAIL;
1967 
1968     if (read_byte(&in, &suffix))
1969     {
1970         prefix = suffix;
1971 
1972         while (read_byte(&in, &suffix))
1973         {
1974             code = lzw_dict_lookup(&state, prefix, suffix);
1975             if (code == -1)
1976             {
1977                 if (!lzw_output_code(&state, prefix))
1978                     return E_FAIL;
1979 
1980                 if (lzw_dict_add(&state, prefix, suffix) == -1)
1981                 {
1982                     if (!lzw_output_clear_code(&state))
1983                         return E_FAIL;
1984                     lzw_dict_reset(&state);
1985                 }
1986 
1987                 prefix = suffix;
1988             }
1989             else
1990                 prefix = code;
1991         }
1992 
1993         if (!lzw_output_code(&state, prefix))
1994             return E_FAIL;
1995         if (!lzw_output_eof_code(&state))
1996             return E_FAIL;
1997         if (!lzw_flush_bits(&state))
1998             return E_FAIL;
1999     }
2000 
2001     return flush_output_data(&out) ? S_OK : E_FAIL;
2002 }
2003 
2004 static HRESULT WINAPI GifFrameEncode_Commit(IWICBitmapFrameEncode *iface)
2005 {
2006     GifFrameEncode *This = impl_from_IWICBitmapFrameEncode(iface);
2007     HRESULT hr;
2008 
2009     TRACE("%p\n", iface);
2010 
2011     EnterCriticalSection(&This->encoder->lock);
2012 
2013     if (This->image_data && This->lines == This->height && !This->committed)
2014     {
2015         BYTE gif_palette[256][3];
2016 
2017         hr = S_OK;
2018 
2019         if (!This->encoder->info_written)
2020         {
2021             struct logical_screen_descriptor lsd;
2022 
2023             /* Logical Screen Descriptor */
2024             memcpy(lsd.signature, "GIF89a", 6);
2025             lsd.width = This->width;
2026             lsd.height = This->height;
2027             lsd.packed = 0;
2028             if (This->encoder->colors)
2029                 lsd.packed |= 0x80; /* global color table flag */
2030             lsd.packed |= 0x07 << 4; /* color resolution */
2031             lsd.packed |= 0x07; /* global color table size */
2032             lsd.background_color_index = 0; /* FIXME */
2033             lsd.pixel_aspect_ratio = 0;
2034             hr = IStream_Write(This->encoder->stream, &lsd, sizeof(lsd), NULL);
2035             if (hr == S_OK && This->encoder->colors)
2036             {
2037                 UINT i;
2038 
2039                 /* Global Color Table */
2040                 memset(gif_palette, 0, sizeof(gif_palette));
2041                 for (i = 0; i < This->encoder->colors; i++)
2042                 {
2043                     gif_palette[i][0] = (This->encoder->palette[i] >> 16) & 0xff;
2044                     gif_palette[i][1] = (This->encoder->palette[i] >> 8) & 0xff;
2045                     gif_palette[i][2] = This->encoder->palette[i] & 0xff;
2046                 }
2047                 hr = IStream_Write(This->encoder->stream, gif_palette, sizeof(gif_palette), NULL);
2048             }
2049 
2050             /* FIXME: write GCE, APE, etc. GIF extensions */
2051 
2052             if (hr == S_OK)
2053                 This->encoder->info_written = TRUE;
2054         }
2055 
2056         if (hr == S_OK)
2057         {
2058             char image_separator = 0x2c;
2059 
2060             hr = IStream_Write(This->encoder->stream, &image_separator, sizeof(image_separator), NULL);
2061             if (hr == S_OK)
2062             {
2063                 struct image_descriptor imd;
2064 
2065                 /* Image Descriptor */
2066                 imd.left = 0;
2067                 imd.top = 0;
2068                 imd.width = This->width;
2069                 imd.height = This->height;
2070                 imd.packed = 0;
2071                 if (This->colors)
2072                 {
2073                     imd.packed |= 0x80; /* local color table flag */
2074                     imd.packed |= 0x07; /* local color table size */
2075                 }
2076                 /* FIXME: interlace flag */
2077                 hr = IStream_Write(This->encoder->stream, &imd, sizeof(imd), NULL);
2078                 if (hr == S_OK && This->colors)
2079                 {
2080                     UINT i;
2081 
2082                     /* Local Color Table */
2083                     memset(gif_palette, 0, sizeof(gif_palette));
2084                     for (i = 0; i < This->colors; i++)
2085                     {
2086                         gif_palette[i][0] = (This->palette[i] >> 16) & 0xff;
2087                         gif_palette[i][1] = (This->palette[i] >> 8) & 0xff;
2088                         gif_palette[i][2] = This->palette[i] & 0xff;
2089                     }
2090                     hr = IStream_Write(This->encoder->stream, gif_palette, sizeof(gif_palette), NULL);
2091                     if (hr == S_OK)
2092                     {
2093                         /* Image Data */
2094                         hr = gif_compress(This->encoder->stream, This->image_data, This->width * This->height);
2095                         if (hr == S_OK)
2096                             This->committed = TRUE;
2097                     }
2098                 }
2099             }
2100         }
2101     }
2102     else
2103         hr = WINCODEC_ERR_WRONGSTATE;
2104 
2105     LeaveCriticalSection(&This->encoder->lock);
2106     return hr;
2107 }
2108 
2109 static HRESULT WINAPI GifFrameEncode_GetMetadataQueryWriter(IWICBitmapFrameEncode *iface, IWICMetadataQueryWriter **writer)
2110 {
2111     FIXME("%p, %p: stub\n", iface, writer);
2112     return E_NOTIMPL;
2113 }
2114 
2115 static const IWICBitmapFrameEncodeVtbl GifFrameEncode_Vtbl =
2116 {
2117     GifFrameEncode_QueryInterface,
2118     GifFrameEncode_AddRef,
2119     GifFrameEncode_Release,
2120     GifFrameEncode_Initialize,
2121     GifFrameEncode_SetSize,
2122     GifFrameEncode_SetResolution,
2123     GifFrameEncode_SetPixelFormat,
2124     GifFrameEncode_SetColorContexts,
2125     GifFrameEncode_SetPalette,
2126     GifFrameEncode_SetThumbnail,
2127     GifFrameEncode_WritePixels,
2128     GifFrameEncode_WriteSource,
2129     GifFrameEncode_Commit,
2130     GifFrameEncode_GetMetadataQueryWriter
2131 };
2132 
2133 static HRESULT WINAPI GifEncoder_QueryInterface(IWICBitmapEncoder *iface, REFIID iid, void **ppv)
2134 {
2135     TRACE("%p,%s,%p\n", iface, debugstr_guid(iid), ppv);
2136 
2137     if (!ppv) return E_INVALIDARG;
2138 
2139     if (IsEqualIID(&IID_IUnknown, iid) ||
2140         IsEqualIID(&IID_IWICBitmapEncoder, iid))
2141     {
2142         IWICBitmapEncoder_AddRef(iface);
2143         *ppv = iface;
2144         return S_OK;
2145     }
2146 
2147     *ppv = NULL;
2148     return E_NOINTERFACE;
2149 }
2150 
2151 static ULONG WINAPI GifEncoder_AddRef(IWICBitmapEncoder *iface)
2152 {
2153     GifEncoder *This = impl_from_IWICBitmapEncoder(iface);
2154     ULONG ref = InterlockedIncrement(&This->ref);
2155 
2156     TRACE("%p -> %u\n", iface, ref);
2157     return ref;
2158 }
2159 
2160 static ULONG WINAPI GifEncoder_Release(IWICBitmapEncoder *iface)
2161 {
2162     GifEncoder *This = impl_from_IWICBitmapEncoder(iface);
2163     ULONG ref = InterlockedDecrement(&This->ref);
2164 
2165     TRACE("%p -> %u\n", iface, ref);
2166 
2167     if (!ref)
2168     {
2169         if (This->stream) IStream_Release(This->stream);
2170         This->lock.DebugInfo->Spare[0] = 0;
2171         DeleteCriticalSection(&This->lock);
2172         HeapFree(GetProcessHeap(), 0, This);
2173     }
2174 
2175     return ref;
2176 }
2177 
2178 static HRESULT WINAPI GifEncoder_Initialize(IWICBitmapEncoder *iface, IStream *stream, WICBitmapEncoderCacheOption option)
2179 {
2180     GifEncoder *This = impl_from_IWICBitmapEncoder(iface);
2181     HRESULT hr;
2182 
2183     TRACE("%p,%p,%#x\n", iface, stream, option);
2184 
2185     if (!stream) return E_INVALIDARG;
2186 
2187     EnterCriticalSection(&This->lock);
2188 
2189     if (!This->initialized)
2190     {
2191         IStream_AddRef(stream);
2192         This->stream = stream;
2193         This->initialized = TRUE;
2194         hr = S_OK;
2195     }
2196     else
2197         hr = WINCODEC_ERR_WRONGSTATE;
2198 
2199     LeaveCriticalSection(&This->lock);
2200 
2201     return hr;
2202 }
2203 
2204 static HRESULT WINAPI GifEncoder_GetContainerFormat(IWICBitmapEncoder *iface, GUID *format)
2205 {
2206     if (!format) return E_INVALIDARG;
2207 
2208     *format = GUID_ContainerFormatGif;
2209     return S_OK;
2210 }
2211 
2212 static HRESULT WINAPI GifEncoder_GetEncoderInfo(IWICBitmapEncoder *iface, IWICBitmapEncoderInfo **info)
2213 {
2214     IWICComponentInfo *comp_info;
2215     HRESULT hr;
2216 
2217     TRACE("%p,%p\n", iface, info);
2218 
2219     if (!info) return E_INVALIDARG;
2220 
2221     hr = CreateComponentInfo(&CLSID_WICGifEncoder, &comp_info);
2222     if (hr == S_OK)
2223     {
2224         hr = IWICComponentInfo_QueryInterface(comp_info, &IID_IWICBitmapEncoderInfo, (void **)info);
2225         IWICComponentInfo_Release(comp_info);
2226     }
2227     return hr;
2228 }
2229 
2230 static HRESULT WINAPI GifEncoder_SetColorContexts(IWICBitmapEncoder *iface, UINT count, IWICColorContext **context)
2231 {
2232     FIXME("%p,%u,%p: stub\n", iface, count, context);
2233     return E_NOTIMPL;
2234 }
2235 
2236 static HRESULT WINAPI GifEncoder_SetPalette(IWICBitmapEncoder *iface, IWICPalette *palette)
2237 {
2238     GifEncoder *This = impl_from_IWICBitmapEncoder(iface);
2239     HRESULT hr;
2240 
2241     TRACE("%p,%p\n", iface, palette);
2242 
2243     if (!palette) return E_INVALIDARG;
2244 
2245     EnterCriticalSection(&This->lock);
2246 
2247     if (This->initialized)
2248         hr = IWICPalette_GetColors(palette, 256, This->palette, &This->colors);
2249     else
2250         hr = WINCODEC_ERR_NOTINITIALIZED;
2251 
2252     LeaveCriticalSection(&This->lock);
2253     return hr;
2254 }
2255 
2256 static HRESULT WINAPI GifEncoder_SetThumbnail(IWICBitmapEncoder *iface, IWICBitmapSource *thumbnail)
2257 {
2258     TRACE("%p,%p\n", iface, thumbnail);
2259     return WINCODEC_ERR_UNSUPPORTEDOPERATION;
2260 }
2261 
2262 static HRESULT WINAPI GifEncoder_SetPreview(IWICBitmapEncoder *iface, IWICBitmapSource *preview)
2263 {
2264     TRACE("%p,%p\n", iface, preview);
2265     return WINCODEC_ERR_UNSUPPORTEDOPERATION;
2266 }
2267 
2268 static HRESULT WINAPI GifEncoder_CreateNewFrame(IWICBitmapEncoder *iface, IWICBitmapFrameEncode **frame, IPropertyBag2 **options)
2269 {
2270     GifEncoder *This = impl_from_IWICBitmapEncoder(iface);
2271     HRESULT hr;
2272 
2273     TRACE("%p,%p,%p\n", iface, frame, options);
2274 
2275     if (!frame) return E_INVALIDARG;
2276 
2277     EnterCriticalSection(&This->lock);
2278 
2279     if (This->initialized && !This->committed)
2280     {
2281         GifFrameEncode *ret = HeapAlloc(GetProcessHeap(), 0, sizeof(*ret));
2282         if (ret)
2283         {
2284             This->n_frames++;
2285 
2286             ret->IWICBitmapFrameEncode_iface.lpVtbl = &GifFrameEncode_Vtbl;
2287             ret->ref = 1;
2288             ret->encoder = This;
2289             ret->initialized = FALSE;
2290             ret->interlace = FALSE; /* FIXME: read from the properties */
2291             ret->committed = FALSE;
2292             ret->width = 0;
2293             ret->height = 0;
2294             ret->lines = 0;
2295             ret->xres = 0.0;
2296             ret->yres = 0.0;
2297             ret->colors = 0;
2298             ret->image_data = NULL;
2299             IWICBitmapEncoder_AddRef(iface);
2300             *frame = &ret->IWICBitmapFrameEncode_iface;
2301 
2302             hr = S_OK;
2303 
2304             if (options)
2305             {
2306                 hr = CreatePropertyBag2(NULL, 0, options);
2307                 if (hr != S_OK)
2308                 {
2309                     IWICBitmapFrameEncode_Release(*frame);
2310                     *frame = NULL;
2311                 }
2312             }
2313         }
2314         else
2315             hr = E_OUTOFMEMORY;
2316     }
2317     else
2318         hr = WINCODEC_ERR_WRONGSTATE;
2319 
2320     LeaveCriticalSection(&This->lock);
2321 
2322     return hr;
2323 
2324 }
2325 
2326 static HRESULT WINAPI GifEncoder_Commit(IWICBitmapEncoder *iface)
2327 {
2328     GifEncoder *This = impl_from_IWICBitmapEncoder(iface);
2329     HRESULT hr;
2330 
2331     TRACE("%p\n", iface);
2332 
2333     EnterCriticalSection(&This->lock);
2334 
2335     if (This->initialized && !This->committed)
2336     {
2337         char gif_trailer = 0x3b;
2338 
2339         /* FIXME: write text, comment GIF extensions */
2340 
2341         hr = IStream_Write(This->stream, &gif_trailer, sizeof(gif_trailer), NULL);
2342         if (hr == S_OK)
2343             This->committed = TRUE;
2344     }
2345     else
2346         hr = WINCODEC_ERR_WRONGSTATE;
2347 
2348     LeaveCriticalSection(&This->lock);
2349     return hr;
2350 }
2351 
2352 static HRESULT WINAPI GifEncoder_GetMetadataQueryWriter(IWICBitmapEncoder *iface, IWICMetadataQueryWriter **writer)
2353 {
2354     FIXME("%p,%p: stub\n", iface, writer);
2355     return E_NOTIMPL;
2356 }
2357 
2358 static const IWICBitmapEncoderVtbl GifEncoder_Vtbl =
2359 {
2360     GifEncoder_QueryInterface,
2361     GifEncoder_AddRef,
2362     GifEncoder_Release,
2363     GifEncoder_Initialize,
2364     GifEncoder_GetContainerFormat,
2365     GifEncoder_GetEncoderInfo,
2366     GifEncoder_SetColorContexts,
2367     GifEncoder_SetPalette,
2368     GifEncoder_SetThumbnail,
2369     GifEncoder_SetPreview,
2370     GifEncoder_CreateNewFrame,
2371     GifEncoder_Commit,
2372     GifEncoder_GetMetadataQueryWriter
2373 };
2374 
2375 HRESULT GifEncoder_CreateInstance(REFIID iid, void **ppv)
2376 {
2377     GifEncoder *This;
2378     HRESULT ret;
2379 
2380     TRACE("%s,%p\n", debugstr_guid(iid), ppv);
2381 
2382     *ppv = NULL;
2383 
2384     This = HeapAlloc(GetProcessHeap(), 0, sizeof(*This));
2385     if (!This) return E_OUTOFMEMORY;
2386 
2387     This->IWICBitmapEncoder_iface.lpVtbl = &GifEncoder_Vtbl;
2388     This->ref = 1;
2389     This->stream = NULL;
2390     InitializeCriticalSection(&This->lock);
2391     This->lock.DebugInfo->Spare[0] = (DWORD_PTR)(__FILE__ ": GifEncoder.lock");
2392     This->initialized = FALSE;
2393     This->info_written = FALSE;
2394     This->committed = FALSE;
2395     This->n_frames = 0;
2396     This->colors = 0;
2397 
2398     ret = IWICBitmapEncoder_QueryInterface(&This->IWICBitmapEncoder_iface, iid, ppv);
2399     IWICBitmapEncoder_Release(&This->IWICBitmapEncoder_iface);
2400 
2401     return ret;
2402 }
2403