blob: 55e777d8fbe5ba3be58a69d8d928b0842d89b4bd [file] [log] [blame]
epoger@google.comec3ed6a2011-07-28 14:26:00 +00001
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 */
reed@google.com84e9c082011-04-13 17:44:24 +00008#include "SkBenchmark.h"
9#include "SkBitmap.h"
10#include "SkCanvas.h"
11#include "SkColorPriv.h"
12#include "SkGradientShader.h"
13#include "SkPaint.h"
14#include "SkShader.h"
15#include "SkString.h"
commit-bot@chromium.org3339ac52014-05-22 02:55:59 +000016#include "SkUnitMapper.h"
reed@google.com84e9c082011-04-13 17:44:24 +000017
18struct GradData {
19 int fCount;
20 const SkColor* fColors;
21 const SkScalar* fPos;
commit-bot@chromium.org5396a042013-07-10 16:13:14 +000022 const char* fName;
reed@google.com84e9c082011-04-13 17:44:24 +000023};
24
25static const SkColor gColors[] = {
commit-bot@chromium.orgb6865392013-07-09 19:43:29 +000026 SK_ColorRED, SK_ColorGREEN, SK_ColorBLUE, SK_ColorWHITE, SK_ColorBLACK,
commit-bot@chromium.org5396a042013-07-10 16:13:14 +000027 SK_ColorRED, SK_ColorGREEN, SK_ColorBLUE, SK_ColorWHITE, SK_ColorBLACK,
28 SK_ColorRED, SK_ColorGREEN, SK_ColorBLUE, SK_ColorWHITE, SK_ColorBLACK,
29 SK_ColorRED, SK_ColorGREEN, SK_ColorBLUE, SK_ColorWHITE, SK_ColorBLACK,
30 SK_ColorRED, SK_ColorGREEN, SK_ColorBLUE, SK_ColorWHITE, SK_ColorBLACK,
31 SK_ColorRED, SK_ColorGREEN, SK_ColorBLUE, SK_ColorWHITE, SK_ColorBLACK,
32 SK_ColorRED, SK_ColorGREEN, SK_ColorBLUE, SK_ColorWHITE, SK_ColorBLACK,
33 SK_ColorRED, SK_ColorGREEN, SK_ColorBLUE, SK_ColorWHITE, SK_ColorBLACK,
34 SK_ColorRED, SK_ColorGREEN, SK_ColorBLUE, SK_ColorWHITE, SK_ColorBLACK,
35 SK_ColorRED, SK_ColorGREEN, SK_ColorBLUE, SK_ColorWHITE, SK_ColorBLACK, // 10 lines, 50 colors
reed@google.com84e9c082011-04-13 17:44:24 +000036};
37
commit-bot@chromium.org3400f4b2013-09-30 19:41:09 +000038// We have several special-cases depending on the number (and spacing) of colors, so
39// try to exercise those here.
reed@google.com84e9c082011-04-13 17:44:24 +000040static const GradData gGradData[] = {
commit-bot@chromium.org5396a042013-07-10 16:13:14 +000041 { 2, gColors, NULL, "" },
42 { 50, gColors, NULL, "_hicolor" }, // many color gradient
commit-bot@chromium.org3400f4b2013-09-30 19:41:09 +000043 { 3, gColors, NULL, "_3color" },
reed@google.com84e9c082011-04-13 17:44:24 +000044};
45
tomhudson@google.com5ea050f2011-09-26 15:03:55 +000046/// Ignores scale
reed@google.com84e9c082011-04-13 17:44:24 +000047static SkShader* MakeLinear(const SkPoint pts[2], const GradData& data,
commit-bot@chromium.org3339ac52014-05-22 02:55:59 +000048 SkShader::TileMode tm, SkUnitMapper* mapper,
49 float scale) {
50 return SkGradientShader::CreateLinear(pts, data.fColors, data.fPos,
51 data.fCount, tm, mapper);
reed@google.com84e9c082011-04-13 17:44:24 +000052}
53
54static SkShader* MakeRadial(const SkPoint pts[2], const GradData& data,
commit-bot@chromium.org3339ac52014-05-22 02:55:59 +000055 SkShader::TileMode tm, SkUnitMapper* mapper,
56 float scale) {
reed@google.com84e9c082011-04-13 17:44:24 +000057 SkPoint center;
58 center.set(SkScalarAve(pts[0].fX, pts[1].fX),
59 SkScalarAve(pts[0].fY, pts[1].fY));
tomhudson@google.com5ea050f2011-09-26 15:03:55 +000060 return SkGradientShader::CreateRadial(center, center.fX * scale,
61 data.fColors,
commit-bot@chromium.org3339ac52014-05-22 02:55:59 +000062 data.fPos, data.fCount, tm, mapper);
reed@google.com84e9c082011-04-13 17:44:24 +000063}
64
tomhudson@google.com5ea050f2011-09-26 15:03:55 +000065/// Ignores scale
reed@google.com84e9c082011-04-13 17:44:24 +000066static SkShader* MakeSweep(const SkPoint pts[2], const GradData& data,
commit-bot@chromium.org3339ac52014-05-22 02:55:59 +000067 SkShader::TileMode tm, SkUnitMapper* mapper,
68 float scale) {
reed@google.com84e9c082011-04-13 17:44:24 +000069 SkPoint center;
70 center.set(SkScalarAve(pts[0].fX, pts[1].fX),
71 SkScalarAve(pts[0].fY, pts[1].fY));
72 return SkGradientShader::CreateSweep(center.fX, center.fY, data.fColors,
commit-bot@chromium.org3339ac52014-05-22 02:55:59 +000073 data.fPos, data.fCount, mapper);
reed@google.com84e9c082011-04-13 17:44:24 +000074}
75
tomhudson@google.com5ea050f2011-09-26 15:03:55 +000076/// Ignores scale
reed@google.com84e9c082011-04-13 17:44:24 +000077static SkShader* Make2Radial(const SkPoint pts[2], const GradData& data,
commit-bot@chromium.org3339ac52014-05-22 02:55:59 +000078 SkShader::TileMode tm, SkUnitMapper* mapper,
79 float scale) {
reed@google.com84e9c082011-04-13 17:44:24 +000080 SkPoint center0, center1;
81 center0.set(SkScalarAve(pts[0].fX, pts[1].fX),
82 SkScalarAve(pts[0].fY, pts[1].fY));
83 center1.set(SkScalarInterp(pts[0].fX, pts[1].fX, SkIntToScalar(3)/5),
84 SkScalarInterp(pts[0].fY, pts[1].fY, SkIntToScalar(1)/4));
85 return SkGradientShader::CreateTwoPointRadial(
reed@google.com258e0bf2012-06-07 14:18:34 +000086 center1, (pts[1].fX - pts[0].fX) / 7,
87 center0, (pts[1].fX - pts[0].fX) / 2,
commit-bot@chromium.org3339ac52014-05-22 02:55:59 +000088 data.fColors, data.fPos, data.fCount, tm, mapper);
reed@google.com258e0bf2012-06-07 14:18:34 +000089}
90
91/// Ignores scale
92static SkShader* MakeConical(const SkPoint pts[2], const GradData& data,
commit-bot@chromium.org3339ac52014-05-22 02:55:59 +000093 SkShader::TileMode tm, SkUnitMapper* mapper,
94 float scale) {
reed@google.com258e0bf2012-06-07 14:18:34 +000095 SkPoint center0, center1;
96 center0.set(SkScalarAve(pts[0].fX, pts[1].fX),
97 SkScalarAve(pts[0].fY, pts[1].fY));
98 center1.set(SkScalarInterp(pts[0].fX, pts[1].fX, SkIntToScalar(3)/5),
99 SkScalarInterp(pts[0].fY, pts[1].fY, SkIntToScalar(1)/4));
100 return SkGradientShader::CreateTwoPointConical(center1, (pts[1].fX - pts[0].fX) / 7,
101 center0, (pts[1].fX - pts[0].fX) / 2,
commit-bot@chromium.org3339ac52014-05-22 02:55:59 +0000102 data.fColors, data.fPos, data.fCount, tm, mapper);
reed@google.com84e9c082011-04-13 17:44:24 +0000103}
104
commit-bot@chromium.org95f79262014-04-14 16:02:31 +0000105/// Ignores scale
106static SkShader* MakeConicalZeroRad(const SkPoint pts[2], const GradData& data,
commit-bot@chromium.org3339ac52014-05-22 02:55:59 +0000107 SkShader::TileMode tm, SkUnitMapper* mapper,
108 float scale) {
commit-bot@chromium.org95f79262014-04-14 16:02:31 +0000109 SkPoint center0, center1;
110 center0.set(SkScalarAve(pts[0].fX, pts[1].fX),
111 SkScalarAve(pts[0].fY, pts[1].fY));
112 center1.set(SkScalarInterp(pts[0].fX, pts[1].fX, SkIntToScalar(3)/5),
113 SkScalarInterp(pts[0].fY, pts[1].fY, SkIntToScalar(1)/4));
114 return SkGradientShader::CreateTwoPointConical(center1, 0.0,
115 center0, (pts[1].fX - pts[0].fX) / 2,
commit-bot@chromium.org3339ac52014-05-22 02:55:59 +0000116 data.fColors, data.fPos, data.fCount, tm, mapper);
commit-bot@chromium.org95f79262014-04-14 16:02:31 +0000117}
118
119/// Ignores scale
120static SkShader* MakeConicalOutside(const SkPoint pts[2], const GradData& data,
commit-bot@chromium.org3339ac52014-05-22 02:55:59 +0000121 SkShader::TileMode tm, SkUnitMapper* mapper,
122 float scale) {
commit-bot@chromium.org95f79262014-04-14 16:02:31 +0000123 SkPoint center0, center1;
124 SkScalar radius0 = SkScalarDiv(pts[1].fX - pts[0].fX, 10);
125 SkScalar radius1 = SkScalarDiv(pts[1].fX - pts[0].fX, 3);
126 center0.set(pts[0].fX + radius0, pts[0].fY + radius0);
127 center1.set(pts[1].fX - radius1, pts[1].fY - radius1);
128 return SkGradientShader::CreateTwoPointConical(center0, radius0,
129 center1, radius1,
130 data.fColors, data.fPos,
commit-bot@chromium.org3339ac52014-05-22 02:55:59 +0000131 data.fCount, tm, mapper);
commit-bot@chromium.org95f79262014-04-14 16:02:31 +0000132}
133
134/// Ignores scale
135static SkShader* MakeConicalOutsideZeroRad(const SkPoint pts[2], const GradData& data,
commit-bot@chromium.org3339ac52014-05-22 02:55:59 +0000136 SkShader::TileMode tm, SkUnitMapper* mapper,
137 float scale) {
commit-bot@chromium.org95f79262014-04-14 16:02:31 +0000138 SkPoint center0, center1;
139 SkScalar radius0 = SkScalarDiv(pts[1].fX - pts[0].fX, 10);
140 SkScalar radius1 = SkScalarDiv(pts[1].fX - pts[0].fX, 3);
141 center0.set(pts[0].fX + radius0, pts[0].fY + radius0);
142 center1.set(pts[1].fX - radius1, pts[1].fY - radius1);
143 return SkGradientShader::CreateTwoPointConical(center0, 0.0,
144 center1, radius1,
145 data.fColors, data.fPos,
commit-bot@chromium.org3339ac52014-05-22 02:55:59 +0000146 data.fCount, tm, mapper);
commit-bot@chromium.org95f79262014-04-14 16:02:31 +0000147}
148
reed@google.com84e9c082011-04-13 17:44:24 +0000149typedef SkShader* (*GradMaker)(const SkPoint pts[2], const GradData& data,
commit-bot@chromium.org3339ac52014-05-22 02:55:59 +0000150 SkShader::TileMode tm, SkUnitMapper* mapper,
151 float scale);
reed@google.com84e9c082011-04-13 17:44:24 +0000152
153static const struct {
154 GradMaker fMaker;
155 const char* fName;
reed@google.com84e9c082011-04-13 17:44:24 +0000156} gGrads[] = {
commit-bot@chromium.org95f79262014-04-14 16:02:31 +0000157 { MakeLinear, "linear" },
158 { MakeRadial, "radial1" },
159 { MakeSweep, "sweep" },
160 { Make2Radial, "radial2" },
161 { MakeConical, "conical" },
162 { MakeConicalZeroRad, "conicalZero" },
163 { MakeConicalOutside, "conicalOut" },
164 { MakeConicalOutsideZeroRad, "conicalOutZero" },
reed@google.com84e9c082011-04-13 17:44:24 +0000165};
166
167enum GradType { // these must match the order in gGrads
168 kLinear_GradType,
169 kRadial_GradType,
170 kSweep_GradType,
reed@google.com258e0bf2012-06-07 14:18:34 +0000171 kRadial2_GradType,
commit-bot@chromium.org95f79262014-04-14 16:02:31 +0000172 kConical_GradType,
173 kConicalZero_GradType,
174 kConicalOut_GradType,
175 kConicalOutZero_GradType
reed@google.com84e9c082011-04-13 17:44:24 +0000176};
177
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000178enum GeomType {
179 kRect_GeomType,
180 kOval_GeomType
181};
182
reed@google.com72415162011-06-15 21:17:37 +0000183static const char* tilemodename(SkShader::TileMode tm) {
184 switch (tm) {
185 case SkShader::kClamp_TileMode:
186 return "clamp";
187 case SkShader::kRepeat_TileMode:
188 return "repeat";
189 case SkShader::kMirror_TileMode:
190 return "mirror";
191 default:
mtklein@google.com330313a2013-08-22 15:37:26 +0000192 SkDEBUGFAIL("unknown tilemode");
reed@google.com72415162011-06-15 21:17:37 +0000193 return "error";
194 }
195}
196
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000197static const char* geomtypename(GeomType gt) {
198 switch (gt) {
199 case kRect_GeomType:
200 return "rectangle";
201 case kOval_GeomType:
202 return "oval";
203 default:
mtklein@google.com330313a2013-08-22 15:37:26 +0000204 SkDEBUGFAIL("unknown geometry type");
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000205 return "error";
206 }
207}
208
reed@google.com84e9c082011-04-13 17:44:24 +0000209///////////////////////////////////////////////////////////////////////////////
210
211class GradientBench : public SkBenchmark {
212 SkString fName;
213 SkShader* fShader;
reed@google.com84e9c082011-04-13 17:44:24 +0000214 enum {
215 W = 400,
216 H = 400,
commit-bot@chromium.orgb120bd92013-10-07 17:18:21 +0000217 kRepeat = 15,
reed@google.com84e9c082011-04-13 17:44:24 +0000218 };
219public:
mtklein@google.com410e6e82013-09-13 19:52:27 +0000220 GradientBench(GradType gradType,
commit-bot@chromium.org5396a042013-07-10 16:13:14 +0000221 GradData data = gGradData[0],
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000222 SkShader::TileMode tm = SkShader::kClamp_TileMode,
223 GeomType geomType = kRect_GeomType,
commit-bot@chromium.org3400f4b2013-09-30 19:41:09 +0000224 float scale = 1.0f) {
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000225 fName.printf("gradient_%s_%s", gGrads[gradType].fName,
226 tilemodename(tm));
227 if (geomType != kRect_GeomType) {
228 fName.append("_");
229 fName.append(geomtypename(geomType));
230 }
reed@google.com84e9c082011-04-13 17:44:24 +0000231
commit-bot@chromium.org3400f4b2013-09-30 19:41:09 +0000232 if (scale != 1.f) {
233 fName.appendf("_scale_%g", scale);
234 }
235
commit-bot@chromium.org5396a042013-07-10 16:13:14 +0000236 fName.append(data.fName);
237
reed@google.com84e9c082011-04-13 17:44:24 +0000238 const SkPoint pts[2] = {
239 { 0, 0 },
240 { SkIntToScalar(W), SkIntToScalar(H) }
241 };
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000242
commit-bot@chromium.org3339ac52014-05-22 02:55:59 +0000243 fShader = gGrads[gradType].fMaker(pts, data, tm, NULL, scale);
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000244 fGeomType = geomType;
reed@google.com84e9c082011-04-13 17:44:24 +0000245 }
246
247 virtual ~GradientBench() {
248 fShader->unref();
249 }
250
251protected:
252 virtual const char* onGetName() {
253 return fName.c_str();
254 }
255
commit-bot@chromium.org33614712013-12-03 18:17:16 +0000256 virtual void onDraw(const int loops, SkCanvas* canvas) {
reed@google.com84e9c082011-04-13 17:44:24 +0000257 SkPaint paint;
258 this->setupPaint(&paint);
259
260 paint.setShader(fShader);
261
262 SkRect r = { 0, 0, SkIntToScalar(W), SkIntToScalar(H) };
commit-bot@chromium.org33614712013-12-03 18:17:16 +0000263 for (int i = 0; i < loops * kRepeat; i++) {
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000264 switch (fGeomType) {
265 case kRect_GeomType:
266 canvas->drawRect(r, paint);
267 break;
268 case kOval_GeomType:
269 canvas->drawOval(r, paint);
270 break;
271 }
reed@google.com84e9c082011-04-13 17:44:24 +0000272 }
273 }
274
275private:
276 typedef SkBenchmark INHERITED;
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000277
278 GeomType fGeomType;
reed@google.com84e9c082011-04-13 17:44:24 +0000279};
280
commit-bot@chromium.orgb120bd92013-10-07 17:18:21 +0000281DEF_BENCH( return new GradientBench(kLinear_GradType); )
282DEF_BENCH( return new GradientBench(kLinear_GradType, gGradData[1]); )
283DEF_BENCH( return new GradientBench(kLinear_GradType, gGradData[2]); )
284DEF_BENCH( return new GradientBench(kLinear_GradType, gGradData[0], SkShader::kMirror_TileMode); )
285
286DEF_BENCH( return new GradientBench(kRadial_GradType, gGradData[0]); )
287DEF_BENCH( return new GradientBench(kRadial_GradType, gGradData[1]); )
288DEF_BENCH( return new GradientBench(kRadial_GradType, gGradData[2]); )
289// Draw a radial gradient of radius 1/2 on a rectangle; half the lines should
290// be completely pinned, the other half should pe partially pinned
291DEF_BENCH( return new GradientBench(kRadial_GradType, gGradData[0], SkShader::kClamp_TileMode, kRect_GeomType, 0.5f); )
292
293// Draw a radial gradient on a circle of equal size; all the lines should
294// hit the unpinned fast path (so long as GradientBench.W == H)
295DEF_BENCH( return new GradientBench(kRadial_GradType, gGradData[0], SkShader::kClamp_TileMode, kOval_GeomType); )
296
297DEF_BENCH( return new GradientBench(kRadial_GradType, gGradData[0], SkShader::kMirror_TileMode); )
commit-bot@chromium.org34150b42013-10-16 18:59:44 +0000298DEF_BENCH( return new GradientBench(kRadial_GradType, gGradData[0], SkShader::kRepeat_TileMode); )
commit-bot@chromium.orgb120bd92013-10-07 17:18:21 +0000299DEF_BENCH( return new GradientBench(kSweep_GradType); )
300DEF_BENCH( return new GradientBench(kSweep_GradType, gGradData[1]); )
301DEF_BENCH( return new GradientBench(kSweep_GradType, gGradData[2]); )
302DEF_BENCH( return new GradientBench(kRadial2_GradType); )
303DEF_BENCH( return new GradientBench(kRadial2_GradType, gGradData[1]); )
304DEF_BENCH( return new GradientBench(kRadial2_GradType, gGradData[0], SkShader::kMirror_TileMode); )
305DEF_BENCH( return new GradientBench(kConical_GradType); )
306DEF_BENCH( return new GradientBench(kConical_GradType, gGradData[1]); )
307DEF_BENCH( return new GradientBench(kConical_GradType, gGradData[2]); )
commit-bot@chromium.org95f79262014-04-14 16:02:31 +0000308DEF_BENCH( return new GradientBench(kConicalZero_GradType); )
309DEF_BENCH( return new GradientBench(kConicalZero_GradType, gGradData[1]); )
310DEF_BENCH( return new GradientBench(kConicalZero_GradType, gGradData[2]); )
311DEF_BENCH( return new GradientBench(kConicalOut_GradType); )
312DEF_BENCH( return new GradientBench(kConicalOut_GradType, gGradData[1]); )
313DEF_BENCH( return new GradientBench(kConicalOut_GradType, gGradData[2]); )
314DEF_BENCH( return new GradientBench(kConicalOutZero_GradType); )
315DEF_BENCH( return new GradientBench(kConicalOutZero_GradType, gGradData[1]); )
316DEF_BENCH( return new GradientBench(kConicalOutZero_GradType, gGradData[2]); )
commit-bot@chromium.orgb120bd92013-10-07 17:18:21 +0000317
318///////////////////////////////////////////////////////////////////////////////
319
reed@google.com8ac7a0f2011-06-16 13:14:19 +0000320class Gradient2Bench : public SkBenchmark {
reed@google.com73349aa2013-02-04 18:09:58 +0000321 SkString fName;
322 bool fHasAlpha;
323
reed@google.com8ac7a0f2011-06-16 13:14:19 +0000324public:
mtklein@google.com410e6e82013-09-13 19:52:27 +0000325 Gradient2Bench(bool hasAlpha) {
reed@google.com73349aa2013-02-04 18:09:58 +0000326 fName.printf("gradient_create_%s", hasAlpha ? "alpha" : "opaque");
327 fHasAlpha = hasAlpha;
328 }
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000329
reed@google.com8ac7a0f2011-06-16 13:14:19 +0000330protected:
331 virtual const char* onGetName() {
reed@google.com73349aa2013-02-04 18:09:58 +0000332 return fName.c_str();
reed@google.com8ac7a0f2011-06-16 13:14:19 +0000333 }
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000334
commit-bot@chromium.org33614712013-12-03 18:17:16 +0000335 virtual void onDraw(const int loops, SkCanvas* canvas) {
reed@google.com8ac7a0f2011-06-16 13:14:19 +0000336 SkPaint paint;
337 this->setupPaint(&paint);
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000338
reed@google.com8ac7a0f2011-06-16 13:14:19 +0000339 const SkRect r = { 0, 0, SkIntToScalar(4), SkIntToScalar(4) };
reed@google.com1ca4f262011-06-24 19:17:37 +0000340 const SkPoint pts[] = {
341 { 0, 0 },
342 { SkIntToScalar(100), SkIntToScalar(100) },
343 };
reed@google.com8ac7a0f2011-06-16 13:14:19 +0000344
commit-bot@chromium.org33614712013-12-03 18:17:16 +0000345 for (int i = 0; i < loops; i++) {
reed@google.com73349aa2013-02-04 18:09:58 +0000346 const int gray = i % 256;
347 const int alpha = fHasAlpha ? gray : 0xFF;
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000348 SkColor colors[] = {
349 SK_ColorBLACK,
reed@google.com73349aa2013-02-04 18:09:58 +0000350 SkColorSetARGB(alpha, gray, gray, gray),
tomhudson@google.com5ea050f2011-09-26 15:03:55 +0000351 SK_ColorWHITE };
reed@google.com8ac7a0f2011-06-16 13:14:19 +0000352 SkShader* s = SkGradientShader::CreateLinear(pts, colors, NULL,
353 SK_ARRAY_COUNT(colors),
354 SkShader::kClamp_TileMode);
355 paint.setShader(s)->unref();
356 canvas->drawRect(r, paint);
357 }
358 }
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000359
reed@google.com8ac7a0f2011-06-16 13:14:19 +0000360private:
361 typedef SkBenchmark INHERITED;
362};
363
mtklein@google.com410e6e82013-09-13 19:52:27 +0000364DEF_BENCH( return new Gradient2Bench(false); )
365DEF_BENCH( return new Gradient2Bench(true); )