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 "SkDashPathEffect.h" 12 #include "SkPaint.h" 13 #include "SkPath.h" 14 #include "SkRandom.h" 15 #include "SkString.h" 16 #include "SkTDArray.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 */ 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 SkBenchmark { 33 protected: 34 SkString fName; 35 SkTDArray<SkScalar> fIntervals; 36 int fWidth; 37 SkPoint fPts[2]; 38 bool fDoClip; 39 40 public: 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 55 virtual void makePath(SkPath* path) { 56 path_hline(path); 57 } 58 59 protected: 60 virtual const char* onGetName() SK_OVERRIDE { 61 return fName.c_str(); 62 } 63 64 virtual void onDraw(const int loops, SkCanvas* canvas) SK_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(new SkDashPathEffect(fIntervals.begin(), 75 fIntervals.count(), 0))->unref(); 76 77 if (fDoClip) { 78 SkRect r = path.getBounds(); 79 r.inset(-SkIntToScalar(20), -SkIntToScalar(20)); 80 // now move it so we don't intersect 81 r.offset(0, r.height() * 3 / 2); 82 canvas->clipRect(r); 83 } 84 85 this->handlePath(canvas, path, paint, loops); 86 } 87 88 virtual void handlePath(SkCanvas* canvas, const SkPath& path, 89 const SkPaint& paint, int N) { 90 for (int i = 0; i < N; ++i) { 91 // canvas->drawPoints(SkCanvas::kLines_PointMode, 2, fPts, paint); 92 canvas->drawPath(path, paint); 93 } 94 } 95 96 private: 97 typedef SkBenchmark INHERITED; 98 }; 99 100 class RectDashBench : public DashBench { 101 public: 102 RectDashBench(const SkScalar intervals[], int count, int width) 103 : INHERITED(intervals, count, width) { 104 fName.append("_rect"); 105 } 106 107 protected: 108 virtual void handlePath(SkCanvas* canvas, const SkPath& path, 109 const SkPaint& paint, int N) SK_OVERRIDE { 110 SkPoint pts[2]; 111 if (!path.isLine(pts) || pts[0].fY != pts[1].fY) { 112 this->INHERITED::handlePath(canvas, path, paint, N); 113 } else { 114 SkRect rect; 115 rect.fLeft = pts[0].fX; 116 rect.fTop = pts[0].fY - paint.getStrokeWidth() / 2; 117 rect.fRight = rect.fLeft + SkIntToScalar(fWidth); 118 rect.fBottom = rect.fTop + paint.getStrokeWidth(); 119 120 SkPaint p(paint); 121 p.setStyle(SkPaint::kFill_Style); 122 p.setPathEffect(NULL); 123 124 int count = SkScalarRoundToInt((pts[1].fX - pts[0].fX) / (2*fWidth)); 125 SkScalar dx = SkIntToScalar(2 * fWidth); 126 127 for (int i = 0; i < N*10; ++i) { 128 SkRect r = rect; 129 for (int j = 0; j < count; ++j) { 130 canvas->drawRect(r, p); 131 r.offset(dx, 0); 132 } 133 } 134 } 135 } 136 137 private: 138 typedef DashBench INHERITED; 139 }; 140 141 static void make_unit_star(SkPath* path, int n) { 142 SkScalar rad = -SK_ScalarPI / 2; 143 const SkScalar drad = (n >> 1) * SK_ScalarPI * 2 / n; 144 145 path->moveTo(0, -SK_Scalar1); 146 for (int i = 1; i < n; i++) { 147 rad += drad; 148 SkScalar cosV, sinV = SkScalarSinCos(rad, &cosV); 149 path->lineTo(cosV, sinV); 150 } 151 path->close(); 152 } 153 154 static void make_poly(SkPath* path) { 155 make_unit_star(path, 9); 156 SkMatrix matrix; 157 matrix.setScale(SkIntToScalar(100), SkIntToScalar(100)); 158 path->transform(matrix); 159 } 160 161 static void make_quad(SkPath* path) { 162 SkScalar x0 = SkIntToScalar(10); 163 SkScalar y0 = SkIntToScalar(10); 164 path->moveTo(x0, y0); 165 path->quadTo(x0, y0 + 400 * SK_Scalar1, 166 x0 + 600 * SK_Scalar1, y0 + 400 * SK_Scalar1); 167 } 168 169 static void make_cubic(SkPath* path) { 170 SkScalar x0 = SkIntToScalar(10); 171 SkScalar y0 = SkIntToScalar(10); 172 path->moveTo(x0, y0); 173 path->cubicTo(x0, y0 + 400 * SK_Scalar1, 174 x0 + 600 * SK_Scalar1, y0 + 400 * SK_Scalar1, 175 x0 + 600 * SK_Scalar1, y0); 176 } 177 178 class MakeDashBench : public SkBenchmark { 179 SkString fName; 180 SkPath fPath; 181 SkAutoTUnref<SkPathEffect> fPE; 182 183 public: 184 MakeDashBench(void (*proc)(SkPath*), const char name[]) { 185 fName.printf("makedash_%s", name); 186 proc(&fPath); 187 188 SkScalar vals[] = { SkIntToScalar(4), SkIntToScalar(4) }; 189 fPE.reset(new SkDashPathEffect(vals, 2, 0)); 190 } 191 192 protected: 193 virtual const char* onGetName() SK_OVERRIDE { 194 return fName.c_str(); 195 } 196 197 virtual void onDraw(const int loops, SkCanvas*) SK_OVERRIDE { 198 SkPath dst; 199 for (int i = 0; i < loops; ++i) { 200 SkStrokeRec rec(SkStrokeRec::kHairline_InitStyle); 201 202 fPE->filterPath(&dst, fPath, &rec, NULL); 203 dst.rewind(); 204 } 205 } 206 207 private: 208 typedef SkBenchmark INHERITED; 209 }; 210 211 /* 212 * We try to special case square dashes (intervals are equal to strokewidth). 213 */ 214 class DashLineBench : public SkBenchmark { 215 SkString fName; 216 SkScalar fStrokeWidth; 217 bool fIsRound; 218 SkAutoTUnref<SkPathEffect> fPE; 219 220 public: 221 DashLineBench(SkScalar width, bool isRound) { 222 fName.printf("dashline_%g_%s", SkScalarToFloat(width), isRound ? "circle" : "square"); 223 fStrokeWidth = width; 224 fIsRound = isRound; 225 226 SkScalar vals[] = { SK_Scalar1, SK_Scalar1 }; 227 fPE.reset(new SkDashPathEffect(vals, 2, 0)); 228 } 229 230 protected: 231 virtual const char* onGetName() SK_OVERRIDE { 232 return fName.c_str(); 233 } 234 235 virtual void onDraw(const int loops, SkCanvas* canvas) SK_OVERRIDE { 236 SkPaint paint; 237 this->setupPaint(&paint); 238 paint.setStrokeWidth(fStrokeWidth); 239 paint.setStrokeCap(fIsRound ? SkPaint::kRound_Cap : SkPaint::kSquare_Cap); 240 paint.setPathEffect(fPE); 241 for (int i = 0; i < loops; ++i) { 242 canvas->drawLine(10 * SK_Scalar1, 10 * SK_Scalar1, 243 640 * SK_Scalar1, 10 * SK_Scalar1, paint); 244 } 245 } 246 247 private: 248 typedef SkBenchmark INHERITED; 249 }; 250 251 class DrawPointsDashingBench : public SkBenchmark { 252 SkString fName; 253 int fStrokeWidth; 254 bool fDoAA; 255 256 SkAutoTUnref<SkPathEffect> fPathEffect; 257 258 public: 259 DrawPointsDashingBench(int dashLength, int strokeWidth, bool doAA) 260 { 261 fName.printf("drawpointsdash_%d_%d%s", dashLength, strokeWidth, doAA ? "_aa" : "_bw"); 262 fStrokeWidth = strokeWidth; 263 fDoAA = doAA; 264 265 SkScalar vals[] = { SkIntToScalar(dashLength), SkIntToScalar(dashLength) }; 266 fPathEffect.reset(new SkDashPathEffect(vals, 2, SK_Scalar1, false)); 267 } 268 269 protected: 270 virtual const char* onGetName() SK_OVERRIDE { 271 return fName.c_str(); 272 } 273 274 virtual void onDraw(const int loops, SkCanvas* canvas) SK_OVERRIDE { 275 SkPaint p; 276 this->setupPaint(&p); 277 p.setColor(SK_ColorBLACK); 278 p.setStyle(SkPaint::kStroke_Style); 279 p.setStrokeWidth(SkIntToScalar(fStrokeWidth)); 280 p.setPathEffect(fPathEffect); 281 p.setAntiAlias(fDoAA); 282 283 SkPoint pts[2] = { 284 { SkIntToScalar(10), 0 }, 285 { SkIntToScalar(640), 0 } 286 }; 287 288 for (int i = 0; i < loops; ++i) { 289 pts[0].fY = pts[1].fY = SkIntToScalar(i % 480); 290 canvas->drawPoints(SkCanvas::kLines_PointMode, 2, pts, p); 291 } 292 } 293 294 private: 295 typedef SkBenchmark INHERITED; 296 }; 297 298 // Want to test how we handle dashing when 99% of the dash is clipped out 299 class GiantDashBench : public SkBenchmark { 300 SkString fName; 301 SkScalar fStrokeWidth; 302 SkPoint fPts[2]; 303 SkAutoTUnref<SkPathEffect> fPathEffect; 304 305 public: 306 enum LineType { 307 kHori_LineType, 308 kVert_LineType, 309 kDiag_LineType, 310 kLineTypeCount 311 }; 312 313 static const char* LineTypeName(LineType lt) { 314 static const char* gNames[] = { "hori", "vert", "diag" }; 315 SK_COMPILE_ASSERT(kLineTypeCount == SK_ARRAY_COUNT(gNames), names_wrong_size); 316 return gNames[lt]; 317 } 318 319 GiantDashBench(LineType lt, SkScalar width) { 320 fName.printf("giantdashline_%s_%g", LineTypeName(lt), width); 321 fStrokeWidth = width; 322 323 // deliberately pick intervals that won't be caught by asPoints(), so 324 // we can test the filterPath code-path. 325 const SkScalar intervals[] = { 2, 1, 1, 1 }; 326 fPathEffect.reset(new SkDashPathEffect(intervals, 327 SK_ARRAY_COUNT(intervals), 0)); 328 329 SkScalar cx = 640 / 2; // center X 330 SkScalar cy = 480 / 2; // center Y 331 SkMatrix matrix; 332 333 switch (lt) { 334 case kHori_LineType: 335 matrix.setIdentity(); 336 break; 337 case kVert_LineType: 338 matrix.setRotate(90, cx, cy); 339 break; 340 case kDiag_LineType: 341 matrix.setRotate(45, cx, cy); 342 break; 343 case kLineTypeCount: 344 // Not a real enum value. 345 break; 346 } 347 348 const SkScalar overshoot = 100*1000; 349 const SkPoint pts[2] = { 350 { -overshoot, cy }, { 640 + overshoot, cy } 351 }; 352 matrix.mapPoints(fPts, pts, 2); 353 } 354 355 protected: 356 virtual const char* onGetName() SK_OVERRIDE { 357 return fName.c_str(); 358 } 359 360 virtual void onDraw(const int loops, SkCanvas* canvas) SK_OVERRIDE { 361 SkPaint p; 362 this->setupPaint(&p); 363 p.setStyle(SkPaint::kStroke_Style); 364 p.setStrokeWidth(fStrokeWidth); 365 p.setPathEffect(fPathEffect); 366 367 for (int i = 0; i < loops; i++) { 368 canvas->drawPoints(SkCanvas::kLines_PointMode, 2, fPts, p); 369 } 370 } 371 372 private: 373 typedef SkBenchmark INHERITED; 374 }; 375 376 377 /////////////////////////////////////////////////////////////////////////////// 378 379 static const SkScalar gDots[] = { SK_Scalar1, SK_Scalar1 }; 380 381 #define PARAM(array) array, SK_ARRAY_COUNT(array) 382 383 DEF_BENCH( return new DashBench(PARAM(gDots), 0); ) 384 DEF_BENCH( return new DashBench(PARAM(gDots), 1); ) 385 DEF_BENCH( return new DashBench(PARAM(gDots), 1, true); ) 386 DEF_BENCH( return new DashBench(PARAM(gDots), 4); ) 387 DEF_BENCH( return new MakeDashBench(make_poly, "poly"); ) 388 DEF_BENCH( return new MakeDashBench(make_quad, "quad"); ) 389 DEF_BENCH( return new MakeDashBench(make_cubic, "cubic"); ) 390 DEF_BENCH( return new DashLineBench(0, false); ) 391 DEF_BENCH( return new DashLineBench(SK_Scalar1, false); ) 392 DEF_BENCH( return new DashLineBench(2 * SK_Scalar1, false); ) 393 DEF_BENCH( return new DashLineBench(0, true); ) 394 DEF_BENCH( return new DashLineBench(SK_Scalar1, true); ) 395 DEF_BENCH( return new DashLineBench(2 * SK_Scalar1, true); ) 396 397 DEF_BENCH( return new DrawPointsDashingBench(1, 1, false); ) 398 DEF_BENCH( return new DrawPointsDashingBench(1, 1, true); ) 399 DEF_BENCH( return new DrawPointsDashingBench(3, 1, false); ) 400 DEF_BENCH( return new DrawPointsDashingBench(3, 1, true); ) 401 DEF_BENCH( return new DrawPointsDashingBench(5, 5, false); ) 402 DEF_BENCH( return new DrawPointsDashingBench(5, 5, true); ) 403 404 /* Disable the GiantDashBench for Android devices until we can better control 405 * the memory usage. (https://code.google.com/p/skia/issues/detail?id=1430) 406 */ 407 #ifndef SK_BUILD_FOR_ANDROID 408 DEF_BENCH( return new GiantDashBench(GiantDashBench::kHori_LineType, 0); ) 409 DEF_BENCH( return new GiantDashBench(GiantDashBench::kVert_LineType, 0); ) 410 DEF_BENCH( return new GiantDashBench(GiantDashBench::kDiag_LineType, 0); ) 411 412 // pass 2 to explicitly avoid any 1-is-the-same-as-hairline special casing 413 414 // hori_2 is just too slow to enable at the moment 415 DEF_BENCH( return new GiantDashBench(GiantDashBench::kHori_LineType, 2); ) 416 DEF_BENCH( return new GiantDashBench(GiantDashBench::kVert_LineType, 2); ) 417 DEF_BENCH( return new GiantDashBench(GiantDashBench::kDiag_LineType, 2); ) 418 #endif 419