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