| // 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_linearized_header.h" |
| |
| #include <algorithm> |
| #include <limits> |
| #include <utility> |
| |
| #include "core/fpdfapi/parser/cpdf_array.h" |
| #include "core/fpdfapi/parser/cpdf_dictionary.h" |
| #include "core/fpdfapi/parser/cpdf_number.h" |
| #include "core/fpdfapi/parser/cpdf_syntax_parser.h" |
| #include "core/fxcrt/fx_safe_types.h" |
| #include "third_party/base/check.h" |
| #include "third_party/base/ptr_util.h" |
| |
| namespace { |
| |
| constexpr FX_FILESIZE kLinearizedHeaderOffset = 9; |
| constexpr size_t kMaxInt = static_cast<size_t>(std::numeric_limits<int>::max()); |
| |
| template <class T> |
| bool IsValidNumericDictionaryValue(const CPDF_Dictionary* pDict, |
| const ByteString& key, |
| T min_value, |
| bool must_exist = true) { |
| if (!pDict->KeyExist(key)) |
| return !must_exist; |
| const CPDF_Number* pNum = ToNumber(pDict->GetObjectFor(key)); |
| if (!pNum || !pNum->IsInteger()) |
| return false; |
| const int raw_value = pNum->GetInteger(); |
| if (!pdfium::base::IsValueInRangeForNumericType<T>(raw_value)) |
| return false; |
| return static_cast<T>(raw_value) >= min_value; |
| } |
| |
| bool IsLinearizedHeaderValid(const CPDF_LinearizedHeader* header, |
| FX_FILESIZE document_size) { |
| DCHECK(header); |
| return header->GetFileSize() == document_size && |
| header->GetFirstPageNo() < kMaxInt && |
| header->GetFirstPageNo() < header->GetPageCount() && |
| header->GetMainXRefTableFirstEntryOffset() < document_size && |
| header->GetFirstPageEndOffset() < document_size && |
| header->GetLastXRefOffset() < document_size && |
| header->GetHintStart() < document_size; |
| } |
| |
| } // namespace |
| |
| // static |
| std::unique_ptr<CPDF_LinearizedHeader> CPDF_LinearizedHeader::Parse( |
| CPDF_SyntaxParser* parser) { |
| parser->SetPos(kLinearizedHeaderOffset); |
| |
| const auto pDict = ToDictionary( |
| parser->GetIndirectObject(nullptr, CPDF_SyntaxParser::ParseType::kLoose)); |
| |
| if (!pDict || !pDict->KeyExist("Linearized") || |
| !IsValidNumericDictionaryValue<FX_FILESIZE>(pDict.Get(), "L", 1) || |
| !IsValidNumericDictionaryValue<uint32_t>(pDict.Get(), "P", 0, false) || |
| !IsValidNumericDictionaryValue<FX_FILESIZE>(pDict.Get(), "T", 1) || |
| !IsValidNumericDictionaryValue<uint32_t>(pDict.Get(), "N", 1) || |
| !IsValidNumericDictionaryValue<FX_FILESIZE>(pDict.Get(), "E", 1) || |
| !IsValidNumericDictionaryValue<uint32_t>(pDict.Get(), "O", 1)) { |
| return nullptr; |
| } |
| // Move parser to the start of the xref table for the documents first page. |
| // (skpping endobj keyword) |
| if (parser->GetNextWord().word != "endobj") |
| return nullptr; |
| |
| auto result = pdfium::WrapUnique( |
| new CPDF_LinearizedHeader(pDict.Get(), parser->GetPos())); |
| |
| if (!IsLinearizedHeaderValid(result.get(), parser->GetDocumentSize())) |
| return nullptr; |
| |
| return result; |
| } |
| |
| CPDF_LinearizedHeader::CPDF_LinearizedHeader(const CPDF_Dictionary* pDict, |
| FX_FILESIZE szLastXRefOffset) |
| : m_szFileSize(pDict->GetIntegerFor("L")), |
| m_dwFirstPageNo(pDict->GetIntegerFor("P")), |
| m_szMainXRefTableFirstEntryOffset(pDict->GetIntegerFor("T")), |
| m_PageCount(pDict->GetIntegerFor("N")), |
| m_szFirstPageEndOffset(pDict->GetIntegerFor("E")), |
| m_FirstPageObjNum(pDict->GetIntegerFor("O")), |
| m_szLastXRefOffset(szLastXRefOffset) { |
| const CPDF_Array* pHintStreamRange = pDict->GetArrayFor("H"); |
| const size_t nHintStreamSize = |
| pHintStreamRange ? pHintStreamRange->size() : 0; |
| if (nHintStreamSize == 2 || nHintStreamSize == 4) { |
| m_szHintStart = std::max(pHintStreamRange->GetIntegerAt(0), 0); |
| const FX_SAFE_UINT32 safe_hint_length = pHintStreamRange->GetIntegerAt(1); |
| if (safe_hint_length.IsValid()) |
| m_HintLength = safe_hint_length.ValueOrDie(); |
| } |
| } |
| |
| CPDF_LinearizedHeader::~CPDF_LinearizedHeader() = default; |
| |
| bool CPDF_LinearizedHeader::HasHintTable() const { |
| return GetPageCount() > 1 && GetHintStart() > 0 && GetHintLength() > 0; |
| } |