// 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_bmpmodule.h" #include "core/fxcodec/codec/codec_int.h" #include "core/fxcodec/fx_codec.h" #include "core/fxcodec/lbmp/fx_bmp.h" #include "core/fxcrt/cfx_unowned_ptr.h" #include "core/fxge/fx_dib.h" class CCodec_BmpModule::Context { public: Context(bmp_decompress_struct_p pBmp, CCodec_BmpModule* pModule, CCodec_BmpModule::Delegate* pDelegate); ~Context(); bmp_decompress_struct_p m_pBmp; CFX_UnownedPtr const m_pModule; CFX_UnownedPtr const m_pDelegate; char m_szLastError[256]; }; extern "C" { static void bmp_error_data(bmp_decompress_struct_p pBmp, const char* err_msg) { strncpy(pBmp->err_ptr, err_msg, BMP_MAX_ERROR_SIZE - 1); longjmp(pBmp->jmpbuf, 1); } static void bmp_read_scanline(bmp_decompress_struct_p pBmp, int32_t row_num, uint8_t* row_buf) { auto* p = reinterpret_cast(pBmp->context_ptr); p->m_pDelegate->BmpReadScanline(row_num, row_buf); } static bool bmp_get_data_position(bmp_decompress_struct_p pBmp, uint32_t rcd_pos) { auto* p = reinterpret_cast(pBmp->context_ptr); return p->m_pDelegate->BmpInputImagePositionBuf(rcd_pos); } } // extern "C" CCodec_BmpModule::Context::Context(bmp_decompress_struct_p pBmp, CCodec_BmpModule* pModule, CCodec_BmpModule::Delegate* pDelegate) : m_pBmp(pBmp), m_pModule(pModule), m_pDelegate(pDelegate) { memset(m_szLastError, 0, sizeof(m_szLastError)); } CCodec_BmpModule::Context::~Context() { bmp_destroy_decompress(&m_pBmp); } CCodec_BmpModule::CCodec_BmpModule() {} CCodec_BmpModule::~CCodec_BmpModule() {} CCodec_BmpModule::Context* CCodec_BmpModule::Start(Delegate* pDelegate) { bmp_decompress_struct_p pBmp = bmp_create_decompress(); if (!pBmp) return nullptr; auto* p = new Context(pBmp, this, pDelegate); p->m_pBmp->context_ptr = p; p->m_pBmp->err_ptr = p->m_szLastError; p->m_pBmp->bmp_error_fn = bmp_error_data; p->m_pBmp->bmp_get_row_fn = bmp_read_scanline; p->m_pBmp->bmp_get_data_position_fn = bmp_get_data_position; return p; } void CCodec_BmpModule::Finish(Context* ctx) { delete ctx; } int32_t CCodec_BmpModule::ReadHeader(Context* ctx, int32_t* width, int32_t* height, bool* tb_flag, int32_t* components, int32_t* pal_num, uint32_t** pal_pp, CFX_DIBAttribute* pAttribute) { if (setjmp(ctx->m_pBmp->jmpbuf)) { return 0; } int32_t ret = bmp_read_header(ctx->m_pBmp); if (ret != 1) { return ret; } *width = ctx->m_pBmp->width; *height = ctx->m_pBmp->height; *tb_flag = ctx->m_pBmp->imgTB_flag; *components = ctx->m_pBmp->components; *pal_num = ctx->m_pBmp->pal_num; *pal_pp = ctx->m_pBmp->pal_ptr; if (pAttribute) { pAttribute->m_wDPIUnit = FXCODEC_RESUNIT_METER; pAttribute->m_nXDPI = ctx->m_pBmp->dpi_x; pAttribute->m_nYDPI = ctx->m_pBmp->dpi_y; pAttribute->m_nBmpCompressType = ctx->m_pBmp->compress_flag; } return 1; } int32_t CCodec_BmpModule::LoadImage(Context* ctx) { if (setjmp(ctx->m_pBmp->jmpbuf)) return 0; return bmp_decode_image(ctx->m_pBmp); } uint32_t CCodec_BmpModule::GetAvailInput(Context* ctx, uint8_t** avail_buf_ptr) { return bmp_get_avail_input(ctx->m_pBmp, avail_buf_ptr); } void CCodec_BmpModule::Input(Context* ctx, const uint8_t* src_buf, uint32_t src_size) { bmp_input_buffer(ctx->m_pBmp, (uint8_t*)src_buf, src_size); }