223 lines
6.1 KiB
C++
223 lines
6.1 KiB
C++
/*
|
|
* Copyright 2018 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/SkPath.h"
|
|
#include "include/core/SkShader.h"
|
|
#include "include/core/SkString.h"
|
|
#include "include/pathops/SkPathOps.h"
|
|
#include "include/private/SkTArray.h"
|
|
#include "include/utils/SkRandom.h"
|
|
|
|
class PathOpsBench : public Benchmark {
|
|
SkString fName;
|
|
SkPath fPath1, fPath2;
|
|
SkPathOp fOp;
|
|
|
|
public:
|
|
PathOpsBench(const char suffix[], SkPathOp op) : fOp(op) {
|
|
fName.printf("pathops_%s", suffix);
|
|
|
|
fPath1.addOval({-10, -20, 10, 20});
|
|
fPath2.addOval({-20, -10, 20, 10});
|
|
}
|
|
|
|
bool isSuitableFor(Backend backend) override {
|
|
return backend == kNonRendering_Backend;
|
|
}
|
|
|
|
protected:
|
|
const char* onGetName() override {
|
|
return fName.c_str();
|
|
}
|
|
|
|
void onDraw(int loops, SkCanvas* canvas) override {
|
|
for (int i = 0; i < loops; i++) {
|
|
for (int j = 0; j < 1000; ++j) {
|
|
SkPath result;
|
|
Op(fPath1, fPath2, fOp, &result);
|
|
}
|
|
}
|
|
}
|
|
|
|
private:
|
|
using INHERITED = Benchmark;
|
|
};
|
|
|
|
class PathOpsSimplifyBench : public Benchmark {
|
|
SkString fName;
|
|
SkPath fPath;
|
|
|
|
public:
|
|
PathOpsSimplifyBench(const char suffix[], const SkPath& path) : fPath(path) {
|
|
fName.printf("pathops_simplify_%s", suffix);
|
|
}
|
|
|
|
bool isSuitableFor(Backend backend) override {
|
|
return backend == kNonRendering_Backend;
|
|
}
|
|
|
|
protected:
|
|
const char* onGetName() override {
|
|
return fName.c_str();
|
|
}
|
|
|
|
void onDraw(int loops, SkCanvas* canvas) override {
|
|
for (int i = 0; i < loops; i++) {
|
|
for (int j = 0; j < 100; ++j) {
|
|
SkPath result;
|
|
Simplify(fPath, &result);
|
|
}
|
|
}
|
|
}
|
|
|
|
private:
|
|
using INHERITED = Benchmark;
|
|
};
|
|
DEF_BENCH( return new PathOpsBench("sect", kIntersect_SkPathOp); )
|
|
DEF_BENCH( return new PathOpsBench("join", kUnion_SkPathOp); )
|
|
|
|
static SkPath makerects() {
|
|
SkRandom rand;
|
|
SkPath path;
|
|
SkScalar scale = 100;
|
|
for (int i = 0; i < 20; ++i) {
|
|
SkScalar x = rand.nextUScalar1() * scale;
|
|
SkScalar y = rand.nextUScalar1() * scale;
|
|
path.addRect({x, y, x + scale, y + scale});
|
|
}
|
|
return path;
|
|
}
|
|
DEF_BENCH( return new PathOpsSimplifyBench("rects", makerects()); )
|
|
|
|
#include "include/core/SkPathBuilder.h"
|
|
|
|
template <size_t N> struct ArrayPath {
|
|
SkPoint fPts[N];
|
|
uint8_t fVbs[N];
|
|
int fPIndex = 0, fVIndex = 0;
|
|
|
|
void moveTo(float x, float y) {
|
|
fVbs[fVIndex++] = (uint8_t)SkPathVerb::kMove;
|
|
fPts[fPIndex++] = {x, y};
|
|
}
|
|
void lineTo(float x, float y) {
|
|
fVbs[fVIndex++] = (uint8_t)SkPathVerb::kLine;
|
|
fPts[fPIndex++] = {x, y};
|
|
}
|
|
void quadTo(float x, float y, float x1, float y1) {
|
|
fVbs[fVIndex++] = (uint8_t)SkPathVerb::kQuad;
|
|
fPts[fPIndex++] = {x, y};
|
|
fPts[fPIndex++] = {x1, y1};
|
|
}
|
|
void cubicTo(float x, float y, float x1, float y1, float x2, float y2) {
|
|
fVbs[fVIndex++] = (uint8_t)SkPathVerb::kCubic;
|
|
fPts[fPIndex++] = {x, y};
|
|
fPts[fPIndex++] = {x1, y1};
|
|
fPts[fPIndex++] = {x2, y2};
|
|
}
|
|
void incReserve(int) {}
|
|
};
|
|
|
|
template <typename T> void run_builder(T& b, bool useReserve, int N) {
|
|
if (useReserve) {
|
|
b.incReserve(N * 12);
|
|
}
|
|
|
|
float x = 0, y = 0;
|
|
b.moveTo(x, y);
|
|
for (int i = 1; i < N; ++i) {
|
|
b.lineTo(x, y);
|
|
b.quadTo(x, y, x, y);
|
|
b.cubicTo(x, y, x, y, x, y);
|
|
}
|
|
}
|
|
|
|
enum class MakeType {
|
|
kPath,
|
|
kSnapshot,
|
|
kDetach,
|
|
kArray,
|
|
};
|
|
|
|
class PathBuilderBench : public Benchmark {
|
|
SkString fName;
|
|
MakeType fMakeType;
|
|
bool fUseReserve;
|
|
|
|
enum { N = 100 };
|
|
ArrayPath<N*12> fArrays;
|
|
|
|
public:
|
|
PathBuilderBench(MakeType mt, bool reserve) : fMakeType(mt), fUseReserve(reserve) {
|
|
const char* typenames[] = { "path", "snapshot", "detach", "arrays" };
|
|
|
|
fName.printf("makepath_%s_%s", typenames[(int)mt], reserve ? "reserve" : "noreserve");
|
|
}
|
|
|
|
bool isSuitableFor(Backend backend) override {
|
|
return backend == kNonRendering_Backend;
|
|
}
|
|
|
|
protected:
|
|
const char* onGetName() override {
|
|
return fName.c_str();
|
|
}
|
|
|
|
void onDelayedSetup() override {
|
|
run_builder(fArrays, false, N);
|
|
}
|
|
|
|
SkPath build() {
|
|
switch (fMakeType) {
|
|
case MakeType::kSnapshot:
|
|
case MakeType::kDetach: {
|
|
SkPathBuilder b;
|
|
run_builder(b, fUseReserve, N);
|
|
return MakeType::kSnapshot == fMakeType ? b.snapshot() : b.detach();
|
|
}
|
|
case MakeType::kPath: {
|
|
SkPath p;
|
|
run_builder(p, fUseReserve, N);
|
|
return p;
|
|
}
|
|
case MakeType::kArray: {
|
|
// ArrayPath<N*12> arrays;
|
|
// run_builder(arrays, false, N);
|
|
return SkPath::Make(fArrays.fPts, fArrays.fPIndex,
|
|
fArrays.fVbs, fArrays.fVIndex,
|
|
nullptr, 0, SkPathFillType::kWinding);
|
|
}
|
|
}
|
|
return SkPath();
|
|
}
|
|
|
|
void onDraw(int loops, SkCanvas* canvas) override {
|
|
for (int i = 0; i < loops; i++) {
|
|
for (int j = 0; j < 100; ++j) {
|
|
SkPath result = this->build();
|
|
// force bounds calc as part of the test
|
|
if (!result.getBounds().isFinite()) {
|
|
SkDebugf("should never get here!\n");
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private:
|
|
using INHERITED = Benchmark;
|
|
};
|
|
DEF_BENCH( return new PathBuilderBench(MakeType::kPath, false); )
|
|
DEF_BENCH( return new PathBuilderBench(MakeType::kSnapshot, false); )
|
|
DEF_BENCH( return new PathBuilderBench(MakeType::kDetach, false); )
|
|
DEF_BENCH( return new PathBuilderBench(MakeType::kPath, true); )
|
|
DEF_BENCH( return new PathBuilderBench(MakeType::kSnapshot, true); )
|
|
DEF_BENCH( return new PathBuilderBench(MakeType::kDetach, true); )
|
|
|
|
DEF_BENCH( return new PathBuilderBench(MakeType::kArray, true); )
|