1 /*
2 * Copyright 2011 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 "bench/Benchmark.h"
9 #include "include/core/SkBitmap.h"
10 #include "include/core/SkCanvas.h"
11 #include "include/core/SkColorPriv.h"
12 #include "include/core/SkPaint.h"
13 #include "include/core/SkPath.h"
14 #include "include/core/SkShader.h"
15 #include "include/core/SkString.h"
16 #include "include/private/SkTArray.h"
17 #include "include/utils/SkRandom.h"
18
19 #include "src/core/SkDraw.h"
20
21 enum Flags {
22 kStroke_Flag = 1 << 0,
23 kBig_Flag = 1 << 1
24 };
25
26 #define FLAGS00 Flags(0)
27 #define FLAGS01 Flags(kStroke_Flag)
28 #define FLAGS10 Flags(kBig_Flag)
29 #define FLAGS11 Flags(kStroke_Flag | kBig_Flag)
30
31 class PathBench : public Benchmark {
32 SkPaint fPaint;
33 SkString fName;
34 Flags fFlags;
35 public:
PathBench(Flags flags)36 PathBench(Flags flags) : fFlags(flags) {
37 fPaint.setStyle(flags & kStroke_Flag ? SkPaint::kStroke_Style :
38 SkPaint::kFill_Style);
39 fPaint.setStrokeWidth(SkIntToScalar(5));
40 fPaint.setStrokeJoin(SkPaint::kBevel_Join);
41 }
42
43 virtual void appendName(SkString*) = 0;
44 virtual void makePath(SkPath*) = 0;
complexity()45 virtual int complexity() { return 0; }
46
47 protected:
onGetName()48 const char* onGetName() override {
49 fName.printf("path_%s_%s_",
50 fFlags & kStroke_Flag ? "stroke" : "fill",
51 fFlags & kBig_Flag ? "big" : "small");
52 this->appendName(&fName);
53 return fName.c_str();
54 }
55
onDraw(int loops,SkCanvas * canvas)56 void onDraw(int loops, SkCanvas* canvas) override {
57 SkPaint paint(fPaint);
58 this->setupPaint(&paint);
59
60 SkPath path;
61 this->makePath(&path);
62 if (fFlags & kBig_Flag) {
63 const SkMatrix m = SkMatrix::Scale(10, 10);
64 path.transform(m);
65 }
66
67 for (int i = 0; i < loops; i++) {
68 canvas->drawPath(path, paint);
69 }
70 }
71
72 private:
73 using INHERITED = Benchmark;
74 };
75
76 class TrianglePathBench : public PathBench {
77 public:
TrianglePathBench(Flags flags)78 TrianglePathBench(Flags flags) : INHERITED(flags) {}
79
appendName(SkString * name)80 void appendName(SkString* name) override {
81 name->append("triangle");
82 }
makePath(SkPath * path)83 void makePath(SkPath* path) override {
84 static const int gCoord[] = {
85 10, 10, 15, 5, 20, 20
86 };
87 path->moveTo(SkIntToScalar(gCoord[0]), SkIntToScalar(gCoord[1]));
88 path->lineTo(SkIntToScalar(gCoord[2]), SkIntToScalar(gCoord[3]));
89 path->lineTo(SkIntToScalar(gCoord[4]), SkIntToScalar(gCoord[5]));
90 path->close();
91 }
92 private:
93 using INHERITED = PathBench;
94 };
95
96 class RectPathBench : public PathBench {
97 public:
RectPathBench(Flags flags)98 RectPathBench(Flags flags) : INHERITED(flags) {}
99
appendName(SkString * name)100 void appendName(SkString* name) override {
101 name->append("rect");
102 }
makePath(SkPath * path)103 void makePath(SkPath* path) override {
104 SkRect r = { 10, 10, 20, 20 };
105 path->addRect(r);
106 }
107 private:
108 using INHERITED = PathBench;
109 };
110
111 class RotatedRectBench : public PathBench {
112 public:
RotatedRectBench(Flags flags,bool aa,int degrees)113 RotatedRectBench(Flags flags, bool aa, int degrees) : INHERITED(flags) {
114 fAA = aa;
115 fDegrees = degrees;
116 }
117
appendName(SkString * name)118 void appendName(SkString* name) override {
119 SkString suffix;
120 suffix.printf("rotated_rect_%s_%d", fAA ? "aa" : "noaa", fDegrees);
121 name->append(suffix);
122 }
123
makePath(SkPath * path)124 void makePath(SkPath* path) override {
125 SkRect r = { 10, 10, 20, 20 };
126 path->addRect(r);
127 SkMatrix rotateMatrix;
128 rotateMatrix.setRotate((SkScalar)fDegrees);
129 path->transform(rotateMatrix);
130 }
131
setupPaint(SkPaint * paint)132 void setupPaint(SkPaint* paint) override {
133 PathBench::setupPaint(paint);
134 paint->setAntiAlias(fAA);
135 }
136 private:
137 using INHERITED = PathBench;
138 int fDegrees;
139 bool fAA;
140 };
141
142 class OvalPathBench : public PathBench {
143 public:
OvalPathBench(Flags flags)144 OvalPathBench(Flags flags) : INHERITED(flags) {}
145
appendName(SkString * name)146 void appendName(SkString* name) override {
147 name->append("oval");
148 }
makePath(SkPath * path)149 void makePath(SkPath* path) override {
150 SkRect r = { 10, 10, 23, 20 };
151 path->addOval(r);
152 }
153 private:
154 using INHERITED = PathBench;
155 };
156
157 class CirclePathBench: public PathBench {
158 public:
CirclePathBench(Flags flags)159 CirclePathBench(Flags flags) : INHERITED(flags) {}
160
appendName(SkString * name)161 void appendName(SkString* name) override {
162 name->append("circle");
163 }
makePath(SkPath * path)164 void makePath(SkPath* path) override {
165 path->addCircle(SkIntToScalar(20), SkIntToScalar(20),
166 SkIntToScalar(10));
167 }
168 private:
169 using INHERITED = PathBench;
170 };
171
172 class NonAACirclePathBench: public CirclePathBench {
173 public:
NonAACirclePathBench(Flags flags)174 NonAACirclePathBench(Flags flags) : INHERITED(flags) {}
175
appendName(SkString * name)176 void appendName(SkString* name) override {
177 name->append("nonaacircle");
178 }
179
setupPaint(SkPaint * paint)180 void setupPaint(SkPaint* paint) override {
181 CirclePathBench::setupPaint(paint);
182 paint->setAntiAlias(false);
183 }
184
185 private:
186 using INHERITED = CirclePathBench;
187 };
188
189 // Test max speedup of Analytic AA for concave paths
190 class AAAConcavePathBench : public PathBench {
191 public:
AAAConcavePathBench(Flags flags)192 AAAConcavePathBench(Flags flags) : INHERITED(flags) {}
193
appendName(SkString * name)194 void appendName(SkString* name) override {
195 name->append("concave_aaa");
196 }
197
makePath(SkPath * path)198 void makePath(SkPath* path) override {
199 path->moveTo(10, 10);
200 path->lineTo(15, 10);
201 path->lineTo(15, 5);
202 path->lineTo(40, 40);
203 path->close();
204 }
205
206 private:
207 using INHERITED = PathBench;
208 };
209
210 // Test max speedup of Analytic AA for convex paths
211 class AAAConvexPathBench : public PathBench {
212 public:
AAAConvexPathBench(Flags flags)213 AAAConvexPathBench(Flags flags) : INHERITED(flags) {}
214
appendName(SkString * name)215 void appendName(SkString* name) override {
216 name->append("convex_aaa");
217 }
218
makePath(SkPath * path)219 void makePath(SkPath* path) override {
220 path->moveTo(10, 10);
221 path->lineTo(15, 10);
222 path->lineTo(40, 50);
223 path->close();
224 }
225
226 private:
227 using INHERITED = PathBench;
228 };
229
230 class SawToothPathBench : public PathBench {
231 public:
SawToothPathBench(Flags flags)232 SawToothPathBench(Flags flags) : INHERITED(flags) {}
233
appendName(SkString * name)234 void appendName(SkString* name) override {
235 name->append("sawtooth");
236 }
makePath(SkPath * path)237 void makePath(SkPath* path) override {
238 SkScalar x = SkIntToScalar(20);
239 SkScalar y = SkIntToScalar(20);
240 const SkScalar x0 = x;
241 const SkScalar dx = SK_Scalar1 * 5;
242 const SkScalar dy = SK_Scalar1 * 10;
243
244 path->moveTo(x, y);
245 for (int i = 0; i < 32; i++) {
246 x += dx;
247 path->lineTo(x, y - dy);
248 x += dx;
249 path->lineTo(x, y + dy);
250 }
251 path->lineTo(x, y + 2 * dy);
252 path->lineTo(x0, y + 2 * dy);
253 path->close();
254 }
complexity()255 int complexity() override { return 1; }
256 private:
257 using INHERITED = PathBench;
258 };
259
260 class LongCurvedPathBench : public PathBench {
261 public:
LongCurvedPathBench(Flags flags)262 LongCurvedPathBench(Flags flags) : INHERITED(flags) {}
263
appendName(SkString * name)264 void appendName(SkString* name) override {
265 name->append("long_curved");
266 }
makePath(SkPath * path)267 void makePath(SkPath* path) override {
268 SkRandom rand (12);
269 int i;
270 for (i = 0; i < 100; i++) {
271 path->quadTo(rand.nextUScalar1() * 640, rand.nextUScalar1() * 480,
272 rand.nextUScalar1() * 640, rand.nextUScalar1() * 480);
273 }
274 path->close();
275 }
complexity()276 int complexity() override { return 2; }
277 private:
278 using INHERITED = PathBench;
279 };
280
281 class LongLinePathBench : public PathBench {
282 public:
LongLinePathBench(Flags flags)283 LongLinePathBench(Flags flags) : INHERITED(flags) {}
284
appendName(SkString * name)285 void appendName(SkString* name) override {
286 name->append("long_line");
287 }
makePath(SkPath * path)288 void makePath(SkPath* path) override {
289 SkRandom rand;
290 path->moveTo(rand.nextUScalar1() * 640, rand.nextUScalar1() * 480);
291 for (size_t i = 1; i < 100; i++) {
292 path->lineTo(rand.nextUScalar1() * 640, rand.nextUScalar1() * 480);
293 }
294 }
complexity()295 int complexity() override { return 2; }
296 private:
297 using INHERITED = PathBench;
298 };
299
300 class RandomPathBench : public Benchmark {
301 public:
isSuitableFor(Backend backend)302 bool isSuitableFor(Backend backend) override {
303 return backend == kNonRendering_Backend;
304 }
305
306 protected:
createData(int minVerbs,int maxVerbs,bool allowMoves=true,SkRect * bounds=nullptr)307 void createData(int minVerbs,
308 int maxVerbs,
309 bool allowMoves = true,
310 SkRect* bounds = nullptr) {
311 SkRect tempBounds;
312 if (nullptr == bounds) {
313 tempBounds.setXYWH(0, 0, SK_Scalar1, SK_Scalar1);
314 bounds = &tempBounds;
315 }
316 fVerbCnts.reset(kNumVerbCnts);
317 for (int i = 0; i < kNumVerbCnts; ++i) {
318 fVerbCnts[i] = fRandom.nextRangeU(minVerbs, maxVerbs + 1);
319 }
320 fVerbs.reset(kNumVerbs);
321 for (int i = 0; i < kNumVerbs; ++i) {
322 do {
323 fVerbs[i] = static_cast<SkPath::Verb>(fRandom.nextULessThan(SkPath::kDone_Verb));
324 } while (!allowMoves && SkPath::kMove_Verb == fVerbs[i]);
325 }
326 fPoints.reset(kNumPoints);
327 for (int i = 0; i < kNumPoints; ++i) {
328 fPoints[i].set(fRandom.nextRangeScalar(bounds->fLeft, bounds->fRight),
329 fRandom.nextRangeScalar(bounds->fTop, bounds->fBottom));
330 }
331 this->restartMakingPaths();
332 }
333
restartMakingPaths()334 void restartMakingPaths() {
335 fCurrPath = 0;
336 fCurrVerb = 0;
337 fCurrPoint = 0;
338 }
339
makePath(SkPath * path)340 void makePath(SkPath* path) {
341 int vCount = fVerbCnts[(fCurrPath++) & (kNumVerbCnts - 1)];
342 for (int v = 0; v < vCount; ++v) {
343 int verb = fVerbs[(fCurrVerb++) & (kNumVerbs - 1)];
344 switch (verb) {
345 case SkPath::kMove_Verb:
346 path->moveTo(fPoints[(fCurrPoint++) & (kNumPoints - 1)]);
347 break;
348 case SkPath::kLine_Verb:
349 path->lineTo(fPoints[(fCurrPoint++) & (kNumPoints - 1)]);
350 break;
351 case SkPath::kQuad_Verb:
352 path->quadTo(fPoints[(fCurrPoint + 0) & (kNumPoints - 1)],
353 fPoints[(fCurrPoint + 1) & (kNumPoints - 1)]);
354 fCurrPoint += 2;
355 break;
356 case SkPath::kConic_Verb:
357 path->conicTo(fPoints[(fCurrPoint + 0) & (kNumPoints - 1)],
358 fPoints[(fCurrPoint + 1) & (kNumPoints - 1)],
359 SK_ScalarHalf);
360 fCurrPoint += 2;
361 break;
362 case SkPath::kCubic_Verb:
363 path->cubicTo(fPoints[(fCurrPoint + 0) & (kNumPoints - 1)],
364 fPoints[(fCurrPoint + 1) & (kNumPoints - 1)],
365 fPoints[(fCurrPoint + 2) & (kNumPoints - 1)]);
366 fCurrPoint += 3;
367 break;
368 case SkPath::kClose_Verb:
369 path->close();
370 break;
371 default:
372 SkDEBUGFAIL("Unexpected path verb");
373 break;
374 }
375 }
376 }
377
finishedMakingPaths()378 void finishedMakingPaths() {
379 fVerbCnts.reset(0);
380 fVerbs.reset(0);
381 fPoints.reset(0);
382 }
383
384 private:
385 enum {
386 // these should all be pow 2
387 kNumVerbCnts = 1 << 5,
388 kNumVerbs = 1 << 5,
389 kNumPoints = 1 << 5,
390 };
391 SkAutoTArray<int> fVerbCnts;
392 SkAutoTArray<SkPath::Verb> fVerbs;
393 SkAutoTArray<SkPoint> fPoints;
394 int fCurrPath;
395 int fCurrVerb;
396 int fCurrPoint;
397 SkRandom fRandom;
398 using INHERITED = Benchmark;
399 };
400
401 class PathCreateBench : public RandomPathBench {
402 public:
PathCreateBench()403 PathCreateBench() {
404 }
405
406 protected:
onGetName()407 const char* onGetName() override {
408 return "path_create";
409 }
410
onDelayedSetup()411 void onDelayedSetup() override {
412 this->createData(10, 100);
413 }
414
onDraw(int loops,SkCanvas *)415 void onDraw(int loops, SkCanvas*) override {
416 for (int i = 0; i < loops; ++i) {
417 if (i % 1000 == 0) {
418 fPath.reset(); // PathRef memory can grow without bound otherwise.
419 }
420 this->makePath(&fPath);
421 }
422 this->restartMakingPaths();
423 }
424
425 private:
426 SkPath fPath;
427
428 using INHERITED = RandomPathBench;
429 };
430
431 class PathCopyBench : public RandomPathBench {
432 public:
PathCopyBench()433 PathCopyBench() {
434 }
435
436 protected:
onGetName()437 const char* onGetName() override {
438 return "path_copy";
439 }
onDelayedSetup()440 void onDelayedSetup() override {
441 this->createData(10, 100);
442 fPaths.reset(kPathCnt);
443 fCopies.reset(kPathCnt);
444 for (int i = 0; i < kPathCnt; ++i) {
445 this->makePath(&fPaths[i]);
446 }
447 this->finishedMakingPaths();
448 }
onDraw(int loops,SkCanvas *)449 void onDraw(int loops, SkCanvas*) override {
450 for (int i = 0; i < loops; ++i) {
451 int idx = i & (kPathCnt - 1);
452 fCopies[idx] = fPaths[idx];
453 }
454 }
455
456 private:
457 enum {
458 // must be a pow 2
459 kPathCnt = 1 << 5,
460 };
461 SkAutoTArray<SkPath> fPaths;
462 SkAutoTArray<SkPath> fCopies;
463
464 using INHERITED = RandomPathBench;
465 };
466
467 class PathTransformBench : public RandomPathBench {
468 public:
PathTransformBench(bool inPlace)469 PathTransformBench(bool inPlace) : fInPlace(inPlace) {}
470
471 protected:
onGetName()472 const char* onGetName() override {
473 return fInPlace ? "path_transform_in_place" : "path_transform_copy";
474 }
475
onDelayedSetup()476 void onDelayedSetup() override {
477 fMatrix.setScale(5 * SK_Scalar1, 6 * SK_Scalar1);
478 this->createData(10, 100);
479 fPaths.reset(kPathCnt);
480 for (int i = 0; i < kPathCnt; ++i) {
481 this->makePath(&fPaths[i]);
482 }
483 this->finishedMakingPaths();
484 if (!fInPlace) {
485 fTransformed.reset(kPathCnt);
486 }
487 }
488
onDraw(int loops,SkCanvas *)489 void onDraw(int loops, SkCanvas*) override {
490 if (fInPlace) {
491 for (int i = 0; i < loops; ++i) {
492 fPaths[i & (kPathCnt - 1)].transform(fMatrix);
493 }
494 } else {
495 for (int i = 0; i < loops; ++i) {
496 int idx = i & (kPathCnt - 1);
497 fPaths[idx].transform(fMatrix, &fTransformed[idx]);
498 }
499 }
500 }
501
502 private:
503 enum {
504 // must be a pow 2
505 kPathCnt = 1 << 5,
506 };
507 SkAutoTArray<SkPath> fPaths;
508 SkAutoTArray<SkPath> fTransformed;
509
510 SkMatrix fMatrix;
511 bool fInPlace;
512 using INHERITED = RandomPathBench;
513 };
514
515 class PathEqualityBench : public RandomPathBench {
516 public:
PathEqualityBench()517 PathEqualityBench() { }
518
519 protected:
onGetName()520 const char* onGetName() override {
521 return "path_equality_50%";
522 }
523
onDelayedSetup()524 void onDelayedSetup() override {
525 fParity = 0;
526 this->createData(10, 100);
527 fPaths.reset(kPathCnt);
528 fCopies.reset(kPathCnt);
529 for (int i = 0; i < kPathCnt; ++i) {
530 this->makePath(&fPaths[i]);
531 fCopies[i] = fPaths[i];
532 }
533 this->finishedMakingPaths();
534 }
535
onDraw(int loops,SkCanvas *)536 void onDraw(int loops, SkCanvas*) override {
537 for (int i = 0; i < loops; ++i) {
538 int idx = i & (kPathCnt - 1);
539 fParity ^= (fPaths[idx] == fCopies[idx & ~0x1]);
540 }
541 }
542
543 private:
544 bool fParity; // attempt to keep compiler from optimizing out the ==
545 enum {
546 // must be a pow 2
547 kPathCnt = 1 << 5,
548 };
549 SkAutoTArray<SkPath> fPaths;
550 SkAutoTArray<SkPath> fCopies;
551 using INHERITED = RandomPathBench;
552 };
553
554 class SkBench_AddPathTest : public RandomPathBench {
555 public:
556 enum AddType {
557 kAdd_AddType,
558 kAddTrans_AddType,
559 kAddMatrix_AddType,
560 kReverseAdd_AddType,
561 kReversePathTo_AddType,
562 };
563
SkBench_AddPathTest(AddType type)564 SkBench_AddPathTest(AddType type) : fType(type) {
565 fMatrix.setRotate(60 * SK_Scalar1);
566 }
567
568 protected:
onGetName()569 const char* onGetName() override {
570 switch (fType) {
571 case kAdd_AddType:
572 return "path_add_path";
573 case kAddTrans_AddType:
574 return "path_add_path_trans";
575 case kAddMatrix_AddType:
576 return "path_add_path_matrix";
577 case kReverseAdd_AddType:
578 return "path_reverse_add_path";
579 case kReversePathTo_AddType:
580 return "path_reverse_path_to";
581 default:
582 SkDEBUGFAIL("Bad add type");
583 return "";
584 }
585 }
586
onDelayedSetup()587 void onDelayedSetup() override {
588 // reversePathTo assumes a single contour path.
589 bool allowMoves = kReversePathTo_AddType != fType;
590 this->createData(10, 100, allowMoves);
591 fPaths0.reset(kPathCnt);
592 fPaths1.reset(kPathCnt);
593 for (int i = 0; i < kPathCnt; ++i) {
594 this->makePath(&fPaths0[i]);
595 this->makePath(&fPaths1[i]);
596 }
597 this->finishedMakingPaths();
598 }
599
onDraw(int loops,SkCanvas *)600 void onDraw(int loops, SkCanvas*) override {
601 switch (fType) {
602 case kAdd_AddType:
603 for (int i = 0; i < loops; ++i) {
604 int idx = i & (kPathCnt - 1);
605 SkPath result = fPaths0[idx];
606 result.addPath(fPaths1[idx]);
607 }
608 break;
609 case kAddTrans_AddType:
610 for (int i = 0; i < loops; ++i) {
611 int idx = i & (kPathCnt - 1);
612 SkPath result = fPaths0[idx];
613 result.addPath(fPaths1[idx], 2 * SK_Scalar1, 5 * SK_Scalar1);
614 }
615 break;
616 case kAddMatrix_AddType:
617 for (int i = 0; i < loops; ++i) {
618 int idx = i & (kPathCnt - 1);
619 SkPath result = fPaths0[idx];
620 result.addPath(fPaths1[idx], fMatrix);
621 }
622 break;
623 case kReverseAdd_AddType:
624 for (int i = 0; i < loops; ++i) {
625 int idx = i & (kPathCnt - 1);
626 SkPath result = fPaths0[idx];
627 result.reverseAddPath(fPaths1[idx]);
628 }
629 break;
630 case kReversePathTo_AddType:
631 for (int i = 0; i < loops; ++i) {
632 int idx = i & (kPathCnt - 1);
633 SkPath result = fPaths0[idx];
634 result.reversePathTo(fPaths1[idx]);
635 }
636 break;
637 }
638 }
639
640 private:
641 AddType fType; // or reverseAddPath
642 enum {
643 // must be a pow 2
644 kPathCnt = 1 << 5,
645 };
646 SkAutoTArray<SkPath> fPaths0;
647 SkAutoTArray<SkPath> fPaths1;
648 SkMatrix fMatrix;
649 using INHERITED = RandomPathBench;
650 };
651
652
653 class CirclesBench : public Benchmark {
654 protected:
655 SkString fName;
656 Flags fFlags;
657
658 public:
CirclesBench(Flags flags)659 CirclesBench(Flags flags) : fFlags(flags) {
660 fName.printf("circles_%s", fFlags & kStroke_Flag ? "stroke" : "fill");
661 }
662
663 protected:
onGetName()664 const char* onGetName() override {
665 return fName.c_str();
666 }
667
onDraw(int loops,SkCanvas * canvas)668 void onDraw(int loops, SkCanvas* canvas) override {
669 SkPaint paint;
670
671 paint.setColor(SK_ColorBLACK);
672 paint.setAntiAlias(true);
673 if (fFlags & kStroke_Flag) {
674 paint.setStyle(SkPaint::kStroke_Style);
675 }
676
677 SkRandom rand;
678
679 SkRect r;
680
681 for (int i = 0; i < loops; ++i) {
682 SkScalar radius = rand.nextUScalar1() * 3;
683 r.fLeft = rand.nextUScalar1() * 300;
684 r.fTop = rand.nextUScalar1() * 300;
685 r.fRight = r.fLeft + 2 * radius;
686 r.fBottom = r.fTop + 2 * radius;
687
688 if (fFlags & kStroke_Flag) {
689 paint.setStrokeWidth(rand.nextUScalar1() * 5.0f);
690 }
691
692 SkPath temp;
693
694 // mimic how Chrome does circles
695 temp.arcTo(r, 0, 0, false);
696 temp.addOval(r, SkPathDirection::kCCW);
697 temp.arcTo(r, 360, 0, true);
698 temp.close();
699
700 canvas->drawPath(temp, paint);
701 }
702 }
703
704 private:
705 using INHERITED = Benchmark;
706 };
707
708
709 // Chrome creates its own round rects with each corner possibly being different.
710 // In its "zero radius" incarnation it creates degenerate round rects.
711 // Note: PathTest::test_arb_round_rect_is_convex and
712 // test_arb_zero_rad_round_rect_is_rect perform almost exactly
713 // the same test (but with no drawing)
714 class ArbRoundRectBench : public Benchmark {
715 protected:
716 SkString fName;
717
718 public:
ArbRoundRectBench(bool zeroRad)719 ArbRoundRectBench(bool zeroRad) : fZeroRad(zeroRad) {
720 if (zeroRad) {
721 fName.printf("zeroradroundrect");
722 } else {
723 fName.printf("arbroundrect");
724 }
725 }
726
727 protected:
onGetName()728 const char* onGetName() override {
729 return fName.c_str();
730 }
731
add_corner_arc(SkPath * path,const SkRect & rect,SkScalar xIn,SkScalar yIn,int startAngle)732 static void add_corner_arc(SkPath* path, const SkRect& rect,
733 SkScalar xIn, SkScalar yIn,
734 int startAngle)
735 {
736
737 SkScalar rx = std::min(rect.width(), xIn);
738 SkScalar ry = std::min(rect.height(), yIn);
739
740 SkRect arcRect;
741 arcRect.setLTRB(-rx, -ry, rx, ry);
742 switch (startAngle) {
743 case 0:
744 arcRect.offset(rect.fRight - arcRect.fRight, rect.fBottom - arcRect.fBottom);
745 break;
746 case 90:
747 arcRect.offset(rect.fLeft - arcRect.fLeft, rect.fBottom - arcRect.fBottom);
748 break;
749 case 180:
750 arcRect.offset(rect.fLeft - arcRect.fLeft, rect.fTop - arcRect.fTop);
751 break;
752 case 270:
753 arcRect.offset(rect.fRight - arcRect.fRight, rect.fTop - arcRect.fTop);
754 break;
755 default:
756 break;
757 }
758
759 path->arcTo(arcRect, SkIntToScalar(startAngle), SkIntToScalar(90), false);
760 }
761
make_arb_round_rect(SkPath * path,const SkRect & r,SkScalar xCorner,SkScalar yCorner)762 static void make_arb_round_rect(SkPath* path, const SkRect& r,
763 SkScalar xCorner, SkScalar yCorner) {
764 // we are lazy here and use the same x & y for each corner
765 add_corner_arc(path, r, xCorner, yCorner, 270);
766 add_corner_arc(path, r, xCorner, yCorner, 0);
767 add_corner_arc(path, r, xCorner, yCorner, 90);
768 add_corner_arc(path, r, xCorner, yCorner, 180);
769 path->close();
770
771 SkASSERT(path->isConvex());
772 }
773
onDraw(int loops,SkCanvas * canvas)774 void onDraw(int loops, SkCanvas* canvas) override {
775 SkRandom rand;
776 SkRect r;
777
778 for (int i = 0; i < loops; ++i) {
779 SkPaint paint;
780 paint.setColor(0xff000000 | rand.nextU());
781 paint.setAntiAlias(true);
782
783 SkScalar size = rand.nextUScalar1() * 30;
784 if (size < SK_Scalar1) {
785 continue;
786 }
787 r.fLeft = rand.nextUScalar1() * 300;
788 r.fTop = rand.nextUScalar1() * 300;
789 r.fRight = r.fLeft + 2 * size;
790 r.fBottom = r.fTop + 2 * size;
791
792 SkPath temp;
793
794 if (fZeroRad) {
795 make_arb_round_rect(&temp, r, 0, 0);
796
797 SkASSERT(temp.isRect(nullptr));
798 } else {
799 make_arb_round_rect(&temp, r, r.width() / 10, r.height() / 15);
800 }
801
802 canvas->drawPath(temp, paint);
803 }
804 }
805
806 private:
807 bool fZeroRad; // should 0 radius rounds rects be tested?
808
809 using INHERITED = Benchmark;
810 };
811
812 class ConservativelyContainsBench : public Benchmark {
813 public:
814 enum Type {
815 kRect_Type,
816 kRoundRect_Type,
817 kOval_Type,
818 };
819
ConservativelyContainsBench(Type type)820 ConservativelyContainsBench(Type type) {
821 fParity = false;
822 fName = "conservatively_contains_";
823 switch (type) {
824 case kRect_Type:
825 fName.append("rect");
826 fPath.addRect(kBaseRect);
827 break;
828 case kRoundRect_Type:
829 fName.append("round_rect");
830 fPath.addRoundRect(kBaseRect, kRRRadii[0], kRRRadii[1]);
831 break;
832 case kOval_Type:
833 fName.append("oval");
834 fPath.addOval(kBaseRect);
835 break;
836 }
837 }
838
isSuitableFor(Backend backend)839 bool isSuitableFor(Backend backend) override {
840 return backend == kNonRendering_Backend;
841 }
842
843 private:
onGetName()844 const char* onGetName() override {
845 return fName.c_str();
846 }
847
onDraw(int loops,SkCanvas *)848 void onDraw(int loops, SkCanvas*) override {
849 for (int i = 0; i < loops; ++i) {
850 const SkRect& rect = fQueryRects[i % kQueryRectCnt];
851 fParity = fParity != fPath.conservativelyContainsRect(rect);
852 }
853 }
854
onDelayedSetup()855 void onDelayedSetup() override {
856 fQueryRects.setCount(kQueryRectCnt);
857
858 SkRandom rand;
859 for (int i = 0; i < kQueryRectCnt; ++i) {
860 SkSize size;
861 SkPoint xy;
862 size.fWidth = rand.nextRangeScalar(kQueryMin.fWidth, kQueryMax.fWidth);
863 size.fHeight = rand.nextRangeScalar(kQueryMin.fHeight, kQueryMax.fHeight);
864 xy.fX = rand.nextRangeScalar(kBounds.fLeft, kBounds.fRight - size.fWidth);
865 xy.fY = rand.nextRangeScalar(kBounds.fTop, kBounds.fBottom - size.fHeight);
866
867 fQueryRects[i] = SkRect::MakeXYWH(xy.fX, xy.fY, size.fWidth, size.fHeight);
868 }
869 }
870
871 enum {
872 kQueryRectCnt = 400,
873 };
874 static const SkRect kBounds; // bounds for all random query rects
875 static const SkSize kQueryMin; // minimum query rect size, should be <= kQueryMax
876 static const SkSize kQueryMax; // max query rect size, should < kBounds
877 static const SkRect kBaseRect; // rect that is used to construct the path
878 static const SkScalar kRRRadii[2]; // x and y radii for round rect
879
880 SkString fName;
881 SkPath fPath;
882 bool fParity;
883 SkTDArray<SkRect> fQueryRects;
884
885 using INHERITED = Benchmark;
886 };
887
888 ///////////////////////////////////////////////////////////////////////////////
889
890 #include "src/core/SkGeometry.h"
891
892 class ConicBench_Chop : public Benchmark {
893 protected:
894 SkConic fRQ, fDst[2];
895 SkString fName;
896 public:
ConicBench_Chop()897 ConicBench_Chop() : fName("conic-chop") {
898 fRQ.fPts[0].set(0, 0);
899 fRQ.fPts[1].set(100, 0);
900 fRQ.fPts[2].set(100, 100);
901 fRQ.fW = SkScalarCos(SK_ScalarPI/4);
902 }
903
isSuitableFor(Backend backend)904 bool isSuitableFor(Backend backend) override {
905 return backend == kNonRendering_Backend;
906 }
907
908 private:
onGetName()909 const char* onGetName() override { return fName.c_str(); }
910
onDraw(int loops,SkCanvas *)911 void onDraw(int loops, SkCanvas*) override {
912 for (int i = 0; i < loops; ++i) {
913 fRQ.chop(fDst);
914 }
915 }
916
917 using INHERITED = Benchmark;
918 };
919 DEF_BENCH( return new ConicBench_Chop; )
920
921 class ConicBench_EvalPos : public ConicBench_Chop {
922 const bool fUseV2;
923 public:
ConicBench_EvalPos(bool useV2)924 ConicBench_EvalPos(bool useV2) : fUseV2(useV2) {
925 fName.printf("conic-eval-pos%d", useV2);
926 }
onDraw(int loops,SkCanvas *)927 void onDraw(int loops, SkCanvas*) override {
928 if (fUseV2) {
929 for (int i = 0; i < loops; ++i) {
930 for (int j = 0; j < 1000; ++j) {
931 fDst[0].fPts[0] = fRQ.evalAt(0.4f);
932 }
933 }
934 } else {
935 for (int i = 0; i < loops; ++i) {
936 for (int j = 0; j < 1000; ++j) {
937 fRQ.evalAt(0.4f, &fDst[0].fPts[0], nullptr);
938 }
939 }
940 }
941 }
942 };
943 DEF_BENCH( return new ConicBench_EvalPos(false); )
944 DEF_BENCH( return new ConicBench_EvalPos(true); )
945
946 class ConicBench_EvalTan : public ConicBench_Chop {
947 const bool fUseV2;
948 public:
ConicBench_EvalTan(bool useV2)949 ConicBench_EvalTan(bool useV2) : fUseV2(useV2) {
950 fName.printf("conic-eval-tan%d", useV2);
951 }
onDraw(int loops,SkCanvas *)952 void onDraw(int loops, SkCanvas*) override {
953 if (fUseV2) {
954 for (int i = 0; i < loops; ++i) {
955 for (int j = 0; j < 1000; ++j) {
956 fDst[0].fPts[0] = fRQ.evalTangentAt(0.4f);
957 }
958 }
959 } else {
960 for (int i = 0; i < loops; ++i) {
961 for (int j = 0; j < 1000; ++j) {
962 fRQ.evalAt(0.4f, nullptr, &fDst[0].fPts[0]);
963 }
964 }
965 }
966 }
967 };
968 DEF_BENCH( return new ConicBench_EvalTan(false); )
969 DEF_BENCH( return new ConicBench_EvalTan(true); )
970
971 class ConicBench_TinyError : public Benchmark {
972 protected:
973 SkString fName;
974
975 public:
ConicBench_TinyError()976 ConicBench_TinyError() : fName("conic-tinyerror") {}
977
978 protected:
onGetName()979 const char* onGetName() override { return fName.c_str(); }
980
onDraw(int loops,SkCanvas *)981 void onDraw(int loops, SkCanvas*) override {
982 SkPaint paint;
983 paint.setColor(SK_ColorBLACK);
984 paint.setAntiAlias(true);
985 paint.setStyle(SkPaint::kStroke_Style);
986 paint.setStrokeWidth(2);
987
988 SkPath path;
989 path.moveTo(-100, 1);
990 path.cubicTo(-101, 1, -118, -47, -138, -44);
991
992 // The large y scale factor produces a tiny error threshold.
993 const SkMatrix mtx = SkMatrix::MakeAll(3.07294035f, 0.833333373f, 361.111115f, 0.0f,
994 6222222.5f, 28333.334f, 0.0f, 0.0f, 1.0f);
995 for (int i = 0; i < loops; ++i) {
996 SkPath dst;
997 paint.getFillPath(path, &dst, nullptr, SkPaintPriv::ComputeResScaleForStroking(mtx));
998 }
999 }
1000
1001 private:
1002 using INHERITED = Benchmark;
1003 };
DEF_BENCH(return new ConicBench_TinyError;)1004 DEF_BENCH( return new ConicBench_TinyError; )
1005
1006 ///////////////////////////////////////////////////////////////////////////////
1007
1008 static void rand_conic(SkConic* conic, SkRandom& rand) {
1009 for (int i = 0; i < 3; ++i) {
1010 conic->fPts[i].set(rand.nextUScalar1() * 100, rand.nextUScalar1() * 100);
1011 }
1012 if (rand.nextUScalar1() > 0.5f) {
1013 conic->fW = rand.nextUScalar1();
1014 } else {
1015 conic->fW = 1 + rand.nextUScalar1() * 4;
1016 }
1017 }
1018
1019 class ConicBench : public Benchmark {
1020 public:
ConicBench()1021 ConicBench() {
1022 SkRandom rand;
1023 for (int i = 0; i < CONICS; ++i) {
1024 rand_conic(&fConics[i], rand);
1025 }
1026 }
1027
isSuitableFor(Backend backend)1028 bool isSuitableFor(Backend backend) override {
1029 return backend == kNonRendering_Backend;
1030 }
1031
1032 protected:
1033 enum {
1034 CONICS = 100
1035 };
1036 SkConic fConics[CONICS];
1037
1038 private:
1039 using INHERITED = Benchmark;
1040 };
1041
1042 class ConicBench_ComputeError : public ConicBench {
1043 public:
ConicBench_ComputeError()1044 ConicBench_ComputeError() {}
1045
1046 protected:
onGetName()1047 const char* onGetName() override {
1048 return "conic-compute-error";
1049 }
1050
onDraw(int loops,SkCanvas *)1051 void onDraw(int loops, SkCanvas*) override {
1052 SkVector err;
1053 for (int i = 0; i < loops; ++i) {
1054 for (int j = 0; j < CONICS; ++j) {
1055 fConics[j].computeAsQuadError(&err);
1056 }
1057 }
1058 }
1059
1060 private:
1061 using INHERITED = ConicBench;
1062 };
1063
1064 class ConicBench_asQuadTol : public ConicBench {
1065 public:
ConicBench_asQuadTol()1066 ConicBench_asQuadTol() {}
1067
1068 protected:
onGetName()1069 const char* onGetName() override {
1070 return "conic-asQuadTol";
1071 }
1072
onDraw(int loops,SkCanvas *)1073 void onDraw(int loops, SkCanvas*) override {
1074 for (int i = 0; i < loops; ++i) {
1075 for (int j = 0; j < CONICS; ++j) {
1076 fConics[j].asQuadTol(SK_ScalarHalf);
1077 }
1078 }
1079 }
1080
1081 private:
1082 using INHERITED = ConicBench;
1083 };
1084
1085 class ConicBench_quadPow2 : public ConicBench {
1086 public:
ConicBench_quadPow2()1087 ConicBench_quadPow2() {}
1088
1089 protected:
onGetName()1090 const char* onGetName() override {
1091 return "conic-quadPow2";
1092 }
1093
onDraw(int loops,SkCanvas *)1094 void onDraw(int loops, SkCanvas*) override {
1095 for (int i = 0; i < loops; ++i) {
1096 for (int j = 0; j < CONICS; ++j) {
1097 fConics[j].computeQuadPOW2(SK_ScalarHalf);
1098 }
1099 }
1100 }
1101
1102 private:
1103 using INHERITED = ConicBench;
1104 };
1105
1106 ///////////////////////////////////////////////////////////////////////////////
1107
1108 class TightBoundsBench : public Benchmark {
1109 SkPath fPath;
1110 SkString fName;
1111 SkRect (*fProc)(const SkPath&);
1112
1113 public:
TightBoundsBench(SkRect (* proc)(const SkPath &),const char suffix[])1114 TightBoundsBench(SkRect (*proc)(const SkPath&), const char suffix[]) : fProc(proc) {
1115 fName.printf("tight_bounds_%s", suffix);
1116
1117 const int N = 100;
1118 SkRandom rand;
1119 for (int i = 0; i < N; ++i) {
1120 fPath.moveTo(rand.nextF()*100, rand.nextF()*100);
1121 fPath.lineTo(rand.nextF()*100, rand.nextF()*100);
1122 fPath.quadTo(rand.nextF()*100, rand.nextF()*100, rand.nextF()*100, rand.nextF()*100);
1123 fPath.conicTo(rand.nextF()*100, rand.nextF()*100, rand.nextF()*100, rand.nextF()*100,
1124 rand.nextF()*10);
1125 fPath.cubicTo(rand.nextF()*100, rand.nextF()*100, rand.nextF()*100, rand.nextF()*100,
1126 rand.nextF()*100, rand.nextF()*100);
1127 }
1128 }
1129
1130 protected:
isSuitableFor(Backend backend)1131 bool isSuitableFor(Backend backend) override {
1132 return backend == kNonRendering_Backend;
1133 }
1134
onGetName()1135 const char* onGetName() override { return fName.c_str(); }
1136
onDraw(int loops,SkCanvas * canvas)1137 void onDraw(int loops, SkCanvas* canvas) override {
1138 for (int i = 0; i < loops*100; ++i) {
1139 fProc(fPath);
1140 }
1141 }
1142
1143 private:
1144 using INHERITED = Benchmark;
1145 };
1146
1147
1148 const SkRect ConservativelyContainsBench::kBounds = SkRect::MakeWH(SkIntToScalar(100), SkIntToScalar(100));
1149 const SkSize ConservativelyContainsBench::kQueryMin = {SkIntToScalar(1), SkIntToScalar(1)};
1150 const SkSize ConservativelyContainsBench::kQueryMax = {SkIntToScalar(40), SkIntToScalar(40)};
1151 const SkRect ConservativelyContainsBench::kBaseRect = SkRect::MakeXYWH(SkIntToScalar(25), SkIntToScalar(25), SkIntToScalar(50), SkIntToScalar(50));
1152 const SkScalar ConservativelyContainsBench::kRRRadii[2] = {SkIntToScalar(5), SkIntToScalar(10)};
1153
1154 DEF_BENCH( return new TrianglePathBench(FLAGS00); )
1155 DEF_BENCH( return new TrianglePathBench(FLAGS01); )
1156 DEF_BENCH( return new TrianglePathBench(FLAGS10); )
1157 DEF_BENCH( return new TrianglePathBench(FLAGS11); )
1158
1159 DEF_BENCH( return new RectPathBench(FLAGS00); )
1160 DEF_BENCH( return new RectPathBench(FLAGS01); )
1161 DEF_BENCH( return new RectPathBench(FLAGS10); )
1162 DEF_BENCH( return new RectPathBench(FLAGS11); )
1163
1164 DEF_BENCH( return new RotatedRectBench(FLAGS00, false, 45));
1165 DEF_BENCH( return new RotatedRectBench(FLAGS10, false, 45));
1166 DEF_BENCH( return new RotatedRectBench(FLAGS00, true, 45));
1167 DEF_BENCH( return new RotatedRectBench(FLAGS10, true, 45));
1168
1169 DEF_BENCH( return new OvalPathBench(FLAGS00); )
1170 DEF_BENCH( return new OvalPathBench(FLAGS01); )
1171 DEF_BENCH( return new OvalPathBench(FLAGS10); )
1172 DEF_BENCH( return new OvalPathBench(FLAGS11); )
1173
1174 DEF_BENCH( return new CirclePathBench(FLAGS00); )
1175 DEF_BENCH( return new CirclePathBench(FLAGS01); )
1176 DEF_BENCH( return new CirclePathBench(FLAGS10); )
1177 DEF_BENCH( return new CirclePathBench(FLAGS11); )
1178
1179 DEF_BENCH( return new NonAACirclePathBench(FLAGS00); )
1180 DEF_BENCH( return new NonAACirclePathBench(FLAGS10); )
1181
1182 DEF_BENCH( return new AAAConcavePathBench(FLAGS00); )
1183 DEF_BENCH( return new AAAConcavePathBench(FLAGS10); )
1184 DEF_BENCH( return new AAAConvexPathBench(FLAGS00); )
1185 DEF_BENCH( return new AAAConvexPathBench(FLAGS10); )
1186
1187 DEF_BENCH( return new SawToothPathBench(FLAGS00); )
1188 DEF_BENCH( return new SawToothPathBench(FLAGS01); )
1189
1190 DEF_BENCH( return new LongCurvedPathBench(FLAGS00); )
1191 DEF_BENCH( return new LongCurvedPathBench(FLAGS01); )
1192 DEF_BENCH( return new LongLinePathBench(FLAGS00); )
1193 DEF_BENCH( return new LongLinePathBench(FLAGS01); )
1194
1195 DEF_BENCH( return new PathCreateBench(); )
1196 DEF_BENCH( return new PathCopyBench(); )
1197 DEF_BENCH( return new PathTransformBench(true); )
1198 DEF_BENCH( return new PathTransformBench(false); )
1199 DEF_BENCH( return new PathEqualityBench(); )
1200
1201 DEF_BENCH( return new SkBench_AddPathTest(SkBench_AddPathTest::kAdd_AddType); )
1202 DEF_BENCH( return new SkBench_AddPathTest(SkBench_AddPathTest::kAddTrans_AddType); )
1203 DEF_BENCH( return new SkBench_AddPathTest(SkBench_AddPathTest::kAddMatrix_AddType); )
1204 DEF_BENCH( return new SkBench_AddPathTest(SkBench_AddPathTest::kReverseAdd_AddType); )
1205 DEF_BENCH( return new SkBench_AddPathTest(SkBench_AddPathTest::kReversePathTo_AddType); )
1206
1207 DEF_BENCH( return new CirclesBench(FLAGS00); )
1208 DEF_BENCH( return new CirclesBench(FLAGS01); )
1209 DEF_BENCH( return new ArbRoundRectBench(false); )
1210 DEF_BENCH( return new ArbRoundRectBench(true); )
1211 DEF_BENCH( return new ConservativelyContainsBench(ConservativelyContainsBench::kRect_Type); )
1212 DEF_BENCH( return new ConservativelyContainsBench(ConservativelyContainsBench::kRoundRect_Type); )
1213 DEF_BENCH( return new ConservativelyContainsBench(ConservativelyContainsBench::kOval_Type); )
1214
1215 #include "include/pathops/SkPathOps.h"
1216 #include "src/core/SkPathPriv.h"
1217
__anon53182ec70802(const SkPath& path)1218 DEF_BENCH( return new TightBoundsBench([](const SkPath& path){ return path.computeTightBounds();},
1219 "priv"); )
__anon53182ec70902(const SkPath& path) 1220 DEF_BENCH( return new TightBoundsBench([](const SkPath& path) {
1221 SkRect bounds; TightBounds(path, &bounds); return bounds;
1222 }, "pathops"); )
1223
1224 // These seem to be optimized away, which is troublesome for timing.
1225 /*
1226 DEF_BENCH( return new ConicBench_Chop5() )
1227 DEF_BENCH( return new ConicBench_ComputeError() )
1228 DEF_BENCH( return new ConicBench_asQuadTol() )
1229 DEF_BENCH( return new ConicBench_quadPow2() )
1230 */
1231
1232 class CommonConvexBench : public Benchmark {
1233 protected:
1234 SkString fName;
1235 SkPath fPath;
1236 const bool fAA;
1237
1238 public:
CommonConvexBench(int w,int h,bool forceConcave,bool aa)1239 CommonConvexBench(int w, int h, bool forceConcave, bool aa) : fAA(aa) {
1240 fName.printf("convex_path_%d_%d_%d_%d", w, h, forceConcave, aa);
1241
1242 SkRect r = SkRect::MakeXYWH(10, 10, w*1.0f, h*1.0f);
1243 fPath.addRRect(SkRRect::MakeRectXY(r, w/8.0f, h/8.0f));
1244
1245 if (forceConcave) {
1246 SkPathPriv::SetConvexity(fPath, SkPathConvexity::kConcave);
1247 SkASSERT(!fPath.isConvex());
1248 } else {
1249 SkASSERT(fPath.isConvex());
1250 }
1251 }
1252
1253 protected:
onGetName()1254 const char* onGetName() override {
1255 return fName.c_str();
1256 }
1257
onDraw(int loops,SkCanvas * canvas)1258 void onDraw(int loops, SkCanvas* canvas) override {
1259 SkPaint paint;
1260 paint.setAntiAlias(fAA);
1261
1262 for (int i = 0; i < loops; ++i) {
1263 for (int inner = 0; inner < 100; ++inner) {
1264 canvas->drawPath(fPath, paint);
1265 }
1266 }
1267 }
1268
1269 private:
1270 using INHERITED = Benchmark;
1271 };
1272
1273 DEF_BENCH( return new CommonConvexBench( 16, 16, false, false); )
1274 DEF_BENCH( return new CommonConvexBench( 16, 16, true, false); )
1275 DEF_BENCH( return new CommonConvexBench( 16, 16, false, true); )
1276 DEF_BENCH( return new CommonConvexBench( 16, 16, true, true); )
1277
1278 DEF_BENCH( return new CommonConvexBench(200, 16, false, false); )
1279 DEF_BENCH( return new CommonConvexBench(200, 16, true, false); )
1280 DEF_BENCH( return new CommonConvexBench(200, 16, false, true); )
1281 DEF_BENCH( return new CommonConvexBench(200, 16, true, true); )
1282