blob: f0eeb83bea1f46222fa0fdadb92ca0bb9ee349e9 [file] [log] [blame]
// Copyright 2016 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/fpdfapi/parser/cpdf_stream_acc.h"
#include <utility>
#include <vector>
#include "core/fdrm/fx_crypt.h"
#include "core/fpdfapi/parser/cpdf_dictionary.h"
#include "core/fpdfapi/parser/cpdf_stream.h"
#include "core/fpdfapi/parser/fpdf_parser_decode.h"
#include "third_party/base/check_op.h"
CPDF_StreamAcc::OwnedData::OwnedData(
std::unique_ptr<uint8_t, FxFreeDeleter> buffer,
uint32_t size)
: buffer(std::move(buffer)), size(size) {}
CPDF_StreamAcc::OwnedData::OwnedData(OwnedData&&) = default;
CPDF_StreamAcc::OwnedData& CPDF_StreamAcc::OwnedData::operator=(OwnedData&&) =
default;
CPDF_StreamAcc::OwnedData::~OwnedData() = default;
CPDF_StreamAcc::CPDF_StreamAcc(RetainPtr<const CPDF_Stream> pStream)
: m_pStream(std::move(pStream)) {}
CPDF_StreamAcc::~CPDF_StreamAcc() = default;
void CPDF_StreamAcc::LoadAllData(bool bRawAccess,
uint32_t estimated_size,
bool bImageAcc) {
if (bRawAccess) {
DCHECK(!estimated_size);
DCHECK(!bImageAcc);
}
if (!m_pStream)
return;
bool bProcessRawData = bRawAccess || !m_pStream->HasFilter();
if (bProcessRawData)
ProcessRawData();
else
ProcessFilteredData(estimated_size, bImageAcc);
}
void CPDF_StreamAcc::LoadAllDataFiltered() {
LoadAllData(false, 0, false);
}
void CPDF_StreamAcc::LoadAllDataFilteredWithEstimatedSize(
uint32_t estimated_size) {
LoadAllData(false, estimated_size, false);
}
void CPDF_StreamAcc::LoadAllDataImageAcc(uint32_t estimated_size) {
LoadAllData(false, estimated_size, true);
}
void CPDF_StreamAcc::LoadAllDataRaw() {
LoadAllData(true, 0, false);
}
RetainPtr<const CPDF_Stream> CPDF_StreamAcc::GetStream() const {
return m_pStream;
}
const CPDF_Dictionary* CPDF_StreamAcc::GetDict() const {
// TODO(tsepez): return retained references.
return m_pStream ? m_pStream->GetDict().Get() : nullptr;
}
const uint8_t* CPDF_StreamAcc::GetData() const {
if (is_owned())
return absl::get<OwnedData>(m_Data).buffer.get();
return (m_pStream && m_pStream->IsMemoryBased())
? m_pStream->GetInMemoryRawData()
: nullptr;
}
uint32_t CPDF_StreamAcc::GetSize() const {
if (is_owned())
return absl::get<OwnedData>(m_Data).size;
return (m_pStream && m_pStream->IsMemoryBased()) ? m_pStream->GetRawSize()
: 0;
}
pdfium::span<const uint8_t> CPDF_StreamAcc::GetSpan() const {
return {GetData(), GetSize()};
}
ByteString CPDF_StreamAcc::ComputeDigest() const {
uint8_t digest[20];
CRYPT_SHA1Generate(GetData(), GetSize(), digest);
return ByteString(digest, 20);
}
std::unique_ptr<uint8_t, FxFreeDeleter> CPDF_StreamAcc::DetachData() {
if (is_owned()) {
auto& data = absl::get<OwnedData>(m_Data);
data.size = 0;
return std::move(data.buffer);
}
auto span = absl::get<pdfium::span<const uint8_t>>(m_Data);
std::unique_ptr<uint8_t, FxFreeDeleter> result(
FX_AllocUninit(uint8_t, span.size()));
memcpy(result.get(), span.data(), span.size());
return result;
}
void CPDF_StreamAcc::ProcessRawData() {
uint32_t dwSrcSize = m_pStream->GetRawSize();
if (dwSrcSize == 0)
return;
if (m_pStream->IsMemoryBased()) {
m_Data = pdfium::make_span(m_pStream->GetInMemoryRawData(), dwSrcSize);
return;
}
std::unique_ptr<uint8_t, FxFreeDeleter> pData = ReadRawStream();
if (!pData)
return;
m_Data.emplace<OwnedData>(std::move(pData), dwSrcSize);
}
void CPDF_StreamAcc::ProcessFilteredData(uint32_t estimated_size,
bool bImageAcc) {
uint32_t dwSrcSize = m_pStream->GetRawSize();
if (dwSrcSize == 0)
return;
absl::variant<pdfium::span<const uint8_t>, OwnedData> src_data;
pdfium::span<const uint8_t> src_span;
if (m_pStream->IsMemoryBased()) {
src_span = pdfium::make_span(m_pStream->GetInMemoryRawData(), dwSrcSize);
src_data = src_span;
} else {
std::unique_ptr<uint8_t, FxFreeDeleter> pTempSrcData = ReadRawStream();
if (!pTempSrcData)
return;
src_span = pdfium::make_span(pTempSrcData.get(), dwSrcSize);
src_data.emplace<OwnedData>(std::move(pTempSrcData), dwSrcSize);
}
std::unique_ptr<uint8_t, FxFreeDeleter> pDecodedData;
uint32_t dwDecodedSize = 0;
absl::optional<std::vector<std::pair<ByteString, const CPDF_Object*>>>
decoder_array = GetDecoderArray(m_pStream->GetDict());
if (!decoder_array.has_value() || decoder_array.value().empty() ||
!PDF_DataDecode(src_span, estimated_size, bImageAcc,
decoder_array.value(), &pDecodedData, &dwDecodedSize,
&m_ImageDecoder, &m_pImageParam)) {
m_Data = std::move(src_data);
return;
}
if (pDecodedData) {
DCHECK_NE(pDecodedData.get(), src_span.data());
m_Data.emplace<OwnedData>(std::move(pDecodedData), dwDecodedSize);
} else {
m_Data = std::move(src_data);
}
}
std::unique_ptr<uint8_t, FxFreeDeleter> CPDF_StreamAcc::ReadRawStream() const {
DCHECK(m_pStream);
DCHECK(!m_pStream->IsMemoryBased());
uint32_t dwSrcSize = m_pStream->GetRawSize();
DCHECK(dwSrcSize);
std::unique_ptr<uint8_t, FxFreeDeleter> pSrcData(
FX_Alloc(uint8_t, dwSrcSize));
if (!m_pStream->ReadRawData(0, pSrcData.get(), dwSrcSize))
return nullptr;
return pSrcData;
}