// 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/fx_codec.h"
#include <algorithm>
#include <cmath>
#include <memory>
#include <utility>
#include "core/fxcodec/codec/codec_int.h"
#include "core/fxcrt/fx_ext.h"
#include "core/fxcrt/fx_safe_types.h"
#include "third_party/base/logging.h"
#include "third_party/base/ptr_util.h"
CCodec_ModuleMgr::CCodec_ModuleMgr()
: m_pBasicModule(new CCodec_BasicModule),
m_pFaxModule(new CCodec_FaxModule),
m_pJpegModule(new CCodec_JpegModule),
m_pJpxModule(new CCodec_JpxModule),
m_pJbig2Module(new CCodec_Jbig2Module),
m_pIccModule(new CCodec_IccModule),
m_pFlateModule(new CCodec_FlateModule) {
}
CCodec_ModuleMgr::~CCodec_ModuleMgr() {}
CCodec_ScanlineDecoder::CCodec_ScanlineDecoder()
: CCodec_ScanlineDecoder(0, 0, 0, 0, 0, 0, 0) {}
CCodec_ScanlineDecoder::CCodec_ScanlineDecoder(int nOrigWidth,
int nOrigHeight,
int nOutputWidth,
int nOutputHeight,
int nComps,
int nBpc,
uint32_t nPitch)
: m_OrigWidth(nOrigWidth),
m_OrigHeight(nOrigHeight),
m_OutputWidth(nOutputWidth),
m_OutputHeight(nOutputHeight),
m_nComps(nComps),
m_bpc(nBpc),
m_Pitch(nPitch),
m_NextLine(-1),
m_pLastScanline(nullptr) {}
CCodec_ScanlineDecoder::~CCodec_ScanlineDecoder() {}
const uint8_t* CCodec_ScanlineDecoder::GetScanline(int line) {
if (m_NextLine == line + 1)
return m_pLastScanline;
if (m_NextLine < 0 || m_NextLine > line) {
if (!v_Rewind())
return nullptr;
m_NextLine = 0;
}
while (m_NextLine < line) {
ReadNextLine();
m_NextLine++;
}
m_pLastScanline = ReadNextLine();
m_NextLine++;
return m_pLastScanline;
}
bool CCodec_ScanlineDecoder::SkipToScanline(int line, IFX_Pause* pPause) {
if (m_NextLine == line || m_NextLine == line + 1)
return false;
if (m_NextLine < 0 || m_NextLine > line) {
v_Rewind();
m_NextLine = 0;
}
m_pLastScanline = nullptr;
while (m_NextLine < line) {
m_pLastScanline = ReadNextLine();
m_NextLine++;
if (pPause && pPause->NeedToPauseNow()) {
return true;
}
}
return false;
}
uint8_t* CCodec_ScanlineDecoder::ReadNextLine() {
return v_GetNextLine();
}
bool CCodec_BasicModule::RunLengthEncode(const uint8_t* src_buf,
uint32_t src_size,
uint8_t** dest_buf,
uint32_t* dest_size) {
// Check inputs
if (!src_buf || !dest_buf || !dest_size || src_size == 0)
return false;
// Edge case
if (src_size == 1) {
*dest_buf = FX_Alloc(uint8_t, 3);
(*dest_buf)[0] = 0;
(*dest_buf)[1] = src_buf[0];
(*dest_buf)[2] = 128;
*dest_size = 3;
return true;
}
// Worst case: 1 nonmatch, 2 match, 1 nonmatch, 2 match, etc. This becomes
// 4 output chars for every 3 input, plus up to 4 more for the 1-2 chars
// rounded off plus the terminating character.
uint32_t est_size = 4 * ((src_size + 2) / 3) + 1;
*dest_buf = FX_Alloc(uint8_t, est_size);
// Set up pointers.
uint8_t* out = *dest_buf;
uint32_t run_start = 0;
uint32_t run_end = 1;
uint8_t x = src_buf[run_start];
uint8_t y = src_buf[run_end];
while (run_end < src_size) {
uint32_t max_len = std::min((uint32_t)128, src_size - run_start);
while (x == y && (run_end - run_start < max_len - 1))
y = src_buf[++run_end];
// Reached end with matched run. Update variables to expected values.
if (x == y) {
run_end++;
if (run_end < src_size)
y = src_buf[run_end];
}
if (run_end - run_start > 1) { // Matched run but not at end of input.
out[0] = 257 - (run_end - run_start);
out[1] = x;
x = y;
run_start = run_end;
run_end++;
if (run_end < src_size)
y = src_buf[run_end];
out += 2;
continue;
}
// Mismatched run
while (x != y && run_end <= run_start + max_len) {
out[run_end - run_start] = x;
x = y;
run_end++;
if (run_end == src_size) {
if (run_end <= run_start + max_len) {
out[run_end - run_start] = x;
run_end++;
}
break;
}
y = src_buf[run_end];
}
out[0] = run_end - run_start - 2;
out += run_end - run_start;
run_start = run_end - 1;
}
if (run_start < src_size) { // 1 leftover character
out[0] = 0;
out[1] = x;
out += 2;
}
*out = 128;
*dest_size = out + 1 - *dest_buf;
return true;
}
bool CCodec_BasicModule::A85Encode(const uint8_t* src_buf,
uint32_t src_size,
uint8_t** dest_buf,
uint32_t* dest_size) {
// Check inputs.
if (!src_buf || !dest_buf || !dest_size)
return false;
if (src_size == 0) {
*dest_size = 0;
return false;
}
// Worst case: 5 output for each 4 input (plus up to 4 from leftover), plus
// 2 character new lines each 75 output chars plus 2 termination chars. May
// have fewer if there are special "z" chars.
uint32_t est_size = 5 * (src_size / 4) + 4 + src_size / 30 + 2;
*dest_buf = FX_Alloc(uint8_t, est_size);
// Set up pointers.
uint8_t* out = *dest_buf;
uint32_t pos = 0;
uint32_t line_length = 0;
while (src_size >= 4 && pos < src_size - 3) {
uint32_t val = ((uint32_t)(src_buf[pos]) << 24) +
((uint32_t)(src_buf[pos + 1]) << 16) +
((uint32_t)(src_buf[pos + 2]) << 8) +
(uint32_t)(src_buf[pos + 3]);
pos += 4;
if (val == 0) { // All zero special case
*out = 'z';
out++;
line_length++;
} else { // Compute base 85 characters and add 33.
for (int i = 4; i >= 0; i--) {
out[i] = (uint8_t)(val % 85) + 33;
val = val / 85;
}
out += 5;
line_length += 5;
}
if (line_length >= 75) { // Add a return.
*out++ = '\r';
*out++ = '\n';
line_length = 0;
}
}
if (pos < src_size) { // Leftover bytes
uint32_t val = 0;
int count = 0;
while (pos < src_size) {
val += (uint32_t)(src_buf[pos]) << (8 * (3 - count));
count++;
pos++;
}
for (int i = 4; i >= 0; i--) {
if (i <= count)
out[i] = (uint8_t)(val % 85) + 33;
val = val / 85;
}
out += count + 1;
}
// Terminating characters.
out[0] = '~';
out[1] = '>';
out += 2;
*dest_size = out - *dest_buf;
return true;
}
#ifdef PDF_ENABLE_XFA
CFX_DIBAttribute::CFX_DIBAttribute()
: m_nXDPI(-1),
m_nYDPI(-1),
m_fAspectRatio(-1.0f),
m_wDPIUnit(0),
m_nGifLeft(0),
m_nGifTop(0),
m_pGifLocalPalette(nullptr),
m_nGifLocalPalNum(0),
m_nBmpCompressType(0) {
FXSYS_memset(m_strTime, 0, sizeof(m_strTime));
}
CFX_DIBAttribute::~CFX_DIBAttribute() {
for (const auto& pair : m_Exif)
FX_Free(pair.second);
}
#endif // PDF_ENABLE_XFA
class CCodec_RLScanlineDecoder : public CCodec_ScanlineDecoder {
public:
CCodec_RLScanlineDecoder();
~CCodec_RLScanlineDecoder() override;
bool Create(const uint8_t* src_buf,
uint32_t src_size,
int width,
int height,
int nComps,
int bpc);
// CCodec_ScanlineDecoder
bool v_Rewind() override;
uint8_t* v_GetNextLine() override;
uint32_t GetSrcOffset() override { return m_SrcOffset; }
protected:
bool CheckDestSize();
void GetNextOperator();
void UpdateOperator(uint8_t used_bytes);
uint8_t* m_pScanline;
const uint8_t* m_pSrcBuf;
uint32_t m_SrcSize;
uint32_t m_dwLineBytes;
uint32_t m_SrcOffset;
bool m_bEOD;
uint8_t m_Operator;
};
CCodec_RLScanlineDecoder::CCodec_RLScanlineDecoder()
: m_pScanline(nullptr),
m_pSrcBuf(nullptr),
m_SrcSize(0),
m_dwLineBytes(0),
m_SrcOffset(0),
m_bEOD(false),
m_Operator(0) {}
CCodec_RLScanlineDecoder::~CCodec_RLScanlineDecoder() {
FX_Free(m_pScanline);
}
bool CCodec_RLScanlineDecoder::CheckDestSize() {
uint32_t i = 0;
uint32_t old_size = 0;
uint32_t dest_size = 0;
while (i < m_SrcSize) {
if (m_pSrcBuf[i] < 128) {
old_size = dest_size;
dest_size += m_pSrcBuf[i] + 1;
if (dest_size < old_size) {
return false;
}
i += m_pSrcBuf[i] + 2;
} else if (m_pSrcBuf[i] > 128) {
old_size = dest_size;
dest_size += 257 - m_pSrcBuf[i];
if (dest_size < old_size) {
return false;
}
i += 2;
} else {
break;
}
}
if (((uint32_t)m_OrigWidth * m_nComps * m_bpc * m_OrigHeight + 7) / 8 >
dest_size) {
return false;
}
return true;
}
bool CCodec_RLScanlineDecoder::Create(const uint8_t* src_buf,
uint32_t src_size,
int width,
int height,
int nComps,
int bpc) {
m_pSrcBuf = src_buf;
m_SrcSize = src_size;
m_OutputWidth = m_OrigWidth = width;
m_OutputHeight = m_OrigHeight = height;
m_nComps = nComps;
m_bpc = bpc;
// Aligning the pitch to 4 bytes requires an integer overflow check.
FX_SAFE_UINT32 pitch = width;
pitch *= nComps;
pitch *= bpc;
pitch += 31;
pitch /= 32;
pitch *= 4;
if (!pitch.IsValid()) {
return false;
}
m_Pitch = pitch.ValueOrDie();
// Overflow should already have been checked before this is called.
m_dwLineBytes = (static_cast<uint32_t>(width) * nComps * bpc + 7) / 8;
m_pScanline = FX_Alloc(uint8_t, m_Pitch);
return CheckDestSize();
}
bool CCodec_RLScanlineDecoder::v_Rewind() {
FXSYS_memset(m_pScanline, 0, m_Pitch);
m_SrcOffset = 0;
m_bEOD = false;
m_Operator = 0;
return true;
}
uint8_t* CCodec_RLScanlineDecoder::v_GetNextLine() {
if (m_SrcOffset == 0) {
GetNextOperator();
} else {
if (m_bEOD) {
return nullptr;
}
}
FXSYS_memset(m_pScanline, 0, m_Pitch);
uint32_t col_pos = 0;
bool eol = false;
while (m_SrcOffset < m_SrcSize && !eol) {
if (m_Operator < 128) {
uint32_t copy_len = m_Operator + 1;
if (col_pos + copy_len >= m_dwLineBytes) {
copy_len = m_dwLineBytes - col_pos;
eol = true;
}
if (copy_len >= m_SrcSize - m_SrcOffset) {
copy_len = m_SrcSize - m_SrcOffset;
m_bEOD = true;
}
FXSYS_memcpy(m_pScanline + col_pos, m_pSrcBuf + m_SrcOffset, copy_len);
col_pos += copy_len;
UpdateOperator((uint8_t)copy_len);
} else if (m_Operator > 128) {
int fill = 0;
if (m_SrcOffset - 1 < m_SrcSize - 1) {
fill = m_pSrcBuf[m_SrcOffset];
}
uint32_t duplicate_len = 257 - m_Operator;
if (col_pos + duplicate_len >= m_dwLineBytes) {
duplicate_len = m_dwLineBytes - col_pos;
eol = true;
}
FXSYS_memset(m_pScanline + col_pos, fill, duplicate_len);
col_pos += duplicate_len;
UpdateOperator((uint8_t)duplicate_len);
} else {
m_bEOD = true;
break;
}
}
return m_pScanline;
}
void CCodec_RLScanlineDecoder::GetNextOperator() {
if (m_SrcOffset >= m_SrcSize) {
m_Operator = 128;
return;
}
m_Operator = m_pSrcBuf[m_SrcOffset];
m_SrcOffset++;
}
void CCodec_RLScanlineDecoder::UpdateOperator(uint8_t used_bytes) {
if (used_bytes == 0) {
return;
}
if (m_Operator < 128) {
ASSERT((uint32_t)m_Operator + 1 >= used_bytes);
if (used_bytes == m_Operator + 1) {
m_SrcOffset += used_bytes;
GetNextOperator();
return;
}
m_Operator -= used_bytes;
m_SrcOffset += used_bytes;
if (m_SrcOffset >= m_SrcSize) {
m_Operator = 128;
}
return;
}
uint8_t count = 257 - m_Operator;
ASSERT((uint32_t)count >= used_bytes);
if (used_bytes == count) {
m_SrcOffset++;
GetNextOperator();
return;
}
count -= used_bytes;
m_Operator = 257 - count;
}
std::unique_ptr<CCodec_ScanlineDecoder>
CCodec_BasicModule::CreateRunLengthDecoder(const uint8_t* src_buf,
uint32_t src_size,
int width,
int height,
int nComps,
int bpc) {
auto pDecoder = pdfium::MakeUnique<CCodec_RLScanlineDecoder>();
if (!pDecoder->Create(src_buf, src_size, width, height, nComps, bpc))
return nullptr;
return std::move(pDecoder);
}