blob: feb6c9617426780a8885947870c8edb7394483ee [file] [log] [blame] [edit]
// Copyright 2014 The PDFium Authors
// 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/tiff/tiff_decoder.h"
#include <memory>
#include <utility>
#include "core/fxcodec/cfx_codec_memory.h"
#include "core/fxcodec/fx_codec.h"
#include "core/fxcodec/fx_codec_def.h"
#include "core/fxcrt/check.h"
#include "core/fxcrt/compiler_specific.h"
#include "core/fxcrt/fx_memcpy_wrappers.h"
#include "core/fxcrt/fx_safe_types.h"
#include "core/fxcrt/fx_stream.h"
#include "core/fxcrt/fx_system.h"
#include "core/fxcrt/notreached.h"
#include "core/fxcrt/numerics/safe_conversions.h"
#include "core/fxcrt/retain_ptr.h"
#include "core/fxcrt/span.h"
#include "core/fxcrt/span_util.h"
#include "core/fxge/dib/cfx_dibitmap.h"
#include "core/fxge/dib/fx_dib.h"
extern "C" {
#if defined(USE_SYSTEM_LIBTIFF)
#include <tiffio.h>
#else
#include "third_party/libtiff/tiffio.h"
#endif
} // extern C
namespace {
// For use with std::unique_ptr<TIFF>.
struct TiffDeleter {
inline void operator()(TIFF* context) { TIFFClose(context); }
};
// For use with std::unique_ptr<TIFFOpenOptions>.
struct TIFFOpenOptionsDeleter {
inline void operator()(TIFFOpenOptions* options) {
TIFFOpenOptionsFree(options);
}
};
} // namespace
class CTiffContext final : public ProgressiveDecoderIface::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(RetainPtr<CFX_DIBitmap> bitmap);
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:
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) {
UNSAFE_TODO(FXSYS_memset(ptr, val, static_cast<size_t>(size)));
}
void _TIFFmemcpy(void* des, const void* src, tmsize_t size) {
UNSAFE_TODO(FXSYS_memcpy(des, src, static_cast<size_t>(size)));
}
int _TIFFmemcmp(const void* ptr1, const void* ptr2, tmsize_t size) {
return UNSAFE_TODO(memcmp(ptr1, ptr2, static_cast<size_t>(size)));
}
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();
// SAFETY: required from caller.
if (!pTiffContext->io_in()->ReadBlockAtOffset(
UNSAFE_BUFFERS(pdfium::make_span(static_cast<uint8_t*>(buf),
static_cast<size_t>(length))),
offset)) {
return 0;
}
pTiffContext->set_offset(increment.ValueOrDie());
if (offset + length > pTiffContext->io_in()->GetSize()) {
return pdfium::checked_cast<tsize_t>(pTiffContext->io_in()->GetSize() -
offset);
}
return length;
}
tsize_t tiff_write(thandle_t context, tdata_t buf, tsize_t length) {
NOTREACHED_NORETURN();
}
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(pdfium::checked_cast<uint32_t>(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(pdfium::checked_cast<uint32_t>(
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) {}
} // namespace
bool CTiffContext::InitDecoder(
const RetainPtr<IFX_SeekableReadStream>& file_ptr) {
// Limit set to make fuzzers happy. If this causes problems in the real world,
// then adjust as needed.
constexpr tmsize_t kMaxTiffAllocBytes = 1536 * 1024 * 1024; // 1.5 GB
std::unique_ptr<TIFFOpenOptions, TIFFOpenOptionsDeleter> options(
TIFFOpenOptionsAlloc());
CHECK(options);
TIFFOpenOptionsSetMaxCumulatedMemAlloc(options.get(), kMaxTiffAllocBytes);
m_io_in = file_ptr;
m_tif_ctx.reset(TIFFClientOpenExt(
/*name=*/"Tiff Image", /*mode=*/"r", /*clientdata=*/this, tiff_read,
tiff_write, tiff_seek, tiff_close, tiff_get_size, tiff_map, tiff_unmap,
options.get()));
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_t)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);
uint16_t tif_resunit = 0;
if (TIFFGetField(m_tif_ctx.get(), TIFFTAG_RESOLUTIONUNIT, &tif_resunit)) {
pAttribute->m_wDPIUnit =
static_cast<CFX_DIBAttribute::ResUnit>(tif_resunit - 1);
} else {
pAttribute->m_wDPIUnit = CFX_DIBAttribute::kResUnitInch;
}
float tif_xdpi = 0.0f;
TIFFGetField(m_tif_ctx.get(), TIFFTAG_XRESOLUTION, &tif_xdpi);
if (tif_xdpi)
pAttribute->m_nXDPI = static_cast<int32_t>(tif_xdpi + 0.5f);
float tif_ydpi = 0.0f;
TIFFGetField(m_tif_ctx.get(), TIFFTAG_YRESOLUTION, &tif_ydpi);
if (tif_ydpi)
pAttribute->m_nYDPI = static_cast<int32_t>(tif_ydpi + 0.5f);
FX_SAFE_INT32 checked_width = tif_width;
FX_SAFE_INT32 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::Decode(RetainPtr<CFX_DIBitmap> bitmap) {
// TODO(crbug.com/355630556): Consider adding support for
// `FXDIB_Format::kBgraPremul`
CHECK_EQ(bitmap->GetFormat(), FXDIB_Format::kBgra);
const uint32_t img_width = bitmap->GetWidth();
const uint32_t img_height = bitmap->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;
}
uint16_t rotation = ORIENTATION_TOPLEFT;
TIFFGetField(m_tif_ctx.get(), TIFFTAG_ORIENTATION, &rotation);
uint32_t* data =
fxcrt::reinterpret_span<uint32_t>(bitmap->GetWritableBuffer()).data();
if (!TIFFReadRGBAImageOriented(m_tif_ctx.get(), img_width, img_height, data,
rotation, 1)) {
return false;
}
for (uint32_t row = 0; row < img_height; row++) {
auto row_span = bitmap->GetWritableScanlineAs<FX_BGRA_STRUCT<uint8_t>>(row);
for (auto& pixel : row_span) {
std::swap(pixel.blue, pixel.red);
}
}
return true;
}
namespace fxcodec {
// static
std::unique_ptr<ProgressiveDecoderIface::Context> TiffDecoder::CreateDecoder(
const RetainPtr<IFX_SeekableReadStream>& file_ptr) {
auto pDecoder = std::make_unique<CTiffContext>();
if (!pDecoder->InitDecoder(file_ptr))
return nullptr;
return pDecoder;
}
// static
bool TiffDecoder::LoadFrameInfo(ProgressiveDecoderIface::Context* pContext,
int32_t frame,
int32_t* width,
int32_t* height,
int32_t* comps,
int32_t* bpc,
CFX_DIBAttribute* pAttribute) {
DCHECK(pAttribute);
auto* ctx = static_cast<CTiffContext*>(pContext);
return ctx->LoadFrameInfo(frame, width, height, comps, bpc, pAttribute);
}
// static
bool TiffDecoder::Decode(ProgressiveDecoderIface::Context* pContext,
RetainPtr<CFX_DIBitmap> bitmap) {
auto* ctx = static_cast<CTiffContext*>(pContext);
return ctx->Decode(std::move(bitmap));
}
} // namespace fxcodec