// 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.h" #include #include "core/fpdfapi/parser/cpdf_dictionary.h" #include "core/fpdfapi/parser/cpdf_number.h" #include "core/fpdfapi/parser/cpdf_stream_acc.h" #include "core/fpdfapi/parser/fpdf_parser_decode.h" #include "third_party/base/numerics/safe_conversions.h" #include "third_party/base/ptr_util.h" #include "third_party/base/stl_util.h" CPDF_Stream::CPDF_Stream() {} CPDF_Stream::CPDF_Stream(std::unique_ptr pData, uint32_t size, std::unique_ptr pDict) : m_dwSize(size), m_pDict(std::move(pDict)), m_pDataBuf(std::move(pData)) {} CPDF_Stream::~CPDF_Stream() { m_ObjNum = kInvalidObjNum; if (m_pDict && m_pDict->GetObjNum() == kInvalidObjNum) m_pDict.release(); // lowercase release, release ownership. } CPDF_Object::Type CPDF_Stream::GetType() const { return STREAM; } CPDF_Dictionary* CPDF_Stream::GetDict() const { return m_pDict.get(); } bool CPDF_Stream::IsStream() const { return true; } CPDF_Stream* CPDF_Stream::AsStream() { return this; } const CPDF_Stream* CPDF_Stream::AsStream() const { return this; } void CPDF_Stream::InitStream(const uint8_t* pData, uint32_t size, std::unique_ptr pDict) { m_pDict = std::move(pDict); m_bMemoryBased = true; m_pFile = nullptr; m_pDataBuf.reset(FX_Alloc(uint8_t, size)); if (pData) memcpy(m_pDataBuf.get(), pData, size); m_dwSize = size; if (m_pDict) m_pDict->SetNewFor("Length", static_cast(m_dwSize)); } void CPDF_Stream::InitStreamFromFile( const CFX_RetainPtr& pFile, std::unique_ptr pDict) { m_pDict = std::move(pDict); m_bMemoryBased = false; m_pDataBuf.reset(); m_pFile = pFile; m_dwSize = pdfium::base::checked_cast(pFile->GetSize()); if (m_pDict) m_pDict->SetNewFor("Length", static_cast(m_dwSize)); } std::unique_ptr CPDF_Stream::Clone() const { return CloneObjectNonCyclic(false); } std::unique_ptr CPDF_Stream::CloneNonCyclic( bool bDirect, std::set* pVisited) const { pVisited->insert(this); auto pAcc = pdfium::MakeRetain(this); pAcc->LoadAllData(true); uint32_t streamSize = pAcc->GetSize(); CPDF_Dictionary* pDict = GetDict(); std::unique_ptr pNewDict; if (pDict && !pdfium::ContainsKey(*pVisited, pDict)) { pNewDict = ToDictionary( static_cast(pDict)->CloneNonCyclic(bDirect, pVisited)); } return pdfium::MakeUnique(pAcc->DetachData(), streamSize, std::move(pNewDict)); } void CPDF_Stream::SetDataAndRemoveFilter(const uint8_t* pData, uint32_t size) { SetData(pData, size); m_pDict->RemoveFor("Filter"); m_pDict->RemoveFor("DecodeParms"); } void CPDF_Stream::SetDataAndRemoveFilter(std::ostringstream* stream) { SetDataAndRemoveFilter( reinterpret_cast(stream->str().c_str()), stream->tellp()); } void CPDF_Stream::SetData(const uint8_t* pData, uint32_t size) { m_bMemoryBased = true; m_pDataBuf.reset(FX_Alloc(uint8_t, size)); if (pData) memcpy(m_pDataBuf.get(), pData, size); m_dwSize = size; if (!m_pDict) m_pDict = pdfium::MakeUnique(); m_pDict->SetNewFor("Length", static_cast(size)); } void CPDF_Stream::SetData(std::ostringstream* stream) { SetData(reinterpret_cast(stream->str().c_str()), stream->tellp()); } bool CPDF_Stream::ReadRawData(FX_FILESIZE offset, uint8_t* buf, uint32_t size) const { if (!m_bMemoryBased && m_pFile) return m_pFile->ReadBlock(buf, offset, size); if (m_pDataBuf) memcpy(buf, m_pDataBuf.get() + offset, size); return true; } bool CPDF_Stream::HasFilter() const { return m_pDict && m_pDict->KeyExist("Filter"); } CFX_WideString CPDF_Stream::GetUnicodeText() const { auto pAcc = pdfium::MakeRetain(this); pAcc->LoadAllData(false); return PDF_DecodeText(pAcc->GetData(), pAcc->GetSize()); } bool CPDF_Stream::WriteTo(IFX_ArchiveStream* archive) const { if (!GetDict()->WriteTo(archive) || !archive->WriteString("stream\r\n")) return false; auto pAcc = pdfium::MakeRetain(this); pAcc->LoadAllData(true); return archive->WriteBlock(pAcc->GetData(), pAcc->GetSize()) && archive->WriteString("\r\nendstream"); }