xref: /aosp_15_r20/external/skia/tests/PDFDocumentTest.cpp (revision c8dee2aa9b3f27cf6c858bd81872bdeb2c07ed17)
1 /*
2  * Copyright 2013 Google Inc.
3  *
4  * Use of this source code is governed by a BSD-style license that can be
5  * found in the LICENSE file.
6  */
7 #include "include/core/SkBitmap.h"
8 #include "include/core/SkCanvas.h"
9 #include "include/core/SkColor.h"
10 #include "include/core/SkData.h"
11 #include "include/core/SkDocument.h"
12 #include "include/core/SkExecutor.h"
13 #include "include/core/SkFont.h"
14 #include "include/core/SkImage.h" // IWYU pragma: keep
15 #include "include/core/SkPaint.h"
16 #include "include/core/SkRefCnt.h"
17 #include "include/core/SkStream.h"
18 #include "include/core/SkString.h"
19 #include "include/docs/SkPDFDocument.h"
20 #include "src/utils/SkOSPath.h"
21 #include "tests/Test.h"
22 #include "tools/fonts/FontToolUtils.h"
23 
24 #include <cstdint>
25 #include <cstdio>
26 #include <cstring>
27 #include <memory>
28 
test_empty(skiatest::Reporter * reporter)29 static void test_empty(skiatest::Reporter* reporter) {
30     SkDynamicMemoryWStream stream;
31 
32     auto doc = SkPDF::MakeDocument(&stream);
33 
34     doc->close();
35 
36     REPORTER_ASSERT(reporter, stream.bytesWritten() == 0);
37 }
38 
test_abort(skiatest::Reporter * reporter)39 static void test_abort(skiatest::Reporter* reporter) {
40     SkDynamicMemoryWStream stream;
41     auto doc = SkPDF::MakeDocument(&stream);
42 
43     SkCanvas* canvas = doc->beginPage(100, 100);
44     canvas->drawColor(SK_ColorRED);
45     doc->endPage();
46 
47     doc->abort();
48 
49     // Test that only the header is written, not the full document.
50     REPORTER_ASSERT(reporter, stream.bytesWritten() < 256);
51 }
52 
test_abortWithFile(skiatest::Reporter * reporter)53 static void test_abortWithFile(skiatest::Reporter* reporter) {
54     SkString tmpDir = skiatest::GetTmpDir();
55 
56     if (tmpDir.isEmpty()) {
57         ERRORF(reporter, "missing tmpDir.");
58         return;
59     }
60 
61     SkString path = SkOSPath::Join(tmpDir.c_str(), "aborted.pdf");
62     if (!SkFILEWStream(path.c_str()).isValid()) {
63         ERRORF(reporter, "unable to write to: %s", path.c_str());
64         return;
65     }
66 
67     // Make sure doc's destructor is called to flush.
68     {
69         SkFILEWStream stream(path.c_str());
70         auto doc = SkPDF::MakeDocument(&stream);
71 
72         SkCanvas* canvas = doc->beginPage(100, 100);
73         canvas->drawColor(SK_ColorRED);
74         doc->endPage();
75 
76         doc->abort();
77     }
78 
79     FILE* file = fopen(path.c_str(), "r");
80     // Test that only the header is written, not the full document.
81     char buffer[256];
82     REPORTER_ASSERT(reporter, fread(buffer, 1, sizeof(buffer), file) < sizeof(buffer));
83     fclose(file);
84 }
85 
test_file(skiatest::Reporter * reporter)86 static void test_file(skiatest::Reporter* reporter) {
87     SkString tmpDir = skiatest::GetTmpDir();
88     if (tmpDir.isEmpty()) {
89         ERRORF(reporter, "missing tmpDir.");
90         return;
91     }
92 
93     SkString path = SkOSPath::Join(tmpDir.c_str(), "file.pdf");
94     if (!SkFILEWStream(path.c_str()).isValid()) {
95         ERRORF(reporter, "unable to write to: %s", path.c_str());
96         return;
97     }
98 
99     {
100         SkFILEWStream stream(path.c_str());
101         auto doc = SkPDF::MakeDocument(&stream);
102         SkCanvas* canvas = doc->beginPage(100, 100);
103 
104         canvas->drawColor(SK_ColorRED);
105         doc->endPage();
106         doc->close();
107     }
108 
109     FILE* file = fopen(path.c_str(), "r");
110     REPORTER_ASSERT(reporter, file != nullptr);
111     char header[100];
112     REPORTER_ASSERT(reporter, fread(header, 4, 1, file) != 0);
113     REPORTER_ASSERT(reporter, strncmp(header, "%PDF", 4) == 0);
114     fclose(file);
115 }
116 
test_close(skiatest::Reporter * reporter)117 static void test_close(skiatest::Reporter* reporter) {
118     SkDynamicMemoryWStream stream;
119     auto doc = SkPDF::MakeDocument(&stream);
120 
121     SkCanvas* canvas = doc->beginPage(100, 100);
122     canvas->drawColor(SK_ColorRED);
123     doc->endPage();
124 
125     doc->close();
126 
127     REPORTER_ASSERT(reporter, stream.bytesWritten() != 0);
128 }
129 
DEF_TEST(SkPDF_document_tests,reporter)130 DEF_TEST(SkPDF_document_tests, reporter) {
131     REQUIRE_PDF_DOCUMENT(document_tests, reporter);
132     test_empty(reporter);
133     test_abort(reporter);
134     test_abortWithFile(reporter);
135     test_file(reporter);
136     test_close(reporter);
137 }
138 
DEF_TEST(SkPDF_document_skbug_4734,r)139 DEF_TEST(SkPDF_document_skbug_4734, r) {
140     REQUIRE_PDF_DOCUMENT(SkPDF_document_skbug_4734, r);
141     SkDynamicMemoryWStream stream;
142     auto doc = SkPDF::MakeDocument(&stream);
143     SkCanvas* canvas = doc->beginPage(64, 64);
144     canvas->scale(10000.0f, 10000.0f);
145     canvas->translate(20.0f, 10.0f);
146     canvas->rotate(30.0f);
147     const char text[] = "HELLO";
148     canvas->drawString(text, 0, 0, ToolUtils::DefaultFont(), SkPaint());
149 }
150 
contains(const uint8_t * result,size_t size,const char expectation[])151 static bool contains(const uint8_t* result, size_t size, const char expectation[]) {
152     size_t len = strlen(expectation);
153     size_t N = 1 + size - len;
154     for (size_t i = 0; i < N; ++i) {
155         if (0 == memcmp(result + i, expectation, len)) {
156             return true;
157         }
158     }
159     return false;
160 }
161 
162 // verify that the PDFA flag does something.
DEF_TEST(SkPDF_pdfa_document,r)163 DEF_TEST(SkPDF_pdfa_document, r) {
164     REQUIRE_PDF_DOCUMENT(SkPDF_pdfa_document, r);
165 
166     SkPDF::Metadata pdfMetadata;
167     pdfMetadata.fTitle = "test document";
168     pdfMetadata.fCreation = {0, 1999, 12, 5, 31, 23, 59, 59};
169     pdfMetadata.fPDFA = true;
170 
171     SkDynamicMemoryWStream buffer;
172     auto doc = SkPDF::MakeDocument(&buffer, pdfMetadata);
173     doc->beginPage(64, 64)->drawColor(SK_ColorRED);
174     doc->close();
175     sk_sp<SkData> data(buffer.detachAsData());
176 
177     static const char* expectations[] = {
178         "sRGB IEC61966-2.1",
179         "<dc:title><rdf:Alt><rdf:li xml:lang=\"x-default\">test document",
180         "<xmp:CreateDate>1999-12-31T23:59:59+00:00</xmp:CreateDate>",
181         "/Subtype /XML",
182         "/CreationDate (D:19991231235959+00'00')>>",
183     };
184     for (const char* expectation : expectations) {
185         if (!contains(data->bytes(), data->size(), expectation)) {
186             ERRORF(r, "PDFA expectation missing: '%s'.", expectation);
187         }
188     }
189     pdfMetadata.fProducer = "phoney library";
190     pdfMetadata.fPDFA = true;
191     doc = SkPDF::MakeDocument(&buffer, pdfMetadata);
192     doc->beginPage(64, 64)->drawColor(SK_ColorRED);
193     doc->close();
194     data = buffer.detachAsData();
195 
196     static const char* moreExpectations[] = {
197         "/Producer (phoney library)",
198         "<pdf:Producer>phoney library</pdf:Producer>",
199     };
200     for (const char* expectation : moreExpectations) {
201         if (!contains(data->bytes(), data->size(), expectation)) {
202             ERRORF(r, "PDFA expectation missing: '%s'.", expectation);
203         }
204     }
205 }
206 
207 
DEF_TEST(SkPDF_unicode_metadata,r)208 DEF_TEST(SkPDF_unicode_metadata, r) {
209     REQUIRE_PDF_DOCUMENT(SkPDF_unicode_metadata, r);
210     SkPDF::Metadata pdfMetadata;
211     pdfMetadata.fTitle   = "���������� ����������"; // Out of basic multilingual plane
212     pdfMetadata.fAuthor  = "ABCDE FGHIJ"; // ASCII
213     pdfMetadata.fSubject = "αβγδε ζηθικ"; // inside  basic multilingual plane
214     pdfMetadata.fPDFA = true;
215     SkDynamicMemoryWStream wStream;
216     {
217         auto doc = SkPDF::MakeDocument(&wStream, pdfMetadata);
218         doc->beginPage(612, 792)->drawColor(SK_ColorCYAN);
219     }
220     sk_sp<SkData> data(wStream.detachAsData());
221     static const char* expectations[] = {
222         ("<</Title <FEFFD835DCD0D835DCD1D835DCD2D835DCD3D835DCD40020"
223             "D835DCD5D835DCD6D835DCD7D835DCD8D835DCD9>"),
224          "/Author (ABCDE FGHIJ)",
225          "Subject <FEFF03B103B203B303B403B5002003B603B703B803B903BA>",
226          "/ViewerPreferences",
227          "/DisplayDocTitle true",
228     };
229     for (const char* expectation : expectations) {
230         if (!contains(data->bytes(), data->size(), expectation)) {
231             ERRORF(r, "PDF expectation missing: '%s'.", expectation);
232         }
233     }
234 }
235 
236 // Make sure we excercise the multi-page functionality without problems.
237 // Add this to args.gn to output the PDF to a file:
238 //   extra_cflags = [ "-DSK_PDF_TEST_MULTIPAGE=\"/tmp/skpdf_test_multipage.pdf\"" ]
DEF_TEST(SkPDF_multiple_pages,r)239 DEF_TEST(SkPDF_multiple_pages, r) {
240     REQUIRE_PDF_DOCUMENT(SkPDF_multiple_pages, r);
241     int n = 100;
242 #ifdef SK_PDF_TEST_MULTIPAGE
243     SkFILEWStream wStream(SK_PDF_TEST_MULTIPAGE);
244 #else
245     SkDynamicMemoryWStream wStream;
246 #endif
247     auto doc = SkPDF::MakeDocument(&wStream);
248     for (int i = 0; i < n; ++i) {
249         doc->beginPage(612, 792)->drawColor(
250                 SkColorSetARGB(0xFF, 0x00, (uint8_t)(255.0f * i / (n - 1)), 0x00));
251     }
252 }
253 
254 // Test to make sure that jobs launched by PDF backend don't cause a segfault
255 // after calling abort().
DEF_TEST(SkPDF_abort_jobs,rep)256 DEF_TEST(SkPDF_abort_jobs, rep) {
257     REQUIRE_PDF_DOCUMENT(SkPDF_abort_jobs, rep);
258     SkBitmap b;
259     b.allocN32Pixels(612, 792);
260     b.eraseColor(0x4F9643A0);
261     SkPDF::Metadata metadata;
262     std::unique_ptr<SkExecutor> executor = SkExecutor::MakeFIFOThreadPool();
263     metadata.fExecutor = executor.get();
264     SkNullWStream dst;
265     auto doc = SkPDF::MakeDocument(&dst, metadata);
266     doc->beginPage(612, 792)->drawImage(b.asImage(), 0, 0);
267     doc->abort();
268 }
269 
270