blob: dcb75c9119679014bf2a6e3f4641d7f8fcf24833 [file] [log] [blame]
cdalton4a6e40d2016-02-10 14:54:21 -08001/*
2 * Copyright 2016 Google Inc.
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
7
Mike Kleinc0bd9f92019-04-23 12:05:21 -05008#include "bench/Benchmark.h"
9#include "include/core/SkCanvas.h"
10#include "include/core/SkPaint.h"
11#include "include/core/SkRRect.h"
12#include "include/core/SkString.h"
13#include "include/utils/SkRandom.h"
14#include "tools/flags/CommandLineFlags.h"
Hal Canary8a001442018-09-19 11:31:27 -040015
cdalton4a6e40d2016-02-10 14:54:21 -080016#include <stdio.h>
17#include <stdlib.h>
18#include <functional>
19
20#define ENABLE_COMMAND_LINE_SHAPES_BENCH 0
21
22#if ENABLE_COMMAND_LINE_SHAPES_BENCH
Mike Klein84836b72019-03-21 11:31:36 -050023static DEFINE_string(shapesType, "mixed",
24 "Type of shape to use in ShapesBench. Must be one of: "
25 "rect, oval, rrect, mixed.");
26static DEFINE_string(innerShapesType, "none",
27 "Type of inner shape to use in ShapesBench. Must be one of: "
28 "none, rect, oval, rrect, mixed.");
Mike Klein5b3f3432019-03-21 11:42:21 -050029static DEFINE_int(numShapes, 10000, "Number of shapes to draw in ShapesBench.");
Mike Klein84836b72019-03-21 11:31:36 -050030static DEFINE_string(shapesSize, "32x32", "Size of shapes to draw in ShapesBench.");
31static DEFINE_bool(shapesPersp, false, "Use slight perspective tilt in ShapesBench?");
cdalton4a6e40d2016-02-10 14:54:21 -080032#endif
33
34/*
35 * This class is used for several benchmarks that draw different primitive Skia shapes at various
36 * sizes. It is used to test both CPU-bound and GPU-bound rendering situations. It draws large
37 * amounts of shapes internally (rather than relying on nanobench selecting lots of loops) in order
38 * to take advantage of instanced rendering approaches.
39 */
40class ShapesBench : public Benchmark {
41public:
42 enum ShapesType {
43 kNone_ShapesType,
44 kRect_ShapesType,
45 kOval_ShapesType,
46 kRRect_ShapesType,
47 kMixed_ShapesType
48 };
49
50 ShapesBench(ShapesType shapesType, ShapesType innerShapesType,
51 int numShapes, const SkISize& shapesSize, bool perspective)
52 : fShapesType(shapesType)
53 , fInnerShapesType(innerShapesType)
54 , fNumShapes(numShapes)
55 , fShapesSize(shapesSize)
56 , fPerspective(perspective) {
57 clampShapeSize();
58 }
59
60#if ENABLE_COMMAND_LINE_SHAPES_BENCH
61 ShapesBench() {
62 if (!strcmp(FLAGS_shapesType[0], "rect")) {
63 fShapesType = kRect_ShapesType;
64 } else if (!strcmp(FLAGS_shapesType[0], "oval")) {
65 fShapesType = kOval_ShapesType;
66 } else if (!strcmp(FLAGS_shapesType[0], "rrect")) {
67 fShapesType = kRRect_ShapesType;
68 } else if (!strcmp(FLAGS_shapesType[0], "mixed")) {
69 fShapesType = kMixed_ShapesType;
70 } else {
71 SkDebugf("Invalid shapesType \"%s\". Must be one of: rect, oval, rrect, mixed.",
72 FLAGS_shapesType[0]);
73 exit(-1);
74 }
75 if (!strcmp(FLAGS_innerShapesType[0], "none")) {
76 fInnerShapesType = kNone_ShapesType;
77 } else if (!strcmp(FLAGS_innerShapesType[0], "rect")) {
78 fInnerShapesType = kRect_ShapesType;
79 } else if (!strcmp(FLAGS_innerShapesType[0], "oval")) {
80 fInnerShapesType = kOval_ShapesType;
81 } else if (!strcmp(FLAGS_innerShapesType[0], "rrect")) {
82 fInnerShapesType = kRRect_ShapesType;
83 } else if (!strcmp(FLAGS_innerShapesType[0], "mixed")) {
84 fInnerShapesType = kMixed_ShapesType;
85 } else {
86 SkDebugf("Invalid innerShapesType \"%s\". Must be one of: "
87 "none, rect, oval, rrect, mixed.", FLAGS_innerShapesType[0]);
88 exit(-1);
89 }
90 if (2 != sscanf(FLAGS_shapesSize[0], "%ix%i", &fShapesSize.fWidth, &fShapesSize.fHeight)) {
91 SkDebugf("Could not parse shapesSize from \"%s\". Expected \"%%ix%%i\"\n",
92 FLAGS_shapesSize[0]);
93 exit(-1);
94 }
95
96 fNumShapes = FLAGS_numShapes;
97 fPerspective = FLAGS_shapesPersp;
98
99 clampShapeSize();
100 }
101#endif
102
cdalton4a6e40d2016-02-10 14:54:21 -0800103private:
104 void clampShapeSize() {
Brian Osman788b9162020-02-07 10:36:46 -0500105 float maxDiagonal = static_cast<float>(std::min(kBenchWidth, kBenchHeight));
cdalton4a6e40d2016-02-10 14:54:21 -0800106 float diagonal = sqrtf(static_cast<float>(fShapesSize.width() * fShapesSize.width()) +
107 static_cast<float>(fShapesSize.height() * fShapesSize.height()));
108 if (diagonal > maxDiagonal) {
109 fShapesSize.fWidth = static_cast<int>(fShapesSize.width() * maxDiagonal / diagonal);
110 fShapesSize.fHeight = static_cast<int>(fShapesSize.height() * maxDiagonal / diagonal);
111 }
112 }
113
114 const char* onGetName() override {
115 const char* shapeTypeNames[] = {
116 "none", "rect", "oval", "rrect", "mixed"
117 };
118
119 fName.printf("shapes_%s", shapeTypeNames[fShapesType]);
120
121 if (kNone_ShapesType != fInnerShapesType) {
122 fName.appendf("_inner_%s", shapeTypeNames[fInnerShapesType]);
123 }
124
125 fName.appendf("_%i_%ix%i", fNumShapes, fShapesSize.width(), fShapesSize.height());
126
127 if (fPerspective) {
128 fName.append("_persp");
129 }
130
131 return fName.c_str();
132 }
133 SkIPoint onGetSize() override { return SkIPoint::Make(kBenchWidth, kBenchHeight); }
134
135 void onDelayedSetup() override {
136 SkScalar w = SkIntToScalar(fShapesSize.width());
137 SkScalar h = SkIntToScalar(fShapesSize.height());
138
139 fRect.setRect(SkRect::MakeXYWH(-w / 2, -h / 2, w, h));
140 fOval.setOval(fRect.rect());
141 fRRect.setNinePatch(fRect.rect(), w / 8, h / 13, w / 11, h / 7);
142
143 if (kNone_ShapesType != fInnerShapesType) {
144 fRect.inset(w / 7, h / 11, &fInnerRect);
145 fInnerRect.offset(w / 28, h / 44);
146 fInnerOval.setOval(fInnerRect.rect());
147 fInnerRRect.setRectXY(fInnerRect.rect(), w / 13, w / 7);
148 }
149
150 SkRandom rand;
151 fShapes.push_back_n(fNumShapes);
152 for (int i = 0; i < fNumShapes; i++) {
153 float pad = sqrtf(static_cast<float>(fShapesSize.width() * fShapesSize.width()) +
154 static_cast<float>(fShapesSize.height() * fShapesSize.height()));
155 fShapes[i].fMatrix.setTranslate(0.5f * pad + rand.nextF() * (kBenchWidth - pad),
156 0.5f * pad + rand.nextF() * (kBenchHeight - pad));
157 fShapes[i].fMatrix.preRotate(rand.nextF() * 360.0f);
158 if (fPerspective) {
159 fShapes[i].fMatrix.setPerspX(0.00015f);
160 fShapes[i].fMatrix.setPerspY(-0.00015f);
161 }
162 fShapes[i].fColor = rand.nextU() | 0xff808080;
163 }
164 for (int i = 0; i < fNumShapes; i++) {
165 // Do this in a separate loop so mixed shapes get the same random numbers during
166 // placement as non-mixed do.
167 int shapeType = fShapesType;
168 if (kMixed_ShapesType == shapeType) {
169 shapeType = rand.nextRangeU(kRect_ShapesType, kRRect_ShapesType);
170 }
171 int innerShapeType = fInnerShapesType;
172 if (kMixed_ShapesType == innerShapeType) {
173 innerShapeType = rand.nextRangeU(kRect_ShapesType, kRRect_ShapesType);
174 }
175 if (kNone_ShapesType == innerShapeType) {
176 switch (shapeType) {
177 using namespace std;
178 using namespace std::placeholders;
179 case kRect_ShapesType:
180 fShapes[i].fDraw = bind(&SkCanvas::drawRect, _1, cref(fRect.rect()), _2);
181 break;
182 case kOval_ShapesType:
183 fShapes[i].fDraw = bind(&SkCanvas::drawOval, _1, cref(fOval.rect()), _2);
184 break;
185 case kRRect_ShapesType:
186 fShapes[i].fDraw = bind(&SkCanvas::drawRRect, _1, cref(fRRect), _2);
187 break;
188 }
189 } else {
Kevin Lubick42846132018-01-05 10:11:11 -0500190 const SkRRect* outer = nullptr;
cdalton4a6e40d2016-02-10 14:54:21 -0800191 switch (shapeType) {
192 case kRect_ShapesType: outer = &fRect; break;
193 case kOval_ShapesType: outer = &fOval; break;
194 case kRRect_ShapesType: outer = &fRRect; break;
195 }
Kevin Lubick42846132018-01-05 10:11:11 -0500196 const SkRRect* inner = nullptr;
cdalton4a6e40d2016-02-10 14:54:21 -0800197 switch (innerShapeType) {
198 case kRect_ShapesType: inner = &fInnerRect; break;
199 case kOval_ShapesType: inner = &fInnerOval; break;
200 case kRRect_ShapesType: inner = &fInnerRRect; break;
201 }
202 fShapes[i].fDraw = std::bind(&SkCanvas::drawDRRect, std::placeholders::_1,
203 std::cref(*outer), std::cref(*inner),
204 std::placeholders::_2);
205 }
206 }
207 }
208
209 void onDraw(int loops, SkCanvas* canvas) override {
210 SkPaint paint;
211 this->setupPaint(&paint);
212 for (int j = 0; j < loops; j++) {
213 for (int i = 0; i < fNumShapes; i++) {
214 canvas->save();
215 canvas->setMatrix(fShapes[i].fMatrix);
216 paint.setColor(fShapes[i].fColor);
217 fShapes[i].fDraw(canvas, paint);
218 canvas->restore();
219 }
220 }
221 }
222
223 enum {
224 kBenchWidth = 1000,
225 kBenchHeight = 1000
226 };
227
228 struct ShapeInfo {
229 SkMatrix fMatrix;
230 SkColor fColor;
231 std::function<void(SkCanvas*, const SkPaint&)> fDraw;
232 };
233
234 ShapesType fShapesType;
235 ShapesType fInnerShapesType;
236 int fNumShapes;
237 SkISize fShapesSize;
238 bool fPerspective;
239 SkString fName;
240 SkRRect fRect;
241 SkRRect fOval;
242 SkRRect fRRect;
243 SkRRect fInnerRect;
244 SkRRect fInnerOval;
245 SkRRect fInnerRRect;
246 SkTArray<ShapeInfo> fShapes;
247
248
249 typedef Benchmark INHERITED;
250};
251
csmartdaltona7f29642016-07-07 08:49:11 -0700252#if ENABLE_COMMAND_LINE_SHAPES_BENCH
253DEF_BENCH(return new ShapesBench;)
254#else
cdalton4a6e40d2016-02-10 14:54:21 -0800255// Small primitives (CPU bound, in theory):
256DEF_BENCH(return new ShapesBench(ShapesBench::kRect_ShapesType, ShapesBench::kNone_ShapesType,
257 10000, SkISize::Make(32, 32), false);)
258DEF_BENCH(return new ShapesBench(ShapesBench::kOval_ShapesType, ShapesBench::kNone_ShapesType,
259 10000, SkISize::Make(32, 32), false);)
260DEF_BENCH(return new ShapesBench(ShapesBench::kOval_ShapesType, ShapesBench::kNone_ShapesType,
261 10000, SkISize::Make(32, 33), false);)
262DEF_BENCH(return new ShapesBench(ShapesBench::kRRect_ShapesType, ShapesBench::kNone_ShapesType,
263 10000, SkISize::Make(32, 32), false);)
264DEF_BENCH(return new ShapesBench(ShapesBench::kMixed_ShapesType, ShapesBench::kNone_ShapesType,
265 10000, SkISize::Make(32, 33), false);)
266
267// Large primitives (GPU bound, in theory):
268DEF_BENCH(return new ShapesBench(ShapesBench::kRect_ShapesType, ShapesBench::kNone_ShapesType,
brianosman32f19082016-02-17 13:43:40 -0800269 100, SkISize::Make(500, 500), false);)
cdalton4a6e40d2016-02-10 14:54:21 -0800270DEF_BENCH(return new ShapesBench(ShapesBench::kOval_ShapesType, ShapesBench::kNone_ShapesType,
brianosman32f19082016-02-17 13:43:40 -0800271 100, SkISize::Make(500, 500), false);)
cdalton4a6e40d2016-02-10 14:54:21 -0800272DEF_BENCH(return new ShapesBench(ShapesBench::kOval_ShapesType, ShapesBench::kNone_ShapesType,
brianosman32f19082016-02-17 13:43:40 -0800273 100, SkISize::Make(500, 501), false);)
cdalton4a6e40d2016-02-10 14:54:21 -0800274DEF_BENCH(return new ShapesBench(ShapesBench::kRRect_ShapesType, ShapesBench::kNone_ShapesType,
brianosman32f19082016-02-17 13:43:40 -0800275 100, SkISize::Make(500, 500), false);)
cdalton4a6e40d2016-02-10 14:54:21 -0800276DEF_BENCH(return new ShapesBench(ShapesBench::kMixed_ShapesType, ShapesBench::kNone_ShapesType,
brianosman32f19082016-02-17 13:43:40 -0800277 100, SkISize::Make(500, 501), false);)
cdalton4a6e40d2016-02-10 14:54:21 -0800278
279// Donuts (small and large). These fall-back to path rendering due to non-orthogonal rotation
280// making them quite slow. Thus, reduce the counts substantially:
281DEF_BENCH(return new ShapesBench(ShapesBench::kRect_ShapesType, ShapesBench::kRect_ShapesType,
brianosmanb8c97ba2016-02-11 09:07:02 -0800282 500, SkISize::Make(32, 32), false);)
cdalton4a6e40d2016-02-10 14:54:21 -0800283DEF_BENCH(return new ShapesBench(ShapesBench::kRRect_ShapesType, ShapesBench::kRRect_ShapesType,
brianosmanb8c97ba2016-02-11 09:07:02 -0800284 500, SkISize::Make(32, 32), false);)
cdalton4a6e40d2016-02-10 14:54:21 -0800285DEF_BENCH(return new ShapesBench(ShapesBench::kRect_ShapesType, ShapesBench::kRect_ShapesType,
brianosmanb8c97ba2016-02-11 09:07:02 -0800286 50, SkISize::Make(500, 500), false);)
cdalton4a6e40d2016-02-10 14:54:21 -0800287DEF_BENCH(return new ShapesBench(ShapesBench::kRRect_ShapesType, ShapesBench::kRRect_ShapesType,
brianosmanb8c97ba2016-02-11 09:07:02 -0800288 50, SkISize::Make(500, 500), false);)
cdalton4a6e40d2016-02-10 14:54:21 -0800289#endif