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