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 "SkCommandLineFlags.h" 12 #include "SkGradientShader.h" 13 #include "SkPaint.h" 14 #include "SkRandom.h" 15 #include "SkShader.h" 16 #include "SkString.h" 17 18 DEFINE_double(strokeWidth, -1.0, "If set, use this stroke width in RectBench."); 19 20 class RectBench : public Benchmark { 21 public: 22 int fShift, fStroke; 23 enum { 24 W = 640, 25 H = 480, 26 N = 300, 27 }; 28 SkRect fRects[N]; 29 SkColor fColors[N]; 30 bool fAA; 31 bool fPerspective; 32 RectBench(int shift,int stroke=0,bool aa=true,bool perspective=false)33 RectBench(int shift, int stroke = 0, bool aa = true, bool perspective = false) 34 : fShift(shift) 35 , fStroke(stroke) 36 , fAA(aa) 37 , fPerspective(perspective) {} 38 computeName(const char root[])39 const char* computeName(const char root[]) { 40 fBaseName.printf("%s_%d", root, fShift); 41 if (fStroke > 0) { 42 fBaseName.appendf("_stroke_%d", fStroke); 43 } 44 if (fAA) { 45 fBaseName.appendf("_aa"); 46 } else { 47 fBaseName.appendf("_bw"); 48 } 49 if (fPerspective) { 50 fBaseName.appendf("_persp"); 51 } 52 return fBaseName.c_str(); 53 } 54 55 protected: 56 drawThisRect(SkCanvas * c,const SkRect & r,const SkPaint & p)57 virtual void drawThisRect(SkCanvas* c, const SkRect& r, const SkPaint& p) { 58 c->drawRect(r, p); 59 } 60 onGetName()61 const char* onGetName() override { return computeName("rects"); } 62 onDelayedSetup()63 void onDelayedSetup() override { 64 SkRandom rand; 65 const SkScalar offset = SK_Scalar1/3; 66 for (int i = 0; i < N; i++) { 67 int x = rand.nextU() % W; 68 int y = rand.nextU() % H; 69 int w = rand.nextU() % W; 70 int h = rand.nextU() % H; 71 w >>= fShift; 72 h >>= fShift; 73 x -= w/2; 74 y -= h/2; 75 fRects[i].set(SkIntToScalar(x), SkIntToScalar(y), 76 SkIntToScalar(x+w), SkIntToScalar(y+h)); 77 fRects[i].offset(offset, offset); 78 fColors[i] = rand.nextU() | 0xFF808080; 79 } 80 } 81 onDraw(int loops,SkCanvas * canvas)82 void onDraw(int loops, SkCanvas* canvas) override { 83 SkPaint paint; 84 if (fStroke > 0) { 85 paint.setStyle(SkPaint::kStroke_Style); 86 paint.setStrokeWidth(SkIntToScalar(fStroke)); 87 } 88 if (fPerspective) { 89 // Apply some fixed perspective to change how ops may draw the rects 90 SkMatrix perspective; 91 perspective.setIdentity(); 92 perspective.setPerspX(1e-4f); 93 perspective.setPerspY(1e-3f); 94 perspective.setSkewX(0.1f); 95 canvas->concat(perspective); 96 } 97 for (int i = 0; i < loops; i++) { 98 paint.setColor(fColors[i % N]); 99 this->setupPaint(&paint); 100 this->drawThisRect(canvas, fRects[i % N], paint); 101 } 102 } 103 setupPaint(SkPaint * paint)104 void setupPaint(SkPaint* paint) override { 105 this->INHERITED::setupPaint(paint); 106 paint->setAntiAlias(fAA); 107 } 108 109 private: 110 SkString fBaseName; 111 typedef Benchmark INHERITED; 112 }; 113 114 class SrcModeRectBench : public RectBench { 115 public: SrcModeRectBench()116 SrcModeRectBench() : INHERITED(1, 0) { 117 fMode = SkBlendMode::kSrc; 118 } 119 120 protected: setupPaint(SkPaint * paint)121 void setupPaint(SkPaint* paint) override { 122 this->INHERITED::setupPaint(paint); 123 // srcmode is most interesting when we're not opaque 124 paint->setAlpha(0x80); 125 paint->setBlendMode(fMode); 126 } 127 onGetName()128 const char* onGetName() override { 129 fName.set(this->INHERITED::onGetName()); 130 fName.prepend("srcmode_"); 131 return fName.c_str(); 132 } 133 134 private: 135 SkBlendMode fMode; 136 SkString fName; 137 138 typedef RectBench INHERITED; 139 }; 140 141 class TransparentRectBench : public RectBench { 142 public: TransparentRectBench()143 TransparentRectBench() : INHERITED(1, 0) {} 144 145 protected: setupPaint(SkPaint * paint)146 void setupPaint(SkPaint* paint) override { 147 this->INHERITED::setupPaint(paint); 148 // draw non opaque rect 149 paint->setAlpha(0x80); 150 } 151 onGetName()152 const char* onGetName() override { 153 fName.set(this->INHERITED::onGetName()); 154 fName.prepend("transparent_"); 155 return fName.c_str(); 156 } 157 158 private: 159 SkString fName; 160 typedef RectBench INHERITED; 161 }; 162 163 // Adds a shader to the paint that requires local coordinates to be used 164 class LocalCoordsRectBench : public RectBench { 165 public: LocalCoordsRectBench(bool aa,bool perspective=false)166 LocalCoordsRectBench(bool aa, bool perspective = false) : INHERITED(1, 0, aa, perspective) { } 167 168 protected: onDelayedSetup()169 void onDelayedSetup() override { 170 this->INHERITED::onDelayedSetup(); 171 // Create the shader once, so that isn't included in the timing 172 SkPoint pts[2] = { {0.f, 0.f}, {50.f, 50.f} }; 173 SkColor colors[] = { SK_ColorWHITE, SK_ColorBLUE }; 174 fShader = SkGradientShader::MakeLinear(pts, colors, nullptr, 2, SkShader::kClamp_TileMode); 175 } 176 setupPaint(SkPaint * paint)177 void setupPaint(SkPaint* paint) override { 178 this->INHERITED::setupPaint(paint); 179 paint->setShader(fShader); 180 } 181 onGetName()182 const char* onGetName() override { 183 fName.set(this->INHERITED::onGetName()); 184 fName.append("_localcoords"); 185 return fName.c_str(); 186 } 187 188 private: 189 SkString fName; 190 sk_sp<SkShader> fShader; 191 192 typedef RectBench INHERITED; 193 }; 194 195 196 class OvalBench : public RectBench { 197 public: OvalBench(int shift,int stroke=0)198 OvalBench(int shift, int stroke = 0) : RectBench(shift, stroke) {} 199 protected: drawThisRect(SkCanvas * c,const SkRect & r,const SkPaint & p)200 void drawThisRect(SkCanvas* c, const SkRect& r, const SkPaint& p) override { 201 c->drawOval(r, p); 202 } onGetName()203 const char* onGetName() override { return computeName("ovals"); } 204 }; 205 206 class RRectBench : public RectBench { 207 public: RRectBench(int shift,int stroke=0)208 RRectBench(int shift, int stroke = 0) : RectBench(shift, stroke) {} 209 protected: drawThisRect(SkCanvas * c,const SkRect & r,const SkPaint & p)210 void drawThisRect(SkCanvas* c, const SkRect& r, const SkPaint& p) override { 211 c->drawRoundRect(r, r.width() / 4, r.height() / 4, p); 212 } onGetName()213 const char* onGetName() override { return computeName("rrects"); } 214 }; 215 216 class PointsBench : public RectBench { 217 public: 218 SkCanvas::PointMode fMode; 219 PointsBench(SkCanvas::PointMode mode,const char * name)220 PointsBench(SkCanvas::PointMode mode, const char* name) 221 : RectBench(2) 222 , fMode(mode) { 223 fName = name; 224 } 225 226 protected: onDraw(int loops,SkCanvas * canvas)227 void onDraw(int loops, SkCanvas* canvas) override { 228 SkScalar gSizes[] = { 229 SkIntToScalar(7), 0 230 }; 231 size_t sizes = SK_ARRAY_COUNT(gSizes); 232 233 if (FLAGS_strokeWidth >= 0) { 234 gSizes[0] = (SkScalar)FLAGS_strokeWidth; 235 sizes = 1; 236 } 237 238 SkPaint paint; 239 paint.setStrokeCap(SkPaint::kRound_Cap); 240 241 for (int loop = 0; loop < loops; loop++) { 242 for (size_t i = 0; i < sizes; i++) { 243 paint.setStrokeWidth(gSizes[i]); 244 this->setupPaint(&paint); 245 canvas->drawPoints(fMode, N * 2, reinterpret_cast<SkPoint*>(fRects), paint); 246 paint.setColor(fColors[i % N]); 247 } 248 } 249 } onGetName()250 const char* onGetName() override { return fName.c_str(); } 251 252 private: 253 SkString fName; 254 255 }; 256 257 /******************************************************************************* 258 * to bench BlitMask [Opaque, Black, color, shader] 259 *******************************************************************************/ 260 261 class BlitMaskBench : public RectBench { 262 public: 263 enum kMaskType { 264 kMaskOpaque = 0, 265 kMaskBlack, 266 kMaskColor, 267 KMaskShader 268 }; 269 SkCanvas::PointMode fMode; 270 BlitMaskBench(SkCanvas::PointMode mode,BlitMaskBench::kMaskType type,const char * name)271 BlitMaskBench(SkCanvas::PointMode mode, 272 BlitMaskBench::kMaskType type, const char* name) : 273 RectBench(2), fMode(mode), _type(type) { 274 fName = name; 275 } 276 277 protected: onDraw(int loops,SkCanvas * canvas)278 void onDraw(int loops, SkCanvas* canvas) override { 279 SkScalar gSizes[] = { 280 SkIntToScalar(13), SkIntToScalar(24) 281 }; 282 size_t sizes = SK_ARRAY_COUNT(gSizes); 283 284 if (FLAGS_strokeWidth >= 0) { 285 gSizes[0] = (SkScalar)FLAGS_strokeWidth; 286 sizes = 1; 287 } 288 SkRandom rand; 289 SkColor color = 0xFF000000; 290 U8CPU alpha = 0xFF; 291 SkPaint paint; 292 paint.setStrokeCap(SkPaint::kRound_Cap); 293 if (_type == KMaskShader) { 294 SkBitmap srcBM; 295 srcBM.allocN32Pixels(10, 1); 296 srcBM.eraseColor(0xFF00FF00); 297 298 paint.setShader(SkShader::MakeBitmapShader(srcBM, SkShader::kClamp_TileMode, 299 SkShader::kClamp_TileMode)); 300 } 301 for (int loop = 0; loop < loops; loop++) { 302 for (size_t i = 0; i < sizes; i++) { 303 switch (_type) { 304 case kMaskOpaque: 305 color = fColors[i]; 306 alpha = 0xFF; 307 break; 308 case kMaskBlack: 309 alpha = 0xFF; 310 color = 0xFF000000; 311 break; 312 case kMaskColor: 313 color = fColors[i]; 314 alpha = rand.nextU() & 255; 315 break; 316 case KMaskShader: 317 break; 318 } 319 paint.setStrokeWidth(gSizes[i]); 320 this->setupPaint(&paint); 321 paint.setColor(color); 322 paint.setAlpha(alpha); 323 canvas->drawPoints(fMode, N * 2, reinterpret_cast<SkPoint*>(fRects), paint); 324 } 325 } 326 } onGetName()327 const char* onGetName() override { return fName.c_str(); } 328 329 private: 330 typedef RectBench INHERITED; 331 kMaskType _type; 332 SkString fName; 333 }; 334 335 // AA rects 336 DEF_BENCH(return new RectBench(1, 0, true);) 337 DEF_BENCH(return new RectBench(1, 4, true);) 338 DEF_BENCH(return new RectBench(3, 0, true);) 339 DEF_BENCH(return new RectBench(3, 4, true);) 340 // Non-AA rects 341 DEF_BENCH(return new RectBench(1, 0, false);) 342 DEF_BENCH(return new RectBench(1, 4, false);) 343 DEF_BENCH(return new RectBench(3, 0, false);) 344 DEF_BENCH(return new RectBench(3, 4, false);) 345 346 DEF_BENCH(return new OvalBench(1);) 347 DEF_BENCH(return new OvalBench(3);) 348 DEF_BENCH(return new OvalBench(1, 4);) 349 DEF_BENCH(return new OvalBench(3, 4);) 350 DEF_BENCH(return new RRectBench(1);) 351 DEF_BENCH(return new RRectBench(1, 4);) 352 DEF_BENCH(return new RRectBench(3);) 353 DEF_BENCH(return new RRectBench(3, 4);) 354 DEF_BENCH(return new PointsBench(SkCanvas::kPoints_PointMode, "points");) 355 DEF_BENCH(return new PointsBench(SkCanvas::kLines_PointMode, "lines");) 356 DEF_BENCH(return new PointsBench(SkCanvas::kPolygon_PointMode, "polygon");) 357 358 DEF_BENCH(return new SrcModeRectBench();) 359 360 DEF_BENCH(return new TransparentRectBench();) 361 362 DEF_BENCH(return new LocalCoordsRectBench(true);) 363 DEF_BENCH(return new LocalCoordsRectBench(false);) 364 365 // Perspective rects 366 DEF_BENCH(return new RectBench(1, 0, true, true);) 367 DEF_BENCH(return new RectBench(1, 0, false, true);) 368 DEF_BENCH(return new LocalCoordsRectBench(true, true);) 369 DEF_BENCH(return new LocalCoordsRectBench(false, true);) 370 371 /* init the blitmask bench 372 */ 373 DEF_BENCH(return new BlitMaskBench(SkCanvas::kPoints_PointMode, 374 BlitMaskBench::kMaskOpaque, 375 "maskopaque");) 376 DEF_BENCH(return new BlitMaskBench(SkCanvas::kPoints_PointMode, 377 BlitMaskBench::kMaskBlack, 378 "maskblack");) 379 DEF_BENCH(return new BlitMaskBench(SkCanvas::kPoints_PointMode, 380 BlitMaskBench::kMaskColor, 381 "maskcolor");) 382 DEF_BENCH(return new BlitMaskBench(SkCanvas::kPoints_PointMode, 383 BlitMaskBench::KMaskShader, 384 "maskshader");) 385