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 #include "bench/Benchmark.h"
8 #include "include/core/SkBitmap.h"
9 #include "include/core/SkCanvas.h"
10 #include "include/core/SkPaint.h"
11 #include "include/core/SkPath.h"
12 #include "include/core/SkString.h"
13 #include "include/core/SkStrokeRec.h"
14 #include "include/effects/SkDashPathEffect.h"
15 #include "include/private/SkTDArray.h"
16 #include "include/utils/SkRandom.h"
17 
18 
19 /*
20  *  Cases to consider:
21  *
22  *  1. antialiasing on/off (esp. width <= 1)
23  *  2. strokewidth == 0, 1, 2
24  *  3. hline, vline, diagonal, rect, oval
25  *  4. dots [1,1] ([N,N] where N=strokeWidth?) or arbitrary (e.g. [2,1] or [1,2,3,2])
26  */
path_hline(SkPath * path)27 static void path_hline(SkPath* path) {
28     path->moveTo(SkIntToScalar(10), SkIntToScalar(10));
29     path->lineTo(SkIntToScalar(600), SkIntToScalar(10));
30 }
31 
32 class DashBench : public Benchmark {
33 protected:
34     SkString            fName;
35     SkTDArray<SkScalar> fIntervals;
36     int                 fWidth;
37     SkPoint             fPts[2];
38     bool                fDoClip;
39 
40 public:
DashBench(const SkScalar intervals[],int count,int width,bool doClip=false)41     DashBench(const SkScalar intervals[], int count, int width,
42               bool doClip = false)  {
43         fIntervals.append(count, intervals);
44         for (int i = 0; i < count; ++i) {
45             fIntervals[i] *= width;
46         }
47         fWidth = width;
48         fName.printf("dash_%d_%s", width, doClip ? "clipped" : "noclip");
49         fDoClip = doClip;
50 
51         fPts[0].set(SkIntToScalar(10), SkIntToScalar(10));
52         fPts[1].set(SkIntToScalar(600), SkIntToScalar(10));
53     }
54 
makePath(SkPath * path)55     virtual void makePath(SkPath* path) {
56         path_hline(path);
57     }
58 
59 protected:
onGetName()60     const char* onGetName() override {
61         return fName.c_str();
62     }
63 
onDraw(int loops,SkCanvas * canvas)64     void onDraw(int loops, SkCanvas* canvas) override {
65         SkPaint paint;
66         this->setupPaint(&paint);
67         paint.setStyle(SkPaint::kStroke_Style);
68         paint.setStrokeWidth(SkIntToScalar(fWidth));
69         paint.setAntiAlias(false);
70 
71         SkPath path;
72         this->makePath(&path);
73 
74         paint.setPathEffect(SkDashPathEffect::Make(fIntervals.begin(), fIntervals.count(), 0));
75 
76         if (fDoClip) {
77             SkRect r = path.getBounds();
78             r.inset(-SkIntToScalar(20), -SkIntToScalar(20));
79             // now move it so we don't intersect
80             r.offset(0, r.height() * 3 / 2);
81             canvas->clipRect(r);
82         }
83 
84         this->handlePath(canvas, path, paint, loops);
85     }
86 
handlePath(SkCanvas * canvas,const SkPath & path,const SkPaint & paint,int N)87     virtual void handlePath(SkCanvas* canvas, const SkPath& path,
88                             const SkPaint& paint, int N) {
89         for (int i = 0; i < N; ++i) {
90 //            canvas->drawPoints(SkCanvas::kLines_PointMode, 2, fPts, paint);
91             canvas->drawPath(path, paint);
92         }
93     }
94 
95 private:
96     using INHERITED = Benchmark;
97 };
98 
99 class RectDashBench : public DashBench {
100 public:
RectDashBench(const SkScalar intervals[],int count,int width)101     RectDashBench(const SkScalar intervals[], int count, int width)
102     : INHERITED(intervals, count, width) {
103         fName.append("_rect");
104     }
105 
106 protected:
handlePath(SkCanvas * canvas,const SkPath & path,const SkPaint & paint,int N)107     void handlePath(SkCanvas* canvas, const SkPath& path, const SkPaint& paint, int N) override {
108         SkPoint pts[2];
109         if (!path.isLine(pts) || pts[0].fY != pts[1].fY) {
110             this->INHERITED::handlePath(canvas, path, paint, N);
111         } else {
112             SkRect rect;
113             rect.fLeft = pts[0].fX;
114             rect.fTop = pts[0].fY - paint.getStrokeWidth() / 2;
115             rect.fRight = rect.fLeft + SkIntToScalar(fWidth);
116             rect.fBottom = rect.fTop + paint.getStrokeWidth();
117 
118             SkPaint p(paint);
119             p.setStyle(SkPaint::kFill_Style);
120             p.setPathEffect(nullptr);
121 
122             int count = SkScalarRoundToInt((pts[1].fX - pts[0].fX) / (2*fWidth));
123             SkScalar dx = SkIntToScalar(2 * fWidth);
124 
125             for (int i = 0; i < N*10; ++i) {
126                 SkRect r = rect;
127                 for (int j = 0; j < count; ++j) {
128                     canvas->drawRect(r, p);
129                     r.offset(dx, 0);
130                 }
131             }
132         }
133     }
134 
135 private:
136     using INHERITED = DashBench;
137 };
138 
make_unit_star(SkPath * path,int n)139 static void make_unit_star(SkPath* path, int n) {
140     SkScalar rad = -SK_ScalarPI / 2;
141     const SkScalar drad = (n >> 1) * SK_ScalarPI * 2 / n;
142 
143     path->moveTo(0, -SK_Scalar1);
144     for (int i = 1; i < n; i++) {
145         rad += drad;
146         path->lineTo(SkScalarCos(rad), SkScalarSin(rad));
147     }
148     path->close();
149 }
150 
make_poly(SkPath * path)151 static void make_poly(SkPath* path) {
152     make_unit_star(path, 9);
153     const SkMatrix matrix = SkMatrix::Scale(100, 100);
154     path->transform(matrix);
155 }
156 
make_quad(SkPath * path)157 static void make_quad(SkPath* path) {
158     SkScalar x0 = SkIntToScalar(10);
159     SkScalar y0 = SkIntToScalar(10);
160     path->moveTo(x0, y0);
161     path->quadTo(x0,                    y0 + 400 * SK_Scalar1,
162                  x0 + 600 * SK_Scalar1, y0 + 400 * SK_Scalar1);
163 }
164 
make_cubic(SkPath * path)165 static void make_cubic(SkPath* path) {
166     SkScalar x0 = SkIntToScalar(10);
167     SkScalar y0 = SkIntToScalar(10);
168     path->moveTo(x0, y0);
169     path->cubicTo(x0,                    y0 + 400 * SK_Scalar1,
170                   x0 + 600 * SK_Scalar1, y0 + 400 * SK_Scalar1,
171                   x0 + 600 * SK_Scalar1, y0);
172 }
173 
174 class MakeDashBench : public Benchmark {
175     SkString fName;
176     SkPath   fPath;
177     sk_sp<SkPathEffect> fPE;
178 
179 public:
MakeDashBench(void (* proc)(SkPath *),const char name[])180     MakeDashBench(void (*proc)(SkPath*), const char name[])  {
181         fName.printf("makedash_%s", name);
182         proc(&fPath);
183 
184         SkScalar vals[] = { SkIntToScalar(4), SkIntToScalar(4) };
185         fPE = SkDashPathEffect::Make(vals, 2, 0);
186     }
187 
188 protected:
onGetName()189     const char* onGetName() override {
190         return fName.c_str();
191     }
192 
onDraw(int loops,SkCanvas *)193     void onDraw(int loops, SkCanvas*) override {
194         SkPath dst;
195         for (int i = 0; i < loops; ++i) {
196             SkStrokeRec rec(SkStrokeRec::kHairline_InitStyle);
197 
198             fPE->filterPath(&dst, fPath, &rec, nullptr);
199             dst.rewind();
200         }
201     }
202 
203 private:
204     using INHERITED = Benchmark;
205 };
206 
207 /*
208  *  We try to special case square dashes (intervals are equal to strokewidth).
209  */
210 class DashLineBench : public Benchmark {
211     SkString fName;
212     SkScalar fStrokeWidth;
213     bool     fIsRound;
214     sk_sp<SkPathEffect> fPE;
215 
216 public:
DashLineBench(SkScalar width,bool isRound)217     DashLineBench(SkScalar width, bool isRound)  {
218         fName.printf("dashline_%g_%s", SkScalarToFloat(width), isRound ? "circle" : "square");
219         fStrokeWidth = width;
220         fIsRound = isRound;
221 
222         SkScalar vals[] = { SK_Scalar1, SK_Scalar1 };
223         fPE = SkDashPathEffect::Make(vals, 2, 0);
224     }
225 
226 protected:
onGetName()227     const char* onGetName() override {
228         return fName.c_str();
229     }
230 
onDraw(int loops,SkCanvas * canvas)231     void onDraw(int loops, SkCanvas* canvas) override {
232         SkPaint paint;
233         this->setupPaint(&paint);
234         paint.setStrokeWidth(fStrokeWidth);
235         paint.setStrokeCap(fIsRound ? SkPaint::kRound_Cap : SkPaint::kSquare_Cap);
236         paint.setPathEffect(fPE);
237         for (int i = 0; i < loops; ++i) {
238             canvas->drawLine(10 * SK_Scalar1, 10 * SK_Scalar1,
239                              640 * SK_Scalar1, 10 * SK_Scalar1, paint);
240         }
241     }
242 
243 private:
244     using INHERITED = Benchmark;
245 };
246 
247 class DrawPointsDashingBench : public Benchmark {
248     SkString fName;
249     int      fStrokeWidth;
250     bool     fDoAA;
251 
252     sk_sp<SkPathEffect> fPathEffect;
253 
254 public:
DrawPointsDashingBench(int dashLength,int strokeWidth,bool doAA)255     DrawPointsDashingBench(int dashLength, int strokeWidth, bool doAA)
256          {
257         fName.printf("drawpointsdash_%d_%d%s", dashLength, strokeWidth, doAA ? "_aa" : "_bw");
258         fStrokeWidth = strokeWidth;
259         fDoAA = doAA;
260 
261         SkScalar vals[] = { SkIntToScalar(dashLength), SkIntToScalar(dashLength) };
262         fPathEffect = SkDashPathEffect::Make(vals, 2, SK_Scalar1);
263     }
264 
265 protected:
onGetName()266     const char* onGetName() override {
267         return fName.c_str();
268     }
269 
onDraw(int loops,SkCanvas * canvas)270     void onDraw(int loops, SkCanvas* canvas) override {
271         SkPaint p;
272         this->setupPaint(&p);
273         p.setColor(SK_ColorBLACK);
274         p.setStyle(SkPaint::kStroke_Style);
275         p.setStrokeWidth(SkIntToScalar(fStrokeWidth));
276         p.setPathEffect(fPathEffect);
277         p.setAntiAlias(fDoAA);
278 
279         SkPoint pts[2] = {
280             { SkIntToScalar(10), 0 },
281             { SkIntToScalar(640), 0 }
282         };
283 
284         for (int i = 0; i < loops; ++i) {
285             pts[0].fY = pts[1].fY = SkIntToScalar(i % 480);
286             canvas->drawPoints(SkCanvas::kLines_PointMode, 2, pts, p);
287         }
288     }
289 
290 private:
291     using INHERITED = Benchmark;
292 };
293 
294 // Want to test how we handle dashing when 99% of the dash is clipped out
295 class GiantDashBench : public Benchmark {
296     SkString fName;
297     SkScalar fStrokeWidth;
298     SkPoint  fPts[2];
299     sk_sp<SkPathEffect> fPathEffect;
300 
301 public:
302     enum LineType {
303         kHori_LineType,
304         kVert_LineType,
305         kDiag_LineType,
306         kLineTypeCount
307     };
308 
LineTypeName(LineType lt)309     static const char* LineTypeName(LineType lt) {
310         static const char* gNames[] = { "hori", "vert", "diag" };
311         static_assert(kLineTypeCount == SK_ARRAY_COUNT(gNames), "names_wrong_size");
312         return gNames[lt];
313     }
314 
GiantDashBench(LineType lt,SkScalar width)315     GiantDashBench(LineType lt, SkScalar width)  {
316         fName.printf("giantdashline_%s_%g", LineTypeName(lt), width);
317         fStrokeWidth = width;
318 
319         // deliberately pick intervals that won't be caught by asPoints(), so
320         // we can test the filterPath code-path.
321         const SkScalar intervals[] = { 20, 10, 10, 10 };
322         fPathEffect = SkDashPathEffect::Make(intervals, SK_ARRAY_COUNT(intervals), 0);
323 
324         SkScalar cx = 640 / 2;  // center X
325         SkScalar cy = 480 / 2;  // center Y
326         SkMatrix matrix;
327 
328         switch (lt) {
329             case kHori_LineType:
330                 matrix.setIdentity();
331                 break;
332             case kVert_LineType:
333                 matrix.setRotate(90, cx, cy);
334                 break;
335             case kDiag_LineType:
336                 matrix.setRotate(45, cx, cy);
337                 break;
338             case kLineTypeCount:
339                 // Not a real enum value.
340                 break;
341         }
342 
343         const SkScalar overshoot = 100*1000;
344         const SkPoint pts[2] = {
345             { -overshoot, cy }, { 640 + overshoot, cy }
346         };
347         matrix.mapPoints(fPts, pts, 2);
348     }
349 
350 protected:
onGetName()351     const char* onGetName() override {
352         return fName.c_str();
353     }
354 
onDraw(int loops,SkCanvas * canvas)355     void onDraw(int loops, SkCanvas* canvas) override {
356         SkPaint p;
357         this->setupPaint(&p);
358         p.setStyle(SkPaint::kStroke_Style);
359         p.setStrokeWidth(fStrokeWidth);
360         p.setPathEffect(fPathEffect);
361 
362         for (int i = 0; i < loops; i++) {
363             canvas->drawPoints(SkCanvas::kLines_PointMode, 2, fPts, p);
364         }
365     }
366 
367 private:
368     using INHERITED = Benchmark;
369 };
370 
371 // Want to test how we draw a dashed grid (like what is used in spreadsheets) of many
372 // small dashed lines switching back and forth between horizontal and vertical
373 class DashGridBench : public Benchmark {
374     SkString fName;
375     int      fStrokeWidth;
376     bool     fDoAA;
377 
378     sk_sp<SkPathEffect> fPathEffect;
379 
380 public:
DashGridBench(int dashLength,int strokeWidth,bool doAA)381     DashGridBench(int dashLength, int strokeWidth, bool doAA) {
382         fName.printf("dashgrid_%d_%d%s", dashLength, strokeWidth, doAA ? "_aa" : "_bw");
383         fStrokeWidth = strokeWidth;
384         fDoAA = doAA;
385 
386         SkScalar vals[] = { SkIntToScalar(dashLength), SkIntToScalar(dashLength) };
387         fPathEffect = SkDashPathEffect::Make(vals, 2, SK_Scalar1);
388     }
389 
390 protected:
onGetName()391     const char* onGetName() override {
392         return fName.c_str();
393     }
394 
onDraw(int loops,SkCanvas * canvas)395     void onDraw(int loops, SkCanvas* canvas) override {
396         SkPaint p;
397         this->setupPaint(&p);
398         p.setColor(SK_ColorBLACK);
399         p.setStyle(SkPaint::kStroke_Style);
400         p.setStrokeWidth(SkIntToScalar(fStrokeWidth));
401         p.setPathEffect(fPathEffect);
402         p.setAntiAlias(fDoAA);
403 
404         SkPoint pts[4] = {
405             { SkIntToScalar(0), 20.5f },
406             { SkIntToScalar(20), 20.5f },
407             { 20.5f, SkIntToScalar(0) },
408             { 20.5f, SkIntToScalar(20) }
409         };
410 
411         for (int i = 0; i < loops; ++i) {
412             for (int j = 0; j < 10; ++j) {
413                 for (int k = 0; k < 10; ++k) {
414                     // Horizontal line
415                     SkPoint horPts[2];
416                     horPts[0].fX = pts[0].fX + k * 22.f;
417                     horPts[0].fY = pts[0].fY + j * 22.f;
418                     horPts[1].fX = pts[1].fX + k * 22.f;
419                     horPts[1].fY = pts[1].fY + j * 22.f;
420                     canvas->drawPoints(SkCanvas::kLines_PointMode, 2, horPts, p);
421 
422                     // Vertical line
423                     SkPoint vertPts[2];
424                     vertPts[0].fX = pts[2].fX + k * 22.f;
425                     vertPts[0].fY = pts[2].fY + j * 22.f;
426                     vertPts[1].fX = pts[3].fX + k * 22.f;
427                     vertPts[1].fY = pts[3].fY + j * 22.f;
428                     canvas->drawPoints(SkCanvas::kLines_PointMode, 2, vertPts, p);
429                 }
430             }
431         }
432     }
433 
434 private:
435     using INHERITED = Benchmark;
436 };
437 
438 ///////////////////////////////////////////////////////////////////////////////
439 
440 static const SkScalar gDots[] = { SK_Scalar1, SK_Scalar1 };
441 
442 #define PARAM(array)    array, SK_ARRAY_COUNT(array)
443 
444 DEF_BENCH( return new DashBench(PARAM(gDots), 0); )
445 DEF_BENCH( return new DashBench(PARAM(gDots), 1); )
446 DEF_BENCH( return new DashBench(PARAM(gDots), 1, true); )
447 DEF_BENCH( return new DashBench(PARAM(gDots), 4); )
448 DEF_BENCH( return new MakeDashBench(make_poly, "poly"); )
449 DEF_BENCH( return new MakeDashBench(make_quad, "quad"); )
450 DEF_BENCH( return new MakeDashBench(make_cubic, "cubic"); )
451 DEF_BENCH( return new DashLineBench(0, false); )
452 DEF_BENCH( return new DashLineBench(SK_Scalar1, false); )
453 DEF_BENCH( return new DashLineBench(2 * SK_Scalar1, false); )
454 DEF_BENCH( return new DashLineBench(0, true); )
455 DEF_BENCH( return new DashLineBench(SK_Scalar1, true); )
456 DEF_BENCH( return new DashLineBench(2 * SK_Scalar1, true); )
457 
458 DEF_BENCH( return new DrawPointsDashingBench(1, 1, false); )
459 DEF_BENCH( return new DrawPointsDashingBench(1, 1, true); )
460 DEF_BENCH( return new DrawPointsDashingBench(3, 1, false); )
461 DEF_BENCH( return new DrawPointsDashingBench(3, 1, true); )
462 DEF_BENCH( return new DrawPointsDashingBench(5, 5, false); )
463 DEF_BENCH( return new DrawPointsDashingBench(5, 5, true); )
464 
465 /* Disable the GiantDashBench for Android devices until we can better control
466  * the memory usage. (https://code.google.com/p/skia/issues/detail?id=1430)
467  */
468 #ifndef SK_BUILD_FOR_ANDROID
469 DEF_BENCH( return new GiantDashBench(GiantDashBench::kHori_LineType, 0); )
470 DEF_BENCH( return new GiantDashBench(GiantDashBench::kVert_LineType, 0); )
471 DEF_BENCH( return new GiantDashBench(GiantDashBench::kDiag_LineType, 0); )
472 
473 // pass 2 to explicitly avoid any 1-is-the-same-as-hairline special casing
474 
475 // hori_2 is just too slow to enable at the moment
476 DEF_BENCH( return new GiantDashBench(GiantDashBench::kHori_LineType, 2); )
477 DEF_BENCH( return new GiantDashBench(GiantDashBench::kVert_LineType, 2); )
478 DEF_BENCH( return new GiantDashBench(GiantDashBench::kDiag_LineType, 2); )
479 
480 DEF_BENCH( return new DashGridBench(1, 1, true); )
481 DEF_BENCH( return new DashGridBench(1, 1, false); )
482 DEF_BENCH( return new DashGridBench(3, 1, true); )
483 DEF_BENCH( return new DashGridBench(3, 1, false); )
484 #endif
485