]>
git.saurik.com Git - wxWidgets.git/blob - src/common/gifdecod.cpp
1 /////////////////////////////////////////////////////////////////////////////
3 // Purpose: wxGIFDecoder, GIF reader for wxImage and wxAnimation
4 // Author: Guillermo Rodriguez Garcia <guille@iies.es>
7 // Copyright: (c) Guillermo Rodriguez Garcia
8 // Licence: wxWindows licence
9 /////////////////////////////////////////////////////////////////////////////
11 #if defined(__GNUG__) && !defined(NO_GCC_PRAGMA)
12 #pragma implementation "gifdecod.h"
15 // For compilers that support precompilation, includes "wx.h".
16 #include "wx/wxprec.h"
24 # include "wx/palette.h"
27 #if wxUSE_STREAMS && wxUSE_GIF
31 #include "wx/gifdecod.h"
34 //---------------------------------------------------------------------------
35 // GIFImage constructor
36 //---------------------------------------------------------------------------
46 p
= (unsigned char *) NULL
;
47 pal
= (unsigned char *) NULL
;
48 next
= (GIFImage
*) NULL
;
49 prev
= (GIFImage
*) NULL
;
52 //---------------------------------------------------------------------------
53 // wxGIFDecoder constructor and destructor
54 //---------------------------------------------------------------------------
56 wxGIFDecoder::wxGIFDecoder(wxInputStream
*s
, bool anim
)
72 wxGIFDecoder::~wxGIFDecoder()
77 void wxGIFDecoder::Destroy()
79 GIFImage
*pimg
, *paux
;
100 //---------------------------------------------------------------------------
101 // Convert this image to a wxImage object
102 //---------------------------------------------------------------------------
104 // This function was designed by Vaclav Slavik
106 bool wxGIFDecoder::ConvertToImage(wxImage
*image
) const
108 unsigned char *src
, *dst
, *pal
;
112 /* just in case... */
115 /* create the image */
116 image
->Create(GetWidth(), GetHeight());
123 dst
= image
->GetData();
124 transparent
= GetTransparentColour();
126 /* set transparent colour mask */
127 if (transparent
!= -1)
129 for (i
= 0; i
< 256; i
++)
131 if ((pal
[3 * i
+ 0] == 255) &&
132 (pal
[3 * i
+ 1] == 0) &&
133 (pal
[3 * i
+ 2] == 255))
135 pal
[3 * i
+ 2] = 254;
139 pal
[3 * transparent
+ 0] = 255,
140 pal
[3 * transparent
+ 1] = 0,
141 pal
[3 * transparent
+ 2] = 255;
143 image
->SetMaskColour(255, 0, 255);
146 image
->SetMask(false);
151 unsigned char r
[256];
152 unsigned char g
[256];
153 unsigned char b
[256];
155 for (i
= 0; i
< 256; i
++)
162 image
->SetPalette(wxPalette(256, r
, g
, b
));
164 #endif // wxUSE_PALETTE
166 /* copy image data */
167 for (i
= 0; i
< (GetWidth() * GetHeight()); i
++, src
++)
169 *(dst
++) = pal
[3 * (*src
) + 0];
170 *(dst
++) = pal
[3 * (*src
) + 1];
171 *(dst
++) = pal
[3 * (*src
) + 2];
178 //---------------------------------------------------------------------------
180 //---------------------------------------------------------------------------
182 // Get data for current frame
184 int wxGIFDecoder::GetFrameIndex() const { return m_image
; }
185 unsigned char* wxGIFDecoder::GetData() const { return (m_pimage
->p
); }
186 unsigned char* wxGIFDecoder::GetPalette() const { return (m_pimage
->pal
); }
187 unsigned int wxGIFDecoder::GetWidth() const { return (m_pimage
->w
); }
188 unsigned int wxGIFDecoder::GetHeight() const { return (m_pimage
->h
); }
189 unsigned int wxGIFDecoder::GetTop() const { return (m_pimage
->top
); }
190 unsigned int wxGIFDecoder::GetLeft() const { return (m_pimage
->left
); }
191 int wxGIFDecoder::GetTransparentColour() const { return (m_pimage
->transparent
); }
192 int wxGIFDecoder::GetDisposalMethod() const { return (m_pimage
->disposal
); }
193 long wxGIFDecoder::GetDelay() const { return (m_pimage
->delay
); }
197 unsigned int wxGIFDecoder::GetLogicalScreenWidth() const { return m_screenw
; }
198 unsigned int wxGIFDecoder::GetLogicalScreenHeight() const { return m_screenh
; }
199 int wxGIFDecoder::GetBackgroundColour() const { return m_background
; }
200 int wxGIFDecoder::GetNumberOfFrames() const { return m_nimages
; }
201 bool wxGIFDecoder::IsAnimation() const { return (m_nimages
> 1); }
204 //---------------------------------------------------------------------------
205 // Functions to move through the animation
206 //---------------------------------------------------------------------------
208 bool wxGIFDecoder::GoFirstFrame()
218 bool wxGIFDecoder::GoLastFrame()
228 bool wxGIFDecoder::GoNextFrame(bool cyclic
)
233 if ((m_image
< m_nimages
) || (cyclic
))
235 m_pimage
= m_pimage
->next
;
250 bool wxGIFDecoder::GoPrevFrame(bool cyclic
)
255 if ((m_image
> 1) || (cyclic
))
257 m_pimage
= m_pimage
->prev
;
272 bool wxGIFDecoder::GoFrame(int which
)
277 if ((which
>= 1) && (which
<= m_nimages
))
282 while (m_image
< which
)
285 m_pimage
= m_pimage
->next
;
295 //---------------------------------------------------------------------------
296 // GIF reading and decoding
297 //---------------------------------------------------------------------------
300 // Reads the next code from the file stream, with size 'bits'
302 int wxGIFDecoder::getcode(int bits
, int ab_fin
)
304 unsigned int mask
; /* bit mask */
305 unsigned int code
; /* code (result) */
308 /* get remaining bits from last byte read */
309 mask
= (1 << bits
) - 1;
310 code
= (m_lastbyte
>> (8 - m_restbits
)) & mask
;
312 /* keep reading new bytes while needed */
313 while (bits
> m_restbits
)
315 /* if no bytes left in this block, read the next block */
318 m_restbyte
= (unsigned char)m_f
->GetC();
320 /* Some encoders are a bit broken: instead of issuing
321 * an end-of-image symbol (ab_fin) they come up with
322 * a zero-length subblock!! We catch this here so
323 * that the decoder sees an ab_fin code.
332 m_f
->Read((void *) m_buffer
, m_restbyte
);
333 if (m_f
->LastRead() != m_restbyte
)
341 /* read next byte and isolate the bits we need */
342 m_lastbyte
= (unsigned char) (*m_bufp
++);
343 mask
= (1 << (bits
- m_restbits
)) - 1;
344 code
= code
+ ((m_lastbyte
& mask
) << m_restbits
);
347 /* adjust total number of bits extracted from the buffer */
348 m_restbits
= m_restbits
+ 8;
351 /* find number of bits remaining for next code */
352 m_restbits
= (m_restbits
- bits
);
359 // GIF decoding function. The initial code size (aka root size)
360 // is 'bits'. Supports interlaced images (interl == 1).
361 // Returns wxGIF_OK (== 0) on success, or an error code if something
362 // fails (see header file for details)
363 int wxGIFDecoder::dgif(GIFImage
*img
, int interl
, int bits
)
365 static const int allocSize
= 4096 + 1;
366 int *ab_prefix
= new int[allocSize
]; /* alphabet (prefixes) */
367 if (ab_prefix
== NULL
)
372 int *ab_tail
= new int[allocSize
]; /* alphabet (tails) */
379 int *stack
= new int[allocSize
]; /* decompression stack */
387 int ab_clr
; /* clear code */
388 int ab_fin
; /* end of info code */
389 int ab_bits
; /* actual symbol width, in bits */
390 int ab_free
; /* first free position in alphabet */
391 int ab_max
; /* last possible character in alphabet */
392 int pass
; /* pass number in interlaced images */
393 int pos
; /* index into decompresion stack */
394 unsigned int x
, y
; /* position in image buffer */
396 int code
, readcode
, lastcode
, abcabca
;
398 /* these won't change */
399 ab_clr
= (1 << bits
);
400 ab_fin
= (1 << bits
) + 1;
402 /* these will change through the decompression proccess */
404 ab_free
= (1 << bits
) + 2;
405 ab_max
= (1 << ab_bits
) - 1;
411 /* reset decoder vars */
419 readcode
= code
= getcode(ab_bits
, ab_fin
);
422 if (code
== ab_fin
) break;
424 /* reset alphabet? */
427 /* reset main variables */
429 ab_free
= (1 << bits
) + 2;
430 ab_max
= (1 << ab_bits
) - 1;
434 /* skip to next code */
438 /* unknown code: special case (like in ABCABCA) */
441 code
= lastcode
; /* take last string */
442 stack
[pos
++] = abcabca
; /* add first character */
445 /* build the string for this code in the stack */
446 while (code
> ab_clr
)
448 stack
[pos
++] = ab_tail
[code
];
449 code
= ab_prefix
[code
];
451 // Don't overflow. This shouldn't happen with normal
452 // GIF files, the allocSize of 4096+1 is enough. This
453 // will only happen with badly formed GIFs.
454 if (pos
>= allocSize
)
459 return wxGIF_INVFORMAT
;
463 if (pos
>= allocSize
)
468 return wxGIF_INVFORMAT
;
471 stack
[pos
] = code
; /* push last code into the stack */
472 abcabca
= code
; /* save for special case */
474 /* make new entry in alphabet (only if NOT just cleared) */
477 ab_prefix
[ab_free
] = lastcode
;
478 ab_tail
[ab_free
] = code
;
481 if ((ab_free
> ab_max
) && (ab_bits
< 12))
484 ab_max
= (1 << ab_bits
) - 1;
488 /* dump stack data to the image buffer */
491 (img
->p
)[x
+ (y
* (img
->w
))] = (char) stack
[pos
];
500 /* support for interlaced images */
503 case 1: y
+= 8; break;
504 case 2: y
+= 8; break;
505 case 3: y
+= 4; break;
506 case 4: y
+= 2; break;
509 /* loop until a valid y coordinate has been
510 found, Or if the maximum number of passes has
511 been reached, exit the loop, and stop image
512 decoding (At this point the image is succesfully
514 If we don't loop, but merely set y to some other
515 value, that new value might still be invalid depending
516 on the height of the image. This would cause out of
519 while (y
>= (img
->h
))
523 case 2: y
= 4; break;
524 case 3: y
= 2; break;
525 case 4: y
= 1; break;
529 It's possible we arrive here. For example this
530 happens when the image is interlaced, and the
531 height is 1. Looking at the above cases, the
532 lowest possible y is 1. While the only valid
533 one would be 0 for an image of height 1. So
534 'eventually' the loop will arrive here.
535 This case makes sure this while loop is
536 exited, as well as the 2 other ones.
539 // Set y to a valid coordinate so the local
540 // while loop will be exited. (y = 0 always
541 // is >= img->h since if img->h == 0 the
542 // image is never decoded)
545 // This will exit the other outer while loop
548 // This will halt image decoding.
560 Normally image decoding is finished when an End of Information code is
561 encountered (code == ab_fin) however some broken encoders write wrong
562 "block byte counts" (The first byte value after the "code size" byte),
563 being one value too high. It might very well be possible other variants
564 of this problem occur as well. The only sensible solution seems to
565 be to check for clipping.
566 Example of wrong encoding:
567 (1 * 1 B/W image, raster data stream follows in hex bytes)
569 02 << B/W images have a code size of 2
570 02 << Block byte count
572 00 << Zero byte count (terminates data stream)
574 Because the block byte count is 2, the zero byte count is used in the
575 decoding process, and decoding is continued after this byte. (While it
576 should signal an end of image)
582 01 << When decoded this correctly includes the End of Information code
590 (The 44 doesn't include an End of Information code, but at least the
591 decoder correctly skips to 00 now after decoding, and signals this
592 as an End of Information itself)
606 while (code
!= ab_fin
);
608 delete [] ab_prefix
;
617 // Returns true if the file looks like a valid GIF, false otherwise.
619 bool wxGIFDecoder::CanRead()
621 unsigned char buf
[3];
623 if ( !m_f
->Read(buf
, WXSIZEOF(buf
)) )
626 m_f
->SeekI(-(wxFileOffset
)WXSIZEOF(buf
), wxFromCurrent
);
628 return memcmp(buf
, "GIF", WXSIZEOF(buf
)) == 0;
633 // Reads and decodes one or more GIF images, depending on whether
634 // animated GIF support is enabled. Can read GIFs with any bit
635 // size (color depth), but the output images are always expanded
636 // to 8 bits per pixel. Also, the image palettes always contain
637 // 256 colors, although some of them may be unused. Returns wxGIF_OK
638 // (== 0) on success, or an error code if something fails (see
639 // header file for details)
641 int wxGIFDecoder::ReadGIF()
643 unsigned int ncolors
;
644 int bits
, interl
, transparent
, disposal
, i
;
647 unsigned char type
= 0;
648 unsigned char pal
[768];
649 unsigned char buf
[16];
651 GIFImage
*pimg
, *pprev
;
653 /* check GIF signature */
655 return wxGIF_INVFORMAT
;
657 /* check for animated GIF support (ver. >= 89a) */
659 static const size_t headerSize
= (3 + 3);
660 m_f
->Read(buf
, headerSize
);
661 if (m_f
->LastRead() != headerSize
)
663 return wxGIF_INVFORMAT
;
666 if (memcmp(buf
+ 3, "89a", 3) < 0)
671 /* read logical screen descriptor block (LSDB) */
672 static const size_t lsdbSize
= (2 + 2 + 1 + 1 + 1);
673 m_f
->Read(buf
, lsdbSize
);
674 if (m_f
->LastRead() != lsdbSize
)
676 return wxGIF_INVFORMAT
;
679 m_screenw
= buf
[0] + 256 * buf
[1];
680 m_screenh
= buf
[2] + 256 * buf
[3];
682 /* load global color map if available */
683 if ((buf
[4] & 0x80) == 0x80)
685 m_background
= buf
[5];
687 ncolors
= 2 << (buf
[4] & 0x07);
688 size_t numBytes
= 3 * ncolors
;
689 m_f
->Read(pal
, numBytes
);
690 if (m_f
->LastRead() != numBytes
)
692 return wxGIF_INVFORMAT
;
696 /* transparent colour, disposal method and delay default to unused */
710 type
= (unsigned char)m_f
->GetC();
713 If the end of file has been reached (or an error) and a ";"
714 (0x3B) hasn't been encountered yet, exit the loop. (Without this
715 check the while loop would loop endlessly.) Later on, in the next while
716 loop, the file will be treated as being truncated (But still
717 be decoded as far as possible). returning wxGIF_TRUNCATED is not
718 possible here since some init code is done after this loop.
720 if (m_f
->Eof())// || !m_f->IsOk())
723 type is set to some bogus value, so there's no
724 need to continue evaluating it.
726 break; // Alternative : "return wxGIF_INVFORMAT;"
735 /* extension block? */
738 if (((unsigned char)m_f
->GetC()) == 0xF9)
739 /* graphics control extension, parse it */
741 static const size_t gceSize
= 6;
742 m_f
->Read(buf
, gceSize
);
743 if (m_f
->LastRead() != gceSize
)
746 return wxGIF_INVFORMAT
;
749 /* read delay and convert from 1/100 of a second to ms */
750 delay
= 10 * (buf
[2] + 256 * buf
[3]);
752 /* read transparent colour index, if used */
754 transparent
= buf
[4];
756 /* read disposal method */
757 disposal
= ((buf
[1] & 0x1C) >> 2) - 1;
760 /* other extension, skip */
762 while ((i
= (unsigned char)m_f
->GetC()) != 0)
764 m_f
->SeekI(i
, wxFromCurrent
);
774 /* image descriptor block? */
777 /* allocate memory for IMAGEN struct */
778 pimg
= (*ppimg
) = new GIFImage();
786 /* fill in the data */
787 static const size_t idbSize
= (2 + 2 + 2 + 2 + 1);
788 m_f
->Read(buf
, idbSize
);
789 if (m_f
->LastRead() != idbSize
)
792 return wxGIF_INVFORMAT
;
795 pimg
->left
= buf
[0] + 256 * buf
[1];
796 pimg
->top
= buf
[2] + 256 * buf
[3];
798 pimg->left = buf[4] + 256 * buf[5];
799 pimg->top = buf[4] + 256 * buf[5];
801 pimg
->w
= buf
[4] + 256 * buf
[5];
802 pimg
->h
= buf
[6] + 256 * buf
[7];
804 if (pimg
->w
== 0 || pimg
->h
== 0)
807 return wxGIF_INVFORMAT
;
810 interl
= ((buf
[8] & 0x40)? 1 : 0);
811 size
= pimg
->w
* pimg
->h
;
813 pimg
->transparent
= transparent
;
814 pimg
->disposal
= disposal
;
821 /* allocate memory for image and palette */
822 pimg
->p
= (unsigned char *) malloc((size_t)size
);
823 pimg
->pal
= (unsigned char *) malloc(768);
825 if ((!pimg
->p
) || (!pimg
->pal
))
831 /* load local color map if available, else use global map */
832 if ((buf
[8] & 0x80) == 0x80)
834 ncolors
= 2 << (buf
[8] & 0x07);
835 size_t numBytes
= 3 * ncolors
;
836 m_f
->Read(pimg
->pal
, numBytes
);
837 if (m_f
->LastRead() != numBytes
)
840 return wxGIF_INVFORMAT
;
845 memcpy(pimg
->pal
, pal
, 768);
848 /* get initial code size from first byte in raster data */
849 bits
= (unsigned char)m_f
->GetC();
852 int result
= dgif(pimg
, interl
, bits
);
853 if (result
!= wxGIF_OK
)
860 /* if this is not an animated GIF, exit after first image */
869 return wxGIF_INVFORMAT
;
872 /* setup image pointers */
877 /* try to read to the end of the stream */
881 return wxGIF_TRUNCATED
;
883 type
= (unsigned char)m_f
->GetC();
891 while ((i
= (unsigned char)m_f
->GetC()) != 0)
893 m_f
->SeekI(i
, wxFromCurrent
);
896 else if (type
== 0x2C)
898 /* image descriptor block */
899 static const size_t idbSize
= (2 + 2 + 2 + 2 + 1);
900 m_f
->Read(buf
, idbSize
);
901 if (m_f
->LastRead() != idbSize
)
904 return wxGIF_INVFORMAT
;
907 /* local color map */
908 if ((buf
[8] & 0x80) == 0x80)
910 ncolors
= 2 << (buf
[8] & 0x07);
911 wxFileOffset pos
= m_f
->TellI();
912 wxFileOffset numBytes
= 3 * ncolors
;
913 m_f
->SeekI(numBytes
, wxFromCurrent
);
914 if (m_f
->TellI() != (pos
+ numBytes
))
917 return wxGIF_INVFORMAT
;
921 /* initial code size */
925 while ((i
= (unsigned char)m_f
->GetC()) != 0)
927 m_f
->SeekI(i
, wxFromCurrent
);
930 else if ((type
!= 0x3B) && (type
!= 00)) /* testing */
932 /* images are OK, but couldn't read to the end of the stream */
933 return wxGIF_TRUNCATED
;
940 #endif // wxUSE_STREAMS && wxUSE_GIF