|  | /* | 
|  | * Copyright 2011 Google Inc. | 
|  | * | 
|  | * Use of this source code is governed by a BSD-style license that can be | 
|  | * found in the LICENSE file. | 
|  | */ | 
|  | #include "bench/Benchmark.h" | 
|  | #include "include/core/SkBitmap.h" | 
|  | #include "include/core/SkCanvas.h" | 
|  | #include "include/core/SkColorPriv.h" | 
|  | #include "include/core/SkPaint.h" | 
|  | #include "include/core/SkPath.h" | 
|  | #include "include/core/SkShader.h" | 
|  | #include "include/core/SkString.h" | 
|  | #include "include/utils/SkRandom.h" | 
|  | #include "src/core/SkPathPriv.h" | 
|  |  | 
|  | enum class PathIterType { | 
|  | kIter, | 
|  | kRaw, | 
|  | kEdge, | 
|  | }; | 
|  | const char* gPathIterNames[] = { | 
|  | "iter", "raw", "edge" | 
|  | }; | 
|  |  | 
|  | static int rand_pts(SkRandom& rand, SkPoint pts[4]) { | 
|  | int n = rand.nextU() & 3; | 
|  | n += 1; | 
|  |  | 
|  | for (int i = 0; i < n; ++i) { | 
|  | pts[i].fX = rand.nextSScalar1(); | 
|  | pts[i].fY = rand.nextSScalar1(); | 
|  | } | 
|  | return n; | 
|  | } | 
|  |  | 
|  | class PathIterBench : public Benchmark { | 
|  | SkString        fName; | 
|  | SkPath          fPath; | 
|  | PathIterType    fType; | 
|  |  | 
|  | int fVerbInc = 0; | 
|  | SkScalar fXInc = 0, fYInc = 0; | 
|  |  | 
|  | public: | 
|  | PathIterBench(PathIterType t) : fType(t) { | 
|  | fName.printf("pathiter_%s", gPathIterNames[static_cast<unsigned>(t)]); | 
|  |  | 
|  | SkRandom rand; | 
|  | for (int i = 0; i < 1000; ++i) { | 
|  | SkPoint pts[4]; | 
|  | int n = rand_pts(rand, pts); | 
|  | switch (n) { | 
|  | case 1: | 
|  | fPath.moveTo(pts[0]); | 
|  | break; | 
|  | case 2: | 
|  | fPath.lineTo(pts[1]); | 
|  | break; | 
|  | case 3: | 
|  | fPath.quadTo(pts[1], pts[2]); | 
|  | break; | 
|  | case 4: | 
|  | fPath.cubicTo(pts[1], pts[2], pts[3]); | 
|  | break; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | bool isSuitableFor(Backend backend) override { | 
|  | return backend == kNonRendering_Backend; | 
|  | } | 
|  |  | 
|  | protected: | 
|  | const char* onGetName() override { | 
|  | return fName.c_str(); | 
|  | } | 
|  |  | 
|  | void onDraw(int loops, SkCanvas*) override { | 
|  | // Need to do *something* with the results, so the compile doesn't elide | 
|  | // away the code we want to time. | 
|  | auto handle = [this](int verb, const SkPoint pts[]) { | 
|  | fVerbInc += verb; | 
|  | fXInc += pts[0].fX; | 
|  | fYInc += pts[0].fY; | 
|  | }; | 
|  |  | 
|  | switch (fType) { | 
|  | case PathIterType::kIter: | 
|  | for (int i = 0; i < loops; ++i) { | 
|  | SkPath::Iter iter(fPath, true); | 
|  | SkPath::Verb verb; | 
|  | SkPoint      pts[4]; | 
|  | while ((verb = iter.next(pts)) != SkPath::kDone_Verb) { | 
|  | handle(verb, pts); | 
|  | } | 
|  | } | 
|  | break; | 
|  | case PathIterType::kRaw: | 
|  | for (int i = 0; i < loops; ++i) { | 
|  | for (auto [verb, pts, w] : SkPathPriv::Iterate(fPath)) { | 
|  | handle((SkPath::Verb)verb, pts); | 
|  | } | 
|  | } | 
|  | break; | 
|  | case PathIterType::kEdge: | 
|  | for (int i = 0; i < loops; ++i) { | 
|  | SkPathEdgeIter iter(fPath); | 
|  | while (auto r = iter.next()) { | 
|  | handle((int)r.fEdge, r.fPts); | 
|  | } | 
|  | } | 
|  | break; | 
|  | } | 
|  | } | 
|  |  | 
|  | private: | 
|  | using INHERITED = Benchmark; | 
|  | }; | 
|  |  | 
|  | /////////////////////////////////////////////////////////////////////////////// | 
|  |  | 
|  | DEF_BENCH( return new PathIterBench(PathIterType::kIter); ) | 
|  | DEF_BENCH( return new PathIterBench(PathIterType::kRaw); ) | 
|  | DEF_BENCH( return new PathIterBench(PathIterType::kEdge); ) |