Leaked source code of windows server 2003
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

1766 lines
59 KiB

  1. //+-----------------------------------------------------------------------------------
  2. //
  3. // Microsoft
  4. // Copyright (c) Microsoft Corporation, 1999
  5. //
  6. // File: src\time\src\loadgif.cpp
  7. //
  8. // Contents: gif decoder, copied from direct animation source: danim\src\appel\util\loadgif.cpp
  9. //
  10. //------------------------------------------------------------------------------------
  11. //#include <wininetp.h>
  12. //bw #include "headers.h"
  13. // #define WIN32_LEAN_AND_MEAN
  14. #include <windows.h>
  15. #include <string.h>
  16. #include <stdio.h>
  17. #include <stdlib.h>
  18. #include <stdarg.h>
  19. #include <ole2.h>
  20. #include <math.h>
  21. #include <windowsx.h>
  22. /*lint ++flb*/
  23. //bw DeclareTag(tagImageDecode, "Image Decode", "Image Decode Filters");
  24. const long COLORKEY_NOT_SET = -1;
  25. /*--
  26. Structs from IE img.hxx
  27. --*/
  28. void * __cdecl
  29. _calloc(size_t num, size_t size)
  30. {
  31. void * pv = malloc(num * size);
  32. if (NULL == pv)
  33. {
  34. return NULL;
  35. }
  36. ZeroMemory(pv, num * size);
  37. return pv;
  38. }
  39. enum
  40. {
  41. gifNoneSpecified = 0, // no disposal method specified
  42. gifNoDispose = 1, // do not dispose, leave the bits there
  43. gifRestoreBkgnd = 2, // replace the image with the background color
  44. gifRestorePrev = 3 // replace the image with the previous pixels
  45. };
  46. typedef struct _GCEDATA // data from GIF Graphic control extension
  47. {
  48. unsigned int uiDelayTime; // frame duration, initialy 1/100ths seconds
  49. // converted to milliseconds
  50. unsigned int uiDisposalMethod; // 0 - none specified.
  51. // 1 - do not dispose - leave bits in place.
  52. // 2 - replace with background color.
  53. // 3 - restore previous bits
  54. // >3 - not yet defined
  55. BOOL fTransparent; // TRUE is ucTransIndex describes transparent color
  56. unsigned char ucTransIndex; // transparent index
  57. } GCEDATA;
  58. typedef struct _GIFFRAME
  59. {
  60. struct _GIFFRAME *pgfNext;
  61. GCEDATA gced; // animation parameters for frame.
  62. int top; // bounds relative to the GIF logical screen
  63. int left;
  64. int width;
  65. int height;
  66. unsigned char *ppixels; // pointer to image pixel data
  67. int cColors; // number of entries in pcolors
  68. PALETTEENTRY *pcolors;
  69. PBITMAPINFO pbmi;
  70. HRGN hrgnVis; // region describing currently visible portion of the frame
  71. int iRgnKind; // region type for hrgnVis
  72. } GIFFRAME, *PGIFFRAME;
  73. typedef struct {
  74. BOOL fAnimating; // TRUE if animation is (still) running
  75. DWORD dwLoopIter; // current iteration of looped animation, not actually used for Netscape compliance reasons
  76. _GIFFRAME * pgfDraw; // last frame we need to draw
  77. DWORD dwNextTimeMS; // Time to display pgfDraw->pgfNext, or next iteration
  78. } GIFANIMATIONSTATE, *PGIFANIMATIONSTATE;
  79. #define dwGIFVerUnknown ((DWORD)0) // unknown version of GIF file
  80. #define dwGIFVer87a ((DWORD)87) // GIF87a file format
  81. #define dwGIFVer89a ((DWORD)89) // GIF89a file format.
  82. typedef struct _GIFANIMDATA
  83. {
  84. BOOL fAnimated; // TRUE if cFrames and pgf define a GIF animation
  85. BOOL fLooped; // TRUE if we've seen a Netscape loop block
  86. BOOL fHasTransparency; // TRUE if a frame is transparent, or if a frame does
  87. // not cover the entire logical screen.
  88. BOOL fNoBWMapping; // TRUE if we saw more than two colors in anywhere in the file.
  89. DWORD dwGIFVer; // GIF Version <see defines above> we need to special case 87a backgrounds
  90. unsigned short cLoops; // A la Netscape, we will treat this as
  91. // "loop forever" if it is zero.
  92. PGIFFRAME pgf; // animation frame entries
  93. PALETTEENTRY *pcolorsGlobal; // GIF global colors - NULL after GIF prepared for screen
  94. PGIFFRAME pgfLastProg; // remember the last frame to be drawn during decoding
  95. DWORD dwLastProgTimeMS; // time at which pgfLastProg was displayed.
  96. } GIFANIMDATA, *PGIFANIMDATA;
  97. /** End Structs **/
  98. #define MAXCOLORMAPSIZE 256
  99. #define TRUE 1
  100. #define FALSE 0
  101. #define CM_RED 0
  102. #define CM_GREEN 1
  103. #define CM_BLUE 2
  104. #define MAX_LWZ_BITS 12
  105. #define INTERLACE 0x40
  106. #define LOCALCOLORMAP 0x80
  107. #define BitSet(byte, bit) (((byte) & (bit)) == (bit))
  108. #define LM_to_uint(a,b) ((((unsigned int) b)<<8)|((unsigned int)a))
  109. #define dwIndefiniteGIFThreshold 300 // 300 seconds == 5 minutes
  110. // If the GIF runs longer than
  111. // this, we will assume the author
  112. // intended an indefinite run.
  113. #define dwMaxGIFBits 13107200 // keep corrupted GIFs from causing
  114. // us to allocate _too_ big a buffer.
  115. // This one is 1280 X 1024 X 10.
  116. typedef struct _GIFSCREEN
  117. {
  118. unsigned long Width;
  119. unsigned long Height;
  120. unsigned char ColorMap[3][MAXCOLORMAPSIZE];
  121. unsigned long BitPixel;
  122. unsigned long ColorResolution;
  123. unsigned long Background;
  124. unsigned long AspectRatio;
  125. }
  126. GIFSCREEN;
  127. typedef struct _GIF89
  128. {
  129. long transparent;
  130. long delayTime;
  131. long inputFlag;
  132. long disposal;
  133. }
  134. GIF89;
  135. #define MAX_STACK_SIZE ((1 << (MAX_LWZ_BITS)) * 2)
  136. #define MAX_TABLE_SIZE (1 << MAX_LWZ_BITS)
  137. typedef struct _GIFINFO
  138. {
  139. IStream *stream;
  140. GIF89 Gif89;
  141. long lGifLoc;
  142. long ZeroDataBlock;
  143. /*
  144. ** Pulled out of nextCode
  145. */
  146. long curbit, lastbit, get_done;
  147. long last_byte;
  148. long return_clear;
  149. /*
  150. ** Out of nextLWZ
  151. */
  152. unsigned short *pstack, *sp;
  153. long stacksize;
  154. long code_size, set_code_size;
  155. long max_code, max_code_size;
  156. long clear_code, end_code;
  157. /*
  158. * Were statics in procedures
  159. */
  160. unsigned char buf[280];
  161. unsigned short *table[2];
  162. long tablesize;
  163. long firstcode, oldcode;
  164. } GIFINFO,*PGIFINFO;
  165. /*
  166. DirectAnimation wrapper class for GIF info
  167. */
  168. class CImgGif
  169. {
  170. // Class methods
  171. public:
  172. CImgGif();
  173. ~CImgGif();
  174. unsigned char * ReadGIFMaster();
  175. BOOL Read(unsigned char *buffer, long len);
  176. long ReadColorMap(long number, unsigned char buffer[3][MAXCOLORMAPSIZE]);
  177. long DoExtension(long label);
  178. long GetDataBlock(unsigned char *buf);
  179. unsigned char * ReadImage(long len, long height, BOOL fInterlace, BOOL fGIFFrame);
  180. long readLWZ();
  181. long nextLWZ();
  182. long nextCode(long code_size);
  183. BOOL initLWZ(long input_code_size);
  184. unsigned short * growStack();
  185. BOOL growTables();
  186. BITMAPINFO * FinishDithering();
  187. // Data members
  188. public:
  189. LPCSTR _szFileName;
  190. BOOL _fInterleaved;
  191. BOOL _fInvalidateAll;
  192. int _yLogRow;
  193. GIFINFO _gifinfo;
  194. GIFANIMATIONSTATE _gas;
  195. GIFANIMDATA _gad;
  196. PALETTEENTRY _ape[256];
  197. int _xWidth;
  198. int _yHeight;
  199. LONG _lTrans;
  200. BYTE * _pbBits;
  201. } GIFIMAGE;
  202. CImgGif::CImgGif() {
  203. _gifinfo.pstack = NULL;
  204. _gifinfo.table[0] = NULL;
  205. _gifinfo.table[1] = NULL;
  206. }
  207. CImgGif::~CImgGif() {
  208. free(_gifinfo.pstack);
  209. free(_gifinfo.table[0]);
  210. free(_gifinfo.table[1]);
  211. PGIFFRAME nextPgf, curPgf;
  212. curPgf = _gad.pgf;
  213. while(curPgf != NULL) {
  214. nextPgf = curPgf->pgfNext;
  215. free(curPgf->ppixels);
  216. free(curPgf->pcolors);
  217. free(curPgf->pbmi);
  218. free(curPgf);
  219. curPgf = nextPgf;
  220. }
  221. }
  222. static int GetColorMode() { return 0; };
  223. #ifndef DEBUG
  224. #pragma optimize("t",on)
  225. #endif
  226. BOOL CImgGif::Read(unsigned char *buffer, long len)
  227. {
  228. DWORD lenout = 0;
  229. /* read len characters into buffer */
  230. _gifinfo.stream->Read(buffer,len,&lenout);
  231. return ((long)lenout == len);
  232. }
  233. long CImgGif::ReadColorMap(long number, unsigned char buffer[3][MAXCOLORMAPSIZE])
  234. {
  235. long i;
  236. unsigned char rgb[3];
  237. for (i = 0; i < number; ++i)
  238. {
  239. if (!Read(rgb, sizeof(rgb)))
  240. {
  241. //bw //bw TraceTag((tagImageDecode, "bad gif colormap."));
  242. return (TRUE);
  243. }
  244. buffer[CM_RED][i] = rgb[0];
  245. buffer[CM_GREEN][i] = rgb[1];
  246. buffer[CM_BLUE][i] = rgb[2];
  247. }
  248. return FALSE;
  249. }
  250. long
  251. CImgGif::GetDataBlock(unsigned char *buf)
  252. {
  253. unsigned char count;
  254. count = 0;
  255. if (!Read(&count, 1))
  256. {
  257. return -1;
  258. }
  259. _gifinfo.ZeroDataBlock = count == 0;
  260. if ((count != 0) && (!Read(buf, count)))
  261. {
  262. return -1;
  263. }
  264. return ((long) count);
  265. }
  266. #define MIN_CODE_BITS 5
  267. #define MIN_STACK_SIZE 64
  268. #define MINIMUM_CODE_SIZE 2
  269. BOOL CImgGif::initLWZ(long input_code_size)
  270. {
  271. if(input_code_size < MINIMUM_CODE_SIZE)
  272. return FALSE;
  273. _gifinfo.set_code_size = input_code_size;
  274. _gifinfo.code_size = _gifinfo.set_code_size + 1;
  275. _gifinfo.clear_code = 1 << _gifinfo.set_code_size;
  276. _gifinfo.end_code = _gifinfo.clear_code + 1;
  277. _gifinfo.max_code_size = 2 * _gifinfo.clear_code;
  278. _gifinfo.max_code = _gifinfo.clear_code + 2;
  279. _gifinfo.curbit = _gifinfo.lastbit = 0;
  280. _gifinfo.last_byte = 2;
  281. _gifinfo.get_done = FALSE;
  282. _gifinfo.return_clear = TRUE;
  283. if(input_code_size >= MIN_CODE_BITS)
  284. _gifinfo.stacksize = ((1 << (input_code_size)) * 2);
  285. else
  286. _gifinfo.stacksize = MIN_STACK_SIZE;
  287. if ( _gifinfo.pstack != NULL )
  288. free( _gifinfo.pstack );
  289. if ( _gifinfo.table[0] != NULL )
  290. free( _gifinfo.table[0] );
  291. if ( _gifinfo.table[1] != NULL )
  292. free( _gifinfo.table[1] );
  293. _gifinfo.table[0] = 0;
  294. _gifinfo.table[1] = 0;
  295. _gifinfo.pstack = 0;
  296. _gifinfo.pstack = (unsigned short *) malloc((_gifinfo.stacksize)*sizeof(unsigned short));
  297. if(_gifinfo.pstack == 0){
  298. goto ErrorExit;
  299. }
  300. _gifinfo.sp = _gifinfo.pstack;
  301. // Initialize the two tables.
  302. _gifinfo.tablesize = (_gifinfo.max_code_size);
  303. _gifinfo.table[0] = (unsigned short *) malloc((_gifinfo.tablesize)*sizeof(unsigned short));
  304. _gifinfo.table[1] = (unsigned short *) malloc((_gifinfo.tablesize)*sizeof(unsigned short));
  305. if((_gifinfo.table[0] == 0) || (_gifinfo.table[1] == 0)){
  306. goto ErrorExit;
  307. }
  308. return TRUE;
  309. ErrorExit:
  310. if(_gifinfo.pstack){
  311. free(_gifinfo.pstack);
  312. _gifinfo.pstack = 0;
  313. }
  314. if(_gifinfo.table[0]){
  315. free(_gifinfo.table[0]);
  316. _gifinfo.table[0] = 0;
  317. }
  318. if(_gifinfo.table[1]){
  319. free(_gifinfo.table[1]);
  320. _gifinfo.table[1] = 0;
  321. }
  322. return FALSE;
  323. }
  324. long CImgGif::nextCode(long code_size)
  325. {
  326. static const long maskTbl[16] =
  327. {
  328. 0x0000, 0x0001, 0x0003, 0x0007,
  329. 0x000f, 0x001f, 0x003f, 0x007f,
  330. 0x00ff, 0x01ff, 0x03ff, 0x07ff,
  331. 0x0fff, 0x1fff, 0x3fff, 0x7fff,
  332. };
  333. long i, j, ret, end;
  334. unsigned char *buf = &_gifinfo.buf[0];
  335. if (_gifinfo.return_clear)
  336. {
  337. _gifinfo.return_clear = FALSE;
  338. return _gifinfo.clear_code;
  339. }
  340. end = _gifinfo.curbit + code_size;
  341. if (end >= _gifinfo.lastbit)
  342. {
  343. long count;
  344. if (_gifinfo.get_done)
  345. {
  346. return -1;
  347. }
  348. buf[0] = buf[_gifinfo.last_byte - 2];
  349. buf[1] = buf[_gifinfo.last_byte - 1];
  350. if ((count = GetDataBlock(&buf[2])) == 0)
  351. _gifinfo.get_done = TRUE;
  352. if (count < 0)
  353. {
  354. return -1;
  355. }
  356. _gifinfo.last_byte = 2 + count;
  357. _gifinfo.curbit = (_gifinfo.curbit - _gifinfo.lastbit) + 16;
  358. _gifinfo.lastbit = (2 + count) * 8;
  359. end = _gifinfo.curbit + code_size;
  360. // Okay, bug 30784 time. It's possible that we only got 1
  361. // measly byte in the last data block. Rare, but it does happen.
  362. // In that case, the additional byte may still not supply us with
  363. // enough bits for the next code, so, as Mars Needs Women, IE
  364. // Needs Data.
  365. if ( end >= _gifinfo.lastbit && !_gifinfo.get_done )
  366. {
  367. // protect ourselve from the ( theoretically impossible )
  368. // case where between the last data block, the 2 bytes from
  369. // the block preceding that, and the potential 0xFF bytes in
  370. // the next block, we overflow the buffer.
  371. // Since count should always be 1,
  372. //bw Assert ( count == 1 );
  373. // there should be enough room in the buffer, so long as someone
  374. // doesn't shrink it.
  375. if ( count + 0x101 >= sizeof( _gifinfo.buf ) )
  376. {
  377. //bw Assert ( FALSE ); //
  378. return -1;
  379. }
  380. if ((count = GetDataBlock(&buf[2 + count])) == 0)
  381. _gifinfo.get_done = TRUE;
  382. if (count < 0)
  383. {
  384. return -1;
  385. }
  386. _gifinfo.last_byte += count;
  387. _gifinfo.lastbit = _gifinfo.last_byte * 8;
  388. end = _gifinfo.curbit + code_size;
  389. }
  390. }
  391. j = end / 8;
  392. i = _gifinfo.curbit / 8;
  393. if (i == j)
  394. ret = buf[i];
  395. else if (i + 1 == j)
  396. ret = buf[i] | (((long) buf[i + 1]) << 8);
  397. else
  398. ret = buf[i] | (((long) buf[i + 1]) << 8) | (((long) buf[i + 2]) << 16);
  399. ret = (ret >> (_gifinfo.curbit % 8)) & maskTbl[code_size];
  400. _gifinfo.curbit += code_size;
  401. return ret;
  402. }
  403. // Grows the stack and returns the top of the stack.
  404. unsigned short *
  405. CImgGif::growStack()
  406. {
  407. long index;
  408. unsigned short *lp;
  409. if (_gifinfo.stacksize >= MAX_STACK_SIZE) return 0;
  410. index = long(_gifinfo.sp - _gifinfo.pstack);
  411. lp = (unsigned short *)realloc(_gifinfo.pstack, (_gifinfo.stacksize)*2*sizeof(unsigned short));
  412. if(lp == 0)
  413. return 0;
  414. _gifinfo.pstack = lp;
  415. _gifinfo.sp = &(_gifinfo.pstack[index]);
  416. _gifinfo.stacksize = (_gifinfo.stacksize)*2;
  417. lp = &(_gifinfo.pstack[_gifinfo.stacksize]);
  418. return lp;
  419. }
  420. BOOL
  421. CImgGif::growTables()
  422. {
  423. unsigned short *lp;
  424. lp = (unsigned short *) realloc(_gifinfo.table[0], (_gifinfo.max_code_size)*sizeof(unsigned short));
  425. if(lp == 0){
  426. return FALSE;
  427. }
  428. _gifinfo.table[0] = lp;
  429. lp = (unsigned short *) realloc(_gifinfo.table[1], (_gifinfo.max_code_size)*sizeof(unsigned short));
  430. if(lp == 0){
  431. return FALSE;
  432. }
  433. _gifinfo.table[1] = lp;
  434. return TRUE;
  435. }
  436. inline
  437. long CImgGif::readLWZ()
  438. {
  439. return((_gifinfo.sp > _gifinfo.pstack) ? *--(_gifinfo.sp) : nextLWZ());
  440. }
  441. #define CODE_MASK 0xffff
  442. long CImgGif::nextLWZ()
  443. {
  444. long code, incode;
  445. unsigned short usi;
  446. unsigned short *table0 = _gifinfo.table[0];
  447. unsigned short *table1 = _gifinfo.table[1];
  448. unsigned short *pstacktop = &(_gifinfo.pstack[_gifinfo.stacksize]);
  449. while ((code = nextCode(_gifinfo.code_size)) >= 0)
  450. {
  451. if (code == _gifinfo.clear_code)
  452. {
  453. /* corrupt GIFs can make this happen */
  454. if (_gifinfo.clear_code >= (1 << MAX_LWZ_BITS))
  455. {
  456. return -2;
  457. }
  458. _gifinfo.code_size = _gifinfo.set_code_size + 1;
  459. _gifinfo.max_code_size = 2 * _gifinfo.clear_code;
  460. _gifinfo.max_code = _gifinfo.clear_code + 2;
  461. if(!growTables())
  462. return -2;
  463. table0 = _gifinfo.table[0];
  464. table1 = _gifinfo.table[1];
  465. _gifinfo.tablesize = _gifinfo.max_code_size;
  466. for (usi = 0; usi < _gifinfo.clear_code; ++usi)
  467. {
  468. table1[usi] = usi;
  469. }
  470. memset(table0,0,sizeof(unsigned short )*(_gifinfo.tablesize));
  471. memset(&table1[_gifinfo.clear_code],0,sizeof(unsigned short)*((_gifinfo.tablesize)-_gifinfo.clear_code));
  472. _gifinfo.sp = _gifinfo.pstack;
  473. do
  474. {
  475. _gifinfo.firstcode = _gifinfo.oldcode = nextCode(_gifinfo.code_size);
  476. }
  477. while (_gifinfo.firstcode == _gifinfo.clear_code);
  478. return _gifinfo.firstcode;
  479. }
  480. if (code == _gifinfo.end_code)
  481. {
  482. long count;
  483. unsigned char buf[260];
  484. if (_gifinfo.ZeroDataBlock)
  485. {
  486. return -2;
  487. }
  488. while ((count = GetDataBlock(buf)) > 0)
  489. ;
  490. if (count != 0)
  491. return -2;
  492. }
  493. incode = code;
  494. if (code >= _gifinfo.max_code)
  495. {
  496. if (_gifinfo.sp >= pstacktop){
  497. pstacktop = growStack();
  498. if(pstacktop == 0)
  499. return -2;
  500. }
  501. *(_gifinfo.sp)++ = (unsigned short)((CODE_MASK ) & (_gifinfo.firstcode));
  502. code = _gifinfo.oldcode;
  503. }
  504. #if FEATURE_FAST
  505. // (andyp) easy speedup here for ie3.1 (too late for ie3.0):
  506. //
  507. // 1. move growStack code out of loop (use max 12-bit/4k slop).
  508. // 2. do "sp = _gifinfo.sp" so it will get enreg'ed.
  509. // 3. un-inline growStack (and growTables).
  510. // 4. change short's to int's (benefits win32) (esp. table1 & table2)
  511. // (n.b. int not long, so we'll keep win3.1 perf)
  512. // 5. change long's to int's (benefits win16) (esp. code).
  513. //
  514. // together these will make the loop very tight w/ everything kept
  515. // enregistered and no 66 overrides.
  516. //
  517. // one caveat is that on average this loop iterates 4x so it's
  518. // not clear how much the speedup will really gain us until we
  519. // look at the outer loop as well.
  520. #endif
  521. while (code >= _gifinfo.clear_code)
  522. {
  523. if (_gifinfo.sp >= pstacktop){
  524. pstacktop = growStack();
  525. if(pstacktop == 0)
  526. return -2;
  527. }
  528. *(_gifinfo.sp)++ = table1[code];
  529. if (code == (long)(table0[code]))
  530. {
  531. return (code);
  532. }
  533. code = (long)(table0[code]);
  534. }
  535. if (_gifinfo.sp >= pstacktop){
  536. pstacktop = growStack();
  537. if(pstacktop == 0)
  538. return -2;
  539. }
  540. _gifinfo.firstcode = (long)table1[code];
  541. *(_gifinfo.sp)++ = table1[code];
  542. if ((code = _gifinfo.max_code) < (1 << MAX_LWZ_BITS))
  543. {
  544. table0[code] = (USHORT)(_gifinfo.oldcode) & CODE_MASK;
  545. table1[code] = (USHORT)(_gifinfo.firstcode) & CODE_MASK;
  546. ++_gifinfo.max_code;
  547. if ((_gifinfo.max_code >= _gifinfo.max_code_size) && (_gifinfo.max_code_size < ((1 << MAX_LWZ_BITS))))
  548. {
  549. _gifinfo.max_code_size *= 2;
  550. ++_gifinfo.code_size;
  551. if(!growTables())
  552. return -2;
  553. table0 = _gifinfo.table[0];
  554. table1 = _gifinfo.table[1];
  555. // Tables have been reallocated to the correct size but initialization
  556. // still remains to be done. This initialization is different from
  557. // the first time initialization of these tables.
  558. memset(&(table0[_gifinfo.tablesize]),0,
  559. sizeof(unsigned short )*(_gifinfo.max_code_size - _gifinfo.tablesize));
  560. memset(&(table1[_gifinfo.tablesize]),0,
  561. sizeof(unsigned short )*(_gifinfo.max_code_size - _gifinfo.tablesize));
  562. _gifinfo.tablesize = (_gifinfo.max_code_size);
  563. }
  564. }
  565. _gifinfo.oldcode = incode;
  566. if (_gifinfo.sp > _gifinfo.pstack)
  567. return ((long)(*--(_gifinfo.sp)));
  568. }
  569. return code;
  570. }
  571. #ifndef DEBUG
  572. // Return to default optimization flags
  573. #pragma optimize("",on)
  574. #endif
  575. unsigned char *
  576. CImgGif::ReadImage(long len, long height, BOOL fInterlace, BOOL fGIFFrame)
  577. {
  578. unsigned char *dp, c;
  579. long v;
  580. long xpos = 0, ypos = 0, pass = 0;
  581. unsigned char *image;
  582. long padlen = ((len + 3) / 4) * 4;
  583. DWORD cbImage = 0;
  584. char buf[256]; // need a buffer to read trailing blocks ( up to terminator ) into
  585. //ULONG ulCoversImg = IMGBITS_PARTIAL;
  586. /*
  587. ** Initialize the Compression routines
  588. */
  589. if (!Read(&c, 1))
  590. {
  591. return (NULL);
  592. }
  593. /*
  594. ** If this is an "uninteresting picture" ignore it.
  595. */
  596. cbImage = padlen * height * sizeof(char);
  597. if ( cbImage > dwMaxGIFBits
  598. || (image = (unsigned char *) _calloc(1, cbImage)) == NULL)
  599. {
  600. //bw TraceTag((tagImageDecode, "Cannot allocate space for gif image data\n"));
  601. return (NULL);
  602. }
  603. if (c == 1)
  604. {
  605. // Netscape seems to field these bogus GIFs by filling treating them
  606. // as transparent. While not the optimal way to simulate this effect,
  607. // we'll fake it by pushing the initial code size up to a safe value,
  608. // consuming the input, and returning a buffer full of the transparent
  609. // color or zero, if no transparency is indicated.
  610. if (initLWZ(MINIMUM_CODE_SIZE))
  611. while (readLWZ() >= 0);
  612. else {
  613. //bw TraceTag((tagImageDecode, "GIF: failed LZW decode.\n"));
  614. return (NULL);
  615. }
  616. if (_gifinfo.Gif89.transparent != -1)
  617. FillMemory(image, cbImage, (BYTE)_gifinfo.Gif89.transparent);
  618. else // fall back on the background color
  619. FillMemory(image, cbImage, 0);
  620. return image;
  621. }
  622. else if (initLWZ(c) == FALSE)
  623. {
  624. free(image);
  625. //bw TraceTag((tagImageDecode, "GIF: failed LZW decode.\n"));
  626. return NULL;
  627. }
  628. if (!fGIFFrame)
  629. _pbBits = image;
  630. if (fInterlace)
  631. {
  632. long i;
  633. long pass = 0, step = 8;
  634. if (!fGIFFrame && (height > 4))
  635. _fInterleaved = TRUE;
  636. for (i = 0; i < height; i++)
  637. {
  638. // message("readimage, logical=%d, offset=%d\n", i, padlen * ((height-1) - ypos));
  639. dp = &image[padlen * ((height-1) - ypos)];
  640. for (xpos = 0; xpos < len; xpos++)
  641. {
  642. if ((v = readLWZ()) < 0)
  643. goto abort;
  644. *dp++ = (unsigned char) v;
  645. }
  646. ypos += step;
  647. while (ypos >= height)
  648. {
  649. if (pass++ > 0)
  650. step /= 2;
  651. ypos = step / 2;
  652. /*if (!fGIFFrame && pass == 1)
  653. {
  654. ulCoversImg = IMGBITS_TOTAL;
  655. }*/
  656. }
  657. if (!fGIFFrame)
  658. {
  659. _yLogRow = i;
  660. /*if ((i & PROG_INTERVAL) == 0)
  661. {
  662. // Post ProgDraw (IE code has delay-logic)
  663. OnProg(FALSE, ulCoversImg);
  664. }*/
  665. }
  666. }
  667. /*if (!fGIFFrame)
  668. {
  669. OnProg(TRUE, ulCoversImg);
  670. }*/
  671. if (!fGIFFrame && height <= 4)
  672. {
  673. _yLogRow = height-1;
  674. }
  675. }
  676. else
  677. {
  678. if (!fGIFFrame)
  679. _yLogRow = -1;
  680. for (ypos = height-1; ypos >= 0; ypos--)
  681. {
  682. dp = &image[padlen * ypos];
  683. for (xpos = 0; xpos < len; xpos++)
  684. {
  685. if ((v = readLWZ()) < 0)
  686. goto abort;
  687. *dp++ = (unsigned char) v;
  688. }
  689. if (!fGIFFrame)
  690. {
  691. _yLogRow++;
  692. // message("readimage, logical=%d, offset=%d\n", _yLogRow, padlen * ypos);
  693. /*if ((_yLogRow & PROG_INTERVAL) == 0)
  694. {
  695. // Post ProgDraw (IE code has delay-logic)
  696. OnProg(FALSE, ulCoversImg);
  697. }*/
  698. }
  699. }
  700. /*if (!fGIFFrame)
  701. {
  702. OnProg(TRUE, ulCoversImg);
  703. }*/
  704. }
  705. // consume blocks up to image block terminator so we can proceed to the next image
  706. while (GetDataBlock((unsigned char *) buf) > 0)
  707. ;
  708. return (image);
  709. abort:
  710. /*if (!fGIFFrame)
  711. OnProg(TRUE, ulCoversImg);*/
  712. return NULL;
  713. }
  714. long CImgGif::DoExtension(long label)
  715. {
  716. unsigned char buf[256];
  717. int count;
  718. switch (label)
  719. {
  720. case 0x01: /* Plain Text Extension */
  721. break;
  722. case 0xff: /* Application Extension */
  723. // Is it the Netscape looping extension
  724. count = GetDataBlock((unsigned char *) buf);
  725. if (count >= 11)
  726. {
  727. char *szNSExt = "NETSCAPE2.0";
  728. if ( memcmp( buf, szNSExt, strlen( szNSExt ) ) == 0 )
  729. { // if it has their signature, get the data subblock with the iter count
  730. count = GetDataBlock((unsigned char *) buf);
  731. if ( count >= 3 )
  732. {
  733. _gad.fLooped = TRUE;
  734. _gad.cLoops = (buf[2] << 8) | buf[1];
  735. }
  736. }
  737. }
  738. while (GetDataBlock((unsigned char *) buf) > 0)
  739. ;
  740. return FALSE;
  741. break;
  742. case 0xfe: /* Comment Extension */
  743. while (GetDataBlock((unsigned char *) buf) > 0)
  744. {
  745. //bw TraceTag((tagImageDecode, "GIF comment: %s\n", buf));
  746. }
  747. return FALSE;
  748. case 0xf9: /* Graphic Control Extension */
  749. count = GetDataBlock((unsigned char *) buf);
  750. if (count >= 3)
  751. {
  752. _gifinfo.Gif89.disposal = (buf[0] >> 2) & 0x7;
  753. _gifinfo.Gif89.inputFlag = (buf[0] >> 1) & 0x1;
  754. _gifinfo.Gif89.delayTime = LM_to_uint(buf[1], buf[2]);
  755. if ((buf[0] & 0x1) != 0)
  756. _gifinfo.Gif89.transparent = buf[3];
  757. else
  758. _gifinfo.Gif89.transparent = -1;
  759. }
  760. while (GetDataBlock((unsigned char *) buf) > 0)
  761. ;
  762. return FALSE;
  763. default:
  764. break;
  765. }
  766. while (GetDataBlock((unsigned char *) buf) > 0)
  767. ;
  768. return FALSE;
  769. }
  770. BOOL IsGifHdr(BYTE * pb)
  771. {
  772. return(pb[0] == 'G' && pb[1] == 'I' && pb[2] == 'F'
  773. && pb[3] == '8' && (pb[4] == '7' || pb[4] == '9') && pb[5] == 'a');
  774. }
  775. PBITMAPINFO x_8BPIBitmap(int xsize, int ysize)
  776. {
  777. PBITMAPINFO pbmi;
  778. if (GetColorMode() == 8)
  779. {
  780. pbmi = (PBITMAPINFO) _calloc(1, sizeof(BITMAPINFOHEADER) + 256 * sizeof(WORD));
  781. if (!pbmi)
  782. {
  783. return NULL;
  784. }
  785. pbmi->bmiHeader.biSize = sizeof(BITMAPINFOHEADER);
  786. pbmi->bmiHeader.biWidth = xsize;
  787. pbmi->bmiHeader.biHeight = ysize;
  788. pbmi->bmiHeader.biPlanes = 1;
  789. pbmi->bmiHeader.biBitCount = 8;
  790. pbmi->bmiHeader.biCompression = BI_RGB; /* no compression */
  791. pbmi->bmiHeader.biSizeImage = 0; /* not needed when not compressed */
  792. pbmi->bmiHeader.biXPelsPerMeter = 0;
  793. pbmi->bmiHeader.biYPelsPerMeter = 0;
  794. pbmi->bmiHeader.biClrUsed = 256;
  795. pbmi->bmiHeader.biClrImportant = 0;
  796. }
  797. else
  798. {
  799. pbmi = (PBITMAPINFO) _calloc(1, sizeof(BITMAPINFOHEADER) + 256 * sizeof(RGBQUAD));
  800. if (!pbmi)
  801. {
  802. return NULL;
  803. }
  804. pbmi->bmiHeader.biSize = sizeof(BITMAPINFOHEADER);
  805. pbmi->bmiHeader.biWidth = xsize;
  806. pbmi->bmiHeader.biHeight = ysize;
  807. pbmi->bmiHeader.biPlanes = 1;
  808. pbmi->bmiHeader.biBitCount = 8;
  809. pbmi->bmiHeader.biCompression = BI_RGB; /* no compression */
  810. pbmi->bmiHeader.biSizeImage = 0; /* not needed when not compressed */
  811. pbmi->bmiHeader.biXPelsPerMeter = 0;
  812. pbmi->bmiHeader.biYPelsPerMeter = 0;
  813. pbmi->bmiHeader.biClrUsed = 256;
  814. pbmi->bmiHeader.biClrImportant = 0;
  815. }
  816. return pbmi;
  817. }
  818. /*
  819. For color images.
  820. This routine should only be used when drawing to an 8 bit palette screen.
  821. It always creates a DIB in DIB_PAL_COLORS format.
  822. */
  823. PBITMAPINFO BIT_Make_DIB_PAL_Header(int xsize, int ysize)
  824. {
  825. int i;
  826. PBITMAPINFO pbmi;
  827. WORD *pw;
  828. pbmi = x_8BPIBitmap(xsize, ysize);
  829. if (!pbmi)
  830. {
  831. return NULL;
  832. }
  833. pw = (WORD *) pbmi->bmiColors;
  834. for (i = 0; i < 256; i++)
  835. {
  836. pw[i] = (WORD)i;
  837. }
  838. return pbmi;
  839. }
  840. /*
  841. For color images.
  842. This routine is used when drawing to the nonpalette screens. It always creates
  843. DIBs in DIB_RGB_COLORS format.
  844. If there is a transparent color, it is modified in the palette to be the
  845. background color for the window.
  846. */
  847. PBITMAPINFO BIT_Make_DIB_RGB_Header_Screen(int xsize, int ysize,
  848. int cEntries, PALETTEENTRY * rgpe, int transparent)
  849. {
  850. int i;
  851. PBITMAPINFO pbmi;
  852. pbmi = x_8BPIBitmap(xsize, ysize);
  853. if (!pbmi)
  854. {
  855. return NULL;
  856. }
  857. for (i = 0; i < cEntries; i++)
  858. {
  859. pbmi->bmiColors[i].rgbRed = rgpe[i].peRed;
  860. pbmi->bmiColors[i].rgbGreen = rgpe[i].peGreen;
  861. pbmi->bmiColors[i].rgbBlue = rgpe[i].peBlue;
  862. pbmi->bmiColors[i].rgbReserved = 0;
  863. }
  864. /*
  865. if (transparent != -1)
  866. {
  867. COLORREF color;
  868. color = PREF_GetBackgroundColor();
  869. pbmi->bmiColors[transparent].rgbRed = GetRValue(color);
  870. pbmi->bmiColors[transparent].rgbGreen = GetGValue(color);
  871. pbmi->bmiColors[transparent].rgbBlue = GetBValue(color);
  872. }
  873. */
  874. return pbmi;
  875. }
  876. unsigned char *
  877. CImgGif::ReadGIFMaster()
  878. {
  879. HRESULT hr = S_OK;
  880. unsigned char buf[16];
  881. unsigned char c;
  882. unsigned char localColorMap[3][MAXCOLORMAPSIZE];
  883. long useGlobalColormap;
  884. long imageCount = 0;
  885. long imageNumber = 1;
  886. unsigned char *image = NULL;
  887. unsigned long i;
  888. GIFSCREEN GifScreen;
  889. long bitPixel;
  890. PGIFFRAME pgfLast = NULL;
  891. PGIFFRAME pgfNew;
  892. _gifinfo.ZeroDataBlock = 0;
  893. /*
  894. * Initialize GIF89 extensions
  895. */
  896. _gifinfo.Gif89.transparent = -1;
  897. _gifinfo.Gif89.delayTime = 5;
  898. _gifinfo.Gif89.inputFlag = -1;
  899. _gifinfo.Gif89.disposal = 0;
  900. _gifinfo.lGifLoc = 0;
  901. // initialize our animation fields
  902. _gad.fAnimated = FALSE; // set to TRUE if we see more than one image
  903. _gad.fLooped = FALSE; // TRUE if we've seen a Netscape loop block
  904. _gad.fHasTransparency = FALSE; // until proven otherwise
  905. _gad.fNoBWMapping = FALSE;
  906. _gad.dwGIFVer = dwGIFVerUnknown;
  907. _gad.cLoops = 0;
  908. _gad.pgf = NULL;
  909. _gad.pcolorsGlobal = NULL;
  910. if (!Read(buf, 6))
  911. {
  912. //bw TraceTag((tagImageDecode, "GIF: error reading magic number\n"));
  913. hr = E_FAIL;
  914. goto done;
  915. }
  916. if (!IsGifHdr(buf)) {
  917. //bw TraceTag((tagImageDecode, "GIF: Malformed header\n"));
  918. hr = E_FAIL;
  919. goto done;
  920. }
  921. _gad.dwGIFVer = (buf[4] == '7') ? dwGIFVer87a : dwGIFVer89a;
  922. if (!Read(buf, 7))
  923. {
  924. //bw TraceTag((tagImageDecode, "GIF: failed to read screen descriptor\n"));
  925. hr = E_FAIL;
  926. goto done;
  927. }
  928. GifScreen.Width = LM_to_uint(buf[0], buf[1]);
  929. GifScreen.Height = LM_to_uint(buf[2], buf[3]);
  930. GifScreen.BitPixel = 2 << (buf[4] & 0x07);
  931. GifScreen.ColorResolution = (((buf[4] & 0x70) >> 3) + 1);
  932. GifScreen.Background = buf[5];
  933. GifScreen.AspectRatio = buf[6];
  934. if (BitSet(buf[4], LOCALCOLORMAP))
  935. { /* Global Colormap */
  936. int scale = 65536 / MAXCOLORMAPSIZE;
  937. if (ReadColorMap(GifScreen.BitPixel, GifScreen.ColorMap))
  938. {
  939. //bw TraceTag((tagImageDecode, "error reading global colormap\n"));
  940. hr = E_FAIL;
  941. goto done;
  942. }
  943. for (i = 0; i < GifScreen.BitPixel; i++)
  944. {
  945. int tmp;
  946. tmp = (BYTE) (GifScreen.ColorMap[0][i]);
  947. _ape[i].peRed = (BYTE) (GifScreen.ColorMap[0][i]);
  948. _ape[i].peGreen = (BYTE) (GifScreen.ColorMap[1][i]);
  949. _ape[i].peBlue = (BYTE) (GifScreen.ColorMap[2][i]);
  950. _ape[i].peFlags = (BYTE) 0;
  951. }
  952. for (i = GifScreen.BitPixel; i < MAXCOLORMAPSIZE; i++)
  953. {
  954. _ape[i].peRed = (BYTE) 0;
  955. _ape[i].peGreen = (BYTE) 0;
  956. _ape[i].peBlue = (BYTE) 0;
  957. _ape[i].peFlags = (BYTE) 0;
  958. }
  959. }
  960. if (GifScreen.AspectRatio != 0 && GifScreen.AspectRatio != 49)
  961. {
  962. float r;
  963. r = ((float) (GifScreen.AspectRatio) + (float) 15.0) / (float) 64.0;
  964. //bw TraceTag((tagImageDecode, "Warning: non-square pixels!\n"));
  965. }
  966. for (;; ) // our appetite now knows no bounds save termination or error
  967. {
  968. if (!Read(&c, 1))
  969. {
  970. //bw TraceTag((tagImageDecode, "EOF / read error on image data\n"));
  971. hr = E_FAIL;
  972. goto done;
  973. }
  974. if (c == ';')
  975. { /* GIF terminator */
  976. if (imageCount < imageNumber)
  977. {
  978. //bw TraceTag((tagImageDecode, "No images found in file\n"));
  979. hr = E_FAIL;
  980. goto done;
  981. }
  982. break;
  983. }
  984. if (c == '!')
  985. { /* Extension */
  986. if (!Read(&c, 1))
  987. {
  988. //bw TraceTag((tagImageDecode, "EOF / read error on extension function code\n"));
  989. hr = E_FAIL;
  990. goto done;
  991. }
  992. DoExtension(c);
  993. continue;
  994. }
  995. if (c != ',')
  996. { /* Not a valid start character */
  997. break;
  998. }
  999. ++imageCount;
  1000. if (!Read(buf, 9))
  1001. {
  1002. //bw TraceTag((tagImageDecode, "couldn't read left/top/width/height\n"));
  1003. hr = E_FAIL;
  1004. goto done;
  1005. }
  1006. useGlobalColormap = !BitSet(buf[8], LOCALCOLORMAP);
  1007. bitPixel = 1 << ((buf[8] & 0x07) + 1);
  1008. /*
  1009. * We only want to set width and height for the imageNumber
  1010. * we are requesting.
  1011. */
  1012. if (imageCount == imageNumber)
  1013. {
  1014. // Replicate some of Netscape's special cases:
  1015. // Don't use the logical screen if it's a GIF87a and the topLeft of the first image is at the origin.
  1016. // Don't use the logical screen if the first image spills out of the logical screen.
  1017. // These are artifacts of primitive authoring tools falling into the hands of hapless users.
  1018. RECT rectImage; // rect defining bounds of GIF
  1019. RECT rectLS; // rect defining bounds of GIF logical screen.
  1020. RECT rectSect; // intersection of image an logical screen
  1021. BOOL fNoSpill; // True if the image doesn't spill out of the logical screen
  1022. BOOL fGoofy87a; // TRUE if its one of the 87a pathologies that Netscape special cases
  1023. rectImage.left = LM_to_uint(buf[0], buf[1]);
  1024. rectImage.top = LM_to_uint(buf[2], buf[3]);
  1025. rectImage.right = rectImage.left + LM_to_uint(buf[4], buf[5]);
  1026. rectImage.bottom = rectImage.top + LM_to_uint(buf[6], buf[7]);
  1027. rectLS.left = rectLS.top = 0;
  1028. rectLS.right = GifScreen.Width;
  1029. rectLS.bottom = GifScreen.Height;
  1030. IntersectRect( &rectSect, &rectImage, &rectLS );
  1031. fNoSpill = EqualRect( &rectImage, &rectSect );
  1032. fGoofy87a = FALSE;
  1033. if (_gad.dwGIFVer == dwGIFVer87a)
  1034. {
  1035. // netscape ignores the logical screen if the image is flush against
  1036. // either the upper left or lower right corner
  1037. fGoofy87a = (rectImage.top == 0 && rectImage.left == 0) ||
  1038. (rectImage.bottom == rectLS.bottom &&
  1039. rectImage.right == rectLS.right);
  1040. }
  1041. if (!fGoofy87a && fNoSpill)
  1042. {
  1043. _xWidth = GifScreen.Width;
  1044. _yHeight = GifScreen.Height;
  1045. }
  1046. else
  1047. {
  1048. // Something is amiss. Fall back to the image's dimensions.
  1049. // If the sizes match, but the image is offset, or we're ignoring
  1050. // the logical screen cuz it's a goofy 87a, then pull it back to
  1051. // to the origin
  1052. if ((LM_to_uint(buf[4], buf[5]) == GifScreen.Width &&
  1053. LM_to_uint(buf[6], buf[7]) == GifScreen.Height) ||
  1054. fGoofy87a)
  1055. {
  1056. buf[0] = buf[1] = 0; // left corner to zero
  1057. buf[2] = buf[3] = 0; // top to zero.
  1058. }
  1059. _xWidth = LM_to_uint(buf[4], buf[5]);
  1060. _yHeight = LM_to_uint(buf[6], buf[7]);
  1061. }
  1062. _lTrans = _gifinfo.Gif89.transparent;
  1063. // Post WHKNOWN
  1064. //OnSize(_xWidth, _yHeight, _lTrans);
  1065. }
  1066. if (!useGlobalColormap)
  1067. {
  1068. if (ReadColorMap(bitPixel, localColorMap))
  1069. {
  1070. //bw TraceTag((tagImageDecode, "error reading local colormap\n"));
  1071. hr = E_FAIL;
  1072. goto done;
  1073. }
  1074. }
  1075. // We allocate a frame record for each imag in the GIF stream, including
  1076. // the first/primary image.
  1077. pgfNew = (PGIFFRAME) _calloc(1, sizeof(GIFFRAME));
  1078. if ( pgfNew == NULL )
  1079. {
  1080. //bw TraceTag((tagImageDecode, "not enough memory for GIF frame\n"));
  1081. hr = E_FAIL;
  1082. goto done;
  1083. }
  1084. if ( _gifinfo.Gif89.delayTime != -1 )
  1085. {
  1086. // we have a fresh control extension for this block
  1087. // convert to milliseconds
  1088. pgfNew->gced.uiDelayTime = _gifinfo.Gif89.delayTime * 10;
  1089. //REVIEW(seanf): crude hack to cope with 'degenerate animations' whose timing is set to some
  1090. // small value becaue of the delays imposed by Netscape's animation process
  1091. if ( pgfNew->gced.uiDelayTime <= 50 ) // assume these small values imply Netscape encoding delay
  1092. pgfNew->gced.uiDelayTime = 100; // pick a larger value s.t. the frame will be visible
  1093. pgfNew->gced.uiDisposalMethod = _gifinfo.Gif89.disposal;
  1094. pgfNew->gced.fTransparent = _gifinfo.Gif89.transparent != -1;
  1095. pgfNew->gced.ucTransIndex = (unsigned char)_gifinfo.Gif89.transparent;
  1096. }
  1097. else
  1098. { // fake one up s.t. GIFs that rely solely on Netscape's delay to time their animations will play
  1099. // The spec says that the scope of one of these blocks is the image after the block.
  1100. // Netscape says 'until further notice'. So we play it their way up to a point. We
  1101. // propagate the disposal method and transparency. Since Netscape doesn't honor the timing
  1102. // we use our default timing for these images.
  1103. pgfNew->gced.uiDelayTime = 100;
  1104. pgfNew->gced.uiDisposalMethod = _gifinfo.Gif89.disposal;
  1105. pgfNew->gced.fTransparent = _gifinfo.Gif89.transparent != -1;
  1106. pgfNew->gced.ucTransIndex = (unsigned char)_gifinfo.Gif89.transparent;
  1107. }
  1108. pgfNew->top = LM_to_uint(buf[2], buf[3]); // bounds relative to the GIF logical screen
  1109. pgfNew->left = LM_to_uint(buf[0], buf[1]);
  1110. pgfNew->width = LM_to_uint(buf[4], buf[5]);
  1111. pgfNew->height = LM_to_uint(buf[6], buf[7]);
  1112. // Images that are offset, or do not cover the full logical screen are 'transparent' in the
  1113. // sense that they require us to matte the frame onto the background.
  1114. if (!_gad.fHasTransparency && (pgfNew->gced.fTransparent ||
  1115. pgfNew->top != 0 ||
  1116. pgfNew->left != 0 ||
  1117. (UINT)pgfNew->width != (UINT)GifScreen.Width ||
  1118. (UINT)pgfNew->height != (UINT)GifScreen.Height))
  1119. {
  1120. _gad.fHasTransparency = TRUE;
  1121. //if (_lTrans == -1)
  1122. // OnTrans(0);
  1123. }
  1124. // We don't need to allocate a handle for the simple region case.
  1125. // FrancisH says Windows is too much of a cheapskate to allow us the simplicity
  1126. // of allocating the region once and modifying as needed. Well, okay, he didn't
  1127. // put it that way...
  1128. pgfNew->hrgnVis = NULL;
  1129. pgfNew->iRgnKind = NULLREGION;
  1130. if (!useGlobalColormap)
  1131. {
  1132. // remember that we saw a local color table and only map two-color images
  1133. // if we have a homogenous color environment
  1134. _gad.fNoBWMapping = _gad.fNoBWMapping || bitPixel > 2;
  1135. // CALLOC will set unused colors to <0,0,0,0>
  1136. pgfNew->pcolors = (PALETTEENTRY *) _calloc(MAXCOLORMAPSIZE, sizeof(PALETTEENTRY));
  1137. if ( pgfNew->pcolors == NULL )
  1138. {
  1139. DeleteRgn( pgfNew->hrgnVis );
  1140. free( pgfNew );
  1141. //bw TraceTag((tagImageDecode, "not enough memory for GIF frame colors\n"));
  1142. hr = E_FAIL;
  1143. goto done;
  1144. }
  1145. else
  1146. {
  1147. for (i = 0; i < (ULONG)bitPixel; ++i)
  1148. {
  1149. pgfNew->pcolors[i].peRed = localColorMap[CM_RED][i];
  1150. pgfNew->pcolors[i].peGreen = localColorMap[CM_GREEN][i];
  1151. pgfNew->pcolors[i].peBlue = localColorMap[CM_BLUE][i];
  1152. }
  1153. pgfNew->cColors = bitPixel;
  1154. }
  1155. }
  1156. else
  1157. {
  1158. if ( _gad.pcolorsGlobal == NULL )
  1159. { // Whoa! Somebody's interested in the global color table
  1160. // CALLOC will set unused colors to <0,0,0,0>
  1161. _gad.pcolorsGlobal = (PALETTEENTRY *) _calloc(MAXCOLORMAPSIZE, sizeof(PALETTEENTRY));
  1162. _gad.fNoBWMapping = _gad.fNoBWMapping || GifScreen.BitPixel > 2;
  1163. if ( _gad.pcolorsGlobal != NULL )
  1164. {
  1165. CopyMemory(_gad.pcolorsGlobal, _ape,
  1166. GifScreen.BitPixel * sizeof(PALETTEENTRY) );
  1167. }
  1168. else
  1169. {
  1170. DeleteRgn( pgfNew->hrgnVis );
  1171. free( pgfNew );
  1172. //bw TraceTag((tagImageDecode, "not enough memory for GIF frame colors\n"));
  1173. hr = E_FAIL;
  1174. goto done;
  1175. }
  1176. }
  1177. pgfNew->cColors = GifScreen.BitPixel;
  1178. pgfNew->pcolors = _gad.pcolorsGlobal;
  1179. }
  1180. // Get this in here so that GifStrectchDIBits can use it during progressive
  1181. // rendering.
  1182. if ( _gad.pgf == NULL )
  1183. _gad.pgf = pgfNew;
  1184. pgfNew->ppixels = ReadImage(LM_to_uint(buf[4], buf[5]), // width
  1185. LM_to_uint(buf[6], buf[7]), // height
  1186. BitSet(buf[8], INTERLACE),
  1187. imageCount != imageNumber);
  1188. if ( pgfNew->ppixels != NULL )
  1189. {
  1190. // Oh JOY of JOYS! We got the pixels!
  1191. if (pgfLast != NULL)
  1192. {
  1193. int transparent = (pgfNew->gced.fTransparent) ? (int) pgfNew->gced.ucTransIndex : -1;
  1194. _gad.fAnimated = TRUE; // say multi-image == animated
  1195. if (GetColorMode() == 8) // palettized, use DIB_PAL_COLORS
  1196. { // This will also dither the bits to the screen palette
  1197. pgfNew->pbmi = BIT_Make_DIB_PAL_Header(pgfNew->width, pgfNew->height);
  1198. //if (x_Dither(pgfNew->ppixels, pgfNew->pcolors, pgfNew->width, pgfNew->height, transparent))
  1199. // goto exitPoint;
  1200. }
  1201. else // give it an RGB header
  1202. {
  1203. pgfNew->pbmi = BIT_Make_DIB_RGB_Header_Screen(
  1204. pgfNew->width,
  1205. pgfNew->height,
  1206. pgfNew->cColors, pgfNew->pcolors,
  1207. transparent);
  1208. }
  1209. // Okay, so we've done any mapping on the GIFFRAME, so there's
  1210. // no need to keep the pcolors around. Let's go can clear out
  1211. // the pcolors.
  1212. // REVIEW(seanf): This assumes a common palette is used by all
  1213. // clients of the image
  1214. if ( pgfNew->pcolors != NULL && pgfNew->pcolors != _gad.pcolorsGlobal )
  1215. free( pgfNew->pcolors );
  1216. pgfNew->pcolors = NULL;
  1217. pgfLast->pgfNext = pgfNew;
  1218. // Do something to here to get the new frame on the screen.
  1219. _fInvalidateAll = TRUE;
  1220. //super::OnProg(FALSE, IMGBITS_TOTAL);
  1221. }
  1222. else
  1223. { // first frame
  1224. _gad.pgf = pgfNew;
  1225. _gad.pgfLastProg = pgfNew;
  1226. _gad.dwLastProgTimeMS = 0;
  1227. // set up a temporary animation state for use in progressive draw
  1228. _gas.fAnimating = TRUE;
  1229. _gas.dwLoopIter = 0;
  1230. _gas.pgfDraw = pgfNew;
  1231. if ( imageCount == imageNumber )
  1232. image = pgfNew->ppixels;
  1233. }
  1234. pgfLast = pgfNew;
  1235. }
  1236. // make the _gifinfo.Gif89.delayTime stale, so we know if we got a new
  1237. // GCE for the next image
  1238. _gifinfo.Gif89.delayTime = -1;
  1239. }
  1240. if ( imageCount > imageNumber )
  1241. _gad.fAnimated = TRUE; // say multi-image == animated
  1242. #ifdef FEATURE_GIF_ANIMATION_LONG_LOOP_GOES_INFINITE
  1243. // RAID #23709 - If an animation is sufficiently long, we treat it as indefinite...
  1244. // Indefinite stays indefinite.
  1245. // 5/29/96 - JCordell sez we shouldn't introduce this gratuitous NS incompatibility.
  1246. // We'll keep it around inside this ifdef in case we decide we want it.
  1247. if ( _gad.fLooped &&
  1248. (_gad.dwLoopDurMS * _gad.cLoops) / 1000 > dwIndefiniteGIFThreshold ) // if longer than five minutes
  1249. _gad.cLoops = 0; // set to indefinite looping.
  1250. #endif // FEATURE_GIF_ANIMATION_LONG_LOOP_GOES_INFINITE
  1251. done:
  1252. return image;
  1253. }
  1254. BITMAPINFO *
  1255. CImgGif::FinishDithering()
  1256. {
  1257. BITMAPINFO * pbmi;
  1258. if (GetColorMode() == 8)
  1259. {
  1260. pbmi = BIT_Make_DIB_PAL_Header(_gad.pgf->width, _gad.pgf->height);
  1261. }
  1262. else
  1263. {
  1264. pbmi = BIT_Make_DIB_RGB_Header_Screen(_gad.pgf->width, _gad.pgf->height,
  1265. _gad.pgf->cColors, _gad.pgf->pcolors, _lTrans);
  1266. }
  1267. return pbmi;
  1268. }
  1269. //#include <vector>
  1270. //#define vector std::vector
  1271. //+-----------------------------------------------------------------------
  1272. //
  1273. // Member: LoadGifImage
  1274. //
  1275. // Overview: Given an IStream, decode an image into an array of bitmaps
  1276. //
  1277. // Arguments: pStream data source
  1278. // colorKeys pointer to where to store colorKey data
  1279. // numBitmaps where to store number of bitmaps
  1280. // delays where to store delay array
  1281. // loop where to store number of times to loop
  1282. // ppBitMaps where to store bitmaps
  1283. //
  1284. // Returns: S_OK on success otherwise error code
  1285. //
  1286. //------------------------------------------------------------------------
  1287. HRESULT
  1288. LoadGifImage(IStream *stream,
  1289. COLORREF **colorKeys,
  1290. int *numBitmaps,
  1291. int **delays,
  1292. double *loop,
  1293. HBITMAP **ppBitMaps)
  1294. {
  1295. HRESULT hr = S_OK;
  1296. /*
  1297. The odd approach here lets us keep the original IE GIF code unchanged while removing
  1298. DA specific inserts (except error reporting). The progressive rendering and palette
  1299. dithering found in the IE code is also not supported yet.
  1300. */
  1301. CImgGif gifimage;
  1302. gifimage._szFileName = NULL;
  1303. gifimage._gifinfo.stream = stream;
  1304. BYTE *pbBits = gifimage.ReadGIFMaster();
  1305. if (pbBits) {
  1306. gifimage._pbBits = pbBits;
  1307. gifimage._gad.pgf->pbmi = gifimage.FinishDithering();
  1308. }
  1309. /*
  1310. Extract information from GIF decoder, and format it into an array of bitmaps.
  1311. */
  1312. *delays = NULL;
  1313. /*vector<>*/HBITMAP vhbmp;
  1314. /*vector<>*/COLORREF vcolorKey;
  1315. /*vector<>*/int vdelay;
  1316. LPVOID image = NULL;
  1317. LPVOID lastBits = pbBits;
  1318. LPVOID bitsBeforeLastBits = NULL;
  1319. HBITMAP *hImage = NULL;
  1320. PBITMAPINFO pbmi = NULL;
  1321. HBITMAP hbm;
  1322. PGIFFRAME pgf = gifimage._gad.pgf;
  1323. PGIFFRAME pgfOld = NULL;
  1324. bool fUseOffset = false;
  1325. bool fFirstFrame = true;
  1326. long pgfWidth,pgfHeight, // animation frame dims
  1327. fullWidth,fullHeight, // main frame dims
  1328. fullPad, pgfPad, // row padding vals
  1329. fullSize, pgfSize;
  1330. unsigned int disp = 0;
  1331. int i = 0;
  1332. // TODO: Dither global palette to display palette
  1333. fullWidth = gifimage._xWidth;
  1334. fullHeight = gifimage._yHeight;
  1335. fullPad = (((fullWidth + 3) / 4) * 4) - fullWidth;
  1336. fullSize = (fullPad+fullWidth)*fullHeight;
  1337. if (NULL == pgf)
  1338. {
  1339. hr = E_FAIL;
  1340. goto done;
  1341. }
  1342. while(1)
  1343. {
  1344. // Assert(pgf);
  1345. pbmi = pgf->pbmi;
  1346. if (pbmi == NULL)
  1347. {
  1348. hr = E_FAIL;
  1349. goto done;
  1350. }
  1351. // TODO: It would be nice to pass local palettes up so they could
  1352. // be mapped to system palettes.
  1353. // Check to see if frame is offset from logical frame
  1354. if(pgf->top != 0 ||
  1355. pgf->left != 0 ||
  1356. pgf->width != fullWidth ||
  1357. pgf->height != fullHeight)
  1358. {
  1359. fUseOffset = true;
  1360. pgfWidth = pbmi->bmiHeader.biWidth;
  1361. pgfHeight = pbmi->bmiHeader.biHeight;
  1362. pbmi->bmiHeader.biWidth = fullWidth;
  1363. pbmi->bmiHeader.biHeight = fullHeight;
  1364. pgfPad = (((pgfWidth + 3) / 4) * 4) - pgfWidth;
  1365. pgfSize = (pgfPad+pgfWidth)*pgfHeight;
  1366. }
  1367. hbm = CreateDIBSection(NULL, pbmi, DIB_RGB_COLORS, (LPVOID *) &image, NULL, 0);
  1368. if(!hbm)
  1369. {
  1370. hr = E_OUTOFMEMORY;
  1371. goto done;
  1372. }
  1373. // Correctly composite bitmaps based on disposal method specified
  1374. disp = pgf->gced.uiDisposalMethod;
  1375. // If the frame is offset, fill it with
  1376. if( (disp == gifRestorePrev) && (bitsBeforeLastBits != NULL) )
  1377. memcpy(image, bitsBeforeLastBits, fullSize);
  1378. else if( (disp == gifRestoreBkgnd) || (disp == gifRestorePrev) || fFirstFrame ) // fill with bgColor
  1379. memset(image, pgf->gced.ucTransIndex, fullSize);
  1380. else // fill with last frames data
  1381. memcpy(image, lastBits, fullSize);
  1382. // For offset gifs allocate an image the size of the first frame
  1383. // and then fill in the bits at the offset location.
  1384. if(fUseOffset) {
  1385. for(i=0; i<pgfHeight; i++) {
  1386. BYTE *dst, *src;
  1387. // the destination is the address of the image data plus the frame and row offset.
  1388. int topOffset = fullHeight - pgfHeight - pgf->top;
  1389. dst = (BYTE*)image +
  1390. ( ((topOffset + i) *(fullPad+fullWidth)) + pgf->left );
  1391. // copy from the frame's nth row
  1392. src = pgf->ppixels + i*(pgfPad+pgfWidth);
  1393. for(int j=0; j<pgfWidth; j++) {
  1394. // copy the frame row data, excluding transparent bytes
  1395. if(src[j] != pgf->gced.ucTransIndex)
  1396. dst[j] = src[j];
  1397. }
  1398. }
  1399. }
  1400. else {
  1401. // Overwritten accumulated bits with current bits. If the
  1402. // new image contains transparency we need to take it into
  1403. // account. Since this is slower, special case it.
  1404. if(pgf->gced.fTransparent) {
  1405. for(i=0; i<((fullPad+fullWidth)*fullHeight); i++) {
  1406. if(pgf->ppixels[i] != pgf->gced.ucTransIndex)
  1407. ((BYTE*)image)[i] = ((BYTE*)pgf->ppixels)[i];
  1408. }
  1409. }
  1410. else // Otherwise, just copy over the offset window's bytes
  1411. memcpy(image, pgf->ppixels, (fullPad+fullWidth)*fullHeight);
  1412. }
  1413. /*
  1414. If we got a transparent color extension, convert it to a COLORREF
  1415. */
  1416. COLORREF colorKey = COLORKEY_NOT_SET;
  1417. if (pgf->gced.fTransparent) {
  1418. int transparent = pgf->gced.ucTransIndex;
  1419. colorKey = RGB(pgf->pbmi->bmiColors[transparent].rgbRed,
  1420. pgf->pbmi->bmiColors[transparent].rgbGreen,
  1421. pgf->pbmi->bmiColors[transparent].rgbBlue);
  1422. }
  1423. // vcolorKey.push_back(colorKey);
  1424. vcolorKey = colorKey;
  1425. // biao change : vhbmp.push_back(hbm);
  1426. vhbmp = hbm;
  1427. /*
  1428. The delay times are frame specific and can be different, these
  1429. should be propagated as an array to the sampling code.
  1430. */
  1431. // vdelay.push_back(pgf->gced.uiDelayTime);
  1432. vdelay = pgf->gced.uiDelayTime;
  1433. bitsBeforeLastBits = lastBits;
  1434. lastBits = image;
  1435. fUseOffset = false;
  1436. if(pgf->pgfNext == NULL)
  1437. break;
  1438. pgfOld = pgf;
  1439. pgf = pgf->pgfNext;
  1440. fFirstFrame = FALSE;
  1441. }
  1442. // The number of times to loop are also propagated. Note we add one because
  1443. // all other GIF decoders appear to treat the loop as the number of times to
  1444. // loop AFTER the first run through the frames.
  1445. if (gifimage._gad.cLoops == 0 && gifimage._gad.fLooped != 0)
  1446. {
  1447. *loop = 0; // HUGE_VAL;
  1448. }
  1449. else
  1450. {
  1451. *loop = gifimage._gad.cLoops;
  1452. }
  1453. *numBitmaps = 1;
  1454. // Since the vector will go out of scope, move contents over to heap
  1455. hImage = (HBITMAP *)malloc(1 * sizeof(HBITMAP));
  1456. if (NULL == hImage)
  1457. {
  1458. hr = E_OUTOFMEMORY;
  1459. goto done;
  1460. }
  1461. *delays = (int*)malloc(1 * sizeof(int));
  1462. if (NULL == *delays)
  1463. {
  1464. hr = E_OUTOFMEMORY;
  1465. goto done;
  1466. }
  1467. *colorKeys = (COLORREF*)malloc( sizeof(COLORREF) * 1 );
  1468. if (NULL == *colorKeys)
  1469. {
  1470. hr = E_OUTOFMEMORY;
  1471. goto done;
  1472. }
  1473. for(i=0; i < 1; i++) {
  1474. hImage[i] = vhbmp; // biao fix [i];
  1475. (*colorKeys)[i] = vcolorKey; // [i];
  1476. (*delays)[i] = vdelay; //[i];
  1477. }
  1478. *ppBitMaps = hImage;
  1479. hr = S_OK;
  1480. done:
  1481. if (FAILED(hr))
  1482. {
  1483. free(hImage);
  1484. free(*delays);
  1485. free(*colorKeys);
  1486. }
  1487. return hr;
  1488. }
  1489. /*lint --flb*/
  1490. BOOL Gif2Bmp(LPSTREAM pStream, HBITMAP** ppBmp)
  1491. {
  1492. HRESULT hr;
  1493. HBITMAP * phBMPs = NULL;
  1494. int numGifs = 0;
  1495. double loop = 0;
  1496. int * pDelays = NULL;
  1497. COLORREF * pColorKeys = NULL;
  1498. hr = LoadGifImage(pStream,
  1499. &pColorKeys,
  1500. &numGifs,
  1501. &pDelays,
  1502. &loop,
  1503. &phBMPs);
  1504. if (FAILED(hr))
  1505. {
  1506. return FALSE;
  1507. }
  1508. *ppBmp = phBMPs;
  1509. return TRUE;
  1510. }