blob: 4020b003bbc63bac31bdb6c2f34267a0ee150730 [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/fpdf_parser/include/cpdf_indirect_object_holder.h"
#include "core/fpdfapi/fpdf_parser/include/cpdf_dictionary.h"
#include "core/fpdfapi/fpdf_parser/include/cpdf_object.h"
#include "core/fpdfapi/fpdf_parser/include/cpdf_parser.h"
CPDF_IndirectObjectHolder::CPDF_IndirectObjectHolder(CPDF_Parser* pParser)
: m_pParser(pParser), m_LastObjNum(0) {
if (pParser)
m_LastObjNum = m_pParser->GetLastObjNum();
}
CPDF_IndirectObjectHolder::~CPDF_IndirectObjectHolder() {
for (const auto& pair : m_IndirectObjs)
pair.second->Destroy();
}
CPDF_Object* CPDF_IndirectObjectHolder::GetIndirectObject(uint32_t objnum) {
if (objnum == 0)
return nullptr;
CPDF_Object* result_obj = nullptr;
auto it = m_IndirectObjs.find(objnum);
if (it != m_IndirectObjs.end()) {
CPDF_Object* obj = it->second;
result_obj =
obj->GetObjNum() != CPDF_Object::kInvalidObjNum ? it->second : nullptr;
// Xref object is not used by the pdf document itself. Some software thus
// reuse an object number for xref object. So when we get an xref object,
// try again to see whether another object with the same number is defined.
// If so, use that object instead. See chromium:596947.
CPDF_Dictionary* dict =
obj->IsStream() ? obj->GetDict() : obj->AsDictionary();
if (!dict || dict->GetStringBy("Type") != "XRef")
return result_obj;
}
if (!m_pParser)
return nullptr;
CPDF_Object* pObj = m_pParser->ParseIndirectObject(this, objnum);
if (!pObj)
return result_obj;
pObj->m_ObjNum = objnum;
m_LastObjNum = std::max(m_LastObjNum, objnum);
if (m_IndirectObjs[objnum])
m_IndirectObjs[objnum]->Destroy();
m_IndirectObjs[objnum] = pObj;
return pObj;
}
uint32_t CPDF_IndirectObjectHolder::AddIndirectObject(CPDF_Object* pObj) {
if (pObj->m_ObjNum)
return pObj->m_ObjNum;
m_LastObjNum++;
m_IndirectObjs[m_LastObjNum] = pObj;
pObj->m_ObjNum = m_LastObjNum;
return m_LastObjNum;
}
void CPDF_IndirectObjectHolder::ReleaseIndirectObject(uint32_t objnum) {
auto it = m_IndirectObjs.find(objnum);
if (it == m_IndirectObjs.end() ||
it->second->GetObjNum() == CPDF_Object::kInvalidObjNum) {
return;
}
it->second->Destroy();
m_IndirectObjs.erase(it);
}
FX_BOOL CPDF_IndirectObjectHolder::InsertIndirectObject(uint32_t objnum,
CPDF_Object* pObj) {
if (!objnum || !pObj)
return FALSE;
auto it = m_IndirectObjs.find(objnum);
if (it != m_IndirectObjs.end()) {
if (pObj->GetGenNum() <= it->second->GetGenNum()) {
pObj->Destroy();
return FALSE;
}
it->second->Destroy();
}
pObj->m_ObjNum = objnum;
m_IndirectObjs[objnum] = pObj;
m_LastObjNum = std::max(m_LastObjNum, objnum);
return TRUE;
}