blob: 0a05d993c2751ef889094b6d7ec95cc75c38e6f7 [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/jbig2/JBig2_ArithIntDecoder.h"
#include <array>
#include <vector>
#include "core/fxcrt/fx_safe_types.h"
#include "core/fxcrt/stl_util.h"
namespace {
int ShiftOr(int val, int bitwise_or_val) {
return (val << 1) | bitwise_or_val;
}
struct ArithIntDecodeData {
int nNeedBits;
int nValue;
};
constexpr auto kArithIntDecodeData = fxcrt::ToArray<ArithIntDecodeData>({
{2, 0},
{4, 4},
{6, 20},
{8, 84},
{12, 340},
{32, 4436},
});
size_t RecursiveDecode(CJBig2_ArithDecoder* decoder,
std::vector<JBig2ArithCtx>* context,
int* prev,
size_t depth) {
static const size_t kDepthEnd = std::size(kArithIntDecodeData) - 1;
if (depth == kDepthEnd)
return kDepthEnd;
JBig2ArithCtx* pCX = &(*context)[*prev];
int D = decoder->Decode(pCX);
*prev = ShiftOr(*prev, D);
if (!D)
return depth;
return RecursiveDecode(decoder, context, prev, depth + 1);
}
} // namespace
CJBig2_ArithIntDecoder::CJBig2_ArithIntDecoder() {
m_IAx.resize(512);
}
CJBig2_ArithIntDecoder::~CJBig2_ArithIntDecoder() = default;
bool CJBig2_ArithIntDecoder::Decode(CJBig2_ArithDecoder* pArithDecoder,
int* nResult) {
// This decoding algorithm is explained in "Annex A - Arithmetic Integer
// Decoding Procedure" on page 113 of the JBIG2 specification (ISO/IEC FCD
// 14492).
int PREV = 1;
const int S = pArithDecoder->Decode(&m_IAx[PREV]);
PREV = ShiftOr(PREV, S);
const size_t nDecodeDataIndex =
RecursiveDecode(pArithDecoder, &m_IAx, &PREV, 0);
int nTemp = 0;
for (int i = 0; i < kArithIntDecodeData[nDecodeDataIndex].nNeedBits; ++i) {
int D = pArithDecoder->Decode(&m_IAx[PREV]);
PREV = ShiftOr(PREV, D);
if (PREV >= 256)
PREV = (PREV & 511) | 256;
nTemp = ShiftOr(nTemp, D);
}
FX_SAFE_INT32 safeValue = kArithIntDecodeData[nDecodeDataIndex].nValue;
safeValue += nTemp;
// Value does not fit in int.
if (!safeValue.IsValid()) {
*nResult = 0;
return false;
}
int nValue = safeValue.ValueOrDie();
if (S == 1 && nValue > 0)
nValue = -nValue;
*nResult = nValue;
return S != 1 || nValue != 0;
}
CJBig2_ArithIaidDecoder::CJBig2_ArithIaidDecoder(unsigned char SBSYMCODELENA)
: SBSYMCODELEN(SBSYMCODELENA) {
m_IAID.resize(static_cast<size_t>(1) << SBSYMCODELEN);
}
CJBig2_ArithIaidDecoder::~CJBig2_ArithIaidDecoder() = default;
void CJBig2_ArithIaidDecoder::Decode(CJBig2_ArithDecoder* pArithDecoder,
uint32_t* nResult) {
int PREV = 1;
for (unsigned char i = 0; i < SBSYMCODELEN; ++i) {
JBig2ArithCtx* pCX = &m_IAID[PREV];
int D = pArithDecoder->Decode(pCX);
PREV = ShiftOr(PREV, D);
}
*nResult = PREV - (1 << SBSYMCODELEN);
}