// Copyright 2016 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/fpdfapi/parser/cpdf_array.h" #include #include #include "core/fpdfapi/parser/cpdf_boolean.h" #include "core/fpdfapi/parser/cpdf_dictionary.h" #include "core/fpdfapi/parser/cpdf_name.h" #include "core/fpdfapi/parser/cpdf_number.h" #include "core/fpdfapi/parser/cpdf_reference.h" #include "core/fpdfapi/parser/cpdf_stream.h" #include "core/fpdfapi/parser/cpdf_string.h" #include "core/fxcrt/fx_stream.h" #include "third_party/base/check.h" #include "third_party/base/containers/contains.h" #include "third_party/base/notreached.h" CPDF_Array::CPDF_Array() = default; CPDF_Array::CPDF_Array(const WeakPtr& pPool) : m_pPool(pPool) {} CPDF_Array::~CPDF_Array() { // Break cycles for cyclic references. m_ObjNum = kInvalidObjNum; for (auto& it : m_Objects) { if (it->GetObjNum() == kInvalidObjNum) it.Leak(); } } CPDF_Object::Type CPDF_Array::GetType() const { return kArray; } CPDF_Array* CPDF_Array::AsMutableArray() { return this; } RetainPtr CPDF_Array::Clone() const { return CloneObjectNonCyclic(false); } RetainPtr CPDF_Array::CloneNonCyclic( bool bDirect, std::set* pVisited) const { pVisited->insert(this); auto pCopy = pdfium::MakeRetain(); for (const auto& pValue : m_Objects) { if (!pdfium::Contains(*pVisited, pValue.Get())) { std::set visited(*pVisited); if (auto obj = pValue->CloneNonCyclic(bDirect, &visited)) pCopy->m_Objects.push_back(std::move(obj)); } } return pCopy; } CFX_FloatRect CPDF_Array::GetRect() const { CFX_FloatRect rect; if (m_Objects.size() != 4) return rect; rect.left = GetFloatAt(0); rect.bottom = GetFloatAt(1); rect.right = GetFloatAt(2); rect.top = GetFloatAt(3); return rect; } CFX_Matrix CPDF_Array::GetMatrix() const { if (m_Objects.size() != 6) return CFX_Matrix(); return CFX_Matrix(GetFloatAt(0), GetFloatAt(1), GetFloatAt(2), GetFloatAt(3), GetFloatAt(4), GetFloatAt(5)); } absl::optional CPDF_Array::Find(const CPDF_Object* pThat) const { for (size_t i = 0; i < size(); ++i) { if (GetDirectObjectAt(i) == pThat) return i; } return absl::nullopt; } bool CPDF_Array::Contains(const CPDF_Object* pThat) const { return Find(pThat).has_value(); } CPDF_Object* CPDF_Array::GetMutableObjectAtInternal(size_t index) { return index < m_Objects.size() ? m_Objects[index].Get() : nullptr; } const CPDF_Object* CPDF_Array::GetObjectAtInternal(size_t index) const { return const_cast(this)->GetMutableObjectAtInternal(index); } RetainPtr CPDF_Array::GetMutableObjectAt(size_t index) { return pdfium::WrapRetain(GetMutableObjectAtInternal(index)); } RetainPtr CPDF_Array::GetObjectAt(size_t index) const { return pdfium::WrapRetain(GetObjectAtInternal(index)); } RetainPtr CPDF_Array::GetDirectObjectAt(size_t index) const { return const_cast(this)->GetMutableDirectObjectAt(index); } RetainPtr CPDF_Array::GetMutableDirectObjectAt(size_t index) { RetainPtr pObj = GetMutableObjectAt(index); return pObj ? pObj->GetMutableDirect() : nullptr; } ByteString CPDF_Array::GetByteStringAt(size_t index) const { if (index >= m_Objects.size()) return ByteString(); return m_Objects[index]->GetString(); } WideString CPDF_Array::GetUnicodeTextAt(size_t index) const { if (index >= m_Objects.size()) return WideString(); return m_Objects[index]->GetUnicodeText(); } bool CPDF_Array::GetBooleanAt(size_t index, bool bDefault) const { if (index >= m_Objects.size()) return bDefault; const CPDF_Object* p = m_Objects[index].Get(); return ToBoolean(p) ? p->GetInteger() != 0 : bDefault; } int CPDF_Array::GetIntegerAt(size_t index) const { if (index >= m_Objects.size()) return 0; return m_Objects[index]->GetInteger(); } float CPDF_Array::GetFloatAt(size_t index) const { if (index >= m_Objects.size()) return 0; return m_Objects[index]->GetNumber(); } RetainPtr CPDF_Array::GetMutableDictAt(size_t index) { RetainPtr p = GetMutableDirectObjectAt(index); if (!p) return nullptr; CPDF_Dictionary* pDict = p->AsMutableDictionary(); if (pDict) return pdfium::WrapRetain(pDict); CPDF_Stream* pStream = p->AsMutableStream(); if (pStream) return pStream->GetMutableDict(); return nullptr; } RetainPtr CPDF_Array::GetDictAt(size_t index) const { return const_cast(this)->GetMutableDictAt(index); } RetainPtr CPDF_Array::GetMutableStreamAt(size_t index) { return ToStream(GetMutableDirectObjectAt(index)); } RetainPtr CPDF_Array::GetStreamAt(size_t index) const { return const_cast(this)->GetMutableStreamAt(index); } RetainPtr CPDF_Array::GetMutableArrayAt(size_t index) { return ToArray(GetMutableDirectObjectAt(index)); } RetainPtr CPDF_Array::GetArrayAt(size_t index) const { return const_cast(this)->GetMutableArrayAt(index); } RetainPtr CPDF_Array::GetNumberAt(size_t index) const { return ToNumber(GetObjectAt(index)); } RetainPtr CPDF_Array::GetStringAt(size_t index) const { return ToString(GetObjectAt(index)); } void CPDF_Array::Clear() { CHECK(!IsLocked()); m_Objects.clear(); } void CPDF_Array::RemoveAt(size_t index) { CHECK(!IsLocked()); if (index < m_Objects.size()) m_Objects.erase(m_Objects.begin() + index); } void CPDF_Array::ConvertToIndirectObjectAt(size_t index, CPDF_IndirectObjectHolder* pHolder) { CHECK(!IsLocked()); if (index >= m_Objects.size()) return; if (!m_Objects[index] || m_Objects[index]->IsReference()) return; pHolder->AddIndirectObject(m_Objects[index]); m_Objects[index] = m_Objects[index]->MakeReference(pHolder); } void CPDF_Array::SetAt(size_t index, RetainPtr pObj) { (void)SetAtInternal(index, std::move(pObj)); } void CPDF_Array::InsertAt(size_t index, RetainPtr pObj) { (void)InsertAtInternal(index, std::move(pObj)); } void CPDF_Array::Append(RetainPtr pObj) { (void)AppendInternal(std::move(pObj)); } CPDF_Object* CPDF_Array::SetAtInternal(size_t index, RetainPtr pObj) { CHECK(!IsLocked()); CHECK(pObj); CHECK(pObj->IsInline()); if (index >= m_Objects.size()) return nullptr; CPDF_Object* pRet = pObj.Get(); m_Objects[index] = std::move(pObj); return pRet; } CPDF_Object* CPDF_Array::InsertAtInternal(size_t index, RetainPtr pObj) { CHECK(!IsLocked()); CHECK(pObj); CHECK(pObj->IsInline()); if (index > m_Objects.size()) return nullptr; CPDF_Object* pRet = pObj.Get(); m_Objects.insert(m_Objects.begin() + index, std::move(pObj)); return pRet; } CPDF_Object* CPDF_Array::AppendInternal(RetainPtr pObj) { CHECK(!IsLocked()); CHECK(pObj); CHECK(pObj->IsInline()); CPDF_Object* pRet = pObj.Get(); m_Objects.push_back(std::move(pObj)); return pRet; } bool CPDF_Array::WriteTo(IFX_ArchiveStream* archive, const CPDF_Encryptor* encryptor) const { if (!archive->WriteString("[")) return false; for (size_t i = 0; i < size(); ++i) { if (!GetObjectAt(i)->WriteTo(archive, encryptor)) return false; } return archive->WriteString("]"); } CPDF_ArrayLocker::CPDF_ArrayLocker(const CPDF_Array* pArray) : m_pArray(pArray) { m_pArray->m_LockCount++; } CPDF_ArrayLocker::CPDF_ArrayLocker(RetainPtr pArray) : m_pArray(std::move(pArray)) { m_pArray->m_LockCount++; } CPDF_ArrayLocker::CPDF_ArrayLocker(RetainPtr pArray) : m_pArray(std::move(pArray)) { m_pArray->m_LockCount++; } CPDF_ArrayLocker::~CPDF_ArrayLocker() { m_pArray->m_LockCount--; }