1 // Copyright 2014 PDFium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com
6 
7 #include "core/include/fxcodec/fx_codec.h"
8 #include "core/include/fxge/fx_dib.h"
9 #include "codec_int.h"
10 
11 extern "C" {
12 #include "third_party/libtiff/tiffiop.h"
13 }
14 
15 void* IccLib_CreateTransform_sRGB(const unsigned char* pProfileData,
16                                   unsigned int dwProfileSize,
17                                   int nComponents,
18                                   int intent,
19                                   FX_DWORD dwSrcFormat = Icc_FORMAT_DEFAULT);
20 void IccLib_TranslateImage(void* pTransform,
21                            unsigned char* pDest,
22                            const unsigned char* pSrc,
23                            int pixels);
24 void IccLib_DestroyTransform(void* pTransform);
25 class CCodec_TiffContext {
26  public:
27   CCodec_TiffContext();
28   ~CCodec_TiffContext();
29 
30   FX_BOOL InitDecoder(IFX_FileRead* file_ptr);
31   void GetFrames(int32_t& frames);
32   FX_BOOL LoadFrameInfo(int32_t frame,
33                         FX_DWORD& width,
34                         FX_DWORD& height,
35                         FX_DWORD& comps,
36                         FX_DWORD& bpc,
37                         CFX_DIBAttribute* pAttribute);
38   FX_BOOL Decode(CFX_DIBitmap* pDIBitmap);
39 
40   union {
41     IFX_FileRead* in;
42     IFX_FileStream* out;
43   } io;
44 
45   FX_DWORD offset;
46 
47   TIFF* tif_ctx;
48   void* icc_ctx;
49   int32_t frame_num;
50   int32_t frame_cur;
51   FX_BOOL isDecoder;
52 
53  private:
54   FX_BOOL isSupport(CFX_DIBitmap* pDIBitmap);
55   void SetPalette(CFX_DIBitmap* pDIBitmap, uint16_t bps);
56   FX_BOOL Decode1bppRGB(CFX_DIBitmap* pDIBitmap,
57                         int32_t height,
58                         int32_t width,
59                         uint16_t bps,
60                         uint16_t spp);
61   FX_BOOL Decode8bppRGB(CFX_DIBitmap* pDIBitmap,
62                         int32_t height,
63                         int32_t width,
64                         uint16_t bps,
65                         uint16_t spp);
66   FX_BOOL Decode24bppRGB(CFX_DIBitmap* pDIBitmap,
67                          int32_t height,
68                          int32_t width,
69                          uint16_t bps,
70                          uint16_t spp);
71 };
CCodec_TiffContext()72 CCodec_TiffContext::CCodec_TiffContext() {
73   offset = 0;
74   frame_num = 0;
75   frame_cur = 0;
76   io.in = NULL;
77   tif_ctx = NULL;
78   icc_ctx = NULL;
79   isDecoder = TRUE;
80 }
~CCodec_TiffContext()81 CCodec_TiffContext::~CCodec_TiffContext() {
82   if (icc_ctx) {
83     IccLib_DestroyTransform(icc_ctx);
84     icc_ctx = NULL;
85   }
86   if (tif_ctx) {
87     TIFFClose(tif_ctx);
88   }
89 }
_tiff_read(thandle_t context,tdata_t buf,tsize_t length)90 static tsize_t _tiff_read(thandle_t context, tdata_t buf, tsize_t length) {
91   CCodec_TiffContext* pTiffContext = (CCodec_TiffContext*)context;
92   FX_BOOL ret = FALSE;
93   if (pTiffContext->isDecoder) {
94     ret = pTiffContext->io.in->ReadBlock(buf, pTiffContext->offset, length);
95   } else {
96     ret = pTiffContext->io.out->ReadBlock(buf, pTiffContext->offset, length);
97   }
98   if (!ret) {
99     return 0;
100   }
101   pTiffContext->offset += (FX_DWORD)length;
102   return length;
103 }
_tiff_write(thandle_t context,tdata_t buf,tsize_t length)104 static tsize_t _tiff_write(thandle_t context, tdata_t buf, tsize_t length) {
105   CCodec_TiffContext* pTiffContext = (CCodec_TiffContext*)context;
106   ASSERT(!pTiffContext->isDecoder);
107   if (!pTiffContext->io.out->WriteBlock(buf, pTiffContext->offset, length)) {
108     return 0;
109   }
110   pTiffContext->offset += (FX_DWORD)length;
111   return length;
112 }
_tiff_seek(thandle_t context,toff_t offset,int whence)113 static toff_t _tiff_seek(thandle_t context, toff_t offset, int whence) {
114   CCodec_TiffContext* pTiffContext = (CCodec_TiffContext*)context;
115   switch (whence) {
116     case 0:
117       pTiffContext->offset = (FX_DWORD)offset;
118       break;
119     case 1:
120       pTiffContext->offset += (FX_DWORD)offset;
121       break;
122     case 2:
123       if (pTiffContext->isDecoder) {
124         if (pTiffContext->io.in->GetSize() < (FX_FILESIZE)offset) {
125           return -1;
126         }
127         pTiffContext->offset =
128             (FX_DWORD)(pTiffContext->io.in->GetSize() - offset);
129       } else {
130         if (pTiffContext->io.out->GetSize() < (FX_FILESIZE)offset) {
131           return -1;
132         }
133         pTiffContext->offset =
134             (FX_DWORD)(pTiffContext->io.out->GetSize() - offset);
135       }
136       break;
137     default:
138       return -1;
139   }
140   ASSERT(pTiffContext->isDecoder ? (pTiffContext->offset <=
141                                     (FX_DWORD)pTiffContext->io.in->GetSize())
142                                  : TRUE);
143   return pTiffContext->offset;
144 }
_tiff_close(thandle_t context)145 static int _tiff_close(thandle_t context) {
146   return 0;
147 }
_tiff_get_size(thandle_t context)148 static toff_t _tiff_get_size(thandle_t context) {
149   CCodec_TiffContext* pTiffContext = (CCodec_TiffContext*)context;
150   return pTiffContext->isDecoder ? (toff_t)pTiffContext->io.in->GetSize()
151                                  : (toff_t)pTiffContext->io.out->GetSize();
152 }
_tiff_map(thandle_t context,tdata_t *,toff_t *)153 static int _tiff_map(thandle_t context, tdata_t*, toff_t*) {
154   return 0;
155 }
_tiff_unmap(thandle_t context,tdata_t,toff_t)156 static void _tiff_unmap(thandle_t context, tdata_t, toff_t) {}
_tiff_open(void * context,const char * mode)157 TIFF* _tiff_open(void* context, const char* mode) {
158   TIFF* tif = TIFFClientOpen("Tiff Image", mode, (thandle_t)context, _tiff_read,
159                              _tiff_write, _tiff_seek, _tiff_close,
160                              _tiff_get_size, _tiff_map, _tiff_unmap);
161   if (tif) {
162     tif->tif_fd = (int)(intptr_t)context;
163   }
164   return tif;
165 }
_TIFFmalloc(tmsize_t size)166 void* _TIFFmalloc(tmsize_t size) {
167   return FXMEM_DefaultAlloc(size, 0);
168 }
_TIFFfree(void * ptr)169 void _TIFFfree(void* ptr) {
170   FXMEM_DefaultFree(ptr, 0);
171 }
_TIFFrealloc(void * ptr,tmsize_t size)172 void* _TIFFrealloc(void* ptr, tmsize_t size) {
173   return FXMEM_DefaultRealloc(ptr, size, 0);
174 }
_TIFFmemset(void * ptr,int val,tmsize_t size)175 void _TIFFmemset(void* ptr, int val, tmsize_t size) {
176   FXSYS_memset(ptr, val, (size_t)size);
177 }
_TIFFmemcpy(void * des,const void * src,tmsize_t size)178 void _TIFFmemcpy(void* des, const void* src, tmsize_t size) {
179   FXSYS_memcpy(des, src, (size_t)size);
180 }
_TIFFmemcmp(const void * ptr1,const void * ptr2,tmsize_t size)181 int _TIFFmemcmp(const void* ptr1, const void* ptr2, tmsize_t size) {
182   return FXSYS_memcmp(ptr1, ptr2, (size_t)size);
183 }
184 
185 TIFFErrorHandler _TIFFwarningHandler = nullptr;
186 TIFFErrorHandler _TIFFerrorHandler = nullptr;
187 
TIFFCmyk2Rgb(thandle_t context,uint8 c,uint8 m,uint8 y,uint8 k,uint8 * r,uint8 * g,uint8 * b)188 int TIFFCmyk2Rgb(thandle_t context,
189                  uint8 c,
190                  uint8 m,
191                  uint8 y,
192                  uint8 k,
193                  uint8* r,
194                  uint8* g,
195                  uint8* b) {
196   if (context == NULL) {
197     return 0;
198   }
199   CCodec_TiffContext* p = (CCodec_TiffContext*)context;
200   if (p->icc_ctx) {
201     unsigned char cmyk[4], bgr[3];
202     cmyk[0] = c, cmyk[1] = m, cmyk[2] = y, cmyk[3] = k;
203     IccLib_TranslateImage(p->icc_ctx, bgr, cmyk, 1);
204     *r = bgr[2], *g = bgr[1], *b = bgr[0];
205   } else {
206     AdobeCMYK_to_sRGB1(c, m, y, k, *r, *g, *b);
207   }
208   return 1;
209 }
InitDecoder(IFX_FileRead * file_ptr)210 FX_BOOL CCodec_TiffContext::InitDecoder(IFX_FileRead* file_ptr) {
211   io.in = file_ptr;
212   tif_ctx = _tiff_open(this, "r");
213   if (tif_ctx == NULL) {
214     return FALSE;
215   }
216   return TRUE;
217 }
GetFrames(int32_t & frames)218 void CCodec_TiffContext::GetFrames(int32_t& frames) {
219   frames = frame_num = TIFFNumberOfDirectories(tif_ctx);
220 }
221 #define TIFF_EXIF_GETINFO(key, T, tag)      \
222   {                                         \
223     T val = (T)0;                           \
224     TIFFGetField(tif_ctx, tag, &val);       \
225     if (val) {                              \
226       (key) = FX_Alloc(uint8_t, sizeof(T)); \
227       if ((key)) {                          \
228         T* ptr = (T*)(key);                 \
229         *ptr = val;                         \
230         pExif->m_TagVal.SetAt(tag, (key));  \
231       }                                     \
232     }                                       \
233   }                                         \
234   (key) = NULL;
235 #define TIFF_EXIF_GETSTRINGINFO(key, tag)    \
236   {                                          \
237     FX_DWORD size = 0;                       \
238     uint8_t* buf = NULL;                     \
239     TIFFGetField(tif_ctx, tag, &size, &buf); \
240     if (size && buf) {                       \
241       (key) = FX_Alloc(uint8_t, size);       \
242       if ((key)) {                           \
243         FXSYS_memcpy((key), buf, size);      \
244         pExif->m_TagVal.SetAt(tag, (key));   \
245       }                                      \
246     }                                        \
247   }                                          \
248   (key) = NULL;
249 
250 namespace {
251 
252 template <class T>
Tiff_Exif_GetInfo(TIFF * tif_ctx,ttag_t tag,CFX_DIBAttribute * pAttr)253 FX_BOOL Tiff_Exif_GetInfo(TIFF* tif_ctx, ttag_t tag, CFX_DIBAttribute* pAttr) {
254   T val = 0;
255   TIFFGetField(tif_ctx, tag, &val);
256   if (!val)
257     return FALSE;
258   T* ptr = FX_Alloc(T, 1);
259   *ptr = val;
260   pAttr->m_Exif[tag] = (void*)ptr;
261   return TRUE;
262 }
Tiff_Exif_GetStringInfo(TIFF * tif_ctx,ttag_t tag,CFX_DIBAttribute * pAttr)263 void Tiff_Exif_GetStringInfo(TIFF* tif_ctx,
264                              ttag_t tag,
265                              CFX_DIBAttribute* pAttr) {
266   FX_CHAR* buf = nullptr;
267   TIFFGetField(tif_ctx, tag, &buf);
268   if (!buf)
269     return;
270   FX_STRSIZE size = FXSYS_strlen(buf);
271   uint8_t* ptr = FX_Alloc(uint8_t, size + 1);
272   FXSYS_memcpy(ptr, buf, size);
273   ptr[size] = 0;
274   pAttr->m_Exif[tag] = ptr;
275 }
276 
277 }  // namespace
278 
LoadFrameInfo(int32_t frame,FX_DWORD & width,FX_DWORD & height,FX_DWORD & comps,FX_DWORD & bpc,CFX_DIBAttribute * pAttribute)279 FX_BOOL CCodec_TiffContext::LoadFrameInfo(int32_t frame,
280                                           FX_DWORD& width,
281                                           FX_DWORD& height,
282                                           FX_DWORD& comps,
283                                           FX_DWORD& bpc,
284                                           CFX_DIBAttribute* pAttribute) {
285   if (!TIFFSetDirectory(tif_ctx, (uint16)frame)) {
286     return FALSE;
287   }
288   FX_WORD tif_cs;
289   FX_DWORD tif_icc_size = 0;
290   uint8_t* tif_icc_buf = NULL;
291   FX_WORD tif_bpc = 0;
292   FX_WORD tif_cps;
293   FX_DWORD tif_rps;
294   width = height = comps = 0;
295   TIFFGetField(tif_ctx, TIFFTAG_IMAGEWIDTH, &width);
296   TIFFGetField(tif_ctx, TIFFTAG_IMAGELENGTH, &height);
297   TIFFGetField(tif_ctx, TIFFTAG_SAMPLESPERPIXEL, &comps);
298   TIFFGetField(tif_ctx, TIFFTAG_BITSPERSAMPLE, &tif_bpc);
299   TIFFGetField(tif_ctx, TIFFTAG_PHOTOMETRIC, &tif_cs);
300   TIFFGetField(tif_ctx, TIFFTAG_COMPRESSION, &tif_cps);
301   TIFFGetField(tif_ctx, TIFFTAG_ROWSPERSTRIP, &tif_rps);
302   TIFFGetField(tif_ctx, TIFFTAG_ICCPROFILE, &tif_icc_size, &tif_icc_buf);
303   if (pAttribute) {
304     pAttribute->m_wDPIUnit = FXCODEC_RESUNIT_INCH;
305     if (TIFFGetField(tif_ctx, TIFFTAG_RESOLUTIONUNIT,
306                      &pAttribute->m_wDPIUnit)) {
307       pAttribute->m_wDPIUnit -= 1;
308     }
309     Tiff_Exif_GetInfo<FX_WORD>(tif_ctx, TIFFTAG_ORIENTATION, pAttribute);
310     if (Tiff_Exif_GetInfo<FX_FLOAT>(tif_ctx, TIFFTAG_XRESOLUTION, pAttribute)) {
311       void* val = pAttribute->m_Exif[TIFFTAG_XRESOLUTION];
312       FX_FLOAT fDpi = val ? *reinterpret_cast<FX_FLOAT*>(val) : 0;
313       pAttribute->m_nXDPI = (int32_t)(fDpi + 0.5f);
314     }
315     if (Tiff_Exif_GetInfo<FX_FLOAT>(tif_ctx, TIFFTAG_YRESOLUTION, pAttribute)) {
316       void* val = pAttribute->m_Exif[TIFFTAG_YRESOLUTION];
317       FX_FLOAT fDpi = val ? *reinterpret_cast<FX_FLOAT*>(val) : 0;
318       pAttribute->m_nYDPI = (int32_t)(fDpi + 0.5f);
319     }
320     Tiff_Exif_GetStringInfo(tif_ctx, TIFFTAG_IMAGEDESCRIPTION, pAttribute);
321     Tiff_Exif_GetStringInfo(tif_ctx, TIFFTAG_MAKE, pAttribute);
322     Tiff_Exif_GetStringInfo(tif_ctx, TIFFTAG_MODEL, pAttribute);
323   }
324   bpc = tif_bpc;
325   if (tif_rps > height) {
326     TIFFSetField(tif_ctx, TIFFTAG_ROWSPERSTRIP, tif_rps = height);
327   }
328   return TRUE;
329 }
_TiffBGRA2RGBA(uint8_t * pBuf,int32_t pixel,int32_t spp)330 void _TiffBGRA2RGBA(uint8_t* pBuf, int32_t pixel, int32_t spp) {
331   for (int32_t n = 0; n < pixel; n++) {
332     uint8_t tmp = pBuf[0];
333     pBuf[0] = pBuf[2];
334     pBuf[2] = tmp;
335     pBuf += spp;
336   }
337 }
isSupport(CFX_DIBitmap * pDIBitmap)338 FX_BOOL CCodec_TiffContext::isSupport(CFX_DIBitmap* pDIBitmap) {
339   if (TIFFIsTiled(tif_ctx)) {
340     return FALSE;
341   }
342   uint16_t photometric;
343   if (!TIFFGetField(tif_ctx, TIFFTAG_PHOTOMETRIC, &photometric)) {
344     return FALSE;
345   }
346   switch (pDIBitmap->GetBPP()) {
347     case 1:
348     case 8:
349       if (photometric != PHOTOMETRIC_PALETTE) {
350         return FALSE;
351       }
352       break;
353     case 24:
354       if (photometric != PHOTOMETRIC_RGB) {
355         return FALSE;
356       }
357       break;
358     default:
359       return FALSE;
360   }
361   uint16_t planarconfig;
362   if (!TIFFGetFieldDefaulted(tif_ctx, TIFFTAG_PLANARCONFIG, &planarconfig)) {
363     return FALSE;
364   }
365   if (planarconfig == PLANARCONFIG_SEPARATE) {
366     return FALSE;
367   }
368   return TRUE;
369 }
SetPalette(CFX_DIBitmap * pDIBitmap,uint16_t bps)370 void CCodec_TiffContext::SetPalette(CFX_DIBitmap* pDIBitmap, uint16_t bps) {
371   uint16_t *red_orig, *green_orig, *blue_orig;
372   TIFFGetField(tif_ctx, TIFFTAG_COLORMAP, &red_orig, &green_orig, &blue_orig);
373   for (int32_t i = (1L << bps) - 1; i >= 0; i--) {
374 #define CVT(x) ((uint16_t)((x) >> 8))
375     red_orig[i] = CVT(red_orig[i]);
376     green_orig[i] = CVT(green_orig[i]);
377     blue_orig[i] = CVT(blue_orig[i]);
378 #undef CVT
379   }
380   int32_t len = 1 << bps;
381   for (int32_t index = 0; index < len; index++) {
382     FX_DWORD r = red_orig[index] & 0xFF;
383     FX_DWORD g = green_orig[index] & 0xFF;
384     FX_DWORD b = blue_orig[index] & 0xFF;
385     FX_DWORD color = (uint32_t)b | ((uint32_t)g << 8) | ((uint32_t)r << 16) |
386                      (((uint32)0xffL) << 24);
387     pDIBitmap->SetPaletteEntry(index, color);
388   }
389 }
Decode1bppRGB(CFX_DIBitmap * pDIBitmap,int32_t height,int32_t width,uint16_t bps,uint16_t spp)390 FX_BOOL CCodec_TiffContext::Decode1bppRGB(CFX_DIBitmap* pDIBitmap,
391                                           int32_t height,
392                                           int32_t width,
393                                           uint16_t bps,
394                                           uint16_t spp) {
395   if (pDIBitmap->GetBPP() != 1 || spp != 1 || bps != 1 ||
396       !isSupport(pDIBitmap)) {
397     return FALSE;
398   }
399   SetPalette(pDIBitmap, bps);
400   int32_t size = (int32_t)TIFFScanlineSize(tif_ctx);
401   uint8_t* buf = (uint8_t*)_TIFFmalloc(size);
402   if (buf == NULL) {
403     TIFFError(TIFFFileName(tif_ctx), "No space for scanline buffer");
404     return FALSE;
405   }
406   uint8_t* bitMapbuffer = (uint8_t*)pDIBitmap->GetBuffer();
407   FX_DWORD pitch = pDIBitmap->GetPitch();
408   for (int32_t row = 0; row < height; row++) {
409     TIFFReadScanline(tif_ctx, buf, row, 0);
410     for (int32_t j = 0; j < size; j++) {
411       bitMapbuffer[row * pitch + j] = buf[j];
412     }
413   }
414   _TIFFfree(buf);
415   return TRUE;
416 }
Decode8bppRGB(CFX_DIBitmap * pDIBitmap,int32_t height,int32_t width,uint16_t bps,uint16_t spp)417 FX_BOOL CCodec_TiffContext::Decode8bppRGB(CFX_DIBitmap* pDIBitmap,
418                                           int32_t height,
419                                           int32_t width,
420                                           uint16_t bps,
421                                           uint16_t spp) {
422   if (pDIBitmap->GetBPP() != 8 || spp != 1 || (bps != 4 && bps != 8) ||
423       !isSupport(pDIBitmap)) {
424     return FALSE;
425   }
426   SetPalette(pDIBitmap, bps);
427   int32_t size = (int32_t)TIFFScanlineSize(tif_ctx);
428   uint8_t* buf = (uint8_t*)_TIFFmalloc(size);
429   if (buf == NULL) {
430     TIFFError(TIFFFileName(tif_ctx), "No space for scanline buffer");
431     return FALSE;
432   }
433   uint8_t* bitMapbuffer = (uint8_t*)pDIBitmap->GetBuffer();
434   FX_DWORD pitch = pDIBitmap->GetPitch();
435   for (int32_t row = 0; row < height; row++) {
436     TIFFReadScanline(tif_ctx, buf, row, 0);
437     for (int32_t j = 0; j < size; j++) {
438       switch (bps) {
439         case 4:
440           bitMapbuffer[row * pitch + 2 * j + 0] = (buf[j] & 0xF0) >> 4;
441           bitMapbuffer[row * pitch + 2 * j + 1] = (buf[j] & 0x0F) >> 0;
442           break;
443         case 8:
444           bitMapbuffer[row * pitch + j] = buf[j];
445           break;
446       }
447     }
448   }
449   _TIFFfree(buf);
450   return TRUE;
451 }
Decode24bppRGB(CFX_DIBitmap * pDIBitmap,int32_t height,int32_t width,uint16_t bps,uint16_t spp)452 FX_BOOL CCodec_TiffContext::Decode24bppRGB(CFX_DIBitmap* pDIBitmap,
453                                            int32_t height,
454                                            int32_t width,
455                                            uint16_t bps,
456                                            uint16_t spp) {
457   if (pDIBitmap->GetBPP() != 24 || !isSupport(pDIBitmap)) {
458     return FALSE;
459   }
460   int32_t size = (int32_t)TIFFScanlineSize(tif_ctx);
461   uint8_t* buf = (uint8_t*)_TIFFmalloc(size);
462   if (buf == NULL) {
463     TIFFError(TIFFFileName(tif_ctx), "No space for scanline buffer");
464     return FALSE;
465   }
466   uint8_t* bitMapbuffer = (uint8_t*)pDIBitmap->GetBuffer();
467   FX_DWORD pitch = pDIBitmap->GetPitch();
468   for (int32_t row = 0; row < height; row++) {
469     TIFFReadScanline(tif_ctx, buf, row, 0);
470     for (int32_t j = 0; j < size - 2; j += 3) {
471       bitMapbuffer[row * pitch + j + 0] = buf[j + 2];
472       bitMapbuffer[row * pitch + j + 1] = buf[j + 1];
473       bitMapbuffer[row * pitch + j + 2] = buf[j + 0];
474     }
475   }
476   _TIFFfree(buf);
477   return TRUE;
478 }
Decode(CFX_DIBitmap * pDIBitmap)479 FX_BOOL CCodec_TiffContext::Decode(CFX_DIBitmap* pDIBitmap) {
480   FX_DWORD img_wid = pDIBitmap->GetWidth();
481   FX_DWORD img_hei = pDIBitmap->GetHeight();
482   FX_DWORD width = 0;
483   FX_DWORD height = 0;
484   TIFFGetField(tif_ctx, TIFFTAG_IMAGEWIDTH, &width);
485   TIFFGetField(tif_ctx, TIFFTAG_IMAGELENGTH, &height);
486   if (img_wid != width || img_hei != height) {
487     return FALSE;
488   }
489   if (pDIBitmap->GetBPP() == 32) {
490     FX_WORD rotation = ORIENTATION_TOPLEFT;
491     TIFFGetField(tif_ctx, TIFFTAG_ORIENTATION, &rotation);
492     if (TIFFReadRGBAImageOriented(tif_ctx, img_wid, img_hei,
493                                   (uint32*)pDIBitmap->GetBuffer(), rotation,
494                                   1)) {
495       for (FX_DWORD row = 0; row < img_hei; row++) {
496         uint8_t* row_buf = (uint8_t*)pDIBitmap->GetScanline(row);
497         _TiffBGRA2RGBA(row_buf, img_wid, 4);
498       }
499       return TRUE;
500     }
501   }
502   uint16_t spp, bps;
503   TIFFGetField(tif_ctx, TIFFTAG_SAMPLESPERPIXEL, &spp);
504   TIFFGetField(tif_ctx, TIFFTAG_BITSPERSAMPLE, &bps);
505   FX_DWORD bpp = bps * spp;
506   if (bpp == 1) {
507     return Decode1bppRGB(pDIBitmap, height, width, bps, spp);
508   } else if (bpp <= 8) {
509     return Decode8bppRGB(pDIBitmap, height, width, bps, spp);
510   } else if (bpp <= 24) {
511     return Decode24bppRGB(pDIBitmap, height, width, bps, spp);
512   }
513   return FALSE;
514 }
CreateDecoder(IFX_FileRead * file_ptr)515 void* CCodec_TiffModule::CreateDecoder(IFX_FileRead* file_ptr) {
516   CCodec_TiffContext* pDecoder = new CCodec_TiffContext;
517   if (!pDecoder->InitDecoder(file_ptr)) {
518     delete pDecoder;
519     return NULL;
520   }
521   return pDecoder;
522 }
GetFrames(void * ctx,int32_t & frames)523 void CCodec_TiffModule::GetFrames(void* ctx, int32_t& frames) {
524   CCodec_TiffContext* pDecoder = (CCodec_TiffContext*)ctx;
525   pDecoder->GetFrames(frames);
526 }
LoadFrameInfo(void * ctx,int32_t frame,FX_DWORD & width,FX_DWORD & height,FX_DWORD & comps,FX_DWORD & bpc,CFX_DIBAttribute * pAttribute)527 FX_BOOL CCodec_TiffModule::LoadFrameInfo(void* ctx,
528                                          int32_t frame,
529                                          FX_DWORD& width,
530                                          FX_DWORD& height,
531                                          FX_DWORD& comps,
532                                          FX_DWORD& bpc,
533                                          CFX_DIBAttribute* pAttribute) {
534   CCodec_TiffContext* pDecoder = (CCodec_TiffContext*)ctx;
535   return pDecoder->LoadFrameInfo(frame, width, height, comps, bpc, pAttribute);
536 }
Decode(void * ctx,class CFX_DIBitmap * pDIBitmap)537 FX_BOOL CCodec_TiffModule::Decode(void* ctx, class CFX_DIBitmap* pDIBitmap) {
538   CCodec_TiffContext* pDecoder = (CCodec_TiffContext*)ctx;
539   return pDecoder->Decode(pDIBitmap);
540 }
DestroyDecoder(void * ctx)541 void CCodec_TiffModule::DestroyDecoder(void* ctx) {
542   CCodec_TiffContext* pDecoder = (CCodec_TiffContext*)ctx;
543   delete pDecoder;
544 }
545