1 /*
2  * Copyright 2016 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 
8 #include "Benchmark.h"
9 
10 #include "Resources.h"
11 #include "SkAutoPixmapStorage.h"
12 #include "SkData.h"
13 #include "SkExecutor.h"
14 #include "SkFloatToDecimal.h"
15 #include "SkGradientShader.h"
16 #include "SkImage.h"
17 #include "SkPDFUnion.h"
18 #include "SkPixmap.h"
19 #include "SkRandom.h"
20 #include "SkStream.h"
21 #include "SkTo.h"
22 
23 namespace {
24 struct WStreamWriteTextBenchmark : public Benchmark {
25     std::unique_ptr<SkWStream> fWStream;
WStreamWriteTextBenchmark__anon0b7e5f620111::WStreamWriteTextBenchmark26     WStreamWriteTextBenchmark() : fWStream(new SkNullWStream) {}
onGetName__anon0b7e5f620111::WStreamWriteTextBenchmark27     const char* onGetName() override { return "WStreamWriteText"; }
isSuitableFor__anon0b7e5f620111::WStreamWriteTextBenchmark28     bool isSuitableFor(Backend backend) override {
29         return backend == kNonRendering_Backend;
30     }
onDraw__anon0b7e5f620111::WStreamWriteTextBenchmark31     void onDraw(int loops, SkCanvas*) override {
32         while (loops-- > 0) {
33             for (int i = 1000; i-- > 0;) {
34                 fWStream->writeText("HELLO SKIA!\n");
35             }
36         }
37     }
38 };
39 }  // namespace
40 
41 DEF_BENCH(return new WStreamWriteTextBenchmark;)
42 
43 // Test speed of SkFloatToDecimal for typical floats that
44 // might be found in a PDF document.
45 struct PDFScalarBench : public Benchmark {
PDFScalarBenchPDFScalarBench46     PDFScalarBench(const char* n, float (*f)(SkRandom*)) : fName(n), fNextFloat(f) {}
47     const char* fName;
48     float (*fNextFloat)(SkRandom*);
isSuitableForPDFScalarBench49     bool isSuitableFor(Backend b) override {
50         return b == kNonRendering_Backend;
51     }
onGetNamePDFScalarBench52     const char* onGetName() override { return fName; }
onDrawPDFScalarBench53     void onDraw(int loops, SkCanvas*) override {
54         SkRandom random;
55         char dst[kMaximumSkFloatToDecimalLength];
56         while (loops-- > 0) {
57             auto f = fNextFloat(&random);
58             (void)SkFloatToDecimal(f, dst);
59         }
60     }
61 };
62 
next_common(SkRandom * random)63 float next_common(SkRandom* random) {
64     return random->nextRangeF(-500.0f, 1500.0f);
65 }
next_any(SkRandom * random)66 float next_any(SkRandom* random) {
67     union { uint32_t u; float f; };
68     u = random->nextU();
69     static_assert(sizeof(float) == sizeof(uint32_t), "");
70     return f;
71 }
72 
73 DEF_BENCH(return new PDFScalarBench("PDFScalar_common", next_common);)
74 DEF_BENCH(return new PDFScalarBench("PDFScalar_random", next_any);)
75 
76 #ifdef SK_SUPPORT_PDF
77 
78 #include "SkPDFBitmap.h"
79 #include "SkPDFDocumentPriv.h"
80 #include "SkPDFShader.h"
81 #include "SkPDFUtils.h"
82 
83 namespace {
84 class PDFImageBench : public Benchmark {
85 public:
PDFImageBench()86     PDFImageBench() {}
~PDFImageBench()87     ~PDFImageBench() override {}
88 
89 protected:
onGetName()90     const char* onGetName() override { return "PDFImage"; }
isSuitableFor(Backend backend)91     bool isSuitableFor(Backend backend) override {
92         return backend == kNonRendering_Backend;
93     }
onDelayedSetup()94     void onDelayedSetup() override {
95         sk_sp<SkImage> img(GetResourceAsImage("images/color_wheel.png"));
96         if (img) {
97             // force decoding, throw away reference to encoded data.
98             SkAutoPixmapStorage pixmap;
99             pixmap.alloc(SkImageInfo::MakeN32Premul(img->dimensions()));
100             if (img->readPixels(pixmap, 0, 0)) {
101                 fImage = SkImage::MakeRasterCopy(pixmap);
102             }
103         }
104     }
onDraw(int loops,SkCanvas *)105     void onDraw(int loops, SkCanvas*) override {
106         if (!fImage) {
107             return;
108         }
109         while (loops-- > 0) {
110             SkNullWStream nullStream;
111             SkPDFDocument doc(&nullStream, SkPDF::Metadata());
112             doc.beginPage(256, 256);
113             (void)SkPDFSerializeImage(fImage.get(), &doc);
114         }
115     }
116 
117 private:
118     sk_sp<SkImage> fImage;
119 };
120 
121 class PDFJpegImageBench : public Benchmark {
122 public:
PDFJpegImageBench()123     PDFJpegImageBench() {}
~PDFJpegImageBench()124     ~PDFJpegImageBench() override {}
125 
126 protected:
onGetName()127     const char* onGetName() override { return "PDFJpegImage"; }
isSuitableFor(Backend backend)128     bool isSuitableFor(Backend backend) override {
129         return backend == kNonRendering_Backend;
130     }
onDelayedSetup()131     void onDelayedSetup() override {
132         sk_sp<SkImage> img(GetResourceAsImage("images/mandrill_512_q075.jpg"));
133         if (!img) { return; }
134         sk_sp<SkData> encoded = img->refEncodedData();
135         SkASSERT(encoded);
136         if (!encoded) { return; }
137         fImage = img;
138     }
onDraw(int loops,SkCanvas *)139     void onDraw(int loops, SkCanvas*) override {
140         if (!fImage) {
141             SkDEBUGFAIL("");
142             return;
143         }
144         while (loops-- > 0) {
145             SkNullWStream nullStream;
146             SkPDFDocument doc(&nullStream, SkPDF::Metadata());
147             doc.beginPage(256, 256);
148             (void)SkPDFSerializeImage(fImage.get(), &doc);
149         }
150     }
151 
152 private:
153     sk_sp<SkImage> fImage;
154 };
155 
156 /** Test calling DEFLATE on a 78k PDF command stream. Used for measuring
157     alternate zlib settings, usage, and library versions. */
158 class PDFCompressionBench : public Benchmark {
159 public:
PDFCompressionBench()160     PDFCompressionBench() {}
~PDFCompressionBench()161     ~PDFCompressionBench() override {}
162 
163 protected:
onGetName()164     const char* onGetName() override { return "PDFCompression"; }
isSuitableFor(Backend backend)165     bool isSuitableFor(Backend backend) override {
166         return backend == kNonRendering_Backend;
167     }
onDelayedSetup()168     void onDelayedSetup() override {
169         fAsset = GetResourceAsStream("pdf_command_stream.txt");
170     }
onDraw(int loops,SkCanvas *)171     void onDraw(int loops, SkCanvas*) override {
172         SkASSERT(fAsset);
173         if (!fAsset) { return; }
174         while (loops-- > 0) {
175             SkNullWStream wStream;
176             SkPDFDocument doc(&wStream, SkPDF::Metadata());
177             doc.beginPage(256, 256);
178             (void)SkPDFStreamOut(nullptr, fAsset->duplicate(), &doc, true);
179        }
180     }
181 
182 private:
183     std::unique_ptr<SkStreamAsset> fAsset;
184 };
185 
186 struct PDFColorComponentBench : public Benchmark {
isSuitableFor__anon0b7e5f620311::PDFColorComponentBench187     bool isSuitableFor(Backend b) override {
188         return b == kNonRendering_Backend;
189     }
onGetName__anon0b7e5f620311::PDFColorComponentBench190     const char* onGetName() override { return "PDFColorComponent"; }
onDraw__anon0b7e5f620311::PDFColorComponentBench191     void onDraw(int loops, SkCanvas*) override {
192         char dst[5];
193         while (loops-- > 0) {
194             for (int i = 0; i < 256; ++i) {
195                 (void)SkPDFUtils::ColorToDecimal(SkToU8(i), dst);
196             }
197         }
198     }
199 };
200 
201 struct PDFShaderBench : public Benchmark {
202     sk_sp<SkShader> fShader;
onGetName__anon0b7e5f620311::PDFShaderBench203     const char* onGetName() final { return "PDFShader"; }
isSuitableFor__anon0b7e5f620311::PDFShaderBench204     bool isSuitableFor(Backend b) final { return b == kNonRendering_Backend; }
onDelayedSetup__anon0b7e5f620311::PDFShaderBench205     void onDelayedSetup() final {
206         const SkPoint pts[2] = {{0.0f, 0.0f}, {100.0f, 100.0f}};
207         const SkColor colors[] = {
208             SK_ColorRED, SK_ColorGREEN, SK_ColorBLUE,
209             SK_ColorWHITE, SK_ColorBLACK,
210         };
211         fShader = SkGradientShader::MakeLinear(
212                 pts, colors, nullptr, SK_ARRAY_COUNT(colors),
213                 SkShader::kClamp_TileMode);
214     }
onDraw__anon0b7e5f620311::PDFShaderBench215     void onDraw(int loops, SkCanvas*) final {
216         SkASSERT(fShader);
217         while (loops-- > 0) {
218             SkNullWStream nullStream;
219             SkPDFDocument doc(&nullStream, SkPDF::Metadata());
220             doc.beginPage(256, 256);
221             (void) SkPDFMakeShader(&doc, fShader.get(), SkMatrix::I(),
222                                    {0, 0, 400, 400}, SK_ColorBLACK);
223         }
224     }
225 };
226 
227 struct WritePDFTextBenchmark : public Benchmark {
228     std::unique_ptr<SkWStream> fWStream;
WritePDFTextBenchmark__anon0b7e5f620311::WritePDFTextBenchmark229     WritePDFTextBenchmark() : fWStream(new SkNullWStream) {}
onGetName__anon0b7e5f620311::WritePDFTextBenchmark230     const char* onGetName() override { return "WritePDFText"; }
isSuitableFor__anon0b7e5f620311::WritePDFTextBenchmark231     bool isSuitableFor(Backend backend) override {
232         return backend == kNonRendering_Backend;
233     }
onDraw__anon0b7e5f620311::WritePDFTextBenchmark234     void onDraw(int loops, SkCanvas*) override {
235         static const char kHello[] = "HELLO SKIA!\n";
236         static const char kBinary[] = "\001\002\003\004\005\006";
237         while (loops-- > 0) {
238             for (int i = 1000; i-- > 0;) {
239                 SkPDFWriteString(fWStream.get(), kHello, strlen(kHello));
240                 SkPDFWriteString(fWStream.get(), kBinary, strlen(kBinary));
241             }
242         }
243     }
244 };
245 
246 }  // namespace
247 DEF_BENCH(return new PDFImageBench;)
248 DEF_BENCH(return new PDFJpegImageBench;)
249 DEF_BENCH(return new PDFCompressionBench;)
250 DEF_BENCH(return new PDFColorComponentBench;)
251 DEF_BENCH(return new PDFShaderBench;)
252 DEF_BENCH(return new WritePDFTextBenchmark;)
253 
254 #ifdef SK_PDF_ENABLE_SLOW_TESTS
255 #include "SkExecutor.h"
256 namespace {
big_pdf_test(SkDocument * doc,const SkBitmap & background)257 void big_pdf_test(SkDocument* doc, const SkBitmap& background) {
258     static const char* kText[] = {
259         "Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do",
260         "eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad",
261         "minim veniam, quis nostrud exercitation ullamco laboris nisi ut",
262         "aliquip ex ea commodo consequat. Duis aute irure dolor in",
263         "reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla",
264         "pariatur. Excepteur sint occaecat cupidatat non proident, sunt in",
265         "culpa qui officia deserunt mollit anim id est laborum.",
266         "",
267         "Sed ut perspiciatis, unde omnis iste natus error sit voluptatem",
268         "accusantium doloremque laudantium, totam rem aperiam eaque ipsa, quae",
269         "ab illo inventore veritatis et quasi architecto beatae vitae dicta",
270         "sunt, explicabo. Nemo enim ipsam voluptatem, quia voluptas sit,",
271         "aspernatur aut odit aut fugit, sed quia consequuntur magni dolores",
272         "eos, qui ratione voluptatem sequi nesciunt, neque porro quisquam est,",
273         "qui dolorem ipsum, quia dolor sit amet consectetur adipiscing velit,",
274         "sed quia non numquam do eius modi tempora incididunt, ut labore et",
275         "dolore magnam aliquam quaerat voluptatem. Ut enim ad minima veniam,",
276         "quis nostrum exercitationem ullam corporis suscipit laboriosam, nisi",
277         "ut aliquid ex ea commodi consequatur? Quis autem vel eum iure",
278         "reprehenderit, qui in ea voluptate velit esse, quam nihil molestiae",
279         "consequatur, vel illum, qui dolorem eum fugiat, quo voluptas nulla",
280         "pariatur?",
281         "",
282         "At vero eos et accusamus et iusto odio dignissimos ducimus, qui",
283         "blanditiis praesentium voluptatum deleniti atque corrupti, quos",
284         "dolores et quas molestias excepturi sint, obcaecati cupiditate non",
285         "provident, similique sunt in culpa, qui officia deserunt mollitia",
286         "animi, id est laborum et dolorum fuga. Et harum quidem rerum facilis",
287         "est et expedita distinctio. Nam libero tempore, cum soluta nobis est",
288         "eligendi optio, cumque nihil impedit, quo minus id, quod maxime",
289         "placeat, facere possimus, omnis voluptas assumenda est, omnis dolor",
290         "repellendus. Temporibus autem quibusdam et aut officiis debitis aut",
291         "rerum necessitatibus saepe eveniet, ut et voluptates repudiandae sint",
292         "et molestiae non recusandae. Itaque earum rerum hic tenetur a sapiente",
293         "delectus, ut aut reiciendis voluptatibus maiores alias consequatur aut",
294         "perferendis doloribus asperiores repellat",
295         "",
296         "Sed ut perspiciatis, unde omnis iste natus error sit voluptatem",
297         "accusantium doloremque laudantium, totam rem aperiam eaque ipsa, quae",
298         "ab illo inventore veritatis et quasi architecto beatae vitae dicta",
299         "sunt, explicabo. Nemo enim ipsam voluptatem, quia voluptas sit,",
300         "aspernatur aut odit aut fugit, sed quia consequuntur magni dolores",
301         "eos, qui ratione voluptatem sequi nesciunt, neque porro quisquam est,",
302         "qui dolorem ipsum, quia dolor sit amet consectetur adipiscing velit,",
303         "sed quia non numquam do eius modi tempora incididunt, ut labore et",
304         "dolore magnam aliquam quaerat voluptatem. Ut enim ad minima veniam,",
305         "quis nostrum exercitationem ullam corporis suscipit laboriosam, nisi",
306         "ut aliquid ex ea commodi consequatur? Quis autem vel eum iure",
307         "reprehenderit, qui in ea voluptate velit esse, quam nihil molestiae",
308         "consequatur, vel illum, qui dolorem eum fugiat, quo voluptas nulla",
309         "pariatur?",
310         "",
311     };
312     SkCanvas* canvas = nullptr;
313     float x = 36;
314     float y = 36;
315     constexpr size_t kLineCount = SK_ARRAY_COUNT(kText);
316     constexpr int kLoopCount = 200;
317     SkFont font;
318     SkPaint paint;
319     for (int loop = 0; loop < kLoopCount; ++loop) {
320         for (size_t line = 0; line < kLineCount; ++line) {
321             y += font.getSpacing();
322             if (!canvas || y > 792 - 36) {
323                 y = 36 + font.getSpacing();
324                 canvas = doc->beginPage(612, 792);
325                 background.notifyPixelsChanged();
326                 canvas->drawBitmap(background, 0, 0);
327             }
328             canvas->drawString(kText[line], x, y, font, paint);
329         }
330     }
331 }
332 
make_background()333 SkBitmap make_background() {
334     SkBitmap background;
335     SkBitmap bitmap;
336     bitmap.allocN32Pixels(32, 32);
337     bitmap.eraseColor(SK_ColorWHITE);
338     SkCanvas tmp(bitmap);
339     SkPaint gray;
340     gray.setColor(SkColorSetARGB(0xFF, 0xEE, 0xEE, 0xEE));
341     tmp.drawRect({0,0,16,16}, gray);
342     tmp.drawRect({16,16,32,32}, gray);
343     SkPaint shader;
344     shader.setShader(
345             SkShader::MakeBitmapShader(
346                 bitmap, SkShader::kRepeat_TileMode, SkShader::kRepeat_TileMode));
347     background.allocN32Pixels(612, 792);
348     SkCanvas tmp2(background);
349     tmp2.drawPaint(shader);
350     return background;
351 }
352 
353 struct PDFBigDocBench : public Benchmark {
354     bool fFast;
355     SkBitmap fBackground;
356     std::unique_ptr<SkExecutor> fExecutor;
PDFBigDocBench__anon0b7e5f620411::PDFBigDocBench357     PDFBigDocBench(bool fast) : fFast(fast) {}
onDelayedSetup__anon0b7e5f620411::PDFBigDocBench358     void onDelayedSetup() override {
359         fBackground = make_background();
360         fExecutor = fFast ? SkExecutor::MakeFIFOThreadPool() : nullptr;
361     }
onGetName__anon0b7e5f620411::PDFBigDocBench362     const char* onGetName() override {
363         static const char kNameFast[] = "PDFBigDocBench_fast";
364         static const char kNameSlow[] = "PDFBigDocBench_slow";
365         return fFast ? kNameFast : kNameSlow;
366     }
isSuitableFor__anon0b7e5f620411::PDFBigDocBench367     bool isSuitableFor(Backend backend) override { return backend == kNonRendering_Backend; }
onDraw__anon0b7e5f620411::PDFBigDocBench368     void onDraw(int loops, SkCanvas*) override {
369         while (loops-- > 0) {
370             #ifdef SK_PDF_TEST_BIGDOCBENCH_OUTPUT
371             SkFILEWStream wStream("/tmp/big_pdf.pdf");
372             #else
373             SkNullWStream wStream;
374             #endif
375             SkPDF::Metadata metadata;
376             metadata.fExecutor = fExecutor.get();
377             auto doc = SkPDF::MakeDocument(&wStream, metadata);
378             big_pdf_test(doc.get(), fBackground);
379         }
380     }
381 };
382 }  // namespace
383 DEF_BENCH(return new PDFBigDocBench(false);)
384 DEF_BENCH(return new PDFBigDocBench(true);)
385 #endif
386 
387 #endif // SK_SUPPORT_PDF
388