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