blob: 547c0f0d06650b74906cf6c129db640c9bd95d2b [file] [log] [blame]
/*
* 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/SkPaint.h"
#include "include/core/SkPath.h"
#include "include/core/SkPathBuilder.h"
#include "include/core/SkPathUtils.h"
#include "include/core/SkRRect.h"
#include "include/core/SkShader.h"
#include "include/core/SkString.h"
#include "include/private/base/SkTArray.h"
#include "include/private/base/SkTDArray.h"
#include "src/base/SkRandom.h"
#include "src/core/SkColorPriv.h"
#include "src/core/SkDraw.h"
#include "src/core/SkMatrixPriv.h"
#include "src/core/SkPathData.h"
using namespace skia_private;
enum Flags {
kStroke_Flag = 1 << 0,
kBig_Flag = 1 << 1
};
#define FLAGS00 Flags(0)
#define FLAGS01 Flags(kStroke_Flag)
#define FLAGS10 Flags(kBig_Flag)
#define FLAGS11 Flags(kStroke_Flag | kBig_Flag)
class PathBench : public Benchmark {
SkPaint fPaint;
SkString fName;
Flags fFlags;
public:
PathBench(Flags flags) : fFlags(flags) {
fPaint.setStyle(flags & kStroke_Flag ? SkPaint::kStroke_Style :
SkPaint::kFill_Style);
fPaint.setStrokeWidth(SkIntToScalar(5));
fPaint.setStrokeJoin(SkPaint::kBevel_Join);
}
virtual void appendName(SkString*) = 0;
virtual SkPath makePath() = 0;
virtual int complexity() { return 0; }
protected:
const char* onGetName() override {
fName.printf("path_%s_%s_",
fFlags & kStroke_Flag ? "stroke" : "fill",
fFlags & kBig_Flag ? "big" : "small");
this->appendName(&fName);
return fName.c_str();
}
void onDraw(int loops, SkCanvas* canvas) override {
SkPaint paint(fPaint);
this->setupPaint(&paint);
SkPath path = this->makePath();
if (fFlags & kBig_Flag) {
path = path.makeTransform(SkMatrix::Scale(10, 10));
}
for (int i = 0; i < loops; i++) {
canvas->drawPath(path, paint);
}
}
private:
using INHERITED = Benchmark;
};
class TrianglePathBench : public PathBench {
public:
TrianglePathBench(Flags flags) : INHERITED(flags) {}
void appendName(SkString* name) override {
name->append("triangle");
}
SkPath makePath() override {
static const SkPoint pts[] = {
{10, 10}, {15, 5}, {20, 20}
};
return SkPath::Polygon(pts, true);
}
private:
using INHERITED = PathBench;
};
class RectPathBench : public PathBench {
public:
RectPathBench(Flags flags) : INHERITED(flags) {}
void appendName(SkString* name) override {
name->append("rect");
}
SkPath makePath() override {
return SkPath::Rect({10, 10, 20, 20});
}
private:
using INHERITED = PathBench;
};
class RotatedRectBench : public PathBench {
public:
RotatedRectBench(Flags flags, bool aa, float degrees) : INHERITED(flags) {
fAA = aa;
fDegrees = degrees;
}
void appendName(SkString* name) override {
SkString suffix;
suffix.printf("rotated_rect_%s_%g", fAA ? "aa" : "noaa", fDegrees);
name->append(suffix);
}
SkPath makePath() override {
SkPath path = SkPath::Rect({10, 10, 20, 20});
return path.makeTransform(SkMatrix::RotateDeg(fDegrees));
}
void setupPaint(SkPaint* paint) override {
PathBench::setupPaint(paint);
paint->setAntiAlias(fAA);
}
private:
using INHERITED = PathBench;
float fDegrees;
bool fAA;
};
class OvalPathBench : public PathBench {
public:
OvalPathBench(Flags flags) : INHERITED(flags) {}
void appendName(SkString* name) override {
name->append("oval");
}
SkPath makePath() override {
return SkPath::Oval({ 10, 10, 23, 20 });
}
private:
using INHERITED = PathBench;
};
class CirclePathBench: public PathBench {
public:
CirclePathBench(Flags flags) : INHERITED(flags) {}
void appendName(SkString* name) override {
name->append("circle");
}
SkPath makePath() override {
return SkPath::Circle(20, 20, 10);
}
private:
using INHERITED = PathBench;
};
class NonAACirclePathBench: public CirclePathBench {
public:
NonAACirclePathBench(Flags flags) : INHERITED(flags) {}
void appendName(SkString* name) override {
name->append("nonaacircle");
}
void setupPaint(SkPaint* paint) override {
CirclePathBench::setupPaint(paint);
paint->setAntiAlias(false);
}
private:
using INHERITED = CirclePathBench;
};
// Test max speedup of Analytic AA for concave paths
class AAAConcavePathBench : public PathBench {
public:
AAAConcavePathBench(Flags flags) : INHERITED(flags) {}
void appendName(SkString* name) override {
name->append("concave_aaa");
}
SkPath makePath() override {
const SkPoint pts[] = {
{10, 10}, {15, 10}, {15, 5}, {40, 40},
};
return SkPath::Polygon(pts, true);
}
private:
using INHERITED = PathBench;
};
// Test max speedup of Analytic AA for convex paths
class AAAConvexPathBench : public PathBench {
public:
AAAConvexPathBench(Flags flags) : INHERITED(flags) {}
void appendName(SkString* name) override {
name->append("convex_aaa");
}
SkPath makePath() override {
const SkPoint pts[] = {
{10, 10}, {15, 10}, {40, 50},
};
return SkPath::Polygon(pts, true);
}
private:
using INHERITED = PathBench;
};
class SawToothPathBench : public PathBench {
public:
SawToothPathBench(Flags flags) : INHERITED(flags) {}
void appendName(SkString* name) override {
name->append("sawtooth");
}
SkPath makePath() override {
SkScalar x = SkIntToScalar(20);
SkScalar y = SkIntToScalar(20);
const SkScalar x0 = x;
const SkScalar dx = SK_Scalar1 * 5;
const SkScalar dy = SK_Scalar1 * 10;
SkPathBuilder builder;
builder.moveTo(x, y);
for (int i = 0; i < 32; i++) {
x += dx;
builder.lineTo(x, y - dy);
x += dx;
builder.lineTo(x, y + dy);
}
builder.lineTo(x, y + 2 * dy);
builder.lineTo(x0, y + 2 * dy);
builder.close();
return builder.detach();
}
int complexity() override { return 1; }
private:
using INHERITED = PathBench;
};
class LongCurvedPathBench : public PathBench {
public:
LongCurvedPathBench(Flags flags) : INHERITED(flags) {}
void appendName(SkString* name) override {
name->append("long_curved");
}
SkPath makePath() override {
SkRandom rand (12);
SkPathBuilder builder;
for (int i = 0; i < 100; i++) {
builder.quadTo(rand.nextUScalar1() * 640, rand.nextUScalar1() * 480,
rand.nextUScalar1() * 640, rand.nextUScalar1() * 480);
}
builder.close();
return builder.detach();
}
int complexity() override { return 2; }
private:
using INHERITED = PathBench;
};
class LongLinePathBench : public PathBench {
public:
LongLinePathBench(Flags flags) : INHERITED(flags) {}
void appendName(SkString* name) override {
name->append("long_line");
}
SkPath makePath() override {
SkRandom rand;
SkPathBuilder builder;
builder.moveTo(rand.nextUScalar1() * 640, rand.nextUScalar1() * 480);
for (size_t i = 1; i < 100; i++) {
builder.lineTo(rand.nextUScalar1() * 640, rand.nextUScalar1() * 480);
}
return builder.detach();
}
int complexity() override { return 2; }
private:
using INHERITED = PathBench;
};
class RandomPathBench : public Benchmark {
public:
bool isSuitableFor(Backend backend) override {
return backend == Backend::kNonRendering;
}
protected:
void createData(int minVerbs,
int maxVerbs,
bool allowMoves = true,
SkRect* bounds = nullptr) {
SkRect tempBounds;
if (nullptr == bounds) {
tempBounds.setXYWH(0, 0, SK_Scalar1, SK_Scalar1);
bounds = &tempBounds;
}
fVerbCnts.reset(kNumVerbCnts);
for (int i = 0; i < kNumVerbCnts; ++i) {
fVerbCnts[i] = fRandom.nextRangeU(minVerbs, maxVerbs + 1);
}
fVerbs.reset(kNumVerbs);
for (int i = 0; i < kNumVerbs; ++i) {
do {
fVerbs[i] = static_cast<SkPathVerb>(fRandom.nextULessThan((int)SkPathVerb::kClose));
} while (!allowMoves && SkPathVerb::kMove == fVerbs[i]);
}
fPoints.reset(kNumPoints);
for (int i = 0; i < kNumPoints; ++i) {
fPoints[i].set(fRandom.nextRangeScalar(bounds->fLeft, bounds->fRight),
fRandom.nextRangeScalar(bounds->fTop, bounds->fBottom));
}
this->restartMakingPaths();
}
void restartMakingPaths() {
fCurrPath = 0;
fCurrVerb = 0;
fCurrPoint = 0;
}
SkPath makePath() {
SkPathBuilder builder;
int vCount = fVerbCnts[(fCurrPath++) & (kNumVerbCnts - 1)];
for (int v = 0; v < vCount; ++v) {
SkPathVerb verb = fVerbs[(fCurrVerb++) & (kNumVerbs - 1)];
switch (verb) {
case SkPathVerb::kMove:
builder.moveTo(fPoints[(fCurrPoint++) & (kNumPoints - 1)]);
break;
case SkPathVerb::kLine:
builder.lineTo(fPoints[(fCurrPoint++) & (kNumPoints - 1)]);
break;
case SkPathVerb::kQuad:
builder.quadTo(fPoints[(fCurrPoint + 0) & (kNumPoints - 1)],
fPoints[(fCurrPoint + 1) & (kNumPoints - 1)]);
fCurrPoint += 2;
break;
case SkPathVerb::kConic:
builder.conicTo(fPoints[(fCurrPoint + 0) & (kNumPoints - 1)],
fPoints[(fCurrPoint + 1) & (kNumPoints - 1)],
SK_ScalarHalf);
fCurrPoint += 2;
break;
case SkPathVerb::kCubic:
builder.cubicTo(fPoints[(fCurrPoint + 0) & (kNumPoints - 1)],
fPoints[(fCurrPoint + 1) & (kNumPoints - 1)],
fPoints[(fCurrPoint + 2) & (kNumPoints - 1)]);
fCurrPoint += 3;
break;
case SkPathVerb::kClose:
builder.close();
break;
default:
SkDEBUGFAIL("Unexpected path verb");
break;
}
}
return builder.detach();
}
void finishedMakingPaths() {
fVerbCnts.reset(0);
fVerbs.reset(0);
fPoints.reset(0);
}
private:
enum {
// these should all be pow 2
kNumVerbCnts = 1 << 5,
kNumVerbs = 1 << 5,
kNumPoints = 1 << 5,
};
AutoTArray<int> fVerbCnts;
AutoTArray<SkPathVerb> fVerbs;
AutoTArray<SkPoint> fPoints;
int fCurrPath;
int fCurrVerb;
int fCurrPoint;
SkRandom fRandom;
using INHERITED = Benchmark;
};
class PathCreateBench : public RandomPathBench {
public:
PathCreateBench() {
}
protected:
const char* onGetName() override {
return "path_create";
}
void onDelayedSetup() override {
this->createData(10, 100);
}
void onDraw(int loops, SkCanvas*) override {
for (int i = 0; i < loops; ++i) {
(void)this->makePath();
}
this->restartMakingPaths();
}
private:
using INHERITED = RandomPathBench;
};
class PathCopyBench : public RandomPathBench {
public:
PathCopyBench() {
}
protected:
const char* onGetName() override {
return "path_copy";
}
void onDelayedSetup() override {
this->createData(10, 100);
fPaths.reset(kPathCnt);
fCopies.reset(kPathCnt);
for (int i = 0; i < kPathCnt; ++i) {
fPaths[i] = this->makePath();
}
this->finishedMakingPaths();
}
void onDraw(int loops, SkCanvas*) override {
for (int i = 0; i < loops; ++i) {
int idx = i & (kPathCnt - 1);
fCopies[idx] = fPaths[idx];
}
}
private:
enum {
// must be a pow 2
kPathCnt = 1 << 5,
};
AutoTArray<SkPath> fPaths;
AutoTArray<SkPath> fCopies;
using INHERITED = RandomPathBench;
};
enum class BenchPathType {
kPath,
kBuilder,
kData,
};
const char* gBenchPathTypeNames[] = { "path", "builder", "data" };
class PathTransformBench : public Benchmark {
public:
PathTransformBench(BenchPathType t, bool p) : fType(t), fPerspective(p) {
const char* mx = fPerspective ? "persp" : "affine";
fName.printf("path_transform_%s_%s", mx, gBenchPathTypeNames[(int)fType]);
}
protected:
const char* onGetName() override {
return fName.c_str();
}
bool isSuitableFor(Backend backend) override {
return backend == Backend::kNonRendering;
}
void onDelayedSetup() override {
const SkRect r = {0, 0, 100, 100};
fBuilderSrc.addOval(r);
fBuilderSrc.addOval(r.makeInset(10, 10));
fPathSrc = fBuilderSrc.snapshot();
fPData = fBuilderSrc.snapshotData();
if (fPerspective) {
fMatrix[6] = 0.0000001f;
} else {
fMatrix[0] = 1.000001f;
}
}
void onDraw(int loops, SkCanvas*) override {
// we ask for bounds each time, to ensure we're playing fair, as some
// techniques compute bounds up front after a transform, and some defer
// it until it is first requested.
for (int i = 0; i < loops; ++i) {
switch (fType) {
case BenchPathType::kPath:
fPathSrc.makeTransform(fMatrix).getBounds();
break;
case BenchPathType::kBuilder:
fBuilderSrc.transform(fMatrix);
fBuilderSrc.snapshot().getBounds();
break;
case BenchPathType::kData:
fPData->makeTransform(fMatrix)->bounds();
break;
}
}
}
private:
SkPath fPathSrc;
SkPathBuilder fBuilderSrc;
sk_sp<SkPathData> fPData;
SkMatrix fMatrix;
BenchPathType fType;
bool fPerspective;
SkString fName;
};
static void builder_from_rect(const SkRRect& r) {
SkPathBuilder bu; bu.addRect(r.rect()); (void)bu.detach();
}
static void builder_from_oval(const SkRRect& r) {
SkPathBuilder bu; bu.addOval(r.rect()); (void)bu.detach();
}
static void builder_from_rrect(const SkRRect& r) {
SkPathBuilder bu; bu.addRRect(r); (void)bu.detach();
}
static void path_from_rect(const SkRRect& r) {
(void)SkPath::Rect(r.rect());
}
static void path_from_oval(const SkRRect& r) {
(void)SkPath::Oval(r.rect());
}
static void path_from_rrect(const SkRRect& r) {
(void)SkPath::RRect(r);
}
static void pdata_from_rect(const SkRRect& r) {
(void)SkPathData::Rect(r.rect());
}
static void pdata_from_oval(const SkRRect& r) {
(void)SkPathData::Oval(r.rect());
}
static void pdata_from_rrect(const SkRRect& r) {
(void)SkPathData::RRect(r);
}
class PathMakeFromBench : public Benchmark {
public:
using MakeFrom = void(const SkRRect&);
PathMakeFromBench(const char name[], MakeFrom* proc) : fMaker(proc) {
fName.printf("pathmaker_%s", name);
}
protected:
const char* onGetName() override {
return fName.c_str();
}
bool isSuitableFor(Backend backend) override {
return backend == Backend::kNonRendering;
}
void onDraw(int loops, SkCanvas*) override {
const SkRRect rr = SkRRect::MakeRectXY({10, 20, 30, 40}, 2, 3);
for (int i = 0; i < loops; ++i) {
fMaker(rr);
}
}
private:
MakeFrom* fMaker;
SkString fName;
};
class PathEqualityBench : public RandomPathBench {
public:
PathEqualityBench() { }
protected:
const char* onGetName() override {
return "path_equality_50%";
}
void onDelayedSetup() override {
fParity = 0;
this->createData(10, 100);
fPaths.reset(kPathCnt);
fCopies.reset(kPathCnt);
for (int i = 0; i < kPathCnt; ++i) {
fPaths[i] = this->makePath();
fCopies[i] = fPaths[i];
}
this->finishedMakingPaths();
}
void onDraw(int loops, SkCanvas*) override {
for (int i = 0; i < loops; ++i) {
int idx = i & (kPathCnt - 1);
fParity ^= (fPaths[idx] == fCopies[idx & ~0x1]);
}
}
private:
bool fParity; // attempt to keep compiler from optimizing out the ==
enum {
// must be a pow 2
kPathCnt = 1 << 5,
};
AutoTArray<SkPath> fPaths;
AutoTArray<SkPath> fCopies;
using INHERITED = RandomPathBench;
};
#ifndef SK_HIDE_PATH_EDIT_METHODS
class SkBench_AddPathTest : public RandomPathBench {
public:
enum AddType {
kAdd_AddType,
kAddTrans_AddType,
kAddMatrix_AddType,
kReverseAdd_AddType,
};
SkBench_AddPathTest(AddType type) : fType(type) {
fMatrix.setRotate(60 * SK_Scalar1);
}
protected:
const char* onGetName() override {
switch (fType) {
case kAdd_AddType:
return "path_add_path";
case kAddTrans_AddType:
return "path_add_path_trans";
case kAddMatrix_AddType:
return "path_add_path_matrix";
case kReverseAdd_AddType:
return "path_reverse_add_path";
default:
SkDEBUGFAIL("Bad add type");
return "";
}
}
void onDelayedSetup() override {
this->createData(10, 100, true);
fPaths0.reset(kPathCnt);
fPaths1.reset(kPathCnt);
for (int i = 0; i < kPathCnt; ++i) {
fPaths0[i] = this->makePath();
fPaths1[i] = this->makePath();
}
this->finishedMakingPaths();
}
void onDraw(int loops, SkCanvas*) override {
switch (fType) {
case kAdd_AddType:
for (int i = 0; i < loops; ++i) {
int idx = i & (kPathCnt - 1);
SkPath result = fPaths0[idx];
result.addPath(fPaths1[idx]);
}
break;
case kAddTrans_AddType:
for (int i = 0; i < loops; ++i) {
int idx = i & (kPathCnt - 1);
SkPath result = fPaths0[idx];
result.addPath(fPaths1[idx], 2 * SK_Scalar1, 5 * SK_Scalar1);
}
break;
case kAddMatrix_AddType:
for (int i = 0; i < loops; ++i) {
int idx = i & (kPathCnt - 1);
SkPath result = fPaths0[idx];
result.addPath(fPaths1[idx], fMatrix);
}
break;
case kReverseAdd_AddType:
for (int i = 0; i < loops; ++i) {
int idx = i & (kPathCnt - 1);
SkPath result = fPaths0[idx];
result.reverseAddPath(fPaths1[idx]);
}
break;
}
}
private:
AddType fType; // or reverseAddPath
enum {
// must be a pow 2
kPathCnt = 1 << 5,
};
AutoTArray<SkPath> fPaths0;
AutoTArray<SkPath> fPaths1;
SkMatrix fMatrix;
using INHERITED = RandomPathBench;
};
#endif
class CirclesBench : public Benchmark {
protected:
SkString fName;
Flags fFlags;
public:
CirclesBench(Flags flags) : fFlags(flags) {
fName.printf("circles_%s", fFlags & kStroke_Flag ? "stroke" : "fill");
}
protected:
const char* onGetName() override {
return fName.c_str();
}
void onDraw(int loops, SkCanvas* canvas) override {
SkPaint paint;
paint.setColor(SK_ColorBLACK);
paint.setAntiAlias(true);
if (fFlags & kStroke_Flag) {
paint.setStyle(SkPaint::kStroke_Style);
}
SkRandom rand;
SkRect r;
for (int i = 0; i < loops; ++i) {
SkScalar radius = rand.nextUScalar1() * 3;
r.fLeft = rand.nextUScalar1() * 300;
r.fTop = rand.nextUScalar1() * 300;
r.fRight = r.fLeft + 2 * radius;
r.fBottom = r.fTop + 2 * radius;
if (fFlags & kStroke_Flag) {
paint.setStrokeWidth(rand.nextUScalar1() * 5.0f);
}
// mimic how Chrome does circles
SkPath temp = SkPathBuilder()
.arcTo(r, 0, 0, false)
.addOval(r, SkPathDirection::kCCW)
.arcTo(r, 360, 0, true)
.close()
.detach();
canvas->drawPath(temp, paint);
}
}
private:
using INHERITED = Benchmark;
};
// Chrome creates its own round rects with each corner possibly being different.
// In its "zero radius" incarnation it creates degenerate round rects.
// Note: PathTest::test_arb_round_rect_is_convex and
// test_arb_zero_rad_round_rect_is_rect perform almost exactly
// the same test (but with no drawing)
class ArbRoundRectBench : public Benchmark {
protected:
SkString fName;
public:
ArbRoundRectBench(bool zeroRad) : fZeroRad(zeroRad) {
if (zeroRad) {
fName.printf("zeroradroundrect");
} else {
fName.printf("arbroundrect");
}
}
protected:
const char* onGetName() override {
return fName.c_str();
}
static void add_corner_arc(SkPathBuilder* builder, const SkRect& rect,
SkScalar xIn, SkScalar yIn,
int startAngle)
{
SkScalar rx = std::min(rect.width(), xIn);
SkScalar ry = std::min(rect.height(), yIn);
SkRect arcRect;
arcRect.setLTRB(-rx, -ry, rx, ry);
switch (startAngle) {
case 0:
arcRect.offset(rect.fRight - arcRect.fRight, rect.fBottom - arcRect.fBottom);
break;
case 90:
arcRect.offset(rect.fLeft - arcRect.fLeft, rect.fBottom - arcRect.fBottom);
break;
case 180:
arcRect.offset(rect.fLeft - arcRect.fLeft, rect.fTop - arcRect.fTop);
break;
case 270:
arcRect.offset(rect.fRight - arcRect.fRight, rect.fTop - arcRect.fTop);
break;
default:
break;
}
builder->arcTo(arcRect, SkIntToScalar(startAngle), SkIntToScalar(90), false);
}
static SkPath make_arb_round_rect(const SkRect& r, SkScalar xCorner, SkScalar yCorner) {
SkPathBuilder builder;
// we are lazy here and use the same x & y for each corner
add_corner_arc(&builder, r, xCorner, yCorner, 270);
add_corner_arc(&builder, r, xCorner, yCorner, 0);
add_corner_arc(&builder, r, xCorner, yCorner, 90);
add_corner_arc(&builder, r, xCorner, yCorner, 180);
builder.close();
SkPath path = builder.detach();
SkASSERT(path.isConvex());
return path;
}
void onDraw(int loops, SkCanvas* canvas) override {
SkRandom rand;
SkRect r;
for (int i = 0; i < loops; ++i) {
SkPaint paint;
paint.setColor(0xff000000 | rand.nextU());
paint.setAntiAlias(true);
SkScalar size = rand.nextUScalar1() * 30;
if (size < SK_Scalar1) {
continue;
}
r.fLeft = rand.nextUScalar1() * 300;
r.fTop = rand.nextUScalar1() * 300;
r.fRight = r.fLeft + 2 * size;
r.fBottom = r.fTop + 2 * size;
SkPath temp;
if (fZeroRad) {
temp = make_arb_round_rect(r, 0, 0);
SkASSERT(temp.isRect(nullptr));
} else {
temp = make_arb_round_rect(r, r.width() / 10, r.height() / 15);
}
canvas->drawPath(temp, paint);
}
}
private:
bool fZeroRad; // should 0 radius rounds rects be tested?
using INHERITED = Benchmark;
};
class ConservativelyContainsBench : public Benchmark {
public:
enum Type {
kRect_Type,
kRoundRect_Type,
kOval_Type,
};
ConservativelyContainsBench(Type type) {
fParity = false;
fName = "conservatively_contains_";
switch (type) {
case kRect_Type:
fName.append("rect");
fPath = SkPath::Rect(kBaseRect);
break;
case kRoundRect_Type:
fName.append("round_rect");
fPath = SkPath::RRect(kBaseRect, kRRRadii[0], kRRRadii[1]);
break;
case kOval_Type:
fName.append("oval");
fPath = SkPath::Oval(kBaseRect);
break;
}
}
bool isSuitableFor(Backend backend) override {
return backend == Backend::kNonRendering;
}
private:
const char* onGetName() override {
return fName.c_str();
}
void onDraw(int loops, SkCanvas*) override {
for (int i = 0; i < loops; ++i) {
const SkRect& rect = fQueryRects[i % kQueryRectCnt];
fParity = fParity != fPath.conservativelyContainsRect(rect);
}
}
void onDelayedSetup() override {
fQueryRects.resize(kQueryRectCnt);
SkRandom rand;
for (int i = 0; i < kQueryRectCnt; ++i) {
SkSize size;
SkPoint xy;
size.fWidth = rand.nextRangeScalar(kQueryMin.fWidth, kQueryMax.fWidth);
size.fHeight = rand.nextRangeScalar(kQueryMin.fHeight, kQueryMax.fHeight);
xy.fX = rand.nextRangeScalar(kBounds.fLeft, kBounds.fRight - size.fWidth);
xy.fY = rand.nextRangeScalar(kBounds.fTop, kBounds.fBottom - size.fHeight);
fQueryRects[i] = SkRect::MakeXYWH(xy.fX, xy.fY, size.fWidth, size.fHeight);
}
}
enum {
kQueryRectCnt = 400,
};
static const SkRect kBounds; // bounds for all random query rects
static const SkSize kQueryMin; // minimum query rect size, should be <= kQueryMax
static const SkSize kQueryMax; // max query rect size, should < kBounds
static const SkRect kBaseRect; // rect that is used to construct the path
static const SkScalar kRRRadii[2]; // x and y radii for round rect
SkString fName;
SkPath fPath;
bool fParity;
SkTDArray<SkRect> fQueryRects;
using INHERITED = Benchmark;
};
///////////////////////////////////////////////////////////////////////////////
#include "src/core/SkGeometry.h"
class ConicBench_Chop : public Benchmark {
protected:
SkConic fRQ, fDst[2];
SkString fName;
public:
ConicBench_Chop() : fName("conic-chop") {
fRQ.fPts[0].set(0, 0);
fRQ.fPts[1].set(100, 0);
fRQ.fPts[2].set(100, 100);
fRQ.fW = SkScalarCos(SK_ScalarPI/4);
}
bool isSuitableFor(Backend backend) override {
return backend == Backend::kNonRendering;
}
private:
const char* onGetName() override { return fName.c_str(); }
void onDraw(int loops, SkCanvas*) override {
for (int i = 0; i < loops; ++i) {
fRQ.chop(fDst);
}
}
using INHERITED = Benchmark;
};
DEF_BENCH( return new ConicBench_Chop; )
class ConicBench_EvalPos : public ConicBench_Chop {
const bool fUseV2;
public:
ConicBench_EvalPos(bool useV2) : fUseV2(useV2) {
fName.printf("conic-eval-pos%d", useV2);
}
void onDraw(int loops, SkCanvas*) override {
if (fUseV2) {
for (int i = 0; i < loops; ++i) {
for (int j = 0; j < 1000; ++j) {
fDst[0].fPts[0] = fRQ.evalAt(0.4f);
}
}
} else {
for (int i = 0; i < loops; ++i) {
for (int j = 0; j < 1000; ++j) {
fRQ.evalAt(0.4f, &fDst[0].fPts[0], nullptr);
}
}
}
}
};
DEF_BENCH( return new ConicBench_EvalPos(false); )
DEF_BENCH( return new ConicBench_EvalPos(true); )
class ConicBench_EvalTan : public ConicBench_Chop {
const bool fUseV2;
public:
ConicBench_EvalTan(bool useV2) : fUseV2(useV2) {
fName.printf("conic-eval-tan%d", useV2);
}
void onDraw(int loops, SkCanvas*) override {
if (fUseV2) {
for (int i = 0; i < loops; ++i) {
for (int j = 0; j < 1000; ++j) {
fDst[0].fPts[0] = fRQ.evalTangentAt(0.4f);
}
}
} else {
for (int i = 0; i < loops; ++i) {
for (int j = 0; j < 1000; ++j) {
fRQ.evalAt(0.4f, nullptr, &fDst[0].fPts[0]);
}
}
}
}
};
DEF_BENCH( return new ConicBench_EvalTan(false); )
DEF_BENCH( return new ConicBench_EvalTan(true); )
class ConicBench_TinyError : public Benchmark {
protected:
SkString fName;
public:
ConicBench_TinyError() : fName("conic-tinyerror") {}
protected:
const char* onGetName() override { return fName.c_str(); }
void onDraw(int loops, SkCanvas*) override {
SkPaint paint;
paint.setColor(SK_ColorBLACK);
paint.setAntiAlias(true);
paint.setStyle(SkPaint::kStroke_Style);
paint.setStrokeWidth(2);
SkPath path = SkPathBuilder()
.moveTo(-100, 1)
.cubicTo(-101, 1, -118, -47, -138, -44)
.detach();
// The large y scale factor produces a tiny error threshold.
const SkMatrix mtx = SkMatrix::MakeAll(3.07294035f, 0.833333373f, 361.111115f, 0.0f,
6222222.5f, 28333.334f, 0.0f, 0.0f, 1.0f);
const SkScalar scale = SkMatrixPriv::ComputeResScaleForStroking(mtx);
const SkMatrix mx = SkMatrix::Scale(scale, scale);
for (int i = 0; i < loops; ++i) {
SkPathBuilder dst;
skpathutils::FillPathWithPaint(path, paint, &dst, nullptr, mx);
}
}
private:
using INHERITED = Benchmark;
};
DEF_BENCH( return new ConicBench_TinyError; )
///////////////////////////////////////////////////////////////////////////////
static void rand_conic(SkConic* conic, SkRandom& rand) {
for (int i = 0; i < 3; ++i) {
conic->fPts[i].set(rand.nextUScalar1() * 100, rand.nextUScalar1() * 100);
}
if (rand.nextUScalar1() > 0.5f) {
conic->fW = rand.nextUScalar1();
} else {
conic->fW = 1 + rand.nextUScalar1() * 4;
}
}
class ConicBench : public Benchmark {
public:
ConicBench() {
SkRandom rand;
for (int i = 0; i < CONICS; ++i) {
rand_conic(&fConics[i], rand);
}
}
bool isSuitableFor(Backend backend) override {
return backend == Backend::kNonRendering;
}
protected:
enum {
CONICS = 100
};
SkConic fConics[CONICS];
private:
using INHERITED = Benchmark;
};
class ConicBench_ComputeError : public ConicBench {
public:
ConicBench_ComputeError() {}
protected:
const char* onGetName() override {
return "conic-compute-error";
}
void onDraw(int loops, SkCanvas*) override {
SkVector err;
for (int i = 0; i < loops; ++i) {
for (int j = 0; j < CONICS; ++j) {
fConics[j].computeAsQuadError(&err);
}
}
}
private:
using INHERITED = ConicBench;
};
class ConicBench_asQuadTol : public ConicBench {
public:
ConicBench_asQuadTol() {}
protected:
const char* onGetName() override {
return "conic-asQuadTol";
}
void onDraw(int loops, SkCanvas*) override {
for (int i = 0; i < loops; ++i) {
for (int j = 0; j < CONICS; ++j) {
fConics[j].asQuadTol(SK_ScalarHalf);
}
}
}
private:
using INHERITED = ConicBench;
};
class ConicBench_quadPow2 : public ConicBench {
public:
ConicBench_quadPow2() {}
protected:
const char* onGetName() override {
return "conic-quadPow2";
}
void onDraw(int loops, SkCanvas*) override {
for (int i = 0; i < loops; ++i) {
for (int j = 0; j < CONICS; ++j) {
fConics[j].computeQuadPOW2(SK_ScalarHalf);
}
}
}
private:
using INHERITED = ConicBench;
};
///////////////////////////////////////////////////////////////////////////////
class TightBoundsBench : public Benchmark {
SkPath fPath;
SkString fName;
SkRect (*fProc)(const SkPath&);
public:
TightBoundsBench(SkRect (*proc)(const SkPath&), const char suffix[]) : fProc(proc) {
fName.printf("tight_bounds_%s", suffix);
SkPathBuilder builder;
const int N = 100;
SkRandom rand;
for (int i = 0; i < N; ++i) {
builder.moveTo(rand.nextF()*100, rand.nextF()*100);
builder.lineTo(rand.nextF()*100, rand.nextF()*100);
builder.quadTo(rand.nextF()*100, rand.nextF()*100, rand.nextF()*100, rand.nextF()*100);
builder.conicTo(rand.nextF()*100, rand.nextF()*100, rand.nextF()*100, rand.nextF()*100,
rand.nextF()*10);
builder.cubicTo(rand.nextF()*100, rand.nextF()*100, rand.nextF()*100, rand.nextF()*100,
rand.nextF()*100, rand.nextF()*100);
}
fPath = builder.detach();
}
protected:
bool isSuitableFor(Backend backend) override {
return backend == Backend::kNonRendering;
}
const char* onGetName() override { return fName.c_str(); }
void onDraw(int loops, SkCanvas* canvas) override {
for (int i = 0; i < loops*100; ++i) {
fProc(fPath);
}
}
private:
using INHERITED = Benchmark;
};
const SkRect ConservativelyContainsBench::kBounds = SkRect::MakeWH(SkIntToScalar(100), SkIntToScalar(100));
const SkSize ConservativelyContainsBench::kQueryMin = {SkIntToScalar(1), SkIntToScalar(1)};
const SkSize ConservativelyContainsBench::kQueryMax = {SkIntToScalar(40), SkIntToScalar(40)};
const SkRect ConservativelyContainsBench::kBaseRect = SkRect::MakeXYWH(SkIntToScalar(25), SkIntToScalar(25), SkIntToScalar(50), SkIntToScalar(50));
const SkScalar ConservativelyContainsBench::kRRRadii[2] = {SkIntToScalar(5), SkIntToScalar(10)};
DEF_BENCH( return new TrianglePathBench(FLAGS00); )
DEF_BENCH( return new TrianglePathBench(FLAGS01); )
DEF_BENCH( return new TrianglePathBench(FLAGS10); )
DEF_BENCH( return new TrianglePathBench(FLAGS11); )
DEF_BENCH( return new RectPathBench(FLAGS00); )
DEF_BENCH( return new RectPathBench(FLAGS01); )
DEF_BENCH( return new RectPathBench(FLAGS10); )
DEF_BENCH( return new RectPathBench(FLAGS11); )
DEF_BENCH( return new RotatedRectBench(FLAGS00, false, 45));
DEF_BENCH( return new RotatedRectBench(FLAGS10, false, 45));
DEF_BENCH( return new RotatedRectBench(FLAGS00, true, 45));
DEF_BENCH( return new RotatedRectBench(FLAGS10, true, 45));
DEF_BENCH( return new OvalPathBench(FLAGS00); )
DEF_BENCH( return new OvalPathBench(FLAGS01); )
DEF_BENCH( return new OvalPathBench(FLAGS10); )
DEF_BENCH( return new OvalPathBench(FLAGS11); )
DEF_BENCH( return new CirclePathBench(FLAGS00); )
DEF_BENCH( return new CirclePathBench(FLAGS01); )
DEF_BENCH( return new CirclePathBench(FLAGS10); )
DEF_BENCH( return new CirclePathBench(FLAGS11); )
DEF_BENCH( return new NonAACirclePathBench(FLAGS00); )
DEF_BENCH( return new NonAACirclePathBench(FLAGS10); )
DEF_BENCH( return new AAAConcavePathBench(FLAGS00); )
DEF_BENCH( return new AAAConcavePathBench(FLAGS10); )
DEF_BENCH( return new AAAConvexPathBench(FLAGS00); )
DEF_BENCH( return new AAAConvexPathBench(FLAGS10); )
DEF_BENCH( return new SawToothPathBench(FLAGS00); )
DEF_BENCH( return new SawToothPathBench(FLAGS01); )
DEF_BENCH( return new LongCurvedPathBench(FLAGS00); )
DEF_BENCH( return new LongCurvedPathBench(FLAGS01); )
DEF_BENCH( return new LongLinePathBench(FLAGS00); )
DEF_BENCH( return new LongLinePathBench(FLAGS01); )
DEF_BENCH( return new PathCreateBench(); )
DEF_BENCH( return new PathCopyBench(); )
DEF_BENCH( return new PathEqualityBench(); )
DEF_BENCH( return new PathTransformBench(BenchPathType::kData, true); )
DEF_BENCH( return new PathTransformBench(BenchPathType::kBuilder, true); )
DEF_BENCH( return new PathTransformBench(BenchPathType::kPath, true); )
DEF_BENCH( return new PathTransformBench(BenchPathType::kData, false); )
DEF_BENCH( return new PathTransformBench(BenchPathType::kBuilder, false); )
DEF_BENCH( return new PathTransformBench(BenchPathType::kPath, false); )
#define MAKEFROM(name) PathMakeFromBench(#name, name);
DEF_BENCH( return new MAKEFROM(pdata_from_rrect) )
DEF_BENCH( return new MAKEFROM(builder_from_rrect) )
DEF_BENCH( return new MAKEFROM(path_from_rrect) )
DEF_BENCH( return new MAKEFROM(pdata_from_oval) )
DEF_BENCH( return new MAKEFROM(builder_from_oval) )
DEF_BENCH( return new MAKEFROM(path_from_oval) )
DEF_BENCH( return new MAKEFROM(pdata_from_rect) )
DEF_BENCH( return new MAKEFROM(builder_from_rect) )
DEF_BENCH( return new MAKEFROM(path_from_rect) )
#undef MAKEFROM
#ifndef SK_HIDE_PATH_EDIT_METHODS
DEF_BENCH( return new SkBench_AddPathTest(SkBench_AddPathTest::kAdd_AddType); )
DEF_BENCH( return new SkBench_AddPathTest(SkBench_AddPathTest::kAddTrans_AddType); )
DEF_BENCH( return new SkBench_AddPathTest(SkBench_AddPathTest::kAddMatrix_AddType); )
DEF_BENCH( return new SkBench_AddPathTest(SkBench_AddPathTest::kReverseAdd_AddType); )
#endif
DEF_BENCH( return new CirclesBench(FLAGS00); )
DEF_BENCH( return new CirclesBench(FLAGS01); )
DEF_BENCH( return new ArbRoundRectBench(false); )
DEF_BENCH( return new ArbRoundRectBench(true); )
DEF_BENCH( return new ConservativelyContainsBench(ConservativelyContainsBench::kRect_Type); )
DEF_BENCH( return new ConservativelyContainsBench(ConservativelyContainsBench::kRoundRect_Type); )
DEF_BENCH( return new ConservativelyContainsBench(ConservativelyContainsBench::kOval_Type); )
#include "include/pathops/SkPathOps.h"
#include "src/core/SkPathPriv.h"
DEF_BENCH( return new TightBoundsBench([](const SkPath& path){ return path.computeTightBounds();},
"priv"); )
// These seem to be optimized away, which is troublesome for timing.
/*
DEF_BENCH( return new ConicBench_Chop5() )
DEF_BENCH( return new ConicBench_ComputeError() )
DEF_BENCH( return new ConicBench_asQuadTol() )
DEF_BENCH( return new ConicBench_quadPow2() )
*/
class CommonConvexBench : public Benchmark {
protected:
SkString fName;
SkPath fPath;
const bool fAA;
public:
CommonConvexBench(int w, int h, bool forceConcave, bool aa) : fAA(aa) {
fName.printf("convex_path_%d_%d_%d_%d", w, h, forceConcave, aa);
SkRect r = SkRect::MakeXYWH(10, 10, w*1.0f, h*1.0f);
fPath = SkPath::RRect(SkRRect::MakeRectXY(r, w/8.0f, h/8.0f));
if (forceConcave) {
SkPathPriv::SetConvexity(fPath, SkPathConvexity::kConcave);
SkASSERT(!fPath.isConvex());
} else {
SkASSERT(fPath.isConvex());
}
}
protected:
const char* onGetName() override {
return fName.c_str();
}
void onDraw(int loops, SkCanvas* canvas) override {
SkPaint paint;
paint.setAntiAlias(fAA);
for (int i = 0; i < loops; ++i) {
for (int inner = 0; inner < 100; ++inner) {
canvas->drawPath(fPath, paint);
}
}
}
private:
using INHERITED = Benchmark;
};
DEF_BENCH( return new CommonConvexBench( 16, 16, false, false); )
DEF_BENCH( return new CommonConvexBench( 16, 16, true, false); )
DEF_BENCH( return new CommonConvexBench( 16, 16, false, true); )
DEF_BENCH( return new CommonConvexBench( 16, 16, true, true); )
DEF_BENCH( return new CommonConvexBench(200, 16, false, false); )
DEF_BENCH( return new CommonConvexBench(200, 16, true, false); )
DEF_BENCH( return new CommonConvexBench(200, 16, false, true); )
DEF_BENCH( return new CommonConvexBench(200, 16, true, true); )
class PathBuildBench : public Benchmark {
public:
using Builder = SkPath(const SkRect&);
Builder* fBuilder;
SkString fName;
PathBuildBench(const char name[], Builder* builder) : fBuilder(builder) {
fName.printf("path_buider_%s", name);
}
protected:
const char* onGetName() override {
return fName.c_str();
}
void onDraw(int loops, SkCanvas* canvas) override {
const SkRect r = {1, 2, 3, 4};
for (int i = 0; i < loops; ++i) {
SkPath path = fBuilder(r);
std::ignore = path.getBounds();
}
}
bool isSuitableFor(Backend backend) override {
return backend == Backend::kNonRendering;
}
private:
using INHERITED = Benchmark;
};
DEF_BENCH( return new PathBuildBench("addRect", [](const SkRect& r) {
return SkPath::Rect(r);
}));
DEF_BENCH( return new PathBuildBench("addOval", [](const SkRect& r) {
return SkPath::Oval(r);
}));
DEF_BENCH( return new PathBuildBench("addRRect", [](const SkRect& r) {
return SkPath::RRect(SkRRect::MakeRectXY(r, 0.1f, 0.1f));
}));