1 // Copyright 2015 PDFium Authors. All rights reserved.
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 <cmath>
6 #include <limits>
7 #include <string>
8 
9 #include "fpdfsdk/fpdfview_c_api_test.h"
10 #include "public/fpdfview.h"
11 #include "testing/embedder_test.h"
12 #include "testing/gtest/include/gtest/gtest.h"
13 #include "testing/utils/path_service.h"
14 
15 namespace {
16 
17 class MockDownloadHints : public FX_DOWNLOADHINTS {
18  public:
SAddSegment(FX_DOWNLOADHINTS * pThis,size_t offset,size_t size)19   static void SAddSegment(FX_DOWNLOADHINTS* pThis, size_t offset, size_t size) {
20   }
21 
MockDownloadHints()22   MockDownloadHints() {
23     FX_DOWNLOADHINTS::version = 1;
24     FX_DOWNLOADHINTS::AddSegment = SAddSegment;
25   }
26 
~MockDownloadHints()27   ~MockDownloadHints() {}
28 };
29 
30 }  // namespace
31 
TEST(fpdf,CApiTest)32 TEST(fpdf, CApiTest) {
33   EXPECT_TRUE(CheckPDFiumCApi());
34 }
35 
36 class FPDFViewEmbeddertest : public EmbedderTest {
37  protected:
38   void TestRenderPageBitmapWithMatrix(FPDF_PAGE page,
39                                       const int bitmap_width,
40                                       const int bitmap_height,
41                                       const FS_MATRIX& matrix,
42                                       const FS_RECTF& rect,
43                                       const char* expected_md5);
44 };
45 
TEST_F(FPDFViewEmbeddertest,Document)46 TEST_F(FPDFViewEmbeddertest, Document) {
47   EXPECT_TRUE(OpenDocument("about_blank.pdf"));
48   EXPECT_EQ(1, GetPageCount());
49   EXPECT_EQ(0, GetFirstPageNum());
50 
51   int version;
52   EXPECT_TRUE(FPDF_GetFileVersion(document(), &version));
53   EXPECT_EQ(14, version);
54 
55   EXPECT_EQ(0xFFFFFFFF, FPDF_GetDocPermissions(document()));
56   EXPECT_EQ(-1, FPDF_GetSecurityHandlerRevision(document()));
57 }
58 
TEST_F(FPDFViewEmbeddertest,LoadNonexistentDocument)59 TEST_F(FPDFViewEmbeddertest, LoadNonexistentDocument) {
60   FPDF_DOCUMENT doc = FPDF_LoadDocument("nonexistent_document.pdf", "");
61   ASSERT_FALSE(doc);
62   EXPECT_EQ(static_cast<int>(FPDF_GetLastError()), FPDF_ERR_FILE);
63 }
64 
65 // See bug 465.
TEST_F(FPDFViewEmbeddertest,EmptyDocument)66 TEST_F(FPDFViewEmbeddertest, EmptyDocument) {
67   EXPECT_TRUE(CreateEmptyDocument());
68 
69   {
70     int version = 42;
71     EXPECT_FALSE(FPDF_GetFileVersion(document(), &version));
72     EXPECT_EQ(0, version);
73   }
74 
75   {
76 #ifndef PDF_ENABLE_XFA
77     const unsigned long kExpected = 0;
78 #else
79     const unsigned long kExpected = static_cast<uint32_t>(-1);
80 #endif
81     EXPECT_EQ(kExpected, FPDF_GetDocPermissions(document()));
82   }
83 
84   EXPECT_EQ(-1, FPDF_GetSecurityHandlerRevision(document()));
85 
86   EXPECT_EQ(0, FPDF_GetPageCount(document()));
87 
88   EXPECT_TRUE(FPDF_VIEWERREF_GetPrintScaling(document()));
89   EXPECT_EQ(1, FPDF_VIEWERREF_GetNumCopies(document()));
90   EXPECT_EQ(DuplexUndefined, FPDF_VIEWERREF_GetDuplex(document()));
91 
92   char buf[100];
93   EXPECT_EQ(0U, FPDF_VIEWERREF_GetName(document(), "foo", nullptr, 0));
94   EXPECT_EQ(0U, FPDF_VIEWERREF_GetName(document(), "foo", buf, sizeof(buf)));
95 
96   EXPECT_EQ(0u, FPDF_CountNamedDests(document()));
97 }
98 
TEST_F(FPDFViewEmbeddertest,LinearizedDocument)99 TEST_F(FPDFViewEmbeddertest, LinearizedDocument) {
100   EXPECT_TRUE(OpenDocumentLinearized("feature_linearized_loading.pdf"));
101   int version;
102   EXPECT_TRUE(FPDF_GetFileVersion(document(), &version));
103   EXPECT_EQ(16, version);
104 }
105 
TEST_F(FPDFViewEmbeddertest,Page)106 TEST_F(FPDFViewEmbeddertest, Page) {
107   EXPECT_TRUE(OpenDocument("about_blank.pdf"));
108   FPDF_PAGE page = LoadPage(0);
109   EXPECT_NE(nullptr, page);
110 
111   EXPECT_EQ(612.0, FPDF_GetPageWidth(page));
112   EXPECT_EQ(792.0, FPDF_GetPageHeight(page));
113 
114   FS_RECTF rect;
115   EXPECT_TRUE(FPDF_GetPageBoundingBox(page, &rect));
116   EXPECT_EQ(0.0, rect.left);
117   EXPECT_EQ(0.0, rect.bottom);
118   EXPECT_EQ(612.0, rect.right);
119   EXPECT_EQ(792.0, rect.top);
120 
121   UnloadPage(page);
122   EXPECT_EQ(nullptr, LoadPage(1));
123 }
124 
TEST_F(FPDFViewEmbeddertest,ViewerRefDummy)125 TEST_F(FPDFViewEmbeddertest, ViewerRefDummy) {
126   EXPECT_TRUE(OpenDocument("about_blank.pdf"));
127   EXPECT_TRUE(FPDF_VIEWERREF_GetPrintScaling(document()));
128   EXPECT_EQ(1, FPDF_VIEWERREF_GetNumCopies(document()));
129   EXPECT_EQ(DuplexUndefined, FPDF_VIEWERREF_GetDuplex(document()));
130 
131   char buf[100];
132   EXPECT_EQ(0U, FPDF_VIEWERREF_GetName(document(), "foo", nullptr, 0));
133   EXPECT_EQ(0U, FPDF_VIEWERREF_GetName(document(), "foo", buf, sizeof(buf)));
134 }
135 
TEST_F(FPDFViewEmbeddertest,ViewerRef)136 TEST_F(FPDFViewEmbeddertest, ViewerRef) {
137   EXPECT_TRUE(OpenDocument("viewer_ref.pdf"));
138   EXPECT_TRUE(FPDF_VIEWERREF_GetPrintScaling(document()));
139   EXPECT_EQ(5, FPDF_VIEWERREF_GetNumCopies(document()));
140   EXPECT_EQ(DuplexUndefined, FPDF_VIEWERREF_GetDuplex(document()));
141 
142   // Test some corner cases.
143   char buf[100];
144   EXPECT_EQ(0U, FPDF_VIEWERREF_GetName(document(), "", buf, sizeof(buf)));
145   EXPECT_EQ(0U, FPDF_VIEWERREF_GetName(document(), "foo", nullptr, 0));
146   EXPECT_EQ(0U, FPDF_VIEWERREF_GetName(document(), "foo", buf, sizeof(buf)));
147 
148   // Make sure |buf| does not get written into when it appears to be too small.
149   // NOLINTNEXTLINE(runtime/printf)
150   strcpy(buf, "ABCD");
151   EXPECT_EQ(4U, FPDF_VIEWERREF_GetName(document(), "Foo", buf, 1));
152   EXPECT_STREQ("ABCD", buf);
153 
154   // Note "Foo" is a different key from "foo".
155   EXPECT_EQ(4U,
156             FPDF_VIEWERREF_GetName(document(), "Foo", nullptr, sizeof(buf)));
157   ASSERT_EQ(4U, FPDF_VIEWERREF_GetName(document(), "Foo", buf, sizeof(buf)));
158   EXPECT_STREQ("foo", buf);
159 
160   // Try to retrieve a boolean and an integer.
161   EXPECT_EQ(
162       0U, FPDF_VIEWERREF_GetName(document(), "HideToolbar", buf, sizeof(buf)));
163   EXPECT_EQ(0U,
164             FPDF_VIEWERREF_GetName(document(), "NumCopies", buf, sizeof(buf)));
165 
166   // Try more valid cases.
167   ASSERT_EQ(4U,
168             FPDF_VIEWERREF_GetName(document(), "Direction", buf, sizeof(buf)));
169   EXPECT_STREQ("R2L", buf);
170   ASSERT_EQ(8U,
171             FPDF_VIEWERREF_GetName(document(), "ViewArea", buf, sizeof(buf)));
172   EXPECT_STREQ("CropBox", buf);
173 }
174 
TEST_F(FPDFViewEmbeddertest,NamedDests)175 TEST_F(FPDFViewEmbeddertest, NamedDests) {
176   EXPECT_TRUE(OpenDocument("named_dests.pdf"));
177   long buffer_size;
178   char fixed_buffer[512];
179   FPDF_DEST dest;
180 
181   // Query the size of the first item.
182   buffer_size = 2000000;  // Absurdly large, check not used for this case.
183   dest = FPDF_GetNamedDest(document(), 0, nullptr, &buffer_size);
184   EXPECT_NE(nullptr, dest);
185   EXPECT_EQ(12, buffer_size);
186 
187   // Try to retrieve the first item with too small a buffer.
188   buffer_size = 10;
189   dest = FPDF_GetNamedDest(document(), 0, fixed_buffer, &buffer_size);
190   EXPECT_NE(nullptr, dest);
191   EXPECT_EQ(-1, buffer_size);
192 
193   // Try to retrieve the first item with correctly sized buffer. Item is
194   // taken from Dests NameTree in named_dests.pdf.
195   buffer_size = 12;
196   dest = FPDF_GetNamedDest(document(), 0, fixed_buffer, &buffer_size);
197   EXPECT_NE(nullptr, dest);
198   EXPECT_EQ(12, buffer_size);
199   EXPECT_EQ(std::string("F\0i\0r\0s\0t\0\0\0", 12),
200             std::string(fixed_buffer, buffer_size));
201 
202   // Try to retrieve the second item with ample buffer. Item is taken
203   // from Dests NameTree but has a sub-dictionary in named_dests.pdf.
204   buffer_size = sizeof(fixed_buffer);
205   dest = FPDF_GetNamedDest(document(), 1, fixed_buffer, &buffer_size);
206   EXPECT_NE(nullptr, dest);
207   EXPECT_EQ(10, buffer_size);
208   EXPECT_EQ(std::string("N\0e\0x\0t\0\0\0", 10),
209             std::string(fixed_buffer, buffer_size));
210 
211   // Try to retrieve third item with ample buffer. Item is taken
212   // from Dests NameTree but has a bad sub-dictionary in named_dests.pdf.
213   // in named_dests.pdf).
214   buffer_size = sizeof(fixed_buffer);
215   dest = FPDF_GetNamedDest(document(), 2, fixed_buffer, &buffer_size);
216   EXPECT_EQ(nullptr, dest);
217   EXPECT_EQ(sizeof(fixed_buffer),
218             static_cast<size_t>(buffer_size));  // unmodified.
219 
220   // Try to retrieve the forth item with ample buffer. Item is taken
221   // from Dests NameTree but has a vale of the wrong type in named_dests.pdf.
222   buffer_size = sizeof(fixed_buffer);
223   dest = FPDF_GetNamedDest(document(), 3, fixed_buffer, &buffer_size);
224   EXPECT_EQ(nullptr, dest);
225   EXPECT_EQ(sizeof(fixed_buffer),
226             static_cast<size_t>(buffer_size));  // unmodified.
227 
228   // Try to retrieve fifth item with ample buffer. Item taken from the
229   // old-style Dests dictionary object in named_dests.pdf.
230   buffer_size = sizeof(fixed_buffer);
231   dest = FPDF_GetNamedDest(document(), 4, fixed_buffer, &buffer_size);
232   EXPECT_NE(nullptr, dest);
233   EXPECT_EQ(30, buffer_size);
234   EXPECT_EQ(std::string("F\0i\0r\0s\0t\0A\0l\0t\0e\0r\0n\0a\0t\0e\0\0\0", 30),
235             std::string(fixed_buffer, buffer_size));
236 
237   // Try to retrieve sixth item with ample buffer. Item istaken from the
238   // old-style Dests dictionary object but has a sub-dictionary in
239   // named_dests.pdf.
240   buffer_size = sizeof(fixed_buffer);
241   dest = FPDF_GetNamedDest(document(), 5, fixed_buffer, &buffer_size);
242   EXPECT_NE(nullptr, dest);
243   EXPECT_EQ(28, buffer_size);
244   EXPECT_EQ(std::string("L\0a\0s\0t\0A\0l\0t\0e\0r\0n\0a\0t\0e\0\0\0", 28),
245             std::string(fixed_buffer, buffer_size));
246 
247   // Try to retrieve non-existent item with ample buffer.
248   buffer_size = sizeof(fixed_buffer);
249   dest = FPDF_GetNamedDest(document(), 6, fixed_buffer, &buffer_size);
250   EXPECT_EQ(nullptr, dest);
251   EXPECT_EQ(sizeof(fixed_buffer),
252             static_cast<size_t>(buffer_size));  // unmodified.
253 
254   // Try to underflow/overflow the integer index.
255   buffer_size = sizeof(fixed_buffer);
256   dest = FPDF_GetNamedDest(document(), std::numeric_limits<int>::max(),
257                            fixed_buffer, &buffer_size);
258   EXPECT_EQ(nullptr, dest);
259   EXPECT_EQ(sizeof(fixed_buffer),
260             static_cast<size_t>(buffer_size));  // unmodified.
261 
262   buffer_size = sizeof(fixed_buffer);
263   dest = FPDF_GetNamedDest(document(), std::numeric_limits<int>::min(),
264                            fixed_buffer, &buffer_size);
265   EXPECT_EQ(nullptr, dest);
266   EXPECT_EQ(sizeof(fixed_buffer),
267             static_cast<size_t>(buffer_size));  // unmodified.
268 
269   buffer_size = sizeof(fixed_buffer);
270   dest = FPDF_GetNamedDest(document(), -1, fixed_buffer, &buffer_size);
271   EXPECT_EQ(nullptr, dest);
272   EXPECT_EQ(sizeof(fixed_buffer),
273             static_cast<size_t>(buffer_size));  // unmodified.
274 }
275 
TEST_F(FPDFViewEmbeddertest,NamedDestsByName)276 TEST_F(FPDFViewEmbeddertest, NamedDestsByName) {
277   EXPECT_TRUE(OpenDocument("named_dests.pdf"));
278 
279   // Null pointer returns nullptr.
280   FPDF_DEST dest = FPDF_GetNamedDestByName(document(), nullptr);
281   EXPECT_EQ(nullptr, dest);
282 
283   // Empty string returns nullptr.
284   dest = FPDF_GetNamedDestByName(document(), "");
285   EXPECT_EQ(nullptr, dest);
286 
287   // Item from Dests NameTree.
288   dest = FPDF_GetNamedDestByName(document(), "First");
289   EXPECT_NE(nullptr, dest);
290 
291   long ignore_len = 0;
292   FPDF_DEST dest_by_index =
293       FPDF_GetNamedDest(document(), 0, nullptr, &ignore_len);
294   EXPECT_EQ(dest_by_index, dest);
295 
296   // Item from Dests dictionary.
297   dest = FPDF_GetNamedDestByName(document(), "FirstAlternate");
298   EXPECT_NE(nullptr, dest);
299 
300   ignore_len = 0;
301   dest_by_index = FPDF_GetNamedDest(document(), 4, nullptr, &ignore_len);
302   EXPECT_EQ(dest_by_index, dest);
303 
304   // Bad value type for item from Dests NameTree array.
305   dest = FPDF_GetNamedDestByName(document(), "WrongType");
306   EXPECT_EQ(nullptr, dest);
307 
308   // No such destination in either Dest NameTree or dictionary.
309   dest = FPDF_GetNamedDestByName(document(), "Bogus");
310   EXPECT_EQ(nullptr, dest);
311 }
312 
313 // The following tests pass if the document opens without crashing.
TEST_F(FPDFViewEmbeddertest,Crasher_113)314 TEST_F(FPDFViewEmbeddertest, Crasher_113) {
315   EXPECT_TRUE(OpenDocument("bug_113.pdf"));
316 }
317 
TEST_F(FPDFViewEmbeddertest,Crasher_451830)318 TEST_F(FPDFViewEmbeddertest, Crasher_451830) {
319   // Document is damaged and can't be opened.
320   EXPECT_FALSE(OpenDocument("bug_451830.pdf"));
321 }
322 
TEST_F(FPDFViewEmbeddertest,Crasher_452455)323 TEST_F(FPDFViewEmbeddertest, Crasher_452455) {
324   EXPECT_TRUE(OpenDocument("bug_452455.pdf"));
325   FPDF_PAGE page = LoadPage(0);
326   EXPECT_NE(nullptr, page);
327   UnloadPage(page);
328 }
329 
TEST_F(FPDFViewEmbeddertest,Crasher_454695)330 TEST_F(FPDFViewEmbeddertest, Crasher_454695) {
331   // Document is damaged and can't be opened.
332   EXPECT_FALSE(OpenDocument("bug_454695.pdf"));
333 }
334 
TEST_F(FPDFViewEmbeddertest,Crasher_572871)335 TEST_F(FPDFViewEmbeddertest, Crasher_572871) {
336   EXPECT_TRUE(OpenDocument("bug_572871.pdf"));
337 }
338 
339 // It tests that document can still be loaded even the trailer has no 'Size'
340 // field if other information is right.
TEST_F(FPDFViewEmbeddertest,Failed_213)341 TEST_F(FPDFViewEmbeddertest, Failed_213) {
342   EXPECT_TRUE(OpenDocument("bug_213.pdf"));
343 }
344 
345 // The following tests pass if the document opens without infinite looping.
TEST_F(FPDFViewEmbeddertest,Hang_298)346 TEST_F(FPDFViewEmbeddertest, Hang_298) {
347   EXPECT_FALSE(OpenDocument("bug_298.pdf"));
348 }
349 
350 // Test if the document opens without infinite looping.
351 // Previously this test will hang in a loop inside LoadAllCrossRefV4. After
352 // the fix, LoadAllCrossRefV4 will return false after detecting a cross
353 // reference loop. Cross references will be rebuilt successfully.
TEST_F(FPDFViewEmbeddertest,CrossRefV4Loop)354 TEST_F(FPDFViewEmbeddertest, CrossRefV4Loop) {
355   EXPECT_TRUE(OpenDocument("bug_xrefv4_loop.pdf"));
356   MockDownloadHints hints;
357 
358   // Make sure calling FPDFAvail_IsDocAvail() on this file does not infinite
359   // loop either. See bug 875.
360   int ret = PDF_DATA_NOTAVAIL;
361   while (ret == PDF_DATA_NOTAVAIL)
362     ret = FPDFAvail_IsDocAvail(avail_, &hints);
363   EXPECT_EQ(PDF_DATA_AVAIL, ret);
364 }
365 
366 // The test should pass when circular references to ParseIndirectObject will not
367 // cause infinite loop.
TEST_F(FPDFViewEmbeddertest,Hang_343)368 TEST_F(FPDFViewEmbeddertest, Hang_343) {
369   EXPECT_FALSE(OpenDocument("bug_343.pdf"));
370 }
371 
372 // The test should pass when the absence of 'Contents' field in a signature
373 // dictionary will not cause an infinite loop in CPDF_SyntaxParser::GetObject().
TEST_F(FPDFViewEmbeddertest,Hang_344)374 TEST_F(FPDFViewEmbeddertest, Hang_344) {
375   EXPECT_FALSE(OpenDocument("bug_344.pdf"));
376 }
377 
378 // The test should pass when there is no infinite recursion in
379 // CPDF_SyntaxParser::GetString().
TEST_F(FPDFViewEmbeddertest,Hang_355)380 TEST_F(FPDFViewEmbeddertest, Hang_355) {
381   EXPECT_FALSE(OpenDocument("bug_355.pdf"));
382 }
383 // The test should pass even when the file has circular references to pages.
TEST_F(FPDFViewEmbeddertest,Hang_360)384 TEST_F(FPDFViewEmbeddertest, Hang_360) {
385   EXPECT_FALSE(OpenDocument("bug_360.pdf"));
386 }
387 
TestRenderPageBitmapWithMatrix(FPDF_PAGE page,const int bitmap_width,const int bitmap_height,const FS_MATRIX & matrix,const FS_RECTF & rect,const char * expected_md5)388 void FPDFViewEmbeddertest::TestRenderPageBitmapWithMatrix(
389     FPDF_PAGE page,
390     const int bitmap_width,
391     const int bitmap_height,
392     const FS_MATRIX& matrix,
393     const FS_RECTF& rect,
394     const char* expected_md5) {
395   FPDF_BITMAP bitmap = FPDFBitmap_Create(bitmap_width, bitmap_height, 0);
396   FPDFBitmap_FillRect(bitmap, 0, 0, bitmap_width, bitmap_height, 0xFFFFFFFF);
397   FPDF_RenderPageBitmapWithMatrix(bitmap, page, &matrix, &rect, 0);
398   CompareBitmap(bitmap, bitmap_width, bitmap_height, expected_md5);
399   FPDFBitmap_Destroy(bitmap);
400 }
401 
TEST_F(FPDFViewEmbeddertest,FPDF_RenderPageBitmapWithMatrix)402 TEST_F(FPDFViewEmbeddertest, FPDF_RenderPageBitmapWithMatrix) {
403   const char kOriginalMD5[] = "0a90de37f52127619c3dfb642b5fa2fe";
404   const char kClippedMD5[] = "a84cab93c102b9b9290fba3047ba702c";
405   const char kTopLeftQuarterMD5[] = "f11a11137c8834389e31cf555a4a6979";
406   const char kHoriStretchedMD5[] = "48ef9205941ed19691ccfa00d717187e";
407   const char kRotated90ClockwiseMD5[] = "d8da2c7bf77521550d0f2752b9cf3482";
408   const char kRotated180ClockwiseMD5[] = "0113386bb0bd45125bacc6dee78bfe78";
409   const char kRotated270ClockwiseMD5[] = "a287e0f74ce203699cda89f9cc97a240";
410   const char kMirrorHoriMD5[] = "6e8d7a6fde39d8e720fb9e620102918c";
411   const char kMirrorVertMD5[] = "8f3a555ef9c0d5031831ae3715273707";
412   const char kLargerTopLeftQuarterMD5[] = "172a2f4adafbadbe98017b1c025b9e27";
413   const char kLargerMD5[] = "c806145641c3e6fc4e022c7065343749";
414   const char kLargerClippedMD5[] = "091d3b1c7933c8f6945eb2cb41e588e9";
415   const char kLargerRotatedMD5[] = "115f13353ebfc82ddb392d1f0059eb12";
416   const char kLargerRotatedLandscapeMD5[] = "c901239d17d84ac84cb6f2124da71b0d";
417   const char kLargerRotatedDiagonalMD5[] = "3d62417468bdaff0eb14391a0c30a3b1";
418   const char kTileMD5[] = "0a190003c97220bf8877684c8d7e89cf";
419 
420   EXPECT_TRUE(OpenDocument("rectangles.pdf"));
421   FPDF_PAGE page = LoadPage(0);
422   EXPECT_NE(nullptr, page);
423   const int page_width = static_cast<int>(FPDF_GetPageWidth(page));
424   const int page_height = static_cast<int>(FPDF_GetPageHeight(page));
425   EXPECT_EQ(200, page_width);
426   EXPECT_EQ(300, page_height);
427 
428   FPDF_BITMAP bitmap = RenderPage(page);
429   CompareBitmap(bitmap, page_width, page_height, kOriginalMD5);
430   FPDFBitmap_Destroy(bitmap);
431 
432   FS_RECTF page_rect{0, 0, page_width, page_height};
433 
434   // Try rendering with an identity matrix. The output should be the same as
435   // the RenderPage() output.
436   FS_MATRIX identity_matrix{1, 0, 0, 1, 0, 0};
437   TestRenderPageBitmapWithMatrix(page, page_width, page_height, identity_matrix,
438                                  page_rect, kOriginalMD5);
439 
440   // Again render with an identity matrix but with a smaller clipping rect.
441   FS_RECTF middle_of_page_rect{page_width / 4, page_height / 4,
442                                page_width * 3 / 4, page_height * 3 / 4};
443   TestRenderPageBitmapWithMatrix(page, page_width, page_height, identity_matrix,
444                                  middle_of_page_rect, kClippedMD5);
445 
446   // Now render again with the image scaled smaller.
447   FS_MATRIX half_scale_matrix{0.5, 0, 0, 0.5, 0, 0};
448   TestRenderPageBitmapWithMatrix(page, page_width, page_height,
449                                  half_scale_matrix, page_rect,
450                                  kTopLeftQuarterMD5);
451 
452   // Now render again with the image scaled larger horizontally (the right half
453   // will be clipped).
454   FS_MATRIX stretch_x_matrix{2, 0, 0, 1, 0, 0};
455   TestRenderPageBitmapWithMatrix(page, page_width, page_height,
456                                  stretch_x_matrix, page_rect,
457                                  kHoriStretchedMD5);
458 
459   // Try a 90 degree rotation clockwise but with the same bitmap size, so part
460   // will be clipped.
461   FS_MATRIX rotate_90_matrix{0, 1, -1, 0, page_width, 0};
462   TestRenderPageBitmapWithMatrix(page, page_width, page_height,
463                                  rotate_90_matrix, page_rect,
464                                  kRotated90ClockwiseMD5);
465 
466   // 180 degree rotation clockwise.
467   FS_MATRIX rotate_180_matrix{-1, 0, 0, -1, page_width, page_height};
468   TestRenderPageBitmapWithMatrix(page, page_width, page_height,
469                                  rotate_180_matrix, page_rect,
470                                  kRotated180ClockwiseMD5);
471 
472   // 270 degree rotation clockwise.
473   FS_MATRIX rotate_270_matrix{0, -1, 1, 0, 0, page_width};
474   TestRenderPageBitmapWithMatrix(page, page_width, page_height,
475                                  rotate_270_matrix, page_rect,
476                                  kRotated270ClockwiseMD5);
477 
478   // Mirror horizontally.
479   FS_MATRIX mirror_hori_matrix{-1, 0, 0, 1, page_width, 0};
480   TestRenderPageBitmapWithMatrix(page, page_width, page_height,
481                                  mirror_hori_matrix, page_rect, kMirrorHoriMD5);
482 
483   // Mirror vertically.
484   FS_MATRIX mirror_vert_matrix{1, 0, 0, -1, 0, page_height};
485   TestRenderPageBitmapWithMatrix(page, page_width, page_height,
486                                  mirror_vert_matrix, page_rect, kMirrorVertMD5);
487 
488   // Tests rendering to a larger bitmap
489   const int bitmap_width = page_width * 2;
490   const int bitmap_height = page_height * 2;
491 
492   // Render using an identity matrix and the whole bitmap area as clipping rect.
493   FS_RECTF bitmap_rect{0, 0, bitmap_width, bitmap_height};
494   TestRenderPageBitmapWithMatrix(page, bitmap_width, bitmap_height,
495                                  identity_matrix, bitmap_rect,
496                                  kLargerTopLeftQuarterMD5);
497 
498   // Render using a scaling matrix to fill the larger bitmap.
499   FS_MATRIX double_scale_matrix{2, 0, 0, 2, 0, 0};
500   TestRenderPageBitmapWithMatrix(page, bitmap_width, bitmap_height,
501                                  double_scale_matrix, bitmap_rect, kLargerMD5);
502 
503   // Render the larger image again but with clipping.
504   FS_RECTF middle_of_bitmap_rect{bitmap_width / 4, bitmap_height / 4,
505                                  bitmap_width * 3 / 4, bitmap_height * 3 / 4};
506   TestRenderPageBitmapWithMatrix(page, bitmap_width, bitmap_height,
507                                  double_scale_matrix, middle_of_bitmap_rect,
508                                  kLargerClippedMD5);
509 
510   // On the larger bitmap, try a 90 degree rotation but with the same bitmap
511   // size, so part will be clipped.
512   FS_MATRIX rotate_90_scale_2_matrix{0, 2, -2, 0, bitmap_width, 0};
513   TestRenderPageBitmapWithMatrix(page, bitmap_width, bitmap_height,
514                                  rotate_90_scale_2_matrix, bitmap_rect,
515                                  kLargerRotatedMD5);
516 
517   // On the larger bitmap, apply 90 degree rotation to a bitmap with the
518   // appropriate dimensions.
519   const int landscape_bitmap_width = bitmap_height;
520   const int landscape_bitmap_height = bitmap_width;
521   FS_RECTF landscape_bitmap_rect{0, 0, landscape_bitmap_width,
522                                  landscape_bitmap_height};
523   FS_MATRIX landscape_rotate_90_scale_2_matrix{
524       0, 2, -2, 0, landscape_bitmap_width, 0};
525   TestRenderPageBitmapWithMatrix(
526       page, landscape_bitmap_width, landscape_bitmap_height,
527       landscape_rotate_90_scale_2_matrix, landscape_bitmap_rect,
528       kLargerRotatedLandscapeMD5);
529 
530   // On the larger bitmap, apply 45 degree rotation to a bitmap with the
531   // appropriate dimensions.
532   const float sqrt2 = 1.41421356f;
533   const int diagonal_bitmap_size = ceil((bitmap_width + bitmap_height) / sqrt2);
534   FS_RECTF diagonal_bitmap_rect{0, 0, diagonal_bitmap_size,
535                                 diagonal_bitmap_size};
536   FS_MATRIX rotate_45_scale_2_matrix{
537       sqrt2, sqrt2, -sqrt2, sqrt2, bitmap_height / sqrt2, 0};
538   TestRenderPageBitmapWithMatrix(page, diagonal_bitmap_size,
539                                  diagonal_bitmap_size, rotate_45_scale_2_matrix,
540                                  diagonal_bitmap_rect,
541                                  kLargerRotatedDiagonalMD5);
542 
543   // Render the (2, 1) tile of the page (third column, second row) when the page
544   // is divided in 50x50 pixel tiles. The tile is scaled by a factor of 7.
545   const float scale = 7.0;
546   const int tile_size = 50;
547   const int tile_x = 2;
548   const int tile_y = 1;
549   int tile_bitmap_size = scale * tile_size;
550   FS_RECTF tile_bitmap_rect{0, 0, tile_bitmap_size, tile_bitmap_size};
551   FS_MATRIX tile_2_1_matrix{scale,
552                             0,
553                             0,
554                             scale,
555                             -tile_x * tile_bitmap_size,
556                             -tile_y * tile_bitmap_size};
557   TestRenderPageBitmapWithMatrix(page, tile_bitmap_size, tile_bitmap_size,
558                                  tile_2_1_matrix, tile_bitmap_rect, kTileMD5);
559 
560   UnloadPage(page);
561 }
562 
563 class UnSupRecordDelegate : public EmbedderTest::Delegate {
564  public:
UnSupRecordDelegate()565   UnSupRecordDelegate() : type_(-1) {}
~UnSupRecordDelegate()566   ~UnSupRecordDelegate() override {}
567 
UnsupportedHandler(int type)568   void UnsupportedHandler(int type) override { type_ = type; }
569 
570   int type_;
571 };
572 
TEST_F(FPDFViewEmbeddertest,UnSupportedOperations_NotFound)573 TEST_F(FPDFViewEmbeddertest, UnSupportedOperations_NotFound) {
574   UnSupRecordDelegate delegate;
575   SetDelegate(&delegate);
576   ASSERT_TRUE(OpenDocument("hello_world.pdf"));
577   EXPECT_EQ(delegate.type_, -1);
578   SetDelegate(nullptr);
579 }
580 
TEST_F(FPDFViewEmbeddertest,UnSupportedOperations_LoadCustomDocument)581 TEST_F(FPDFViewEmbeddertest, UnSupportedOperations_LoadCustomDocument) {
582   UnSupRecordDelegate delegate;
583   SetDelegate(&delegate);
584   ASSERT_TRUE(OpenDocument("unsupported_feature.pdf"));
585   EXPECT_EQ(FPDF_UNSP_DOC_PORTABLECOLLECTION, delegate.type_);
586   SetDelegate(nullptr);
587 }
588 
TEST_F(FPDFViewEmbeddertest,UnSupportedOperations_LoadDocument)589 TEST_F(FPDFViewEmbeddertest, UnSupportedOperations_LoadDocument) {
590   std::string file_path;
591   ASSERT_TRUE(
592       PathService::GetTestFilePath("unsupported_feature.pdf", &file_path));
593 
594   UnSupRecordDelegate delegate;
595   SetDelegate(&delegate);
596   FPDF_DOCUMENT doc = FPDF_LoadDocument(file_path.c_str(), "");
597   EXPECT_TRUE(doc != nullptr);
598   EXPECT_EQ(FPDF_UNSP_DOC_PORTABLECOLLECTION, delegate.type_);
599   FPDF_CloseDocument(doc);
600   SetDelegate(nullptr);
601 }
602