Qt
Internal/Contributor docs for the Qt SDK. Note: These are NOT official API docs; those are found at https://doc.qt.io/
Loading...
Searching...
No Matches
cpdf_hint_tables_unittest.cpp
Go to the documentation of this file.
1// Copyright 2017 The PDFium Authors
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5#include "core/fpdfapi/parser/cpdf_hint_tables.h"
6
7#include <memory>
8#include <string>
9#include <utility>
10
11#include "core/fpdfapi/page/test_with_page_module.h"
12#include "core/fpdfapi/parser/cpdf_data_avail.h"
13#include "core/fpdfapi/parser/cpdf_dictionary.h"
14#include "core/fpdfapi/parser/cpdf_linearized_header.h"
15#include "core/fpdfapi/parser/cpdf_object.h"
16#include "core/fpdfapi/parser/cpdf_read_validator.h"
17#include "core/fpdfapi/parser/cpdf_stream.h"
18#include "core/fpdfapi/parser/cpdf_syntax_parser.h"
19#include "core/fxcrt/bytestring.h"
20#include "core/fxcrt/cfx_read_only_string_stream.h"
21#include "core/fxcrt/fx_stream.h"
22#include "testing/gmock/include/gmock/gmock.h"
23#include "testing/gtest/include/gtest/gtest.h"
24#include "testing/utils/path_service.h"
25#include "third_party/base/check.h"
26
27namespace {
28
29RetainPtr<CPDF_ReadValidator> MakeValidatorFromFile(
30 const std::string& file_name) {
31 std::string file_path = PathService::GetTestFilePath(file_name);
32 CHECK(!file_path.empty());
33 return pdfium::MakeRetain<CPDF_ReadValidator>(
34 IFX_SeekableReadStream::CreateFromFilename(file_path.c_str()), nullptr);
35}
36
37std::unique_ptr<CPDF_DataAvail> MakeDataAvailFromFile(
38 const std::string& file_name) {
39 return std::make_unique<CPDF_DataAvail>(nullptr,
40 MakeValidatorFromFile(file_name));
41}
42
43class TestLinearizedHeader final : public CPDF_LinearizedHeader {
44 public:
45 TestLinearizedHeader(const CPDF_Dictionary* pDict,
46 FX_FILESIZE szLastXRefOffset)
47 : CPDF_LinearizedHeader(pDict, szLastXRefOffset) {}
48
49 static std::unique_ptr<CPDF_LinearizedHeader> MakeHeader(
50 ByteString inline_data) {
51 CPDF_SyntaxParser parser(
52 pdfium::MakeRetain<CFX_ReadOnlyStringStream>(std::move(inline_data)));
53 RetainPtr<CPDF_Dictionary> dict =
54 ToDictionary(parser.GetObjectBody(nullptr));
55 DCHECK(dict);
56 return std::make_unique<TestLinearizedHeader>(dict.Get(), 0);
57 }
58};
59
60} // namespace
61
62// Needs page module for encoding Hint table stream.
63using HintTablesTest = TestWithPageModule;
64
65TEST_F(HintTablesTest, Load) {
66 auto data_avail = MakeDataAvailFromFile("feature_linearized_loading.pdf");
67 ASSERT_EQ(CPDF_DataAvail::kDataAvailable, data_avail->IsDocAvail(nullptr));
68
69 ASSERT_TRUE(data_avail->GetHintTablesForTest());
70
71 const CPDF_HintTables* hint_tables = data_avail->GetHintTablesForTest();
72 FX_FILESIZE page_start = 0;
73 FX_FILESIZE page_length = 0;
74 uint32_t page_obj_num = 0;
75
76 ASSERT_TRUE(
77 hint_tables->GetPagePos(0, &page_start, &page_length, &page_obj_num));
78 EXPECT_EQ(777, page_start);
79 EXPECT_EQ(4328, page_length);
80 EXPECT_EQ(39u, page_obj_num);
81
82 ASSERT_TRUE(
83 hint_tables->GetPagePos(1, &page_start, &page_length, &page_obj_num));
84 EXPECT_EQ(5105, page_start);
85 EXPECT_EQ(767, page_length);
86 EXPECT_EQ(1u, page_obj_num);
87
88 ASSERT_FALSE(
89 hint_tables->GetPagePos(2, &page_start, &page_length, &page_obj_num));
90}
91
92TEST_F(HintTablesTest, PageAndGroupInfos) {
93 auto data_avail = MakeDataAvailFromFile("feature_linearized_loading.pdf");
94 ASSERT_EQ(CPDF_DataAvail::kDataAvailable, data_avail->IsDocAvail(nullptr));
95
96 const CPDF_HintTables* hint_tables = data_avail->GetHintTablesForTest();
97 ASSERT_TRUE(hint_tables);
98 ASSERT_EQ(2u, hint_tables->PageInfos().size());
99
100 EXPECT_EQ(5u, hint_tables->PageInfos()[0].objects_count());
101 EXPECT_EQ(777, hint_tables->PageInfos()[0].page_offset());
102 EXPECT_EQ(4328u, hint_tables->PageInfos()[0].page_length());
103 EXPECT_EQ(39u, hint_tables->PageInfos()[0].start_obj_num());
104 ASSERT_EQ(2u, hint_tables->PageInfos()[0].Identifiers().size());
105
106 EXPECT_EQ(0u, hint_tables->PageInfos()[0].Identifiers()[0]);
107 EXPECT_EQ(0u, hint_tables->PageInfos()[0].Identifiers()[1]);
108
109 EXPECT_EQ(3u, hint_tables->PageInfos()[1].objects_count());
110 EXPECT_EQ(5105, hint_tables->PageInfos()[1].page_offset());
111 EXPECT_EQ(767u, hint_tables->PageInfos()[1].page_length());
112 EXPECT_EQ(1u, hint_tables->PageInfos()[1].start_obj_num());
113 ASSERT_EQ(3u, hint_tables->PageInfos()[1].Identifiers().size());
114
115 EXPECT_EQ(2u, hint_tables->PageInfos()[1].Identifiers()[0]);
116 EXPECT_EQ(5u, hint_tables->PageInfos()[1].Identifiers()[1]);
117 EXPECT_EQ(3u, hint_tables->PageInfos()[1].Identifiers()[2]);
118
119 // SharedGroupInfo
120 ASSERT_EQ(6u, hint_tables->SharedGroupInfos().size());
121
122 EXPECT_EQ(777, hint_tables->SharedGroupInfos()[0].m_szOffset);
123 EXPECT_EQ(254u, hint_tables->SharedGroupInfos()[0].m_dwLength);
124 EXPECT_EQ(39u, hint_tables->SharedGroupInfos()[0].m_dwStartObjNum);
125 EXPECT_EQ(1u, hint_tables->SharedGroupInfos()[0].m_dwObjectsCount);
126
127 EXPECT_EQ(1031, hint_tables->SharedGroupInfos()[1].m_szOffset);
128 EXPECT_EQ(389u, hint_tables->SharedGroupInfos()[1].m_dwLength);
129 EXPECT_EQ(40u, hint_tables->SharedGroupInfos()[1].m_dwStartObjNum);
130 EXPECT_EQ(1u, hint_tables->SharedGroupInfos()[1].m_dwObjectsCount);
131
132 EXPECT_EQ(1420, hint_tables->SharedGroupInfos()[2].m_szOffset);
133 EXPECT_EQ(726u, hint_tables->SharedGroupInfos()[2].m_dwLength);
134 EXPECT_EQ(41u, hint_tables->SharedGroupInfos()[2].m_dwStartObjNum);
135 EXPECT_EQ(1u, hint_tables->SharedGroupInfos()[2].m_dwObjectsCount);
136
137 EXPECT_EQ(2146, hint_tables->SharedGroupInfos()[3].m_szOffset);
138 EXPECT_EQ(290u, hint_tables->SharedGroupInfos()[3].m_dwLength);
139 EXPECT_EQ(42u, hint_tables->SharedGroupInfos()[3].m_dwStartObjNum);
140 EXPECT_EQ(1u, hint_tables->SharedGroupInfos()[3].m_dwObjectsCount);
141
142 EXPECT_EQ(2436, hint_tables->SharedGroupInfos()[4].m_szOffset);
143 EXPECT_EQ(2669u, hint_tables->SharedGroupInfos()[4].m_dwLength);
144 EXPECT_EQ(43u, hint_tables->SharedGroupInfos()[4].m_dwStartObjNum);
145 EXPECT_EQ(1u, hint_tables->SharedGroupInfos()[4].m_dwObjectsCount);
146
147 EXPECT_EQ(10939, hint_tables->SharedGroupInfos()[5].m_szOffset);
148 EXPECT_EQ(544u, hint_tables->SharedGroupInfos()[5].m_dwLength);
149 EXPECT_EQ(4u, hint_tables->SharedGroupInfos()[5].m_dwStartObjNum);
150 EXPECT_EQ(1u, hint_tables->SharedGroupInfos()[5].m_dwObjectsCount);
151}
152
153TEST_F(HintTablesTest, FirstPageOffset) {
154 // Test that valid hint table is loaded, and have correct offset of first page
155 // object.
156 const auto linearized_header = TestLinearizedHeader::MakeHeader(
157 "<< /Linearized 1 /L 19326762 /H [ 123730 3816 ] /O 5932 /E 639518 /N "
158 "102 /T 19220281 >>");
159 ASSERT_TRUE(linearized_header);
160 // This hint table is extracted from linearized file, generated by qpdf tool.
161 RetainPtr<CPDF_ReadValidator> validator =
162 MakeValidatorFromFile("hint_table_102p.bin");
163 CPDF_SyntaxParser parser(validator, 0);
164 RetainPtr<CPDF_Stream> stream = ToStream(parser.GetObjectBody(nullptr));
165 ASSERT_TRUE(stream);
166 auto hint_tables = std::make_unique<CPDF_HintTables>(validator.Get(),
167 linearized_header.get());
168 // Check that hint table will load.
169 ASSERT_TRUE(hint_tables->LoadHintStream(stream.Get()));
170 // Check that hint table have correct first page offset.
171 // 127546 is predefined real value from original file.
172 EXPECT_EQ(127546, hint_tables->GetFirstPageObjOffset());
173}
bool GetPagePos(uint32_t index, FX_FILESIZE *szPageStartPos, FX_FILESIZE *szPageLength, uint32_t *dwObjNum) const
CPDF_LinearizedHeader(const CPDF_Dictionary *pDict, FX_FILESIZE szLastXRefOffset)
static RetainPtr< IFX_SeekableReadStream > CreateFromFilename(const char *filename)
Definition fx_stream.cpp:68
TEST_F(CPDF_CIDFontTest, BUG_920636)
#define FX_FILESIZE
Definition fx_types.h:19
#define CHECK(cvref)