|  | // Copyright 2014 PDFium Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com | 
|  |  | 
|  | #include "core/fxcodec/codec/ccodec_tiffmodule.h" | 
|  |  | 
|  | #include <limits> | 
|  | #include <memory> | 
|  |  | 
|  | #include "core/fxcodec/codec/cfx_codec_memory.h" | 
|  | #include "core/fxcodec/codec/codec_int.h" | 
|  | #include "core/fxcodec/fx_codec.h" | 
|  | #include "core/fxcrt/fx_safe_types.h" | 
|  | #include "core/fxcrt/fx_stream.h" | 
|  | #include "core/fxcrt/retain_ptr.h" | 
|  | #include "core/fxge/dib/cfx_dibitmap.h" | 
|  | #include "core/fxge/fx_dib.h" | 
|  | #include "third_party/base/logging.h" | 
|  | #include "third_party/base/ptr_util.h" | 
|  |  | 
|  | extern "C" { | 
|  | #include "third_party/libtiff/tiffiop.h" | 
|  | }  // extern C | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | // For use with std::unique_ptr<TIFF>. | 
|  | struct TiffDeleter { | 
|  | inline void operator()(TIFF* context) { TIFFClose(context); } | 
|  | }; | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | class CTiffContext final : public CodecModuleIface::Context { | 
|  | public: | 
|  | CTiffContext() = default; | 
|  | ~CTiffContext() override = default; | 
|  |  | 
|  | bool InitDecoder(const RetainPtr<IFX_SeekableReadStream>& file_ptr); | 
|  | bool LoadFrameInfo(int32_t frame, | 
|  | int32_t* width, | 
|  | int32_t* height, | 
|  | int32_t* comps, | 
|  | int32_t* bpc, | 
|  | CFX_DIBAttribute* pAttribute); | 
|  | bool Decode(const RetainPtr<CFX_DIBitmap>& pDIBitmap); | 
|  |  | 
|  | RetainPtr<IFX_SeekableReadStream> io_in() const { return m_io_in; } | 
|  | uint32_t offset() const { return m_offset; } | 
|  | void set_offset(uint32_t offset) { m_offset = offset; } | 
|  |  | 
|  | private: | 
|  | bool IsSupport(const RetainPtr<CFX_DIBitmap>& pDIBitmap) const; | 
|  | void SetPalette(const RetainPtr<CFX_DIBitmap>& pDIBitmap, uint16_t bps); | 
|  | bool Decode1bppRGB(const RetainPtr<CFX_DIBitmap>& pDIBitmap, | 
|  | int32_t height, | 
|  | int32_t width, | 
|  | uint16_t bps, | 
|  | uint16_t spp); | 
|  | bool Decode8bppRGB(const RetainPtr<CFX_DIBitmap>& pDIBitmap, | 
|  | int32_t height, | 
|  | int32_t width, | 
|  | uint16_t bps, | 
|  | uint16_t spp); | 
|  | bool Decode24bppRGB(const RetainPtr<CFX_DIBitmap>& pDIBitmap, | 
|  | int32_t height, | 
|  | int32_t width, | 
|  | uint16_t bps, | 
|  | uint16_t spp); | 
|  |  | 
|  | RetainPtr<IFX_SeekableReadStream> m_io_in; | 
|  | uint32_t m_offset = 0; | 
|  | std::unique_ptr<TIFF, TiffDeleter> m_tif_ctx; | 
|  | }; | 
|  |  | 
|  | void* _TIFFcalloc(tmsize_t nmemb, tmsize_t siz) { | 
|  | return FXMEM_DefaultCalloc(nmemb, siz); | 
|  | } | 
|  |  | 
|  | void* _TIFFmalloc(tmsize_t size) { | 
|  | return FXMEM_DefaultAlloc(size); | 
|  | } | 
|  |  | 
|  | void _TIFFfree(void* ptr) { | 
|  | if (ptr) | 
|  | FXMEM_DefaultFree(ptr); | 
|  | } | 
|  |  | 
|  | void* _TIFFrealloc(void* ptr, tmsize_t size) { | 
|  | return FXMEM_DefaultRealloc(ptr, size); | 
|  | } | 
|  |  | 
|  | void _TIFFmemset(void* ptr, int val, tmsize_t size) { | 
|  | memset(ptr, val, static_cast<size_t>(size)); | 
|  | } | 
|  |  | 
|  | void _TIFFmemcpy(void* des, const void* src, tmsize_t size) { | 
|  | memcpy(des, src, static_cast<size_t>(size)); | 
|  | } | 
|  |  | 
|  | int _TIFFmemcmp(const void* ptr1, const void* ptr2, tmsize_t size) { | 
|  | return memcmp(ptr1, ptr2, static_cast<size_t>(size)); | 
|  | } | 
|  |  | 
|  | int _TIFFIfMultiplicationOverflow(tmsize_t op1, tmsize_t op2) { | 
|  | return op1 > std::numeric_limits<tmsize_t>::max() / op2; | 
|  | } | 
|  |  | 
|  | TIFFErrorHandler _TIFFwarningHandler = nullptr; | 
|  | TIFFErrorHandler _TIFFerrorHandler = nullptr; | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | tsize_t tiff_read(thandle_t context, tdata_t buf, tsize_t length) { | 
|  | CTiffContext* pTiffContext = reinterpret_cast<CTiffContext*>(context); | 
|  | FX_SAFE_UINT32 increment = pTiffContext->offset(); | 
|  | increment += length; | 
|  | if (!increment.IsValid()) | 
|  | return 0; | 
|  |  | 
|  | FX_FILESIZE offset = pTiffContext->offset(); | 
|  | if (!pTiffContext->io_in()->ReadBlock(buf, offset, length)) | 
|  | return 0; | 
|  |  | 
|  | pTiffContext->set_offset(increment.ValueOrDie()); | 
|  | if (offset + length > pTiffContext->io_in()->GetSize()) | 
|  | return pTiffContext->io_in()->GetSize() - offset; | 
|  |  | 
|  | return length; | 
|  | } | 
|  |  | 
|  | tsize_t tiff_write(thandle_t context, tdata_t buf, tsize_t length) { | 
|  | NOTREACHED(); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | toff_t tiff_seek(thandle_t context, toff_t offset, int whence) { | 
|  | CTiffContext* pTiffContext = reinterpret_cast<CTiffContext*>(context); | 
|  | FX_SAFE_FILESIZE safe_offset = offset; | 
|  | if (!safe_offset.IsValid()) | 
|  | return static_cast<toff_t>(-1); | 
|  | FX_FILESIZE file_offset = safe_offset.ValueOrDie(); | 
|  |  | 
|  | switch (whence) { | 
|  | case 0: { | 
|  | if (file_offset > pTiffContext->io_in()->GetSize()) | 
|  | return static_cast<toff_t>(-1); | 
|  | pTiffContext->set_offset(file_offset); | 
|  | return pTiffContext->offset(); | 
|  | } | 
|  | case 1: { | 
|  | FX_SAFE_UINT32 new_increment = pTiffContext->offset(); | 
|  | new_increment += file_offset; | 
|  | if (!new_increment.IsValid()) | 
|  | return static_cast<toff_t>(-1); | 
|  | pTiffContext->set_offset(new_increment.ValueOrDie()); | 
|  | return pTiffContext->offset(); | 
|  | } | 
|  | case 2: { | 
|  | if (pTiffContext->io_in()->GetSize() < file_offset) | 
|  | return static_cast<toff_t>(-1); | 
|  | pTiffContext->set_offset(pTiffContext->io_in()->GetSize() - file_offset); | 
|  | return pTiffContext->offset(); | 
|  | } | 
|  | default: | 
|  | return static_cast<toff_t>(-1); | 
|  | } | 
|  | } | 
|  |  | 
|  | int tiff_close(thandle_t context) { | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | toff_t tiff_get_size(thandle_t context) { | 
|  | CTiffContext* pTiffContext = reinterpret_cast<CTiffContext*>(context); | 
|  | return static_cast<toff_t>(pTiffContext->io_in()->GetSize()); | 
|  | } | 
|  |  | 
|  | int tiff_map(thandle_t context, tdata_t*, toff_t*) { | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | void tiff_unmap(thandle_t context, tdata_t, toff_t) {} | 
|  |  | 
|  | TIFF* tiff_open(void* context, const char* mode) { | 
|  | TIFF* tif = TIFFClientOpen("Tiff Image", mode, (thandle_t)context, tiff_read, | 
|  | tiff_write, tiff_seek, tiff_close, tiff_get_size, | 
|  | tiff_map, tiff_unmap); | 
|  | if (tif) { | 
|  | tif->tif_fd = (int)(intptr_t)context; | 
|  | } | 
|  | return tif; | 
|  | } | 
|  |  | 
|  | template <class T> | 
|  | bool Tiff_Exif_GetInfo(TIFF* tif_ctx, ttag_t tag, CFX_DIBAttribute* pAttr) { | 
|  | T val = 0; | 
|  | TIFFGetField(tif_ctx, tag, &val); | 
|  | if (!val) | 
|  | return false; | 
|  | T* ptr = FX_Alloc(T, 1); | 
|  | *ptr = val; | 
|  | pAttr->m_Exif[tag] = ptr; | 
|  | return true; | 
|  | } | 
|  |  | 
|  | void Tiff_Exif_GetStringInfo(TIFF* tif_ctx, | 
|  | ttag_t tag, | 
|  | CFX_DIBAttribute* pAttr) { | 
|  | char* buf = nullptr; | 
|  | TIFFGetField(tif_ctx, tag, &buf); | 
|  | if (!buf) | 
|  | return; | 
|  | size_t size = strlen(buf); | 
|  | uint8_t* ptr = FX_Alloc(uint8_t, size + 1); | 
|  | memcpy(ptr, buf, size); | 
|  | ptr[size] = 0; | 
|  | pAttr->m_Exif[tag] = ptr; | 
|  | } | 
|  |  | 
|  | void TiffBGRA2RGBA(uint8_t* pBuf, int32_t pixel, int32_t spp) { | 
|  | for (int32_t n = 0; n < pixel; n++) { | 
|  | uint8_t tmp = pBuf[0]; | 
|  | pBuf[0] = pBuf[2]; | 
|  | pBuf[2] = tmp; | 
|  | pBuf += spp; | 
|  | } | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | bool CTiffContext::InitDecoder( | 
|  | const RetainPtr<IFX_SeekableReadStream>& file_ptr) { | 
|  | m_io_in = file_ptr; | 
|  | m_tif_ctx.reset(tiff_open(this, "r")); | 
|  | return !!m_tif_ctx; | 
|  | } | 
|  |  | 
|  | bool CTiffContext::LoadFrameInfo(int32_t frame, | 
|  | int32_t* width, | 
|  | int32_t* height, | 
|  | int32_t* comps, | 
|  | int32_t* bpc, | 
|  | CFX_DIBAttribute* pAttribute) { | 
|  | if (!TIFFSetDirectory(m_tif_ctx.get(), (uint16)frame)) | 
|  | return false; | 
|  |  | 
|  | uint32_t tif_width = 0; | 
|  | uint32_t tif_height = 0; | 
|  | uint16_t tif_comps = 0; | 
|  | uint16_t tif_bpc = 0; | 
|  | uint32_t tif_rps = 0; | 
|  | TIFFGetField(m_tif_ctx.get(), TIFFTAG_IMAGEWIDTH, &tif_width); | 
|  | TIFFGetField(m_tif_ctx.get(), TIFFTAG_IMAGELENGTH, &tif_height); | 
|  | TIFFGetField(m_tif_ctx.get(), TIFFTAG_SAMPLESPERPIXEL, &tif_comps); | 
|  | TIFFGetField(m_tif_ctx.get(), TIFFTAG_BITSPERSAMPLE, &tif_bpc); | 
|  | TIFFGetField(m_tif_ctx.get(), TIFFTAG_ROWSPERSTRIP, &tif_rps); | 
|  |  | 
|  | if (pAttribute) { | 
|  | pAttribute->m_wDPIUnit = FXCODEC_RESUNIT_INCH; | 
|  | if (TIFFGetField(m_tif_ctx.get(), TIFFTAG_RESOLUTIONUNIT, | 
|  | &pAttribute->m_wDPIUnit)) { | 
|  | pAttribute->m_wDPIUnit--; | 
|  | } | 
|  | Tiff_Exif_GetInfo<uint16_t>(m_tif_ctx.get(), TIFFTAG_ORIENTATION, | 
|  | pAttribute); | 
|  | if (Tiff_Exif_GetInfo<float>(m_tif_ctx.get(), TIFFTAG_XRESOLUTION, | 
|  | pAttribute)) { | 
|  | void* val = pAttribute->m_Exif[TIFFTAG_XRESOLUTION]; | 
|  | float fDpi = val ? *reinterpret_cast<float*>(val) : 0; | 
|  | pAttribute->m_nXDPI = (int32_t)(fDpi + 0.5f); | 
|  | } | 
|  | if (Tiff_Exif_GetInfo<float>(m_tif_ctx.get(), TIFFTAG_YRESOLUTION, | 
|  | pAttribute)) { | 
|  | void* val = pAttribute->m_Exif[TIFFTAG_YRESOLUTION]; | 
|  | float fDpi = val ? *reinterpret_cast<float*>(val) : 0; | 
|  | pAttribute->m_nYDPI = (int32_t)(fDpi + 0.5f); | 
|  | } | 
|  | Tiff_Exif_GetStringInfo(m_tif_ctx.get(), TIFFTAG_IMAGEDESCRIPTION, | 
|  | pAttribute); | 
|  | Tiff_Exif_GetStringInfo(m_tif_ctx.get(), TIFFTAG_MAKE, pAttribute); | 
|  | Tiff_Exif_GetStringInfo(m_tif_ctx.get(), TIFFTAG_MODEL, pAttribute); | 
|  | } | 
|  | pdfium::base::CheckedNumeric<int32_t> checked_width = tif_width; | 
|  | pdfium::base::CheckedNumeric<int32_t> checked_height = tif_height; | 
|  | if (!checked_width.IsValid() || !checked_height.IsValid()) | 
|  | return false; | 
|  |  | 
|  | *width = checked_width.ValueOrDie(); | 
|  | *height = checked_height.ValueOrDie(); | 
|  | *comps = tif_comps; | 
|  | *bpc = tif_bpc; | 
|  | if (tif_rps > tif_height) { | 
|  | tif_rps = tif_height; | 
|  | TIFFSetField(m_tif_ctx.get(), TIFFTAG_ROWSPERSTRIP, tif_rps); | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool CTiffContext::IsSupport(const RetainPtr<CFX_DIBitmap>& pDIBitmap) const { | 
|  | if (TIFFIsTiled(m_tif_ctx.get())) | 
|  | return false; | 
|  |  | 
|  | uint16_t photometric = 0; | 
|  | if (!TIFFGetField(m_tif_ctx.get(), TIFFTAG_PHOTOMETRIC, &photometric)) | 
|  | return false; | 
|  |  | 
|  | switch (pDIBitmap->GetBPP()) { | 
|  | case 1: | 
|  | case 8: | 
|  | if (photometric != PHOTOMETRIC_PALETTE) { | 
|  | return false; | 
|  | } | 
|  | break; | 
|  | case 24: | 
|  | if (photometric != PHOTOMETRIC_RGB) { | 
|  | return false; | 
|  | } | 
|  | break; | 
|  | default: | 
|  | return false; | 
|  | } | 
|  | uint16_t planarconfig = 0; | 
|  | if (!TIFFGetFieldDefaulted(m_tif_ctx.get(), TIFFTAG_PLANARCONFIG, | 
|  | &planarconfig)) | 
|  | return false; | 
|  |  | 
|  | return planarconfig != PLANARCONFIG_SEPARATE; | 
|  | } | 
|  |  | 
|  | void CTiffContext::SetPalette(const RetainPtr<CFX_DIBitmap>& pDIBitmap, | 
|  | uint16_t bps) { | 
|  | uint16_t* red_orig = nullptr; | 
|  | uint16_t* green_orig = nullptr; | 
|  | uint16_t* blue_orig = nullptr; | 
|  | TIFFGetField(m_tif_ctx.get(), TIFFTAG_COLORMAP, &red_orig, &green_orig, | 
|  | &blue_orig); | 
|  | for (int32_t i = (1L << bps) - 1; i >= 0; i--) { | 
|  | #define CVT(x) ((uint16_t)((x) >> 8)) | 
|  | red_orig[i] = CVT(red_orig[i]); | 
|  | green_orig[i] = CVT(green_orig[i]); | 
|  | blue_orig[i] = CVT(blue_orig[i]); | 
|  | #undef CVT | 
|  | } | 
|  | int32_t len = 1 << bps; | 
|  | for (int32_t index = 0; index < len; index++) { | 
|  | uint32_t r = red_orig[index] & 0xFF; | 
|  | uint32_t g = green_orig[index] & 0xFF; | 
|  | uint32_t b = blue_orig[index] & 0xFF; | 
|  | uint32_t color = (uint32_t)b | ((uint32_t)g << 8) | ((uint32_t)r << 16) | | 
|  | (((uint32)0xffL) << 24); | 
|  | pDIBitmap->SetPaletteArgb(index, color); | 
|  | } | 
|  | } | 
|  |  | 
|  | bool CTiffContext::Decode1bppRGB(const RetainPtr<CFX_DIBitmap>& pDIBitmap, | 
|  | int32_t height, | 
|  | int32_t width, | 
|  | uint16_t bps, | 
|  | uint16_t spp) { | 
|  | if (pDIBitmap->GetBPP() != 1 || spp != 1 || bps != 1 || | 
|  | !IsSupport(pDIBitmap)) { | 
|  | return false; | 
|  | } | 
|  | SetPalette(pDIBitmap, bps); | 
|  | int32_t size = (int32_t)TIFFScanlineSize(m_tif_ctx.get()); | 
|  | uint8_t* buf = (uint8_t*)_TIFFmalloc(size); | 
|  | if (!buf) { | 
|  | TIFFError(TIFFFileName(m_tif_ctx.get()), "No space for scanline buffer"); | 
|  | return false; | 
|  | } | 
|  | uint8_t* bitMapbuffer = (uint8_t*)pDIBitmap->GetBuffer(); | 
|  | uint32_t pitch = pDIBitmap->GetPitch(); | 
|  | for (int32_t row = 0; row < height; row++) { | 
|  | TIFFReadScanline(m_tif_ctx.get(), buf, row, 0); | 
|  | for (int32_t j = 0; j < size; j++) { | 
|  | bitMapbuffer[row * pitch + j] = buf[j]; | 
|  | } | 
|  | } | 
|  | _TIFFfree(buf); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool CTiffContext::Decode8bppRGB(const RetainPtr<CFX_DIBitmap>& pDIBitmap, | 
|  | int32_t height, | 
|  | int32_t width, | 
|  | uint16_t bps, | 
|  | uint16_t spp) { | 
|  | if (pDIBitmap->GetBPP() != 8 || spp != 1 || (bps != 4 && bps != 8) || | 
|  | !IsSupport(pDIBitmap)) { | 
|  | return false; | 
|  | } | 
|  | SetPalette(pDIBitmap, bps); | 
|  | int32_t size = (int32_t)TIFFScanlineSize(m_tif_ctx.get()); | 
|  | uint8_t* buf = (uint8_t*)_TIFFmalloc(size); | 
|  | if (!buf) { | 
|  | TIFFError(TIFFFileName(m_tif_ctx.get()), "No space for scanline buffer"); | 
|  | return false; | 
|  | } | 
|  | uint8_t* bitMapbuffer = (uint8_t*)pDIBitmap->GetBuffer(); | 
|  | uint32_t pitch = pDIBitmap->GetPitch(); | 
|  | for (int32_t row = 0; row < height; row++) { | 
|  | TIFFReadScanline(m_tif_ctx.get(), buf, row, 0); | 
|  | for (int32_t j = 0; j < size; j++) { | 
|  | switch (bps) { | 
|  | case 4: | 
|  | bitMapbuffer[row * pitch + 2 * j + 0] = (buf[j] & 0xF0) >> 4; | 
|  | bitMapbuffer[row * pitch + 2 * j + 1] = (buf[j] & 0x0F) >> 0; | 
|  | break; | 
|  | case 8: | 
|  | bitMapbuffer[row * pitch + j] = buf[j]; | 
|  | break; | 
|  | } | 
|  | } | 
|  | } | 
|  | _TIFFfree(buf); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool CTiffContext::Decode24bppRGB(const RetainPtr<CFX_DIBitmap>& pDIBitmap, | 
|  | int32_t height, | 
|  | int32_t width, | 
|  | uint16_t bps, | 
|  | uint16_t spp) { | 
|  | if (pDIBitmap->GetBPP() != 24 || !IsSupport(pDIBitmap)) | 
|  | return false; | 
|  |  | 
|  | int32_t size = (int32_t)TIFFScanlineSize(m_tif_ctx.get()); | 
|  | uint8_t* buf = (uint8_t*)_TIFFmalloc(size); | 
|  | if (!buf) { | 
|  | TIFFError(TIFFFileName(m_tif_ctx.get()), "No space for scanline buffer"); | 
|  | return false; | 
|  | } | 
|  | uint8_t* bitMapbuffer = (uint8_t*)pDIBitmap->GetBuffer(); | 
|  | uint32_t pitch = pDIBitmap->GetPitch(); | 
|  | for (int32_t row = 0; row < height; row++) { | 
|  | TIFFReadScanline(m_tif_ctx.get(), buf, row, 0); | 
|  | for (int32_t j = 0; j < size - 2; j += 3) { | 
|  | bitMapbuffer[row * pitch + j + 0] = buf[j + 2]; | 
|  | bitMapbuffer[row * pitch + j + 1] = buf[j + 1]; | 
|  | bitMapbuffer[row * pitch + j + 2] = buf[j + 0]; | 
|  | } | 
|  | } | 
|  | _TIFFfree(buf); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool CTiffContext::Decode(const RetainPtr<CFX_DIBitmap>& pDIBitmap) { | 
|  | uint32_t img_width = pDIBitmap->GetWidth(); | 
|  | uint32_t img_height = pDIBitmap->GetHeight(); | 
|  | uint32_t width = 0; | 
|  | uint32_t height = 0; | 
|  | TIFFGetField(m_tif_ctx.get(), TIFFTAG_IMAGEWIDTH, &width); | 
|  | TIFFGetField(m_tif_ctx.get(), TIFFTAG_IMAGELENGTH, &height); | 
|  | if (img_width != width || img_height != height) | 
|  | return false; | 
|  |  | 
|  | if (pDIBitmap->GetBPP() == 32) { | 
|  | uint16_t rotation = ORIENTATION_TOPLEFT; | 
|  | TIFFGetField(m_tif_ctx.get(), TIFFTAG_ORIENTATION, &rotation); | 
|  | if (TIFFReadRGBAImageOriented(m_tif_ctx.get(), img_width, img_height, | 
|  | (uint32*)pDIBitmap->GetBuffer(), rotation, | 
|  | 1)) { | 
|  | for (uint32_t row = 0; row < img_height; row++) { | 
|  | uint8_t* row_buf = pDIBitmap->GetWritableScanline(row); | 
|  | TiffBGRA2RGBA(row_buf, img_width, 4); | 
|  | } | 
|  | return true; | 
|  | } | 
|  | } | 
|  | uint16_t spp = 0; | 
|  | uint16_t bps = 0; | 
|  | TIFFGetField(m_tif_ctx.get(), TIFFTAG_SAMPLESPERPIXEL, &spp); | 
|  | TIFFGetField(m_tif_ctx.get(), TIFFTAG_BITSPERSAMPLE, &bps); | 
|  | FX_SAFE_UINT32 safe_bpp = bps; | 
|  | safe_bpp *= spp; | 
|  | if (!safe_bpp.IsValid()) | 
|  | return false; | 
|  | uint32_t bpp = safe_bpp.ValueOrDie(); | 
|  | if (bpp == 1) | 
|  | return Decode1bppRGB(pDIBitmap, height, width, bps, spp); | 
|  | if (bpp <= 8) | 
|  | return Decode8bppRGB(pDIBitmap, height, width, bps, spp); | 
|  | if (bpp <= 24) | 
|  | return Decode24bppRGB(pDIBitmap, height, width, bps, spp); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | std::unique_ptr<CodecModuleIface::Context> CCodec_TiffModule::CreateDecoder( | 
|  | const RetainPtr<IFX_SeekableReadStream>& file_ptr) { | 
|  | auto pDecoder = pdfium::MakeUnique<CTiffContext>(); | 
|  | if (!pDecoder->InitDecoder(file_ptr)) | 
|  | return nullptr; | 
|  |  | 
|  | return pDecoder; | 
|  | } | 
|  |  | 
|  | FX_FILESIZE CCodec_TiffModule::GetAvailInput(Context* pContext) const { | 
|  | NOTREACHED(); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | bool CCodec_TiffModule::Input(Context* pContext, | 
|  | RetainPtr<CFX_CodecMemory> codec_memory, | 
|  | CFX_DIBAttribute*) { | 
|  | NOTREACHED(); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | bool CCodec_TiffModule::LoadFrameInfo(Context* pContext, | 
|  | int32_t frame, | 
|  | int32_t* width, | 
|  | int32_t* height, | 
|  | int32_t* comps, | 
|  | int32_t* bpc, | 
|  | CFX_DIBAttribute* pAttribute) { | 
|  | auto* ctx = static_cast<CTiffContext*>(pContext); | 
|  | return ctx->LoadFrameInfo(frame, width, height, comps, bpc, pAttribute); | 
|  | } | 
|  |  | 
|  | bool CCodec_TiffModule::Decode(Context* pContext, | 
|  | const RetainPtr<CFX_DIBitmap>& pDIBitmap) { | 
|  | auto* ctx = static_cast<CTiffContext*>(pContext); | 
|  | return ctx->Decode(pDIBitmap); | 
|  | } |